- made reading of objects from the savegame work.

It turned out this may not be done automatically when opening the savegame - it has to be done later, after the pre-spawned map thinkers and all connected objects have been destroyed.
The object deserializer also has to be rather careful about dealing with parse errors, because if something goes wrong a whole batch of uninitialized or partially initialized objects will be left behind to destroy.
This means that no object class may assume that anything but the default constructor has been run on it and needs to check any variable it may reference.
This commit is contained in:
Christoph Oelckers 2016-09-23 09:38:55 +02:00
parent a83ea4ddd2
commit 5a3f1dcdb6
4 changed files with 62 additions and 28 deletions

View file

@ -2941,6 +2941,7 @@ void DACSThinker::Serialize(FSerializer &arc)
arc(nullptr, srs); arc(nullptr, srs);
RunningScripts[srs.scriptnum] = srs.lscript; RunningScripts[srs.scriptnum] = srs.lscript;
} }
arc.EndArray();
} }
} }
} }

View file

@ -904,7 +904,7 @@ void G_SerializeLevel(FSerializer &arc, bool hubload)
if (arc.isReading()) if (arc.isReading())
{ {
P_DestroyThinkers(hubload); P_DestroyThinkers(hubload);
// ReadObjects arc.ReadObjects();
} }
arc("level.flags", level.flags) arc("level.flags", level.flags)

View file

