Merge remote-tracking branch 'upstream/master'
commit
af344e57bd
@ -0,0 +1,129 @@
|
||||
/*
|
||||
https://github.com/peterix/dfhack
|
||||
Copyright (c) 2009-2011 Petr Mrázek (peterix@gmail.com)
|
||||
|
||||
This software is provided 'as-is', without any express or implied
|
||||
warranty. In no event will the authors be held liable for any
|
||||
damages arising from the use of this software.
|
||||
|
||||
Permission is granted to anyone to use this software for any
|
||||
purpose, including commercial applications, and to alter it and
|
||||
redistribute it freely, subject to the following restrictions:
|
||||
|
||||
1. The origin of this software must not be misrepresented; you must
|
||||
not claim that you wrote the original software. If you use this
|
||||
software in a product, an acknowledgment in the product documentation
|
||||
would be appreciated but is not required.
|
||||
|
||||
2. Altered source versions must be plainly marked as such, and
|
||||
must not be misrepresented as being the original software.
|
||||
|
||||
3. This notice may not be removed or altered from any source
|
||||
distribution.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <string>
|
||||
#include <sstream>
|
||||
#include <vector>
|
||||
#include <map>
|
||||
|
||||
#include "DataIdentity.h"
|
||||
#include "LuaWrapper.h"
|
||||
|
||||
namespace df {
|
||||
template<class T> struct function_wrapper {};
|
||||
|
||||
/*
|
||||
* Since templates can't match variable arg count,
|
||||
* a separate specialization is needed for every
|
||||
* supported count value...
|
||||
*
|
||||
* The FW_TARGS ugliness is needed because of
|
||||
* commas not wrapped in ()
|
||||
*/
|
||||
|
||||
#define INVOKE_VOID(call) \
|
||||
call; lua_pushnil(state);
|
||||
#define INVOKE_RV(call) \
|
||||
RT rv = call; df::identity_traits<RT>::get()->lua_read(state, UPVAL_METHOD_NAME, &rv);
|
||||
#define LOAD_CLASS() \
|
||||
CT *self = (CT*)DFHack::LuaWrapper::get_object_addr(state, base++, UPVAL_METHOD_NAME, "invoke");
|
||||
#define LOAD_ARG(type) \
|
||||
type v##type; df::identity_traits<type>::get()->lua_write(state, UPVAL_METHOD_NAME, &v##type, base++);
|
||||
|
||||
#define INSTANTIATE_WRAPPERS(Count, FArgs, Args, Loads) \
|
||||
template<FW_TARGS> struct function_wrapper<void (*) FArgs> { \
|
||||
static const bool is_method = false; \
|
||||
static const int num_args = Count; \
|
||||
static void execute(lua_State *state, int base, void (*cb) FArgs) { Loads; INVOKE_VOID(cb Args); } \
|
||||
}; \
|
||||
template<FW_TARGSC class RT> struct function_wrapper<RT (*) FArgs> { \
|
||||
static const bool is_method = false; \
|
||||
static const int num_args = Count; \
|
||||
static void execute(lua_State *state, int base, RT (*cb) FArgs) { Loads; INVOKE_RV(cb Args); } \
|
||||
}; \
|
||||
template<FW_TARGSC class CT> struct function_wrapper<void (CT::*) FArgs> { \
|
||||
static const bool is_method = true; \
|
||||
static const int num_args = Count+1; \
|
||||
static void execute(lua_State *state, int base, void (CT::*cb) FArgs) { \
|
||||
LOAD_CLASS() Loads; INVOKE_VOID((self->*cb) Args); } \
|
||||
}; \
|
||||
template<FW_TARGSC class RT, class CT> struct function_wrapper<RT (CT::*) FArgs> { \
|
||||
static const bool is_method = true; \
|
||||
static const int num_args = Count+1; \
|
||||
static void execute(lua_State *state, int base, RT (CT::*cb) FArgs) { \
|
||||
LOAD_CLASS(); Loads; INVOKE_RV((self->*cb) Args); } \
|
||||
};
|
||||
|
||||
#define FW_TARGSC
|
||||
#define FW_TARGS
|
||||
INSTANTIATE_WRAPPERS(0, (), (), ;)
|
||||
#undef FW_TARGS
|
||||
|
||||
#undef FW_TARGSC
|
||||
#define FW_TARGSC FW_TARGS,
|
||||
#define FW_TARGS class A1
|
||||
INSTANTIATE_WRAPPERS(1, (A1), (vA1), LOAD_ARG(A1);)
|
||||
#undef FW_TARGS
|
||||
|
||||
#define FW_TARGS class A1, class A2
|
||||
INSTANTIATE_WRAPPERS(2, (A1,A2), (vA1,vA2), LOAD_ARG(A1); LOAD_ARG(A2);)
|
||||
#undef FW_TARGS
|
||||
|
||||
#define FW_TARGS class A1, class A2, class A3
|
||||
INSTANTIATE_WRAPPERS(3, (A1,A2,A3), (vA1,vA2,vA3), LOAD_ARG(A1); LOAD_ARG(A2); LOAD_ARG(A3);)
|
||||
#undef FW_TARGS
|
||||
|
||||
#define FW_TARGS class A1, class A2, class A3, class A4
|
||||
INSTANTIATE_WRAPPERS(4, (A1,A2,A3,A4), (vA1,vA2,vA3,vA4),
|
||||
LOAD_ARG(A1); LOAD_ARG(A2); LOAD_ARG(A3); LOAD_ARG(A4);)
|
||||
#undef FW_TARGS
|
||||
|
||||
#undef FW_TARGSC
|
||||
#undef INSTANTIATE_WRAPPERS
|
||||
#undef INVOKE_VOID
|
||||
#undef INVOKE_RV
|
||||
#undef LOAD_CLASS
|
||||
#undef LOAD_ARG
|
||||
|
||||
template<class T>
|
||||
class function_identity : public function_identity_base {
|
||||
T ptr;
|
||||
|
||||
public:
|
||||
typedef function_wrapper<T> wrapper;
|
||||
|
||||
function_identity(T ptr)
|
||||
: function_identity_base(wrapper::num_args), ptr(ptr) {};
|
||||
|
||||
virtual void invoke(lua_State *state, int base) { wrapper::execute(state, base, ptr); }
|
||||
};
|
||||
|
||||
template<class T>
|
||||
inline function_identity_base *wrap_function(T ptr) {
|
||||
// bah, but didn't have any idea how to allocate statically
|
||||
return new function_identity<T>(ptr);
|
||||
}
|
||||
}
|
@ -1 +1 @@
|
||||
Subproject commit 55a120d5bc823b783aeb8b6931ccaa31402fd850
|
||||
Subproject commit f7c535a64b0858acbb56d38f13dde0c820e2c4bc
|
@ -1 +1 @@
|
||||
Subproject commit 7525c003089367823183eaf5093a90271a5eb9b4
|
||||
Subproject commit a0f6808254168063449b4f7e93afbd879bd1fafd
|
Loading…
Reference in New Issue