12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331 |
- //-----------------------------------------------------------------------------
- // Copyright (c) 2012 GarageGames, LLC
- //
- // 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 _ENGINEAPI_H_
- #define _ENGINEAPI_H_
- #include <tuple>
- #include <utility>
- #ifndef _FIXEDTUPLE_H_
- #include "fixedTuple.h"
- #endif
- #ifndef _CONSOLETYPES_H_
- #include "console/consoleTypes.h"
- #endif
- #ifndef _CONSOLE_H_
- #include "console/console.h"
- #endif
- #ifndef _STRINGFUNCTIONS_H_
- #include "core/strings/stringFunctions.h"
- #endif
- #ifndef _SIMOBJECT_H_
- #include "console/simObject.h"
- #endif
- #ifndef _ENGINEFUNCTIONS_H_
- #include "console/engineFunctions.h"
- #endif
- // Whatever types are used in API definitions, their DECLAREs must be visible to the
- // macros. We include the basic primitive and struct types here.
- #ifndef _ENGINEPRIMITIVES_H_
- #include "console/enginePrimitives.h"
- #endif
- #ifndef _ENGINESTRUCTS_H_
- #include "console/engineStructs.h"
- #endif
- // Needed for the executef macros. Blame GCC.
- #ifndef _SIMEVENTS_H_
- #include "console/simEvents.h"
- #endif
- /// @file
- /// Definitions for exposing engine functionality to the control layer.
- ///
- /// This file provides a convenience layer around the underlying engine interop system (which at
- /// the moment still includes the legacy TorqueScript interop a.k.a. "console system"). The
- /// macros exposed here will automatically take care of all marshalling, value type constraints,
- /// reflection info instancing, etc. involved in defining engine API call-ins and call-outs.
- ///
- /// @note At the moment, this file supplies both the legacy TorqueScript console system as well
- /// as the new engine export system with the structures and information they need. In the
- /// near future, the console-based parts will get purged. This will not result in visible
- /// changes to users of the functionality here except for the string-based marshalling
- /// functions currently exposed (which will also disappear).
- //TODO: Disable warning for extern "C" functions returning UDTs for now; need to take a closer look at this
- #pragma warning( disable : 4190 )
- // Disable some VC warnings that are irrelevant to us.
- #pragma warning( push )
- #pragma warning( disable : 4510 ) // default constructor could not be generated; all the Args structures are never constructed by us
- #pragma warning( disable : 4610 ) // can never be instantiated; again Args is never constructed by us
- namespace engineAPI {
- /// Flag for enabling legacy console behavior in the interop system while
- /// we still have it around. Will disappear along with console.
- extern bool gUseConsoleInterop;
-
- /// Flag to allow engine functions to detect whether the engine had been
- /// initialized or shut down.
- extern bool gIsInitialized;
- }
- //FIXME: this allows const char* to be used as a struct field type
- // Temp support for allowing const char* to remain in the API functions as long as we
- // still have the console system around. When that is purged, these definitions should
- // be deleted and all const char* uses be replaced with String.
- template<> struct EngineTypeTraits< const char* > : public EngineTypeTraits< String > {};
- template<> inline const EngineTypeInfo* TYPE< const char* >() { return TYPE< String >(); }
- /// @name Marshalling
- ///
- /// Functions for converting to/from string-based data representations.
- ///
- /// @note This functionality is specific to the console interop.
- /// @{
- /// Marshal a single piece of data from native into client form.
- template< typename T >
- inline const char* EngineMarshallData( const T& value )
- {
- return castConsoleTypeToString( value );
- }
- inline const char* EngineMarshallData( bool value )
- {
- if( value )
- return "1";
- else
- return "0";
- }
- inline const char* EngineMarshallData( const char* str )
- {
- // The API assumes that if you pass a plain "const char*" through it, then you are referring
- // to string storage with non-local lifetime that can be safely passed to the control layer.
- return str;
- }
- template< typename T >
- inline const char* EngineMarshallData( T* object )
- {
- return ( object ? object->getIdString() : "0" );
- }
- template< typename T >
- inline const char* EngineMarshallData( const T* object )
- {
- return ( object ? object->getIdString() : "0" );
- }
- inline const char* EngineMarshallData( U32 value )
- {
- return EngineMarshallData( S32( value ) );
- }
- /// Marshal data from native into client form stored directly in
- /// client function invocation vector.
- template< typename T >
- inline void EngineMarshallData( const T& arg, S32& argc, ConsoleValueRef *argv )
- {
- argv[ argc ] = castConsoleTypeToString( arg );
- argc ++;
- }
- inline void EngineMarshallData( bool arg, S32& argc, ConsoleValueRef *argv )
- {
- if( arg )
- argv[ argc ] = 1;
- else
- argv[ argc ] = 0;
- argc ++;
- }
- inline void EngineMarshallData( S32 arg, S32& argc, ConsoleValueRef *argv )
- {
- argv[ argc ] = arg;
- argc ++;
- }
- inline void EngineMarshallData( U32 arg, S32& argc, ConsoleValueRef *argv )
- {
- EngineMarshallData( S32( arg ), argc, argv );
- }
- inline void EngineMarshallData( F32 arg, S32& argc, ConsoleValueRef *argv )
- {
- argv[ argc ] = arg;
- argc ++;
- }
- inline void EngineMarshallData( const char* arg, S32& argc, ConsoleValueRef *argv )
- {
- argv[ argc ] = arg;
- argc ++;
- }
- inline void EngineMarshallData( char* arg, S32& argc, ConsoleValueRef *argv )
- {
- argv[ argc ] = arg;
- argc ++;
- }
- template< typename T >
- inline void EngineMarshallData( T* object, S32& argc, ConsoleValueRef *argv )
- {
- argv[ argc ] = object ? object->getId() : 0;
- argc ++;
- }
- template< typename T >
- inline void EngineMarshallData( const T* object, S32& argc, ConsoleValueRef *argv )
- {
- argv[ argc ] = object ? object->getId() : 0;
- argc ++;
- }
- /// Unmarshal data from client form to engine form.
- ///
- /// This is wrapped in an a struct as partial specializations on function
- /// templates are not allowed in C++.
- template< typename T >
- struct EngineUnmarshallData
- {
- T operator()( const char* str ) const
- {
- T value;
- castConsoleTypeFromString( value, str );
- return value;
- }
- };
- template<>
- struct EngineUnmarshallData< S32 >
- {
- S32 operator()( ConsoleValueRef &ref ) const
- {
- return (S32)ref;
- }
- S32 operator()( const char* str ) const
- {
- return dAtoi( str );
- }
- };
- template<>
- struct EngineUnmarshallData< U32 >
- {
- U32 operator()( ConsoleValueRef &ref ) const
- {
- return (U32)((S32)ref);
- }
- U32 operator()( const char* str ) const
- {
- return dAtoui( str );
- }
- };
- template<>
- struct EngineUnmarshallData< F32 >
- {
- F32 operator()( ConsoleValueRef &ref ) const
- {
- return (F32)ref;
- }
- F32 operator()( const char* str ) const
- {
- return dAtof( str );
- }
- };
- template<>
- struct EngineUnmarshallData< U8 >
- {
- U8 operator()( ConsoleValueRef &ref ) const
- {
- return (U8)((S32)ref);
- }
- U8 operator()( const char* str ) const
- {
- return dAtoui( str );
- }
- };
- template<>
- struct EngineUnmarshallData< const char* >
- {
- const char* operator()( ConsoleValueRef &ref ) const
- {
- return ref.getStringValue();
- }
- const char* operator()( const char* str ) const
- {
- return str;
- }
- };
- template< typename T >
- struct EngineUnmarshallData< T* >
- {
- T* operator()( ConsoleValueRef &ref ) const
- {
- return dynamic_cast< T* >( Sim::findObject( ref.getStringValue() ) );
- }
- T* operator()( const char* str ) const
- {
- return dynamic_cast< T* >( Sim::findObject( str ) );
- }
- };
- template<>
- struct EngineUnmarshallData< void >
- {
- void operator()( ConsoleValueRef& ) const {}
- void operator()( const char* ) const {}
- };
- template<>
- struct EngineUnmarshallData< ConsoleValueRef >
- {
- ConsoleValueRef operator()( ConsoleValueRef ref ) const
- {
- return ref;
- }
- };
- /// @}
- /// @name C to C++ Trampolines
- ///
- /// The trampolines serve two purposes:
- ///
- /// For one, they ensure that no matter what argument types are specified by users of the engine API macros, the correct
- /// argument value types are enforced on the functions exported by the engine. Let's say, for example, the user writes
- /// a function that takes a "Point3F direction" argument, then the template machinery here will automatically expose an
- /// API function that takes a "Point3F& direction" argument.
- ///
- /// Secondly, the templates jump the incoming calls from extern "C" space into C++ space. This is mostly relevant for
- /// methods only as they will need an implicit object type argument.
- ///
- /// @{
- // Helper type to factor out commonalities between function and method trampolines.
- template<typename T> struct _EngineTrampoline {
- struct Args {};
- };
- template< typename R, typename ...ArgTs >
- struct _EngineTrampoline< R( ArgTs ... ) >
- {
- typedef std::tuple<ArgTs ...> Args;
- std::tuple<ArgTs ...> argT;
- typedef fixed_tuple<ArgTs ...> FixedArgs;
- fixed_tuple<ArgTs ...> fixedArgT;
- };
- template< typename T >
- struct _EngineFunctionTrampolineBase : public _EngineTrampoline< T >
- {
- typedef T FunctionType;
- };
- // Trampolines for any call-ins that aren't methods.
- template< typename T >
- struct _EngineFunctionTrampoline {};
- template< typename R, typename ...ArgTs >
- struct _EngineFunctionTrampoline< R(ArgTs...) > : public _EngineFunctionTrampolineBase< R(ArgTs...) >
- {
- private:
- using Super = _EngineFunctionTrampolineBase< R(ArgTs...) >;
- using ArgsType = typename Super::Args;
- using FixedArgsType = typename Super::FixedArgs;
-
- template<size_t ...> struct Seq {};
- template<size_t N, size_t ...S> struct Gens : Gens<N-1, N-1, S...> {};
- template<size_t ...I> struct Gens<0, I...>{ typedef Seq<I...> type; };
-
- template<size_t ...I>
- static R dispatchHelper(typename Super::FunctionType fn, const ArgsType& args, Seq<I...>) {
- return R( fn(std::get<I>(args) ...) );
- }
-
- template<size_t ...I>
- static R dispatchHelper(typename Super::FunctionType fn, const FixedArgsType& args, Seq<I...>) {
- return R( fn(fixed_tuple_accessor<I>::get(args) ...) );
- }
- using SeqType = typename Gens<sizeof...(ArgTs)>::type;
- public:
- static R jmp(typename Super::FunctionType fn, const ArgsType& args )
- {
- return dispatchHelper(fn, args, SeqType());
- }
- static R jmp(typename Super::FunctionType fn, const FixedArgsType& args )
- {
- return dispatchHelper(fn, args, SeqType());
- }
- };
- // Trampolines for engine methods
- template< typename T >
- struct _EngineMethodTrampolineBase : public _EngineTrampoline< T > {};
- template< typename Frame, typename T >
- struct _EngineMethodTrampoline {};
- template< typename Frame, typename R, typename ...ArgTs >
- struct _EngineMethodTrampoline< Frame, R(ArgTs ...) > : public _EngineMethodTrampolineBase< R(ArgTs ...) >
- {
- using FunctionType = R( typename Frame::ObjectType*, ArgTs ...);
- private:
- using Super = _EngineMethodTrampolineBase< R(ArgTs ...) >;
- using ArgsType = typename _EngineFunctionTrampolineBase< R(ArgTs ...) >::Args;
- using FixedArgsType = typename Super::FixedArgs;
-
- template<size_t ...> struct Seq {};
- template<size_t N, size_t ...S> struct Gens : Gens<N-1, N-1, S...> {};
- template<size_t ...I> struct Gens<0, I...>{ typedef Seq<I...> type; };
-
- template<size_t ...I>
- static R dispatchHelper(Frame f, const ArgsType& args, Seq<I...>) {
- return R( f._exec(std::get<I>(args) ...) );
- }
-
- template<size_t ...I>
- static R dispatchHelper(Frame f, const FixedArgsType& args, Seq<I...>) {
- return R( f._exec(fixed_tuple_accessor<I>::get(args) ...) );
- }
-
- using SeqType = typename Gens<sizeof...(ArgTs)>::type;
- public:
- static R jmp( typename Frame::ObjectType* object, const ArgsType& args )
- {
-
- Frame f;
- f.object = object;
- return dispatchHelper(f, args, SeqType());
- }
- static R jmp( typename Frame::ObjectType* object, const FixedArgsType& args )
- {
-
- Frame f;
- f.object = object;
- return dispatchHelper(f, args, SeqType());
- }
- };
- /// @}
- /// @name Thunking
- ///
- /// Internal functionality for thunks placed between TorqueScript calls of engine functions and their native
- /// implementations.
- ///
- /// @note The functionality in this group is specific to the console interop system.
- /// @{
- // Helper function to return data from a thunk.
- template< typename T >
- inline const char* _EngineConsoleThunkReturnValue( const T& value )
- {
- return EngineMarshallData( value );
- }
- inline bool _EngineConsoleThunkReturnValue( bool value )
- {
- return value;
- }
- inline S32 _EngineConsoleThunkReturnValue( S32 value )
- {
- return value;
- }
- inline F32 _EngineConsoleThunkReturnValue( F32 value )
- {
- return value;
- }
- inline const char* _EngineConsoleThunkReturnValue( const String& str )
- {
- return Con::getReturnBuffer( str );
- }
- inline const char* _EngineConsoleThunkReturnValue( const char* value )
- {
- return EngineMarshallData( value );
- }
- template< typename T >
- inline const char* _EngineConsoleThunkReturnValue( T* value )
- {
- return ( value ? value->getIdString() : "" );
- }
- template< typename T >
- inline const char* _EngineConsoleThunkReturnValue( const T* value )
- {
- return ( value ? value->getIdString() : "" );
- }
- // Helper class to determine the type of callback registered with the console system.
- template< typename R >
- struct _EngineConsoleThunkType
- {
- typedef const char* ReturnType;
- typedef StringCallback CallbackType;
- };
- template<>
- struct _EngineConsoleThunkType< S32 >
- {
- typedef S32 ReturnType;
- typedef IntCallback CallbackType;
- };
- template<>
- struct _EngineConsoleThunkType< U32 >
- {
- typedef U32 ReturnType;
- typedef IntCallback CallbackType;
- };
- template<>
- struct _EngineConsoleThunkType< F32 >
- {
- typedef F32 ReturnType;
- typedef FloatCallback CallbackType;
- };
- template<>
- struct _EngineConsoleThunkType< bool >
- {
- typedef bool ReturnType;
- typedef BoolCallback CallbackType;
- };
- template<>
- struct _EngineConsoleThunkType< void >
- {
- typedef void ReturnType;
- typedef VoidCallback CallbackType;
- };
- // Helper struct to count the number of parameters in a function list.
- // The setup through operator () allows omitting the the argument list entirely.
- struct _EngineConsoleThunkCountArgs
- {
- template<typename ...ArgTs> U32 operator()(ArgTs... args){
- return sizeof...(ArgTs);
- }
-
- operator U32() const{ // FIXME: WHAT IS THIS?? I'm pretty sure it's incorrect, and it's the version that is invoked by all the macros
- return 0;
- }
- };
- // Encapsulation of a legacy console function invocation.
- namespace engineAPI{
- namespace detail{
- template<S32 startArgc, typename R, typename ...ArgTs>
- struct ThunkHelpers {
- using SelfType = ThunkHelpers<startArgc, R, ArgTs...>;
- using FunctionType = R(*)(ArgTs...);
- template<typename Frame> using MethodType = R(Frame::*)(ArgTs ...) const;
- template<size_t I> using IthArgType = typename std::tuple_element<I, std::tuple<ArgTs ...> >::type;
-
- template<size_t ...> struct Seq {};
- template<size_t N, size_t ...S> struct Gens : Gens<N-1, N-1, S...> {};
- template<size_t ...I> struct Gens<0, I...>{ typedef Seq<I...> type; };
-
- typedef typename _EngineConsoleThunkType< R >::ReturnType ReturnType;
- static const S32 NUM_ARGS = sizeof...(ArgTs) + startArgc;
-
- template<size_t index, size_t method_offset = 0, typename ...RealArgTs>
- static IthArgType<index> getRealArgValue(S32 argc, ConsoleValueRef *argv, const _EngineFunctionDefaultArguments< void(RealArgTs...) >& defaultArgs)
- {
- if((startArgc + index) < argc)
- {
- return EngineUnmarshallData< IthArgType<index> >()( argv[ startArgc + index ] );
- } else {
- return std::get<index + method_offset>(defaultArgs.mArgs);
- }
- }
-
- template<size_t ...I>
- static R dispatchHelper(S32 argc, ConsoleValueRef *argv, FunctionType fn, const _EngineFunctionDefaultArguments< void(ArgTs...) >& defaultArgs, Seq<I...>){
- return fn(SelfType::getRealArgValue<I>(argc, argv, defaultArgs) ...);
- }
-
- template<typename Frame, size_t ...I>
- static R dispatchHelper(S32 argc, ConsoleValueRef *argv, MethodType<Frame> fn, Frame* frame, const _EngineFunctionDefaultArguments< void( typename Frame::ObjectType*, ArgTs...) >& defaultArgs, Seq<I...>){
- return (frame->*fn)(SelfType::getRealArgValue<I,1>(argc, argv, defaultArgs) ...);
- }
-
- using SeqType = typename Gens<sizeof...(ArgTs)>::type;
- };
-
- template<typename ArgVT> struct MarshallHelpers {
- template<typename ...ArgTs> static void marshallEach(S32 &argc, ArgVT *argv, const ArgTs& ...args){}
- template<typename H, typename ...Tail> static void marshallEach(S32 &argc, ArgVT *argv, const H& head, const Tail& ...tail){
- argv[argc++] = EngineMarshallData(head);
- marshallEach(argc, argv, tail...);
- }
- };
-
- template<> struct MarshallHelpers<ConsoleValueRef> {
- template<typename ...ArgTs> static void marshallEach(S32 &argc, ConsoleValueRef *argv, const ArgTs& ...args){}
- template<typename H, typename ...Tail> static void marshallEach(S32 &argc, ConsoleValueRef *argv, const H& head, const Tail& ...tail){
- EngineMarshallData(head, argc, argv);
- marshallEach(argc, argv, tail...);
- }
- };
- }
- }
- template< S32 startArgc, typename T >
- struct _EngineConsoleThunk {};
- template< S32 startArgc, typename R, typename ...ArgTs >
- struct _EngineConsoleThunk< startArgc, R(ArgTs...) >
- {
- private:
- using Helper = engineAPI::detail::ThunkHelpers<startArgc, R, ArgTs...>;
- using SeqType = typename Helper::SeqType;
- public:
- typedef typename Helper::FunctionType FunctionType;
- typedef typename Helper::ReturnType ReturnType;
- template<typename Frame> using MethodType = typename Helper::template MethodType<Frame>;
- static const S32 NUM_ARGS = Helper::NUM_ARGS;
-
- static ReturnType thunk( S32 argc, ConsoleValueRef *argv, FunctionType fn, const _EngineFunctionDefaultArguments< void(ArgTs...) >& defaultArgs)
- {
- return _EngineConsoleThunkReturnValue( Helper::dispatchHelper(argc, argv, fn, defaultArgs, SeqType()));
- }
- template< typename Frame >
- static ReturnType thunk( S32 argc, ConsoleValueRef *argv, MethodType<Frame> fn, Frame* frame, const _EngineFunctionDefaultArguments< void( typename Frame::ObjectType*, ArgTs...) >& defaultArgs)
- {
- return _EngineConsoleThunkReturnValue( Helper::dispatchHelper(argc, argv, fn, frame, defaultArgs, SeqType()));
- }
- };
- // Have to do a partial specialization for void-returning functions :(
- template<S32 startArgc, typename ...ArgTs>
- struct _EngineConsoleThunk<startArgc, void(ArgTs...)> {
- private:
- using Helper = engineAPI::detail::ThunkHelpers<startArgc, void, ArgTs...>;
- using SeqType = typename Helper::SeqType;
- public:
- typedef typename Helper::FunctionType FunctionType;
- typedef typename Helper::ReturnType ReturnType;
- template<typename Frame> using MethodType = typename Helper::template MethodType<Frame>;
- static const S32 NUM_ARGS = Helper::NUM_ARGS;
-
- static void thunk( S32 argc, ConsoleValueRef *argv, FunctionType fn, const _EngineFunctionDefaultArguments< void(ArgTs...) >& defaultArgs)
- {
- Helper::dispatchHelper(argc, argv, fn, defaultArgs, SeqType());
- }
- template< typename Frame >
- static void thunk( S32 argc, ConsoleValueRef *argv, MethodType<Frame> fn, Frame* frame, const _EngineFunctionDefaultArguments< void( typename Frame::ObjectType*, ArgTs...) >& defaultArgs)
- {
- Helper::dispatchHelper(argc, argv, fn, frame, defaultArgs, SeqType());
- }
- };
- /// @}
- /// @name API Definition Macros
- ///
- /// The macros in this group allow to create engine API functions that work both with the
- /// legacy console system as well as with the new engine export system. As such, they only
- /// support those function features that are available in both systems. This means that for
- /// console-style variadic functions, the ConsoleXXX must be used and that for overloaded
- /// and/or C-style variadic functions as well as for placing functions in export scopes,
- /// DEFINE_CALLIN must be used directly.
- ///
- /// When the console system is removed, the console thunking functionality will be removed
- /// from these macros but otherwise they will remain unchanged and in place.
- ///
- /// @{
- // Helpers to implement initialization checks. Pulled out into separate macros so this can be deactivated easily.
- // Especially important for the initialize() function itself.
- #define _CHECK_ENGINE_INITIALIZED_IMPL( fnName, returnType ) \
- if( !engineAPI::gIsInitialized ) \
- { \
- Con::errorf( "EngineAPI: Engine not initialized when calling " #fnName ); \
- return EngineTypeTraits< returnType >::ReturnValue( EngineTypeTraits< returnType >::ReturnValueType() ); \
- }
- #define _CHECK_ENGINE_INITIALIZED( fnName, returnType ) _CHECK_ENGINE_INITIALIZED_IMPL( fnName, returnType )
- /// Define a call-in point for calling into the engine.
- ///
- /// @param name The name of the function as it should be seen by the control layer.
- /// @param returnType The value type returned to the control layer.
- /// @param args The argument list as it would appear on the function definition
- /// @param defaultArgs The list of default argument values.
- /// @param usage The usage doc string for the engine API reference.
- ///
- /// @code
- /// DefineEngineFunction( myFunction, int, ( float f, const String& s ), ( "value for s" ), "This is my function." )
- /// {
- /// return int( f ) + dAtoi( s );
- /// }
- /// @endcode
- #define DefineEngineFunction( name, returnType, args, defaultArgs, usage ) \
- static inline returnType _fn ## name ## impl args; \
- TORQUE_API EngineTypeTraits< returnType >::ReturnValueType fn ## name \
- ( _EngineFunctionTrampoline< returnType args >::FixedArgs a ) \
- { \
- _CHECK_ENGINE_INITIALIZED( name, returnType ); \
- return EngineTypeTraits< returnType >::ReturnValue( \
- _EngineFunctionTrampoline< returnType args >::jmp( _fn ## name ## impl, a ) \
- ); \
- } \
- static _EngineFunctionDefaultArguments< void args > _fn ## name ## DefaultArgs defaultArgs; \
- static EngineFunctionInfo _fn ## name ## FunctionInfo( \
- #name, \
- &_SCOPE<>()(), \
- usage, \
- #returnType " " #name #args, \
- "fn" #name, \
- TYPE< returnType args >(), \
- &_fn ## name ## DefaultArgs, \
- ( void* ) &fn ## name, \
- 0 \
- ); \
- static _EngineConsoleThunkType< returnType >::ReturnType _ ## name ## caster( SimObject*, S32 argc, ConsoleValueRef *argv ) \
- { \
- return _EngineConsoleThunkType< returnType >::ReturnType( _EngineConsoleThunk< 1, returnType args >::thunk( \
- argc, argv, &_fn ## name ## impl, _fn ## name ## DefaultArgs \
- ) ); \
- } \
- static ConsoleFunctionHeader _ ## name ## header \
- ( #returnType, #args, #defaultArgs ); \
- static ConsoleConstructor \
- _ ## name ## obj( NULL, #name, _EngineConsoleThunkType< returnType >::CallbackType( _ ## name ## caster ), usage, \
- _EngineConsoleThunk< 1, returnType args >::NUM_ARGS - _EngineConsoleThunkCountArgs() defaultArgs, \
- _EngineConsoleThunk< 1, returnType args >::NUM_ARGS, \
- false, &_ ## name ## header \
- ); \
- static inline returnType _fn ## name ## impl args
-
-
- // The next thing is a bit tricky. DefineEngineMethod allows to make the 'object' (=this) argument to the function
- // implicit which presents quite an obstacle for the macro internals as the engine export system requires the
- // name of a DLL symbol that represents an extern "C" function with an explicit first object pointer argument.
- //
- // Even if we ignored the fact that we don't have a guarantee how the various C++ compilers implement implicit 'this' arguments,
- // we could still not just use a C++ method for this as then we would have to get past the C++ compiler's mangling to
- // get to the function symbol name (let alone the fact that typing this method correctly would be tricky).
- //
- // So, the trick employed here is to package all but the implicit 'this' argument in a structure and then define an
- // extern "C" function that takes the object pointer as a first argument and the struct type as the second argument.
- // This will result in a function with an identical stack call frame layout to the function we want.
- //
- // Unfortunately, that still requires that function to chain on to the real user-defined function. To do this
- // cleanly and portably, _EngineMethodTrampoline is used to unpack and jump the call from extern "C" into C++ space.
- // In optimized builds, the compiler should be smart enough to pretty much optimize all our trickery here away.
- #define _DefineMethodTrampoline( className, name, returnType, args ) \
- TORQUE_API EngineTypeTraits< returnType >::ReturnValueType \
- fn ## className ## _ ## name ( className* object, _EngineMethodTrampoline< _ ## className ## name ## frame, returnType args >::FixedArgs a )\
- { \
- _CHECK_ENGINE_INITIALIZED( className::name, returnType ); \
- return EngineTypeTraits< returnType >::ReturnValue( \
- _EngineMethodTrampoline< _ ## className ## name ## frame, returnType args >::jmp( object, a ) \
- ); \
- }
- /// Define a call-in point for calling a method on an engine object.
- ///
- /// @param name The name of the C++ class.
- /// @param name The name of the method as it should be seen by the control layer.
- /// @param returnType The value type returned to the control layer.
- /// @param args The argument list as it would appear on the function definition
- /// @param defaultArgs The list of default argument values.
- /// @param usage The usage doc string for the engine API reference.
- ///
- /// @code
- /// DefineEngineMethod( MyClass, myMethod, int, ( float f, const String& s ), ( "value for s" ), "This is my method." )
- /// {
- /// return object->someMethod( f, s );
- /// }
- /// @endcode
- #define DefineEngineMethod( className, name, returnType, args, defaultArgs, usage ) \
- struct _ ## className ## name ## frame \
- { \
- typedef className ObjectType; \
- className* object; \
- inline returnType _exec args const; \
- }; \
- _DefineMethodTrampoline( className, name, returnType, args ); \
- static _EngineFunctionDefaultArguments< _EngineMethodTrampoline< _ ## className ## name ## frame, void args >::FunctionType > \
- _fn ## className ## name ## DefaultArgs defaultArgs; \
- static EngineFunctionInfo _fn ## className ## name ## FunctionInfo( \
- #name, \
- &_SCOPE< className >()(), \
- usage, \
- "virtual " #returnType " " #name #args, \
- "fn" #className "_" #name, \
- TYPE< _EngineMethodTrampoline< _ ## className ## name ## frame, returnType args >::FunctionType >(), \
- &_fn ## className ## name ## DefaultArgs, \
- ( void* ) &fn ## className ## _ ## name, \
- 0 \
- ); \
- static _EngineConsoleThunkType< returnType >::ReturnType _ ## className ## name ## caster( SimObject* object, S32 argc, ConsoleValueRef *argv ) \
- { \
- _ ## className ## name ## frame frame; \
- frame.object = static_cast< className* >( object ); \
- return _EngineConsoleThunkType< returnType >::ReturnType( _EngineConsoleThunk< 2, returnType args >::thunk( \
- argc, argv, &_ ## className ## name ## frame::_exec, &frame, _fn ## className ## name ## DefaultArgs \
- ) ); \
- } \
- static ConsoleFunctionHeader _ ## className ## name ## header \
- ( #returnType, #args, #defaultArgs ); \
- static ConsoleConstructor \
- className ## name ## obj( #className, #name, \
- _EngineConsoleThunkType< returnType >::CallbackType( _ ## className ## name ## caster ), usage, \
- _EngineConsoleThunk< 2, returnType args >::NUM_ARGS - _EngineConsoleThunkCountArgs() defaultArgs, \
- _EngineConsoleThunk< 2, returnType args >::NUM_ARGS, \
- false, &_ ## className ## name ## header \
- ); \
- returnType _ ## className ## name ## frame::_exec args const
-
-
- /// Define a call-in point for calling into the engine. Unlike with DefineEngineFunction, the statically
- /// callable function will be confined to the namespace of the given class.
- ///
- /// @param name The name of the C++ class (or a registered export scope).
- /// @param name The name of the method as it should be seen by the control layer.
- /// @param returnType The value type returned to the control layer.
- /// @param args The argument list as it would appear on the function definition
- /// @param defaultArgs The list of default argument values.
- /// @param usage The usage doc string for the engine API reference.
- ///
- /// @code
- /// DefineEngineStaticMethod( MyClass, myMethod, int, ( float f, string s ), ( "value for s" ), "This is my method." )
- /// {
- /// }
- /// @endcode
- #define DefineEngineStaticMethod( className, name, returnType, args, defaultArgs, usage ) \
- static inline returnType _fn ## className ## name ## impl args; \
- TORQUE_API EngineTypeTraits< returnType >::ReturnValueType fn ## className ## _ ## name \
- ( _EngineFunctionTrampoline< returnType args >::FixedArgs a ) \
- { \
- _CHECK_ENGINE_INITIALIZED( className::name, returnType ); \
- return EngineTypeTraits< returnType >::ReturnValue( \
- _EngineFunctionTrampoline< returnType args >::jmp( _fn ## className ## name ## impl, a ) \
- ); \
- } \
- static _EngineFunctionDefaultArguments< void args > _fn ## className ## name ## DefaultArgs defaultArgs; \
- static EngineFunctionInfo _fn ## name ## FunctionInfo( \
- #name, \
- &_SCOPE< className >()(), \
- usage, \
- #returnType " " #name #args, \
- "fn" #className "_" #name, \
- TYPE< returnType args >(), \
- &_fn ## className ## name ## DefaultArgs, \
- ( void* ) &fn ## className ## _ ## name, \
- 0 \
- ); \
- static _EngineConsoleThunkType< returnType >::ReturnType _ ## className ## name ## caster( SimObject*, S32 argc, ConsoleValueRef *argv )\
- { \
- return _EngineConsoleThunkType< returnType >::ReturnType( _EngineConsoleThunk< 1, returnType args >::thunk( \
- argc, argv, &_fn ## className ## name ## impl, _fn ## className ## name ## DefaultArgs \
- ) ); \
- } \
- static ConsoleFunctionHeader _ ## className ## name ## header \
- ( #returnType, #args, #defaultArgs, true ); \
- static ConsoleConstructor \
- _ ## className ## name ## obj( #className, #name, _EngineConsoleThunkType< returnType >::CallbackType( _ ## className ## name ## caster ), usage, \
- _EngineConsoleThunk< 1, returnType args >::NUM_ARGS - _EngineConsoleThunkCountArgs() defaultArgs, \
- _EngineConsoleThunk< 1, returnType args >::NUM_ARGS, \
- false, &_ ## className ## name ## header \
- ); \
- static inline returnType _fn ## className ## name ## impl args
- # define DefineEngineStringlyVariadicFunction(name,returnType,minArgs,maxArgs,usage) \
- static inline returnType _fn ## name ## impl (SimObject *, S32 argc, ConsoleValueRef *argv); \
- TORQUE_API EngineTypeTraits< returnType >::ReturnValueType fn ## name \
- (S32 argc, const char** argv) \
- { \
- _CHECK_ENGINE_INITIALIZED( name, returnType ); \
- StringStackConsoleWrapper args(argc, argv); \
- return EngineTypeTraits< returnType >::ReturnValue( \
- _fn ## name ## impl(NULL, args.count(), args) \
- ); \
- } \
- static _EngineFunctionDefaultArguments< void (S32 argc, const char** argv) > _fn ## name ## DefaultArgs; \
- static EngineFunctionInfo _fn ## name ## FunctionInfo( \
- #name, \
- &_SCOPE<>()(), \
- usage, \
- #returnType " " #name "(S32 argc, const char** argv)", \
- "fn" #name, \
- TYPE< returnType (S32 argc, const char** argv) >(), \
- &_fn ## name ## DefaultArgs, \
- ( void* ) &fn ## name, \
- 0 \
- ); \
- ConsoleConstructor cc_##name##_obj(NULL,#name,_fn ## name ## impl,usage,minArgs,maxArgs); \
- returnType _fn ## name ## impl(SimObject *, S32 argc, ConsoleValueRef *argv)
- # define DefineEngineStringlyVariadicMethod(className, name,returnType,minArgs,maxArgs,usage) \
- static inline returnType _fn ## className ## _ ## name ## impl (className* object, S32 argc, ConsoleValueRef* argv); \
- TORQUE_API EngineTypeTraits< returnType >::ReturnValueType fn ## className ## _ ## name \
- (className* object, S32 argc, const char** argv) \
- { \
- _CHECK_ENGINE_INITIALIZED( name, returnType ); \
- StringStackConsoleWrapper args(argc, argv); \
- return EngineTypeTraits< returnType >::ReturnValue( \
- _fn ## className ## _ ## name ## impl(object, args.count(), args) \
- ); \
- } \
- static _EngineFunctionDefaultArguments< void (className* object, S32 argc, const char** argv) > _fn ## className ## _ ## name ## DefaultArgs; \
- static EngineFunctionInfo _fn ## className ## _ ## name ## FunctionInfo( \
- #name, \
- &_SCOPE<>()(), \
- usage, \
- #returnType " " #name "(SimObject* object, S32 argc, const char** argv)", \
- "fn" #className "_" #name, \
- TYPE< returnType (SimObject* object, S32 argc, const char** argv) >(), \
- &_fn ## className ## _ ## name ## DefaultArgs, \
- ( void* ) &fn ## className ## _ ## name, \
- 0 \
- ); \
- returnType cm_##className##_##name##_caster(SimObject* object, S32 argc, ConsoleValueRef* argv) { \
- AssertFatal( dynamic_cast<className*>( object ), "Object passed to " #name " is not a " #className "!" ); \
- conmethod_return_##returnType ) _fn ## className ## _ ## name ## impl(static_cast<className*>(object),argc,argv); \
- }; \
- ConsoleConstructor cc_##className##_##name##_obj(#className,#name,cm_##className##_##name##_caster,usage,minArgs,maxArgs); \
- static inline returnType _fn ## className ## _ ## name ## impl(className *object, S32 argc, ConsoleValueRef *argv)
- // The following three macros are only temporary. They allow to define engineAPI functions using the framework
- // here in this file while being visible only in the new API. When the console interop is removed, these macros
- // can be removed and all their uses be replaced with their corresponding versions that now still include support
- // for the console (e.g. DefineNewEngineFunction should become DefineEngineFunction).
- #define DefineNewEngineFunction( name, returnType, args, defaultArgs, usage ) \
- static inline returnType _fn ## name ## impl args; \
- TORQUE_API EngineTypeTraits< returnType >::ReturnValueType fn ## name \
- ( _EngineFunctionTrampoline< returnType args >::FixedArgs a ) \
- { \
- _CHECK_ENGINE_INITIALIZED( name, returnType ); \
- return EngineTypeTraits< returnType >::ReturnValue( \
- _EngineFunctionTrampoline< returnType args >::jmp( _fn ## name ## impl, a ) \
- ); \
- } \
- static _EngineFunctionDefaultArguments< void args > _fn ## name ## DefaultArgs defaultArgs; \
- static EngineFunctionInfo _fn ## name ## FunctionInfo( \
- #name, \
- &_SCOPE<>()(), \
- usage, \
- #returnType " " #name #args, \
- "fn" #name, \
- TYPE< returnType args >(), \
- &_fn ## name ## DefaultArgs, \
- ( void* ) &fn ## name, \
- 0 \
- ); \
- static inline returnType _fn ## name ## impl args
- #define DefineNewEngineMethod( className, name, returnType, args, defaultArgs, usage ) \
- struct _ ## className ## name ## frame \
- { \
- typedef className ObjectType; \
- className* object; \
- inline returnType _exec args const; \
- }; \
- _DefineMethodTrampoline( className, name, returnType, args ); \
- static _EngineFunctionDefaultArguments< _EngineMethodTrampoline< _ ## className ## name ## frame, void args >::FunctionType > \
- _fn ## className ## name ## DefaultArgs defaultArgs; \
- static EngineFunctionInfo _fn ## className ## name ## FunctionInfo( \
- #name, \
- &_SCOPE< className >()(), \
- usage, \
- "virtual " #returnType " " #name #args, \
- "fn" #className "_" #name, \
- TYPE< _EngineMethodTrampoline< _ ## className ## name ## frame, returnType args >::FunctionType >(), \
- &_fn ## className ## name ## DefaultArgs, \
- ( void* ) &fn ## className ## _ ## name, \
- 0 \
- ); \
- returnType _ ## className ## name ## frame::_exec args const
- #define DefineNewEngineStaticMethod( className, name, returnType, args, defaultArgs, usage ) \
- static inline returnType _fn ## className ## name ## impl args; \
- TORQUE_API EngineTypeTraits< returnType >::ReturnValueType fn ## className ## _ ## name \
- ( _EngineFunctionTrampoline< returnType args >::FixedArgs a ) \
- { \
- _CHECK_ENGINE_INITIALIZED( className::name, returnType ); \
- return EngineTypeTraits< returnType >::ReturnValue( \
- _EngineFunctionTrampoline< returnType args >::jmp( _fn ## className ## name ## impl, a ) \
- ); \
- } \
- static _EngineFunctionDefaultArguments< void args > _fn ## className ## name ## DefaultArgs defaultArgs; \
- static EngineFunctionInfo _fn ## name ## FunctionInfo( \
- #name, \
- &_SCOPE< className >()(), \
- usage, \
- #returnType " " #name #args, \
- "fn" #className "_" #name, \
- TYPE< returnType args >(), \
- &_fn ## className ## name ## DefaultArgs, \
- ( void* ) &fn ## className ## _ ## name, \
- 0 \
- ); \
- static inline returnType _fn ## className ## name ## impl args
- /// @}
- //=============================================================================
- // Callbacks.
- //=============================================================================
- /// Matching implement for DECLARE_CALLBACK.
- ///
- ///
- /// @warn With the new interop system, method-style callbacks <em>must not</em> be triggered on object
- /// that are being created! This is because the control layer will likely not yet have a fully valid wrapper
- /// object in place for the EngineObject under construction.
- #define IMPLEMENT_CALLBACK( class, name, returnType, args, argNames, usageString ) \
- struct _ ## class ## name ## frame { typedef class ObjectType; }; \
- TORQUE_API _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType* cb ## class ## _ ## name; \
- TORQUE_API void set_cb ## class ## _ ## name( \
- _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType fn ) \
- { cb ## class ## _ ## name = fn; } \
- _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType* cb ## class ## _ ## name; \
- namespace { \
- ::EngineFunctionInfo _cb ## class ## name( \
- #name, \
- &::_SCOPE< class >()(), \
- usageString, \
- "virtual " #returnType " " #name #args, \
- "cb" #class "_" #name, \
- ::TYPE< _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType >(), \
- NULL, \
- ( void* ) &cb ## class ## _ ## name, \
- EngineFunctionCallout \
- ); \
- } \
- returnType class::name ## _callback args \
- { \
- if( cb ## class ## _ ## name ) { \
- _EngineCallbackHelper cbh( this, reinterpret_cast< const void* >( cb ## class ## _ ## name ) ); \
- return returnType( cbh.call< returnType > argNames ); \
- } \
- if( engineAPI::gUseConsoleInterop ) \
- { \
- static StringTableEntry sName = StringTable->insert( #name ); \
- _EngineConsoleCallbackHelper cbh( sName, this ); \
- return returnType( cbh.call< returnType > argNames ); \
- } \
- return returnType(); \
- } \
- namespace { \
- ConsoleFunctionHeader _ ## class ## name ## header( \
- #returnType, #args, "" ); \
- ConsoleConstructor _ ## class ## name ## obj( #class, #name, usageString, &_ ## class ## name ## header ); \
- }
- /// Used to define global callbacks not associated with
- /// any particular class or namespace.
- #define IMPLEMENT_GLOBAL_CALLBACK( name, returnType, args, argNames, usageString ) \
- DEFINE_CALLOUT( cb ## name, name,, returnType, args, 0, usageString ); \
- returnType name ## _callback args \
- { \
- if( cb ## name ) \
- return returnType( cb ## name argNames ); \
- if( engineAPI::gUseConsoleInterop ) \
- { \
- static StringTableEntry sName = StringTable->insert( #name ); \
- _EngineConsoleCallbackHelper cbh( sName, NULL ); \
- return returnType( cbh.call< returnType > argNames ); \
- } \
- return returnType(); \
- } \
- namespace { \
- ConsoleFunctionHeader _ ## name ## header( \
- #returnType, #args, "" ); \
- ConsoleConstructor _ ## name ## obj( NULL, #name, usageString, &_ ## name ## header ); \
- }
-
-
- // Again, temporary macros to allow splicing the API while we still have the console interop around.
- #define IMPLEMENT_CONSOLE_CALLBACK( class, name, returnType, args, argNames, usageString ) \
- returnType class::name ## _callback args \
- { \
- if( engineAPI::gUseConsoleInterop ) \
- { \
- static StringTableEntry sName = StringTable->insert( #name ); \
- _EngineConsoleCallbackHelper cbh( sName, this ); \
- return returnType( cbh.call< returnType > argNames ); \
- } \
- return returnType(); \
- } \
- namespace { \
- ConsoleFunctionHeader _ ## class ## name ## header( \
- #returnType, #args, "" ); \
- ConsoleConstructor _ ## class ## name ## obj( #class, #name, usageString, &_ ## class ## name ## header ); \
- }
-
- #define IMPLEMENT_NEW_CALLBACK( class, name, returnType, args, argNames, usageString ) \
- struct _ ## class ## name ## frame { typedef class ObjectType; }; \
- TORQUE_API _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType* cb ## class ## _ ## name; \
- TORQUE_API void set_cb ## class ## _ ## name( \
- _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType fn ) \
- { cb ## class ## _ ## name = fn; } \
- _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType* cb ## class ## _ ## name; \
- namespace { \
- ::EngineFunctionInfo _cb ## class ## name( \
- #name, \
- &::_SCOPE< class >()(), \
- usageString, \
- "virtual " #returnType " " #name #args, \
- "cb" #class "_" #name, \
- ::TYPE< _EngineMethodTrampoline< _ ## class ## name ## frame, returnType args >::FunctionType >(), \
- NULL, \
- &cb ## class ## _ ## name, \
- EngineFunctionCallout \
- ); \
- } \
- returnType class::name ## _callback args \
- { \
- if( cb ## class ## _ ## name ) { \
- _EngineCallbackHelper cbh( this, reinterpret_cast< const void* >( cb ## class ## _ ## name ) ); \
- return returnType( cbh.call< returnType > argNames ); \
- } \
- return returnType(); \
- }
- // Internal helper class for doing call-outs in the new interop.
- struct _EngineCallbackHelper
- {
- protected:
-
- EngineObject* mThis;
- const void* mFn;
-
- public:
- _EngineCallbackHelper( EngineObject* pThis, const void* fn )
- : mThis( pThis ),
- mFn( fn ) {}
-
- template< typename R, typename ...ArgTs >
- R call(ArgTs ...args) const
- {
- typedef R( FunctionType )( EngineObject*, ArgTs... );
- return R( reinterpret_cast< FunctionType* >( const_cast<void*>(mFn) )( mThis, args... ) );
- }
- };
- #include "console/stringStack.h"
- // Internal helper for callback support in legacy console system.
- struct _BaseEngineConsoleCallbackHelper
- {
- public:
- /// Matches up to storeArgs.
- static const U32 MAX_ARGUMENTS = 11;
- SimObject* mThis;
- S32 mInitialArgc;
- S32 mArgc;
- StringTableEntry mCallbackName;
- ConsoleValueRef mArgv[ MAX_ARGUMENTS + 2 ];
- ConsoleValueRef _exec();
- ConsoleValueRef _execLater(SimConsoleThreadExecEvent *evt);
- _BaseEngineConsoleCallbackHelper() {;}
- };
- // Base helper for console callbacks
- struct _EngineConsoleCallbackHelper : public _BaseEngineConsoleCallbackHelper
- {
- private:
- using Helper = engineAPI::detail::MarshallHelpers<ConsoleValueRef>;
- public:
- _EngineConsoleCallbackHelper( StringTableEntry callbackName, SimObject* pThis )
- {
- mThis = pThis;
- mArgc = mInitialArgc = pThis ? 2 : 1 ;
- mCallbackName = callbackName;
- }
-
- template< typename R, typename ...ArgTs >
- R call(ArgTs ...args)
- {
- if (Con::isMainThread())
- {
- ConsoleStackFrameSaver sav; sav.save();
- CSTK.reserveValues(mArgc + sizeof...(ArgTs), mArgv);
- mArgv[ 0 ].value->setStackStringValue(mCallbackName);
-
- Helper::marshallEach(mArgc, mArgv, args...);
-
- return R( EngineUnmarshallData< R >()( _exec() ) );
- }
- else
- {
- SimConsoleThreadExecCallback cb;
- SimConsoleThreadExecEvent *evt = new SimConsoleThreadExecEvent(mArgc + sizeof...(ArgTs), NULL, false, &cb);
- evt->populateArgs(mArgv);
- mArgv[ 0 ].value->setStackStringValue(mCallbackName);
-
- Helper::marshallEach(mArgc, mArgv, args...);
-
- Sim::postEvent((SimObject*)Sim::getRootGroup(), evt, Sim::getCurrentTime());
- return R( EngineUnmarshallData< R >()( cb.waitForResult() ) );
- }
- }
-
- };
- // Override for when first parameter is presumably a SimObject*, in which case A will be absorbed as the callback
- template<typename P1> struct _EngineConsoleExecCallbackHelper : public _BaseEngineConsoleCallbackHelper
- {
- private:
- using Helper = engineAPI::detail::MarshallHelpers<ConsoleValueRef>;
- public:
- _EngineConsoleExecCallbackHelper( SimObject* pThis )
- {
- mThis = pThis;
- mArgc = mInitialArgc = 2;
- mCallbackName = NULL;
- }
-
- template< typename R, typename SCB, typename ...ArgTs >
- R call( SCB simCB , ArgTs ...args )
- {
- if (Con::isMainThread())
- {
- ConsoleStackFrameSaver sav; sav.save();
- CSTK.reserveValues(mArgc+sizeof...(ArgTs), mArgv);
- mArgv[ 0 ].value->setStackStringValue(simCB);
- Helper::marshallEach(mArgc, mArgv, args...);
- return R( EngineUnmarshallData< R >()( _exec() ) );
- }
- else
- {
- SimConsoleThreadExecCallback cb;
- SimConsoleThreadExecEvent *evt = new SimConsoleThreadExecEvent(mArgc+sizeof...(ArgTs), NULL, true, &cb);
- evt->populateArgs(mArgv);
- mArgv[ 0 ].value->setStackStringValue(simCB);
-
- Helper::marshallEach(mArgc, mArgv, args...);
- Sim::postEvent(mThis, evt, Sim::getCurrentTime());
- return R( EngineUnmarshallData< R >()( cb.waitForResult() ) );
- }
- }
- };
- // Override for when first parameter is const char*
- template<> struct _EngineConsoleExecCallbackHelper<const char*> : public _BaseEngineConsoleCallbackHelper
- {
- private:
- using Helper = engineAPI::detail::MarshallHelpers<ConsoleValueRef>;
- public:
- _EngineConsoleExecCallbackHelper( const char *callbackName )
- {
- mThis = NULL;
- mArgc = mInitialArgc = 1;
- mCallbackName = StringTable->insert(callbackName);
- }
- template< typename R, typename ...ArgTs >
- R call(ArgTs ...args)
- {
- if (Con::isMainThread())
- {
- ConsoleStackFrameSaver sav; sav.save();
- CSTK.reserveValues(mArgc+sizeof...(ArgTs), mArgv);
- mArgv[ 0 ].value->setStackStringValue(mCallbackName);
-
- Helper::marshallEach(mArgc, mArgv, args...);
-
- return R( EngineUnmarshallData< R >()( _exec() ) );
- }
- else
- {
- SimConsoleThreadExecCallback cb;
- SimConsoleThreadExecEvent *evt = new SimConsoleThreadExecEvent(mArgc+sizeof...(ArgTs), NULL, false, &cb);
- evt->populateArgs(mArgv);
- mArgv[ 0 ].value->setStackStringValue(mCallbackName);
-
- Helper::marshallEach(mArgc, mArgv, args...);
- Sim::postEvent((SimObject*)Sim::getRootGroup(), evt, Sim::getCurrentTime());
- return R( EngineUnmarshallData< R >()( cb.waitForResult() ) );
- }
- }
- };
- // Re-enable some VC warnings we disabled for this file.
- #pragma warning( pop ) // 4510 and 4610
- #endif // !_ENGINEAPI_H_
|