@ -199,6 +199,7 @@ struct FReader
rapidjson::Document mDoc; rapidjson::Document mDoc;
TArray<DObject *> mDObjects; TArray<DObject *> mDObjects;
int mPlayers[MAXPLAYERS]; int mPlayers[MAXPLAYERS];
bool mObjectsRead;
FReader(const char *buffer, size_t length, bool randomaccess) FReader(const char *buffer, size_t length, bool randomaccess)
{ {
@ -206,6 +207,7 @@ struct FReader
mDoc.Parse(buffer, length); mDoc.Parse(buffer, length);
mObjects.Push(FJSONObject(&mDoc, randomaccess)); mObjects.Push(FJSONObject(&mDoc, randomaccess));
memset(mPlayers, 0, sizeof(mPlayers)); memset(mPlayers, 0, sizeof(mPlayers));
mObjectsRead = false;
} }
rapidjson::Value *FindKey(const char *key) rapidjson::Value *FindKey(const char *key)
@ -266,7 +268,6 @@ bool FSerializer::OpenReader(const char *buffer, size_t length)
{ {
if (w != nullptr || r != nullptr) return false; if (w != nullptr || r != nullptr) return false;
r = new FReader(buffer, length, true); r = new FReader(buffer, length, true);
ReadObjects();
return true; return true;
} }
@ -291,7 +292,6 @@ bool FSerializer::OpenReader(FCompressedBuffer *input)
input->Decompress(unpacked); input->Decompress(unpacked);
r = new FReader(unpacked, input->mSize, true); r = new FReader(unpacked, input->mSize, true);
} }
ReadObjects();
return true; return true;
} }
@ -416,6 +416,7 @@ void FSerializer::EndObject(bool endwarning)
} }
else else
{ {
assert(false && "EndObject call not inside an object");
I_Error("EndObject call not inside an object"); I_Error("EndObject call not inside an object");
} }
} }
@ -425,6 +426,7 @@ void FSerializer::EndObject(bool endwarning)
{ {
if (r->mObjects.Last().mIterator != r->mObjects.Last().mObject->MemberEnd()) if (r->mObjects.Last().mIterator != r->mObjects.Last().mObject->MemberEnd())
{ {
assert(false && "Incomplete read of sequential object");
I_Error("Incomplete read of sequential object"); I_Error("Incomplete read of sequential object");
} }
} }
@ -797,6 +799,7 @@ void FSerializer::WriteObjects()
void FSerializer::ReadObjects() void FSerializer::ReadObjects()
{ {
bool founderrors = false; bool founderrors = false;
unsigned i;
if (isReading() && BeginArray("objects")) if (isReading() && BeginArray("objects"))
{ {
@ -827,6 +830,7 @@ void FSerializer::ReadObjects()
} }
} }
// Now that everything has been created and we can retrieve the pointers we can deserialize it. // Now that everything has been created and we can retrieve the pointers we can deserialize it.
r->mObjectsRead = true;
if (!founderrors) if (!founderrors)
{ {
@ -834,14 +838,17 @@ void FSerializer::ReadObjects()
r->mObjects.Last().mIndex = 0; r->mObjects.Last().mIndex = 0;
for (unsigned i = 0; i < r->mDObjects.Size(); i++) for (unsigned i = 0; i < r->mDObjects.Size(); i++)
{
auto obj = r->mDObjects[i];
try
{ {
if (BeginObject(nullptr)) if (BeginObject(nullptr))
{ {
Discard("classtype");
int pindex = -1; int pindex = -1;
Discard("classtype");
Serialize(*this, "playerindex", pindex, nullptr); Serialize(*this, "playerindex", pindex, nullptr);
auto obj = r->mDObjects[i]; if (obj != nullptr)
{
if (pindex >= 0 && pindex < MAXPLAYERS) if (pindex >= 0 && pindex < MAXPLAYERS)
{ {
obj->ObjectFlags |= OF_LoadedPlayer; obj->ObjectFlags |= OF_LoadedPlayer;
@ -849,31 +856,37 @@ void FSerializer::ReadObjects()
} }
obj->SerializeUserVars(*this); obj->SerializeUserVars(*this);
obj->Serialize(*this); obj->Serialize(*this);
try }
{
EndObject(true); EndObject(true);
} }
}
catch (CRecoverableError &err) catch (CRecoverableError &err)
{ {
I_Error("%s\n while restoring %s", err.GetMessage(), obj->GetClass()->TypeName.GetChars()); I_Error("%s\n while restoring %s", err.GetMessage(),obj? obj->GetClass()->TypeName.GetChars() : "invalid object");
}
} }
} }
} }
EndArray(); EndArray();
DThinker::bSerialOverride = false; DThinker::bSerialOverride = false;
if (founderrors)
{
I_Error("Failed to restore all objects in savegame");
}
} }
catch(...) catch(...)
{ {
// nuke all objects we created here.
for (auto obj : r->mDObjects)
{
obj->Destroy();
}
r->mDObjects.Clear();
// make sure this flag gets unset, even if something in here throws an error. // make sure this flag gets unset, even if something in here throws an error.
DThinker::bSerialOverride = false; DThinker::bSerialOverride = false;
throw; throw;
} }
} }
if (founderrors)
{
I_Error("Failed to restore all objects in savegame");
}
} }
//========================================================================== //==========================================================================
@ -1367,13 +1380,17 @@ FSerializer &Serialize(FSerializer &arc, const char *key, DObject *&value, DObje
if (retcode) *retcode = true; if (retcode) *retcode = true;
if (arc.isWriting()) if (arc.isWriting())
{ {
if (value != nullptr && !(value->ObjectFlags & OF_EuthanizeMe)) if (value != nullptr)
{ {
int ndx; int ndx;
if (value == WP_NOCHANGE) if (value == WP_NOCHANGE)
{ {
ndx = -1; ndx = -1;
} }
else if (value->ObjectFlags & OF_EuthanizeMe)
{
return arc;
}
else else
{ {
int *pndx = arc.w->mObjectMap.CheckKey(value); int *pndx = arc.w->mObjectMap.CheckKey(value);
@ -1392,15 +1409,31 @@ FSerializer &Serialize(FSerializer &arc, const char *key, DObject *&value, DObje
} }
else else
{ {
if (!arc.r->mObjectsRead)
{
// If you want to read objects, you MUST call ReadObjects first, even if there's only nullptr's.
assert(false && "Attempt to read object reference without calling ReadObjects first");
I_Error("Attempt to read object reference without calling ReadObjects first");
}
auto val = arc.r->FindKey(key); auto val = arc.r->FindKey(key);
if (val != nullptr && val->IsInt()) if (val != nullptr && val->IsInt())
{ {
if (val->GetInt() == -1) int index = val->GetInt();
if (index == -1)
{ {
value = WP_NOCHANGE; value = WP_NOCHANGE;
} }
else else
{ {
assert(index >= 0 && index < (int)arc.r->mDObjects.Size());
if (index >= 0 && index < (int)arc.r->mDObjects.Size())
{
value = arc.r->mDObjects[index];
}
else
{
I_Error("Invalid object reference for '%s'", key);
}
} }
} }
else if (!retcode) else if (!retcode)

View file

@ -61,7 +61,6 @@ public:
int ArraySize(); int ArraySize();
void WriteKey(const char *key); void WriteKey(const char *key);
void WriteObjects(); void WriteObjects();
void ReadObjects();
public: public:
@ -73,6 +72,7 @@ public:
bool OpenReader(const char *buffer, size_t length); bool OpenReader(const char *buffer, size_t length);
bool OpenReader(FCompressedBuffer *input); bool OpenReader(FCompressedBuffer *input);
void Close(); void Close();
void ReadObjects();
bool BeginObject(const char *name, bool randomaccess = false); bool BeginObject(const char *name, bool randomaccess = false);
void EndObject(bool endwarning = false); void EndObject(bool endwarning = false);
bool BeginArray(const char *name); bool BeginArray(const char *name);