222 lines
6.7 KiB
C++
222 lines
6.7 KiB
C++
// This file is distributed under the BSD License.
|
|
// See "license.txt" for details.
|
|
// Copyright 2009-2012, Jonathan Turner (jonathan@emptycrate.com)
|
|
// Copyright 2009-2014, Jason Turner (jason@emptycrate.com)
|
|
// http://www.chaiscript.com
|
|
|
|
#ifndef CHAISCRIPT_PROXY_FUNCTIONS_DETAIL_HPP_
|
|
#define CHAISCRIPT_PROXY_FUNCTIONS_DETAIL_HPP_
|
|
|
|
#include <algorithm>
|
|
#include <functional>
|
|
#include <stdexcept>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "../chaiscript_defines.hpp"
|
|
#include "boxed_cast.hpp"
|
|
#include "boxed_value.hpp"
|
|
#include "handle_return.hpp"
|
|
#include "type_info.hpp"
|
|
|
|
namespace chaiscript {
|
|
class Dynamic_Cast_Conversions;
|
|
namespace exception {
|
|
class bad_boxed_cast;
|
|
} // namespace exception
|
|
} // namespace chaiscript
|
|
|
|
namespace chaiscript
|
|
{
|
|
namespace exception
|
|
{
|
|
/**
|
|
* Exception thrown when there is a mismatch in number of
|
|
* parameters during Proxy_Function execution
|
|
*/
|
|
struct arity_error : std::range_error
|
|
{
|
|
arity_error(int t_got, int t_expected)
|
|
: std::range_error("Function dispatch arity mismatch"),
|
|
got(t_got), expected(t_expected)
|
|
{
|
|
}
|
|
|
|
virtual ~arity_error() CHAISCRIPT_NOEXCEPT {}
|
|
|
|
int got;
|
|
int expected;
|
|
};
|
|
}
|
|
|
|
namespace dispatch
|
|
{
|
|
namespace detail
|
|
{
|
|
template<typename ... Rest>
|
|
struct Build_Param_Type_List;
|
|
|
|
template<typename Param, typename ... Rest>
|
|
struct Build_Param_Type_List<Param, Rest...>
|
|
{
|
|
static void build(std::vector<Type_Info> &t_params)
|
|
{
|
|
t_params.push_back(chaiscript::detail::Get_Type_Info<Param>::get());
|
|
Build_Param_Type_List<Rest...>::build(t_params);
|
|
}
|
|
};
|
|
|
|
// 0th case
|
|
template<>
|
|
struct Build_Param_Type_List<>
|
|
{
|
|
static void build(std::vector<Type_Info> &)
|
|
{
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Used by Proxy_Function_Impl to return a list of all param types
|
|
* it contains.
|
|
*/
|
|
template<typename Ret, typename ... Params>
|
|
std::vector<Type_Info> build_param_type_list(Ret (*)(Params...))
|
|
{
|
|
/// \todo this code was previously using { chaiscript::detail::Get_Type_Info<Ret>::get()... }
|
|
/// but this seems to indicate another bug with MSVC's uniform initializer lists
|
|
std::vector<Type_Info> params;
|
|
params.push_back(chaiscript::detail::Get_Type_Info<Ret>::get());
|
|
Build_Param_Type_List<Params...>::build(params);
|
|
return params;
|
|
}
|
|
|
|
|
|
// Forward declaration
|
|
template<typename ... Rest>
|
|
struct Try_Cast;
|
|
|
|
template<typename Param, typename ... Rest>
|
|
struct Try_Cast<Param, Rest...>
|
|
{
|
|
static void do_try(const std::vector<Boxed_Value> ¶ms, int generation, const Dynamic_Cast_Conversions &t_conversions)
|
|
{
|
|
boxed_cast<Param>(params[generation], &t_conversions);
|
|
Try_Cast<Rest...>::do_try(params, generation+1, t_conversions);
|
|
}
|
|
};
|
|
|
|
// 0th case
|
|
template<>
|
|
struct Try_Cast<>
|
|
{
|
|
static void do_try(const std::vector<Boxed_Value> &, int, const Dynamic_Cast_Conversions &)
|
|
{
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Used by Proxy_Function_Impl to determine if it is equivalent to another
|
|
* Proxy_Function_Impl object. This function is primarily used to prevent
|
|
* registration of two functions with the exact same signatures
|
|
*/
|
|
template<typename Ret, typename ... Params>
|
|
bool compare_types_cast(Ret (*)(Params...),
|
|
const std::vector<Boxed_Value> ¶ms, const Dynamic_Cast_Conversions &t_conversions)
|
|
{
|
|
try {
|
|
Try_Cast<Params...>::do_try(params, 0, t_conversions);
|
|
} catch (const exception::bad_boxed_cast &) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template<typename Ret, int count, typename ... Params>
|
|
struct Call_Func
|
|
{
|
|
|
|
template<typename ... InnerParams>
|
|
static Ret do_call(const std::function<Ret (Params...)> &f,
|
|
const std::vector<Boxed_Value> ¶ms, const Dynamic_Cast_Conversions &t_conversions, InnerParams &&... innerparams)
|
|
{
|
|
return Call_Func<Ret, count - 1, Params...>::do_call(f, params, t_conversions, std::forward<InnerParams>(innerparams)..., params[sizeof...(Params) - count]);
|
|
}
|
|
};
|
|
|
|
template<typename Ret, typename ... Params>
|
|
struct Call_Func<Ret, 0, Params...>
|
|
{
|
|
#ifdef CHAISCRIPT_MSVC
|
|
#pragma warning(push)
|
|
#pragma warning(disable : 4100) /// Disable unreferenced formal parameter warning, which only shows up in MSVC I don't think there's any way around it \todo evaluate this
|
|
#endif
|
|
template<typename ... InnerParams>
|
|
static Ret do_call(const std::function<Ret (Params...)> &f,
|
|
const std::vector<Boxed_Value> &, const Dynamic_Cast_Conversions &t_conversions, InnerParams &&... innerparams)
|
|
{
|
|
return f(boxed_cast<Params>(std::forward<InnerParams>(innerparams), &t_conversions)...);
|
|
}
|
|
#ifdef CHAISCRIPT_MSVC
|
|
#pragma warning(pop)
|
|
#endif
|
|
};
|
|
|
|
/**
|
|
* Used by Proxy_Function_Impl to perform typesafe execution of a function.
|
|
* The function attempts to unbox each parameter to the expected type.
|
|
* if any unboxing fails the execution of the function fails and
|
|
* the bad_boxed_cast is passed up to the caller.
|
|
*/
|
|
template<typename Ret, typename ... Params>
|
|
Ret call_func(const std::function<Ret (Params...)> &f,
|
|
const std::vector<Boxed_Value> ¶ms, const Dynamic_Cast_Conversions &t_conversions)
|
|
{
|
|
if (params.size() == sizeof...(Params))
|
|
{
|
|
return Call_Func<Ret, sizeof...(Params), Params...>::do_call(f, params, t_conversions);
|
|
}
|
|
|
|
throw exception::arity_error(static_cast<int>(params.size()), sizeof...(Params));
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
namespace chaiscript
|
|
{
|
|
namespace dispatch
|
|
{
|
|
namespace detail
|
|
{
|
|
template<typename Ret>
|
|
struct Do_Call
|
|
{
|
|
template<typename Fun>
|
|
static Boxed_Value go(const std::function<Fun> &fun, const std::vector<Boxed_Value> ¶ms, const Dynamic_Cast_Conversions &t_conversions)
|
|
{
|
|
return Handle_Return<Ret>::handle(call_func(fun, params, t_conversions));
|
|
}
|
|
};
|
|
|
|
template<>
|
|
struct Do_Call<void>
|
|
{
|
|
template<typename Fun>
|
|
static Boxed_Value go(const std::function<Fun> &fun, const std::vector<Boxed_Value> ¶ms, const Dynamic_Cast_Conversions &t_conversions)
|
|
{
|
|
call_func(fun, params, t_conversions);
|
|
return Handle_Return<void>::handle();
|
|
}
|
|
};
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|