2016-09-14 18:01:13 +00:00
//
//---------------------------------------------------------------------------
//
// Copyright(C) 2005-2016 Christoph Oelckers
// All rights reserved.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 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 Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with this program. If not, see http://www.gnu.org/licenses/
//
//--------------------------------------------------------------------------
//
2013-06-23 07:49:34 +00:00
# include "doomtype.h"
# include "p_local.h"
# include "p_lnspec.h"
# include "c_dispatch.h"
# include "a_sharedglobal.h"
2017-01-08 17:45:30 +00:00
# include "g_levellocals.h"
2018-04-01 22:39:04 +00:00
# include "r_utility.h"
2013-06-23 07:49:34 +00:00
2018-04-01 20:26:57 +00:00
//==========================================================================
//
// Helper types for portal grouping
//
//==========================================================================
2013-06-23 07:49:34 +00:00
struct FPortalID
{
2016-04-02 21:17:16 +00:00
DVector2 mDisplacement ;
2013-06-23 07:49:34 +00:00
// for the hash code
2016-04-02 21:17:16 +00:00
operator intptr_t ( ) const { return ( FLOAT2FIXED ( mDisplacement . X ) > > 8 ) + ( FLOAT2FIXED ( mDisplacement . Y ) < < 8 ) ; }
2013-06-23 07:49:34 +00:00
bool operator ! = ( const FPortalID & other ) const
{
2016-04-02 21:17:16 +00:00
return mDisplacement ! = other . mDisplacement ;
2013-06-23 07:49:34 +00:00
}
} ;
struct FPortalSector
{
sector_t * mSub ;
int mPlane ;
} ;
typedef TArray < FPortalSector > FPortalSectors ;
typedef TMap < FPortalID , FPortalSectors > FPortalMap ;
//==========================================================================
//
2016-04-08 10:38:09 +00:00
// this is left as fixed_t because the nodes also are, it makes no sense
// to convert this without converting the nodes as well.
2013-06-23 07:49:34 +00:00
//
//==========================================================================
struct FCoverageVertex
{
fixed_t x , y ;
bool operator ! = ( FCoverageVertex & other )
{
return x ! = other . x | | y ! = other . y ;
}
} ;
struct FCoverageLine
{
FCoverageVertex v [ 2 ] ;
} ;
struct FCoverageBuilder
{
subsector_t * target ;
TArray < int > collect ;
FCoverageVertex center ;
//==========================================================================
//
//
2018-04-01 22:14:53 +00:00
//
2013-06-23 07:49:34 +00:00
//==========================================================================
2016-04-20 09:39:41 +00:00
FCoverageBuilder ( subsector_t * sub )
2013-06-23 07:49:34 +00:00
{
target = sub ;
}
//==========================================================================
//
// GetIntersection
//
// adapted from P_InterceptVector
//
//==========================================================================
bool GetIntersection ( FCoverageVertex * v1 , FCoverageVertex * v2 , node_t * bsp , FCoverageVertex * v )
{
double frac ;
double num ;
double den ;
double v2x = ( double ) v1 - > x ;
double v2y = ( double ) v1 - > y ;
double v2dx = ( double ) ( v2 - > x - v1 - > x ) ;
double v2dy = ( double ) ( v2 - > y - v1 - > y ) ;
double v1x = ( double ) bsp - > x ;
double v1y = ( double ) bsp - > y ;
double v1dx = ( double ) bsp - > dx ;
double v1dy = ( double ) bsp - > dy ;
den = v1dy * v2dx - v1dx * v2dy ;
if ( den = = 0 )
return false ; // parallel
num = ( v1x - v2x ) * v1dy + ( v2y - v1y ) * v1dx ;
frac = num / den ;
if ( frac < 0. | | frac > 1. ) return false ;
v - > x = xs_RoundToInt ( v2x + frac * v2dx ) ;
v - > y = xs_RoundToInt ( v2y + frac * v2dy ) ;
return true ;
}
//==========================================================================
//
//
//
//==========================================================================
double PartitionDistance ( FCoverageVertex * vt , node_t * node )
{
2016-04-16 23:16:46 +00:00
return fabs ( double ( - node - > dy ) * ( vt - > x - node - > x ) + double ( node - > dx ) * ( vt - > y - node - > y ) ) / ( node - > len * 65536. ) ;
2013-06-23 07:49:34 +00:00
}
//==========================================================================
//
//
//
//==========================================================================
int PointOnSide ( FCoverageVertex * vt , node_t * node )
{
return R_PointOnSide ( vt - > x , vt - > y , node ) ;
}
//==========================================================================
//
// adapted from polyobject splitter
//
//==========================================================================
void CollectNode ( void * node , TArray < FCoverageVertex > & shape )
{
static TArray < FCoverageLine > lists [ 2 ] ;
const double COVERAGE_EPSILON = 6. ; // same epsilon as the node builder
if ( ! ( ( size_t ) node & 1 ) ) // Keep going until found a subsector
{
node_t * bsp = ( node_t * ) node ;
int centerside = R_PointOnSide ( center . x , center . y , bsp ) ;
lists [ 0 ] . Clear ( ) ;
lists [ 1 ] . Clear ( ) ;
for ( unsigned i = 0 ; i < shape . Size ( ) ; i + + )
{
FCoverageVertex * v1 = & shape [ i ] ;
FCoverageVertex * v2 = & shape [ ( i + 1 ) % shape . Size ( ) ] ;
2013-08-28 06:33:11 +00:00
FCoverageLine vl = { { * v1 , * v2 } } ;
2013-06-23 07:49:34 +00:00
double dist_v1 = PartitionDistance ( v1 , bsp ) ;
double dist_v2 = PartitionDistance ( v2 , bsp ) ;
if ( dist_v1 < = COVERAGE_EPSILON )
{
if ( dist_v2 < = COVERAGE_EPSILON )
{
lists [ centerside ] . Push ( vl ) ;
}
else
{
int side = PointOnSide ( v2 , bsp ) ;
lists [ side ] . Push ( vl ) ;
}
}
else if ( dist_v2 < = COVERAGE_EPSILON )
{
int side = PointOnSide ( v1 , bsp ) ;
lists [ side ] . Push ( vl ) ;
}
else
{
int side1 = PointOnSide ( v1 , bsp ) ;
int side2 = PointOnSide ( v2 , bsp ) ;
if ( side1 ! = side2 )
{
// if the partition line crosses this seg, we must split it.
FCoverageVertex vert ;
if ( GetIntersection ( v1 , v2 , bsp , & vert ) )
{
lists [ 0 ] . Push ( vl ) ;
lists [ 1 ] . Push ( vl ) ;
lists [ side1 ] . Last ( ) . v [ 1 ] = vert ;
lists [ side2 ] . Last ( ) . v [ 0 ] = vert ;
}
else
{
// should never happen
lists [ side1 ] . Push ( vl ) ;
}
}
else
{
// both points on the same side.
lists [ side1 ] . Push ( vl ) ;
}
}
}
if ( lists [ 1 ] . Size ( ) = = 0 )
{
CollectNode ( bsp - > children [ 0 ] , shape ) ;
}
else if ( lists [ 0 ] . Size ( ) = = 0 )
{
CollectNode ( bsp - > children [ 1 ] , shape ) ;
}
else
{
// copy the static arrays into local ones
TArray < FCoverageVertex > locallists [ 2 ] ;
for ( int l = 0 ; l < 2 ; l + + )
{
for ( unsigned i = 0 ; i < lists [ l ] . Size ( ) ; i + + )
{
locallists [ l ] . Push ( lists [ l ] [ i ] . v [ 0 ] ) ;
unsigned i1 = ( i + 1 ) % lists [ l ] . Size ( ) ;
if ( lists [ l ] [ i1 ] . v [ 0 ] ! = lists [ l ] [ i ] . v [ 1 ] )
{
locallists [ l ] . Push ( lists [ l ] [ i ] . v [ 1 ] ) ;
}
}
}
CollectNode ( bsp - > children [ 0 ] , locallists [ 0 ] ) ;
CollectNode ( bsp - > children [ 1 ] , locallists [ 1 ] ) ;
}
}
else
{
// we reached a subsector so we can link the node with this subsector
2017-03-09 18:54:41 +00:00
subsector_t * sub = ( subsector_t * ) ( ( uint8_t * ) node - 1 ) ;
2017-03-16 23:22:52 +00:00
collect . Push ( int ( sub - > Index ( ) ) ) ;
2013-06-23 07:49:34 +00:00
}
}
} ;
//==========================================================================
//
// Calculate portal coverage for a single subsector
2018-04-01 20:26:57 +00:00
// This data is used by the clipper to free up the ranges covered by a portal.
//
// This also gets called by the render hack code because ZDoom was really lax
// with its stacked sector things and allowed partial tagging of affected sectors
// Any such sector will only be found during rendering and must create its
// coverage info then.
2013-06-23 07:49:34 +00:00
//
//==========================================================================
2018-04-01 20:26:57 +00:00
void BuildPortalCoverage ( FPortalCoverage * coverage , subsector_t * subsector , const DVector2 & displacement )
2013-06-23 07:49:34 +00:00
{
TArray < FCoverageVertex > shape ;
double centerx = 0 , centery = 0 ;
shape . Resize ( subsector - > numlines ) ;
for ( unsigned i = 0 ; i < subsector - > numlines ; i + + )
{
2016-04-20 09:39:41 +00:00
centerx + = ( shape [ i ] . x = FLOAT2FIXED ( subsector - > firstline [ i ] . v1 - > fX ( ) + displacement . X ) ) ;
centery + = ( shape [ i ] . y = FLOAT2FIXED ( subsector - > firstline [ i ] . v1 - > fY ( ) + displacement . Y ) ) ;
2013-06-23 07:49:34 +00:00
}
2016-04-20 09:39:41 +00:00
FCoverageBuilder build ( subsector ) ;
2013-06-23 07:49:34 +00:00
build . center . x = xs_CRoundToInt ( centerx / subsector - > numlines ) ;
build . center . y = xs_CRoundToInt ( centery / subsector - > numlines ) ;
2017-03-17 00:42:37 +00:00
build . CollectNode ( level . HeadNode ( ) , shape ) ;
2017-03-09 19:19:55 +00:00
coverage - > subsectors = new uint32_t [ build . collect . Size ( ) ] ;
2013-06-23 07:49:34 +00:00
coverage - > sscount = build . collect . Size ( ) ;
2017-03-09 19:19:55 +00:00
memcpy ( coverage - > subsectors , & build . collect [ 0 ] , build . collect . Size ( ) * sizeof ( uint32_t ) ) ;
2013-06-23 07:49:34 +00:00
}
//==========================================================================
//
2018-04-01 20:26:57 +00:00
//
2013-06-23 07:49:34 +00:00
//
//==========================================================================
static void CollectPortalSectors ( FPortalMap & collection )
{
2017-01-07 18:32:24 +00:00
for ( auto & sec : level . sectors )
2013-06-23 07:49:34 +00:00
{
2016-01-12 23:15:04 +00:00
for ( int j = 0 ; j < 2 ; j + + )
2013-06-23 07:49:34 +00:00
{
2017-01-07 18:32:24 +00:00
int ptype = sec . GetPortalType ( j ) ;
2016-04-20 18:08:53 +00:00
if ( ptype = = PORTS_STACKEDSECTORTHING | | ptype = = PORTS_PORTAL | | ptype = = PORTS_LINKEDPORTAL ) // only offset-displacing portal types
2016-01-12 23:15:04 +00:00
{
2017-01-07 18:32:24 +00:00
FPortalID id = { sec . GetPortalDisplacement ( j ) } ;
2013-06-23 07:49:34 +00:00
2016-01-12 23:15:04 +00:00
FPortalSectors & sss = collection [ id ] ;
2017-01-07 18:32:24 +00:00
FPortalSector ss = { & sec , j } ;
2016-01-12 23:15:04 +00:00
sss . Push ( ss ) ;
}
2013-06-23 07:49:34 +00:00
}
}
}
2018-04-01 20:26:57 +00:00
//==========================================================================
//
// group sector portals by displacement
// The renderer can handle such a group in one go to avoid multiple
// BSP traversals
//
//==========================================================================
static void GroupSectorPortals ( )
2013-06-23 07:49:34 +00:00
{
FPortalMap collection ;
CollectPortalSectors ( collection ) ;
2018-04-01 20:26:57 +00:00
level . portalGroups . Clear ( ) ;
2013-06-23 07:49:34 +00:00
FPortalMap : : Iterator it ( collection ) ;
FPortalMap : : Pair * pair ;
int c = 0 ;
int planeflags = 0 ;
while ( it . NextPair ( pair ) )
{
2018-04-01 20:26:57 +00:00
for ( unsigned i = 0 ; i < pair - > Value . Size ( ) ; i + + )
2013-06-23 07:49:34 +00:00
{
if ( pair - > Value [ i ] . mPlane = = sector_t : : floor ) planeflags | = 1 ;
else if ( pair - > Value [ i ] . mPlane = = sector_t : : ceiling ) planeflags | = 2 ;
}
2018-04-01 20:26:57 +00:00
for ( int i = 1 ; i < = 2 ; i < < = 1 )
2013-06-23 07:49:34 +00:00
{
2018-04-01 20:26:57 +00:00
// add separate portals for floor and ceiling.
2013-06-23 07:49:34 +00:00
if ( planeflags & i )
{
2018-04-01 20:26:57 +00:00
FSectorPortalGroup * portal = new FSectorPortalGroup ;
2016-04-02 21:17:16 +00:00
portal - > mDisplacement = pair - > Key . mDisplacement ;
2018-04-01 20:26:57 +00:00
portal - > plane = ( i = = 1 ? sector_t : : floor : sector_t : : ceiling ) ; /**/
2013-06-23 07:49:34 +00:00
portal - > glportal = NULL ;
2018-04-01 20:26:57 +00:00
level . portalGroups . Push ( portal ) ;
for ( unsigned j = 0 ; j < pair - > Value . Size ( ) ; j + + )
2013-06-23 07:49:34 +00:00
{
sector_t * sec = pair - > Value [ j ] . mSub ;
int plane = pair - > Value [ j ] . mPlane ;
if ( portal - > plane = = plane )
{
2018-04-01 20:26:57 +00:00
for ( int k = 0 ; k < sec - > subsectorcount ; k + + )
2013-06-23 07:49:34 +00:00
{
subsector_t * sub = sec - > subsectors [ k ] ;
2018-04-01 20:26:57 +00:00
BuildPortalCoverage ( & sub - > portalcoverage [ plane ] , sub , pair - > Key . mDisplacement ) ;
2013-06-23 07:49:34 +00:00
}
sec - > portals [ plane ] = portal ;
}
}
}
}
}
2018-04-01 20:26:57 +00:00
}
//==========================================================================
//
// Group the line portals
// Each group must be a continuous set of colinear linedefs with no gaps
//
//==========================================================================
2016-03-04 13:10:13 +00:00
2018-04-01 20:26:57 +00:00
static void GroupLinePortals ( )
{
level . linePortalSpans . Clear ( ) ;
2016-03-04 13:10:13 +00:00
TArray < int > tempindex ;
2018-04-01 18:17:39 +00:00
tempindex . Reserve ( level . linePortals . Size ( ) ) ;
memset ( & tempindex [ 0 ] , - 1 , level . linePortals . Size ( ) * sizeof ( int ) ) ;
2016-03-04 13:10:13 +00:00
2018-04-01 18:17:39 +00:00
for ( unsigned i = 0 ; i < level . linePortals . Size ( ) ; i + + )
2016-03-04 13:10:13 +00:00
{
2018-04-01 18:17:39 +00:00
auto port = level . linePortals [ i ] ;
2016-03-04 13:10:13 +00:00
bool gotsome ;
if ( tempindex [ i ] = = - 1 )
{
2018-04-01 20:26:57 +00:00
tempindex [ i ] = level . linePortalSpans . Size ( ) ;
2018-04-01 18:17:39 +00:00
line_t * pSrcLine = level . linePortals [ i ] . mOrigin ;
line_t * pLine = level . linePortals [ i ] . mDestination ;
2018-04-01 20:26:57 +00:00
FLinePortalSpan & glport = level . linePortalSpans [ level . linePortalSpans . Reserve ( 1 ) ] ;
2018-04-01 18:17:39 +00:00
glport . lines . Push ( & level . linePortals [ i ] ) ;
2016-03-04 13:10:13 +00:00
2018-04-01 20:26:57 +00:00
// We cannot do this grouping for non-linked portals because they can be changed at run time.
2018-04-01 18:17:39 +00:00
if ( level . linePortals [ i ] . mType = = PORTT_LINKED & & pLine ! = nullptr )
2016-03-04 13:10:13 +00:00
{
2016-07-16 10:45:49 +00:00
glport . v1 = pLine - > v1 ;
glport . v2 = pLine - > v2 ;
do
2016-03-04 13:10:13 +00:00
{
2016-07-16 10:45:49 +00:00
// now collect all other colinear lines connected to this one. We run this loop as long as it still finds a match
gotsome = false ;
2018-04-01 18:17:39 +00:00
for ( unsigned j = 0 ; j < level . linePortals . Size ( ) ; j + + )
2016-03-04 13:10:13 +00:00
{
2016-07-16 10:45:49 +00:00
if ( tempindex [ j ] = = - 1 )
2016-03-04 13:10:13 +00:00
{
2018-04-01 18:17:39 +00:00
line_t * pSrcLine2 = level . linePortals [ j ] . mOrigin ;
line_t * pLine2 = level . linePortals [ j ] . mDestination ;
2018-04-01 20:26:57 +00:00
// angular precision is intentionally reduced to 32 bit BAM to account for precision problems (otherwise many not perfectly horizontal or vertical portals aren't found here.)
2016-07-16 10:45:49 +00:00
unsigned srcang = pSrcLine - > Delta ( ) . Angle ( ) . BAMs ( ) ;
unsigned dstang = pLine - > Delta ( ) . Angle ( ) . BAMs ( ) ;
if ( ( pSrcLine - > v2 = = pSrcLine2 - > v1 & & pLine - > v1 = = pLine2 - > v2 ) | |
( pSrcLine - > v1 = = pSrcLine2 - > v2 & & pLine - > v2 = = pLine2 - > v1 ) )
2016-03-04 13:10:13 +00:00
{
2016-07-16 10:45:49 +00:00
// The line connects, now check the translation
unsigned srcang2 = pSrcLine2 - > Delta ( ) . Angle ( ) . BAMs ( ) ;
unsigned dstang2 = pLine2 - > Delta ( ) . Angle ( ) . BAMs ( ) ;
if ( srcang = = srcang2 & & dstang = = dstang2 )
{
// The lines connect and both source and destination are colinear, so this is a match
gotsome = true ;
tempindex [ j ] = tempindex [ i ] ;
if ( pLine - > v1 = = pLine2 - > v2 ) glport . v1 = pLine2 - > v1 ;
else glport . v2 = pLine2 - > v2 ;
2018-04-01 18:17:39 +00:00
glport . lines . Push ( & level . linePortals [ j ] ) ;
2016-07-16 10:45:49 +00:00
}
2016-03-04 13:10:13 +00:00
}
}
}
2016-07-16 10:45:49 +00:00
} while ( gotsome ) ;
}
2016-03-04 13:10:13 +00:00
}
}
2018-04-01 20:26:57 +00:00
// Final assignment can only be done when all allocations are finished. Otherwise the array may be moved.
2018-04-01 18:17:39 +00:00
for ( unsigned i = 0 ; i < level . linePortals . Size ( ) ; i + + )
2016-03-04 13:10:13 +00:00
{
2018-04-01 20:26:57 +00:00
level . linePortals [ i ] . mGroup = & level . linePortalSpans [ tempindex [ i ] ] ;
2016-03-04 13:10:13 +00:00
}
2018-04-01 20:26:57 +00:00
}
void InitPortalGroups ( )
{
if ( level . nodes . Size ( ) = = 0 ) return ;
GroupSectorPortals ( ) ;
GroupLinePortals ( ) ;
2013-06-23 07:49:34 +00:00
}
CCMD ( dumpportals )
{
2018-04-01 20:26:57 +00:00
for ( unsigned i = 0 ; i < level . portalGroups . Size ( ) ; i + + )
2013-06-23 07:49:34 +00:00
{
2018-04-01 20:26:57 +00:00
auto p = level . portalGroups [ i ] ;
double xdisp = p - > mDisplacement . X ;
double ydisp = p - > mDisplacement . Y ;
Printf ( PRINT_LOG , " Portal #%d, %s, displacement = (%f,%f) \n " , i , p - > plane = = 0 ? " floor " : " ceiling " ,
2013-06-23 07:49:34 +00:00
xdisp , ydisp ) ;
Printf ( PRINT_LOG , " Coverage: \n " ) ;
2017-03-16 23:22:52 +00:00
for ( auto & sub : level . subsectors )
2013-06-23 07:49:34 +00:00
{
2018-04-01 20:26:57 +00:00
auto port = sub . render_sector - > GetPortalGroup ( p - > plane ) ;
if ( port = = p )
2013-06-23 07:49:34 +00:00
{
2017-03-16 23:22:52 +00:00
Printf ( PRINT_LOG , " \t Subsector %d (%d): \n \t \t " , sub . Index ( ) , sub . render_sector - > sectornum ) ;
for ( unsigned k = 0 ; k < sub . numlines ; k + + )
2013-06-23 07:49:34 +00:00
{
2017-03-16 23:22:52 +00:00
Printf ( PRINT_LOG , " (%.3f,%.3f), " , sub . firstline [ k ] . v1 - > fX ( ) + xdisp , sub . firstline [ k ] . v1 - > fY ( ) + ydisp ) ;
2013-06-23 07:49:34 +00:00
}
Printf ( PRINT_LOG , " \n \t \t Covered by subsectors: \n " ) ;
2018-04-01 20:26:57 +00:00
FPortalCoverage * cov = & sub . portalcoverage [ p - > plane ] ;
2013-06-23 07:49:34 +00:00
for ( int l = 0 ; l < cov - > sscount ; l + + )
{
2017-03-16 23:22:52 +00:00
subsector_t * csub = & level . subsectors [ cov - > subsectors [ l ] ] ;
2013-06-23 07:49:34 +00:00
Printf ( PRINT_LOG , " \t \t \t %5d (%4d): " , cov - > subsectors [ l ] , csub - > render_sector - > sectornum ) ;
for ( unsigned m = 0 ; m < csub - > numlines ; m + + )
{
2016-04-20 09:39:41 +00:00
Printf ( PRINT_LOG , " (%.3f,%.3f), " , csub - > firstline [ m ] . v1 - > fX ( ) , csub - > firstline [ m ] . v1 - > fY ( ) ) ;
2013-06-23 07:49:34 +00:00
}
Printf ( PRINT_LOG , " \n " ) ;
}
}
}
}
}