// 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 { 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 = *static_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* inheritancedata = lua_touserdata( L, lua_upvalueindex( 1 ) ); if ( inheritancedata == nullptr ) throw sol_error( "call from lua to c++ function has null data" ); lua_func& fx = *static_cast( inheritancedata ); int r = fx( L ); return r; } 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); } }; 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); } }; 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(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); } }; } // sol #endif // SOL_LUA_FUNC_HPP