mirror of
https://github.com/ThePhD/sol2.git
synced 2024-03-22 13:10:44 +08:00
fixes to a lot of tests and proper subclassing for internal metatable data and work done to all of the usertypes to make them behave as expected
This commit is contained in:
parent
7c29964339
commit
3342e65b38
|
@ -20,8 +20,8 @@
|
||||||
// CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
// CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
// This file was generated with a script.
|
// This file was generated with a script.
|
||||||
// Generated 2017-03-23 14:12:13.335349 UTC
|
// Generated 2017-03-24 01:12:07.656967 UTC
|
||||||
// This header was generated with sol v2.16.0 (revision 6ceb715)
|
// This header was generated with sol v2.16.0 (revision 7c29964)
|
||||||
// https://github.com/ThePhD/sol2
|
// https://github.com/ThePhD/sol2
|
||||||
|
|
||||||
#ifndef SOL_SINGLE_INCLUDE_HPP
|
#ifndef SOL_SINGLE_INCLUDE_HPP
|
||||||
|
@ -10125,6 +10125,43 @@ namespace sol {
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
|
|
||||||
namespace sol {
|
namespace sol {
|
||||||
|
namespace usertype_detail {
|
||||||
|
typedef void(*base_walk)(lua_State*, bool&, int&, string_detail::string_shim&);
|
||||||
|
typedef int(*member_search)(lua_State*, void*, int);
|
||||||
|
|
||||||
|
struct call_information {
|
||||||
|
member_search first;
|
||||||
|
member_search second;
|
||||||
|
int runtime_target;
|
||||||
|
|
||||||
|
call_information(member_search first, member_search second) : call_information(first, second, -1) {}
|
||||||
|
call_information(member_search first, member_search second, int runtimetarget) : first(first), second(second), runtime_target(runtimetarget) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef std::unordered_map<std::string, call_information> mapping_t;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct usertype_metatable_core {
|
||||||
|
usertype_detail::mapping_t mapping;
|
||||||
|
lua_CFunction indexfunc;
|
||||||
|
lua_CFunction newindexfunc;
|
||||||
|
std::vector<object> runtime;
|
||||||
|
bool mustindex;
|
||||||
|
|
||||||
|
usertype_metatable_core(lua_CFunction ifx, lua_CFunction nifx) :
|
||||||
|
mapping(), indexfunc(ifx),
|
||||||
|
newindexfunc(nifx), runtime(), mustindex(false)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
usertype_metatable_core(const usertype_metatable_core&) = default;
|
||||||
|
usertype_metatable_core(usertype_metatable_core&&) = default;
|
||||||
|
usertype_metatable_core& operator=(const usertype_metatable_core&) = default;
|
||||||
|
usertype_metatable_core& operator=(usertype_metatable_core&&) = default;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
namespace usertype_detail {
|
namespace usertype_detail {
|
||||||
const lua_Integer toplevel_magic = static_cast<lua_Integer>(0x00020001);
|
const lua_Integer toplevel_magic = static_cast<lua_Integer>(0x00020001);
|
||||||
|
|
||||||
|
@ -10169,20 +10206,6 @@ namespace sol {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef void(*base_walk)(lua_State*, bool&, int&, string_detail::string_shim&);
|
|
||||||
typedef int(*member_search)(lua_State*, void*, int);
|
|
||||||
|
|
||||||
struct call_information {
|
|
||||||
member_search first;
|
|
||||||
member_search second;
|
|
||||||
int runtime_target;
|
|
||||||
|
|
||||||
call_information(member_search first, member_search second) : call_information(first, second, -1) {}
|
|
||||||
call_information(member_search first, member_search second, int runtimetarget) : first(first), second(second), runtime_target(runtimetarget) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef std::unordered_map<std::string, call_information> mapping_t;
|
|
||||||
|
|
||||||
inline bool is_indexer(string_detail::string_shim s) {
|
inline bool is_indexer(string_detail::string_shim s) {
|
||||||
return s == name_of(meta_function::index) || s == name_of(meta_function::new_index);
|
return s == name_of(meta_function::index) || s == name_of(meta_function::new_index);
|
||||||
}
|
}
|
||||||
|
@ -10238,7 +10261,8 @@ namespace sol {
|
||||||
};
|
};
|
||||||
|
|
||||||
inline int runtime_object_call(lua_State* L, void*, int runtimetarget) {
|
inline int runtime_object_call(lua_State* L, void*, int runtimetarget) {
|
||||||
std::vector<object>& runtime = stack::get<light<std::vector<object>>>(L, lua_upvalueindex(2));
|
usertype_metatable_core& umc = stack::get<light<usertype_metatable_core>>(L, upvalue_index(2));
|
||||||
|
std::vector<object>& runtime = umc.runtime;
|
||||||
return stack::push(L, runtime[runtimetarget]);
|
return stack::push(L, runtime[runtimetarget]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10266,12 +10290,17 @@ namespace sol {
|
||||||
int isnum = 0;
|
int isnum = 0;
|
||||||
lua_Integer magic = lua_tointegerx(L, upvalue_index(4), &isnum);
|
lua_Integer magic = lua_tointegerx(L, upvalue_index(4), &isnum);
|
||||||
if (isnum != 0 && magic == toplevel_magic) {
|
if (isnum != 0 && magic == toplevel_magic) {
|
||||||
bool mustindex = lua_isboolean(L, upvalue_index(5)) != 0 && (lua_toboolean(L, upvalue_index(5)) != 0);
|
auto non_simple = [&L]() {
|
||||||
if (!is_simple && mustindex) {
|
if (is_simple)
|
||||||
mapping_t& mapping = stack::get<light<mapping_t>>(L, upvalue_index(3));
|
return;
|
||||||
std::vector<object>& runtime = stack::get<light<std::vector<object>>>(L, upvalue_index(2));
|
usertype_metatable_core& umc = stack::get<light<usertype_metatable_core>>(L, upvalue_index(2));
|
||||||
int target = static_cast<int>(runtime.size());
|
bool mustindex = umc.mustindex;
|
||||||
|
if (!mustindex)
|
||||||
|
return;
|
||||||
std::string accessor = stack::get<std::string>(L, 2);
|
std::string accessor = stack::get<std::string>(L, 2);
|
||||||
|
mapping_t& mapping = umc.mapping;
|
||||||
|
std::vector<object>& runtime = umc.runtime;
|
||||||
|
int target = static_cast<int>(runtime.size());
|
||||||
auto preexistingit = mapping.find(accessor);
|
auto preexistingit = mapping.find(accessor);
|
||||||
if (preexistingit == mapping.cend()) {
|
if (preexistingit == mapping.cend()) {
|
||||||
runtime.emplace_back(L, 3);
|
runtime.emplace_back(L, 3);
|
||||||
|
@ -10282,7 +10311,8 @@ namespace sol {
|
||||||
runtime[target] = sol::object(L, 3);
|
runtime[target] = sol::object(L, 3);
|
||||||
preexistingit->second = call_information(&runtime_object_call, &runtime_object_call, target);
|
preexistingit->second = call_information(&runtime_object_call, &runtime_object_call, target);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
non_simple();
|
||||||
for (std::size_t i = 0; i < 4; lua_pop(L, 1), ++i) {
|
for (std::size_t i = 0; i < 4; lua_pop(L, 1), ++i) {
|
||||||
const char* metakey = nullptr;
|
const char* metakey = nullptr;
|
||||||
switch (i) {
|
switch (i) {
|
||||||
|
@ -10396,7 +10426,7 @@ namespace sol {
|
||||||
struct usertype_metatable : usertype_detail::registrar {};
|
struct usertype_metatable : usertype_detail::registrar {};
|
||||||
|
|
||||||
template <typename T, std::size_t... I, typename... Tn>
|
template <typename T, std::size_t... I, typename... Tn>
|
||||||
struct usertype_metatable<T, std::index_sequence<I...>, Tn...> : usertype_detail::registrar {
|
struct usertype_metatable<T, std::index_sequence<I...>, Tn...> : usertype_metatable_core, usertype_detail::registrar {
|
||||||
typedef std::make_index_sequence<sizeof...(I) * 2> indices;
|
typedef std::make_index_sequence<sizeof...(I) * 2> indices;
|
||||||
typedef std::index_sequence<I...> half_indices;
|
typedef std::index_sequence<I...> half_indices;
|
||||||
typedef std::array<luaL_Reg, sizeof...(Tn) / 2 + 1 + 3> regs_t;
|
typedef std::array<luaL_Reg, sizeof...(Tn) / 2 + 1 + 3> regs_t;
|
||||||
|
@ -10404,11 +10434,7 @@ namespace sol {
|
||||||
typedef std::tuple<clean_type_t<Tn> ...> Tuple;
|
typedef std::tuple<clean_type_t<Tn> ...> Tuple;
|
||||||
template <std::size_t Idx>
|
template <std::size_t Idx>
|
||||||
struct check_binding : is_variable_binding<meta::unqualified_tuple_element_t<Idx, Tuple>> {};
|
struct check_binding : is_variable_binding<meta::unqualified_tuple_element_t<Idx, Tuple>> {};
|
||||||
usertype_detail::mapping_t mapping;
|
|
||||||
std::vector<object> runtime;
|
|
||||||
Tuple functions;
|
Tuple functions;
|
||||||
lua_CFunction indexfunc;
|
|
||||||
lua_CFunction newindexfunc;
|
|
||||||
lua_CFunction destructfunc;
|
lua_CFunction destructfunc;
|
||||||
lua_CFunction callconstructfunc;
|
lua_CFunction callconstructfunc;
|
||||||
lua_CFunction indexbase;
|
lua_CFunction indexbase;
|
||||||
|
@ -10417,7 +10443,6 @@ namespace sol {
|
||||||
usertype_detail::base_walk newindexbaseclasspropogation;
|
usertype_detail::base_walk newindexbaseclasspropogation;
|
||||||
void* baseclasscheck;
|
void* baseclasscheck;
|
||||||
void* baseclasscast;
|
void* baseclasscast;
|
||||||
bool mustindex;
|
|
||||||
bool secondarymeta;
|
bool secondarymeta;
|
||||||
bool hasequals;
|
bool hasequals;
|
||||||
bool hasless;
|
bool hasless;
|
||||||
|
@ -10523,34 +10548,26 @@ namespace sol {
|
||||||
++index;
|
++index;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <std::size_t Idx>
|
|
||||||
static std::pair<std::string, usertype_detail::call_information> make_call_info(std::string n) {
|
|
||||||
return{ n, (n == "__newindex" || n == "__index") ?
|
|
||||||
usertype_detail::call_information(&usertype_metatable::real_meta_call<Idx * 2 + 1, false>,
|
|
||||||
&usertype_metatable::real_meta_call<Idx * 2 + 1, true>)
|
|
||||||
:
|
|
||||||
usertype_detail::call_information(&usertype_metatable::real_find_call<Idx * 2, Idx * 2 + 1, false>,
|
|
||||||
&usertype_metatable::real_find_call<Idx * 2, Idx * 2 + 1, true>)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args, typename = std::enable_if_t<sizeof...(Args) == sizeof...(Tn)>>
|
template <typename... Args, typename = std::enable_if_t<sizeof...(Args) == sizeof...(Tn)>>
|
||||||
usertype_metatable(Args&&... args) :
|
usertype_metatable(Args&&... args) : usertype_metatable_core(&usertype_detail::indexing_fail<true>, &usertype_detail::metatable_newindex<T, false>), usertype_detail::registrar(),
|
||||||
mapping(),
|
|
||||||
functions(std::forward<Args>(args)...),
|
functions(std::forward<Args>(args)...),
|
||||||
indexfunc(&usertype_detail::indexing_fail<true>), newindexfunc(&usertype_detail::metatable_newindex<T, false>),
|
|
||||||
destructfunc(nullptr), callconstructfunc(nullptr),
|
destructfunc(nullptr), callconstructfunc(nullptr),
|
||||||
indexbase(&core_indexing_call<true>), newindexbase(&core_indexing_call<false>),
|
indexbase(&core_indexing_call<true>), newindexbase(&core_indexing_call<false>),
|
||||||
indexbaseclasspropogation(usertype_detail::walk_all_bases<true>), newindexbaseclasspropogation(usertype_detail::walk_all_bases<false>),
|
indexbaseclasspropogation(usertype_detail::walk_all_bases<true>), newindexbaseclasspropogation(usertype_detail::walk_all_bases<false>),
|
||||||
baseclasscheck(nullptr), baseclasscast(nullptr),
|
baseclasscheck(nullptr), baseclasscast(nullptr),
|
||||||
mustindex(contains_variable() || contains_index()), secondarymeta(contains_variable()),
|
secondarymeta(contains_variable()),
|
||||||
hasequals(false), hasless(false), haslessequals(false) {
|
hasequals(false), hasless(false), haslessequals(false) {
|
||||||
std::initializer_list<typename usertype_detail::mapping_t::value_type> ilist{ {
|
std::initializer_list<typename usertype_detail::mapping_t::value_type> ilist{ {
|
||||||
std::pair<std::string, usertype_detail::call_information>(
|
std::pair<std::string, usertype_detail::call_information>( usertype_detail::make_string(std::get<I * 2>(functions)),
|
||||||
make_call_info<I>(usertype_detail::make_string(std::get<I * 2>(functions)))
|
usertype_detail::call_information(&usertype_metatable::real_find_call<I * 2, I * 2 + 1, false>,
|
||||||
|
&usertype_metatable::real_find_call<I * 2, I * 2 + 1, true>)
|
||||||
)
|
)
|
||||||
}... };
|
}... };
|
||||||
mapping.insert(ilist);
|
this->mapping.insert(ilist);
|
||||||
|
for (const auto& n : meta_function_names) {
|
||||||
|
this->mapping.erase(n);
|
||||||
|
}
|
||||||
|
this->mustindex = contains_variable() || contains_index();
|
||||||
}
|
}
|
||||||
|
|
||||||
usertype_metatable(const usertype_metatable&) = default;
|
usertype_metatable(const usertype_metatable&) = default;
|
||||||
|
@ -10569,10 +10586,10 @@ namespace sol {
|
||||||
return stack::push(L, c_closure(cfunc, upvalues));
|
return stack::push(L, c_closure(cfunc, upvalues));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <std::size_t I1, bool is_index>
|
template <bool is_index>
|
||||||
static int real_meta_call(lua_State* L, void* um, int) {
|
static int real_meta_call(lua_State* L, void* um, int) {
|
||||||
auto& f = *static_cast<usertype_metatable*>(um);
|
auto& f = *static_cast<usertype_metatable*>(um);
|
||||||
return real_call_with<I1, is_index>(L, f);
|
return is_index ? f.indexfunc(L) : f.newindexfunc(L);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <bool is_index, bool toplevel = false>
|
template <bool is_index, bool toplevel = false>
|
||||||
|
@ -10693,6 +10710,7 @@ namespace sol {
|
||||||
static int push(lua_State* L, umt_t&& umx) {
|
static int push(lua_State* L, umt_t&& umx) {
|
||||||
|
|
||||||
umt_t& um = make_cleanup(L, std::move(umx));
|
umt_t& um = make_cleanup(L, std::move(umx));
|
||||||
|
usertype_metatable_core& umc = um;
|
||||||
regs_t value_table{ {} };
|
regs_t value_table{ {} };
|
||||||
int lastreg = 0;
|
int lastreg = 0;
|
||||||
(void)detail::swallow{ 0, (um.template make_regs<(I * 2)>(value_table, lastreg, std::get<(I * 2)>(um.functions), std::get<(I * 2 + 1)>(um.functions)), 0)... };
|
(void)detail::swallow{ 0, (um.template make_regs<(I * 2)>(value_table, lastreg, std::get<(I * 2)>(um.functions), std::get<(I * 2 + 1)>(um.functions)), 0)... };
|
||||||
|
@ -10707,7 +10725,7 @@ namespace sol {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now use um
|
// Now use um
|
||||||
const bool& mustindex = um.mustindex;
|
const bool& mustindex = umc.mustindex;
|
||||||
for (std::size_t i = 0; i < 3; ++i) {
|
for (std::size_t i = 0; i < 3; ++i) {
|
||||||
// Pointer types, AKA "references" from C++
|
// Pointer types, AKA "references" from C++
|
||||||
const char* metakey = nullptr;
|
const char* metakey = nullptr;
|
||||||
|
@ -10739,14 +10757,14 @@ namespace sol {
|
||||||
stack::set_field(L, detail::base_class_cast_key(), um.baseclasscast, t.stack_index());
|
stack::set_field(L, detail::base_class_cast_key(), um.baseclasscast, t.stack_index());
|
||||||
}
|
}
|
||||||
|
|
||||||
stack::set_field(L, detail::base_class_index_propogation_key(), make_closure(um.indexbase, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, detail::base_class_index_propogation_key(), make_closure(um.indexbase, make_light(um), make_light(umc)), t.stack_index());
|
||||||
stack::set_field(L, detail::base_class_new_index_propogation_key(), make_closure(um.newindexbase, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, detail::base_class_new_index_propogation_key(), make_closure(um.newindexbase, make_light(um), make_light(umc)), t.stack_index());
|
||||||
|
|
||||||
if (mustindex) {
|
if (mustindex) {
|
||||||
// Basic index pushing: specialize
|
// Basic index pushing: specialize
|
||||||
// index and newindex to give variables and stuff
|
// index and newindex to give variables and stuff
|
||||||
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(umc)), t.stack_index());
|
||||||
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(umc)), t.stack_index());
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// If there's only functions, we can use the fast index version
|
// If there's only functions, we can use the fast index version
|
||||||
|
@ -10757,11 +10775,11 @@ namespace sol {
|
||||||
lua_createtable(L, 0, 3);
|
lua_createtable(L, 0, 3);
|
||||||
stack_reference metabehind(L, -1);
|
stack_reference metabehind(L, -1);
|
||||||
if (um.callconstructfunc != nullptr) {
|
if (um.callconstructfunc != nullptr) {
|
||||||
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), make_light(um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
}
|
}
|
||||||
if (um.secondarymeta) {
|
if (um.secondarymeta) {
|
||||||
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
}
|
}
|
||||||
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
||||||
metabehind.pop();
|
metabehind.pop();
|
||||||
|
@ -10779,11 +10797,11 @@ namespace sol {
|
||||||
lua_createtable(L, 0, 3);
|
lua_createtable(L, 0, 3);
|
||||||
stack_reference metabehind(L, -1);
|
stack_reference metabehind(L, -1);
|
||||||
if (um.callconstructfunc != nullptr) {
|
if (um.callconstructfunc != nullptr) {
|
||||||
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), static_cast<void*>(&um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
}
|
}
|
||||||
|
|
||||||
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), static_cast<void*>(&um.runtime), static_cast<void*>(&um.mapping), usertype_detail::toplevel_magic, um.mustindex), metabehind.stack_index());
|
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(umc), 0, usertype_detail::toplevel_magic), metabehind.stack_index());
|
||||||
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), static_cast<void*>(&um.runtime), static_cast<void*>(&um.mapping), usertype_detail::toplevel_magic, um.mustindex), metabehind.stack_index());
|
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(umc), 0, usertype_detail::toplevel_magic), metabehind.stack_index());
|
||||||
|
|
||||||
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
||||||
metabehind.pop();
|
metabehind.pop();
|
||||||
|
|
|
@ -35,6 +35,43 @@
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
|
|
||||||
namespace sol {
|
namespace sol {
|
||||||
|
namespace usertype_detail {
|
||||||
|
typedef void(*base_walk)(lua_State*, bool&, int&, string_detail::string_shim&);
|
||||||
|
typedef int(*member_search)(lua_State*, void*, int);
|
||||||
|
|
||||||
|
struct call_information {
|
||||||
|
member_search first;
|
||||||
|
member_search second;
|
||||||
|
int runtime_target;
|
||||||
|
|
||||||
|
call_information(member_search first, member_search second) : call_information(first, second, -1) {}
|
||||||
|
call_information(member_search first, member_search second, int runtimetarget) : first(first), second(second), runtime_target(runtimetarget) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef std::unordered_map<std::string, call_information> mapping_t;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct usertype_metatable_core {
|
||||||
|
usertype_detail::mapping_t mapping;
|
||||||
|
lua_CFunction indexfunc;
|
||||||
|
lua_CFunction newindexfunc;
|
||||||
|
std::vector<object> runtime;
|
||||||
|
bool mustindex;
|
||||||
|
|
||||||
|
usertype_metatable_core(lua_CFunction ifx, lua_CFunction nifx) :
|
||||||
|
mapping(), indexfunc(ifx),
|
||||||
|
newindexfunc(nifx), runtime(), mustindex(false)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
usertype_metatable_core(const usertype_metatable_core&) = default;
|
||||||
|
usertype_metatable_core(usertype_metatable_core&&) = default;
|
||||||
|
usertype_metatable_core& operator=(const usertype_metatable_core&) = default;
|
||||||
|
usertype_metatable_core& operator=(usertype_metatable_core&&) = default;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
namespace usertype_detail {
|
namespace usertype_detail {
|
||||||
const lua_Integer toplevel_magic = static_cast<lua_Integer>(0x00020001);
|
const lua_Integer toplevel_magic = static_cast<lua_Integer>(0x00020001);
|
||||||
|
|
||||||
|
@ -79,20 +116,6 @@ namespace sol {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef void(*base_walk)(lua_State*, bool&, int&, string_detail::string_shim&);
|
|
||||||
typedef int(*member_search)(lua_State*, void*, int);
|
|
||||||
|
|
||||||
struct call_information {
|
|
||||||
member_search first;
|
|
||||||
member_search second;
|
|
||||||
int runtime_target;
|
|
||||||
|
|
||||||
call_information(member_search first, member_search second) : call_information(first, second, -1) {}
|
|
||||||
call_information(member_search first, member_search second, int runtimetarget) : first(first), second(second), runtime_target(runtimetarget) {}
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef std::unordered_map<std::string, call_information> mapping_t;
|
|
||||||
|
|
||||||
inline bool is_indexer(string_detail::string_shim s) {
|
inline bool is_indexer(string_detail::string_shim s) {
|
||||||
return s == name_of(meta_function::index) || s == name_of(meta_function::new_index);
|
return s == name_of(meta_function::index) || s == name_of(meta_function::new_index);
|
||||||
}
|
}
|
||||||
|
@ -148,7 +171,8 @@ namespace sol {
|
||||||
};
|
};
|
||||||
|
|
||||||
inline int runtime_object_call(lua_State* L, void*, int runtimetarget) {
|
inline int runtime_object_call(lua_State* L, void*, int runtimetarget) {
|
||||||
std::vector<object>& runtime = stack::get<light<std::vector<object>>>(L, lua_upvalueindex(2));
|
usertype_metatable_core& umc = stack::get<light<usertype_metatable_core>>(L, upvalue_index(2));
|
||||||
|
std::vector<object>& runtime = umc.runtime;
|
||||||
return stack::push(L, runtime[runtimetarget]);
|
return stack::push(L, runtime[runtimetarget]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,12 +200,17 @@ namespace sol {
|
||||||
int isnum = 0;
|
int isnum = 0;
|
||||||
lua_Integer magic = lua_tointegerx(L, upvalue_index(4), &isnum);
|
lua_Integer magic = lua_tointegerx(L, upvalue_index(4), &isnum);
|
||||||
if (isnum != 0 && magic == toplevel_magic) {
|
if (isnum != 0 && magic == toplevel_magic) {
|
||||||
bool mustindex = lua_isboolean(L, upvalue_index(5)) != 0 && (lua_toboolean(L, upvalue_index(5)) != 0);
|
auto non_simple = [&L]() {
|
||||||
if (!is_simple && mustindex) {
|
if (is_simple)
|
||||||
mapping_t& mapping = stack::get<light<mapping_t>>(L, upvalue_index(3));
|
return;
|
||||||
std::vector<object>& runtime = stack::get<light<std::vector<object>>>(L, upvalue_index(2));
|
usertype_metatable_core& umc = stack::get<light<usertype_metatable_core>>(L, upvalue_index(2));
|
||||||
int target = static_cast<int>(runtime.size());
|
bool mustindex = umc.mustindex;
|
||||||
|
if (!mustindex)
|
||||||
|
return;
|
||||||
std::string accessor = stack::get<std::string>(L, 2);
|
std::string accessor = stack::get<std::string>(L, 2);
|
||||||
|
mapping_t& mapping = umc.mapping;
|
||||||
|
std::vector<object>& runtime = umc.runtime;
|
||||||
|
int target = static_cast<int>(runtime.size());
|
||||||
auto preexistingit = mapping.find(accessor);
|
auto preexistingit = mapping.find(accessor);
|
||||||
if (preexistingit == mapping.cend()) {
|
if (preexistingit == mapping.cend()) {
|
||||||
runtime.emplace_back(L, 3);
|
runtime.emplace_back(L, 3);
|
||||||
|
@ -192,7 +221,8 @@ namespace sol {
|
||||||
runtime[target] = sol::object(L, 3);
|
runtime[target] = sol::object(L, 3);
|
||||||
preexistingit->second = call_information(&runtime_object_call, &runtime_object_call, target);
|
preexistingit->second = call_information(&runtime_object_call, &runtime_object_call, target);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
non_simple();
|
||||||
for (std::size_t i = 0; i < 4; lua_pop(L, 1), ++i) {
|
for (std::size_t i = 0; i < 4; lua_pop(L, 1), ++i) {
|
||||||
const char* metakey = nullptr;
|
const char* metakey = nullptr;
|
||||||
switch (i) {
|
switch (i) {
|
||||||
|
@ -306,7 +336,7 @@ namespace sol {
|
||||||
struct usertype_metatable : usertype_detail::registrar {};
|
struct usertype_metatable : usertype_detail::registrar {};
|
||||||
|
|
||||||
template <typename T, std::size_t... I, typename... Tn>
|
template <typename T, std::size_t... I, typename... Tn>
|
||||||
struct usertype_metatable<T, std::index_sequence<I...>, Tn...> : usertype_detail::registrar {
|
struct usertype_metatable<T, std::index_sequence<I...>, Tn...> : usertype_metatable_core, usertype_detail::registrar {
|
||||||
typedef std::make_index_sequence<sizeof...(I) * 2> indices;
|
typedef std::make_index_sequence<sizeof...(I) * 2> indices;
|
||||||
typedef std::index_sequence<I...> half_indices;
|
typedef std::index_sequence<I...> half_indices;
|
||||||
typedef std::array<luaL_Reg, sizeof...(Tn) / 2 + 1 + 3> regs_t;
|
typedef std::array<luaL_Reg, sizeof...(Tn) / 2 + 1 + 3> regs_t;
|
||||||
|
@ -314,11 +344,7 @@ namespace sol {
|
||||||
typedef std::tuple<clean_type_t<Tn> ...> Tuple;
|
typedef std::tuple<clean_type_t<Tn> ...> Tuple;
|
||||||
template <std::size_t Idx>
|
template <std::size_t Idx>
|
||||||
struct check_binding : is_variable_binding<meta::unqualified_tuple_element_t<Idx, Tuple>> {};
|
struct check_binding : is_variable_binding<meta::unqualified_tuple_element_t<Idx, Tuple>> {};
|
||||||
usertype_detail::mapping_t mapping;
|
|
||||||
std::vector<object> runtime;
|
|
||||||
Tuple functions;
|
Tuple functions;
|
||||||
lua_CFunction indexfunc;
|
|
||||||
lua_CFunction newindexfunc;
|
|
||||||
lua_CFunction destructfunc;
|
lua_CFunction destructfunc;
|
||||||
lua_CFunction callconstructfunc;
|
lua_CFunction callconstructfunc;
|
||||||
lua_CFunction indexbase;
|
lua_CFunction indexbase;
|
||||||
|
@ -327,7 +353,6 @@ namespace sol {
|
||||||
usertype_detail::base_walk newindexbaseclasspropogation;
|
usertype_detail::base_walk newindexbaseclasspropogation;
|
||||||
void* baseclasscheck;
|
void* baseclasscheck;
|
||||||
void* baseclasscast;
|
void* baseclasscast;
|
||||||
bool mustindex;
|
|
||||||
bool secondarymeta;
|
bool secondarymeta;
|
||||||
bool hasequals;
|
bool hasequals;
|
||||||
bool hasless;
|
bool hasless;
|
||||||
|
@ -433,34 +458,26 @@ namespace sol {
|
||||||
++index;
|
++index;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <std::size_t Idx>
|
|
||||||
static std::pair<std::string, usertype_detail::call_information> make_call_info(std::string n) {
|
|
||||||
return{ n, (n == "__newindex" || n == "__index") ?
|
|
||||||
usertype_detail::call_information(&usertype_metatable::real_meta_call<Idx * 2 + 1, false>,
|
|
||||||
&usertype_metatable::real_meta_call<Idx * 2 + 1, true>)
|
|
||||||
:
|
|
||||||
usertype_detail::call_information(&usertype_metatable::real_find_call<Idx * 2, Idx * 2 + 1, false>,
|
|
||||||
&usertype_metatable::real_find_call<Idx * 2, Idx * 2 + 1, true>)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args, typename = std::enable_if_t<sizeof...(Args) == sizeof...(Tn)>>
|
template <typename... Args, typename = std::enable_if_t<sizeof...(Args) == sizeof...(Tn)>>
|
||||||
usertype_metatable(Args&&... args) :
|
usertype_metatable(Args&&... args) : usertype_metatable_core(&usertype_detail::indexing_fail<true>, &usertype_detail::metatable_newindex<T, false>), usertype_detail::registrar(),
|
||||||
mapping(),
|
|
||||||
functions(std::forward<Args>(args)...),
|
functions(std::forward<Args>(args)...),
|
||||||
indexfunc(&usertype_detail::indexing_fail<true>), newindexfunc(&usertype_detail::metatable_newindex<T, false>),
|
|
||||||
destructfunc(nullptr), callconstructfunc(nullptr),
|
destructfunc(nullptr), callconstructfunc(nullptr),
|
||||||
indexbase(&core_indexing_call<true>), newindexbase(&core_indexing_call<false>),
|
indexbase(&core_indexing_call<true>), newindexbase(&core_indexing_call<false>),
|
||||||
indexbaseclasspropogation(usertype_detail::walk_all_bases<true>), newindexbaseclasspropogation(usertype_detail::walk_all_bases<false>),
|
indexbaseclasspropogation(usertype_detail::walk_all_bases<true>), newindexbaseclasspropogation(usertype_detail::walk_all_bases<false>),
|
||||||
baseclasscheck(nullptr), baseclasscast(nullptr),
|
baseclasscheck(nullptr), baseclasscast(nullptr),
|
||||||
mustindex(contains_variable() || contains_index()), secondarymeta(contains_variable()),
|
secondarymeta(contains_variable()),
|
||||||
hasequals(false), hasless(false), haslessequals(false) {
|
hasequals(false), hasless(false), haslessequals(false) {
|
||||||
std::initializer_list<typename usertype_detail::mapping_t::value_type> ilist{ {
|
std::initializer_list<typename usertype_detail::mapping_t::value_type> ilist{ {
|
||||||
std::pair<std::string, usertype_detail::call_information>(
|
std::pair<std::string, usertype_detail::call_information>( usertype_detail::make_string(std::get<I * 2>(functions)),
|
||||||
make_call_info<I>(usertype_detail::make_string(std::get<I * 2>(functions)))
|
usertype_detail::call_information(&usertype_metatable::real_find_call<I * 2, I * 2 + 1, false>,
|
||||||
|
&usertype_metatable::real_find_call<I * 2, I * 2 + 1, true>)
|
||||||
)
|
)
|
||||||
}... };
|
}... };
|
||||||
mapping.insert(ilist);
|
this->mapping.insert(ilist);
|
||||||
|
for (const auto& n : meta_function_names) {
|
||||||
|
this->mapping.erase(n);
|
||||||
|
}
|
||||||
|
this->mustindex = contains_variable() || contains_index();
|
||||||
}
|
}
|
||||||
|
|
||||||
usertype_metatable(const usertype_metatable&) = default;
|
usertype_metatable(const usertype_metatable&) = default;
|
||||||
|
@ -479,10 +496,10 @@ namespace sol {
|
||||||
return stack::push(L, c_closure(cfunc, upvalues));
|
return stack::push(L, c_closure(cfunc, upvalues));
|
||||||
}
|
}
|
||||||
|
|
||||||
template <std::size_t I1, bool is_index>
|
template <bool is_index>
|
||||||
static int real_meta_call(lua_State* L, void* um, int) {
|
static int real_meta_call(lua_State* L, void* um, int) {
|
||||||
auto& f = *static_cast<usertype_metatable*>(um);
|
auto& f = *static_cast<usertype_metatable*>(um);
|
||||||
return real_call_with<I1, is_index>(L, f);
|
return is_index ? f.indexfunc(L) : f.newindexfunc(L);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <bool is_index, bool toplevel = false>
|
template <bool is_index, bool toplevel = false>
|
||||||
|
@ -603,6 +620,7 @@ namespace sol {
|
||||||
static int push(lua_State* L, umt_t&& umx) {
|
static int push(lua_State* L, umt_t&& umx) {
|
||||||
|
|
||||||
umt_t& um = make_cleanup(L, std::move(umx));
|
umt_t& um = make_cleanup(L, std::move(umx));
|
||||||
|
usertype_metatable_core& umc = um;
|
||||||
regs_t value_table{ {} };
|
regs_t value_table{ {} };
|
||||||
int lastreg = 0;
|
int lastreg = 0;
|
||||||
(void)detail::swallow{ 0, (um.template make_regs<(I * 2)>(value_table, lastreg, std::get<(I * 2)>(um.functions), std::get<(I * 2 + 1)>(um.functions)), 0)... };
|
(void)detail::swallow{ 0, (um.template make_regs<(I * 2)>(value_table, lastreg, std::get<(I * 2)>(um.functions), std::get<(I * 2 + 1)>(um.functions)), 0)... };
|
||||||
|
@ -617,7 +635,7 @@ namespace sol {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now use um
|
// Now use um
|
||||||
const bool& mustindex = um.mustindex;
|
const bool& mustindex = umc.mustindex;
|
||||||
for (std::size_t i = 0; i < 3; ++i) {
|
for (std::size_t i = 0; i < 3; ++i) {
|
||||||
// Pointer types, AKA "references" from C++
|
// Pointer types, AKA "references" from C++
|
||||||
const char* metakey = nullptr;
|
const char* metakey = nullptr;
|
||||||
|
@ -649,14 +667,14 @@ namespace sol {
|
||||||
stack::set_field(L, detail::base_class_cast_key(), um.baseclasscast, t.stack_index());
|
stack::set_field(L, detail::base_class_cast_key(), um.baseclasscast, t.stack_index());
|
||||||
}
|
}
|
||||||
|
|
||||||
stack::set_field(L, detail::base_class_index_propogation_key(), make_closure(um.indexbase, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, detail::base_class_index_propogation_key(), make_closure(um.indexbase, make_light(um), make_light(umc)), t.stack_index());
|
||||||
stack::set_field(L, detail::base_class_new_index_propogation_key(), make_closure(um.newindexbase, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, detail::base_class_new_index_propogation_key(), make_closure(um.newindexbase, make_light(um), make_light(umc)), t.stack_index());
|
||||||
|
|
||||||
if (mustindex) {
|
if (mustindex) {
|
||||||
// Basic index pushing: specialize
|
// Basic index pushing: specialize
|
||||||
// index and newindex to give variables and stuff
|
// index and newindex to give variables and stuff
|
||||||
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(umc)), t.stack_index());
|
||||||
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(um.runtime)), t.stack_index());
|
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(umc)), t.stack_index());
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// If there's only functions, we can use the fast index version
|
// If there's only functions, we can use the fast index version
|
||||||
|
@ -667,11 +685,11 @@ namespace sol {
|
||||||
lua_createtable(L, 0, 3);
|
lua_createtable(L, 0, 3);
|
||||||
stack_reference metabehind(L, -1);
|
stack_reference metabehind(L, -1);
|
||||||
if (um.callconstructfunc != nullptr) {
|
if (um.callconstructfunc != nullptr) {
|
||||||
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), make_light(um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
}
|
}
|
||||||
if (um.secondarymeta) {
|
if (um.secondarymeta) {
|
||||||
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
}
|
}
|
||||||
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
||||||
metabehind.pop();
|
metabehind.pop();
|
||||||
|
@ -689,11 +707,11 @@ namespace sol {
|
||||||
lua_createtable(L, 0, 3);
|
lua_createtable(L, 0, 3);
|
||||||
stack_reference metabehind(L, -1);
|
stack_reference metabehind(L, -1);
|
||||||
if (um.callconstructfunc != nullptr) {
|
if (um.callconstructfunc != nullptr) {
|
||||||
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), static_cast<void*>(&um.runtime)), metabehind.stack_index());
|
stack::set_field(L, meta_function::call_function, make_closure(um.callconstructfunc, make_light(um), make_light(umc)), metabehind.stack_index());
|
||||||
}
|
}
|
||||||
|
|
||||||
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), static_cast<void*>(&um.runtime), static_cast<void*>(&um.mapping), usertype_detail::toplevel_magic, um.mustindex), metabehind.stack_index());
|
stack::set_field(L, meta_function::index, make_closure(umt_t::index_call, make_light(um), make_light(umc), 0, usertype_detail::toplevel_magic), metabehind.stack_index());
|
||||||
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), static_cast<void*>(&um.runtime), static_cast<void*>(&um.mapping), usertype_detail::toplevel_magic, um.mustindex), metabehind.stack_index());
|
stack::set_field(L, meta_function::new_index, make_closure(umt_t::new_index_call, make_light(um), make_light(umc), 0, usertype_detail::toplevel_magic), metabehind.stack_index());
|
||||||
|
|
||||||
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
stack::set_field(L, metatable_key, metabehind, t.stack_index());
|
||||||
metabehind.pop();
|
metabehind.pop();
|
||||||
|
|
|
@ -700,8 +700,8 @@ end
|
||||||
TEST_CASE("simple_usertype/meta-key-retrievals", "allow for special meta keys (__index, __newindex) to trigger methods even if overwritten directly") {
|
TEST_CASE("simple_usertype/meta-key-retrievals", "allow for special meta keys (__index, __newindex) to trigger methods even if overwritten directly") {
|
||||||
SECTION("dynamically") {
|
SECTION("dynamically") {
|
||||||
static int writes = 0;
|
static int writes = 0;
|
||||||
static std::string keys[2] = {};
|
static std::string keys[4] = {};
|
||||||
static int values[2] = {};
|
static int values[4] = {};
|
||||||
struct d_sample {
|
struct d_sample {
|
||||||
void foo(std::string k, int v) {
|
void foo(std::string k, int v) {
|
||||||
keys[writes] = k;
|
keys[writes] = k;
|
||||||
|
@ -710,25 +710,32 @@ TEST_CASE("simple_usertype/meta-key-retrievals", "allow for special meta keys (_
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
sol::state state;
|
sol::state lua;
|
||||||
state.new_simple_usertype<d_sample>("sample");
|
lua.new_simple_usertype<d_sample>("sample");
|
||||||
sol::table s = state["sample"]["new"]();
|
sol::table s = lua["sample"]["new"]();
|
||||||
s[sol::metatable_key][sol::meta_function::new_index] = &d_sample::foo;
|
s[sol::metatable_key][sol::meta_function::new_index] = &d_sample::foo;
|
||||||
state["var"] = s;
|
lua["var"] = s;
|
||||||
|
|
||||||
|
|
||||||
state.script("var.key = 2");
|
lua.script("var = sample.new()");
|
||||||
state.script("var.__newindex = 4");
|
lua.script("var.key = 2");
|
||||||
|
lua.script("var.__newindex = 4");
|
||||||
|
lua.script("var.__index = 3");
|
||||||
|
lua.script("var.__call = 1");
|
||||||
REQUIRE(values[0] == 2);
|
REQUIRE(values[0] == 2);
|
||||||
REQUIRE(keys[0] == "key");
|
|
||||||
REQUIRE(values[1] == 4);
|
REQUIRE(values[1] == 4);
|
||||||
|
REQUIRE(values[2] == 3);
|
||||||
|
REQUIRE(values[3] == 1);
|
||||||
|
REQUIRE(keys[0] == "key");
|
||||||
REQUIRE(keys[1] == "__newindex");
|
REQUIRE(keys[1] == "__newindex");
|
||||||
|
REQUIRE(keys[2] == "__index");
|
||||||
|
REQUIRE(keys[3] == "__call");
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION("statically") {
|
SECTION("statically") {
|
||||||
static int writes = 0;
|
static int writes = 0;
|
||||||
static std::string keys[2] = {};
|
static std::string keys[4] = {};
|
||||||
static int values[2] = {};
|
static int values[4] = {};
|
||||||
struct sample {
|
struct sample {
|
||||||
void foo(std::string k, int v) {
|
void foo(std::string k, int v) {
|
||||||
keys[writes] = k;
|
keys[writes] = k;
|
||||||
|
@ -737,15 +744,21 @@ TEST_CASE("simple_usertype/meta-key-retrievals", "allow for special meta keys (_
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
sol::state state;
|
sol::state lua;
|
||||||
state.new_simple_usertype<sample>("sample", sol::meta_function::new_index, &sample::foo);
|
lua.new_simple_usertype<sample>("sample", sol::meta_function::new_index, &sample::foo);
|
||||||
|
|
||||||
state.script("var = sample.new()");
|
lua.script("var = sample.new()");
|
||||||
state.script("var.key = 2");
|
lua.script("var.key = 2");
|
||||||
state.script("var.__newindex = 4");
|
lua.script("var.__newindex = 4");
|
||||||
|
lua.script("var.__index = 3");
|
||||||
|
lua.script("var.__call = 1");
|
||||||
REQUIRE(values[0] == 2);
|
REQUIRE(values[0] == 2);
|
||||||
REQUIRE(keys[0] == "key");
|
|
||||||
REQUIRE(values[1] == 4);
|
REQUIRE(values[1] == 4);
|
||||||
|
REQUIRE(values[2] == 3);
|
||||||
|
REQUIRE(values[3] == 1);
|
||||||
|
REQUIRE(keys[0] == "key");
|
||||||
REQUIRE(keys[1] == "__newindex");
|
REQUIRE(keys[1] == "__newindex");
|
||||||
|
REQUIRE(keys[2] == "__index");
|
||||||
|
REQUIRE(keys[3] == "__call");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1626,8 +1626,8 @@ end
|
||||||
TEST_CASE("usertype/meta-key-retrievals", "allow for special meta keys (__index, __newindex) to trigger methods even if overwritten directly") {
|
TEST_CASE("usertype/meta-key-retrievals", "allow for special meta keys (__index, __newindex) to trigger methods even if overwritten directly") {
|
||||||
SECTION("dynamically") {
|
SECTION("dynamically") {
|
||||||
static int writes = 0;
|
static int writes = 0;
|
||||||
static std::string keys[2] = {};
|
static std::string keys[4] = {};
|
||||||
static int values[2] = {};
|
static int values[4] = {};
|
||||||
struct d_sample {
|
struct d_sample {
|
||||||
void foo(std::string k, int v) {
|
void foo(std::string k, int v) {
|
||||||
keys[writes] = k;
|
keys[writes] = k;
|
||||||
|
@ -1636,25 +1636,32 @@ TEST_CASE("usertype/meta-key-retrievals", "allow for special meta keys (__index,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
sol::state state;
|
sol::state lua;
|
||||||
state.new_usertype<d_sample>("sample");
|
lua.new_usertype<d_sample>("sample");
|
||||||
sol::table s = state["sample"]["new"]();
|
sol::table s = lua["sample"]["new"]();
|
||||||
s[sol::metatable_key][sol::meta_function::new_index] = &d_sample::foo;
|
s[sol::metatable_key][sol::meta_function::new_index] = &d_sample::foo;
|
||||||
state["var"] = s;
|
lua["var"] = s;
|
||||||
|
|
||||||
|
|
||||||
state.script("var.key = 2");
|
lua.script("var = sample.new()");
|
||||||
state.script("var.__newindex = 4");
|
lua.script("var.key = 2");
|
||||||
|
lua.script("var.__newindex = 4");
|
||||||
|
lua.script("var.__index = 3");
|
||||||
|
lua.script("var.__call = 1");
|
||||||
REQUIRE(values[0] == 2);
|
REQUIRE(values[0] == 2);
|
||||||
REQUIRE(keys[0] == "key");
|
|
||||||
REQUIRE(values[1] == 4);
|
REQUIRE(values[1] == 4);
|
||||||
|
REQUIRE(values[2] == 3);
|
||||||
|
REQUIRE(values[3] == 1);
|
||||||
|
REQUIRE(keys[0] == "key");
|
||||||
REQUIRE(keys[1] == "__newindex");
|
REQUIRE(keys[1] == "__newindex");
|
||||||
|
REQUIRE(keys[2] == "__index");
|
||||||
|
REQUIRE(keys[3] == "__call");
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION("statically") {
|
SECTION("statically") {
|
||||||
static int writes = 0;
|
static int writes = 0;
|
||||||
static std::string keys[2] = {};
|
static std::string keys[4] = {};
|
||||||
static int values[2] = {};
|
static int values[4] = {};
|
||||||
struct sample {
|
struct sample {
|
||||||
void foo(std::string k, int v) {
|
void foo(std::string k, int v) {
|
||||||
keys[writes] = k;
|
keys[writes] = k;
|
||||||
|
@ -1663,15 +1670,21 @@ TEST_CASE("usertype/meta-key-retrievals", "allow for special meta keys (__index,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
sol::state state;
|
sol::state lua;
|
||||||
state.new_usertype<sample>("sample", sol::meta_function::new_index, &sample::foo);
|
lua.new_usertype<sample>("sample", sol::meta_function::new_index, &sample::foo);
|
||||||
|
|
||||||
state.script("var = sample.new()");
|
lua.script("var = sample.new()");
|
||||||
state.script("var.key = 2");
|
lua.script("var.key = 2");
|
||||||
state.script("var.__newindex = 4");
|
lua.script("var.__newindex = 4");
|
||||||
|
lua.script("var.__index = 3");
|
||||||
|
lua.script("var.__call = 1");
|
||||||
REQUIRE(values[0] == 2);
|
REQUIRE(values[0] == 2);
|
||||||
REQUIRE(keys[0] == "key");
|
|
||||||
REQUIRE(values[1] == 4);
|
REQUIRE(values[1] == 4);
|
||||||
|
REQUIRE(values[2] == 3);
|
||||||
|
REQUIRE(values[3] == 1);
|
||||||
|
REQUIRE(keys[0] == "key");
|
||||||
REQUIRE(keys[1] == "__newindex");
|
REQUIRE(keys[1] == "__newindex");
|
||||||
|
REQUIRE(keys[2] == "__index");
|
||||||
|
REQUIRE(keys[3] == "__call");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user