2010-08-27 15:20:05 +00:00
|
|
|
/*
|
|
|
|
** gl_nodes.cpp
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 2005-2010 Christoph Oelckers
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
#include <math.h>
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#include <malloc.h> // for alloca()
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#else
|
2010-08-31 04:35:13 +00:00
|
|
|
#include <direct.h>
|
2010-08-27 15:20:05 +00:00
|
|
|
|
|
|
|
#define rmdir _rmdir
|
|
|
|
|
|
|
|
// TODO, maybe: stop using DWORD so I don't need to worry about conflicting
|
|
|
|
// with Windows' typedef. Then I could just include the header file instead
|
|
|
|
// of declaring everything here.
|
|
|
|
#define MAX_PATH 260
|
|
|
|
#define CSIDL_LOCAL_APPDATA 0x001c
|
|
|
|
extern "C" __declspec(dllimport) long __stdcall SHGetFolderPathA(void *hwnd, int csidl, void *hToken, unsigned long dwFlags, char *pszPath);
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <CoreServices/CoreServices.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "templates.h"
|
|
|
|
#include "m_alloc.h"
|
|
|
|
#include "m_argv.h"
|
|
|
|
#include "c_dispatch.h"
|
|
|
|
#include "m_swap.h"
|
|
|
|
#include "g_game.h"
|
|
|
|
#include "i_system.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "doomdef.h"
|
|
|
|
#include "p_local.h"
|
|
|
|
#include "nodebuild.h"
|
|
|
|
#include "doomstat.h"
|
|
|
|
#include "vectors.h"
|
|
|
|
#include "stats.h"
|
|
|
|
#include "doomerrors.h"
|
|
|
|
#include "p_setup.h"
|
|
|
|
#include "x86.h"
|
|
|
|
#include "version.h"
|
|
|
|
#include "md5.h"
|
|
|
|
|
|
|
|
void P_GetPolySpots (MapData * lump, TArray<FNodeBuilder::FPolyStart> &spots, TArray<FNodeBuilder::FPolyStart> &anchors);
|
|
|
|
|
|
|
|
CVAR(Bool, gl_cachenodes, true, CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
|
|
|
CVAR(Float, gl_cachetime, 0.6f, CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
|
|
|
|
|
|
|
void P_LoadZNodes (FileReader &dalump, DWORD id);
|
|
|
|
static bool CheckCachedNodes(MapData *map);
|
|
|
|
static void CreateCachedNodes(MapData *map);
|
|
|
|
|
|
|
|
|
|
|
|
// fixed 32 bit gl_vert format v2.0+ (glBsp 1.91)
|
|
|
|
struct mapglvertex_t
|
|
|
|
{
|
|
|
|
fixed_t x,y;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct gl3_mapsubsector_t
|
|
|
|
{
|
|
|
|
SDWORD numsegs;
|
|
|
|
SDWORD firstseg; // Index of first one; segs are stored sequentially.
|
|
|
|
};
|
|
|
|
|
|
|
|
struct glseg_t
|
|
|
|
{
|
|
|
|
WORD v1; // start vertex (16 bit)
|
|
|
|
WORD v2; // end vertex (16 bit)
|
|
|
|
WORD linedef; // linedef, or -1 for minisegs
|
|
|
|
WORD side; // side on linedef: 0 for right, 1 for left
|
|
|
|
WORD partner; // corresponding partner seg, or 0xffff on one-sided walls
|
|
|
|
};
|
|
|
|
|
|
|
|
struct glseg3_t
|
|
|
|
{
|
|
|
|
SDWORD v1;
|
|
|
|
SDWORD v2;
|
|
|
|
WORD linedef;
|
|
|
|
WORD side;
|
|
|
|
SDWORD partner;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct gl5_mapnode_t
|
|
|
|
{
|
|
|
|
SWORD x,y,dx,dy; // partition line
|
|
|
|
SWORD bbox[2][4]; // bounding box for each child
|
|
|
|
// If NF_SUBSECTOR is or'ed in, it's a subsector,
|
|
|
|
// else it's a node of another subtree.
|
|
|
|
DWORD children[2];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Collect all sidedefs which are not entirely covered by segs
|
|
|
|
// Old ZDBSPs could create such maps. If such a BSP is discovered
|
|
|
|
// a node rebuild must be done to ensure proper rendering
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static int CheckForMissingSegs()
|
|
|
|
{
|
|
|
|
float *added_seglen = new float[numsides];
|
|
|
|
int missing = 0;
|
|
|
|
|
|
|
|
memset(added_seglen, 0, sizeof(float)*numsides);
|
|
|
|
for(int i=0;i<numsegs;i++)
|
|
|
|
{
|
|
|
|
seg_t * seg = &segs[i];
|
|
|
|
|
|
|
|
if (seg->sidedef!=NULL)
|
|
|
|
{
|
|
|
|
// check all the segs and calculate the length they occupy on their sidedef
|
|
|
|
TVector2<double> vec1(seg->v2->x - seg->v1->x, seg->v2->y - seg->v1->y);
|
|
|
|
added_seglen[seg->sidedef - sides] += float(vec1.Length());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for(int i=0;i<numsides;i++)
|
|
|
|
{
|
|
|
|
side_t * side =&sides[i];
|
|
|
|
line_t * line = side->linedef;
|
|
|
|
|
|
|
|
TVector2<double> lvec(line->dx, line->dy);
|
|
|
|
float linelen = float(lvec.Length());
|
|
|
|
|
|
|
|
missing += (added_seglen[i] < linelen - FRACUNIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete [] added_seglen;
|
|
|
|
return missing;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Checks whether the nodes are suitable for GL rendering
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
bool P_CheckForGLNodes()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for(i=0;i<numsubsectors;i++)
|
|
|
|
{
|
|
|
|
subsector_t * sub = &subsectors[i];
|
|
|
|
seg_t * firstseg = sub->firstline;
|
|
|
|
seg_t * lastseg = sub->firstline + sub->numlines - 1;
|
|
|
|
|
|
|
|
if (firstseg->v1 != lastseg->v2)
|
|
|
|
{
|
|
|
|
// This subsector is incomplete which means that these
|
|
|
|
// are normal nodes
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for(DWORD j=0;j<sub->numlines;j++)
|
|
|
|
{
|
|
|
|
if (segs[j].linedef==NULL) // miniseg
|
|
|
|
{
|
|
|
|
// We already have GL nodes. Great!
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// all subsectors were closed but there are no minisegs
|
|
|
|
// Although unlikely this can happen. Such nodes are not a problem.
|
|
|
|
// all that is left is to check whether the BSP covers all sidedefs completely.
|
|
|
|
int missing = CheckForMissingSegs();
|
|
|
|
if (missing > 0)
|
|
|
|
{
|
|
|
|
Printf("%d missing segs counted\nThe BSP needs to be rebuilt", missing);
|
|
|
|
}
|
|
|
|
return missing == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// LoadGLVertexes
|
|
|
|
//
|
|
|
|
// loads GL vertices
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
#define gNd2 MAKE_ID('g','N','d','2')
|
|
|
|
#define gNd4 MAKE_ID('g','N','d','4')
|
|
|
|
#define gNd5 MAKE_ID('g','N','d','5')
|
|
|
|
|
|
|
|
#define GL_VERT_OFFSET 4
|
|
|
|
static int firstglvertex;
|
|
|
|
static bool format5;
|
|
|
|
|
|
|
|
static bool LoadGLVertexes(FileReader * f, wadlump_t * lump)
|
|
|
|
{
|
|
|
|
BYTE *gldata;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
firstglvertex = numvertexes;
|
|
|
|
|
|
|
|
int gllen=lump->Size;
|
|
|
|
|
|
|
|
gldata = new BYTE[gllen];
|
|
|
|
f->Seek(lump->FilePos, SEEK_SET);
|
|
|
|
f->Read(gldata, gllen);
|
|
|
|
|
|
|
|
if (*(int *)gldata == gNd5)
|
|
|
|
{
|
|
|
|
format5=true;
|
|
|
|
}
|
|
|
|
else if (*(int *)gldata != gNd2)
|
|
|
|
{
|
|
|
|
// GLNodes V1 and V4 are unsupported.
|
|
|
|
// V1 because the precision is insufficient and
|
|
|
|
// V4 due to the missing partner segs
|
|
|
|
Printf("GL nodes v%d found. This format is not supported by "GAMENAME"\n",
|
|
|
|
(*(int *)gldata == gNd4)? 4:1);
|
|
|
|
|
|
|
|
delete [] gldata;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else format5=false;
|
|
|
|
|
|
|
|
mapglvertex_t* mgl;
|
|
|
|
|
|
|
|
vertex_t * oldvertexes = vertexes;
|
|
|
|
numvertexes += (gllen - GL_VERT_OFFSET)/sizeof(mapglvertex_t);
|
|
|
|
vertexes = new vertex_t[numvertexes];
|
|
|
|
mgl = (mapglvertex_t *) (gldata + GL_VERT_OFFSET);
|
|
|
|
|
|
|
|
memcpy(vertexes, oldvertexes, firstglvertex * sizeof(vertex_t));
|
|
|
|
for(i=0;i<numlines;i++)
|
|
|
|
{
|
|
|
|
lines[i].v1 = vertexes + (lines[i].v1 - oldvertexes);
|
|
|
|
lines[i].v2 = vertexes + (lines[i].v2 - oldvertexes);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = firstglvertex; i < numvertexes; i++)
|
|
|
|
{
|
|
|
|
vertexes[i].x = LittleLong(mgl->x);
|
|
|
|
vertexes[i].y = LittleLong(mgl->y);
|
|
|
|
mgl++;
|
|
|
|
}
|
|
|
|
delete[] gldata;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// GL Nodes utilities
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static inline int checkGLVertex(int num)
|
|
|
|
{
|
|
|
|
if (num & 0x8000)
|
|
|
|
num = (num&0x7FFF)+firstglvertex;
|
|
|
|
return num;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int checkGLVertex3(int num)
|
|
|
|
{
|
|
|
|
if (num & 0xc0000000)
|
|
|
|
num = (num&0x3FFFFFFF)+firstglvertex;
|
|
|
|
return num;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// LoadGLSegs
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static bool LoadGLSegs(FileReader * f, wadlump_t * lump)
|
|
|
|
{
|
|
|
|
char *data;
|
|
|
|
int i;
|
|
|
|
line_t *ldef=NULL;
|
|
|
|
|
|
|
|
numsegs = lump->Size;
|
|
|
|
data= new char[numsegs];
|
|
|
|
f->Seek(lump->FilePos, SEEK_SET);
|
|
|
|
f->Read(data, lump->Size);
|
|
|
|
segs=NULL;
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
__try
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
if (!format5 && memcmp(data, "gNd3", 4))
|
|
|
|
{
|
|
|
|
numsegs/=sizeof(glseg_t);
|
|
|
|
segs = new seg_t[numsegs];
|
|
|
|
memset(segs,0,sizeof(seg_t)*numsegs);
|
|
|
|
glsegextras = new glsegextra_t[numsegs];
|
|
|
|
|
|
|
|
glseg_t * ml = (glseg_t*)data;
|
|
|
|
for(i = 0; i < numsegs; i++)
|
|
|
|
{ // check for gl-vertices
|
|
|
|
segs[i].v1 = &vertexes[checkGLVertex(LittleShort(ml->v1))];
|
|
|
|
segs[i].v2 = &vertexes[checkGLVertex(LittleShort(ml->v2))];
|
|
|
|
|
|
|
|
glsegextras[i].PartnerSeg = ml->partner == 0xFFFF ? DWORD_MAX : LittleShort(ml->partner);
|
|
|
|
if(ml->linedef != 0xffff)
|
|
|
|
{
|
|
|
|
ldef = &lines[LittleShort(ml->linedef)];
|
|
|
|
segs[i].linedef = ldef;
|
|
|
|
|
|
|
|
|
|
|
|
ml->side=LittleShort(ml->side);
|
|
|
|
segs[i].sidedef = ldef->sidedef[ml->side];
|
|
|
|
segs[i].frontsector = ldef->sidedef[ml->side]->sector;
|
|
|
|
if (ldef->flags & ML_TWOSIDED && ldef->sidedef[ml->side^1] != NULL)
|
|
|
|
{
|
|
|
|
segs[i].backsector = ldef->sidedef[ml->side^1]->sector;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ldef->flags &= ~ML_TWOSIDED;
|
|
|
|
segs[i].backsector = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
segs[i].linedef = NULL;
|
|
|
|
segs[i].sidedef = NULL;
|
|
|
|
|
|
|
|
segs[i].frontsector = NULL;
|
|
|
|
segs[i].backsector = NULL;
|
|
|
|
}
|
|
|
|
ml++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!format5) numsegs-=4;
|
|
|
|
numsegs/=sizeof(glseg3_t);
|
|
|
|
segs = new seg_t[numsegs];
|
|
|
|
memset(segs,0,sizeof(seg_t)*numsegs);
|
|
|
|
glsegextras = new glsegextra_t[numsegs];
|
|
|
|
|
|
|
|
glseg3_t * ml = (glseg3_t*)(data+ (format5? 0:4));
|
|
|
|
for(i = 0; i < numsegs; i++)
|
|
|
|
{ // check for gl-vertices
|
|
|
|
segs[i].v1 = &vertexes[checkGLVertex3(LittleLong(ml->v1))];
|
|
|
|
segs[i].v2 = &vertexes[checkGLVertex3(LittleLong(ml->v2))];
|
|
|
|
|
|
|
|
glsegextras[i].PartnerSeg = LittleLong(ml->partner);
|
|
|
|
|
|
|
|
if(ml->linedef != 0xffff) // skip minisegs
|
|
|
|
{
|
|
|
|
ldef = &lines[LittleLong(ml->linedef)];
|
|
|
|
segs[i].linedef = ldef;
|
|
|
|
|
|
|
|
|
|
|
|
ml->side=LittleShort(ml->side);
|
|
|
|
segs[i].sidedef = ldef->sidedef[ml->side];
|
|
|
|
segs[i].frontsector = ldef->sidedef[ml->side]->sector;
|
|
|
|
if (ldef->flags & ML_TWOSIDED && ldef->sidedef[ml->side^1] != NULL)
|
|
|
|
{
|
|
|
|
segs[i].backsector = ldef->sidedef[ml->side^1]->sector;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ldef->flags &= ~ML_TWOSIDED;
|
|
|
|
segs[i].backsector = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
segs[i].linedef = NULL;
|
|
|
|
segs[i].sidedef = NULL;
|
|
|
|
segs[i].frontsector = NULL;
|
|
|
|
segs[i].backsector = NULL;
|
|
|
|
}
|
|
|
|
ml++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete [] data;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
__except(1)
|
|
|
|
{
|
|
|
|
// Invalid data has the bad habit of requiring extensive checks here
|
|
|
|
// so let's just catch anything invalid and output a message.
|
|
|
|
// (at least under MSVC. GCC can't do SEH even for Windows... :( )
|
|
|
|
Printf("Invalid GL segs. The BSP will have to be rebuilt.\n");
|
|
|
|
delete [] data;
|
|
|
|
delete [] segs;
|
|
|
|
segs = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// LoadGLSubsectors
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static bool LoadGLSubsectors(FileReader * f, wadlump_t * lump)
|
|
|
|
{
|
|
|
|
char * datab;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
numsubsectors = lump->Size;
|
|
|
|
datab = new char[numsubsectors];
|
|
|
|
f->Seek(lump->FilePos, SEEK_SET);
|
|
|
|
f->Read(datab, lump->Size);
|
|
|
|
|
|
|
|
if (numsubsectors == 0)
|
|
|
|
{
|
|
|
|
delete [] datab;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!format5 && memcmp(datab, "gNd3", 4))
|
|
|
|
{
|
|
|
|
mapsubsector_t * data = (mapsubsector_t*) datab;
|
|
|
|
numsubsectors /= sizeof(mapsubsector_t);
|
|
|
|
subsectors = new subsector_t[numsubsectors];
|
|
|
|
memset(subsectors,0,numsubsectors * sizeof(subsector_t));
|
|
|
|
|
|
|
|
for (i=0; i<numsubsectors; i++)
|
|
|
|
{
|
|
|
|
subsectors[i].numlines = LittleShort(data[i].numsegs );
|
|
|
|
subsectors[i].firstline = segs + LittleShort(data[i].firstseg);
|
|
|
|
|
|
|
|
if (subsectors[i].numlines == 0)
|
|
|
|
{
|
|
|
|
delete [] datab;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gl3_mapsubsector_t * data = (gl3_mapsubsector_t*) (datab+(format5? 0:4));
|
|
|
|
numsubsectors /= sizeof(gl3_mapsubsector_t);
|
|
|
|
subsectors = new subsector_t[numsubsectors];
|
|
|
|
memset(subsectors,0,numsubsectors * sizeof(subsector_t));
|
|
|
|
|
|
|
|
for (i=0; i<numsubsectors; i++)
|
|
|
|
{
|
|
|
|
subsectors[i].numlines = LittleLong(data[i].numsegs );
|
|
|
|
subsectors[i].firstline = segs + LittleLong(data[i].firstseg);
|
|
|
|
|
|
|
|
if (subsectors[i].numlines == 0)
|
|
|
|
{
|
|
|
|
delete [] datab;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i=0; i<numsubsectors; i++)
|
|
|
|
{
|
|
|
|
for(unsigned j=0;j<subsectors[i].numlines;j++)
|
|
|
|
{
|
|
|
|
seg_t * seg = subsectors[i].firstline + j;
|
|
|
|
if (seg->linedef==NULL) seg->frontsector = seg->backsector = subsectors[i].firstline->frontsector;
|
|
|
|
}
|
|
|
|
seg_t *firstseg = subsectors[i].firstline;
|
|
|
|
seg_t *lastseg = subsectors[i].firstline + subsectors[i].numlines - 1;
|
|
|
|
// The subsector must be closed. If it isn't we can't use these nodes and have to do a rebuild.
|
|
|
|
if (lastseg->v2 != firstseg->v1)
|
|
|
|
{
|
|
|
|
delete [] datab;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
delete [] datab;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// P_LoadNodes
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static bool LoadNodes (FileReader * f, wadlump_t * lump)
|
|
|
|
{
|
|
|
|
const int NF_SUBSECTOR = 0x8000;
|
|
|
|
const int GL5_NF_SUBSECTOR = (1 << 31);
|
|
|
|
|
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
int k;
|
|
|
|
node_t* no;
|
|
|
|
WORD* used;
|
|
|
|
|
|
|
|
if (!format5)
|
|
|
|
{
|
|
|
|
mapnode_t* mn, * basemn;
|
|
|
|
numnodes = lump->Size / sizeof(mapnode_t);
|
|
|
|
|
|
|
|
if (numnodes == 0) return false;
|
|
|
|
|
|
|
|
nodes = new node_t[numnodes];
|
|
|
|
f->Seek(lump->FilePos, SEEK_SET);
|
|
|
|
|
|
|
|
basemn = mn = new mapnode_t[numnodes];
|
|
|
|
f->Read(mn, lump->Size);
|
|
|
|
|
|
|
|
used = (WORD *)alloca (sizeof(WORD)*numnodes);
|
|
|
|
memset (used, 0, sizeof(WORD)*numnodes);
|
|
|
|
|
|
|
|
no = nodes;
|
|
|
|
|
|
|
|
for (i = 0; i < numnodes; i++, no++, mn++)
|
|
|
|
{
|
|
|
|
no->x = LittleShort(mn->x)<<FRACBITS;
|
|
|
|
no->y = LittleShort(mn->y)<<FRACBITS;
|
|
|
|
no->dx = LittleShort(mn->dx)<<FRACBITS;
|
|
|
|
no->dy = LittleShort(mn->dy)<<FRACBITS;
|
|
|
|
for (j = 0; j < 2; j++)
|
|
|
|
{
|
|
|
|
WORD child = LittleShort(mn->children[j]);
|
|
|
|
if (child & NF_SUBSECTOR)
|
|
|
|
{
|
|
|
|
child &= ~NF_SUBSECTOR;
|
|
|
|
if (child >= numsubsectors)
|
|
|
|
{
|
|
|
|
delete [] basemn;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
no->children[j] = (BYTE *)&subsectors[child] + 1;
|
|
|
|
}
|
|
|
|
else if (child >= numnodes)
|
|
|
|
{
|
|
|
|
delete [] basemn;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if (used[child])
|
|
|
|
{
|
|
|
|
delete [] basemn;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
no->children[j] = &nodes[child];
|
|
|
|
used[child] = j + 1;
|
|
|
|
}
|
|
|
|
for (k = 0; k < 4; k++)
|
|
|
|
{
|
|
|
|
no->bbox[j][k] = LittleShort(mn->bbox[j][k])<<FRACBITS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete [] basemn;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gl5_mapnode_t* mn, * basemn;
|
|
|
|
numnodes = lump->Size / sizeof(gl5_mapnode_t);
|
|
|
|
|
|
|
|
if (numnodes == 0) return false;
|
|
|
|
|
|
|
|
nodes = new node_t[numnodes];
|
|
|
|
f->Seek(lump->FilePos, SEEK_SET);
|
|
|
|
|
|
|
|
basemn = mn = new gl5_mapnode_t[numnodes];
|
|
|
|
f->Read(mn, lump->Size);
|
|
|
|
|
|
|
|
used = (WORD *)alloca (sizeof(WORD)*numnodes);
|
|
|
|
memset (used, 0, sizeof(WORD)*numnodes);
|
|
|
|
|
|
|
|
no = nodes;
|
|
|
|
|
|
|
|
for (i = 0; i < numnodes; i++, no++, mn++)
|
|
|
|
{
|
|
|
|
no->x = LittleShort(mn->x)<<FRACBITS;
|
|
|
|
no->y = LittleShort(mn->y)<<FRACBITS;
|
|
|
|
no->dx = LittleShort(mn->dx)<<FRACBITS;
|
|
|
|
no->dy = LittleShort(mn->dy)<<FRACBITS;
|
|
|
|
for (j = 0; j < 2; j++)
|
|
|
|
{
|
|
|
|
SDWORD child = LittleLong(mn->children[j]);
|
|
|
|
if (child & GL5_NF_SUBSECTOR)
|
|
|
|
{
|
|
|
|
child &= ~GL5_NF_SUBSECTOR;
|
|
|
|
if (child >= numsubsectors)
|
|
|
|
{
|
|
|
|
delete [] basemn;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
no->children[j] = (BYTE *)&subsectors[child] + 1;
|
|
|
|
}
|
|
|
|
else if (child >= numnodes)
|
|
|
|
{
|
|
|
|
delete [] basemn;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if (used[child])
|
|
|
|
{
|
|
|
|
delete [] basemn;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
no->children[j] = &nodes[child];
|
|
|
|
used[child] = j + 1;
|
|
|
|
}
|
|
|
|
for (k = 0; k < 4; k++)
|
|
|
|
{
|
|
|
|
no->bbox[j][k] = LittleShort(mn->bbox[j][k])<<FRACBITS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
delete [] basemn;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// loads the GL node data
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static bool DoLoadGLNodes(FileReader * f, wadlump_t * lumps)
|
|
|
|
{
|
|
|
|
if (!LoadGLVertexes(f, &lumps[0]))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!LoadGLSegs(f, &lumps[1]))
|
|
|
|
{
|
|
|
|
delete [] segs;
|
|
|
|
segs = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!LoadGLSubsectors(f, &lumps[2]))
|
|
|
|
{
|
|
|
|
delete [] subsectors;
|
|
|
|
subsectors = NULL;
|
|
|
|
delete [] segs;
|
|
|
|
segs = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!LoadNodes(f, &lumps[3]))
|
|
|
|
{
|
|
|
|
delete [] nodes;
|
|
|
|
nodes = NULL;
|
|
|
|
delete [] subsectors;
|
|
|
|
subsectors = NULL;
|
|
|
|
delete [] segs;
|
|
|
|
segs = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Quick check for the validity of the nodes
|
|
|
|
// For invalid nodes there is a high chance that this test will fail
|
|
|
|
|
|
|
|
for (int i = 0; i < numsubsectors; i++)
|
|
|
|
{
|
|
|
|
seg_t * seg = subsectors[i].firstline;
|
|
|
|
if (!seg->sidedef)
|
|
|
|
{
|
|
|
|
Printf("GL nodes contain invalid data. The BSP has to be rebuilt.\n");
|
|
|
|
delete [] nodes;
|
|
|
|
nodes = NULL;
|
|
|
|
delete [] subsectors;
|
|
|
|
subsectors = NULL;
|
|
|
|
delete [] segs;
|
|
|
|
segs = NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// check whether the BSP covers all sidedefs completely.
|
|
|
|
int missing = CheckForMissingSegs();
|
|
|
|
if (missing > 0)
|
|
|
|
{
|
|
|
|
Printf("%d missing segs counted in GL nodes.\nThe BSP has to be rebuilt", missing);
|
|
|
|
}
|
|
|
|
return missing == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// MatchHeader
|
|
|
|
//
|
|
|
|
// Checks whether a GL_LEVEL header belongs to this level
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
static bool MatchHeader(const char * label, const char * hdata)
|
|
|
|
{
|
|
|
|
if (!memcmp(hdata, "LEVEL=", 6) == 0)
|
|
|
|
{
|
|
|
|
size_t labellen = strlen(label);
|
|
|
|
|
|
|
|
if (strnicmp(hdata+6, label, labellen)==0 &&
|
|
|
|
(hdata[6+labellen]==0xa || hdata[6+labellen]==0xd))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// FindGLNodesInWAD
|
|
|
|
//
|
|
|
|
// Looks for GL nodes in the same WAD as the level itself
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
static int FindGLNodesInWAD(int labellump)
|
|
|
|
{
|
|
|
|
int wadfile = Wads.GetLumpFile(labellump);
|
|
|
|
FString glheader;
|
|
|
|
|
|
|
|
glheader.Format("GL_%s", Wads.GetLumpFullName(labellump));
|
|
|
|
if (glheader.Len()<=8)
|
|
|
|
{
|
|
|
|
int gllabel = Wads.CheckNumForName(glheader, ns_global, wadfile);
|
|
|
|
if (gllabel >= 0) return gllabel;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Before scanning the entire WAD directory let's check first whether
|
|
|
|
// it is necessary.
|
|
|
|
int gllabel = Wads.CheckNumForName("GL_LEVEL", ns_global, wadfile);
|
|
|
|
|
|
|
|
if (gllabel >= 0)
|
|
|
|
{
|
|
|
|
int lastlump=0;
|
|
|
|
int lump;
|
|
|
|
while ((lump=Wads.FindLump("GL_LEVEL", &lastlump))>=0)
|
|
|
|
{
|
|
|
|
if (Wads.GetLumpFile(lump)==wadfile)
|
|
|
|
{
|
|
|
|
FMemLump mem = Wads.ReadLump(lump);
|
|
|
|
if (MatchHeader(Wads.GetLumpFullName(labellump), (const char *)mem.GetMem())) return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// FindGLNodesInWAD
|
|
|
|
//
|
|
|
|
// Looks for GL nodes in the same WAD as the level itself
|
|
|
|
// When this function returns the file pointer points to
|
|
|
|
// the directory entry of the GL_VERTS lump
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
static int FindGLNodesInFile(FileReader * f, const char * label)
|
|
|
|
{
|
|
|
|
FString glheader;
|
|
|
|
bool mustcheck=false;
|
|
|
|
DWORD id, dirofs, numentries;
|
|
|
|
DWORD offset, size;
|
|
|
|
char lumpname[9];
|
|
|
|
|
|
|
|
glheader.Format("GL_%.8s", label);
|
|
|
|
if (glheader.Len()>8)
|
|
|
|
{
|
|
|
|
glheader="GL_LEVEL";
|
|
|
|
mustcheck=true;
|
|
|
|
}
|
|
|
|
|
|
|
|
f->Seek(0, SEEK_SET);
|
|
|
|
(*f) >> id >> numentries >> dirofs;
|
|
|
|
|
|
|
|
if ((id == IWAD_ID || id == PWAD_ID) && numentries > 4)
|
|
|
|
{
|
|
|
|
f->Seek(dirofs, SEEK_SET);
|
|
|
|
for(DWORD i=0;i<numentries-4;i++)
|
|
|
|
{
|
|
|
|
(*f) >> offset >> size;
|
|
|
|
f->Read(lumpname, 8);
|
|
|
|
if (!strnicmp(lumpname, glheader, 8))
|
|
|
|
{
|
|
|
|
if (mustcheck)
|
|
|
|
{
|
|
|
|
char check[16]={0};
|
|
|
|
int filepos = f->Tell();
|
|
|
|
f->Seek(offset, SEEK_SET);
|
|
|
|
f->Read(check, 16);
|
|
|
|
f->Seek(filepos, SEEK_SET);
|
|
|
|
if (MatchHeader(label, check)) return i;
|
|
|
|
}
|
|
|
|
else return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Checks for the presence of GL nodes in the loaded WADs or a .GWA file
|
|
|
|
// returns true if successful
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
bool P_LoadGLNodes(MapData * map)
|
|
|
|
{
|
2010-09-16 14:26:56 +00:00
|
|
|
|
|
|
|
if (map->MapLumps[ML_GLZNODES].Size != 0)
|
|
|
|
{
|
|
|
|
const int idcheck = MAKE_ID('Z','G','L','N');
|
|
|
|
const int idcheck2 = MAKE_ID('Z','G','L','2');
|
|
|
|
const int idcheck3 = MAKE_ID('X','G','L','N');
|
|
|
|
const int idcheck4 = MAKE_ID('X','G','L','2');
|
|
|
|
int id;
|
|
|
|
|
|
|
|
map->Seek(ML_GLZNODES);
|
|
|
|
map->file->Read (&id, 4);
|
|
|
|
if (id == idcheck || id == idcheck2 || id == idcheck3 || id == idcheck4)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
subsectors = NULL;
|
|
|
|
segs = NULL;
|
|
|
|
nodes = NULL;
|
|
|
|
P_LoadZNodes (*map->file, id);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
catch (CRecoverableError &)
|
|
|
|
{
|
|
|
|
if (subsectors != NULL)
|
|
|
|
{
|
|
|
|
delete[] subsectors;
|
|
|
|
subsectors = NULL;
|
|
|
|
}
|
|
|
|
if (segs != NULL)
|
|
|
|
{
|
|
|
|
delete[] segs;
|
|
|
|
segs = NULL;
|
|
|
|
}
|
|
|
|
if (nodes != NULL)
|
|
|
|
{
|
|
|
|
delete[] nodes;
|
|
|
|
nodes = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-27 15:20:05 +00:00
|
|
|
if (!CheckCachedNodes(map))
|
|
|
|
{
|
|
|
|
wadlump_t gwalumps[4];
|
|
|
|
char path[256];
|
|
|
|
int li;
|
|
|
|
int lumpfile = Wads.GetLumpFile(map->lumpnum);
|
|
|
|
bool mapinwad = map->file == Wads.GetFileReader(lumpfile);
|
|
|
|
FileReader * fr = map->file;
|
|
|
|
FILE * f_gwa = NULL;
|
|
|
|
|
|
|
|
const char * name = Wads.GetWadFullName(lumpfile);
|
|
|
|
|
|
|
|
if (mapinwad)
|
|
|
|
{
|
|
|
|
li = FindGLNodesInWAD(map->lumpnum);
|
|
|
|
|
|
|
|
if (li>=0)
|
|
|
|
{
|
|
|
|
// GL nodes are loaded with a WAD
|
|
|
|
for(int i=0;i<4;i++)
|
|
|
|
{
|
|
|
|
gwalumps[i].FilePos=Wads.GetLumpOffset(li+i+1);
|
|
|
|
gwalumps[i].Size=Wads.LumpLength(li+i+1);
|
|
|
|
}
|
|
|
|
return DoLoadGLNodes(fr, gwalumps);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
strcpy(path, name);
|
|
|
|
|
|
|
|
char * ext = strrchr(path, '.');
|
|
|
|
if (ext)
|
|
|
|
{
|
|
|
|
strcpy(ext, ".gwa");
|
|
|
|
// Todo: Compare file dates
|
|
|
|
|
|
|
|
f_gwa = fopen(path, "rb");
|
|
|
|
if (f_gwa==NULL) return false;
|
|
|
|
|
|
|
|
fr = new FileReader(f_gwa);
|
|
|
|
|
|
|
|
strncpy(map->MapLumps[0].Name, Wads.GetLumpFullName(map->lumpnum), 8);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool result = false;
|
|
|
|
li = FindGLNodesInFile(fr, map->MapLumps[0].Name);
|
|
|
|
if (li!=-1)
|
|
|
|
{
|
|
|
|
static const char check[][9]={"GL_VERT","GL_SEGS","GL_SSECT","GL_NODES"};
|
|
|
|
result=true;
|
|
|
|
for(unsigned i=0; i<4;i++)
|
|
|
|
{
|
|
|
|
(*fr) >> gwalumps[i].FilePos;
|
|
|
|
(*fr) >> gwalumps[i].Size;
|
|
|
|
fr->Read(gwalumps[i].Name, 8);
|
|
|
|
if (strnicmp(gwalumps[i].Name, check[i], 8))
|
|
|
|
{
|
|
|
|
result=false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (result) result = DoLoadGLNodes(fr, gwalumps);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (f_gwa)
|
|
|
|
{
|
|
|
|
delete fr;
|
|
|
|
fclose(f_gwa);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
else return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Checks whether nodes are GL friendly or not
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
bool P_CheckNodes(MapData * map, bool rebuilt, int buildtime)
|
|
|
|
{
|
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
// If the map loading code has performed a node rebuild we don't need to check for it again.
|
|
|
|
if (!rebuilt && !P_CheckForGLNodes())
|
|
|
|
{
|
|
|
|
ret = true; // we are not using the level's original nodes if we get here.
|
|
|
|
for (int i = 0; i < numsubsectors; i++)
|
|
|
|
{
|
|
|
|
gamesubsectors[i].sector = gamesubsectors[i].firstline->sidedef->sector;
|
|
|
|
}
|
|
|
|
|
|
|
|
nodes = NULL;
|
|
|
|
numnodes = 0;
|
|
|
|
subsectors = NULL;
|
|
|
|
numsubsectors = 0;
|
|
|
|
if (segs) delete [] segs;
|
|
|
|
segs = NULL;
|
|
|
|
numsegs = 0;
|
|
|
|
|
|
|
|
// Try to load GL nodes (cached or GWA)
|
|
|
|
if (!P_LoadGLNodes(map))
|
|
|
|
{
|
|
|
|
// none found - we have to build new ones!
|
|
|
|
unsigned int startTime, endTime;
|
|
|
|
|
|
|
|
startTime = I_FPSTime ();
|
|
|
|
TArray<FNodeBuilder::FPolyStart> polyspots, anchors;
|
|
|
|
P_GetPolySpots (map, polyspots, anchors);
|
|
|
|
FNodeBuilder::FLevel leveldata =
|
|
|
|
{
|
|
|
|
vertexes, numvertexes,
|
|
|
|
sides, numsides,
|
|
|
|
lines, numlines
|
|
|
|
};
|
|
|
|
leveldata.FindMapBounds ();
|
|
|
|
FNodeBuilder builder (leveldata, polyspots, anchors, true);
|
|
|
|
delete[] vertexes;
|
|
|
|
builder.Extract (nodes, numnodes,
|
|
|
|
segs, glsegextras, numsegs,
|
|
|
|
subsectors, numsubsectors,
|
|
|
|
vertexes, numvertexes);
|
|
|
|
endTime = I_FPSTime ();
|
|
|
|
DPrintf ("BSP generation took %.3f sec (%d segs)\n", (endTime - startTime) * 0.001, numsegs);
|
|
|
|
buildtime = endTime - startTime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
// Building nodes in debug is much slower so let's cache them only if cachetime is 0
|
|
|
|
buildtime = 0;
|
|
|
|
#endif
|
|
|
|
if (gl_cachenodes && buildtime/1000.f >= gl_cachetime)
|
|
|
|
{
|
|
|
|
DPrintf("Caching nodes\n");
|
|
|
|
CreateCachedNodes(map);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DPrintf("Not caching nodes (time = %f)\n", buildtime/1000.f);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!gamenodes)
|
|
|
|
{
|
|
|
|
gamenodes = nodes;
|
|
|
|
numgamenodes = numnodes;
|
|
|
|
gamesubsectors = subsectors;
|
|
|
|
numgamesubsectors = numsubsectors;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Node caching
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
typedef TArray<BYTE> MemFile;
|
|
|
|
|
|
|
|
static FString GetCachePath()
|
|
|
|
{
|
|
|
|
FString path;
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
char pathstr[MAX_PATH];
|
|
|
|
if (0 != SHGetFolderPathA(NULL, CSIDL_LOCAL_APPDATA, NULL, 0, pathstr))
|
|
|
|
{ // Failed (e.g. On Win9x): use program directory
|
|
|
|
path = progdir;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
path = pathstr;
|
|
|
|
}
|
|
|
|
path += "/zdoom/cache";
|
|
|
|
#elif defined(__APPLE__)
|
|
|
|
char pathstr[PATH_MAX];
|
|
|
|
FSRef folder;
|
|
|
|
|
|
|
|
if (noErr == FSFindFolder(kLocalDomain, kApplicationSupportFolderType, kCreateFolder, &folder) &&
|
2010-09-18 20:13:56 +00:00
|
|
|
noErr == FSRefMakePath(&folder, (UInt8*)path.GetChars(), PATH_MAX))
|
2010-08-27 15:20:05 +00:00
|
|
|
{
|
|
|
|
path = pathstr;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
path = progdir;
|
|
|
|
}
|
|
|
|
path += "/zdoom/cache";
|
|
|
|
#else
|
|
|
|
// Don't use GAME_DIR and such so that ZDoom and its child ports can share the node cache.
|
|
|
|
path = NicePath("~/.zdoom/cache");
|
|
|
|
#endif
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
static FString CreateCacheName(MapData *map, bool create)
|
|
|
|
{
|
|
|
|
FString path = GetCachePath();
|
|
|
|
FString lumpname = Wads.GetLumpFullPath(map->lumpnum);
|
|
|
|
int separator = lumpname.IndexOf(':');
|
|
|
|
path << '/' << lumpname.Left(separator);
|
|
|
|
if (create) CreatePath(path);
|
|
|
|
|
|
|
|
lumpname.ReplaceChars('/', '%');
|
|
|
|
path << '/' << lumpname.Right(lumpname.Len() - separator - 1) << ".gzc";
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void WriteByte(MemFile &f, BYTE b)
|
|
|
|
{
|
|
|
|
f.Push(b);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void WriteWord(MemFile &f, WORD b)
|
|
|
|
{
|
|
|
|
int v = f.Reserve(2);
|
|
|
|
f[v] = (BYTE)b;
|
|
|
|
f[v+1] = (BYTE)(b>>8);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void WriteLong(MemFile &f, DWORD b)
|
|
|
|
{
|
|
|
|
int v = f.Reserve(4);
|
|
|
|
f[v] = (BYTE)b;
|
|
|
|
f[v+1] = (BYTE)(b>>8);
|
|
|
|
f[v+2] = (BYTE)(b>>16);
|
|
|
|
f[v+3] = (BYTE)(b>>24);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void CreateCachedNodes(MapData *map)
|
|
|
|
{
|
|
|
|
MemFile ZNodes;
|
|
|
|
|
|
|
|
WriteLong(ZNodes, 0);
|
|
|
|
WriteLong(ZNodes, numvertexes);
|
|
|
|
for(int i=0;i<numvertexes;i++)
|
|
|
|
{
|
|
|
|
WriteLong(ZNodes, vertexes[i].x);
|
|
|
|
WriteLong(ZNodes, vertexes[i].y);
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteLong(ZNodes, numsubsectors);
|
|
|
|
for(int i=0;i<numsubsectors;i++)
|
|
|
|
{
|
|
|
|
WriteLong(ZNodes, subsectors[i].numlines);
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteLong(ZNodes, numsegs);
|
|
|
|
for(int i=0;i<numsegs;i++)
|
|
|
|
{
|
|
|
|
WriteLong(ZNodes, DWORD(segs[i].v1 - vertexes));
|
|
|
|
WriteLong(ZNodes, DWORD(glsegextras[i].PartnerSeg));
|
|
|
|
if (segs[i].linedef)
|
|
|
|
{
|
|
|
|
WriteLong(ZNodes, DWORD(segs[i].linedef - lines));
|
|
|
|
WriteByte(ZNodes, segs[i].sidedef == segs[i].linedef->sidedef[0]? 0:1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WriteLong(ZNodes, 0xffffffffu);
|
|
|
|
WriteByte(ZNodes, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteLong(ZNodes, numnodes);
|
|
|
|
for(int i=0;i<numnodes;i++)
|
|
|
|
{
|
|
|
|
WriteWord(ZNodes, nodes[i].x >> FRACBITS);
|
|
|
|
WriteWord(ZNodes, nodes[i].y >> FRACBITS);
|
|
|
|
WriteWord(ZNodes, nodes[i].dx >> FRACBITS);
|
|
|
|
WriteWord(ZNodes, nodes[i].dy >> FRACBITS);
|
|
|
|
for (int j = 0; j < 2; ++j)
|
|
|
|
{
|
|
|
|
for (int k = 0; k < 4; ++k)
|
|
|
|
{
|
|
|
|
WriteWord(ZNodes, nodes[i].bbox[j][k] >> FRACBITS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int j = 0; j < 2; ++j)
|
|
|
|
{
|
|
|
|
DWORD child;
|
|
|
|
if ((size_t)nodes[i].children[j] & 1)
|
|
|
|
{
|
|
|
|
child = 0x80000000 | DWORD((subsector_t *)((BYTE *)nodes[i].children[j] - 1) - subsectors);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
child = DWORD((node_t *)nodes[i].children[j] - nodes);
|
|
|
|
}
|
|
|
|
WriteLong(ZNodes, child);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uLongf outlen = ZNodes.Size();
|
|
|
|
BYTE *compressed;
|
|
|
|
int offset = numlines * 8 + 12 + 16;
|
|
|
|
int r;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
compressed = new Bytef[outlen + offset];
|
|
|
|
r = compress (compressed + offset, &outlen, &ZNodes[0], ZNodes.Size());
|
|
|
|
if (r == Z_BUF_ERROR)
|
|
|
|
{
|
|
|
|
delete[] compressed;
|
|
|
|
outlen += 1024;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (r == Z_BUF_ERROR);
|
|
|
|
|
|
|
|
memcpy(compressed, "CACH", 4);
|
|
|
|
DWORD len = LittleLong(numlines);
|
|
|
|
memcpy(compressed+4, &len, 4);
|
|
|
|
map->GetChecksum(compressed+8);
|
|
|
|
for(int i=0;i<numlines;i++)
|
|
|
|
{
|
|
|
|
DWORD ndx[2] = {LittleLong(DWORD(lines[i].v1 - vertexes)), LittleLong(DWORD(lines[i].v2 - vertexes)) };
|
|
|
|
memcpy(compressed+8+16+8*i, ndx, 8);
|
|
|
|
}
|
|
|
|
memcpy(compressed + offset - 4, "ZGL2", 4);
|
|
|
|
|
|
|
|
FString path = CreateCacheName(map, true);
|
|
|
|
FILE *f = fopen(path, "wb");
|
|
|
|
fwrite(compressed, 1, outlen+offset, f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool CheckCachedNodes(MapData *map)
|
|
|
|
{
|
|
|
|
char magic[4] = {0,0,0,0};
|
|
|
|
BYTE md5[16];
|
|
|
|
BYTE md5map[16];
|
|
|
|
DWORD numlin;
|
|
|
|
DWORD *verts;
|
|
|
|
|
|
|
|
FString path = CreateCacheName(map, false);
|
|
|
|
FILE *f = fopen(path, "rb");
|
|
|
|
if (f == NULL) return false;
|
|
|
|
|
|
|
|
if (fread(magic, 1, 4, f) != 4) goto errorout;
|
|
|
|
if (memcmp(magic, "CACH", 4)) goto errorout;
|
|
|
|
|
|
|
|
if (fread(&numlin, 4, 1, f) != 1) goto errorout;
|
|
|
|
numlin = LittleLong(numlin);
|
2010-08-31 04:35:13 +00:00
|
|
|
if ((int)numlin != numlines) goto errorout;
|
2010-08-27 15:20:05 +00:00
|
|
|
|
|
|
|
if (fread(md5, 1, 16, f) != 16) goto errorout;
|
|
|
|
map->GetChecksum(md5map);
|
|
|
|
if (memcmp(md5, md5map, 16)) goto errorout;
|
|
|
|
|
|
|
|
verts = new DWORD[numlin * 8];
|
|
|
|
if (fread(verts, 8, numlin, f) != numlin) goto errorout;
|
|
|
|
|
|
|
|
if (fread(magic, 1, 4, f) != 4) goto errorout;
|
|
|
|
if (memcmp(magic, "ZGL2", 4)) goto errorout;
|
|
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
long pos = ftell(f);
|
|
|
|
FileReader fr(f);
|
|
|
|
fr.Seek(pos, SEEK_SET);
|
|
|
|
P_LoadZNodes (fr, MAKE_ID('Z','G','L','2'));
|
|
|
|
}
|
|
|
|
catch (CRecoverableError &error)
|
|
|
|
{
|
|
|
|
Printf ("Error loading nodes: %s\n", error.GetMessage());
|
|
|
|
|
|
|
|
if (subsectors != NULL)
|
|
|
|
{
|
|
|
|
delete[] subsectors;
|
|
|
|
subsectors = NULL;
|
|
|
|
}
|
|
|
|
if (segs != NULL)
|
|
|
|
{
|
|
|
|
delete[] segs;
|
|
|
|
segs = NULL;
|
|
|
|
}
|
|
|
|
if (nodes != NULL)
|
|
|
|
{
|
|
|
|
delete[] nodes;
|
|
|
|
nodes = NULL;
|
|
|
|
}
|
|
|
|
goto errorout;
|
|
|
|
}
|
|
|
|
|
|
|
|
for(int i=0;i<numlines;i++)
|
|
|
|
{
|
|
|
|
lines[i].v1 = &vertexes[LittleLong(verts[i*2])];
|
|
|
|
lines[i].v2 = &vertexes[LittleLong(verts[i*2+1])];
|
|
|
|
}
|
|
|
|
delete [] verts;
|
|
|
|
|
|
|
|
fclose(f);
|
|
|
|
return true;
|
|
|
|
|
|
|
|
errorout:
|
|
|
|
fclose(f);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
CCMD(clearnodecache)
|
|
|
|
{
|
|
|
|
TArray<FFileList> list;
|
|
|
|
FString path = GetCachePath();
|
|
|
|
path += "/";
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
ScanDirectory(list, path);
|
|
|
|
}
|
|
|
|
catch (CRecoverableError &err)
|
|
|
|
{
|
|
|
|
Printf("%s", err.GetMessage());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan list backwards so that when we reach a directory
|
|
|
|
// all files within are already deleted.
|
|
|
|
for(int i = list.Size()-1; i >= 0; i--)
|
|
|
|
{
|
|
|
|
if (list[i].isDirectory)
|
|
|
|
{
|
|
|
|
rmdir(list[i].Filename);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
remove(list[i].Filename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Keep both the original nodes from the WAD and the GL nodes created here.
|
|
|
|
// The original set is only being used to get the sector for in-game
|
|
|
|
// positioning of actors but not for rendering.
|
|
|
|
//
|
|
|
|
// This is necessary because ZDBSP is much more sensitive
|
|
|
|
// to sloppy mapping practices that produce overlapping sectors.
|
|
|
|
// The crane in P:AR E1M3 is a good example that would be broken if
|
|
|
|
// this wasn't done.
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// P_PointInSubsector
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
subsector_t *P_PointInSubsector (fixed_t x, fixed_t y)
|
|
|
|
{
|
|
|
|
node_t *node;
|
|
|
|
int side;
|
|
|
|
|
|
|
|
// single subsector is a special case
|
|
|
|
if (numgamenodes == 0)
|
|
|
|
return gamesubsectors;
|
|
|
|
|
|
|
|
node = gamenodes + numgamenodes - 1;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
side = R_PointOnSide (x, y, node);
|
|
|
|
node = (node_t *)node->children[side];
|
|
|
|
}
|
|
|
|
while (!((size_t)node & 1));
|
|
|
|
|
|
|
|
return (subsector_t *)((BYTE *)node - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// PointOnLine
|
|
|
|
//
|
|
|
|
// Same as the one im the node builder, but not part of a specific class
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
static bool PointOnLine (int x, int y, int x1, int y1, int dx, int dy)
|
|
|
|
{
|
|
|
|
const double SIDE_EPSILON = 6.5536;
|
|
|
|
|
|
|
|
// For most cases, a simple dot product is enough.
|
|
|
|
double d_dx = double(dx);
|
|
|
|
double d_dy = double(dy);
|
|
|
|
double d_x = double(x);
|
|
|
|
double d_y = double(y);
|
|
|
|
double d_x1 = double(x1);
|
|
|
|
double d_y1 = double(y1);
|
|
|
|
|
|
|
|
double s_num = (d_y1-d_y)*d_dx - (d_x1-d_x)*d_dy;
|
|
|
|
|
|
|
|
if (fabs(s_num) < 17179869184.0) // 4<<32
|
|
|
|
{
|
|
|
|
// Either the point is very near the line, or the segment defining
|
|
|
|
// the line is very short: Do a more expensive test to determine
|
|
|
|
// just how far from the line the point is.
|
|
|
|
double l = sqrt(d_dx*d_dx+d_dy*d_dy);
|
|
|
|
double dist = fabs(s_num)/l;
|
|
|
|
if (dist < SIDE_EPSILON)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// SetRenderSector
|
|
|
|
//
|
|
|
|
// Sets the render sector for each GL subsector so that the proper flat
|
|
|
|
// information can be retrieved
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
void P_SetRenderSector()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
DWORD j;
|
|
|
|
TArray<subsector_t *> undetermined;
|
|
|
|
subsector_t * ss;
|
|
|
|
|
2010-08-27 17:49:27 +00:00
|
|
|
#if 0 // doesn't work as expected :(
|
|
|
|
|
2010-08-27 15:20:05 +00:00
|
|
|
// hide all sectors on textured automap that only have hidden lines.
|
|
|
|
bool *hidesec = new bool[numsectors];
|
|
|
|
for(i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
hidesec[i] = true;
|
|
|
|
}
|
|
|
|
for(i = 0; i < numlines; i++)
|
|
|
|
{
|
|
|
|
if (!(lines[i].flags & ML_DONTDRAW))
|
|
|
|
{
|
|
|
|
hidesec[lines[i].frontsector - sectors] = false;
|
|
|
|
if (lines[i].backsector != NULL)
|
|
|
|
{
|
|
|
|
hidesec[lines[i].backsector - sectors] = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(i = 0; i < numsectors; i++)
|
|
|
|
{
|
|
|
|
if (hidesec[i]) sectors[i].MoreFlags |= SECF_HIDDEN;
|
|
|
|
}
|
|
|
|
delete [] hidesec;
|
2010-08-27 17:49:27 +00:00
|
|
|
#endif
|
2010-08-27 15:20:05 +00:00
|
|
|
|
|
|
|
// Check for incorrect partner seg info so that the following code does not crash.
|
2010-08-28 12:00:42 +00:00
|
|
|
if (glsegextras == NULL)
|
|
|
|
{
|
|
|
|
// This can be normal nodes, mistakenly identified as GL nodes so we must fill
|
|
|
|
// in the missing pieces differently.
|
|
|
|
|
|
|
|
for (i = 0; i < numsubsectors; i++)
|
|
|
|
{
|
|
|
|
ss = &subsectors[i];
|
|
|
|
ss->render_sector = ss->sector;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-27 15:20:05 +00:00
|
|
|
for(i=0;i<numsegs;i++)
|
|
|
|
{
|
|
|
|
int partner = (int)glsegextras[i].PartnerSeg;
|
|
|
|
|
|
|
|
if (partner<0 || partner>=numsegs/*eh? || &segs[partner]!=glsegextras[i].PartnerSeg*/)
|
|
|
|
{
|
|
|
|
glsegextras[i].PartnerSeg=DWORD_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
// glbsp creates such incorrect references for Strife.
|
|
|
|
if (segs[i].linedef && glsegextras[i].PartnerSeg != DWORD_MAX && !segs[glsegextras[i].PartnerSeg].linedef)
|
|
|
|
{
|
|
|
|
glsegextras[i].PartnerSeg = glsegextras[glsegextras[i].PartnerSeg].PartnerSeg = DWORD_MAX;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for(i=0;i<numsegs;i++)
|
|
|
|
{
|
2010-08-31 04:35:13 +00:00
|
|
|
if (glsegextras[i].PartnerSeg != DWORD_MAX && glsegextras[glsegextras[i].PartnerSeg].PartnerSeg!=(DWORD)i)
|
2010-08-27 15:20:05 +00:00
|
|
|
{
|
|
|
|
glsegextras[i].PartnerSeg=DWORD_MAX;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// look up sector number for each subsector
|
|
|
|
for (i = 0; i < numsubsectors; i++)
|
|
|
|
{
|
|
|
|
// For rendering pick the sector from the first seg that is a sector boundary
|
|
|
|
// this takes care of self-referencing sectors
|
|
|
|
ss = &subsectors[i];
|
|
|
|
seg_t *seg = ss->firstline;
|
|
|
|
|
|
|
|
// Check for one-dimensional subsectors. These should be ignored when
|
|
|
|
// being processed for automap drawinng etc.
|
|
|
|
ss->flags |= SSECF_DEGENERATE;
|
|
|
|
for(j=2; j<ss->numlines; j++)
|
|
|
|
{
|
|
|
|
if (!PointOnLine(seg[j].v1->x, seg[j].v1->y, seg->v1->x, seg->v1->y, seg->v2->x-seg->v1->x, seg->v2->y-seg->v1->y))
|
|
|
|
{
|
|
|
|
// Not on the same line
|
|
|
|
ss->flags &= ~SSECF_DEGENERATE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
seg = ss->firstline;
|
|
|
|
for(j=0; j<ss->numlines; j++)
|
|
|
|
{
|
|
|
|
if(seg->sidedef && (glsegextras[seg - segs].PartnerSeg == DWORD_MAX || seg->sidedef->sector!=segs[glsegextras[seg - segs].PartnerSeg].sidedef->sector))
|
|
|
|
{
|
|
|
|
ss->render_sector = seg->sidedef->sector;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
seg++;
|
|
|
|
}
|
|
|
|
if(ss->render_sector == NULL)
|
|
|
|
{
|
|
|
|
undetermined.Push(ss);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// assign a vaild render sector to all subsectors which haven't been processed yet.
|
|
|
|
while (undetermined.Size())
|
|
|
|
{
|
|
|
|
bool deleted=false;
|
|
|
|
for(i=undetermined.Size()-1;i>=0;i--)
|
|
|
|
{
|
|
|
|
ss=undetermined[i];
|
|
|
|
seg_t * seg = ss->firstline;
|
|
|
|
|
|
|
|
for(j=0; j<ss->numlines; j++)
|
|
|
|
{
|
|
|
|
DWORD partner = glsegextras[seg - segs].PartnerSeg;
|
|
|
|
if (partner != DWORD_MAX && glsegextras[partner].Subsector)
|
|
|
|
{
|
|
|
|
sector_t * backsec = glsegextras[partner].Subsector->render_sector;
|
|
|
|
if (backsec)
|
|
|
|
{
|
|
|
|
ss->render_sector=backsec;
|
|
|
|
undetermined.Delete(i);
|
|
|
|
deleted=1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
seg++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// We still got some left but the loop above was unable to assign them.
|
|
|
|
// This only happens when a subsector is off the map.
|
|
|
|
// Don't bother and just assign the real sector for rendering
|
|
|
|
if (!deleted && undetermined.Size())
|
|
|
|
{
|
|
|
|
for(i=undetermined.Size()-1;i>=0;i--)
|
|
|
|
{
|
|
|
|
ss=undetermined[i];
|
|
|
|
ss->render_sector=ss->sector;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0 // may be useful later so let's keep it here for now
|
|
|
|
// now group the subsectors by sector
|
|
|
|
subsector_t ** subsectorbuffer = new subsector_t * [numsubsectors];
|
|
|
|
|
|
|
|
for(i=0, ss=subsectors; i<numsubsectors; i++, ss++)
|
|
|
|
{
|
|
|
|
ss->render_sector->subsectorcount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
|
|
{
|
|
|
|
sectors[i].subsectors = subsectorbuffer;
|
|
|
|
subsectorbuffer += sectors[i].subsectorcount;
|
|
|
|
sectors[i].subsectorcount = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for(i=0, ss = subsectors; i<numsubsectors; i++, ss++)
|
|
|
|
{
|
|
|
|
ss->render_sector->subsectors[ss->render_sector->subsectorcount++]=ss;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|