// sol3 // The MIT License (MIT) // Copyright (c) 2013-2018 Rapptz, ThePhD and contributors // Permission is hereby granted, free of charge, to any person obtaining a copy of // this software and associated documentation files (the "Software"), to deal in // the Software without restriction, including without limitation the rights to // use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of // the Software, and to permit persons to whom the Software is furnished to do so, // subject to the following conditions: // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS // FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR // COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #ifndef SOL_STACK_QUALIFIED_GET_HPP #define SOL_STACK_QUALIFIED_GET_HPP #include "stack_get_unqualified.hpp" namespace sol { namespace stack { template struct qualified_getter::value && is_unique_usertype>::value && !std::is_void>::template rebind_base>::value >> { typedef unique_usertype_traits> u_traits; typedef typename u_traits::type T; typedef typename u_traits::actual_type Real; typedef typename u_traits::template rebind_base rebind_t; static Real get(lua_State* L, int index, record& tracking) { tracking.use(1); void* memory = lua_touserdata(L, index); memory = detail::align_usertype_unique_destructor(memory); detail::unique_destructor& pdx = *static_cast(memory); if (&detail::usertype_unique_alloc_destroy == pdx) { memory = detail::align_usertype_unique_tag(memory); memory = detail::align_usertype_unique(memory); Real* mem = static_cast(memory); return *mem; } Real r(nullptr); if constexpr (!derive::value) { // TODO: abort / terminate, maybe only in debug modes? return r; } else { memory = detail::align_usertype_unique_tag(memory); detail::unique_tag& ic = *reinterpret_cast(memory); memory = detail::align_usertype_unique(memory); string_view ti = usertype_traits::qualified_name(); string_view rebind_ti = usertype_traits::qualified_name(); int cast_operation = ic(memory, &r, ti, rebind_ti); switch (cast_operation) { case 1: { // it's a perfect match, // alias memory directly Real* mem = static_cast(memory); return *mem; } case 2: // it's a base match, return the // aliased creation return std::move(r); default: // uh oh.. break; } // TODO: abort / terminate, maybe only in debug modes? return r; } } }; template struct qualified_getter::value && is_container>::value && std::is_default_constructible>::value && !is_lua_primitive::value && !is_transparent_argument::value >> { static T get(lua_State* L, int index, record& tracking) { if (type_of(L, index) == type::userdata) { return stack_detail::unchecked_unqualified_get(L, index, tracking); } else { return stack_detail::unchecked_unqualified_get>(L, index, tracking); } } }; } } // namespace sol::stack #endif // SOL_STACK_QUALIFIED_GET_HPP