- plugged more memory leaks.

I can now start the first Duke Nukem level, exit with Alt-F4 and no leaked memory blocks get reported.
This commit is contained in:
Christoph Oelckers 2019-12-24 18:53:29 +01:00
parent 1c46c6da9d
commit 2820dc85a8
21 changed files with 139 additions and 138 deletions

View file

@ -108,6 +108,7 @@ struct GameInterface
virtual ~GameInterface() {} virtual ~GameInterface() {}
virtual void faketimerhandler() {} // This is a remnant of older versions, but Blood backend has not updated yet. virtual void faketimerhandler() {} // This is a remnant of older versions, but Blood backend has not updated yet.
virtual int app_main() = 0; virtual int app_main() = 0;
virtual void FreeGameData() {}
virtual bool validate_hud(int) = 0; virtual bool validate_hud(int) = 0;
virtual void set_hud_layout(int size) = 0; virtual void set_hud_layout(int size) = 0;
virtual void set_hud_scale(int size) = 0; virtual void set_hud_scale(int size) = 0;

View file

@ -459,7 +459,7 @@ EXTERN tspriteptr_t tspriteptr[MAXSPRITESONSCREEN + 1];
EXTERN int32_t wx1, wy1, wx2, wy2; EXTERN int32_t wx1, wy1, wx2, wy2;
EXTERN int32_t xdim, ydim, numpages, upscalefactor; EXTERN int32_t xdim, ydim, numpages, upscalefactor;
EXTERN int32_t yxaspect, viewingrange; EXTERN int32_t yxaspect, viewingrange;
EXTERN intptr_t *ylookup; EXTERN TArray<intptr_t> ylookup;
EXTERN int32_t rotatesprite_y_offset; EXTERN int32_t rotatesprite_y_offset;
EXTERN int32_t rotatesprite_yxaspect; EXTERN int32_t rotatesprite_yxaspect;
@ -527,7 +527,7 @@ EXTERN int32_t g_visibility, parallaxvisibility;
EXTERN uint8_t numalphatabs; EXTERN uint8_t numalphatabs;
EXTERN vec2_t windowxy1, windowxy2; EXTERN vec2_t windowxy1, windowxy2;
EXTERN int16_t *startumost, *startdmost; EXTERN TArray<int16_t> startumost, startdmost;
// The maximum tile offset ever used in any tiled parallaxed multi-sky. // The maximum tile offset ever used in any tiled parallaxed multi-sky.
#define PSKYOFF_MAX 8 #define PSKYOFF_MAX 8

View file

@ -27,9 +27,7 @@ void calc_ylookup(int32_t bpl, int32_t lastyidx)
if (lastyidx > ylookupsiz) if (lastyidx > ylookupsiz)
{ {
Xaligned_free(ylookup); ylookup.Resize(lastyidx);
ylookup = (intptr_t *)Xaligned_alloc(16, lastyidx * sizeof(intptr_t));
ylookupsiz = lastyidx; ylookupsiz = lastyidx;
} }

View file

