ref: 5ac7ac324198351603d07f8895a53ec906653ff7
parent: 056415e2d68e261583a6fcfe1df24d5724ef91a8
author: Simon Howard <fraggle@gmail.com>
date: Fri Sep 14 18:20:08 EDT 2007
Add P_SubstNullMobj, substitute NULL mobjs for a dummy mobj where mo->target is not checked for NULL. Subversion-branch: /trunk/chocolate-doom Subversion-revision: 971
--- a/src/p_enemy.c
+++ b/src/p_enemy.c
@@ -1258,14 +1258,17 @@
void A_Fire (mobj_t* actor)
{
mobj_t* dest;
+ mobj_t* target;
unsigned an;
dest = actor->tracer;
if (!dest)
return;
+
+ target = P_SubstNullMobj(actor->target);
// don't move it if the vile lost sight
- if (!P_CheckSight (actor->target, dest) )
+ if (!P_CheckSight (target, dest) )
return;
an = dest->angle >> ANGLETOFINESHIFT;
@@ -1359,14 +1362,17 @@
void A_FatAttack1 (mobj_t* actor)
{
mobj_t* mo;
+ mobj_t* target;
int an;
-
+
A_FaceTarget (actor);
+
// Change direction to ...
actor->angle += FATSPREAD;
- P_SpawnMissile (actor, actor->target, MT_FATSHOT);
+ target = P_SubstNullMobj(actor->target);
+ P_SpawnMissile (actor, target, MT_FATSHOT);
- mo = P_SpawnMissile (actor, actor->target, MT_FATSHOT);
+ mo = P_SpawnMissile (actor, target, MT_FATSHOT);
mo->angle += FATSPREAD;
an = mo->angle >> ANGLETOFINESHIFT;
mo->momx = FixedMul (mo->info->speed, finecosine[an]);
@@ -1376,14 +1382,16 @@
void A_FatAttack2 (mobj_t* actor)
{
mobj_t* mo;
+ mobj_t* target;
int an;
A_FaceTarget (actor);
// Now here choose opposite deviation.
actor->angle -= FATSPREAD;
- P_SpawnMissile (actor, actor->target, MT_FATSHOT);
+ target = P_SubstNullMobj(actor->target);
+ P_SpawnMissile (actor, target, MT_FATSHOT);
- mo = P_SpawnMissile (actor, actor->target, MT_FATSHOT);
+ mo = P_SpawnMissile (actor, target, MT_FATSHOT);
mo->angle -= FATSPREAD*2;
an = mo->angle >> ANGLETOFINESHIFT;
mo->momx = FixedMul (mo->info->speed, finecosine[an]);
@@ -1393,17 +1401,20 @@
void A_FatAttack3 (mobj_t* actor)
{
mobj_t* mo;
+ mobj_t* target;
int an;
A_FaceTarget (actor);
+
+ target = P_SubstNullMobj(actor->target);
- mo = P_SpawnMissile (actor, actor->target, MT_FATSHOT);
+ mo = P_SpawnMissile (actor, target, MT_FATSHOT);
mo->angle -= FATSPREAD/2;
an = mo->angle >> ANGLETOFINESHIFT;
mo->momx = FixedMul (mo->info->speed, finecosine[an]);
mo->momy = FixedMul (mo->info->speed, finesine[an]);
- mo = P_SpawnMissile (actor, actor->target, MT_FATSHOT);
+ mo = P_SpawnMissile (actor, target, MT_FATSHOT);
mo->angle += FATSPREAD/2;
an = mo->angle >> ANGLETOFINESHIFT;
mo->momx = FixedMul (mo->info->speed, finecosine[an]);
@@ -1598,7 +1609,11 @@
//
void A_Explode (mobj_t* thingy)
{
- P_RadiusAttack ( thingy, thingy->target, 128 );
+ mobj_t *target;
+
+ target = P_SubstNullMobj(thingy->target);
+
+ P_RadiusAttack(thingy, target, 128);
}
@@ -1823,7 +1838,7 @@
mobj_t* braintargets[32];
int numbraintargets;
-int braintargeton;
+int braintargeton = 0;
void A_BrainAwake (mobj_t* mo)
{
@@ -1959,7 +1974,7 @@
if (--mo->reactiontime)
return; // still flying
- targ = mo->target;
+ targ = P_SubstNullMobj(mo->target);
// First spawn teleport fog.
fog = P_SpawnMobj (targ->x, targ->y, targ->z, MT_SPAWNFIRE);
--- a/src/p_local.h
+++ b/src/p_local.h
@@ -119,6 +119,7 @@
mobjtype_t type );
void P_RemoveMobj (mobj_t* th);
+mobj_t* P_SubstNullMobj (mobj_t* th);
boolean P_SetMobjState (mobj_t* mobj, statenum_t state);
void P_MobjThinker (mobj_t* mobj);
--- a/src/p_mobj.c
+++ b/src/p_mobj.c
@@ -922,7 +922,29 @@
P_ExplodeMissile (th);
}
+// Certain functions assume that a mobj_t pointer is non-NULL,
+// causing a crash in some situations where it is NULL. Vanilla
+// Doom did not crash because of the lack of proper memory
+// protection. This function substitutes NULL pointers for
+// pointers to a dummy mobj, to avoid a crash.
+mobj_t *P_SubstNullMobj(mobj_t *mobj)
+{
+ if (mobj == NULL)
+ {
+ static mobj_t dummy_mobj;
+
+ dummy_mobj.x = 0;
+ dummy_mobj.y = 0;
+ dummy_mobj.z = 0;
+ dummy_mobj.flags = 0;
+
+ mobj = &dummy_mobj;
+ }
+
+ return mobj;
+}
+
//
// P_SpawnMissile
//
@@ -935,32 +957,7 @@
mobj_t* th;
angle_t an;
int dist;
- fixed_t dest_x, dest_y, dest_z, dest_flags;
- // fraggle: This prevents against *crashes* when dest == NULL.
- // For example, when loading a game saved when a mancubus was
- // in the middle of firing, mancubus->target == NULL. SpawnMissile
- // then gets called with dest == NULL.
- //
- // However, this is not the *correct* behavior. At the moment,
- // the missile is aimed at 0,0,0. In reality, monsters seem to aim
- // somewhere else.
-
- if (dest)
- {
- dest_x = dest->x;
- dest_y = dest->y;
- dest_z = dest->z;
- dest_flags = dest->flags;
- }
- else
- {
- dest_x = 0;
- dest_y = 0;
- dest_z = 0;
- dest_flags = 0;
- }
-
th = P_SpawnMobj (source->x,
source->y,
source->z + 4*8*FRACUNIT, type);
@@ -969,10 +966,10 @@
S_StartSound (th, th->info->seesound);
th->target = source; // where it came from
- an = R_PointToAngle2 (source->x, source->y, dest_x, dest_y);
+ an = R_PointToAngle2 (source->x, source->y, dest->x, dest->y);
// fuzzy player
- if (dest_flags & MF_SHADOW)
+ if (dest->flags & MF_SHADOW)
an += (P_Random()-P_Random())<<20;
th->angle = an;
@@ -980,13 +977,13 @@
th->momx = FixedMul (th->info->speed, finecosine[an]);
th->momy = FixedMul (th->info->speed, finesine[an]);
- dist = P_AproxDistance (dest_x - source->x, dest_y - source->y);
+ dist = P_AproxDistance (dest->x - source->x, dest->y - source->y);
dist = dist / th->info->speed;
if (dist < 1)
dist = 1;
- th->momz = (dest_z - source->z) / dist;
+ th->momz = (dest->z - source->z) / dist;
P_CheckMissileSpawn (th);
return th;