1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-16 15:29:55 +00:00

Fix crash when throwing in index meta methods

This commit is contained in:
uramer 2024-02-12 22:25:12 +01:00 committed by Anton Uramer
parent ec1cf46ec7
commit aa4303fc38
2 changed files with 23 additions and 2 deletions

View file

@ -477,6 +477,7 @@ if (NOT WIN32)
endif()
# C++ library binding to Lua
add_compile_definitions(SOL_ALL_SAFETIES_ON)
set(SOL_INCLUDE_DIR ${OpenMW_SOURCE_DIR}/extern/sol3)
set(SOL_CONFIG_DIR ${OpenMW_SOURCE_DIR}/extern/sol_config)

View file

@ -232,16 +232,36 @@ namespace LuaUtil
}
}
// work around for a (likely) sol3 bug
// when the index meta method throws, simply calling table.get crashes instead of re-throwing the error
template <class Key>
sol::object safe_get(const sol::table& table, const Key& key)
{
auto index = table.traverse_raw_get<sol::optional<sol::main_protected_function>>(
sol::metatable_key, sol::meta_function::index);
if (index)
{
sol::protected_function_result result = index.value()(table, key);
if (result.valid())
return result.get<sol::object>();
else
throw result.get<sol::error>();
}
else
return table.raw_get<sol::object>(key);
}
// getFieldOrNil(table, "a", "b", "c") returns table["a"]["b"]["c"] or nil if some of the fields doesn't exist.
template <class... Str>
sol::object getFieldOrNil(const sol::object& table, std::string_view first, const Str&... str)
{
if (!table.is<sol::table>())
return sol::nil;
sol::object value = safe_get(table.as<sol::table>(), first);
if constexpr (sizeof...(str) == 0)
return table.as<sol::table>()[first];
return value;
else
return getFieldOrNil(table.as<sol::table>()[first], str...);
return getFieldOrNil(value, str...);
}
// String representation of a Lua object. Should be used for debugging/logging purposes only.