2002-08-25 23:06:23 +00:00
|
|
|
/*
|
|
|
|
flow.c
|
|
|
|
|
|
|
|
PVS PHS generator tool
|
|
|
|
|
2002-09-25 01:51:58 +00:00
|
|
|
Copyright (C) 1996-1997 Id Software, Inc.
|
2002-08-25 23:06:23 +00:00
|
|
|
Copyright (C) 2002 Colin Thompson
|
|
|
|
|
|
|
|
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:
|
|
|
|
|
|
|
|
Free Software Foundation, Inc.
|
|
|
|
59 Temple Place - Suite 330
|
2013-03-07 05:31:00 +00:00
|
|
|
Boston, MA 02111-1307, USA
|
2002-08-25 23:06:23 +00:00
|
|
|
|
|
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <getopt.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
# include <unistd.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_IO_H
|
|
|
|
# include <io.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_STRING_H
|
|
|
|
# include <string.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_STRINGS_H
|
|
|
|
# include <strings.h>
|
|
|
|
#endif
|
|
|
|
#include <getopt.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2013-03-08 13:20:29 +00:00
|
|
|
#include "QF/alloc.h"
|
2002-08-25 23:06:23 +00:00
|
|
|
#include "QF/bspfile.h"
|
2002-08-27 07:16:28 +00:00
|
|
|
#include "QF/cmd.h"
|
2002-08-25 23:06:23 +00:00
|
|
|
#include "QF/mathlib.h"
|
2002-08-27 07:16:28 +00:00
|
|
|
#include "QF/quakefs.h"
|
|
|
|
#include "QF/sys.h"
|
|
|
|
|
2002-08-25 23:06:23 +00:00
|
|
|
#include "vis.h"
|
|
|
|
#include "options.h"
|
|
|
|
|
2003-09-24 00:59:53 +00:00
|
|
|
static int
|
2003-02-04 23:26:26 +00:00
|
|
|
CheckStack (cluster_t *cluster, threaddata_t *thread)
|
2002-08-25 23:06:23 +00:00
|
|
|
{
|
2013-03-07 05:31:00 +00:00
|
|
|
pstack_t *portal;
|
2012-05-21 23:23:22 +00:00
|
|
|
|
2013-03-07 05:31:00 +00:00
|
|
|
for (portal = thread->pstack_head.next; portal; portal = portal->next)
|
2003-09-24 00:59:53 +00:00
|
|
|
if (portal->cluster == cluster) {
|
|
|
|
printf ("CheckStack: cluster recursion\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
|
|
|
|
2013-03-08 13:20:29 +00:00
|
|
|
static sep_t *
|
|
|
|
new_separator (threaddata_t *thread)
|
|
|
|
{
|
|
|
|
sep_t *sep;
|
|
|
|
|
|
|
|
ALLOC (128, sep_t, thread->sep, sep);
|
|
|
|
return sep;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
delete_separator (threaddata_t *thread, sep_t *sep)
|
|
|
|
{
|
|
|
|
FREE (thread->sep, sep);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-03-15 07:00:39 +00:00
|
|
|
free_separators (threaddata_t *thread, sep_t *sep_list)
|
2013-03-08 13:20:29 +00:00
|
|
|
{
|
2013-03-15 07:00:39 +00:00
|
|
|
while (sep_list) {
|
|
|
|
sep_t *sep = sep_list;
|
|
|
|
sep_list = sep->next;
|
|
|
|
delete_separator (thread, sep);
|
2013-03-08 13:20:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-08-25 23:06:23 +00:00
|
|
|
/*
|
2013-03-15 07:00:39 +00:00
|
|
|
Find the planes separating source from pass. The planes form a double
|
|
|
|
pyramid with source as the base (ie, source's edges will all be in one
|
|
|
|
plane each) and the vertex of the pyramid is between source and pass.
|
|
|
|
Edges from pass may or may not be in a plane, but each vertex will be in
|
|
|
|
at least one plane.
|
|
|
|
|
|
|
|
If flip is false, the planes will be such that the space enclosed by the
|
|
|
|
planes and on the pass side of the vertex are on the front sides of the
|
|
|
|
planes. If flip is true, then the space on the source side of the vertex
|
|
|
|
and enclosed by the planes is on the front side of the planes.
|
2002-08-25 23:06:23 +00:00
|
|
|
*/
|
2013-03-15 07:00:39 +00:00
|
|
|
static sep_t *
|
|
|
|
FindSeparators (threaddata_t *thread, winding_t *source, winding_t *pass,
|
|
|
|
int flip)
|
2002-08-25 23:06:23 +00:00
|
|
|
{
|
2013-03-07 05:31:00 +00:00
|
|
|
float d;
|
|
|
|
int i, j, k, l;
|
|
|
|
int counts[3];
|
|
|
|
qboolean fliptest;
|
|
|
|
plane_t plane;
|
|
|
|
vec3_t v1, v2;
|
|
|
|
vec_t length;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2013-03-15 07:00:39 +00:00
|
|
|
sep_t *separators = 0;
|
2013-03-08 13:20:29 +00:00
|
|
|
|
2013-03-07 05:31:00 +00:00
|
|
|
for (i = 0; i < source->numpoints; i++) {
|
2002-08-25 23:06:23 +00:00
|
|
|
l = (i + 1) % source->numpoints;
|
|
|
|
VectorSubtract (source->points[l], source->points[i], v1);
|
|
|
|
|
2002-09-19 02:37:52 +00:00
|
|
|
// find a vertex of pass that makes a plane that puts all of the
|
2002-08-25 23:06:23 +00:00
|
|
|
// vertexes of pass on the front side and all of the vertexes of
|
|
|
|
// source on the back side
|
|
|
|
for (j = 0; j < pass->numpoints; j++) {
|
|
|
|
VectorSubtract (pass->points[j], source->points[i], v2);
|
|
|
|
|
2002-09-26 03:58:22 +00:00
|
|
|
plane.normal[0] = v1[1] * v2[2] - v1[2] * v2[1];
|
|
|
|
plane.normal[1] = v1[2] * v2[0] - v1[0] * v2[2];
|
|
|
|
plane.normal[2] = v1[0] * v2[1] - v1[1] * v2[0];
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2002-09-24 21:48:47 +00:00
|
|
|
length = DotProduct (plane.normal, plane.normal);
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2002-09-24 21:48:47 +00:00
|
|
|
// if points don't make a valid plane, skip it
|
2002-08-25 23:06:23 +00:00
|
|
|
if (length < ON_EPSILON)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
length = 1 / sqrt (length);
|
2002-09-24 21:48:47 +00:00
|
|
|
VectorScale (plane.normal, length, plane.normal);
|
2002-08-25 23:06:23 +00:00
|
|
|
|
|
|
|
plane.dist = DotProduct (pass->points[j], plane.normal);
|
|
|
|
|
|
|
|
// find out which side of the generated seperating plane has the
|
|
|
|
// source portal
|
|
|
|
fliptest = false;
|
|
|
|
for (k = 0; k < source->numpoints; k++) {
|
|
|
|
if (k == i || k == l)
|
|
|
|
continue;
|
|
|
|
d = DotProduct (source->points[k], plane.normal) - plane.dist;
|
2012-05-21 23:23:22 +00:00
|
|
|
if (d < -ON_EPSILON) {
|
2002-08-25 23:06:23 +00:00
|
|
|
// source is on the negative side, so we want all
|
|
|
|
// pass and target on the positive side
|
|
|
|
fliptest = false;
|
|
|
|
break;
|
2012-05-21 23:23:22 +00:00
|
|
|
} else if (d > ON_EPSILON) {
|
2002-08-25 23:06:23 +00:00
|
|
|
// source is on the positive side, so we want all
|
|
|
|
// pass and target on the negative side
|
|
|
|
fliptest = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (k == source->numpoints)
|
|
|
|
continue; // planar with source portal
|
|
|
|
|
|
|
|
// flip the normal if the source portal is backwards
|
|
|
|
if (fliptest) {
|
2003-09-03 22:17:04 +00:00
|
|
|
VectorNegate (plane.normal, plane.normal);
|
2002-08-25 23:06:23 +00:00
|
|
|
plane.dist = -plane.dist;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if all of the pass portal points are now on the positive side,
|
|
|
|
// this is the seperating plane
|
|
|
|
counts[0] = counts[1] = counts[2] = 0;
|
|
|
|
for (k = 0; k < pass->numpoints; k++) {
|
|
|
|
if (k == j)
|
|
|
|
continue;
|
|
|
|
d = DotProduct (pass->points[k], plane.normal) - plane.dist;
|
|
|
|
if (d < -ON_EPSILON)
|
|
|
|
break;
|
|
|
|
else if (d > ON_EPSILON)
|
|
|
|
counts[0]++;
|
|
|
|
else
|
|
|
|
counts[2]++;
|
|
|
|
}
|
|
|
|
if (k != pass->numpoints)
|
|
|
|
continue; // points on negative side, not a seperating plane
|
|
|
|
|
|
|
|
if (!counts[0]) {
|
|
|
|
continue; // planar with seperating plane
|
|
|
|
}
|
|
|
|
|
|
|
|
// flip the normal if we want the back side
|
2013-03-15 07:00:39 +00:00
|
|
|
if (flip) {
|
2003-09-03 22:17:04 +00:00
|
|
|
VectorNegate (plane.normal, plane.normal);
|
2002-08-25 23:06:23 +00:00
|
|
|
plane.dist = -plane.dist;
|
|
|
|
}
|
|
|
|
|
2013-03-15 07:00:39 +00:00
|
|
|
sep_t *sep = new_separator (thread);
|
|
|
|
sep->plane = plane;
|
|
|
|
sep->next = separators;
|
|
|
|
separators = sep;
|
2013-03-08 13:20:29 +00:00
|
|
|
break;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
2013-03-07 05:31:00 +00:00
|
|
|
}
|
2013-03-15 07:00:39 +00:00
|
|
|
return separators;
|
|
|
|
}
|
|
|
|
|
|
|
|
static winding_t *
|
|
|
|
ClipToSeparators (const sep_t *separators, winding_t *target)
|
|
|
|
{
|
|
|
|
const sep_t *sep;
|
|
|
|
|
|
|
|
for (sep = separators; target && sep; sep = sep->next) {
|
|
|
|
target = ClipWinding (target, &sep->plane, false);
|
|
|
|
}
|
2013-03-07 05:31:00 +00:00
|
|
|
return target;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
|
|
|
|
2013-03-14 10:39:27 +00:00
|
|
|
static inline set_t *
|
|
|
|
select_test_set (portal_t *portal, threaddata_t *thread)
|
|
|
|
{
|
|
|
|
set_t *test;
|
|
|
|
|
|
|
|
if (portal->status == stat_done) {
|
|
|
|
thread->stats.vistest++;
|
|
|
|
test = portal->visbits;
|
|
|
|
} else {
|
|
|
|
thread->stats.mighttest++;
|
|
|
|
test = portal->mightsee;
|
|
|
|
}
|
|
|
|
return test;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
mightsee_more (set_t *might, const set_t *prev_might, const set_t *test,
|
|
|
|
const set_t *vis)
|
|
|
|
{
|
|
|
|
set_assign (might, prev_might);
|
|
|
|
set_intersection (might, test);
|
|
|
|
return !set_is_subset (vis, might);
|
|
|
|
}
|
|
|
|
|
2002-08-25 23:06:23 +00:00
|
|
|
/*
|
2003-02-04 23:26:26 +00:00
|
|
|
RecursiveClusterFlow
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2003-02-04 23:26:26 +00:00
|
|
|
Flood fill through the clusters
|
|
|
|
If src_portal is NULL, this is the originating cluster
|
2002-08-25 23:06:23 +00:00
|
|
|
*/
|
2003-01-06 18:28:13 +00:00
|
|
|
static void
|
2013-03-14 10:35:43 +00:00
|
|
|
RecursiveClusterFlow (int clusternum, threaddata_t *thread, pstack_t *prevstack)
|
2002-08-25 23:06:23 +00:00
|
|
|
{
|
2013-03-07 05:31:00 +00:00
|
|
|
int i;
|
2013-03-07 05:28:15 +00:00
|
|
|
set_t *might;
|
2013-03-07 05:31:00 +00:00
|
|
|
const set_t *test, *vis;
|
|
|
|
cluster_t *cluster;
|
|
|
|
pstack_t stack;
|
|
|
|
portal_t *portal;
|
|
|
|
plane_t backplane;
|
|
|
|
winding_t *source, *target;
|
|
|
|
|
2013-03-14 02:33:47 +00:00
|
|
|
thread->stats.chains++;
|
2013-03-07 05:31:00 +00:00
|
|
|
|
|
|
|
cluster = &clusters[clusternum];
|
|
|
|
if (CheckStack(cluster, thread))
|
2003-09-24 00:59:53 +00:00
|
|
|
return;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2003-02-04 23:26:26 +00:00
|
|
|
// mark the cluster as visible
|
2013-03-07 02:06:55 +00:00
|
|
|
if (!set_is_member (thread->clustervis, clusternum)) {
|
|
|
|
set_add (thread->clustervis, clusternum);
|
2002-08-25 23:06:23 +00:00
|
|
|
thread->base->numcansee++;
|
2013-03-07 05:31:00 +00:00
|
|
|
}
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2013-03-07 05:31:00 +00:00
|
|
|
prevstack->next = &stack;
|
|
|
|
stack.next = NULL;
|
|
|
|
stack.cluster = cluster;
|
|
|
|
stack.portal = NULL;
|
2013-03-07 02:06:55 +00:00
|
|
|
LOCK;
|
2013-03-07 05:31:00 +00:00
|
|
|
stack.mightsee = set_new_size (portalclusters);
|
2013-03-07 02:06:55 +00:00
|
|
|
UNLOCK;
|
2013-03-08 13:20:29 +00:00
|
|
|
stack.separators[0] = 0;
|
|
|
|
stack.separators[1] = 0;
|
2013-03-07 05:31:00 +00:00
|
|
|
might = stack.mightsee;
|
|
|
|
vis = thread->clustervis;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2012-05-21 23:23:22 +00:00
|
|
|
// check all portals for flowing into other clusters
|
2013-03-07 05:31:00 +00:00
|
|
|
for (i = 0; i < cluster->numportals; i++) {
|
2003-02-04 23:26:26 +00:00
|
|
|
portal = cluster->portals[i];
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2013-03-07 02:06:55 +00:00
|
|
|
if (!set_is_member (prevstack->mightsee, portal->cluster))
|
2002-08-25 23:06:23 +00:00
|
|
|
continue; // can't possibly see it
|
|
|
|
|
2013-03-14 10:39:27 +00:00
|
|
|
// if the portal can't see anything we haven't already seen, skip it
|
|
|
|
test = select_test_set (portal, thread);
|
|
|
|
if (!mightsee_more (might, prevstack->mightsee, test, vis)) {
|
|
|
|
// can't see anything new
|
2002-08-25 23:06:23 +00:00
|
|
|
continue;
|
2013-03-14 10:39:27 +00:00
|
|
|
}
|
2002-09-23 22:54:28 +00:00
|
|
|
|
2003-02-04 23:26:26 +00:00
|
|
|
// get plane of portal, point normal into the neighbor cluster
|
2002-09-24 02:46:03 +00:00
|
|
|
stack.portalplane = portal->plane;
|
2003-09-03 22:17:04 +00:00
|
|
|
VectorNegate (portal->plane.normal, backplane.normal);
|
2002-09-24 02:46:03 +00:00
|
|
|
backplane.dist = -portal->plane.dist;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2002-09-20 17:02:53 +00:00
|
|
|
if (_VectorCompare (prevstack->portalplane.normal, backplane.normal))
|
2002-08-25 23:06:23 +00:00
|
|
|
continue; // can't go out a coplanar face
|
|
|
|
|
2013-03-14 02:33:47 +00:00
|
|
|
thread->stats.portalcheck++;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2002-09-24 02:46:03 +00:00
|
|
|
stack.portal = portal;
|
2002-08-25 23:06:23 +00:00
|
|
|
stack.next = NULL;
|
|
|
|
|
2004-02-08 02:53:58 +00:00
|
|
|
target = ClipWinding (portal->winding,
|
|
|
|
&thread->pstack_head.portalplane, false);
|
2002-08-25 23:06:23 +00:00
|
|
|
if (!target)
|
|
|
|
continue;
|
|
|
|
|
2002-08-26 15:05:23 +00:00
|
|
|
if (!prevstack->pass) {
|
2010-01-13 06:42:26 +00:00
|
|
|
// the second cluster can be blocked only if coplanar
|
2002-08-25 23:06:23 +00:00
|
|
|
|
|
|
|
stack.source = prevstack->source;
|
|
|
|
stack.pass = target;
|
2013-03-14 10:35:43 +00:00
|
|
|
RecursiveClusterFlow (portal->cluster, thread, &stack);
|
2002-08-25 23:06:23 +00:00
|
|
|
FreeWinding (target);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
target = ClipWinding (target, &prevstack->portalplane, false);
|
|
|
|
if (!target)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
source = CopyWinding (prevstack->source);
|
|
|
|
|
|
|
|
source = ClipWinding (source, &backplane, false);
|
|
|
|
if (!source) {
|
|
|
|
FreeWinding (target);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-03-14 02:33:47 +00:00
|
|
|
thread->stats.portaltest++;
|
2013-03-14 10:40:31 +00:00
|
|
|
thread->stats.targettested++;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
|
|
|
if (options.level > 0) {
|
2002-09-24 21:48:47 +00:00
|
|
|
// clip target to the image that would be formed by a laser
|
|
|
|
// pointing from the edges of source passing though the corners of
|
|
|
|
// pass
|
2013-03-14 10:40:31 +00:00
|
|
|
winding_t *old = target;
|
2013-03-15 07:00:39 +00:00
|
|
|
if (!stack.separators[0])
|
|
|
|
stack.separators[0] = FindSeparators (thread, source,
|
|
|
|
prevstack->pass, 0);
|
|
|
|
target = ClipToSeparators (stack.separators[0], target);
|
2002-08-25 23:06:23 +00:00
|
|
|
if (!target) {
|
2013-03-14 10:40:31 +00:00
|
|
|
thread->stats.targetclipped++;
|
2002-08-25 23:06:23 +00:00
|
|
|
FreeWinding (source);
|
|
|
|
continue;
|
|
|
|
}
|
2013-03-14 10:40:31 +00:00
|
|
|
if (target != old)
|
|
|
|
thread->stats.targettrimmed++;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (options.level > 1) {
|
2002-09-24 21:48:47 +00:00
|
|
|
// now pass the laser along the edges of pass from the corners of
|
|
|
|
// source. the resulting image will have a smaller aree. The
|
|
|
|
// resulting shape will be the light image produced by a backlit
|
|
|
|
// source shining past pass. eg, if source and pass are equilateral
|
|
|
|
// triangles rotated 60 (or 180) degrees relative to each other,
|
|
|
|
// parallel and in line, target will wind up being a hexagon.
|
2013-03-14 10:40:31 +00:00
|
|
|
winding_t *old = target;
|
2013-03-15 07:00:39 +00:00
|
|
|
if (!stack.separators[1])
|
|
|
|
stack.separators[1] = FindSeparators (thread, prevstack->pass,
|
|
|
|
source, 1);
|
|
|
|
target = ClipToSeparators (stack.separators[1], target);
|
2002-08-25 23:06:23 +00:00
|
|
|
if (!target) {
|
2013-03-14 10:40:31 +00:00
|
|
|
thread->stats.targetclipped++;
|
2002-08-25 23:06:23 +00:00
|
|
|
FreeWinding (source);
|
|
|
|
continue;
|
|
|
|
}
|
2013-03-14 10:40:31 +00:00
|
|
|
if (target != old)
|
|
|
|
thread->stats.targettrimmed++;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
|
|
|
|
2013-03-14 10:40:31 +00:00
|
|
|
thread->stats.sourcetested++;
|
2012-05-21 23:23:22 +00:00
|
|
|
// now do the same as for levels 1 and 2, but trimming source using
|
2002-09-24 21:48:47 +00:00
|
|
|
// the trimmed target
|
2002-08-25 23:06:23 +00:00
|
|
|
if (options.level > 2) {
|
2013-03-14 10:40:31 +00:00
|
|
|
winding_t *old = source;
|
2013-03-15 07:00:39 +00:00
|
|
|
sep_t *sep;
|
|
|
|
sep = FindSeparators (thread, target, prevstack->pass, 0);
|
|
|
|
source = ClipToSeparators (sep, source);
|
|
|
|
free_separators (thread, sep);
|
2002-08-25 23:06:23 +00:00
|
|
|
if (!source) {
|
2013-03-14 10:40:31 +00:00
|
|
|
thread->stats.sourceclipped++;
|
2002-08-25 23:06:23 +00:00
|
|
|
FreeWinding (target);
|
|
|
|
continue;
|
|
|
|
}
|
2013-03-14 10:40:31 +00:00
|
|
|
if (source != old)
|
|
|
|
thread->stats.sourcetrimmed++;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (options.level > 3) {
|
2013-03-14 10:40:31 +00:00
|
|
|
winding_t *old = source;
|
2013-03-15 07:00:39 +00:00
|
|
|
sep_t *sep;
|
|
|
|
sep = FindSeparators (thread, prevstack->pass, target, 1);
|
|
|
|
source = ClipToSeparators (sep, source);
|
|
|
|
free_separators (thread, sep);
|
2002-08-25 23:06:23 +00:00
|
|
|
if (!source) {
|
2013-03-14 10:40:31 +00:00
|
|
|
thread->stats.sourceclipped++;
|
2002-08-25 23:06:23 +00:00
|
|
|
FreeWinding (target);
|
|
|
|
continue;
|
|
|
|
}
|
2013-03-14 10:40:31 +00:00
|
|
|
if (source != old)
|
|
|
|
thread->stats.sourcetrimmed++;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
stack.source = source;
|
|
|
|
stack.pass = target;
|
|
|
|
|
2013-03-14 02:33:47 +00:00
|
|
|
thread->stats.portalpass++;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
|
|
|
// flow through it for real
|
2013-03-14 10:35:43 +00:00
|
|
|
RecursiveClusterFlow (portal->cluster, thread, &stack);
|
2002-08-25 23:06:23 +00:00
|
|
|
|
|
|
|
FreeWinding (source);
|
|
|
|
FreeWinding (target);
|
2013-03-07 05:31:00 +00:00
|
|
|
}
|
2013-03-15 07:00:39 +00:00
|
|
|
free_separators (thread, stack.separators[1]);
|
|
|
|
free_separators (thread, stack.separators[0]);
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2013-03-07 02:06:55 +00:00
|
|
|
LOCK;
|
2013-03-07 05:31:00 +00:00
|
|
|
set_delete (stack.mightsee);
|
2013-03-07 02:06:55 +00:00
|
|
|
UNLOCK;
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2013-03-14 02:33:47 +00:00
|
|
|
PortalFlow (threaddata_t *data, portal_t *portal)
|
2002-08-25 23:06:23 +00:00
|
|
|
{
|
2002-09-22 21:54:41 +00:00
|
|
|
LOCK;
|
2013-03-07 05:31:00 +00:00
|
|
|
if (portal->status != stat_selected)
|
2002-09-22 21:32:36 +00:00
|
|
|
Sys_Error ("PortalFlow: reflowed");
|
2013-03-07 05:31:00 +00:00
|
|
|
portal->status = stat_working;
|
|
|
|
portal->visbits = set_new_size (portalclusters);
|
2002-09-22 21:54:41 +00:00
|
|
|
UNLOCK;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2013-03-14 02:33:47 +00:00
|
|
|
data->clustervis = portal->visbits;
|
|
|
|
data->base = portal;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2013-03-14 02:33:47 +00:00
|
|
|
memset (&data->pstack_head, 0, sizeof (data->pstack_head));
|
|
|
|
data->pstack_head.portal = portal;
|
|
|
|
data->pstack_head.source = portal->winding;
|
|
|
|
data->pstack_head.portalplane = portal->plane;
|
|
|
|
data->pstack_head.mightsee = portal->mightsee;
|
2002-08-25 23:06:23 +00:00
|
|
|
|
2013-03-14 10:35:43 +00:00
|
|
|
RecursiveClusterFlow (portal->cluster, data, &data->pstack_head);
|
2002-08-25 23:06:23 +00:00
|
|
|
}
|