mirror of
https://github.com/ZDoom/gzdoom.git
synced 2025-01-18 15:42:34 +00:00
- made PSprites submission GC aware.
This commit is contained in:
parent
8c205ebac3
commit
d767d10322
3 changed files with 47 additions and 16 deletions
|
@ -436,6 +436,7 @@ public:
|
|||
|
||||
AWeapon *ReadyWeapon;
|
||||
AWeapon *PendingWeapon; // WP_NOCHANGE if not changing
|
||||
TObjPtr<DPSprite> psprites; // view sprites (gun, etc)
|
||||
|
||||
int cheats; // bit flags
|
||||
int timefreezer; // Player has an active time freezer
|
||||
|
@ -528,7 +529,6 @@ public:
|
|||
int GetSpawnClass();
|
||||
|
||||
// PSprite layers
|
||||
DPSprite *psprites; // view sprites (gun, etc)
|
||||
void TickPSprites();
|
||||
void DestroyPSprites();
|
||||
DPSprite *GetPSprite(psprnum_t layer); // Used ONLY for compatibility with the old hardcoded layers.
|
||||
|
|
|
@ -113,12 +113,25 @@ END_POINTERS
|
|||
DPSprite::DPSprite(player_t *owner, AInventory *caller, int id)
|
||||
: processPending(true), firstTic(true), Owner(owner), Caller(caller), ID(id)
|
||||
{
|
||||
DPSprite **prev = &Owner->psprites;
|
||||
while (*prev && (*prev)->ID < ID)
|
||||
prev = &(*prev)->Next;
|
||||
|
||||
Next = *prev;
|
||||
*prev = this;
|
||||
DPSprite *prev = nullptr;
|
||||
DPSprite *next = Owner->psprites;
|
||||
while (next != nullptr && next->ID < ID)
|
||||
{
|
||||
prev = next;
|
||||
next = next->Next;
|
||||
}
|
||||
Next = next;
|
||||
GC::WriteBarrier(this, next);
|
||||
if (prev == NULL)
|
||||
{
|
||||
Owner->psprites = this;
|
||||
GC::WriteBarrier(this);
|
||||
}
|
||||
else
|
||||
{
|
||||
prev->Next = this;
|
||||
GC::WriteBarrier(prev, this);
|
||||
}
|
||||
|
||||
if (Next && Next->ID == ID && ID != 0)
|
||||
Next->Destroy(); // Replace it.
|
||||
|
@ -1349,10 +1362,13 @@ void DPSprite::Serialize(FArchive &arc)
|
|||
void player_t::DestroyPSprites()
|
||||
{
|
||||
DPSprite *pspr = psprites;
|
||||
psprites = nullptr;
|
||||
while (pspr)
|
||||
{
|
||||
DPSprite *next = pspr->Next;
|
||||
pspr->Next = nullptr;
|
||||
pspr->Destroy();
|
||||
pspr = pspr->Next;
|
||||
pspr = next;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1364,12 +1380,27 @@ void player_t::DestroyPSprites()
|
|||
|
||||
void DPSprite::Destroy()
|
||||
{
|
||||
DPSprite **prev = &Owner->psprites;
|
||||
while (*prev != this)
|
||||
prev = &(*prev)->Next;
|
||||
|
||||
*prev = Next;
|
||||
// Do not crash if this gets called on partially initialized objects.
|
||||
if (Owner != nullptr && Owner->psprites != nullptr)
|
||||
{
|
||||
if (Owner->psprites != this)
|
||||
{
|
||||
DPSprite *prev = Owner->psprites;
|
||||
while (prev != nullptr && prev->Next != this)
|
||||
prev = prev->Next;
|
||||
|
||||
if (prev != nullptr && prev->Next == this)
|
||||
{
|
||||
prev->Next = Next;
|
||||
GC::WriteBarrier(prev, Next);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
Owner->psprites = Next;
|
||||
GC::WriteBarrier(Next);
|
||||
}
|
||||
}
|
||||
Super::Destroy();
|
||||
}
|
||||
|
||||
|
|
|
@ -66,8 +66,8 @@ public:
|
|||
int GetSprite() const { return Sprite; }
|
||||
int GetFrame() const { return Frame; }
|
||||
FState* GetState() const { return State; }
|
||||
DPSprite* GetNext() const { return Next; }
|
||||
TObjPtr<AInventory> GetCaller() const { return Caller; }
|
||||
DPSprite* GetNext() { return Next; }
|
||||
AInventory* GetCaller() { return Caller; }
|
||||
|
||||
double x, y;
|
||||
double oldx, oldy;
|
||||
|
@ -82,7 +82,7 @@ private:
|
|||
void Destroy();
|
||||
|
||||
TObjPtr<AInventory> Caller;
|
||||
DPSprite *Next;
|
||||
TObjPtr<DPSprite> Next;
|
||||
player_t *Owner;
|
||||
FState *State;
|
||||
int Sprite;
|
||||
|
|
Loading…
Reference in a new issue