/*
* Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com)
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include "runtime.h"
#include
namespace AddOn {
LuaError::LuaError(const QString &reason): BaseError(reason) {}
RaiiLuaState::RaiiLuaState(const QString &name, std::chrono::microseconds timeLimit)
: mLua(luaL_newstate()) {
mExtraState.insert(mLua, {name, timeLimit, /* .timeStart = */ {}});
}
RaiiLuaState::RaiiLuaState(RaiiLuaState &&rhs)
: mLua(rhs.mLua) {
rhs.mLua = nullptr;
}
RaiiLuaState &RaiiLuaState::operator=(RaiiLuaState &&rhs) {
// do not check self assignment intentionally, following STL semantics
this->~RaiiLuaState();
// if self assignment, move constructor guarantees to place the object
// in a valid (but unspecified) state
new (this) RaiiLuaState(std::move(rhs));
return *this;
}
RaiiLuaState::~RaiiLuaState() {
if (mLua) {
mExtraState.remove(mLua);
lua_close(mLua);
}
}
bool RaiiLuaState::fetchBoolean(int index)
{
return lua_toboolean(mLua, index);
}
long long RaiiLuaState::fetchInteger(int index)
{
return lua_tointeger(mLua, index);
}
double RaiiLuaState::fetchNumber(int index)
{
return lua_tonumber(mLua, index);
}
QString RaiiLuaState::fetchString(int index)
{
return lua_tostring(mLua, index);
}
QJsonValue RaiiLuaState::fetch(int index)
{
return fetchValueImpl(index, 0);
}
QString RaiiLuaState::fetchString(lua_State *L, int index)
{
return lua_tostring(L, index);
}
bool RaiiLuaState::popBoolean()
{
bool value = lua_toboolean(mLua, -1);
lua_pop(mLua, 1);
return value;
}
QString RaiiLuaState::popString()
{
QString value = fetchString(-1);
lua_pop(mLua, 1);
return value;
}
QJsonValue RaiiLuaState::pop()
{
QJsonValue value = fetch(-1);
lua_pop(mLua, 1);
return value;
}
void RaiiLuaState::push(decltype(nullptr))
{
lua_pushnil(mLua);
}
void RaiiLuaState::push(const QMap &value)
{
lua_newtable(mLua);
for (auto it = value.cbegin(); it != value.cend(); ++it) {
lua_pushstring(mLua, it.key().toUtf8().constData());
lua_pushcfunction(mLua, it.value());
lua_settable(mLua, -3);
}
}
void RaiiLuaState::push(lua_State *L, const QString &value)
{
lua_pushstring(L, value.toUtf8().constData());
}
void RaiiLuaState::push(lua_State *L, const QStringList &value)
{
lua_newtable(L);
for (int i = 0; i < value.length(); i++) {
lua_pushinteger(L, i + 1);
lua_pushstring(L, value[i].toUtf8().constData());
lua_settable(L, -3);
}
}
int RaiiLuaState::loadBuffer(const QByteArray &buff, const QString &name)
{
return luaL_loadbuffer(mLua, buff.constData(), buff.size(), name.toUtf8().constData());
}
void RaiiLuaState::openLibs()
{
luaL_openlibs(mLua);
}
int RaiiLuaState::pCall(int nargs, int nresults, int msgh)
{
return lua_pcall(mLua, nargs, nresults, msgh);
}
void RaiiLuaState::setGlobal(const QString &name)
{
return lua_setglobal(mLua, name.toUtf8().constData());
}
void RaiiLuaState::setHook(lua_Hook f, int mask, int count)
{
lua_sethook(mLua, f, mask, count);
}
void RaiiLuaState::setTimeStart() {
extraState().timeStart = std::chrono::system_clock::now();
}
LuaExtraState &RaiiLuaState::extraState() {
return mExtraState[mLua];
}
LuaExtraState &RaiiLuaState::extraState(lua_State *lua) {
return mExtraState[lua];
}
QJsonValue RaiiLuaState::fetchTableImpl(int index, int depth)
{
if (depth == 1)
// check stack size at first recursion to avoid multiple reallocations
lua_checkstack(mLua, LUA_STACK_SIZE);
if (depth > TABLE_MAX_DEPTH)
throw LuaError("Lua runtime error: table nested too deeply");
push(nullptr); // make sure lua_next starts at beginning
int newIndex = index < 0 ? index - 1 : index; // after push negative index changes
QJsonObject hashPart;
QJsonArray arrayPart;
// here we take the fact that Lua iterates array part first
bool processingArrayPart = true;
while (lua_next(mLua, newIndex)) {
QJsonValue v = pop();
if (processingArrayPart && lua_isinteger(mLua, -1) && fetchInteger(-1) == arrayPart.size() + 1)
// we are still in array part
arrayPart.push_back(v);
else {
// we have stepped in hash part
processingArrayPart = false;
QString k = fetchString(-1);
hashPart[k] = v;
}
}
if (!arrayPart.empty())
if (!hashPart.empty())
// table contains both array part and hash part, malformed
throw LuaError("Lua type error: table contains both array part and hash part.");
else
// is array
return arrayPart;
else
if (!hashPart.empty())
// is object
return hashPart;
else
// empty table, okay
// return null since we cannot determine
return {};
}
QJsonValue RaiiLuaState::fetchValueImpl(int index, int depth)
{
if (lua_isnil(mLua, index))
return {};
else if (lua_isboolean(mLua, index))
return fetchBoolean(index);
else if (lua_isinteger(mLua, index))
return fetchInteger(index);
else if (lua_isnumber(mLua, index))
return fetchNumber(index);
else if (lua_isstring(mLua, index))
return fetchString(index);
else if (lua_istable(mLua, index))
return fetchTableImpl(index, depth + 1);
else
throw LuaError(QString("Lua type error: unknown type %1.").arg(lua_typename(mLua, index)));
}
QHash RaiiLuaState::mExtraState;
} // namespace AddOn