1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-08 09:37:53 +00:00
OpenMW/apps/openmw/mwlua/userdataserializer.cpp

118 lines
4.4 KiB
C++

#include "userdataserializer.hpp"
#include <cstring>
#include <components/lua/serialization.hpp>
#include <components/misc/endianness.hpp>
#include "object.hpp"
namespace MWLua
{
class Serializer final : public LuaUtil::UserdataSerializer
{
public:
explicit Serializer(bool localSerializer, std::map<int, int>* contentFileMapping)
: mLocalSerializer(localSerializer)
, mContentFileMapping(contentFileMapping)
{
}
private:
// Appends serialized sol::userdata to the end of BinaryData.
// Returns false if this type of userdata is not supported by this serializer.
bool serialize(LuaUtil::BinaryData& out, const sol::userdata& data) const override
{
if (data.is<GObject>() || data.is<LObject>())
{
appendRefNum(out, data.as<Object>().id());
return true;
}
if (data.is<GObjectList>())
{
appendObjectIdList(out, data.as<GObjectList>().mIds);
return true;
}
if (data.is<LObjectList>())
{
appendObjectIdList(out, data.as<LObjectList>().mIds);
return true;
}
return false;
}
constexpr static std::string_view sObjListTypeName = "objlist";
void appendObjectIdList(LuaUtil::BinaryData& out, const ObjectIdList& objList) const
{
static_assert(sizeof(ESM::RefNum) == 8);
if constexpr (Misc::IS_LITTLE_ENDIAN)
append(out, sObjListTypeName, objList->data(), objList->size() * sizeof(ESM::RefNum));
else
{
std::vector<ESM::RefNum> buf;
buf.reserve(objList->size());
for (const ESM::RefNum& v : *objList)
buf.push_back({ Misc::toLittleEndian(v.mIndex), Misc::toLittleEndian(v.mContentFile) });
append(out, sObjListTypeName, buf.data(), buf.size() * sizeof(ESM::RefNum));
}
}
void adjustRefNum(ESM::RefNum& refNum) const
{
if (refNum.hasContentFile() && mContentFileMapping)
{
auto iter = mContentFileMapping->find(refNum.mContentFile);
if (iter != mContentFileMapping->end())
refNum.mContentFile = iter->second;
}
}
// Deserializes userdata of type "typeName" from binaryData. Should push the result on stack using
// sol::stack::push. Returns false if this type is not supported by this serializer.
bool deserialize(std::string_view typeName, std::string_view binaryData, lua_State* lua) const override
{
if (typeName == sRefNumTypeName)
{
ObjectId id = loadRefNum(binaryData);
adjustRefNum(id);
if (mLocalSerializer)
sol::stack::push<LObject>(lua, LObject(id));
else
sol::stack::push<GObject>(lua, GObject(id));
return true;
}
if (typeName == sObjListTypeName)
{
if (binaryData.size() % sizeof(ESM::RefNum) != 0)
throw std::runtime_error("Invalid size of ObjectIdList in MWLua::Serializer");
ObjectIdList objList = std::make_shared<std::vector<ESM::RefNum>>();
objList->resize(binaryData.size() / sizeof(ESM::RefNum));
std::memcpy(objList->data(), binaryData.data(), binaryData.size());
for (ESM::RefNum& id : *objList)
{
id.mIndex = Misc::fromLittleEndian(id.mIndex);
id.mContentFile = Misc::fromLittleEndian(id.mContentFile);
adjustRefNum(id);
}
if (mLocalSerializer)
sol::stack::push<LObjectList>(lua, LObjectList{ std::move(objList) });
else
sol::stack::push<GObjectList>(lua, GObjectList{ std::move(objList) });
return true;
}
return false;
}
bool mLocalSerializer;
std::map<int, int>* mContentFileMapping;
};
std::unique_ptr<LuaUtil::UserdataSerializer> createUserdataSerializer(
bool local, std::map<int, int>* contentFileMapping)
{
return std::make_unique<Serializer>(local, contentFileMapping);
}
}