// 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" namespace sol { namespace detail { struct lua_func { 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 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); } }; template::value> struct explicit_lua_func : public lua_func { typedef typename std::remove_pointer::type>::type fx_t; typedef 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); } }; template struct explicit_lua_func : public lua_func { typedef typename std::remove_pointer::type>::type fx_t; typedef function_traits fx_traits; struct lambda { T* member; TFx invocation; template lambda(T* m, FxArgs&&... fxargs): member(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(m, std::forward(fxargs)...) {} template explicit_lua_func(T& m, FxArgs&&... fxargs): fx(std::addressof(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); } }; int lua_cfun(lua_State* L) { void* bridgedata = lua_touserdata(L, lua_upvalueindex(1)); auto* fx = static_cast(bridgedata); int r = fx->operator()(L); return r; } } // detail } // sol #endif // SOL_LUA_FUNC_HPP