mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-09 21:42:13 +00:00
185 lines
6.5 KiB
C++
185 lines
6.5 KiB
C++
#include "luastate.hpp"
|
|
|
|
#ifndef NO_LUAJIT
|
|
#include <luajit.h>
|
|
#endif // NO_LUAJIT
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
namespace LuaUtil
|
|
{
|
|
|
|
static std::string packageNameToPath(std::string_view packageName)
|
|
{
|
|
std::string res(packageName);
|
|
std::replace(res.begin(), res.end(), '.', '/');
|
|
res.append(".lua");
|
|
return res;
|
|
}
|
|
|
|
static const std::string safeFunctions[] = {
|
|
"assert", "error", "ipairs", "next", "pairs", "pcall", "select", "tonumber", "tostring",
|
|
"type", "unpack", "xpcall", "rawequal", "rawget", "rawset", "getmetatable", "setmetatable"};
|
|
static const std::string safePackages[] = {"coroutine", "math", "string", "table"};
|
|
|
|
LuaState::LuaState(const VFS::Manager* vfs) : mVFS(vfs)
|
|
{
|
|
mLua.open_libraries(sol::lib::base, sol::lib::coroutine, sol::lib::math,
|
|
sol::lib::string, sol::lib::table, sol::lib::debug);
|
|
|
|
mLua["math"]["randomseed"](static_cast<unsigned>(std::time(nullptr)));
|
|
mLua["math"]["randomseed"] = sol::nil;
|
|
|
|
mLua["writeToLog"] = [](std::string_view s) { Log(Debug::Level::Info) << s; };
|
|
mLua.script(R"(printToLog = function(name, ...)
|
|
local msg = name
|
|
for _, v in ipairs({...}) do
|
|
msg = msg .. '\t' .. tostring(v)
|
|
end
|
|
return writeToLog(msg)
|
|
end)");
|
|
mLua.script("printGen = function(name) return function(...) return printToLog(name, ...) end end");
|
|
|
|
// Some fixes for compatibility between different Lua versions
|
|
if (mLua["unpack"] == sol::nil)
|
|
mLua["unpack"] = mLua["table"]["unpack"];
|
|
else if (mLua["table"]["unpack"] == sol::nil)
|
|
mLua["table"]["unpack"] = mLua["unpack"];
|
|
|
|
mSandboxEnv = sol::table(mLua, sol::create);
|
|
mSandboxEnv["_VERSION"] = mLua["_VERSION"];
|
|
for (const std::string& s : safeFunctions)
|
|
{
|
|
if (mLua[s] == sol::nil) throw std::logic_error("Lua function not found: " + s);
|
|
mSandboxEnv[s] = mLua[s];
|
|
}
|
|
for (const std::string& s : safePackages)
|
|
{
|
|
if (mLua[s] == sol::nil) throw std::logic_error("Lua package not found: " + s);
|
|
mCommonPackages[s] = mSandboxEnv[s] = makeReadOnly(mLua[s]);
|
|
}
|
|
}
|
|
|
|
LuaState::~LuaState()
|
|
{
|
|
// Should be cleaned before destructing mLua.
|
|
mCommonPackages.clear();
|
|
mSandboxEnv = sol::nil;
|
|
}
|
|
|
|
sol::table makeReadOnly(sol::table table)
|
|
{
|
|
if (table == sol::nil)
|
|
return table;
|
|
if (table.is<sol::userdata>())
|
|
return table; // it is already userdata, no sense to wrap it again
|
|
|
|
lua_State* lua = table.lua_state();
|
|
table[sol::meta_function::index] = table;
|
|
sol::stack::push(lua, std::move(table));
|
|
lua_newuserdata(lua, 0);
|
|
lua_pushvalue(lua, -2);
|
|
lua_setmetatable(lua, -2);
|
|
return sol::stack::pop<sol::table>(lua);
|
|
}
|
|
|
|
sol::table getMutableFromReadOnly(const sol::userdata& ro)
|
|
{
|
|
lua_State* lua = ro.lua_state();
|
|
sol::stack::push(lua, ro);
|
|
int ok = lua_getmetatable(lua, -1);
|
|
assert(ok);
|
|
(void)ok;
|
|
sol::table res = sol::stack::pop<sol::table>(lua);
|
|
lua_pop(lua, 1);
|
|
return res;
|
|
}
|
|
|
|
void LuaState::addCommonPackage(const std::string& packageName, const sol::object& package)
|
|
{
|
|
if (package.is<sol::function>())
|
|
mCommonPackages[packageName] = package;
|
|
else
|
|
mCommonPackages[packageName] = makeReadOnly(package);
|
|
}
|
|
|
|
sol::protected_function_result LuaState::runInNewSandbox(
|
|
const std::string& path, const std::string& namePrefix,
|
|
const std::map<std::string, sol::object>& packages, const sol::object& hiddenData)
|
|
{
|
|
sol::protected_function script = loadScript(path);
|
|
|
|
sol::environment env(mLua, sol::create, mSandboxEnv);
|
|
std::string envName = namePrefix + "[" + path + "]:";
|
|
env["print"] = mLua["printGen"](envName);
|
|
|
|
sol::table loaded(mLua, sol::create);
|
|
for (const auto& [key, value] : mCommonPackages)
|
|
loaded[key] = value;
|
|
for (const auto& [key, value] : packages)
|
|
loaded[key] = value;
|
|
env["require"] = [this, env, loaded, hiddenData](std::string_view packageName)
|
|
{
|
|
sol::table packages = loaded;
|
|
sol::object package = packages[packageName];
|
|
if (package == sol::nil)
|
|
{
|
|
sol::protected_function packageLoader = loadScript(packageNameToPath(packageName));
|
|
sol::set_environment(env, packageLoader);
|
|
package = throwIfError(packageLoader());
|
|
if (!package.is<sol::table>())
|
|
throw std::runtime_error("Lua package must return a table.");
|
|
packages[packageName] = package;
|
|
}
|
|
else if (package.is<sol::function>())
|
|
package = packages[packageName] = call(package.as<sol::protected_function>(), hiddenData);
|
|
return package;
|
|
};
|
|
|
|
sol::set_environment(env, script);
|
|
return call(script);
|
|
}
|
|
|
|
sol::protected_function_result LuaState::throwIfError(sol::protected_function_result&& res)
|
|
{
|
|
if (!res.valid() && static_cast<int>(res.get_type()) == LUA_TSTRING)
|
|
throw std::runtime_error("Lua error: " + res.get<std::string>());
|
|
else
|
|
return std::move(res);
|
|
}
|
|
|
|
sol::protected_function LuaState::loadScript(const std::string& path)
|
|
{
|
|
auto iter = mCompiledScripts.find(path);
|
|
if (iter != mCompiledScripts.end())
|
|
return mLua.load(iter->second.as_string_view(), path, sol::load_mode::binary);
|
|
|
|
std::string fileContent(std::istreambuf_iterator<char>(*mVFS->get(path)), {});
|
|
sol::load_result res = mLua.load(fileContent, path, sol::load_mode::text);
|
|
if (!res.valid())
|
|
throw std::runtime_error("Lua error: " + res.get<std::string>());
|
|
mCompiledScripts[path] = res.get<sol::function>().dump();
|
|
return res;
|
|
}
|
|
|
|
std::string getLuaVersion()
|
|
{
|
|
#ifdef NO_LUAJIT
|
|
return LUA_RELEASE;
|
|
#else
|
|
return LUA_RELEASE " (" LUAJIT_VERSION ")";
|
|
#endif
|
|
}
|
|
|
|
std::string toString(const sol::object& obj)
|
|
{
|
|
if (obj == sol::nil)
|
|
return "nil";
|
|
else if (obj.get_type() == sol::type::string)
|
|
return "\"" + obj.as<std::string>() + "\"";
|
|
else
|
|
return call(sol::state_view(obj.lua_state())["tostring"], obj);
|
|
}
|
|
|
|
}
|