From 0373dd4eb2f2b9714af244b667329d7809868594 Mon Sep 17 00:00:00 2001 From: ThePhD Date: Fri, 19 Sep 2014 08:22:21 -0400 Subject: [PATCH 1/4] Change `userdata` to `usertype` names. We don't need to make the function names `open_usertype` now, since `new_usertype` makes sense. --- sol/function.hpp | 4 +- sol/function_types.hpp | 28 +++---- sol/stack.hpp | 8 +- sol/state.hpp | 18 ++--- sol/table.hpp | 8 +- sol/{userdata.hpp => usertype.hpp} | 74 ++++++++---------- ...serdata_traits.hpp => usertype_traits.hpp} | 14 ++-- tests.cpp | 75 ++++++++++--------- 8 files changed, 109 insertions(+), 120 deletions(-) rename sol/{userdata.hpp => usertype.hpp} (85%) rename sol/{userdata_traits.hpp => usertype_traits.hpp} (80%) diff --git a/sol/function.hpp b/sol/function.hpp index d349f9c9..cbdf5109 100644 --- a/sol/function.hpp +++ b/sol/function.hpp @@ -26,7 +26,7 @@ #include "tuple.hpp" #include "stack.hpp" #include "function_types.hpp" -#include "userdata_traits.hpp" +#include "usertype_traits.hpp" #include "resolve.hpp" #include #include @@ -191,7 +191,7 @@ struct pusher> { template static void set_fx(lua_State* L, std::unique_ptr luafunc) { - auto&& metakey = userdata_traits>::metatable; + auto&& metakey = usertype_traits>::metatable; const char* metatablename = std::addressof(metakey[0]); base_function* target = luafunc.release(); void* userdata = reinterpret_cast(target); diff --git a/sol/function_types.hpp b/sol/function_types.hpp index 8ea6c390..49bbd1ec 100644 --- a/sol/function_types.hpp +++ b/sol/function_types.hpp @@ -248,7 +248,7 @@ struct base_function { } template - struct userdata { + struct usertype { static int call(lua_State* L) { // Zero-based template parameter, but upvalues start at 1 return base_call(L, stack::get(L, I + 1)); @@ -373,7 +373,7 @@ struct member_function : public base_function { }; template -struct userdata_function_core : public base_function { +struct usertype_function_core : public base_function { typedef typename std::remove_pointer::type T; typedef typename std::remove_pointer::type>::type function_type; typedef detail::functor fx_t; @@ -384,7 +384,7 @@ struct userdata_function_core : public base_function { fx_t fx; template - userdata_function_core(FxArgs&&... fxargs): fx(std::forward(fxargs)...) {} + usertype_function_core(FxArgs&&... fxargs): fx(std::forward(fxargs)...) {} template> typename std::enable_if::value, void>::type push(lua_State* L, Return&& r) { @@ -432,15 +432,15 @@ struct userdata_function_core : public base_function { }; template -struct userdata_function : public userdata_function_core { - typedef userdata_function_core base_t; +struct usertype_function : public usertype_function_core { + typedef usertype_function_core base_t; typedef typename std::remove_pointer::type T; typedef typename base_t::traits_type traits_type; typedef typename base_t::args_type args_type; typedef typename base_t::return_type return_type; template - userdata_function(FxArgs&&... fxargs): base_t(std::forward(fxargs)...) {} + usertype_function(FxArgs&&... fxargs): base_t(std::forward(fxargs)...) {} template int fx_call(lua_State* L) { @@ -461,15 +461,15 @@ struct userdata_function : public userdata_function_core { }; template -struct userdata_variable_function : public userdata_function_core { - typedef userdata_function_core base_t; +struct usertype_variable_function : public usertype_function_core { + typedef usertype_function_core base_t; typedef typename std::remove_pointer::type T; typedef typename base_t::traits_type traits_type; typedef typename base_t::args_type args_type; typedef typename base_t::return_type return_type; template - userdata_variable_function(FxArgs&&... fxargs): base_t(std::forward(fxargs)...) {} + usertype_variable_function(FxArgs&&... fxargs): base_t(std::forward(fxargs)...) {} template int fx_call(lua_State* L) { @@ -500,8 +500,8 @@ struct userdata_variable_function : public userdata_function_core }; template -struct userdata_indexing_function : public userdata_function_core { - typedef userdata_function_core base_t; +struct usertype_indexing_function : public usertype_function_core { + typedef usertype_function_core base_t; typedef typename std::remove_pointer::type T; typedef typename base_t::traits_type traits_type; typedef typename base_t::args_type args_type; @@ -511,7 +511,7 @@ struct userdata_indexing_function : public userdata_function_core std::unordered_map, bool>> functions; template - userdata_indexing_function(std::string name, FxArgs&&... fxargs): base_t(std::forward(fxargs)...), name(std::move(name)) {} + usertype_indexing_function(std::string name, FxArgs&&... fxargs): base_t(std::forward(fxargs)...), name(std::move(name)) {} template int fx_call(lua_State* L) { @@ -521,10 +521,10 @@ struct userdata_indexing_function : public userdata_function_core if(function->second.second) { stack::push(L, function->second.first.get()); if(std::is_same::value) { - stack::push(L, &base_function::userdata<0>::ref_call, 1); + stack::push(L, &base_function::usertype<0>::ref_call, 1); } else { - stack::push(L, &base_function::userdata<0>::call, 1); + stack::push(L, &base_function::usertype<0>::call, 1); } return 1; } diff --git a/sol/stack.hpp b/sol/stack.hpp index 160a2515..77dd6296 100644 --- a/sol/stack.hpp +++ b/sol/stack.hpp @@ -26,7 +26,7 @@ #include "reference.hpp" #include "tuple.hpp" #include "traits.hpp" -#include "userdata_traits.hpp" +#include "usertype_traits.hpp" #include #include #include @@ -259,19 +259,19 @@ struct pusher { template, EnableIf>, Not>, Not>, Not>> = 0> static void push(lua_State* L, T& t) { - detail::push_userdata(L, userdata_traits::metatable, t); + detail::push_userdata(L, usertype_traits::metatable, t); } template, EnableIf>, Not>, Not>, Not>> = 0> static void push(lua_State* L, T&& t) { - detail::push_userdata(L, userdata_traits::metatable, std::move(t)); + detail::push_userdata(L, usertype_traits::metatable, std::move(t)); } }; template struct pusher { static void push(lua_State* L, T* obj) { - detail::push_userdata(L, userdata_traits::metatable, obj); + detail::push_userdata(L, usertype_traits::metatable, obj); } }; diff --git a/sol/state.hpp b/sol/state.hpp index b7401888..77a11ed8 100644 --- a/sol/state.hpp +++ b/sol/state.hpp @@ -148,26 +148,26 @@ public: } template - state& set_userdata(userdata& user) { - return set_userdata(user.name(), user); + state& set_usertype(usertype& user) { + return set_usertype(usertype_traits::name, user); } template - state& set_userdata(Key&& key, userdata& user) { - global.set_userdata(std::forward(key), user); + state& set_usertype(Key&& key, usertype& user) { + global.set_usertype(std::forward(key), user); return *this; } template - state& new_userdata(const std::string& name, Args&&... args) { + state& new_usertype(const std::string& name, Args&&... args) { constructors> ctor{}; - return new_userdata(name, ctor, std::forward(args)...); + return new_usertype(name, ctor, std::forward(args)...); } template - state& new_userdata(const std::string& name, constructors ctor, Args&&... args) { - userdata udata(name, ctor, std::forward(args)...); - global.set_userdata(udata); + state& new_usertype(const std::string& name, constructors ctor, Args&&... args) { + usertype utype(ctor, std::forward(args)...); + set_usertype(name, utype); return *this; } diff --git a/sol/table.hpp b/sol/table.hpp index 1420855b..04290db9 100644 --- a/sol/table.hpp +++ b/sol/table.hpp @@ -25,7 +25,7 @@ #include "proxy.hpp" #include "stack.hpp" #include "function_types.hpp" -#include "userdata.hpp" +#include "usertype.hpp" namespace sol { class table : public reference { @@ -84,12 +84,12 @@ public: } template - table& set_userdata(userdata& user) { - return set_userdata(user.name(), user); + table& set_usertype(usertype& user) { + return set_userdata(usertype_traits::name, user); } template - table& set_userdata(Key&& key, userdata& user) { + table& set_usertype(Key&& key, usertype& user) { push(); stack::push(state(), std::forward(key)); stack::push(state(), user); diff --git a/sol/userdata.hpp b/sol/usertype.hpp similarity index 85% rename from sol/userdata.hpp rename to sol/usertype.hpp index 48ddb023..f49806bc 100644 --- a/sol/userdata.hpp +++ b/sol/usertype.hpp @@ -20,11 +20,11 @@ // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #ifndef SOL_USERDATA_HPP -#define SOL_USERDATA_HPP +#define SOL_USERTYPE_HPP #include "state.hpp" #include "function_types.hpp" -#include "userdata_traits.hpp" +#include "usertype_traits.hpp" #include "default_construct.hpp" #include #include @@ -88,7 +88,7 @@ enum class meta_function { }; template -class userdata { +class usertype { private: typedef std::unordered_map, bool>> function_map_t; function_map_t indexmetafunctions, newindexmetafunctions; @@ -97,7 +97,6 @@ private: std::vector metafunctiontable; std::vector ptrmetafunctiontable; lua_CFunction cleanup; - std::string luaname; template struct constructor { @@ -121,7 +120,7 @@ private: } static int construct(lua_State* L) { - auto&& meta = userdata_traits::metatable; + auto&& meta = usertype_traits::metatable; call_syntax syntax = stack::get_call_syntax(L, meta); int argcount = lua_gettop(L); @@ -131,7 +130,7 @@ private: if(luaL_newmetatable(L, std::addressof(meta[0])) == 1) { lua_pop(L, 1); - std::string err = "Unable to get userdata metatable for "; + std::string err = "Unable to get usertype metatable for "; err += meta; throw error(err); } @@ -153,7 +152,7 @@ private: template void build_cleanup() { - cleanup = &base_function::userdata::gc; + cleanup = &base_function::usertype::gc; } template @@ -161,13 +160,13 @@ private: int extracount = 0; if(!indexmetafunctions.empty()) { if(index == nullptr) { - auto idxptr = detail::make_unique>("__index", nullptr); + auto idxptr = detail::make_unique>("__index", nullptr); index = &(idxptr->functions); functionnames.emplace_back("__index"); metafunctions.emplace_back(std::move(idxptr)); std::string& name = functionnames.back(); - metafunctiontable.push_back({ name.c_str(), &base_function::userdata::call }); - ptrmetafunctiontable.push_back({ name.c_str(), &base_function::userdata::ref_call }); + metafunctiontable.push_back({ name.c_str(), &base_function::usertype::call }); + ptrmetafunctiontable.push_back({ name.c_str(), &base_function::usertype::ref_call }); ++extracount; } auto& idx = *index; @@ -177,18 +176,18 @@ private: } if(!newindexmetafunctions.empty()) { if(newindex == nullptr) { - auto idxptr = detail::make_unique>("__newindex", nullptr); + auto idxptr = detail::make_unique>("__newindex", nullptr); newindex = &(idxptr->functions); functionnames.emplace_back("__newindex"); metafunctions.emplace_back(std::move(idxptr)); std::string& name = functionnames.back(); if(extracount > 0) { - metafunctiontable.push_back({ name.c_str(), &base_function::userdata::call }); - ptrmetafunctiontable.push_back({ name.c_str(), &base_function::userdata::ref_call }); + metafunctiontable.push_back({ name.c_str(), &base_function::usertype::call }); + ptrmetafunctiontable.push_back({ name.c_str(), &base_function::usertype::ref_call }); } else { - metafunctiontable.push_back({ name.c_str(), &base_function::userdata::call }); - ptrmetafunctiontable.push_back({ name.c_str(), &base_function::userdata::ref_call }); + metafunctiontable.push_back({ name.c_str(), &base_function::usertype::call }); + ptrmetafunctiontable.push_back({ name.c_str(), &base_function::usertype::ref_call }); } ++extracount; } @@ -215,8 +214,8 @@ private: bool build_function(std::true_type, function_map_t*&, function_map_t*&, std::string funcname, Ret Base::* func) { static_assert(std::is_base_of::value, "Any registered function must be part of the class"); typedef typename std::decay::type function_type; - indexmetafunctions.emplace(funcname, std::make_pair(detail::make_unique>(func), false)); - newindexmetafunctions.emplace(funcname, std::make_pair(detail::make_unique>(func), false)); + indexmetafunctions.emplace(funcname, std::make_pair(detail::make_unique>(func), false)); + newindexmetafunctions.emplace(funcname, std::make_pair(detail::make_unique>(func), false)); return false; } @@ -225,21 +224,21 @@ private: typedef Unqualified Argu; static_assert(std::is_base_of::value, "Any non-member-function must have a first argument which is covariant with the desired userdata type."); typedef typename std::decay::type function_type; - return detail::make_unique>(func); + return detail::make_unique>(func); } template std::unique_ptr make_variable_function(std::true_type, const std::string&, Ret Base::* func) { static_assert(std::is_base_of::value, "Any registered function must be part of the class"); typedef typename std::decay::type function_type; - return detail::make_unique>(func); + return detail::make_unique>(func); } template std::unique_ptr make_variable_function(std::false_type, const std::string&, Ret Base::* func) { static_assert(std::is_base_of::value, "Any registered function must be part of the class"); typedef typename std::decay::type function_type; - return detail::make_unique>(func); + return detail::make_unique>(func); } template @@ -255,7 +254,7 @@ private: typedef Unqualified TArgu; static_assert(std::is_base_of::value, "Any non-member-function must have a first argument which is covariant with the desired userdata type."); typedef typename std::decay::type function_type; - return detail::make_unique>(func); + return detail::make_unique>(func); } template @@ -268,7 +267,7 @@ private: auto indexmetamethod = std::find(meta_variable_names.begin(), meta_variable_names.end(), name); std::unique_ptr ptr(nullptr); if(indexmetamethod != meta_variable_names.end()) { - auto idxptr = detail::make_unique>(name, func); + auto idxptr = detail::make_unique>(name, func); switch(std::distance(indexmetamethod, meta_variable_names.end())) { case 0: index = &(idxptr->functions); @@ -285,8 +284,8 @@ private: ptr = make_function(funcname, std::forward(func)); } metafunctions.emplace_back(std::move(ptr)); - metafunctiontable.push_back( { name.c_str(), &base_function::userdata::call } ); - ptrmetafunctiontable.push_back( { name.c_str(), &base_function::userdata::ref_call } ); + metafunctiontable.push_back( { name.c_str(), &base_function::usertype::call } ); + ptrmetafunctiontable.push_back( { name.c_str(), &base_function::usertype::ref_call } ); return true; } indexmetafunctions.emplace(funcname, std::make_pair(make_function(funcname, std::forward(func)), true)); @@ -314,13 +313,10 @@ private: public: template - userdata(Args&&... args): userdata(userdata_traits::name, default_constructor, std::forward(args)...) {} + usertype(Args&&... args): usertype(default_constructor, std::forward(args)...) {} template - userdata(constructors c, Args&&... args): userdata(userdata_traits::name, std::move(c), std::forward(args)...) {} - - template - userdata(std::string name, constructors, Args&&... args): luaname(std::move(name)) { + usertype(constructors, Args&&... args) { functionnames.reserve(sizeof...(args) + 2); metafunctiontable.reserve(sizeof...(args)); ptrmetafunctiontable.reserve(sizeof...(args)); @@ -343,23 +339,15 @@ public: ptrmetafunctiontable.push_back({ nullptr, nullptr }); } - template - userdata(const char* name, constructors c, Args&&... args) : - userdata(std::string(name), std::move(c), std::forward(args)...) {} - - const std::string& name() const { - return luaname; - } - void push(lua_State* L) { // push pointer tables first, // but leave the regular T table on last // so it can be linked to a type for usage with `.new(...)` or `:new(...)` - push_metatable(L, userdata_traits::metatable, + push_metatable(L, usertype_traits::metatable, metafunctions, ptrmetafunctiontable); lua_pop(L, 1); - push_metatable(L, userdata_traits::metatable, + push_metatable(L, usertype_traits::metatable, metafunctions, metafunctiontable); set_global_deleter(L); } @@ -386,7 +374,7 @@ private: lua_setfield(L, -2, "__gc"); lua_setmetatable(L, -2); // gctable name by default has ♻ part of it - lua_setglobal(L, std::addressof(userdata_traits::gctable[0])); + lua_setglobal(L, std::addressof(usertype_traits::gctable[0])); } template @@ -407,12 +395,12 @@ private: namespace stack { template -struct pusher> { - static void push(lua_State* L, userdata& user) { +struct pusher> { + static void push(lua_State* L, usertype& user) { user.push(L); } }; } // stack } // sol -#endif // SOL_USERDATA_HPP +#endif // SOL_USERTYPE_HPP diff --git a/sol/userdata_traits.hpp b/sol/usertype_traits.hpp similarity index 80% rename from sol/userdata_traits.hpp rename to sol/usertype_traits.hpp index 18d28811..34871f55 100644 --- a/sol/userdata_traits.hpp +++ b/sol/usertype_traits.hpp @@ -19,29 +19,29 @@ // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -#ifndef SOL_USERDATA_TRAITS_HPP -#define SOL_USERDATA_TRAITS_HPP +#ifndef SOL_USERTYPE_TRAITS_HPP +#define SOL_USERTYPE_TRAITS_HPP #include "demangle.hpp" namespace sol { template -struct userdata_traits { +struct usertype_traits { static const std::string name; static const std::string metatable; static const std::string gctable; }; template -const std::string userdata_traits::name = detail::demangle(typeid(T)); +const std::string usertype_traits::name = detail::demangle(typeid(T)); template -const std::string userdata_traits::metatable = std::string("sol.").append(detail::demangle(typeid(T))); +const std::string usertype_traits::metatable = std::string("sol.").append(detail::demangle(typeid(T))); template -const std::string userdata_traits::gctable = std::string("sol.").append(detail::demangle(typeid(T))).append(".\xE2\x99\xBB"); +const std::string usertype_traits::gctable = std::string("sol.").append(detail::demangle(typeid(T))).append(".\xE2\x99\xBB"); } -#endif // SOL_USERDATA_TRAITS_HPP +#endif // SOL_USERTYPE_TRAITS_HPP diff --git a/tests.cpp b/tests.cpp index 3bc3aeb0..feec805f 100644 --- a/tests.cpp +++ b/tests.cpp @@ -568,12 +568,12 @@ TEST_CASE("tables/operator[]", "Check if operator[] retrieval and setting works REQUIRE_NOTHROW(assert1(lua.global_table())); } -TEST_CASE("tables/userdata", "Show that we can create classes from userdata and use them") { +TEST_CASE("tables/usertype", "Show that we can create classes from usertype and use them") { sol::state lua; - sol::userdata lc{ "add", &fuser::add, "add2", &fuser::add2 }; - lua.set_userdata(lc); + sol::usertype lc{ "add", &fuser::add, "add2", &fuser::add2 }; + lua.set_usertype(lc); lua.script("a = fuser:new()\n" "b = a:add(1)\n" @@ -595,13 +595,13 @@ TEST_CASE("tables/userdata", "Show that we can create classes from userdata and REQUIRE(cresult == 3); } -TEST_CASE("tables/userdata constructors", "Show that we can create classes from userdata and use them with multiple destructors") { +TEST_CASE("tables/usertype constructors", "Show that we can create classes from usertype and use them with multiple destructors") { sol::state lua; sol::constructors, sol::types, sol::types> con; - sol::userdata lc("crapola_fuser", con, "add", &crapola::fuser::add, "add2", &crapola::fuser::add2); - lua.set_userdata(lc); + sol::usertype lc(con, "add", &crapola::fuser::add, "add2", &crapola::fuser::add2); + lua.set_usertype(lc); lua.script( "a = crapola_fuser.new(2)\n" @@ -638,10 +638,10 @@ TEST_CASE("tables/userdata constructors", "Show that we can create classes from REQUIRE((z.as() == 9)); } -TEST_CASE("tables/userdata utility", "Show internal management of classes registered through new_userdata") { +TEST_CASE("tables/usertype utility", "Show internal management of classes registered through new_usertype") { sol::state lua; - lua.new_userdata("fuser", "add", &fuser::add, "add2", &fuser::add2); + lua.new_usertype("fuser", "add", &fuser::add, "add2", &fuser::add2); lua.script("a = fuser.new()\n" "b = a:add(1)\n" @@ -663,21 +663,21 @@ TEST_CASE("tables/userdata utility", "Show internal management of classes regist REQUIRE(cresult == 3); } -TEST_CASE("tables/userdata utility derived", "userdata classes must play nice when a derived class does not overload a publically visible base function") { +TEST_CASE("tables/usertype utility derived", "usertype classes must play nice when a derived class does not overload a publically visible base function") { sol::state lua; lua.open_libraries(sol::lib::base); sol::constructors> basector; - sol::userdata baseuserdata("Base", basector, "get_num", &Base::get_num); + sol::usertype baseusertype(basector, "get_num", &Base::get_num); - lua.set_userdata(baseuserdata); + lua.set_usertype(baseusertype); lua.script("base = Base.new(5)"); lua.script("print(base:get_num())"); sol::constructors> derivedctor; - sol::userdata deriveduserdata("Derived", derivedctor, "get_num", &Derived::get_num, "get_num_10", &Derived::get_num_10); + sol::usertype derivedusertype(derivedctor, "get_num", &Derived::get_num, "get_num_10", &Derived::get_num_10); - lua.set_userdata(deriveduserdata); + lua.set_usertype(derivedusertype); lua.script("derived = Derived.new(7)"); lua.script("dgn10 = derived:get_num_10()\nprint(dgn10)"); @@ -687,11 +687,11 @@ TEST_CASE("tables/userdata utility derived", "userdata classes must play nice wh REQUIRE((lua.get("dgn") == 7)); } -TEST_CASE("tables/self-referential userdata", "userdata classes must play nice when C++ object types are requested for C++ code") { +TEST_CASE("tables/self-referential usertype", "usertype classes must play nice when C++ object types are requested for C++ code") { sol::state lua; lua.open_libraries(sol::lib::base); - lua.new_userdata("test", "g", &self_test::g, "f", &self_test::f); + lua.new_usertype("test", "g", &self_test::g, "f", &self_test::f); lua.script( "local a = test.new()\n" @@ -757,7 +757,7 @@ TEST_CASE("tables/issue-number-twenty-five", "Using pointers and references from sol::state lua; lua.open_libraries(sol::lib::base); - lua.new_userdata("test", "set", &test::set, "get", &test::get, "pointer_get", &test::pget, "fun", &test::fun, "create_get", &test::create_get); + lua.new_usertype("test", "set", &test::set, "get", &test::get, "pointer_get", &test::pget, "fun", &test::fun, "create_get", &test::create_get); REQUIRE_NOTHROW(lua.script("x = test.new()\n" "x:set():get()")); REQUIRE_NOTHROW(lua.script("y = x:pointer_get()")); @@ -770,7 +770,7 @@ TEST_CASE("tables/issue-number-twenty-five", "Using pointers and references from REQUIRE_NOTHROW(lua.script("assert(y:set():get() == 10, '...')")); } -TEST_CASE("userdata/issue-number-thirty-five", "using value types created from lua-called C++ code, fixing user-defined types with constructors") { +TEST_CASE("usertype/issue-number-thirty-five", "using value types created from lua-called C++ code, fixing user-defined types with constructors") { struct Line { Vec p1, p2; Line() : p1{0, 0, 0}, p2{0, 0, 0} {} @@ -783,13 +783,13 @@ TEST_CASE("userdata/issue-number-thirty-five", "using value types created from l lua.open_libraries(sol::lib::base); sol::constructors, sol::types, sol::types> lctor; - sol::userdata ludata("Line", lctor); - lua.set_userdata(ludata); + sol::usertype ludata(lctor); + lua.set_usertype("Line", ludata); sol::constructors> ctor; - sol::userdata udata("Vec", ctor, "normalized", &Vec::normalized, "length", &Vec::length); + sol::usertype udata(ctor, "normalized", &Vec::normalized, "length", &Vec::length); - lua.set_userdata(udata); + lua.set_usertype(udata); REQUIRE_NOTHROW(lua.script("v = Vec.new(1, 2, 3)\n" "print(v:length())")); @@ -797,18 +797,18 @@ TEST_CASE("userdata/issue-number-thirty-five", "using value types created from l "print(v:normalized():length())" )); } -TEST_CASE("userdata/lua-stored-userdata", "ensure userdata values can be stored without keeping userdata object alive") { +TEST_CASE("usertype/lua-stored-usertype", "ensure usertype values can be stored without keeping usertype object alive") { sol::state lua; lua.open_libraries(sol::lib::base); { sol::constructors> ctor; - sol::userdata udata("Vec", ctor, + sol::usertype udata(ctor, "normalized", &Vec::normalized, "length", &Vec::length); - lua.set_userdata(udata); - // userdata dies, but still usable in lua! + lua.set_usertype(udata); + // usertype dies, but still usable in lua! } REQUIRE_NOTHROW(lua.script("collectgarbage()\n" @@ -819,17 +819,17 @@ TEST_CASE("userdata/lua-stored-userdata", "ensure userdata values can be stored "print(v:normalized():length())" )); } -TEST_CASE("userdata/member-variables", "allow table-like accessors to behave as member variables for userdata") { +TEST_CASE("usertype/member-variables", "allow table-like accessors to behave as member variables for usertype") { sol::state lua; lua.open_libraries(sol::lib::base); sol::constructors> ctor; - sol::userdata udata("Vec", ctor, + sol::usertype udata(ctor, "x", &Vec::x, "y", &Vec::y, "z", &Vec::z, "normalized", &Vec::normalized, "length", &Vec::length); - lua.set_userdata(udata); + lua.set_usertype(udata); REQUIRE_NOTHROW(lua.script("v = Vec.new(1, 2, 3)\n" "v2 = Vec.new(0, 1, 0)\n" @@ -847,16 +847,16 @@ TEST_CASE("userdata/member-variables", "allow table-like accessors to behave as )); } -TEST_CASE("userdata/nonmember functions implement functionality", "let users set non-member functions that take unqualified T as first parameter to userdata") { +TEST_CASE("usertype/nonmember functions implement functionality", "let users set non-member functions that take unqualified T as first parameter to usertype") { sol::state lua; lua.open_libraries( sol::lib::base ); - lua.new_userdata( "giver", - "gief_stuff", giver::gief_stuff, - "gief", &giver::gief, - "__tostring", [](const giver& t) { - return std::to_string(t.a) + ": giving value"; - } + lua.new_usertype( "giver", + "gief_stuff", giver::gief_stuff, + "gief", &giver::gief, + "__tostring", [](const giver& t) { + return std::to_string(t.a) + ": giving value"; + } ).get( "giver" ) .set_function( "stuff", giver::stuff ); @@ -870,7 +870,8 @@ TEST_CASE("userdata/nonmember functions implement functionality", "let users set TEST_CASE("regressions/one", "issue number 48") { sol::state lua; - lua.new_userdata("vars", "boop", &vars::boop); + lua.new_usertype("vars", + "boop", &vars::boop); REQUIRE_NOTHROW(lua.script("beep = vars.new()\n" "beep.boop = 1")); // test for segfault @@ -883,7 +884,7 @@ TEST_CASE("regressions/one", "issue number 48") { TEST_CASE("references/get-set", "properly get and set with std::ref semantics. Note that to get, we must not use Unqualified on the type...") { sol::state lua; - lua.new_userdata("vars", + lua.new_usertype("vars", "boop", &vars::boop); vars var{}; From a767859e24adaee587e6aa326020ab9d7005277d Mon Sep 17 00:00:00 2001 From: ThePhD Date: Fri, 19 Sep 2014 10:45:13 -0400 Subject: [PATCH 2/4] Headerguard derp; fixed. --- sol/usertype.hpp | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/sol/usertype.hpp b/sol/usertype.hpp index f49806bc..4150dda8 100644 --- a/sol/usertype.hpp +++ b/sol/usertype.hpp @@ -19,7 +19,7 @@ // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -#ifndef SOL_USERDATA_HPP +#ifndef SOL_USERTYPE_HPP #define SOL_USERTYPE_HPP #include "state.hpp" @@ -351,8 +351,8 @@ public: metafunctions, metafunctiontable); set_global_deleter(L); } -private: +private: template static void push_metatable(lua_State* L, Meta&& metakey, MetaFuncs&& metafuncs, MetaFuncTable&& metafunctable) { luaL_newmetatable(L, std::addressof(metakey[0])); @@ -361,7 +361,6 @@ private: int up = push_upvalues(L, metafuncs); luaL_setfuncs(L, metafunctable.data(), up); } - } void set_global_deleter(lua_State* L) { From 79b04acd66b1675d96e54f8c61f07575c5e4111e Mon Sep 17 00:00:00 2001 From: ThePhD Date: Fri, 19 Sep 2014 11:08:44 -0400 Subject: [PATCH 3/4] Proper deprecation to alert users to API name changes. --- sol/deprecate.hpp | 11 ++++++++++- sol/state.hpp | 20 ++++++++++++++++++++ sol/table.hpp | 12 +++++++++++- sol/types.hpp | 2 +- sol/usertype.hpp | 16 ++++++++++++++++ 5 files changed, 58 insertions(+), 3 deletions(-) diff --git a/sol/deprecate.hpp b/sol/deprecate.hpp index f00a6e12..498e4807 100644 --- a/sol/deprecate.hpp +++ b/sol/deprecate.hpp @@ -32,4 +32,13 @@ #endif // compilers #endif // SOL_DEPRECATED -#endif // SOL_DEPRECATE_HPP +namespace sol { +namespace detail { + template + struct SOL_DEPRECATED deprecate_type { + using type = T; + }; +} // detail +} // sol + +#endif // SOL_DEPRECATE_HPP diff --git a/sol/state.hpp b/sol/state.hpp index 77a11ed8..8417f717 100644 --- a/sol/state.hpp +++ b/sol/state.hpp @@ -147,6 +147,26 @@ public: return *this; } + template + SOL_DEPRECATED table& set_userdata(usertype& user) { + return set_usertype(user); + } + + template + SOL_DEPRECATED table& set_userdata(Key&& key, usertype& user) { + return set_usertype(std::forward(key), user); + } + + template + SOL_DEPRECATED state& new_userdata(const std::string& name, Args&&... args) { + return new_usertype(name, std::forward(args)...); + } + + template + SOL_DEPRECATED state& new_userdata(const std::string& name, constructors ctor, Args&&... args) { + return new_usertype(name, std::move(ctor), std::forward(args)...); + } + template state& set_usertype(usertype& user) { return set_usertype(usertype_traits::name, user); diff --git a/sol/table.hpp b/sol/table.hpp index 04290db9..df2aa70a 100644 --- a/sol/table.hpp +++ b/sol/table.hpp @@ -83,9 +83,19 @@ public: return *this; } + template + SOL_DEPRECATED table& set_userdata(usertype& user) { + return set_usertype(user); + } + + template + SOL_DEPRECATED table& set_userdata(Key&& key, usertype& user) { + return set_usertype(std::forward(key), user); + } + template table& set_usertype(usertype& user) { - return set_userdata(usertype_traits::name, user); + return set_usertype(usertype_traits::name, user); } template diff --git a/sol/types.hpp b/sol/types.hpp index a72f09ce..31439924 100644 --- a/sol/types.hpp +++ b/sol/types.hpp @@ -90,7 +90,7 @@ inline std::string type_name(lua_State*L, type t) { } template -class userdata; +class usertype; class table; class function; class object; diff --git a/sol/usertype.hpp b/sol/usertype.hpp index 4150dda8..75468168 100644 --- a/sol/usertype.hpp +++ b/sol/usertype.hpp @@ -26,6 +26,7 @@ #include "function_types.hpp" #include "usertype_traits.hpp" #include "default_construct.hpp" +#include "deprecate.hpp" #include #include #include @@ -315,6 +316,18 @@ public: template usertype(Args&&... args): usertype(default_constructor, std::forward(args)...) {} + template + SOL_DEPRECATED usertype(std::string, Args&&... args): usertype(default_constructor, std::forward(args)...) {} + + template + SOL_DEPRECATED usertype(const char*, Args&&... args): usertype(default_constructor, std::forward(args)...) {} + + template + SOL_DEPRECATED usertype(std::string, constructors c, Args&&... args) : usertype(std::move(c), std::forward(args)...) {} + + template + SOL_DEPRECATED usertype(const char*, constructors c, Args&&... args) : usertype(std::move(c), std::forward(args)...) {} + template usertype(constructors, Args&&... args) { functionnames.reserve(sizeof...(args) + 2); @@ -392,6 +405,9 @@ private: } }; +template +using userdata = typename detail::deprecate_type>::type; + namespace stack { template struct pusher> { From 792b61321d1845bb085660d0f6276facdca13f7e Mon Sep 17 00:00:00 2001 From: ThePhD Date: Fri, 19 Sep 2014 11:39:35 -0400 Subject: [PATCH 4/4] Use 2 string types to properly filter bad name-based usertype overloads. --- sol/usertype.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/sol/usertype.hpp b/sol/usertype.hpp index 75468168..e2316029 100644 --- a/sol/usertype.hpp +++ b/sol/usertype.hpp @@ -317,10 +317,10 @@ public: usertype(Args&&... args): usertype(default_constructor, std::forward(args)...) {} template - SOL_DEPRECATED usertype(std::string, Args&&... args): usertype(default_constructor, std::forward(args)...) {} + SOL_DEPRECATED usertype(std::string, std::string, Args&&... args): usertype(default_constructor, std::forward(args)...) {} template - SOL_DEPRECATED usertype(const char*, Args&&... args): usertype(default_constructor, std::forward(args)...) {} + SOL_DEPRECATED usertype(const char*, std::string, Args&&... args): usertype(default_constructor, std::forward(args)...) {} template SOL_DEPRECATED usertype(std::string, constructors c, Args&&... args) : usertype(std::move(c), std::forward(args)...) {}