// The MIT License (MIT) // Copyright (c) 2013 Danny Y., Rapptz // Permission is hereby granted, free of charge, to any person obtaining a copy of // this software and associated documentation files (the "Software"), to deal in // the Software without restriction, including without limitation the rights to // use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of // the Software, and to permit persons to whom the Software is furnished to do so, // subject to the following conditions: // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS // FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR // COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #ifndef SOL_LUA_FUNC_HPP #define SOL_LUA_FUNC_HPP #include "functional.hpp" #include "stack.hpp" #include namespace sol { template struct static_lua_func { typedef typename std::remove_pointer::type>::type fx_t; typedef detail::function_traits fx_traits; template static int typed_call(types, types t, fx_t* fx, lua_State* L) { stack::pop_call(L, fx, t); return 0; } template static int typed_call(types, types t, fx_t* fx, lua_State* L) { auto r = stack::pop_call(L, fx, t); stack::push(L, std::move(r)); return sizeof...(TRn); } static int call(lua_State* L) { void* functiondata = lua_touserdata(L, lua_upvalueindex(1)); //if (functiondata == nullptr) // throw sol_error("call from lua to c++ function has null function pointer data"); fx_t* fx = reinterpret_cast(functiondata); int r = typed_call(tuple_types(), typename fx_traits::args_type(), fx, L); return r; } int operator()(lua_State* L) { return call(L); } }; template struct static_object_lua_func { typedef typename std::decay::type fx_t; typedef detail::function_traits fx_traits; template static int typed_call(types, types, T& item, fx_t& ifx, lua_State* L) { auto fx = [ &item, &ifx ] (Args&&... args) { (item.*ifx)(std::forward(args)...); }; stack::pop_call(L, fx, types()); return 0; } template static int typed_call(types, types, T& item, fx_t& ifx, lua_State* L) { auto fx = [ &item, &ifx ] (Args&&... args) -> TR { return (item.*ifx)(std::forward(args)...); }; auto r = stack::pop_call(L, fx, types()); stack::push(L, std::move(r)); return 1; } template static int typed_call(types, types, T& item, fx_t& ifx, lua_State* L) { auto fx = [ &item, &ifx ] (Args&&... args) -> std::tuple { return (item.*ifx)(std::forward(args)...); }; auto r = stack::pop_call(L, fx, types()); stack::push(L, std::move(r)); return sizeof...(TRn); } static int call(lua_State* L) { const static std::size_t data_t_count = (sizeof(fx_t)+(sizeof(void*)-1)) / sizeof(void*); typedef std::array data_t; data_t fxptrdata; int upvalue = 1; for (std::size_t i = 0; i < fxptrdata.size(); ++i) { fxptrdata[ i ] = lua_touserdata(L, lua_upvalueindex(upvalue++)); } void* objectdata = lua_touserdata(L, lua_upvalueindex(upvalue++)); //if (objectdata == nullptr) // throw sol_error("call from lua to c++ function has null object data"); fx_t* fxptr = reinterpret_cast(static_cast(fxptrdata.data())); fx_t& mem_ptr = *fxptr; T& obj = *static_cast(objectdata); int r = typed_call(tuple_types(), typename fx_traits::args_type(), obj, mem_ptr, L); return r; } int operator()(lua_State* L) { return call(L); } }; struct lua_func { static int call(lua_State* L) { void** pinheritancedata = static_cast(lua_touserdata(L, lua_upvalueindex(1))); void* inheritancedata = *pinheritancedata; if (inheritancedata == nullptr) throw sol_error("call from lua to c++ function has null data"); lua_func* pfx = static_cast(inheritancedata); lua_func& fx = *pfx; int r = fx(L); return r; } static int gc(lua_State* L) { void** puserdata = static_cast(lua_touserdata(L, 1)); void* userdata = *puserdata; lua_func* ptr = static_cast(userdata); std::default_delete dx{ }; dx(ptr); return 0; } virtual int operator()(lua_State*) { throw sol_error("Failure to call specialized wrapped C++ function from lua"); } virtual ~lua_func() {}; }; template struct lambda_lua_func : public lua_func { typedef decltype(&TFx::operator()) fx_t; typedef detail::function_traits fx_traits; TFx fx; template lambda_lua_func(FxArgs&&... fxargs): fx(std::forward(fxargs)...) {} virtual int operator()(lua_State* L) override { return (*this)(tuple_types(), typename fx_traits::args_type(), L); } template int operator()(types, types t, lua_State* L) { stack::pop_call(L, fx, t); return 0; } template int operator()(types, types t, lua_State* L) { auto r = stack::pop_call(L, fx, t); stack::push(L, r); return sizeof...(TRn); } ~lambda_lua_func() { } }; template::value> struct explicit_lua_func : public lua_func { typedef typename std::remove_pointer::type>::type fx_t; typedef detail::function_traits fx_traits; TFx fx; template explicit_lua_func(FxArgs&&... fxargs): fx(std::forward(fxargs)...) {} virtual int operator()(lua_State* L) override { return (*this)(tuple_types(), typename fx_traits::args_type(), L); } template int operator()(types, types t, lua_State* L) { stack::pop_call(L, fx, t); return 0; } template int operator()(types, types t, lua_State* L) { auto r = stack::pop_call(L, fx, t); stack::push(L, std::move(r)); return sizeof...(TRn); } ~explicit_lua_func() { } }; template struct explicit_lua_func : public lua_func { typedef typename std::remove_pointer::type>::type fx_t; typedef detail::function_traits fx_traits; struct lambda { T member; TFx invocation; template lambda(T m, FxArgs&&... fxargs): member(std::move(m)), invocation(std::forward(fxargs)...) {} template typename fx_traits::return_type operator()(Args&&... args) { return (member.*invocation)(std::forward(args)...); } } fx; template explicit_lua_func(T m, FxArgs&&... fxargs): fx(std::move(m), std::forward(fxargs)...) {} virtual int operator()(lua_State* L) override { return (*this)(tuple_types(), typename fx_traits::args_type(), L); } template int operator()(types, types t, lua_State* L) { stack::pop_call(L, fx, t); return 0; } template int operator()(types, types t, lua_State* L) { auto r = stack::pop_call(L, fx, t); stack::push(L, r); return sizeof...(TRn); } ~explicit_lua_func() { } }; } // sol #endif // SOL_LUA_FUNC_HPP