@ -130,7 +130,7 @@ static struct {
} distrecipcache[DISTRECIPCACHESIZE]; } distrecipcache[DISTRECIPCACHESIZE];
static int32_t distrecipagecnt = 0; static int32_t distrecipagecnt = 0;
static int32_t *lookups = NULL; static TArray<int32_t> lookups;
static int32_t beforedrawrooms = 1; static int32_t beforedrawrooms = 1;
static int32_t oxdimen = -1, oviewingrange = -1, oxyaspect = -1; static int32_t oxdimen = -1, oviewingrange = -1, oxyaspect = -1;
@ -145,7 +145,7 @@ int32_t globalflags;
//Textured Map variables //Textured Map variables
static char globalpolytype; static char globalpolytype;
static int16_t **dotp1, **dotp2; static TArray<int16_t *>dotp1, dotp2;
static int8_t tempbuf[MAXWALLS]; static int8_t tempbuf[MAXWALLS];
@ -165,7 +165,8 @@ static intptr_t slopalookup[SLOPALOOKUPSIZ]; // was 2048
static int32_t no_radarang2 = 0; static int32_t no_radarang2 = 0;
static int16_t radarang[1280]; static int16_t radarang[1280];
static int32_t qradarang[10240], *radarang2; static int32_t qradarang[10240];
static TArray<int32_t> radarang2;
const char ATTRIBUTE((used)) pow2char_[8] = {1,2,4,8,16,32,64,128}; const char ATTRIBUTE((used)) pow2char_[8] = {1,2,4,8,16,32,64,128};
uint16_t ATTRIBUTE((used)) sqrtable[4096], ATTRIBUTE((used)) shlookup[4096+256], ATTRIBUTE((used)) sqrtable_old[2048]; uint16_t ATTRIBUTE((used)) sqrtable[4096], ATTRIBUTE((used)) shlookup[4096+256], ATTRIBUTE((used)) sqrtable_old[2048];
@ -1331,14 +1332,14 @@ int16_t bunchp2[MAXWALLSB], thesector[MAXWALLSB];
int16_t bunchfirst[MAXWALLSB], bunchlast[MAXWALLSB]; int16_t bunchfirst[MAXWALLSB], bunchlast[MAXWALLSB];
static int32_t nodesperline, ysavecnt; static int32_t nodesperline, ysavecnt;
static int16_t *smost, *umost, *dmost, *bakumost, *bakdmost; static TArray<int16_t> smost, umost, dmost, bakumost, bakdmost;
static int16_t *uplc, *dplc, *uwall, *dwall; static TArray<int16_t> uplc, dplc, uwall, dwall;
static int32_t *swplc, *lplc, *swall, *lwall; static TArray<int32_t> swplc, lplc, swall, lwall;
#ifdef HIGH_PRECISION_SPRITE #ifdef HIGH_PRECISION_SPRITE
static float *swallf; static TArray<float> swallf;
#endif #endif
uint8_t* mirrorBuffer; TArray<uint8_t> mirrorBuffer;
static int32_t smostcnt; static int32_t smostcnt;
static int32_t smoststart[MAXWALLSB]; static int32_t smoststart[MAXWALLSB];
@ -1396,7 +1397,7 @@ static int32_t globaly1, globalx2;
int16_t sectorborder[256]; int16_t sectorborder[256];
int32_t ydim16, qsetmode = 0; int32_t ydim16, qsetmode = 0;
int16_t pointhighlight=-1, linehighlight=-1, highlightcnt=0; int16_t pointhighlight=-1, linehighlight=-1, highlightcnt=0;
static int32_t *lastx; static TArray<int32_t> lastx;
int32_t halfxdim16, midydim16; int32_t halfxdim16, midydim16;
@ -1926,7 +1927,7 @@ static void maskwallscan(int32_t x1, int32_t x2, int32_t saturatevplc)
palookupoffse[0] = fpalookup + getpalookupsh(mulscale16(swall[x],globvis)); palookupoffse[0] = fpalookup + getpalookupsh(mulscale16(swall[x],globvis));
calc_bufplc(&bufplce[0], lwall[x], tsiz); calc_bufplc(&bufplce[0], lwall[x], tsiz);
calc_vplcinc(&vplce[0], &vince[0], swall, x, y1ve[0]); calc_vplcinc(&vplce[0], &vince[0], swall.Data(), x, y1ve[0]);
mvlineasm1(vince[0],palookupoffse[0],y2ve[0]-y1ve[0]-1,vplce[0],bufplce[0],p+ylookup[y1ve[0]]); mvlineasm1(vince[0],palookupoffse[0],y2ve[0]-y1ve[0]-1,vplce[0],bufplce[0],p+ylookup[y1ve[0]]);
} }
@ -1941,7 +1942,7 @@ static void maskwallscan(int32_t x1, int32_t x2, int32_t saturatevplc)
if (y2ve[z] < y1ve[z]) { bad += pow2char[z]; continue; } if (y2ve[z] < y1ve[z]) { bad += pow2char[z]; continue; }
calc_bufplc(&bufplce[z], lwall[dax], tsiz); calc_bufplc(&bufplce[z], lwall[dax], tsiz);
calc_vplcinc(&vplce[z], &vince[z], swall, dax, y1ve[z]); calc_vplcinc(&vplce[z], &vince[z], swall.Data(), dax, y1ve[z]);
} }
if (bad == 15) continue; if (bad == 15) continue;
@ -1999,7 +2000,7 @@ do_mvlineasm1:
palookupoffse[0] = fpalookup + getpalookupsh(mulscale16(swall[x],globvis)); palookupoffse[0] = fpalookup + getpalookupsh(mulscale16(swall[x],globvis));
calc_bufplc(&bufplce[0], lwall[x], tsiz); calc_bufplc(&bufplce[0], lwall[x], tsiz);
calc_vplcinc(&vplce[0], &vince[0], swall, x, y1ve[0]); calc_vplcinc(&vplce[0], &vince[0], swall.Data(), x, y1ve[0]);
#ifdef NONPOW2_YSIZE_ASM #ifdef NONPOW2_YSIZE_ASM
if (globalshiftval==0) if (globalshiftval==0)
@ -3042,7 +3043,7 @@ static void transmaskvline(int32_t x)
calc_bufplc(&bufplc, lwall[x], ntsiz); calc_bufplc(&bufplc, lwall[x], ntsiz);
uint32_t vplc; uint32_t vplc;
int32_t vinc; int32_t vinc;
calc_vplcinc(&vplc, &vinc, swall, x, y1v); calc_vplcinc(&vplc, &vinc, swall.Data(), x, y1v);
intptr_t p = ylookup[y1v]+x+frameoffset; intptr_t p = ylookup[y1v]+x+frameoffset;
@ -3083,8 +3084,8 @@ static void transmaskvline2(int32_t x)
calc_bufplc(&bufplce[0], lwall[x], ntsiz); calc_bufplc(&bufplce[0], lwall[x], ntsiz);
calc_bufplc(&bufplce[1], lwall[x2], ntsiz); calc_bufplc(&bufplce[1], lwall[x2], ntsiz);
calc_vplcinc(&vplce[0], &vince[0], swall, x, y1ve[0]); calc_vplcinc(&vplce[0], &vince[0], swall.Data(), x, y1ve[0]);
calc_vplcinc(&vplce[1], &vince[1], swall, x2, y1ve[1]); calc_vplcinc(&vplce[1], &vince[1], swall.Data(), x2, y1ve[1]);
int32_t const y1 = max(y1ve[0],y1ve[1]); int32_t const y1 = max(y1ve[0],y1ve[1]);
int32_t const y2 = min(y2ve[0],y2ve[1]); int32_t const y2 = min(y2ve[0],y2ve[1]);
@ -3931,8 +3932,8 @@ static void parascan(char dastat, int32_t bunch)
globalshade = (int32_t)sec->ceilingshade; globalshade = (int32_t)sec->ceilingshade;
globalxpanning = (int32_t)sec->ceilingxpanning; globalxpanning = (int32_t)sec->ceilingxpanning;
globalypanning = (int32_t)sec->ceilingypanning; globalypanning = (int32_t)sec->ceilingypanning;
topptr = umost; topptr = umost.Data();
botptr = uplc; botptr = uplc.Data();
} }
else else
{ {
@ -3941,8 +3942,8 @@ static void parascan(char dastat, int32_t bunch)
globalshade = (int32_t)sec->floorshade; globalshade = (int32_t)sec->floorshade;
globalxpanning = (int32_t)sec->floorxpanning; globalxpanning = (int32_t)sec->floorxpanning;
globalypanning = (int32_t)sec->floorypanning; globalypanning = (int32_t)sec->floorypanning;
topptr = dplc; topptr = dplc.Data();
botptr = dmost; botptr = dmost.Data();
} }
if ((unsigned)globalpicnum >= MAXTILES) globalpicnum = 0; if ((unsigned)globalpicnum >= MAXTILES) globalpicnum = 0;
@ -4040,7 +4041,7 @@ static void parascan(char dastat, int32_t bunch)
globalpicnum = l + dapskyoff[lplc[x]>>m]; globalpicnum = l + dapskyoff[lplc[x]>>m];
if (((lplc[x]^lplc[xb1[z]-1])>>m) == 0) if (((lplc[x]^lplc[xb1[z]-1])>>m) == 0)
wallscan(x,xb1[z]-1,topptr,botptr,swplc,lplc); wallscan(x,xb1[z]-1,topptr,botptr,swplc.Data(),lplc.Data());
else else
{ {
j = x; j = x;
@ -4049,14 +4050,14 @@ static void parascan(char dastat, int32_t bunch)
n = l + dapskyoff[lplc[x]>>m]; n = l + dapskyoff[lplc[x]>>m];
if (n != globalpicnum) if (n != globalpicnum)
{ {
wallscan(j,x-1,topptr,botptr,swplc,lplc); wallscan(j,x-1,topptr,botptr,swplc.Data(),lplc.Data());
j = x; j = x;
globalpicnum = n; globalpicnum = n;
} }
x++; x++;
} }
if (j < x) if (j < x)
wallscan(j,x-1,topptr,botptr,swplc,lplc); wallscan(j,x-1,topptr,botptr,swplc.Data(),lplc.Data());
} }
globalpicnum = l; globalpicnum = l;
@ -4070,7 +4071,7 @@ static void parascan(char dastat, int32_t bunch)
globalpicnum = l + dapskyoff[lplc[x]>>m]; globalpicnum = l + dapskyoff[lplc[x]>>m];
if (((lplc[x]^lplc[xb2[bunchlast[bunch]]])>>m) == 0) if (((lplc[x]^lplc[xb2[bunchlast[bunch]]])>>m) == 0)
wallscan(x,xb2[bunchlast[bunch]],topptr,botptr,swplc,lplc); wallscan(x,xb2[bunchlast[bunch]],topptr,botptr,swplc.Data(),lplc.Data());
else else
{ {
j = x; j = x;
@ -4079,14 +4080,14 @@ static void parascan(char dastat, int32_t bunch)
n = l + dapskyoff[lplc[x]>>m]; n = l + dapskyoff[lplc[x]>>m];
if (n != globalpicnum) if (n != globalpicnum)
{ {
wallscan(j,x-1,topptr,botptr,swplc,lplc); wallscan(j,x-1,topptr,botptr,swplc.Data(),lplc.Data());
j = x; j = x;
globalpicnum = n; globalpicnum = n;
} }
x++; x++;
} }
if (j <= x) if (j <= x)
wallscan(j,x-1,topptr,botptr,swplc,lplc); wallscan(j,x-1,topptr,botptr,swplc.Data(),lplc.Data());
} }
globalpicnum = l; globalpicnum = l;
} }
@ -4221,8 +4222,8 @@ static void classicDrawBunches(int32_t bunch)
for (; z>=0; z=bunchp2[z]) //uplc/dplc calculation for (; z>=0; z=bunchp2[z]) //uplc/dplc calculation
{ {
andwstat1 &= wallmost(uplc,z,sectnum,(uint8_t)0); andwstat1 &= wallmost(uplc.Data(),z,sectnum,(uint8_t)0);
andwstat2 &= wallmost(dplc,z,sectnum,(uint8_t)1); andwstat2 &= wallmost(dplc.Data(),z,sectnum,(uint8_t)1);
} }
#ifdef YAX_ENABLE #ifdef YAX_ENABLE
@ -4414,7 +4415,7 @@ static void classicDrawBunches(int32_t bunch)
} }
else else
{ {
wallmost(dwall,z,nextsectnum,(uint8_t)0); wallmost(dwall.Data(),z,nextsectnum,(uint8_t)0);
if ((cz[2] > fz[0]) || (cz[3] > fz[1])) if ((cz[2] > fz[0]) || (cz[3] > fz[1]))
for (i=x1; i<=x2; i++) if (dwall[i] > dplc[i]) dwall[i] = dplc[i]; for (i=x1; i<=x2; i++) if (dwall[i] > dplc[i]) dwall[i] = dplc[i];
@ -4435,7 +4436,7 @@ static void classicDrawBunches(int32_t bunch)
gotswall = 1; gotswall = 1;
prepwall(z,wal); prepwall(z,wal);
wallscan(x1,x2,uplc,dwall,swall,lwall); wallscan(x1,x2,uplc.Data(),dwall.Data(),swall.Data(),lwall.Data());
if ((cz[2] >= cz[0]) && (cz[3] >= cz[1])) if ((cz[2] >= cz[0]) && (cz[3] >= cz[1]))
{ {
@ -4499,7 +4500,7 @@ static void classicDrawBunches(int32_t bunch)
} }
else else
{ {
wallmost(uwall,z,nextsectnum,(uint8_t)1); wallmost(uwall.Data(),z,nextsectnum,(uint8_t)1);
if ((fz[2] < cz[0]) || (fz[3] < cz[1])) if ((fz[2] < cz[0]) || (fz[3] < cz[1]))
for (i=x1; i<=x2; i++) if (uwall[i] < uplc[i]) uwall[i] = uplc[i]; for (i=x1; i<=x2; i++) if (uwall[i] < uplc[i]) uwall[i] = uplc[i];
@ -4522,7 +4523,7 @@ static void classicDrawBunches(int32_t bunch)
setup_globals_wall2(wal, sec->visibility, nextsec->floorz, sec->ceilingz); setup_globals_wall2(wal, sec->visibility, nextsec->floorz, sec->ceilingz);
if (gotswall == 0) { gotswall = 1; prepwall(z,wal); } if (gotswall == 0) { gotswall = 1; prepwall(z,wal); }
wallscan(x1,x2,uwall,dplc,swall,lwall); wallscan(x1,x2,uwall.Data(),dplc.Data(),swall.Data(),lwall.Data());
if ((fz[2] <= fz[0]) && (fz[3] <= fz[1])) if ((fz[2] <= fz[0]) && (fz[3] <= fz[1]))
{ {
@ -4604,7 +4605,7 @@ static void classicDrawBunches(int32_t bunch)
(nextsectnum >= 0) ? sec->ceilingz : sec->floorz); (nextsectnum >= 0) ? sec->ceilingz : sec->floorz);
if (gotswall == 0) { gotswall = 1; prepwall(z,wal); } if (gotswall == 0) { gotswall = 1; prepwall(z,wal); }
wallscan(x1,x2,uplc,dplc,swall,lwall); wallscan(x1,x2,uplc.Data(),dplc.Data(),swall.Data(),lwall.Data());
#ifdef YAX_ENABLE #ifdef YAX_ENABLE
// TODO: slopes? // TODO: slopes?
@ -5447,8 +5448,8 @@ draw_as_face_sprite:
xb2[MAXWALLSB-1] = sx2; xb2[MAXWALLSB-1] = sx2;
yb1[MAXWALLSB-1] = sy1; yb1[MAXWALLSB-1] = sy1;
yb2[MAXWALLSB-1] = sy2; yb2[MAXWALLSB-1] = sy2;
owallmost(uwall, MAXWALLSB-1, z1-globalposz); owallmost(uwall.Data(), MAXWALLSB-1, z1-globalposz);
owallmost(dwall, MAXWALLSB-1, z2-globalposz); owallmost(dwall.Data(), MAXWALLSB-1, z2-globalposz);
int32_t hplc = divscale19(xdimenscale,sy1); int32_t hplc = divscale19(xdimenscale,sy1);
const int32_t hplc2 = divscale19(xdimenscale,sy2); const int32_t hplc2 = divscale19(xdimenscale,sy2);
@ -6156,7 +6157,7 @@ draw_as_face_sprite:
const int32_t floorz = (sec->floorstat&3) == 0 ? sec->floorz : INT32_MAX; const int32_t floorz = (sec->floorstat&3) == 0 ? sec->floorz : INT32_MAX;
classicDrawVoxel(tspr->x,tspr->y,tspr->z,i,daxrepeat,(int32_t)tspr->yrepeat,vtilenum, classicDrawVoxel(tspr->x,tspr->y,tspr->z,i,daxrepeat,(int32_t)tspr->yrepeat,vtilenum,
tspr->shade,tspr->pal,lwall,swall,tspr->cstat,(tspr->cstat&48)!=48,floorz,ceilingz); tspr->shade,tspr->pal,lwall.Data(),swall.Data(),tspr->cstat,(tspr->cstat&48)!=48,floorz,ceilingz);
} }
if (automapping == 1 && (unsigned)spritenum < MAXSPRITES) if (automapping == 1 && (unsigned)spritenum < MAXSPRITES)
@ -6198,13 +6199,13 @@ static void renderDrawMaskedWall(int16_t damaskwallcnt)
int32_t z1 = max(nsec->ceilingz,sec->ceilingz); int32_t z1 = max(nsec->ceilingz,sec->ceilingz);
int32_t z2 = min(nsec->floorz,sec->floorz); int32_t z2 = min(nsec->floorz,sec->floorz);
wallmost(uwall,z,sectnum,(uint8_t)0); wallmost(uwall.Data(),z,sectnum,(uint8_t)0);
wallmost(uplc,z,(int32_t)wal->nextsector,(uint8_t)0); wallmost(uplc.Data(),z,(int32_t)wal->nextsector,(uint8_t)0);
for (bssize_t x=xb1[z]; x<=xb2[z]; x++) for (bssize_t x=xb1[z]; x<=xb2[z]; x++)
if (uplc[x] > uwall[x]) if (uplc[x] > uwall[x])
uwall[x] = uplc[x]; uwall[x] = uplc[x];
wallmost(dwall,z,sectnum,(uint8_t)1); wallmost(dwall.Data(),z,sectnum,(uint8_t)1);
wallmost(dplc,z,(int32_t)wal->nextsector,(uint8_t)1); wallmost(dplc.Data(),z,(int32_t)wal->nextsector,(uint8_t)1);
for (bssize_t x=xb1[z]; x<=xb2[z]; x++) for (bssize_t x=xb1[z]; x<=xb2[z]; x++)
if (dplc[x] < dwall[x]) if (dplc[x] < dwall[x])
dwall[x] = dplc[x]; dwall[x] = dplc[x];
@ -8025,7 +8026,6 @@ void engineUnInit(void)
TileFiles.CloseAll(); TileFiles.CloseAll();
DO_FREE_AND_NULL(lookups);
for (bssize_t i=0; i<DISTRECIPCACHESIZE; i++) for (bssize_t i=0; i<DISTRECIPCACHESIZE; i++)
ALIGNED_FREE_AND_NULL(distrecipcache[i].distrecip); ALIGNED_FREE_AND_NULL(distrecipcache[i].distrecip);
Bmemset(distrecipcache, 0, sizeof(distrecipcache)); Bmemset(distrecipcache, 0, sizeof(distrecipcache));
@ -8285,8 +8285,8 @@ int32_t renderDrawRoomsQ16(int32_t daposx, int32_t daposy, int32_t daposz,
{ {
j = yax_globalbunch*xdimen; j = yax_globalbunch*xdimen;
Bmemcpy(umost, yumost+j, xdimen*sizeof(int16_t)); Bmemcpy(umost.Data(), yumost+j, xdimen*sizeof(int16_t));
Bmemcpy(dmost, ydmost+j, xdimen*sizeof(int16_t)); Bmemcpy(dmost.Data(), ydmost+j, xdimen*sizeof(int16_t));
for (i=0; i<xdimen; i++) for (i=0; i<xdimen; i++)
if (umost[i] > dmost[i]) if (umost[i] > dmost[i])
@ -9835,43 +9835,29 @@ static int32_t get_mapversion(void)
static void videoAllocateBuffers(void) static void videoAllocateBuffers(void)
{ {
int32_t i;
// Needed for the game's TILT_SETVIEWTOTILE_320. // Needed for the game's TILT_SETVIEWTOTILE_320.
const int32_t clamped_ydim = max(ydim, 320); const int32_t clamped_ydim = max(ydim, 320);
struct smost.Resize(YSAVES);
{ umost.Resize(xdim);
void **ptr; dmost.Resize(xdim);
size_t size; startumost.Resize(xdim);
} dynarray[] = { startdmost.Resize(xdim);
{ (void **)&smost, YSAVES * sizeof(int16_t) }, bakumost.Resize(xdim);
{ (void **)&umost, xdim * sizeof(int16_t) }, bakdmost.Resize(xdim);
{ (void **)&dmost, xdim * sizeof(int16_t) }, uplc.Resize(xdim);
{ (void **)&startumost, xdim * sizeof(int16_t) }, dplc.Resize(xdim);
{ (void **)&startdmost, xdim * sizeof(int16_t) }, uwall.Resize(xdim);
{ (void **)&bakumost, xdim * sizeof(int16_t) }, dwall.Resize(xdim);
{ (void **)&bakdmost, xdim * sizeof(int16_t) }, swplc.Resize(xdim);
{ (void **)&uplc, xdim * sizeof(int16_t) }, lplc.Resize(xdim);
{ (void **)&dplc, xdim * sizeof(int16_t) }, swall.Resize(xdim);
{ (void **)&uwall, xdim * sizeof(int16_t) }, lwall.Resize((xdim + 4));
{ (void **)&dwall, xdim * sizeof(int16_t) }, radarang2.Resize(xdim);
{ (void **)&swplc, xdim * sizeof(int32_t) }, dotp1.Resize(clamped_ydim);
{ (void **)&lplc, xdim * sizeof(int32_t) }, dotp2.Resize(clamped_ydim);
{ (void **)&swall, xdim * sizeof(int32_t) }, lastx.Resize(clamped_ydim);
{ (void **)&lwall, (xdim + 4) * sizeof(int32_t) }, mirrorBuffer.Resize(xdim * ydim);
{ (void **)&radarang2, xdim * sizeof(int32_t) },
{ (void **)&dotp1, clamped_ydim * sizeof(intptr_t) },
{ (void **)&dotp2, clamped_ydim * sizeof(intptr_t) },
{ (void **)&lastx, clamped_ydim * sizeof(int32_t) },
{ (void **)&mirrorBuffer, (size_t) (xdim * ydim)},
};
for (i = 0; i < (signed)ARRAY_SIZE(dynarray); i++)
{
Xaligned_free(*dynarray[i].ptr);
*dynarray[i].ptr = Xaligned_alloc(16, dynarray[i].size);
}
ysavecnt = YSAVES; ysavecnt = YSAVES;
nodesperline = tabledivide32_noinline(YSAVES, ydim); nodesperline = tabledivide32_noinline(YSAVES, ydim);
@ -9941,16 +9927,14 @@ int32_t videoSetGameMode(char davidoption, int32_t daupscaledxdim, int32_t daups
videoAllocateBuffers(); videoAllocateBuffers();
#ifdef HIGH_PRECISION_SPRITE #ifdef HIGH_PRECISION_SPRITE
swallf = (float *) Xrealloc(swallf, xdim * sizeof(float)); swallf.Resize(xdim);
#endif #endif
Xfree(lookups);
j = ydim*4; //Leave room for horizlookup&horizlookup2 j = ydim*4; //Leave room for horizlookup&horizlookup2
lookups = (int32_t *)Xmalloc(2*j*sizeof(lookups[0])); lookups.Resize(2 * j);
horizlookup = lookups; horizlookup = lookups.Data();
horizlookup2 = lookups + j; horizlookup2 = lookups.Data() + j;
horizycent = ((ydim*4)>>1); horizycent = ((ydim*4)>>1);
//Force drawrooms to call dosetaspect & recalculate stuff //Force drawrooms to call dosetaspect & recalculate stuff
@ -11700,7 +11684,7 @@ void renderCompleteMirror(void)
int const height = mirrorsy2-mirrorsy1; int const height = mirrorsy2-mirrorsy1;
// Address of the mirror wall's top left corner in the source scene: // Address of the mirror wall's top left corner in the source scene:
intptr_t s = (intptr_t) mirrorBuffer + ylookup[windowxy1.y+mirrorsy1] + windowxy1.x+mirrorsx1; intptr_t s = (intptr_t) mirrorBuffer.Data() + ylookup[windowxy1.y+mirrorsy1] + windowxy1.x+mirrorsx1;
// Pointer to the mirror line's left corner in the destination: // Pointer to the mirror line's left corner in the destination:
intptr_t d = (intptr_t) frameplace + ylookup[windowxy1.y+mirrorsy1] + windowxy2.x-mirrorsx2; intptr_t d = (intptr_t) frameplace + ylookup[windowxy1.y+mirrorsy1] + windowxy2.x-mirrorsx2;

View file

@ -92,7 +92,7 @@ extern int16_t thesector[MAXWALLSB], thewall[MAXWALLSB];
extern int16_t bunchfirst[MAXWALLSB], bunchlast[MAXWALLSB]; extern int16_t bunchfirst[MAXWALLSB], bunchlast[MAXWALLSB];
extern int16_t maskwall[MAXWALLSB], maskwallcnt; extern int16_t maskwall[MAXWALLSB], maskwallcnt;
extern tspriteptr_t tspriteptr[MAXSPRITESONSCREEN + 1]; extern tspriteptr_t tspriteptr[MAXSPRITESONSCREEN + 1];
extern uint8_t* mirrorBuffer; extern TArray<uint8_t> mirrorBuffer;
extern int32_t xdimen, xdimenrecip, halfxdimen, xdimenscale, xdimscale, ydimen; extern int32_t xdimen, xdimenrecip, halfxdimen, xdimenscale, xdimscale, ydimen;
extern float fxdimen; extern float fxdimen;
extern intptr_t frameoffset; extern intptr_t frameoffset;

View file

@ -236,12 +236,12 @@ void videoBeginDrawing(void)
{ {
if (!backupFrameplace) if (!backupFrameplace)
backupFrameplace = frameplace; backupFrameplace = frameplace;
else if (frameplace != (intptr_t)mirrorBuffer && else if (frameplace != (intptr_t)mirrorBuffer.Data() &&
frameplace != backupFrameplace) frameplace != backupFrameplace)
return; return;
} }
frameplace = (intptr_t)mirrorBuffer; frameplace = (intptr_t)mirrorBuffer.Data();
if (offscreenrendering) if (offscreenrendering)
return; return;

View file

@ -726,6 +726,7 @@ void C_DeinitConsole ()
// Make sure all tab commands are cleared before the memory for // Make sure all tab commands are cleared before the memory for
// their names is deallocated. // their names is deallocated.
C_ClearTabCommands (); C_ClearTabCommands ();
C_ClearDynCCmds();
// Free AddToConsole()'s work buffer // Free AddToConsole()'s work buffer
if (work != NULL) if (work != NULL)

View file

@ -1055,6 +1055,16 @@ int OSD_RegisterFunction(const char* pszName, const char* pszDesc, int (*func)(o
return 0; return 0;
} }
void C_ClearDynCCmds()
{
for (auto ccmd : dynccmds)
{
delete ccmd;
}
dynccmds.Clear();
}
CCMD (quit) CCMD (quit)
{ {
throw ExitEvent(0); throw ExitEvent(0);

View file

@ -70,6 +70,7 @@ void C_DoCommand (const char *cmd, int keynum=0);
FExecList *C_ParseExecFile(const char *file, FExecList *source); FExecList *C_ParseExecFile(const char *file, FExecList *source);
bool C_ExecFile(const char *file); bool C_ExecFile(const char *file);
void C_ClearDynCCmds();
// Write out alias commands to a file for all current aliases. // Write out alias commands to a file for all current aliases.
void C_ArchiveAliases (FConfigFile *f); void C_ArchiveAliases (FConfigFile *f);

View file

@ -51,6 +51,8 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#include "v_video.h" #include "v_video.h"
#include "st_start.h" #include "st_start.h"
#include "s_music.h" #include "s_music.h"
#include "i_video.h"
#include "glbackend/glbackend.h"
#ifndef NETCODE_DISABLE #ifndef NETCODE_DISABLE
#include "enet.h" #include "enet.h"
#endif #endif
@ -60,6 +62,9 @@ MapRecord *currentLevel; // level that is currently played. (The real level, not
MapRecord* lastLevel; // Same here, for the last level. MapRecord* lastLevel; // Same here, for the last level.
MapRecord userMapRecord; // stand-in for the user map. MapRecord userMapRecord; // stand-in for the user map.
FMemArena dump; // this is for memory blocks than cannot be deallocated without some huge effort. Put them in here so that they do not register on shutdown.
void C_CON_SetAliases(); void C_CON_SetAliases();
InputState inputState; InputState inputState;
void SetClipshapes(); void SetClipshapes();
@ -305,10 +310,17 @@ int GameMain()
} }
S_StopMusic(true); S_StopMusic(true);
if (soundEngine) delete soundEngine; if (soundEngine) delete soundEngine;
soundEngine = nullptr;
I_CloseSound();
I_ShutdownInput(); I_ShutdownInput();
G_SaveConfig(); G_SaveConfig();
C_DeinitConsole(); C_DeinitConsole();
V_ClearFonts(); V_ClearFonts();
TileFiles.ClearTextureCache();
TileFiles.CloseAll(); // do this before shutting down graphics.
GLInterface.Deinit();
I_ShutdownGraphics();
gi->FreeGameData();
if (gi) delete gi; if (gi) delete gi;
#ifndef NETCODE_DISABLE #ifndef NETCODE_DISABLE
if (gHaveNetworking) enet_deinitialize(); if (gHaveNetworking) enet_deinitialize();

View file

@ -6,6 +6,7 @@
#include "gamecvars.h" #include "gamecvars.h"
#include "tarray.h" #include "tarray.h"
#include "name.h" #include "name.h"
#include "memarena.h"
EXTERN_CVAR(Int, cl_defaultconfiguration) EXTERN_CVAR(Int, cl_defaultconfiguration)
@ -13,6 +14,8 @@ extern FString currentGame;
extern FString LumpFilter; extern FString LumpFilter;
class FArgs; class FArgs;
extern FMemArena dump; // this is for memory blocks than cannot be deallocated without some huge effort. Put them in here so that they do not register on shutdown.
extern TMap<FName, int32_t> NameToTileIndex; extern TMap<FName, int32_t> NameToTileIndex;
void D_AddWildFile(TArray<FString>& wadfiles, const char* value); void D_AddWildFile(TArray<FString>& wadfiles, const char* value);

View file

@ -469,3 +469,10 @@ template<> struct THashTraits<FString>
// Compares two keys, returning zero if they are the same. // Compares two keys, returning zero if they are the same.
int Compare(const FString &left, const FString &right) { return left.Compare(right); } int Compare(const FString &left, const FString &right) { return left.Compare(right); }
}; };
struct StringNoCaseHashTraits
{
hash_t Hash(const FString& key) { return (hash_t)SuperFastHashI(key.GetChars(), key.Len()); }
// Compares two keys, returning zero if they are the same.
int Compare(const FString& left, const FString& right) { return left.CompareNoCase(right); }
};

View file

@ -208,7 +208,7 @@ void Anim_Init(void)
if (anm.numsounds) if (anm.numsounds)
{ {
anim->sounds = (animsound_t *)Xmalloc(anm.numsounds * sizeof(animsound_t)); anim->sounds = (animsound_t *)dump.Alloc(anm.numsounds * sizeof(animsound_t));
int const numsounds = anm.numsounds; int const numsounds = anm.numsounds;
for (int i = 0; i < numsounds; ++i) for (int i = 0; i < numsounds; ++i)
{ {

View file

@ -32,8 +32,9 @@ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#include "pragmas.h" #include "pragmas.h"
#include "polymost.h" #include "polymost.h"
#include "gamecvars.h" #include "gamecvars.h"
#include "gamecontrol.h"
#include "menu/menu.h" #include "menu/menu.h"
#include "memarena.h"
#define HEAD2 APPNAME #define HEAD2 APPNAME
@ -144,6 +145,7 @@ static inline int32_t G_DefaultActorHealth(int spriteNum)
struct GameInterface : ::GameInterface struct GameInterface : ::GameInterface
{ {
int app_main() override; int app_main() override;
void FreeGameData() override;
bool validate_hud(int) override; bool validate_hud(int) override;
void set_hud_layout(int size) override; void set_hud_layout(int size) override;
void set_hud_scale(int size) override; void set_hud_scale(int size) override;

View file

@ -259,11 +259,6 @@ void G_GameExit(const char *msg)
G_DisplayExtraScreens(); G_DisplayExtraScreens();
} }
if (*msg != 0) initprintf("%s\n",msg);
if (in3dmode())
G_Shutdown();
if (*msg != 0) if (*msg != 0)
{ {
if (!(msg[0] == ' ' && msg[1] == 0)) if (!(msg[0] == ' ' && msg[1] == 0))
@ -5316,6 +5311,7 @@ static void G_Cleanup(void)
#if !defined LUNATIC #if !defined LUNATIC
if (label != (char *)&sprite[0]) Xfree(label); if (label != (char *)&sprite[0]) Xfree(label);
if (labelcode != (int32_t *)&sector[0]) Xfree(labelcode); if (labelcode != (int32_t *)&sector[0]) Xfree(labelcode);
if (labeltype != (int32_t*)&wall[0]) Xfree(labeltype);
Xfree(apScript); Xfree(apScript);
Xfree(bitptr); Xfree(bitptr);
@ -5342,8 +5338,6 @@ static void G_Cleanup(void)
void G_Shutdown(void) void G_Shutdown(void)
{ {
engineUnInit();
G_Cleanup();
} }
/* /*
@ -5513,21 +5507,13 @@ void G_PostCreateGameState(void)
static void G_HandleMemErr(int32_t lineNum, const char *fileName, const char *funcName) static void G_HandleMemErr(int32_t lineNum, const char *fileName, const char *funcName)
{ {
static char msg[128]; I_FatalError("Out of memory in %s:%d (%s)\n", fileName, lineNum, funcName);
Bsnprintf(msg, sizeof(msg), "Out of memory in %s:%d (%s)\n", fileName, lineNum, funcName);
#ifdef DEBUGGINGAIDS
Bassert(0);
#endif
G_GameExit(msg);
} }
static void G_FatalEngineError(void) static void G_FatalEngineError(void)
{ {
wm_msgbox("Fatal Engine Initialization Error", I_FatalError("Fatal Engine Initialization Error",
"There was a problem initializing the engine: %s\n\nThe application will now close.", engineerrstr); "There was a problem initializing the engine: %s\n\nThe application will now close.", engineerrstr);
G_Cleanup();
ERRprintf("G_Startup: There was a problem initializing the engine: %s\n", engineerrstr);
exit(6);
} }
static void G_Startup(void) static void G_Startup(void)
@ -5734,6 +5720,7 @@ void G_MaybeAllocPlayer(int32_t pnum)
} }
// TODO: reorder (net)actor_t to eliminate slop and update assertion // TODO: reorder (net)actor_t to eliminate slop and update assertion
EDUKE32_STATIC_ASSERT(sizeof(actor_t)%4 == 0); EDUKE32_STATIC_ASSERT(sizeof(actor_t)%4 == 0);
EDUKE32_STATIC_ASSERT(sizeof(DukePlayer_t)%4 == 0); EDUKE32_STATIC_ASSERT(sizeof(DukePlayer_t)%4 == 0);
@ -6430,7 +6417,8 @@ void GameInterface::SendMessage(const char* msg)
typebuf[0] = 0; typebuf[0] = 0;
} }
} }
else
void nix()
{ {
int32_t const hitstate = I_EnterText(typebuf, 120, 0); int32_t const hitstate = I_EnterText(typebuf, 120, 0);
@ -6466,9 +6454,15 @@ void GameInterface::SendMessage(const char* msg)
else else
pub = NUMPAGES; pub = NUMPAGES;
} }
}
#endif #endif
void GameInterface::FreeGameData()
{
engineUnInit();
G_Cleanup();
}
::GameInterface* CreateInterface() ::GameInterface* CreateInterface()
{ {
return new GameInterface; return new GameInterface;

View file

@ -1385,9 +1385,7 @@ void Gv_InitWeaponPointers(void)
if (!aplWeaponClip[i]) if (!aplWeaponClip[i])
{ {
initprintf("ERROR: NULL weapon! WTF?! %s\n", aszBuf); I_FatalError("ERROR: NULL weapon! WTF?! %s\n", aszBuf);
// Bexit(EXIT_SUCCESS);
G_Shutdown();
} }
Bsprintf(aszBuf, "WEAPON%d_RELOAD", i); Bsprintf(aszBuf, "WEAPON%d_RELOAD", i);

View file

@ -1172,7 +1172,8 @@ int WINAPI wWinMain (HINSTANCE hInstance, HINSTANCE nothing, LPWSTR cmdline, int
_CrtSetDbgFlag (_CrtSetDbgFlag(0) | _CRTDBG_LEAK_CHECK_DF); _CrtSetDbgFlag (_CrtSetDbgFlag(0) | _CRTDBG_LEAK_CHECK_DF);
// Use this to break at a specific allocation number. // Use this to break at a specific allocation number.
//_crtBreakAlloc = 10563;
// _crtBreakAlloc = 257481;
#endif #endif
int ret = DoMain (hInstance); int ret = DoMain (hInstance);

View file

@ -148,6 +148,7 @@ static inline int32_t G_DefaultActorHealth(int spriteNum)
struct GameInterface : ::GameInterface struct GameInterface : ::GameInterface
{ {
int app_main() override; int app_main() override;
void FreeGameData() override;
bool validate_hud(int) override; bool validate_hud(int) override;
void set_hud_layout(int size) override; void set_hud_layout(int size) override;
void set_hud_scale(int size) override; void set_hud_scale(int size) override;

View file

@ -367,11 +367,6 @@ void G_GameExit(const char *msg)
G_DisplayExtraScreens(); G_DisplayExtraScreens();
} }
if (*msg != 0) initprintf("%s\n",msg);
if (in3dmode())
G_Shutdown();
if (*msg != 0) if (*msg != 0)
{ {
if (!(msg[0] == ' ' && msg[1] == 0)) if (!(msg[0] == ' ' && msg[1] == 0))
@ -6732,6 +6727,7 @@ static void G_Cleanup(void)
if (label != (char *)&sprite[0]) Bfree(label); if (label != (char *)&sprite[0]) Bfree(label);
if (labelcode != (int32_t *)&sector[0]) Bfree(labelcode); if (labelcode != (int32_t *)&sector[0]) Bfree(labelcode);
if (labeltype != (int32_t*)&wall[0]) Bfree(labeltype);
Bfree(apScript); Bfree(apScript);
Bfree(bitptr); Bfree(bitptr);
@ -6743,21 +6739,6 @@ static void G_Cleanup(void)
hash_free(&h_dukeanim); hash_free(&h_dukeanim);
} }
/*
===================
=
= ShutDown
=
===================
*/
void G_Shutdown(void)
{
G_SetFog(0);
engineUnInit();
G_Cleanup();
}
/* /*
=================== ===================
= =
@ -7933,6 +7914,13 @@ void A_SpawnRandomGlass(int spriteNum, int wallNum, int glassCnt)
} }
} }
void GameInterface::FreeGameData()
{
G_SetFog(0);
engineUnInit();
G_Cleanup();
}
::GameInterface* CreateInterface() ::GameInterface* CreateInterface()
{ {
return new GameInterface; return new GameInterface;

View file

@ -180,7 +180,7 @@ void ClearStartMost(void)
for (i = 0; i < xdim; i++) for (i = 0; i < xdim; i++)
startdmost[i] = ydim; startdmost[i] = ydim;
memset(startumost, 0, xdim * sizeof(int16_t)); memset(startumost.Data(), 0, xdim * sizeof(int16_t));
} }
void void