mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2025-01-22 08:51:13 +00:00
fcd1897421
Also, make doxygen extract static functions as well, so qfbsp's docs can be more complete.
390 lines
8.7 KiB
C
390 lines
8.7 KiB
C
/*
|
|
Copyright (C) 1996-1997 Id Software, Inc.
|
|
|
|
This program 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.
|
|
|
|
This program 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 this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
See file, 'COPYING', for details.
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
static __attribute__ ((used)) const char rcsid[] =
|
|
"$Id$";
|
|
|
|
#ifdef HAVE_STRING_H
|
|
# include <string.h>
|
|
#endif
|
|
|
|
#include "QF/sys.h"
|
|
|
|
#include "brush.h"
|
|
#include "bsp5.h"
|
|
#include "csg4.h"
|
|
#include "draw.h"
|
|
#include "merge.h"
|
|
#include "solidbsp.h"
|
|
#include "surfaces.h"
|
|
#include "winding.h"
|
|
|
|
/** \addtogroup qfbsp_csg4
|
|
*/
|
|
//@{
|
|
|
|
/*
|
|
NOTES
|
|
|
|
Brushes that touch still need to be split at the cut point to make a
|
|
tjunction
|
|
*/
|
|
|
|
face_t *validfaces[MAX_MAP_PLANES];
|
|
face_t *inside, *outside;
|
|
int brushfaces;
|
|
int csgfaces;
|
|
int csgmergefaces;
|
|
|
|
face_t *
|
|
NewFaceFromFace (const face_t *in)
|
|
{
|
|
face_t *newf;
|
|
|
|
newf = AllocFace ();
|
|
|
|
newf->planenum = in->planenum;
|
|
newf->texturenum = in->texturenum;
|
|
newf->planeside = in->planeside;
|
|
newf->original = in->original;
|
|
newf->contents[0] = in->contents[0];
|
|
newf->contents[1] = in->contents[1];
|
|
newf->detail = in->detail;
|
|
|
|
return newf;
|
|
}
|
|
|
|
void
|
|
SplitFace (face_t *in, plane_t *split, face_t **front, face_t **back)
|
|
{
|
|
int i;
|
|
int counts[3];
|
|
plane_t plane;
|
|
vec_t dot;
|
|
winding_t *tmp;
|
|
|
|
if (in->points->numpoints < 0)
|
|
Sys_Error ("SplitFace: freed face");
|
|
counts[0] = counts[1] = counts[2] = 0;
|
|
|
|
// determine sides for each point
|
|
for (i = 0; i < in->points->numpoints; i++) {
|
|
dot = DotProduct (in->points->points[i], split->normal) - split->dist;
|
|
if (dot > ON_EPSILON)
|
|
counts[SIDE_FRONT]++;
|
|
else if (dot < -ON_EPSILON)
|
|
counts[SIDE_BACK]++;
|
|
}
|
|
|
|
if (!counts[SIDE_FRONT]) {
|
|
*front = NULL;
|
|
*back = in;
|
|
return;
|
|
}
|
|
if (!counts[SIDE_BACK]) {
|
|
*front = in;
|
|
*back = NULL;
|
|
return;
|
|
}
|
|
|
|
*back = NewFaceFromFace (in);
|
|
*front = NewFaceFromFace (in);
|
|
|
|
tmp = CopyWinding (in->points);
|
|
(*front)->points = ClipWinding (tmp, split, 0);
|
|
|
|
plane.dist = -split->dist;
|
|
VectorNegate (split->normal, plane.normal);
|
|
(*back)->points = ClipWinding (in->points, &plane, 0);
|
|
|
|
in->points = 0; // freed by ClipWinding
|
|
FreeFace (in);
|
|
}
|
|
|
|
/** Clips all of the faces in the ::inside list.
|
|
|
|
Faces will be moved to the ::outside list or split into a piece in each
|
|
list. Faces exactly on the plane will stay inside unless overdrawn by
|
|
a later brush
|
|
|
|
\param splitplane Index of the plane by which faces will be clipped.
|
|
\param frontside The side of the plane that holds the outside list.
|
|
\param precedence XXX
|
|
*/
|
|
static void
|
|
ClipInside (int splitplane, int frontside, qboolean precedence)
|
|
{
|
|
face_t *insidelist, *next, *f;
|
|
face_t *frags[2];
|
|
plane_t *split;
|
|
|
|
split = &planes[splitplane];
|
|
|
|
insidelist = NULL;
|
|
for (f = inside; f; f = next) {
|
|
next = f->next; // f->next will get mashed by SplitFace
|
|
|
|
if (f->planenum == splitplane) {
|
|
// exactly on, handle special
|
|
|
|
// always clip off opposite facing
|
|
if (frontside != f->planeside || precedence) {
|
|
frags[frontside] = NULL;
|
|
frags[!frontside] = f;
|
|
} else {
|
|
// leave it on the outside
|
|
frags[frontside] = f;
|
|
frags[!frontside] = NULL;
|
|
}
|
|
} else {
|
|
// proper split
|
|
SplitFace (f, split, &frags[0], &frags[1]);
|
|
}
|
|
|
|
if (frags[frontside]) {
|
|
frags[frontside]->next = outside;
|
|
outside = frags[frontside];
|
|
}
|
|
if (frags[!frontside]) {
|
|
frags[!frontside]->next = insidelist;
|
|
insidelist = frags[!frontside];
|
|
}
|
|
}
|
|
|
|
inside = insidelist;
|
|
}
|
|
|
|
/** Saves all of the faces in the ::outside list to the bsp plane list
|
|
(::validfaces).
|
|
|
|
\param mirror If true, add extra faces that face the opposite direction.
|
|
*/
|
|
static void
|
|
SaveOutside (qboolean mirror)
|
|
{
|
|
face_t *f, *next, *newf;
|
|
int planenum;
|
|
|
|
for (f = outside; f; f = next) {
|
|
next = f->next;
|
|
csgfaces++;
|
|
Draw_DrawFace (f);
|
|
planenum = f->planenum;
|
|
|
|
if (mirror) {
|
|
newf = NewFaceFromFace (f);
|
|
|
|
newf->points = CopyWindingReverse (f->points);
|
|
newf->planeside = f->planeside ^ 1; // reverse side
|
|
newf->contents[0] = f->contents[1];
|
|
newf->contents[1] = f->contents[0];
|
|
|
|
validfaces[planenum] = MergeFaceToList (newf,
|
|
validfaces[planenum]);
|
|
}
|
|
|
|
validfaces[planenum] = MergeFaceToList (f, validfaces[planenum]);
|
|
validfaces[planenum] = FreeMergeListScraps (validfaces[planenum]);
|
|
}
|
|
}
|
|
|
|
/** Free the faces that are inside the clipping brush.
|
|
|
|
If the clipping brush is non-solid, then the faces will be moved to
|
|
::outside rather than being freed, thus allowing the faces to continue
|
|
to exist.
|
|
|
|
\param contents The contents of the clipping brush.
|
|
*/
|
|
static void
|
|
FreeInside (int contents)
|
|
{
|
|
face_t *next, *f;
|
|
|
|
for (f = inside; f; f = next) {
|
|
next = f->next;
|
|
|
|
if (contents != CONTENTS_SOLID) {
|
|
f->contents[0] = contents;
|
|
f->next = outside;
|
|
outside = f;
|
|
} else
|
|
FreeFace (f);
|
|
}
|
|
}
|
|
|
|
surface_t *
|
|
BuildSurfaces (void)
|
|
{
|
|
face_t *count;
|
|
face_t **f;
|
|
int i;
|
|
surface_t *surfhead, *s;
|
|
|
|
surfhead = NULL;
|
|
|
|
f = validfaces;
|
|
for (i = 0; i < numbrushplanes; i++, f++) {
|
|
if (!*f)
|
|
continue; // nothing left on this plane
|
|
|
|
// create a new surface to hold the faces on this plane
|
|
s = AllocSurface ();
|
|
s->planenum = i;
|
|
s->next = surfhead;
|
|
surfhead = s;
|
|
s->faces = *f;
|
|
for (count = s->faces; count; count = count->next) {
|
|
csgmergefaces++;
|
|
if (count->detail)
|
|
s->has_detail = 1;
|
|
else
|
|
s->has_struct = 1;
|
|
}
|
|
CalcSurfaceInfo (s); // bounding box and flags
|
|
}
|
|
|
|
return surfhead;
|
|
}
|
|
|
|
/** Create faces using the faces of the provided brush.
|
|
|
|
The ::outside list will be set to the list of created faces. The faces
|
|
will be such that their front is empty and their back is the contents of
|
|
the brush.
|
|
|
|
\param b The brush from which to create the faces.
|
|
*/
|
|
static void
|
|
CopyFacesToOutside (brush_t *b)
|
|
{
|
|
face_t *newf, *f;
|
|
|
|
outside = NULL;
|
|
|
|
for (f = b->faces; f; f = f->next) {
|
|
if (f->texturenum == TEX_SKIP)
|
|
continue;
|
|
|
|
brushfaces++;
|
|
newf = AllocFace ();
|
|
*newf = *f;
|
|
newf->points = CopyWinding (f->points);
|
|
newf->next = outside;
|
|
newf->contents[0] = CONTENTS_EMPTY;
|
|
newf->contents[1] = b->contents;
|
|
outside = newf;
|
|
}
|
|
}
|
|
|
|
surface_t *
|
|
CSGFaces (brushset_t *bs)
|
|
{
|
|
brush_t *b1, *b2;
|
|
face_t *f;
|
|
int i;
|
|
qboolean overwrite;
|
|
surface_t *surfhead;
|
|
|
|
qprintf ("---- CSGFaces ----\n");
|
|
|
|
memset (validfaces, 0, sizeof (validfaces));
|
|
|
|
csgfaces = brushfaces = csgmergefaces = 0;
|
|
|
|
Draw_ClearWindow ();
|
|
|
|
// do the solid faces
|
|
for (b1 = bs->brushes; b1; b1 = b1->next) {
|
|
// set outside to a copy of the brush's faces
|
|
CopyFacesToOutside (b1);
|
|
|
|
if (b1->faces->texturenum < 0) {
|
|
// Don't split HINT and SKIP brushes.
|
|
SaveOutside (false);
|
|
continue;
|
|
}
|
|
|
|
// earlier brushes do NOT overwrite
|
|
overwrite = false;
|
|
|
|
for (b2 = bs->brushes; b2; b2 = b2->next) {
|
|
// see if b2 needs to clip a chunk out of b1
|
|
if (b2->faces->texturenum < 0)
|
|
continue;
|
|
|
|
if (b1 == b2) {
|
|
// later brushes DO overwrite
|
|
overwrite = true;
|
|
continue;
|
|
}
|
|
// check bounding box first
|
|
for (i = 0; i < 3; i++)
|
|
if (b1->mins[i] > b2->maxs[i] || b1->maxs[i] < b2->mins[i])
|
|
break;
|
|
if (i < 3)
|
|
continue;
|
|
|
|
// divide faces by the planes of the new brush
|
|
|
|
inside = outside;
|
|
outside = NULL;
|
|
|
|
for (f = b2->faces; f; f = f->next)
|
|
ClipInside (f->planenum, f->planeside, overwrite);
|
|
|
|
// these faces are contained in another brush, so get rid of them
|
|
if (b1->contents == CONTENTS_SOLID
|
|
&& b2->contents <= CONTENTS_WATER) {
|
|
// Faces from a solid brush are allowed to exist inside a
|
|
// non-solid brush. This forms an intrusion into the clipping
|
|
// brush.
|
|
FreeInside (b2->contents);
|
|
} else {
|
|
// Unconditionally treat the clipping brush as solid to force
|
|
// contents merging, or to make an empty clipping brush clip
|
|
// away parts of other brushes.
|
|
FreeInside (CONTENTS_SOLID);
|
|
}
|
|
}
|
|
|
|
// all of the faces left in outside are real surface faces
|
|
// if the brush is not solid, mirror the faces for the inside view
|
|
SaveOutside (b1->contents != CONTENTS_SOLID);
|
|
}
|
|
|
|
#if 0
|
|
if (!csgfaces)
|
|
Sys_Error ("No faces");
|
|
#endif
|
|
|
|
surfhead = BuildSurfaces ();
|
|
|
|
qprintf ("%5i brushfaces\n", brushfaces);
|
|
qprintf ("%5i csgfaces\n", csgfaces);
|
|
qprintf ("%5i mergedfaces\n", csgmergefaces);
|
|
|
|
return surfhead;
|
|
}
|
|
|
|
//@}
|