ioq3quest/code/bspc/brushbsp.c

1871 lines
45 KiB
C

/*
===========================================================================
Copyright (C) 1999-2005 Id Software, Inc.
This file is part of Quake III Arena source code.
Quake III Arena source code is free software; you can redistribute it
and/or modify it under the terms of the GNU General Public License as
published by the Free Software Foundation; either version 2 of the License,
or (at your option) any later version.
Quake III Arena source code is distributed in the hope that it will be
useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Foobar; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
===========================================================================
*/
#include "qbsp.h"
#include "l_mem.h"
#include "../botlib/aasfile.h"
#include "aas_store.h"
#include "aas_cfg.h"
#include <assert.h>
/*
each side has a count of the other sides it splits
the best split will be the one that minimizes the total split counts
of all remaining sides
precalc side on plane table
evaluate split side
{
cost = 0
for all sides
for all sides
get
if side splits side and splitside is on same child
cost++;
}
*/
int c_nodes;
int c_nonvis;
int c_active_brushes;
int c_solidleafnodes;
int c_totalsides;
int c_brushmemory;
int c_peak_brushmemory;
int c_nodememory;
int c_peak_totalbspmemory;
// if a brush just barely pokes onto the other side,
// let it slide by without chopping
#define PLANESIDE_EPSILON 0.001
//0.1
//#ifdef DEBUG
typedef struct cname_s
{
int value;
char *name;
} cname_t;
cname_t contentnames[] =
{
{CONTENTS_SOLID,"CONTENTS_SOLID"},
{CONTENTS_WINDOW,"CONTENTS_WINDOW"},
{CONTENTS_AUX,"CONTENTS_AUX"},
{CONTENTS_LAVA,"CONTENTS_LAVA"},
{CONTENTS_SLIME,"CONTENTS_SLIME"},
{CONTENTS_WATER,"CONTENTS_WATER"},
{CONTENTS_MIST,"CONTENTS_MIST"},
{LAST_VISIBLE_CONTENTS,"LAST_VISIBLE_CONTENTS"},
{CONTENTS_AREAPORTAL,"CONTENTS_AREAPORTAL"},
{CONTENTS_PLAYERCLIP,"CONTENTS_PLAYERCLIP"},
{CONTENTS_MONSTERCLIP,"CONTENTS_MONSTERCLIP"},
{CONTENTS_CURRENT_0,"CONTENTS_CURRENT_0"},
{CONTENTS_CURRENT_90,"CONTENTS_CURRENT_90"},
{CONTENTS_CURRENT_180,"CONTENTS_CURRENT_180"},
{CONTENTS_CURRENT_270,"CONTENTS_CURRENT_270"},
{CONTENTS_CURRENT_UP,"CONTENTS_CURRENT_UP"},
{CONTENTS_CURRENT_DOWN,"CONTENTS_CURRENT_DOWN"},
{CONTENTS_ORIGIN,"CONTENTS_ORIGIN"},
{CONTENTS_MONSTER,"CONTENTS_MONSTER"},
{CONTENTS_DEADMONSTER,"CONTENTS_DEADMONSTER"},
{CONTENTS_DETAIL,"CONTENTS_DETAIL"},
{CONTENTS_Q2TRANSLUCENT,"CONTENTS_TRANSLUCENT"},
{CONTENTS_LADDER,"CONTENTS_LADDER"},
{0, 0}
};
void PrintContents(int contents)
{
int i;
for (i = 0; contentnames[i].value; i++)
{
if (contents & contentnames[i].value)
{
Log_Write("%s,", contentnames[i].name);
} //end if
} //end for
} //end of the function PrintContents
//#endif DEBUG
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void ResetBrushBSP(void)
{
c_nodes = 0;
c_nonvis = 0;
c_active_brushes = 0;
c_solidleafnodes = 0;
c_totalsides = 0;
c_brushmemory = 0;
c_peak_brushmemory = 0;
c_nodememory = 0;
c_peak_totalbspmemory = 0;
} //end of the function ResetBrushBSP
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void FindBrushInTree (node_t *node, int brushnum)
{
bspbrush_t *b;
if (node->planenum == PLANENUM_LEAF)
{
for (b=node->brushlist ; b ; b=b->next)
if (b->original->brushnum == brushnum)
Log_Print ("here\n");
return;
}
FindBrushInTree(node->children[0], brushnum);
FindBrushInTree(node->children[1], brushnum);
} //end of the function FindBrushInTree
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void DrawBrushList (bspbrush_t *brush, node_t *node)
{
int i;
side_t *s;
GLS_BeginScene ();
for ( ; brush ; brush=brush->next)
{
for (i=0 ; i<brush->numsides ; i++)
{
s = &brush->sides[i];
if (!s->winding)
continue;
if (s->texinfo == TEXINFO_NODE)
GLS_Winding (s->winding, 1);
else if (!(s->flags & SFL_VISIBLE))
GLS_Winding (s->winding, 2);
else
GLS_Winding (s->winding, 0);
}
}
GLS_EndScene ();
} //end of the function DrawBrushList
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void WriteBrushList (char *name, bspbrush_t *brush, qboolean onlyvis)
{
int i;
side_t *s;
FILE *f;
qprintf ("writing %s\n", name);
f = SafeOpenWrite (name);
for ( ; brush ; brush=brush->next)
{
for (i=0 ; i<brush->numsides ; i++)
{
s = &brush->sides[i];
if (!s->winding)
continue;
if (onlyvis && !(s->flags & SFL_VISIBLE))
continue;
OutputWinding (brush->sides[i].winding, f);
}
}
fclose (f);
} //end of the function WriteBrushList
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void PrintBrush (bspbrush_t *brush)
{
int i;
printf ("brush: %p\n", brush);
for (i=0;i<brush->numsides ; i++)
{
pw(brush->sides[i].winding);
printf ("\n");
} //end for
} //end of the function PrintBrush
//===========================================================================
// Sets the mins/maxs based on the windings
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void BoundBrush (bspbrush_t *brush)
{
int i, j;
winding_t *w;
ClearBounds (brush->mins, brush->maxs);
for (i=0 ; i<brush->numsides ; i++)
{
w = brush->sides[i].winding;
if (!w)
continue;
for (j=0 ; j<w->numpoints ; j++)
AddPointToBounds (w->p[j], brush->mins, brush->maxs);
}
} //end of the function BoundBrush
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void CreateBrushWindings (bspbrush_t *brush)
{
int i, j;
winding_t *w;
side_t *side;
plane_t *plane;
for (i=0 ; i<brush->numsides ; i++)
{
side = &brush->sides[i];
plane = &mapplanes[side->planenum];
w = BaseWindingForPlane (plane->normal, plane->dist);
for (j=0 ; j<brush->numsides && w; j++)
{
if (i == j)
continue;
if (brush->sides[j].flags & SFL_BEVEL)
continue;
plane = &mapplanes[brush->sides[j].planenum^1];
ChopWindingInPlace (&w, plane->normal, plane->dist, 0); //CLIP_EPSILON);
}
side->winding = w;
}
BoundBrush (brush);
} //end of the function CreateBrushWindings
//===========================================================================
// Creates a new axial brush
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
bspbrush_t *BrushFromBounds (vec3_t mins, vec3_t maxs)
{
bspbrush_t *b;
int i;
vec3_t normal;
vec_t dist;
b = AllocBrush (6);
b->numsides = 6;
for (i=0 ; i<3 ; i++)
{
VectorClear (normal);
normal[i] = 1;
dist = maxs[i];
b->sides[i].planenum = FindFloatPlane (normal, dist);
normal[i] = -1;
dist = -mins[i];
b->sides[3+i].planenum = FindFloatPlane (normal, dist);
}
CreateBrushWindings (b);
return b;
} //end of the function BrushFromBounds
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
int BrushOutOfBounds(bspbrush_t *brush, vec3_t mins, vec3_t maxs, float epsilon)
{
int i, j, n;
winding_t *w;
side_t *side;
for (i = 0; i < brush->numsides; i++)
{
side = &brush->sides[i];
w = side->winding;
for (j = 0; j < w->numpoints; j++)
{
for (n = 0; n < 3; n++)
{
if (w->p[j][n] < (mins[n] + epsilon) || w->p[j][n] > (maxs[n] - epsilon)) return true;
} //end for
} //end for
} //end for
return false;
} //end of the function BrushOutOfBounds
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
vec_t BrushVolume (bspbrush_t *brush)
{
int i;
winding_t *w;
vec3_t corner;
vec_t d, area, volume;
plane_t *plane;
if (!brush) return 0;
// grab the first valid point as the corner
w = NULL;
for (i = 0; i < brush->numsides; i++)
{
w = brush->sides[i].winding;
if (w) break;
} //end for
if (!w) return 0;
VectorCopy (w->p[0], corner);
// make tetrahedrons to all other faces
volume = 0;
for ( ; i < brush->numsides; i++)
{
w = brush->sides[i].winding;
if (!w) continue;
plane = &mapplanes[brush->sides[i].planenum];
d = -(DotProduct (corner, plane->normal) - plane->dist);
area = WindingArea(w);
volume += d * area;
} //end for
volume /= 3;
return volume;
} //end of the function BrushVolume
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
int CountBrushList (bspbrush_t *brushes)
{
int c;
c = 0;
for ( ; brushes; brushes = brushes->next) c++;
return c;
} //end of the function CountBrushList
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
node_t *AllocNode (void)
{
node_t *node;
node = GetMemory(sizeof(*node));
memset (node, 0, sizeof(*node));
if (numthreads == 1)
{
c_nodememory += MemorySize(node);
} //end if
return node;
} //end of the function AllocNode
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
bspbrush_t *AllocBrush (int numsides)
{
bspbrush_t *bb;
int c;
c = (int)&(((bspbrush_t *)0)->sides[numsides]);
bb = GetMemory(c);
memset (bb, 0, c);
if (numthreads == 1)
{
c_active_brushes++;
c_brushmemory += MemorySize(bb);
if (c_brushmemory > c_peak_brushmemory)
c_peak_brushmemory = c_brushmemory;
} //end if
return bb;
} //end of the function AllocBrush
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void FreeBrush (bspbrush_t *brushes)
{
int i;
for (i=0 ; i<brushes->numsides ; i++)
if (brushes->sides[i].winding)
FreeWinding(brushes->sides[i].winding);
if (numthreads == 1)
{
c_active_brushes--;
c_brushmemory -= MemorySize(brushes);
if (c_brushmemory < 0) c_brushmemory = 0;
} //end if
FreeMemory(brushes);
} //end of the function FreeBrush
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void FreeBrushList (bspbrush_t *brushes)
{
bspbrush_t *next;
for ( ; brushes; brushes = next)
{
next = brushes->next;
FreeBrush(brushes);
} //end for
} //end of the function FreeBrushList
//===========================================================================
// Duplicates the brush, the sides, and the windings
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
bspbrush_t *CopyBrush (bspbrush_t *brush)
{
bspbrush_t *newbrush;
int size;
int i;
size = (int)&(((bspbrush_t *)0)->sides[brush->numsides]);
newbrush = AllocBrush (brush->numsides);
memcpy (newbrush, brush, size);
for (i=0 ; i<brush->numsides ; i++)
{
if (brush->sides[i].winding)
newbrush->sides[i].winding = CopyWinding (brush->sides[i].winding);
}
return newbrush;
} //end of the function CopyBrush
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
node_t *PointInLeaf (node_t *node, vec3_t point)
{
vec_t d;
plane_t *plane;
while (node->planenum != PLANENUM_LEAF)
{
plane = &mapplanes[node->planenum];
d = DotProduct (point, plane->normal) - plane->dist;
if (d > 0)
node = node->children[0];
else
node = node->children[1];
}
return node;
} //end of the function PointInLeaf
//===========================================================================
// Returns PSIDE_FRONT, PSIDE_BACK, or PSIDE_BOTH
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
#if 0
int BoxOnPlaneSide (vec3_t mins, vec3_t maxs, plane_t *plane)
{
int side;
int i;
vec3_t corners[2];
vec_t dist1, dist2;
// axial planes are easy
if (plane->type < 3)
{
side = 0;
if (maxs[plane->type] > plane->dist+PLANESIDE_EPSILON)
side |= PSIDE_FRONT;
if (mins[plane->type] < plane->dist-PLANESIDE_EPSILON)
side |= PSIDE_BACK;
return side;
}
// create the proper leading and trailing verts for the box
for (i=0 ; i<3 ; i++)
{
if (plane->normal[i] < 0)
{
corners[0][i] = mins[i];
corners[1][i] = maxs[i];
}
else
{
corners[1][i] = mins[i];
corners[0][i] = maxs[i];
}
}
dist1 = DotProduct (plane->normal, corners[0]) - plane->dist;
dist2 = DotProduct (plane->normal, corners[1]) - plane->dist;
side = 0;
if (dist1 >= PLANESIDE_EPSILON)
side = PSIDE_FRONT;
if (dist2 < PLANESIDE_EPSILON)
side |= PSIDE_BACK;
return side;
}
#else
int BoxOnPlaneSide (vec3_t emins, vec3_t emaxs, plane_t *p)
{
float dist1, dist2;
int sides;
// axial planes are easy
if (p->type < 3)
{
sides = 0;
if (emaxs[p->type] > p->dist+PLANESIDE_EPSILON) sides |= PSIDE_FRONT;
if (emins[p->type] < p->dist-PLANESIDE_EPSILON) sides |= PSIDE_BACK;
return sides;
} //end if
// general case
switch (p->signbits)
{
case 0:
dist1 = p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2];
dist2 = p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2];
break;
case 1:
dist1 = p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2];
dist2 = p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2];
break;
case 2:
dist1 = p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2];
dist2 = p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2];
break;
case 3:
dist1 = p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2];
dist2 = p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2];
break;
case 4:
dist1 = p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2];
dist2 = p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2];
break;
case 5:
dist1 = p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emins[2];
dist2 = p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emaxs[2];
break;
case 6:
dist1 = p->normal[0]*emaxs[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2];
dist2 = p->normal[0]*emins[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2];
break;
case 7:
dist1 = p->normal[0]*emins[0] + p->normal[1]*emins[1] + p->normal[2]*emins[2];
dist2 = p->normal[0]*emaxs[0] + p->normal[1]*emaxs[1] + p->normal[2]*emaxs[2];
break;
default:
dist1 = dist2 = 0; // shut up compiler
// assert( 0 );
break;
}
sides = 0;
if (dist1 - p->dist >= PLANESIDE_EPSILON) sides = PSIDE_FRONT;
if (dist2 - p->dist < PLANESIDE_EPSILON) sides |= PSIDE_BACK;
// assert(sides != 0);
return sides;
}
#endif
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
int QuickTestBrushToPlanenum (bspbrush_t *brush, int planenum, int *numsplits)
{
int i, num;
plane_t *plane;
int s;
*numsplits = 0;
plane = &mapplanes[planenum];
#ifdef ME
//fast axial cases
if (plane->type < 3)
{
if (plane->dist + PLANESIDE_EPSILON < brush->mins[plane->type])
return PSIDE_FRONT;
if (plane->dist - PLANESIDE_EPSILON > brush->maxs[plane->type])
return PSIDE_BACK;
} //end if
#endif //ME*/
// if the brush actually uses the planenum,
// we can tell the side for sure
for (i = 0; i < brush->numsides; i++)
{
num = brush->sides[i].planenum;
if (num >= MAX_MAPFILE_PLANES)
Error ("bad planenum");
if (num == planenum)
return PSIDE_BACK|PSIDE_FACING;
if (num == (planenum ^ 1) )
return PSIDE_FRONT|PSIDE_FACING;
}
// box on plane side
s = BoxOnPlaneSide (brush->mins, brush->maxs, plane);
// if both sides, count the visible faces split
if (s == PSIDE_BOTH)
{
*numsplits += 3;
}
return s;
} //end of the function QuickTestBrushToPlanenum
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
int TestBrushToPlanenum (bspbrush_t *brush, int planenum,
int *numsplits, qboolean *hintsplit, int *epsilonbrush)
{
int i, j, num;
plane_t *plane;
int s = 0;
winding_t *w;
vec_t d, d_front, d_back;
int front, back;
int type;
float dist;
*numsplits = 0;
*hintsplit = false;
plane = &mapplanes[planenum];
#ifdef ME
//fast axial cases
type = plane->type;
if (type < 3)
{
dist = plane->dist;
if (dist + PLANESIDE_EPSILON < brush->mins[type]) return PSIDE_FRONT;
if (dist - PLANESIDE_EPSILON > brush->maxs[type]) return PSIDE_BACK;
if (brush->mins[type] < dist - PLANESIDE_EPSILON &&
brush->maxs[type] > dist + PLANESIDE_EPSILON) s = PSIDE_BOTH;
} //end if
if (s != PSIDE_BOTH)
#endif //ME
{
// if the brush actually uses the planenum,
// we can tell the side for sure
for (i = 0; i < brush->numsides; i++)
{
num = brush->sides[i].planenum;
if (num >= MAX_MAPFILE_PLANES) Error ("bad planenum");
if (num == planenum)
{
//we don't need to test this side plane again
brush->sides[i].flags |= SFL_TESTED;
return PSIDE_BACK|PSIDE_FACING;
} //end if
if (num == (planenum ^ 1) )
{
//we don't need to test this side plane again
brush->sides[i].flags |= SFL_TESTED;
return PSIDE_FRONT|PSIDE_FACING;
} //end if
} //end for
// box on plane side
s = BoxOnPlaneSide (brush->mins, brush->maxs, plane);
if (s != PSIDE_BOTH) return s;
} //end if
// if both sides, count the visible faces split
d_front = d_back = 0;
for (i = 0; i < brush->numsides; i++)
{
if (brush->sides[i].texinfo == TEXINFO_NODE)
continue; // on node, don't worry about splits
if (!(brush->sides[i].flags & SFL_VISIBLE))
continue; // we don't care about non-visible
w = brush->sides[i].winding;
if (!w) continue;
front = back = 0;
for (j = 0; j < w->numpoints; j++)
{
d = DotProduct(w->p[j], plane->normal) - plane->dist;
if (d > d_front) d_front = d;
if (d < d_back) d_back = d;
if (d > 0.1) // PLANESIDE_EPSILON)
front = 1;
if (d < -0.1) // PLANESIDE_EPSILON)
back = 1;
} //end for
if (front && back)
{
if ( !(brush->sides[i].surf & SURF_SKIP) )
{
(*numsplits)++;
if (brush->sides[i].surf & SURF_HINT)
{
*hintsplit = true;
} //end if
} //end if
} //end if
} //end for
if ( (d_front > 0.0 && d_front < 1.0)
|| (d_back < 0.0 && d_back > -1.0) )
(*epsilonbrush)++;
#if 0
if (*numsplits == 0)
{ // didn't really need to be split
if (front) s = PSIDE_FRONT;
else if (back) s = PSIDE_BACK;
else s = 0;
}
#endif
return s;
} //end of the function TestBrushToPlanenum
//===========================================================================
// Returns true if the winding would be crunched out of
// existance by the vertex snapping.
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
#define EDGE_LENGTH 0.2
qboolean WindingIsTiny (winding_t *w)
{
#if 0
if (WindingArea (w) < 1)
return true;
return false;
#else
int i, j;
vec_t len;
vec3_t delta;
int edges;
edges = 0;
for (i=0 ; i<w->numpoints ; i++)
{
j = i == w->numpoints - 1 ? 0 : i+1;
VectorSubtract (w->p[j], w->p[i], delta);
len = VectorLength (delta);
if (len > EDGE_LENGTH)
{
if (++edges == 3)
return false;
}
}
return true;
#endif
} //end of the function WindingIsTiny
//===========================================================================
// Returns true if the winding still has one of the points
// from basewinding for plane
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
qboolean WindingIsHuge (winding_t *w)
{
int i, j;
for (i=0 ; i<w->numpoints ; i++)
{
for (j=0 ; j<3 ; j++)
if (w->p[i][j] < -BOGUS_RANGE+1 || w->p[i][j] > BOGUS_RANGE-1)
return true;
}
return false;
} //end of the function WindingIsHuge
//===========================================================================
// creates a leaf out of the given nodes with the given brushes
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void LeafNode(node_t *node, bspbrush_t *brushes)
{
bspbrush_t *b;
int i;
node->side = NULL;
node->planenum = PLANENUM_LEAF;
node->contents = 0;
for (b = brushes; b; b = b->next)
{
// if the brush is solid and all of its sides are on nodes,
// it eats everything
if (b->original->contents & CONTENTS_SOLID)
{
for (i=0 ; i<b->numsides ; i++)
if (b->sides[i].texinfo != TEXINFO_NODE)
break;
if (i == b->numsides)
{
node->contents = CONTENTS_SOLID;
break;
} //end if
} //end if
node->contents |= b->original->contents;
} //end for
if (create_aas)
{
node->expansionbboxes = 0;
node->contents = 0;
for (b = brushes; b; b = b->next)
{
node->expansionbboxes |= b->original->expansionbbox;
node->contents |= b->original->contents;
if (b->original->modelnum)
node->modelnum = b->original->modelnum;
} //end for
if (node->contents & CONTENTS_SOLID)
{
if (node->expansionbboxes != cfg.allpresencetypes)
{
node->contents &= ~CONTENTS_SOLID;
} //end if
} //end if
} //end if
node->brushlist = brushes;
} //end of the function LeafNode
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void CheckPlaneAgainstParents (int pnum, node_t *node)
{
node_t *p;
for (p = node->parent; p; p = p->parent)
{
if (p->planenum == pnum) Error("Tried parent");
} //end for
} //end of the function CheckPlaneAgainstParants
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
qboolean CheckPlaneAgainstVolume (int pnum, node_t *node)
{
bspbrush_t *front, *back;
qboolean good;
SplitBrush (node->volume, pnum, &front, &back);
good = (front && back);
if (front) FreeBrush (front);
if (back) FreeBrush (back);
return good;
} //end of the function CheckPlaneAgaintsVolume
//===========================================================================
// Using a hueristic, choses one of the sides out of the brushlist
// to partition the brushes with.
// Returns NULL if there are no valid planes to split with..
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
side_t *SelectSplitSide (bspbrush_t *brushes, node_t *node)
{
int value, bestvalue;
bspbrush_t *brush, *test;
side_t *side, *bestside;
int i, pass, numpasses;
int pnum;
int s;
int front, back, both, facing, splits;
int bsplits;
int bestsplits;
int epsilonbrush;
qboolean hintsplit = false;
bestside = NULL;
bestvalue = -99999;
bestsplits = 0;
// the search order goes: visible-structural, visible-detail,
// nonvisible-structural, nonvisible-detail.
// If any valid plane is available in a pass, no further
// passes will be tried.
numpasses = 2;
for (pass = 0; pass < numpasses; pass++)
{
for (brush = brushes; brush; brush = brush->next)
{
// only check detail the second pass
// if ( (pass & 1) && !(brush->original->contents & CONTENTS_DETAIL) )
// continue;
// if ( !(pass & 1) && (brush->original->contents & CONTENTS_DETAIL) )
// continue;
for (i = 0; i < brush->numsides; i++)
{
side = brush->sides + i;
// if (side->flags & SFL_BEVEL)
// continue; // never use a bevel as a spliter
if (!side->winding)
continue; // nothing visible, so it can't split
if (side->texinfo == TEXINFO_NODE)
continue; // allready a node splitter
if (side->flags & SFL_TESTED)
continue; // we allready have metrics for this plane
// if (side->surf & SURF_SKIP)
// continue; // skip surfaces are never chosen
// if (!(side->flags & SFL_VISIBLE) && (pass < 2))
// continue; // only check visible faces on first pass
if ((side->flags & SFL_CURVE) && (pass < 1))
continue; // only check curves the second pass
pnum = side->planenum;
pnum &= ~1; // allways use positive facing plane
CheckPlaneAgainstParents (pnum, node);
if (!CheckPlaneAgainstVolume (pnum, node))
continue; // would produce a tiny volume
front = 0;
back = 0;
both = 0;
facing = 0;
splits = 0;
epsilonbrush = 0;
//inner loop: optimize
for (test = brushes; test; test = test->next)
{
s = TestBrushToPlanenum (test, pnum, &bsplits, &hintsplit, &epsilonbrush);
splits += bsplits;
// if (bsplits && (s&PSIDE_FACING) )
// Error ("PSIDE_FACING with splits");
test->testside = s;
//
if (s & PSIDE_FACING) facing++;
if (s & PSIDE_FRONT) front++;
if (s & PSIDE_BACK) back++;
if (s == PSIDE_BOTH) both++;
} //end for
// give a value estimate for using this plane
value = 5*facing - 5*splits - abs(front-back);
// value = -5*splits;
// value = 5*facing - 5*splits;
if (mapplanes[pnum].type < 3)
value+=5; // axial is better
value -= epsilonbrush * 1000; // avoid!
// never split a hint side except with another hint
if (hintsplit && !(side->surf & SURF_HINT) )
value = -9999999;
// save off the side test so we don't need
// to recalculate it when we actually seperate
// the brushes
if (value > bestvalue)
{
bestvalue = value;
bestside = side;
bestsplits = splits;
for (test = brushes; test ; test = test->next)
test->side = test->testside;
} //end if
} //end for
} //end for (brush = brushes;
// if we found a good plane, don't bother trying any
// other passes
if (bestside)
{
if (pass > 1)
{
if (numthreads == 1) c_nonvis++;
}
if (pass > 0) node->detail_seperator = true; // not needed for vis
break;
} //end if
} //end for (pass = 0;
//
// clear all the tested flags we set
//
for (brush = brushes ; brush ; brush=brush->next)
{
for (i = 0; i < brush->numsides; i++)
{
brush->sides[i].flags &= ~SFL_TESTED;
} //end for
} //end for
return bestside;
} //end of the function SelectSplitSide
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
int BrushMostlyOnSide (bspbrush_t *brush, plane_t *plane)
{
int i, j;
winding_t *w;
vec_t d, max;
int side;
max = 0;
side = PSIDE_FRONT;
for (i=0 ; i<brush->numsides ; i++)
{
w = brush->sides[i].winding;
if (!w)
continue;
for (j=0 ; j<w->numpoints ; j++)
{
d = DotProduct (w->p[j], plane->normal) - plane->dist;
if (d > max)
{
max = d;
side = PSIDE_FRONT;
}
if (-d > max)
{
max = -d;
side = PSIDE_BACK;
}
}
}
return side;
} //end of the function BrushMostlyOnSide
//===========================================================================
// Generates two new brushes, leaving the original
// unchanged
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void SplitBrush (bspbrush_t *brush, int planenum,
bspbrush_t **front, bspbrush_t **back)
{
bspbrush_t *b[2];
int i, j;
winding_t *w, *cw[2], *midwinding;
plane_t *plane, *plane2;
side_t *s, *cs;
float d, d_front, d_back;
*front = *back = NULL;
plane = &mapplanes[planenum];
// check all points
d_front = d_back = 0;
for (i=0 ; i<brush->numsides ; i++)
{
w = brush->sides[i].winding;
if (!w)
continue;
for (j=0 ; j<w->numpoints ; j++)
{
d = DotProduct (w->p[j], plane->normal) - plane->dist;
if (d > 0 && d > d_front)
d_front = d;
if (d < 0 && d < d_back)
d_back = d;
}
}
if (d_front < 0.2) // PLANESIDE_EPSILON)
{ // only on back
*back = CopyBrush (brush);
return;
}
if (d_back > -0.2) // PLANESIDE_EPSILON)
{ // only on front
*front = CopyBrush (brush);
return;
}
// create a new winding from the split plane
w = BaseWindingForPlane (plane->normal, plane->dist);
for (i=0 ; i<brush->numsides && w ; i++)
{
plane2 = &mapplanes[brush->sides[i].planenum ^ 1];
ChopWindingInPlace (&w, plane2->normal, plane2->dist, 0); // PLANESIDE_EPSILON);
}
if (!w || WindingIsTiny(w))
{ // the brush isn't really split
int side;
side = BrushMostlyOnSide (brush, plane);
if (side == PSIDE_FRONT)
*front = CopyBrush (brush);
if (side == PSIDE_BACK)
*back = CopyBrush (brush);
//free a possible winding
if (w) FreeWinding(w);
return;
}
if (WindingIsHuge (w))
{
Log_Write("WARNING: huge winding\n");
}
midwinding = w;
// split it for real
for (i=0 ; i<2 ; i++)
{
b[i] = AllocBrush (brush->numsides+1);
b[i]->original = brush->original;
}
// split all the current windings
for (i=0 ; i<brush->numsides ; i++)
{
s = &brush->sides[i];
w = s->winding;
if (!w)
continue;
ClipWindingEpsilon (w, plane->normal, plane->dist,
0 /*PLANESIDE_EPSILON*/, &cw[0], &cw[1]);
for (j=0 ; j<2 ; j++)
{
if (!cw[j])
continue;
#if 0
if (WindingIsTiny (cw[j]))
{
FreeWinding (cw[j]);
continue;
}
#endif
cs = &b[j]->sides[b[j]->numsides];
b[j]->numsides++;
*cs = *s;
// cs->planenum = s->planenum;
// cs->texinfo = s->texinfo;
// cs->original = s->original;
cs->winding = cw[j];
cs->flags &= ~SFL_TESTED;
}
}
// see if we have valid polygons on both sides
for (i=0 ; i<2 ; i++)
{
BoundBrush (b[i]);
for (j=0 ; j<3 ; j++)
{
if (b[i]->mins[j] < -MAX_MAP_BOUNDS || b[i]->maxs[j] > MAX_MAP_BOUNDS)
{
Log_Write("bogus brush after clip");
break;
}
}
if (b[i]->numsides < 3 || j < 3)
{
FreeBrush (b[i]);
b[i] = NULL;
}
}
if ( !(b[0] && b[1]) )
{
if (!b[0] && !b[1])
Log_Write("split removed brush\r\n");
else
Log_Write("split not on both sides\r\n");
if (b[0])
{
FreeBrush (b[0]);
*front = CopyBrush (brush);
}
if (b[1])
{
FreeBrush (b[1]);
*back = CopyBrush (brush);
}
return;
}
// add the midwinding to both sides
for (i=0 ; i<2 ; i++)
{
cs = &b[i]->sides[b[i]->numsides];
b[i]->numsides++;
cs->planenum = planenum^i^1;
cs->texinfo = TEXINFO_NODE; //never use these sides as splitters
cs->flags &= ~SFL_VISIBLE;
cs->flags &= ~SFL_TESTED;
if (i==0)
cs->winding = CopyWinding (midwinding);
else
cs->winding = midwinding;
}
{
vec_t v1;
int i;
for (i = 0; i < 2; i++)
{
v1 = BrushVolume (b[i]);
if (v1 < 1.0)
{
FreeBrush(b[i]);
b[i] = NULL;
//Log_Write("tiny volume after clip");
}
}
if (!b[0] && !b[1])
{
Log_Write("two tiny brushes\r\n");
} //end if
}
*front = b[0];
*back = b[1];
} //end of the function SplitBrush
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void SplitBrushList (bspbrush_t *brushes,
node_t *node, bspbrush_t **front, bspbrush_t **back)
{
bspbrush_t *brush, *newbrush, *newbrush2;
side_t *side;
int sides;
int i;
*front = *back = NULL;
for (brush = brushes; brush; brush = brush->next)
{
sides = brush->side;
if (sides == PSIDE_BOTH)
{ // split into two brushes
SplitBrush (brush, node->planenum, &newbrush, &newbrush2);
if (newbrush)
{
newbrush->next = *front;
*front = newbrush;
} //end if
if (newbrush2)
{
newbrush2->next = *back;
*back = newbrush2;
} //end if
continue;
} //end if
newbrush = CopyBrush (brush);
// if the planenum is actualy a part of the brush
// find the plane and flag it as used so it won't be tried
// as a splitter again
if (sides & PSIDE_FACING)
{
for (i=0 ; i<newbrush->numsides ; i++)
{
side = newbrush->sides + i;
if ( (side->planenum& ~1) == node->planenum)
side->texinfo = TEXINFO_NODE;
} //end for
} //end if
if (sides & PSIDE_FRONT)
{
newbrush->next = *front;
*front = newbrush;
continue;
} //end if
if (sides & PSIDE_BACK)
{
newbrush->next = *back;
*back = newbrush;
continue;
} //end if
} //end for
} //end of the function SplitBrushList
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void CheckBrushLists(bspbrush_t *brushlist1, bspbrush_t *brushlist2)
{
bspbrush_t *brush1, *brush2;
for (brush1 = brushlist1; brush1; brush1 = brush1->next)
{
for (brush2 = brushlist2; brush2; brush2 = brush2->next)
{
assert(brush1 != brush2);
} //end for
} //end for
} //end of the function CheckBrushLists
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
int numrecurse = 0;
node_t *BuildTree_r (node_t *node, bspbrush_t *brushes)
{
node_t *newnode;
side_t *bestside;
int i, totalmem;
bspbrush_t *children[2];
qprintf("\r%6d", numrecurse);
numrecurse++;
if (numthreads == 1)
{
totalmem = WindingMemory() + c_nodememory + c_brushmemory;
if (totalmem > c_peak_totalbspmemory)
c_peak_totalbspmemory = totalmem;
c_nodes++;
} //endif
if (drawflag)
DrawBrushList(brushes, node);
// find the best plane to use as a splitter
bestside = SelectSplitSide (brushes, node);
if (!bestside)
{
// leaf node
node->side = NULL;
node->planenum = -1;
LeafNode(node, brushes);
if (node->contents & CONTENTS_SOLID) c_solidleafnodes++;
if (create_aas)
{
//free up memory!!!
FreeBrushList(node->brushlist);
node->brushlist = NULL;
//free the node volume brush
if (node->volume)
{
FreeBrush(node->volume);
node->volume = NULL;
} //end if
} //end if
return node;
} //end if
// this is a splitplane node
node->side = bestside;
node->planenum = bestside->planenum & ~1; // always use front facing
//split the brush list in two for both children
SplitBrushList (brushes, node, &children[0], &children[1]);
//free the old brush list
FreeBrushList (brushes);
// allocate children before recursing
for (i = 0; i < 2; i++)
{
newnode = AllocNode ();
newnode->parent = node;
node->children[i] = newnode;
} //end for
//split the volume brush of the node for the children
SplitBrush (node->volume, node->planenum, &node->children[0]->volume,
&node->children[1]->volume);
if (create_aas)
{
//free the volume brush
if (node->volume)
{
FreeBrush(node->volume);
node->volume = NULL;
} //end if
} //end if
// recursively process children
for (i = 0; i < 2; i++)
{
node->children[i] = BuildTree_r(node->children[i], children[i]);
} //end for
return node;
} //end of the function BuildTree_r
//===========================================================================
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
node_t *firstnode; //first node in the list
node_t *lastnode; //last node in the list
int nodelistsize; //number of nodes in the list
int use_nodequeue = 0; //use nodequeue, otherwise a node stack is used
int numwaiting = 0;
void (*AddNodeToList)(node_t *node);
//add the node to the front of the node list
//(effectively using a node stack)
void AddNodeToStack(node_t *node)
{
ThreadLock();
node->next = firstnode;
firstnode = node;
if (!lastnode) lastnode = node;
nodelistsize++;
ThreadUnlock();
//
ThreadSemaphoreIncrease(1);
} //end of the function AddNodeToStack
//add the node to the end of the node list
//(effectively using a node queue)
void AddNodeToQueue(node_t *node)
{
ThreadLock();
node->next = NULL;
if (lastnode) lastnode->next = node;
else firstnode = node;
lastnode = node;
nodelistsize++;
ThreadUnlock();
//
ThreadSemaphoreIncrease(1);
} //end of the function AddNodeToQueue
//get the first node from the front of the node list
node_t *NextNodeFromList(void)
{
node_t *node;
ThreadLock();
numwaiting++;
if (!firstnode)
{
if (numwaiting >= GetNumThreads()) ThreadSemaphoreIncrease(GetNumThreads());
} //end if
ThreadUnlock();
ThreadSemaphoreWait();
ThreadLock();
numwaiting--;
node = firstnode;
if (firstnode)
{
firstnode = firstnode->next;
nodelistsize--;
} //end if
if (!firstnode) lastnode = NULL;
ThreadUnlock();
return node;
} //end of the function NextNodeFromList
//returns the size of the node list
int NodeListSize(void)
{
int size;
ThreadLock();
size = nodelistsize;
ThreadUnlock();
return size;
} //end of the function NodeListSize
//
void IncreaseNodeCounter(void)
{
ThreadLock();
//if (verbose) printf("\r%6d", numrecurse++);
qprintf("\r%6d", numrecurse++);
//qprintf("\r%6d %d, %5d ", numrecurse++, GetNumThreads(), nodelistsize);
ThreadUnlock();
} //end of the function IncreaseNodeCounter
//thread function, gets nodes from the nodelist and processes them
void BuildTreeThread(int threadid)
{
node_t *newnode, *node;
side_t *bestside;
int i, totalmem;
bspbrush_t *brushes;
for (node = NextNodeFromList(); node; )
{
//if the nodelist isn't empty try to add another thread
//if (NodeListSize() > 10) AddThread(BuildTreeThread);
//display the number of nodes processed so far
if (numthreads == 1)
IncreaseNodeCounter();
brushes = node->brushlist;
if (numthreads == 1)
{
totalmem = WindingMemory() + c_nodememory + c_brushmemory;
if (totalmem > c_peak_totalbspmemory)
{
c_peak_totalbspmemory = totalmem;
} //end if
c_nodes++;
} //endif
if (drawflag)
{
DrawBrushList(brushes, node);
} //end if
if (cancelconversion)
{
bestside = NULL;
} //end if
else
{
// find the best plane to use as a splitter
bestside = SelectSplitSide(brushes, node);
} //end else
//if there's no split side left
if (!bestside)
{
//create a leaf out of the node
LeafNode(node, brushes);
if (node->contents & CONTENTS_SOLID) c_solidleafnodes++;
if (create_aas)
{
//free up memory!!!
FreeBrushList(node->brushlist);
node->brushlist = NULL;
} //end if
//free the node volume brush (it is not used anymore)
if (node->volume)
{
FreeBrush(node->volume);
node->volume = NULL;
} //end if
node = NextNodeFromList();
continue;
} //end if
// this is a splitplane node
node->side = bestside;
node->planenum = bestside->planenum & ~1; //always use front facing
//allocate children
for (i = 0; i < 2; i++)
{
newnode = AllocNode();
newnode->parent = node;
node->children[i] = newnode;
} //end for
//split the brush list in two for both children
SplitBrushList(brushes, node, &node->children[0]->brushlist, &node->children[1]->brushlist);
CheckBrushLists(node->children[0]->brushlist, node->children[1]->brushlist);
//free the old brush list
FreeBrushList(brushes);
node->brushlist = NULL;
//split the volume brush of the node for the children
SplitBrush(node->volume, node->planenum, &node->children[0]->volume,
&node->children[1]->volume);
if (!node->children[0]->volume || !node->children[1]->volume)
{
Error("child without volume brush");
} //end if
//free the volume brush
if (node->volume)
{
FreeBrush(node->volume);
node->volume = NULL;
} //end if
//add both children to the node list
//AddNodeToList(node->children[0]);
AddNodeToList(node->children[1]);
node = node->children[0];
} //end while
RemoveThread(threadid);
} //end of the function BuildTreeThread
//===========================================================================
// build the bsp tree using a node list
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
void BuildTree(tree_t *tree)
{
int i;
firstnode = NULL;
lastnode = NULL;
//use a node queue or node stack
if (use_nodequeue) AddNodeToList = AddNodeToQueue;
else AddNodeToList = AddNodeToStack;
//setup thread locking
ThreadSetupLock();
ThreadSetupSemaphore();
numwaiting = 0;
//
Log_Print("%6d threads max\n", numthreads);
if (use_nodequeue) Log_Print("breadth first bsp building\n");
else Log_Print("depth first bsp building\n");
qprintf("%6d splits", 0);
//add the first node to the list
AddNodeToList(tree->headnode);
//start the threads
for (i = 0; i < numthreads; i++)
AddThread(BuildTreeThread);
//wait for all added threads to be finished
WaitForAllThreadsFinished();
//shutdown the thread locking
ThreadShutdownLock();
ThreadShutdownSemaphore();
} //end of the function BuildTree
//===========================================================================
// The incoming brush list will be freed before exiting
//
// Parameter: -
// Returns: -
// Changes Globals: -
//===========================================================================
tree_t *BrushBSP(bspbrush_t *brushlist, vec3_t mins, vec3_t maxs)
{
int i, c_faces, c_nonvisfaces, c_brushes;
bspbrush_t *b;
node_t *node;
tree_t *tree;
vec_t volume;
// vec3_t point;
Log_Print("-------- Brush BSP ---------\n");
tree = Tree_Alloc();
c_faces = 0;
c_nonvisfaces = 0;
c_brushes = 0;
c_totalsides = 0;
for (b = brushlist; b; b = b->next)
{
c_brushes++;
volume = BrushVolume(b);
if (volume < microvolume)
{
Log_Print("WARNING: entity %i, brush %i: microbrush\n",
b->original->entitynum, b->original->brushnum);
} //end if
for (i=0 ; i<b->numsides ; i++)
{
if (b->sides[i].flags & SFL_BEVEL)
continue;
if (!b->sides[i].winding)
continue;
if (b->sides[i].texinfo == TEXINFO_NODE)
continue;
if (b->sides[i].flags & SFL_VISIBLE)
{
c_faces++;
} //end if
else
{
c_nonvisfaces++;
//if (create_aas) b->sides[i].texinfo = TEXINFO_NODE;
} //end if
} //end for
c_totalsides += b->numsides;
AddPointToBounds (b->mins, tree->mins, tree->maxs);
AddPointToBounds (b->maxs, tree->mins, tree->maxs);
} //end for
Log_Print("%6i brushes\n", c_brushes);
Log_Print("%6i visible faces\n", c_faces);
Log_Print("%6i nonvisible faces\n", c_nonvisfaces);
Log_Print("%6i total sides\n", c_totalsides);
c_active_brushes = c_brushes;
c_nodememory = 0;
c_brushmemory = 0;
c_peak_brushmemory = 0;
c_nodes = 0;
c_nonvis = 0;
node = AllocNode ();
//volume of first node (head node)
node->volume = BrushFromBounds (mins, maxs);
//
tree->headnode = node;
//just get some statistics and the mins/maxs of the node
numrecurse = 0;
// qprintf("%6d splits", numrecurse);
tree->headnode->brushlist = brushlist;
BuildTree(tree);
//build the bsp tree with the start node from the brushlist
// node = BuildTree_r(node, brushlist);
//if the conversion is cancelled
if (cancelconversion) return tree;
qprintf("\n");
Log_Write("%6d splits\r\n", numrecurse);
// Log_Print("%6i visible nodes\n", c_nodes/2 - c_nonvis);
// Log_Print("%6i nonvis nodes\n", c_nonvis);
// Log_Print("%6i leaves\n", (c_nodes+1)/2);
// Log_Print("%6i solid leaf nodes\n", c_solidleafnodes);
// Log_Print("%6i active brushes\n", c_active_brushes);
if (numthreads == 1)
{
// Log_Print("%6i KB of node memory\n", c_nodememory >> 10);
// Log_Print("%6i KB of brush memory\n", c_brushmemory >> 10);
// Log_Print("%6i KB of peak brush memory\n", c_peak_brushmemory >> 10);
// Log_Print("%6i KB of winding memory\n", WindingMemory() >> 10);
// Log_Print("%6i KB of peak winding memory\n", WindingPeakMemory() >> 10);
Log_Print("%6i KB of peak total bsp memory\n", c_peak_totalbspmemory >> 10);
} //end if
/*
point[0] = 1485;
point[1] = 956.125;
point[2] = 352.125;
node = PointInLeaf(tree->headnode, point);
if (node->planenum != PLANENUM_LEAF)
{
Log_Print("node not a leaf\n");
} //end if
Log_Print("at %f %f %f:\n", point[0], point[1], point[2]);
PrintContents(node->contents);
Log_Print("node->expansionbboxes = %d\n", node->expansionbboxes);
//*/
return tree;
} //end of the function BrushBSP