|
|
|
/*==LICENSE==*
|
|
|
|
|
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools
|
|
|
|
Copyright (C) 2011 Cyan Worlds, 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 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
Additional permissions under GNU GPL version 3 section 7
|
|
|
|
|
|
|
|
If you modify this Program, or any covered work, by linking or
|
|
|
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
|
|
|
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
|
|
|
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
|
|
|
|
(or a modified version of those libraries),
|
|
|
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
|
|
|
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
|
|
|
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
|
|
|
|
licensors of this Program grant you additional
|
|
|
|
permission to convey the resulting work. Corresponding Source for a
|
|
|
|
non-source form of such a combination shall include the source code for
|
|
|
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
|
|
|
|
work.
|
|
|
|
|
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
|
|
|
or by snail mail at:
|
|
|
|
Cyan Worlds, Inc.
|
|
|
|
14617 N Newport Hwy
|
|
|
|
Mead, WA 99021
|
|
|
|
|
|
|
|
*==LICENSE==*/
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
// //
|
|
|
|
// plDrawableSpans Class Functions //
|
|
|
|
// //
|
|
|
|
//// Version History /////////////////////////////////////////////////////////
|
|
|
|
// //
|
|
|
|
// 4.3.2001 mcn - Created. //
|
|
|
|
// 5.3.2001 mcn - Completely revamped. Now plDrawableSpans *IS* the group //
|
|
|
|
// of spans, and each span is either an icicle or patch. //
|
|
|
|
// This eliminates the need entirely for separate drawables,//
|
|
|
|
// at the cost of having to do a bit of extra work to //
|
|
|
|
// maintain different types of spans in the same drawable. //
|
|
|
|
// //
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#include "hsTypes.h"
|
|
|
|
|
|
|
|
#include "plAccessSpan.h"
|
|
|
|
#include "plAccessTriSpan.h"
|
|
|
|
|
|
|
|
#include "plDrawableSpans.h"
|
|
|
|
#include "hsStream.h"
|
|
|
|
#include "hsResMgr.h"
|
|
|
|
#include "plPipeline.h"
|
|
|
|
#include "plGeometrySpan.h"
|
|
|
|
#include "plSpaceTree.h"
|
|
|
|
#include "plParticleFiller.h"
|
|
|
|
#include "plSpaceTreeMaker.h"
|
|
|
|
|
|
|
|
#include "plClusterGroup.h"
|
|
|
|
#include "plCluster.h"
|
|
|
|
#include "plSpanTemplate.h"
|
|
|
|
|
|
|
|
#include "plMath/hsRadixSort.h"
|
|
|
|
#include "plSurface/hsGMaterial.h"
|
|
|
|
#include "plSurface/plLayerInterface.h"
|
|
|
|
#include "plPipeline/plFogEnvironment.h"
|
|
|
|
#include "plPipeline/hsGDeviceRef.h"
|
|
|
|
#include "plPipeline/plPipeDebugFlags.h"
|
|
|
|
#include "pnMessage/plRefMsg.h"
|
|
|
|
#include "pnMessage/plNodeRefMsg.h"
|
|
|
|
#include "pnMessage/plDISpansMsg.h"
|
|
|
|
#include "plMessage/plDeviceRecreateMsg.h"
|
|
|
|
#include "plMessage/plRenderMsg.h"
|
|
|
|
#include "plPipeline/plGBufferGroup.h"
|
|
|
|
#include "pnSceneObject/plDrawInterface.h"
|
|
|
|
#include "pnKeyedObject/plKey.h"
|
|
|
|
#include "plParticleSystem/plParticleEmitter.h"
|
|
|
|
#include "plParticleSystem/plParticle.h"
|
|
|
|
#include "plGLight/plLightInfo.h"
|
|
|
|
#include "plgDispatch.h"
|
|
|
|
#include "plProfile.h"
|
|
|
|
|
|
|
|
#include "plMath/plTriUtils.h"
|
|
|
|
|
|
|
|
#include "pnMessage/plPipeResMakeMsg.h"
|
|
|
|
|
|
|
|
#include "plScene/plVisMgr.h"
|
|
|
|
#include "plScene/plVisRegion.h"
|
|
|
|
|
|
|
|
#include "plStatusLog/plStatusLog.h"
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
//// Local Konstants /////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
const uint32_t plDrawableSpans::kSpanTypeMask = 0xc0000000;
|
|
|
|
const uint32_t plDrawableSpans::kSpanIDMask = ~kSpanTypeMask;
|
|
|
|
const uint32_t plDrawableSpans::kSpanTypeIcicle = 0x00000000;
|
|
|
|
const uint32_t plDrawableSpans::kSpanTypeParticleSpan = 0xc0000000;
|
|
|
|
|
|
|
|
//// Constructor & Destructor ////////////////////////////////////////////////
|
|
|
|
|
|
|
|
plDrawableSpans::plDrawableSpans() :
|
|
|
|
fSceneNode(nil),
|
|
|
|
fSpaceTree(nil)
|
|
|
|
{
|
|
|
|
fReadyToRender = false;
|
|
|
|
fProps = 0;
|
|
|
|
fCriteria = 0;
|
|
|
|
fRegisteredForRecreate = false;
|
|
|
|
fRegisteredForRender = false;
|
|
|
|
fNeedCleanup = false;
|
|
|
|
|
|
|
|
fOptimized = true;
|
|
|
|
|
|
|
|
fSettingMatIdxLock = false;
|
|
|
|
|
|
|
|
fSkinTime = 0;
|
|
|
|
|
|
|
|
fType = kNormal;
|
|
|
|
fMaterials.Reset();
|
|
|
|
|
|
|
|
fLocalToWorld.Reset();
|
|
|
|
fWorldToLocal.Reset();
|
|
|
|
|
|
|
|
fVisSet.SetBit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
plDrawableSpans::~plDrawableSpans()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
for( i = 0; i < fGroups.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
delete fGroups[ i ];
|
|
|
|
}
|
|
|
|
fGroups.Reset();
|
|
|
|
|
|
|
|
/// Loop and delete both our types of spans
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
fSpans[ i ]->Destroy();
|
|
|
|
fSpans.Reset();
|
|
|
|
fIcicles.Reset();
|
|
|
|
fParticleSpans.Reset();
|
|
|
|
|
|
|
|
for( i = 0; i < fSourceSpans.GetCount(); i++ )
|
|
|
|
delete fSourceSpans[ i ];
|
|
|
|
fSourceSpans.Reset();
|
|
|
|
|
|
|
|
/// Loop and unref our materials
|
|
|
|
if( GetKey() != nil )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fMaterials.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fMaterials[ i ] != nil && fMaterials[ i ]->GetKey() != nil )
|
|
|
|
GetKey()->Release( fMaterials[ i ]->GetKey() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fMaterials.Reset();
|
|
|
|
|
|
|
|
delete fSpaceTree;
|
|
|
|
|
|
|
|
for( i = 0; i < fDIIndices.GetCount(); i++ )
|
|
|
|
delete fDIIndices[ i ];
|
|
|
|
fDIIndices.Reset();
|
|
|
|
|
|
|
|
if( fRegisteredForRecreate )
|
|
|
|
plgDispatch::Dispatch()->UnRegisterForExactType( plDeviceRecreateMsg::Index(), GetKey() );
|
|
|
|
|
|
|
|
if( fRegisteredForRender )
|
|
|
|
plgDispatch::Dispatch()->UnRegisterForExactType( plRenderMsg::Index(), GetKey() );
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::SetKey(plKey k)
|
|
|
|
{
|
|
|
|
hsKeyedObject::SetKey(k);
|
|
|
|
if( k )
|
|
|
|
{
|
|
|
|
fRegisteredForRecreate = true;
|
|
|
|
plgDispatch::Dispatch()->RegisterForExactType(plDeviceRecreateMsg::Index(), GetKey());
|
|
|
|
plgDispatch::Dispatch()->RegisterForExactType(plPipeGeoMakeMsg::Index(), GetKey());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ChangeSceneNode /////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::SetSceneNode( plKey newNode )
|
|
|
|
{
|
|
|
|
plKey curNode=GetSceneNode();
|
|
|
|
if( curNode == newNode )
|
|
|
|
return;
|
|
|
|
if( newNode )
|
|
|
|
{
|
|
|
|
plNodeRefMsg* refMsg = TRACKED_NEW plNodeRefMsg(newNode, plNodeRefMsg::kOnRequest, -1, plNodeRefMsg::kDrawable);
|
|
|
|
hsgResMgr::ResMgr()->SendRef(GetKey(), refMsg, plRefFlags::kActiveRef);
|
|
|
|
}
|
|
|
|
if( curNode )
|
|
|
|
{
|
|
|
|
curNode->Release(GetKey());
|
|
|
|
}
|
|
|
|
fSceneNode = newNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// PrepForRender ///////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::PrepForRender( plPipeline *p )
|
|
|
|
{
|
|
|
|
/// If we're not registered for this message, register for it so we know when
|
|
|
|
/// we need to refresh the buffers
|
|
|
|
if( !fRegisteredForRecreate && GetKey() != nil )
|
|
|
|
{
|
|
|
|
fRegisteredForRecreate = true;
|
|
|
|
plgDispatch::Dispatch()->RegisterForExactType( plDeviceRecreateMsg::Index(), GetKey() );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( !fReadyToRender )
|
|
|
|
{
|
|
|
|
uint32_t i;
|
|
|
|
|
|
|
|
for( i = 0; i < fGroups.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
// Each group will decide whether it needs to be prepped
|
|
|
|
fGroups[ i ]->PrepForRendering( p, false );
|
|
|
|
}
|
|
|
|
|
|
|
|
fReadyToRender = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fParticleSpans.GetCount() )
|
|
|
|
{
|
|
|
|
uint32_t i;
|
|
|
|
#if 0
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[ i ]->fTypeMask & plSpan::kParticleSpan )
|
|
|
|
{
|
|
|
|
plParticleSpan* span = (plParticleSpan*)fSpans[i];
|
|
|
|
|
|
|
|
plParticleFiller::FillParticles(p, this, span);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for( i = 0; i < fParticleSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plParticleFiller::FillParticles(p, this, &fParticleSpans[i]);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::SetDISpanVisSet(uint32_t diIndex, hsKeyedObject* ref, hsBool on)
|
|
|
|
{
|
|
|
|
// Could actually do something here to neutralize bones, but we're not.
|
|
|
|
// Main thing is that if it's Matrix Only, then the indices are into
|
|
|
|
// the LocalToWorlds, not into fSpans
|
|
|
|
if( fDIIndices[diIndex]->IsMatrixOnly() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
plVisRegion* reg = plVisRegion::ConvertNoRef(ref);
|
|
|
|
if( !reg )
|
|
|
|
return;
|
|
|
|
hsBool isNot = reg->GetProperty(plVisRegion::kIsNot);
|
|
|
|
uint32_t visRegIndex = reg->GetIndex();
|
|
|
|
|
|
|
|
if( isNot )
|
|
|
|
{
|
|
|
|
fVisNot.SetBit(visRegIndex, on);
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fDIIndices[ diIndex ]->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
int spanIndex = (*fDIIndices[diIndex])[i];
|
|
|
|
fSpans[spanIndex]->SetVisNot(visRegIndex, on);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fVisSet.SetBit(visRegIndex, on);
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fDIIndices[ diIndex ]->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
int spanIndex = (*fDIIndices[diIndex])[i];
|
|
|
|
fSpans[spanIndex]->SetVisBit(visRegIndex, on);
|
|
|
|
|
|
|
|
// HACKAGE
|
|
|
|
// We need to be more careful about when we set and clear this bit,
|
|
|
|
// but not today.
|
|
|
|
// Okay, today. mf
|
|
|
|
if( reg->ReplaceNormal() )
|
|
|
|
fSpans[spanIndex]->SetVisBit(plVisMgr::kNormal, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::SetVisSet(plVisMgr* visMgr)
|
|
|
|
{
|
|
|
|
plProfile_Extern(VisSelect);
|
|
|
|
plProfile_BeginTiming(VisSelect);
|
|
|
|
if( visMgr )
|
|
|
|
{
|
|
|
|
const hsBitVector& visSet = visMgr->GetVisSet();
|
|
|
|
const hsBitVector& visNot = visMgr->GetVisNot();
|
|
|
|
|
|
|
|
// Go through some contortions to not be new[]'ing and delete[]'ing.
|
|
|
|
static hsBitVector myVis;
|
|
|
|
static hsBitVector myNot;
|
|
|
|
|
|
|
|
// myVis = (visSet ^ fLastVisSet) & fVisSet;
|
|
|
|
myVis = visSet;
|
|
|
|
myVis ^= fLastVisSet;
|
|
|
|
myVis &= fVisSet;
|
|
|
|
|
|
|
|
// myNot = (visNot ^ fLastVisNot) & fVisNot;
|
|
|
|
myNot = visNot;
|
|
|
|
myNot ^= fLastVisNot;
|
|
|
|
myNot &= fVisNot;
|
|
|
|
|
|
|
|
// myVis = myVis | myNot
|
|
|
|
myVis |= myNot;
|
|
|
|
|
|
|
|
GetSpaceTree()->SetCache(&fVisCache);
|
|
|
|
if( !myVis.Empty() )
|
|
|
|
{
|
|
|
|
fVisCache.Clear();
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( !fSpans[i]->GetVisNot().Overlap(visNot) && fSpans[i]->GetVisSet().Overlap(visSet) )
|
|
|
|
{
|
|
|
|
GetSpaceTree()->EnableLeaf(i, fVisCache);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fLastVisSet = visSet;
|
|
|
|
fLastVisNot = visNot;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GetSpaceTree()->SetCache(nil);
|
|
|
|
}
|
|
|
|
plProfile_EndTiming(VisSelect);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Here's the problem. When we update one of the matrices in the matrix palette,
|
|
|
|
// we've invalidated the current bounds of all the spans that use it. The skinning
|
|
|
|
// matrix may have moved the spans, or may just have stretched them, but either
|
|
|
|
// way, they aren't right any more. Wait, it gets worse. We're going to have a bunch
|
|
|
|
// of SetTransform calls come in, and then some time later, without our knowledge,
|
|
|
|
// our bounds (via the space tree and PageTreeMgr) will be used to evaluate whether
|
|
|
|
// to bother drawing us. Clearly, if we're going to keep our bounds up to date, we
|
|
|
|
// want to wait until all the SetTransforms have come in, and then update our bounds
|
|
|
|
// all at once, but before the visibility determination phase kicks in.
|
|
|
|
// The alternative is to make the local bounds conservative. We bloat out the local
|
|
|
|
// bounds on export so that whereever the skinning matrices push us around, we're
|
|
|
|
// still inside the local bounds. This is certainly cheaper to compute (all computation
|
|
|
|
// is offline), but gives a less tight bounds. Whether the conservative bounds is
|
|
|
|
// tight enough will depend on how the skinning is actually being used.
|
|
|
|
//
|
|
|
|
// Therefore I'm deferring decision on which way to go, and putting in this temp hack
|
|
|
|
// which does it the dumbest way possible while still being correct.
|
|
|
|
//
|
|
|
|
// So recap:
|
|
|
|
// If we need tight up-to-date bounds, we'll probably send a system message between
|
|
|
|
// the update phase (Eval Msg) and the render phase, only marking as dirty the changed
|
|
|
|
// palette matrices.
|
|
|
|
// If conservative bounds are good enough, we'll need to compute those from the skinning
|
|
|
|
// animations offline.
|
|
|
|
void plDrawableSpans::IUpdateMatrixPaletteBoundsHack()
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[i]->fNumMatrices && !(fSpans[i]->fProps & plSpan::kPropNoDraw) )
|
|
|
|
{
|
|
|
|
hsBounds3Ext bnd = fSpans[i]->fLocalBounds;
|
|
|
|
const hsMatrix44& xfm0 = fSpans[i]->fMaxBoneIdx ? fLocalToWorlds[fSpans[i]->fBaseMatrix + fSpans[i]->fMaxBoneIdx] : fSpans[i]->fLocalToWorld;
|
|
|
|
bnd.Transform(&xfm0);
|
|
|
|
fSpans[i]->fWorldBounds = bnd;
|
|
|
|
|
|
|
|
bnd = fSpans[i]->fLocalBounds;
|
|
|
|
const hsMatrix44& xfm1 = fSpans[i]->fPenBoneIdx ? fLocalToWorlds[fSpans[i]->fBaseMatrix + fSpans[i]->fPenBoneIdx] : fSpans[i]->fLocalToWorld;
|
|
|
|
bnd.Transform(&xfm1);
|
|
|
|
fSpans[i]->fWorldBounds.Union(&bnd);
|
|
|
|
|
|
|
|
GetSpaceTree()->MoveLeaf(i, fSpans[i]->fWorldBounds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::IBoundsInvalid(const hsBounds3Ext& bnd) const
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < 3; i++ )
|
|
|
|
{
|
|
|
|
const float kLimit(1.e5f);
|
|
|
|
|
|
|
|
if( bnd.GetMaxs()[i] > kLimit )
|
|
|
|
return true;
|
|
|
|
if( bnd.GetMins()[i] < -kLimit )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// SetTransform ////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
static inline hsMatrix44 IMatrixMul34(const hsMatrix44& lhs, const hsMatrix44& rhs)
|
|
|
|
{
|
|
|
|
hsMatrix44 ret;
|
|
|
|
ret.NotIdentity();
|
|
|
|
ret.fMap[3][0] = ret.fMap[3][1] = ret.fMap[3][2] = 0;
|
|
|
|
ret.fMap[3][3] = 1.f;
|
|
|
|
|
|
|
|
ret.fMap[0][0] = lhs.fMap[0][0] * rhs.fMap[0][0]
|
|
|
|
+ lhs.fMap[0][1] * rhs.fMap[1][0]
|
|
|
|
+ lhs.fMap[0][2] * rhs.fMap[2][0];
|
|
|
|
|
|
|
|
ret.fMap[0][1] = lhs.fMap[0][0] * rhs.fMap[0][1]
|
|
|
|
+ lhs.fMap[0][1] * rhs.fMap[1][1]
|
|
|
|
+ lhs.fMap[0][2] * rhs.fMap[2][1];
|
|
|
|
|
|
|
|
ret.fMap[0][2] = lhs.fMap[0][0] * rhs.fMap[0][2]
|
|
|
|
+ lhs.fMap[0][1] * rhs.fMap[1][2]
|
|
|
|
+ lhs.fMap[0][2] * rhs.fMap[2][2];
|
|
|
|
|
|
|
|
ret.fMap[0][3] = lhs.fMap[0][0] * rhs.fMap[0][3]
|
|
|
|
+ lhs.fMap[0][1] * rhs.fMap[1][3]
|
|
|
|
+ lhs.fMap[0][2] * rhs.fMap[2][3]
|
|
|
|
+ lhs.fMap[0][3];
|
|
|
|
|
|
|
|
ret.fMap[1][0] = lhs.fMap[1][0] * rhs.fMap[0][0]
|
|
|
|
+ lhs.fMap[1][1] * rhs.fMap[1][0]
|
|
|
|
+ lhs.fMap[1][2] * rhs.fMap[2][0];
|
|
|
|
|
|
|
|
ret.fMap[1][1] = lhs.fMap[1][0] * rhs.fMap[0][1]
|
|
|
|
+ lhs.fMap[1][1] * rhs.fMap[1][1]
|
|
|
|
+ lhs.fMap[1][2] * rhs.fMap[2][1];
|
|
|
|
|
|
|
|
ret.fMap[1][2] = lhs.fMap[1][0] * rhs.fMap[0][2]
|
|
|
|
+ lhs.fMap[1][1] * rhs.fMap[1][2]
|
|
|
|
+ lhs.fMap[1][2] * rhs.fMap[2][2];
|
|
|
|
|
|
|
|
ret.fMap[1][3] = lhs.fMap[1][0] * rhs.fMap[0][3]
|
|
|
|
+ lhs.fMap[1][1] * rhs.fMap[1][3]
|
|
|
|
+ lhs.fMap[1][2] * rhs.fMap[2][3]
|
|
|
|
+ lhs.fMap[1][3];
|
|
|
|
|
|
|
|
ret.fMap[2][0] = lhs.fMap[2][0] * rhs.fMap[0][0]
|
|
|
|
+ lhs.fMap[2][1] * rhs.fMap[1][0]
|
|
|
|
+ lhs.fMap[2][2] * rhs.fMap[2][0];
|
|
|
|
|
|
|
|
ret.fMap[2][1] = lhs.fMap[2][0] * rhs.fMap[0][1]
|
|
|
|
+ lhs.fMap[2][1] * rhs.fMap[1][1]
|
|
|
|
+ lhs.fMap[2][2] * rhs.fMap[2][1];
|
|
|
|
|
|
|
|
ret.fMap[2][2] = lhs.fMap[2][0] * rhs.fMap[0][2]
|
|
|
|
+ lhs.fMap[2][1] * rhs.fMap[1][2]
|
|
|
|
+ lhs.fMap[2][2] * rhs.fMap[2][2];
|
|
|
|
|
|
|
|
ret.fMap[2][3] = lhs.fMap[2][0] * rhs.fMap[0][3]
|
|
|
|
+ lhs.fMap[2][1] * rhs.fMap[1][3]
|
|
|
|
+ lhs.fMap[2][2] * rhs.fMap[2][3]
|
|
|
|
+ lhs.fMap[2][3];
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_CreateCounter("DSSetTrans", "Update", DSSetTrans);
|
|
|
|
plProfile_CreateCounter("DSMatSpans", "Update", DSMatSpans);
|
|
|
|
plProfile_CreateCounter("DSRegSpans", "Update", DSRegSpans);
|
|
|
|
|
|
|
|
plProfile_CreateTimer("DSSetTransT", "Update", DSSetTransT);
|
|
|
|
plProfile_CreateTimer("DSMatTransT", "Update", DSMatTransT);
|
|
|
|
plProfile_CreateTimer("DSRegTransT", "Update", DSRegTransT);
|
|
|
|
plProfile_CreateTimer("DSBndTransT", "Update", DSBndTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
|
|
|
|
|
|
|
|
plDrawable& plDrawableSpans::SetTransform( uint32_t index, const hsMatrix44& l2w, const hsMatrix44& w2l )
|
|
|
|
{
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_IncCount(DSSetTrans, 1);
|
|
|
|
plProfile_BeginTiming(DSSetTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
|
|
|
|
if( index == (uint32_t)-1 )
|
|
|
|
{
|
|
|
|
fLocalToWorld = l2w;
|
|
|
|
fWorldToLocal = w2l;
|
|
|
|
|
|
|
|
fWorldBounds = fLocalBounds;
|
|
|
|
fWorldBounds.Transform( &l2w );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
uint32_t idx;
|
|
|
|
plDISpanIndex *spans = fDIIndices[ index ];
|
|
|
|
|
|
|
|
|
|
|
|
if( spans->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_IncCount(DSMatSpans, spans->GetCount());
|
|
|
|
plProfile_BeginTiming(DSMatTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
for( i = 0; i < spans->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
fLocalToWorlds[ (*spans)[ i ] ] = l2w * fLocalToBones[ (*spans)[ i ] ];
|
|
|
|
fWorldToLocals[ (*spans)[ i ] ] = fBoneToLocals[ (*spans)[ i ] ] * w2l;
|
|
|
|
#else
|
|
|
|
fLocalToWorlds[ (*spans)[ i ] ] = IMatrixMul34(l2w, fLocalToBones[ (*spans)[ i ] ]);
|
|
|
|
fWorldToLocals[ (*spans)[ i ] ] = IMatrixMul34(fBoneToLocals[ (*spans)[ i ] ], w2l);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_EndTiming(DSMatTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
}
|
|
|
|
else if( !spans->DontTransform() )
|
|
|
|
{
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_IncCount(DSRegSpans, spans->GetCount());
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
for( i = 0; i < spans->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_BeginTiming(DSRegTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
|
|
|
|
idx = (*spans)[ i ];
|
|
|
|
plSpan *mSpan = fSpans[ idx ];
|
|
|
|
mSpan->fLocalToWorld = l2w;
|
|
|
|
mSpan->fWorldToLocal = w2l;
|
|
|
|
|
|
|
|
mSpan->fWorldBounds = mSpan->fLocalBounds;
|
|
|
|
mSpan->fWorldBounds.Transform( &l2w );
|
|
|
|
|
|
|
|
if( fSourceSpans.GetCount() > idx )
|
|
|
|
{
|
|
|
|
/// If we have a geoSpan for this, update its transform as well,
|
|
|
|
/// just in case we need to use it later (<cough> SceneViewer reshade <cough>)
|
|
|
|
if( fSourceSpans[ idx ] == nil )
|
|
|
|
{
|
|
|
|
plStatusLog::AddLineS( "pipeline.log", 0xffffffff, "Nil source spans found in SetTransform()" );
|
|
|
|
}
|
|
|
|
|
|
|
|
fSourceSpans[ idx ]->fLocalToWorld = l2w;
|
|
|
|
fSourceSpans[ idx ]->fWorldToLocal = w2l;
|
|
|
|
}
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_EndTiming(DSRegTransT);
|
|
|
|
|
|
|
|
plProfile_BeginTiming(DSBndTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
if( IBoundsInvalid(mSpan->fWorldBounds) )
|
|
|
|
{
|
|
|
|
mSpan->fProps |= kPropNoDraw;
|
|
|
|
GetSpaceTree()->SetLeafFlag((int16_t)idx, plSpaceTreeNode::kDisabled, true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GetSpaceTree()->MoveLeaf((int16_t)((*spans)[i]), mSpan->fWorldBounds);
|
|
|
|
}
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_EndTiming(DSBndTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_BeginTiming(DSBndTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
fWorldBounds = GetSpaceTree()->GetNode(GetSpaceTree()->GetRoot()).GetWorldBounds();
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_EndTiming(DSBndTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MF_TEST_UPDATE
|
|
|
|
plProfile_EndTiming(DSSetTransT);
|
|
|
|
#endif // MF_TEST_UPDATE
|
|
|
|
// Might want to assert that MaxWorldBounds still contains WorldBounds.
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::SetNativeTransform(uint32_t idx, const hsMatrix44& l2w, const hsMatrix44& w2l)
|
|
|
|
{
|
|
|
|
if( idx == uint32_t(-1) )
|
|
|
|
{
|
|
|
|
hsAssert(false, "Invalid index to SetNativeTransform");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
plSpan* span = fSpans[idx];
|
|
|
|
span->fLocalToWorld = l2w;
|
|
|
|
span->fWorldToLocal = w2l;
|
|
|
|
|
|
|
|
span->fWorldBounds = span->fLocalBounds;
|
|
|
|
span->fWorldBounds.Transform(&l2w);
|
|
|
|
|
|
|
|
if( fSourceSpans.GetCount() > idx )
|
|
|
|
{
|
|
|
|
fSourceSpans[idx]->fLocalToWorld = l2w;
|
|
|
|
fSourceSpans[idx]->fWorldToLocal = w2l;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( IBoundsInvalid(span->fWorldBounds) )
|
|
|
|
{
|
|
|
|
SetNativeProperty( idx, kPropNoDraw, true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GetSpaceTree()->MoveLeaf((int16_t)idx, span->fWorldBounds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// GetLocalToWorld & GetWorldToLocal ///////////////////////////////////////
|
|
|
|
|
|
|
|
const hsMatrix44& plDrawableSpans::GetLocalToWorld( uint32_t span ) const
|
|
|
|
{
|
|
|
|
if( span == (uint32_t)-1 )
|
|
|
|
return fLocalToWorld;
|
|
|
|
|
|
|
|
return fSpans[ span ]->fLocalToWorld;
|
|
|
|
}
|
|
|
|
|
|
|
|
const hsMatrix44& plDrawableSpans::GetWorldToLocal( uint32_t span ) const
|
|
|
|
{
|
|
|
|
if( span == (uint32_t)-1 )
|
|
|
|
return fWorldToLocal;
|
|
|
|
|
|
|
|
return fSpans[ span ]->fWorldToLocal;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//// Set/GetNativeProperty ///////////////////////////////////////////////////
|
|
|
|
|
|
|
|
plDrawable& plDrawableSpans::SetNativeProperty( uint32_t index, int prop, hsBool on)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
if( index == (uint32_t)-1 )
|
|
|
|
{
|
|
|
|
hsAssert(false, "Invalid index to SetNativeProperty");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
plDISpanIndex *spans = fDIIndices[ index ];
|
|
|
|
|
|
|
|
if( !spans->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
if( on )
|
|
|
|
{
|
|
|
|
for( i = 0; i < spans->GetCount(); i++ )
|
|
|
|
fSpans[ (*spans)[ i ] ]->fProps |= prop;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( i = 0; i < spans->GetCount(); i++ )
|
|
|
|
fSpans[ (*spans)[ i ] ]->fProps &= ~prop;
|
|
|
|
}
|
|
|
|
if( (prop & kPropNoDraw) )
|
|
|
|
{
|
|
|
|
for( i = 0; i < spans->GetCount(); i++ )
|
|
|
|
GetSpaceTree()->SetLeafFlag((int16_t)((*spans)[ i ]), plSpaceTreeNode::kDisabled, on);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::GetNativeProperty( uint32_t index, int prop ) const
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
uint32_t ret = false;
|
|
|
|
|
|
|
|
|
|
|
|
if( index == (uint32_t)-1 )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
ret |= ( fSpans[ i ]->fProps & prop );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
plDISpanIndex& spans = *fDIIndices[ index ];
|
|
|
|
|
|
|
|
if( !spans.IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( i = 0; i < spans.GetCount(); i++ )
|
|
|
|
ret |= ( fSpans[ spans[ i ] ]->fProps & prop );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
plDrawable& plDrawableSpans::SetSubType(uint32_t index, plSubDrawableType t, hsBool on)
|
|
|
|
{
|
|
|
|
if( uint32_t(-1) == index )
|
|
|
|
{
|
|
|
|
if( on )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
fSpans[i]->fSubType |= t;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
fSpans[i]->fSubType &= ~t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
plDISpanIndex& spans = *fDIIndices[ index ];
|
|
|
|
|
|
|
|
if( on )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < spans.GetCount(); i++ )
|
|
|
|
fSpans[ spans[i] ]->fSubType |= t;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < spans.GetCount(); i++ )
|
|
|
|
fSpans[ spans[i] ]->fSubType &= ~t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::GetSubType(uint32_t index) const
|
|
|
|
{
|
|
|
|
uint32_t retVal = 0;
|
|
|
|
|
|
|
|
if( uint32_t(-1) == index )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
retVal |= fSpans[i]->fSubType;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
plDISpanIndex& spans = *fDIIndices[ index ];
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < spans.GetCount(); i++ )
|
|
|
|
retVal |= fSpans[ spans[i] ]->fSubType;
|
|
|
|
}
|
|
|
|
return retVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IXlateSpanProps /////////////////////////////////////////////////////////
|
|
|
|
// Never used yet--just here in case we ever need it
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::IXlateSpanProps( uint32_t props, hsBool xlateToSpan )
|
|
|
|
{
|
|
|
|
uint32_t retProps = 0;
|
|
|
|
|
|
|
|
|
|
|
|
if( xlateToSpan )
|
|
|
|
{
|
|
|
|
/// Drawable props to plSpan props
|
|
|
|
if( props & kPropNoDraw ) retProps |= plSpan::kPropNoDraw;
|
|
|
|
if( props & kPropSortFaces ) retProps |= plSpan::kPropFacesSortable;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// plSpan props to Drawable props
|
|
|
|
if( props & plSpan::kPropNoDraw ) retProps |= kPropNoDraw;
|
|
|
|
if( props & plSpan::kPropFacesSortable ) retProps |= kPropSortFaces;
|
|
|
|
}
|
|
|
|
|
|
|
|
return retProps;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// Set/GetProperty /////////////////////////////////////////////////////////
|
|
|
|
// Sets/gets a property just like the normal Set/GetNativeProperty, but the
|
|
|
|
// flag taken in is from plDrawInterface, not our props flags. So we have to
|
|
|
|
// translate...
|
|
|
|
|
|
|
|
plDrawable& plDrawableSpans::SetProperty( uint32_t index, int diProp, hsBool on )
|
|
|
|
{
|
|
|
|
switch( diProp )
|
|
|
|
{
|
|
|
|
case plDrawInterface::kDisable:
|
|
|
|
return SetNativeProperty( index, kPropNoDraw, on );
|
|
|
|
default:
|
|
|
|
hsAssert( false, "Bad property passed to SetProperty" );
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::GetProperty( uint32_t index, int diProp ) const
|
|
|
|
{
|
|
|
|
switch( diProp )
|
|
|
|
{
|
|
|
|
case plDrawInterface::kDisable:
|
|
|
|
return GetNativeProperty( index, kPropNoDraw );
|
|
|
|
default:
|
|
|
|
hsAssert( false, "Bad property passed to SetProperty" );
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
plDrawable& plDrawableSpans::SetProperty( int prop, hsBool on )
|
|
|
|
{
|
|
|
|
switch( prop )
|
|
|
|
{
|
|
|
|
case plDrawInterface::kDisable:
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i=0; i<fIcicles.Count(); i++)
|
|
|
|
if (on)
|
|
|
|
fIcicles[i].fProps |= kPropNoDraw;
|
|
|
|
else
|
|
|
|
fIcicles[i].fProps &= ~kPropNoDraw;
|
|
|
|
IQuickSpaceTree();
|
|
|
|
}
|
|
|
|
return SetNativeProperty( kPropNoDraw, on );
|
|
|
|
default:
|
|
|
|
hsAssert( false, "Bad property passed to SetProperty" );
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::GetProperty( int prop ) const
|
|
|
|
{
|
|
|
|
switch( prop )
|
|
|
|
{
|
|
|
|
case plDrawInterface::kDisable:
|
|
|
|
return GetNativeProperty( kPropNoDraw );
|
|
|
|
default:
|
|
|
|
hsAssert( false, "Bad property passed to SetProperty" );
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//// Get*Bounds //////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
const hsBounds3Ext& plDrawableSpans::GetLocalBounds( uint32_t index ) const
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
static hsBounds3Ext bnd;
|
|
|
|
|
|
|
|
|
|
|
|
if( index == (uint32_t)-1 )
|
|
|
|
return fLocalBounds;
|
|
|
|
|
|
|
|
plDISpanIndex *spans = fDIIndices[ index ];
|
|
|
|
|
|
|
|
bnd.MakeEmpty();
|
|
|
|
if( !spans->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( i = 0; i < spans->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
bnd.Union( &fSpans[ (*spans)[ i ] ]->fLocalBounds );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return bnd;
|
|
|
|
}
|
|
|
|
|
|
|
|
const hsBounds3Ext& plDrawableSpans::GetWorldBounds( uint32_t index ) const
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
static hsBounds3Ext bnd;
|
|
|
|
|
|
|
|
|
|
|
|
if( index == (uint32_t)-1 )
|
|
|
|
return fWorldBounds;
|
|
|
|
|
|
|
|
plDISpanIndex *spans = fDIIndices[ index ];
|
|
|
|
|
|
|
|
bnd.MakeEmpty();
|
|
|
|
if( !spans->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( i = 0; i < spans->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
bnd.Union( &fSpans[ (*spans)[ i ] ]->fWorldBounds );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return bnd;
|
|
|
|
}
|
|
|
|
|
|
|
|
const hsBounds3Ext& plDrawableSpans::GetMaxWorldBounds( uint32_t index ) const
|
|
|
|
{
|
|
|
|
return GetWorldBounds( index );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// Read ////////////////////////////////////////////////////////////////////
|
|
|
|
// We read each in the array of icicles,
|
|
|
|
// then we read in an array of indices that we translate into
|
|
|
|
// pointers. Note: since materials and fog environments are shared,
|
|
|
|
// we read those keys last, so we don't have to have separate messages for
|
|
|
|
// each.
|
|
|
|
|
|
|
|
void plDrawableSpans::Read( hsStream* s, hsResMgr* mgr )
|
|
|
|
{
|
|
|
|
uint32_t i, j, count, count2;
|
|
|
|
hsBool gotSkin = false;
|
|
|
|
plGBufferGroup *group;
|
|
|
|
plRefMsg *refMsg;
|
|
|
|
|
|
|
|
|
|
|
|
plDrawable::Read(s, mgr);
|
|
|
|
|
|
|
|
fProps = s->ReadLE32();
|
|
|
|
fCriteria = s->ReadLE32();
|
|
|
|
fRenderLevel.fLevel = s->ReadLE32();
|
|
|
|
|
|
|
|
/// Read in the material keys
|
|
|
|
count = s->ReadLE32();
|
|
|
|
fMaterials.SetCountAndZero( count );
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
refMsg = TRACKED_NEW plGenRefMsg( GetKey(), plRefMsg::kOnCreate, i, kMsgMaterial );
|
|
|
|
mgr->ReadKeyNotifyMe( s, refMsg, plRefFlags::kActiveRef );
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read the icicles in
|
|
|
|
count = s->ReadLE32();
|
|
|
|
fIcicles.SetCount( count );
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
fIcicles[ i ].Read( s );
|
|
|
|
if( fIcicles[ i ].fNumMatrices )
|
|
|
|
gotSkin = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read the patches in
|
|
|
|
// FIXME MAJOR VERSION
|
|
|
|
// no more patches, remove this line
|
|
|
|
count = s->ReadLE32();
|
|
|
|
|
|
|
|
/// Now read the index array in and use it to create a pointer table
|
|
|
|
fSpanSourceIndices.Reset();
|
|
|
|
fSpans.Reset();
|
|
|
|
count = s->ReadLE32();
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
j = s->ReadLE32();
|
|
|
|
switch( j & kSpanTypeMask )
|
|
|
|
{
|
|
|
|
case kSpanTypeIcicle: fSpans.Append( (plSpan *)&fIcicles[ j & kSpanIDMask ] ); break;
|
|
|
|
case kSpanTypeParticleSpan: fSpans.Append( (plSpan *)&fParticleSpans[ j & kSpanIDMask ] ); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
fSpanSourceIndices.Append( j );
|
|
|
|
|
|
|
|
if( fSpans[ fSpans.GetCount() - 1 ]->fTypeMask & plSpan::kParticleSpan )
|
|
|
|
{
|
|
|
|
plParticleSpan *span = (plParticleSpan *)fSpans[ fSpans.GetCount() - 1 ];
|
|
|
|
span->fSrcSpanIdx = fSpans.GetCount() - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rebuild bit vectors for various span types
|
|
|
|
IBuildVectors();
|
|
|
|
|
|
|
|
/// Now that we have our pointer array, read in the common keys (fog environs, etc)
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
// Ref message for the fog environment
|
|
|
|
refMsg = TRACKED_NEW plGenRefMsg( GetKey(), plRefMsg::kOnCreate, i, kMsgFogEnviron );
|
|
|
|
mgr->ReadKeyNotifyMe( s, refMsg, plRefFlags::kActiveRef );
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read in bounds and stuff
|
|
|
|
if( count > 0 )
|
|
|
|
{
|
|
|
|
fLocalBounds.Read(s);
|
|
|
|
fWorldBounds.Read(s);
|
|
|
|
fMaxWorldBounds.Read(s);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fLocalBounds.MakeEmpty();
|
|
|
|
fWorldBounds.MakeEmpty();
|
|
|
|
fMaxWorldBounds.MakeEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[i]->fProps & plSpan::kPropHasPermaLights )
|
|
|
|
{
|
|
|
|
uint32_t lcnt = s->ReadLE32();
|
|
|
|
int j;
|
|
|
|
for( j = 0; j < lcnt; j++ )
|
|
|
|
{
|
|
|
|
mgr->ReadKeyNotifyMe( s, TRACKED_NEW plGenRefMsg(GetKey(), plRefMsg::kOnCreate, i, kMsgPermaLight), plRefFlags::kPassiveRef);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if( fSpans[i]->fProps & plSpan::kPropHasPermaProjs )
|
|
|
|
{
|
|
|
|
uint32_t lcnt = s->ReadLE32();
|
|
|
|
int j;
|
|
|
|
for( j = 0; j < lcnt; j++ )
|
|
|
|
{
|
|
|
|
mgr->ReadKeyNotifyMe( s, TRACKED_NEW plGenRefMsg(GetKey(), plRefMsg::kOnCreate, i, kMsgPermaProj), plRefFlags::kPassiveRef);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read in the source spans if necessary
|
|
|
|
count = s->ReadLE32();
|
|
|
|
if( count > 0 )
|
|
|
|
{
|
|
|
|
fSourceSpans.SetCount( count );
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
fSourceSpans[ i ] = TRACKED_NEW plGeometrySpan;
|
|
|
|
fSourceSpans[ i ]->Read( s );
|
|
|
|
fSourceSpans[ i ]->fMaterial = GetMaterial( fSpans[ i ]->fMaterialIdx );
|
|
|
|
fSourceSpans[ i ]->fFogEnviron = fSpans[ i ]->fFogEnvironment;
|
|
|
|
fSourceSpans[ i ]->fSpanRefIndex = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
fSourceSpans.Reset();
|
|
|
|
|
|
|
|
/// Read in the matrix palette (if any)
|
|
|
|
count = s->ReadLE32();
|
|
|
|
fLocalToWorlds.SetCount(count);
|
|
|
|
fWorldToLocals.SetCount(count);
|
|
|
|
fLocalToBones.SetCount(count);
|
|
|
|
fBoneToLocals.SetCount(count);
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
fLocalToWorlds[i].Read(s);
|
|
|
|
fWorldToLocals[i].Read(s);
|
|
|
|
|
|
|
|
fLocalToBones[i].Read(s);
|
|
|
|
fBoneToLocals[i].Read(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read in the drawInterface index arrays
|
|
|
|
count = s->ReadLE32();
|
|
|
|
fDIIndices.SetCountAndZero( count );
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
fDIIndices[ i ] = TRACKED_NEW plDISpanIndex;
|
|
|
|
|
|
|
|
fDIIndices[ i ]->fFlags = (uint8_t)(s->ReadLE32());
|
|
|
|
count2 = s->ReadLE32();
|
|
|
|
fDIIndices[ i ]->SetCountAndZero( count2 );
|
|
|
|
for( j = 0; j < count2; j++ )
|
|
|
|
(*fDIIndices[ i ])[ j ] = s->ReadLE32();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read the groups in
|
|
|
|
count = s->ReadLE32();
|
|
|
|
while( count-- )
|
|
|
|
{
|
|
|
|
group = TRACKED_NEW plGBufferGroup(0, fProps & kPropVolatile, fProps & kPropSortFaces);
|
|
|
|
group->Read( s );
|
|
|
|
|
|
|
|
fGroups.Append( group );
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
IMakeSpanSortable(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Other stuff now
|
|
|
|
fSpaceTree = plSpaceTree::ConvertNoRef(mgr->ReadCreatable(s));
|
|
|
|
|
|
|
|
fSceneNode = mgr->ReadKey(s);
|
|
|
|
plNodeRefMsg* nRefMsg = TRACKED_NEW plNodeRefMsg(fSceneNode, plRefMsg::kOnCreate, -1, plNodeRefMsg::kDrawable);
|
|
|
|
mgr->AddViaNotify(GetKey(), nRefMsg, plRefFlags::kActiveRef);
|
|
|
|
|
|
|
|
if( GetNativeProperty(plDrawable::kPropCharacter) )
|
|
|
|
{
|
|
|
|
fVisSet.SetBit(plVisMgr::kCharacter, true);
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
fSpans[i]->SetVisBit(plVisMgr::kCharacter, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Placeholder hack - see IUpdateMatrixPaletteBoundsHack() for comments
|
|
|
|
if( gotSkin )
|
|
|
|
{
|
|
|
|
fRegisteredForRender = true;
|
|
|
|
plgDispatch::Dispatch()->RegisterForExactType( plRenderMsg::Index(), GetKey() );
|
|
|
|
}
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ITestMatForSpecularity //////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::ITestMatForSpecularity( hsGMaterial *mat )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
for( i = 0; i < mat->GetNumLayers(); i++ )
|
|
|
|
{
|
|
|
|
if( mat->GetLayer( i )->GetShadeFlags() && hsGMatState::kShadeSpecular )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#include "plProfile.h"
|
|
|
|
plProfile_CreateTimer("MatrixPalleteHack", "RenderSetup", PalletteHack);
|
|
|
|
//// MsgReceive //////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::MsgReceive( plMessage* msg )
|
|
|
|
{
|
|
|
|
plGenRefMsg *refMsg = plGenRefMsg::ConvertNoRef( msg );
|
|
|
|
int i;
|
|
|
|
hsBool hasSpec;
|
|
|
|
|
|
|
|
|
|
|
|
if( refMsg )
|
|
|
|
{
|
|
|
|
if( refMsg->fType == kMsgMaterial )
|
|
|
|
{
|
|
|
|
/// Material add/remove on this drawable
|
|
|
|
if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest|plRefMsg::kOnReplace) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fMaterials.GetCount(), "Invalid material index" );
|
|
|
|
|
|
|
|
fMaterials[ refMsg->fWhich ] = hsGMaterial::ConvertNoRef( refMsg->GetRef() );
|
|
|
|
|
|
|
|
if( !fSettingMatIdxLock )
|
|
|
|
{
|
|
|
|
// Now find all spans with this material and mark them as using or not using specular
|
|
|
|
hasSpec = ITestMatForSpecularity( fMaterials[ refMsg->fWhich ] );
|
|
|
|
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[ i ] != nil && fSpans[ i ]->fMaterialIdx == refMsg->fWhich )
|
|
|
|
{
|
|
|
|
if( hasSpec )
|
|
|
|
fSpans[ i ]->fProps |= plSpan::kPropMatHasSpecular;
|
|
|
|
else
|
|
|
|
fSpans[ i ]->fProps &= ~plSpan::kPropMatHasSpecular;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( refMsg->GetContext() & (plRefMsg::kOnDestroy|plRefMsg::kOnRemove) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fMaterials.GetCount(), "Invalid material index" );
|
|
|
|
|
|
|
|
fMaterials[ refMsg->fWhich ] = nil;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if( refMsg->fType == kMsgFogEnviron )
|
|
|
|
{
|
|
|
|
/// Fog environment add/remove on this drawable
|
|
|
|
if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest|plRefMsg::kOnReplace) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fSpans.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
|
|
|
|
fSpans[ refMsg->fWhich ]->fFogEnvironment = plFogEnvironment::ConvertNoRef( refMsg->GetRef() );
|
|
|
|
}
|
|
|
|
else if( refMsg->GetContext() & (plRefMsg::kOnDestroy|plRefMsg::kOnRemove) )
|
|
|
|
{
|
|
|
|
uint32_t i;
|
|
|
|
plFogEnvironment *fog = plFogEnvironment::ConvertNoRef( refMsg->GetRef() );
|
|
|
|
|
|
|
|
for( i = 0; i < GetNumSpans(); i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[ i ]->fFogEnvironment == fog )
|
|
|
|
{
|
|
|
|
fSpans[ i ]->fFogEnvironment = nil;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if( refMsg->fType == kMsgPermaLight )
|
|
|
|
{
|
|
|
|
if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest|plRefMsg::kOnReplace) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fSpans.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = plLightInfo::ConvertNoRef(refMsg->GetRef());
|
|
|
|
fSpans[refMsg->fWhich]->AddPermaLight(li, false);
|
|
|
|
}
|
|
|
|
else if( refMsg->GetContext() & (plRefMsg::kOnDestroy|plRefMsg::kOnRemove) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fSpans.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = (plLightInfo*)refMsg->GetRef();
|
|
|
|
fSpans[refMsg->fWhich]->RemovePermaLight(li, false);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if( refMsg->fType == kMsgPermaProj )
|
|
|
|
{
|
|
|
|
if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest|plRefMsg::kOnReplace) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fSpans.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = plLightInfo::ConvertNoRef(refMsg->GetRef());
|
|
|
|
fSpans[refMsg->fWhich]->AddPermaLight(li, true);
|
|
|
|
}
|
|
|
|
else if( refMsg->GetContext() & (plRefMsg::kOnDestroy|plRefMsg::kOnRemove) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fSpans.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = (plLightInfo*)refMsg->GetRef();
|
|
|
|
fSpans[refMsg->fWhich]->RemovePermaLight(li, true);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if( refMsg->fType == kMsgPermaLightDI )
|
|
|
|
{
|
|
|
|
if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest|plRefMsg::kOnReplace) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fDIIndices.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = plLightInfo::ConvertNoRef(refMsg->GetRef());
|
|
|
|
|
|
|
|
int diIndex = int(refMsg->fWhich);
|
|
|
|
if( (diIndex >= 0)
|
|
|
|
&& !fDIIndices[ diIndex ]->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fDIIndices[ diIndex ]->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
int spanIndex = (*fDIIndices[diIndex])[i];
|
|
|
|
fSpans[spanIndex]->AddPermaLight(li, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( refMsg->GetContext() & (plRefMsg::kOnDestroy|plRefMsg::kOnRemove) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fDIIndices.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = plLightInfo::ConvertNoRef(refMsg->GetRef());
|
|
|
|
|
|
|
|
int diIndex = int(refMsg->fWhich);
|
|
|
|
if( (diIndex >= 0)
|
|
|
|
&& !fDIIndices[ diIndex ]->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fDIIndices[ diIndex ]->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
int spanIndex = (*fDIIndices[diIndex])[i];
|
|
|
|
fSpans[spanIndex]->RemovePermaLight(li, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if( refMsg->fType == kMsgPermaProjDI )
|
|
|
|
{
|
|
|
|
if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest|plRefMsg::kOnReplace) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fDIIndices.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = plLightInfo::ConvertNoRef(refMsg->GetRef());
|
|
|
|
|
|
|
|
int diIndex = int(refMsg->fWhich);
|
|
|
|
if( (diIndex >= 0)
|
|
|
|
&& !fDIIndices[ diIndex ]->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fDIIndices[ diIndex ]->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
int spanIndex = (*fDIIndices[diIndex])[i];
|
|
|
|
fSpans[spanIndex]->AddPermaLight(li, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( refMsg->GetContext() & (plRefMsg::kOnDestroy|plRefMsg::kOnRemove) )
|
|
|
|
{
|
|
|
|
hsAssert( refMsg->fWhich < fDIIndices.GetCount(), "Shesh, send us valid data, will ya??" );
|
|
|
|
plLightInfo* li = plLightInfo::ConvertNoRef(refMsg->GetRef());
|
|
|
|
|
|
|
|
int diIndex = int(refMsg->fWhich);
|
|
|
|
if( (diIndex >= 0)
|
|
|
|
&& !fDIIndices[ diIndex ]->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fDIIndices[ diIndex ]->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
int spanIndex = (*fDIIndices[diIndex])[i];
|
|
|
|
fSpans[spanIndex]->RemovePermaLight(li, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( plDeviceRecreateMsg::ConvertNoRef( msg ) != nil )
|
|
|
|
{
|
|
|
|
/// Device recreation message--just reset our flag so we refresh buffer groups
|
|
|
|
fReadyToRender = false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if( plRenderMsg::ConvertNoRef( msg ) )
|
|
|
|
{
|
|
|
|
plProfile_BeginLap(PalletteHack, this->GetKey()->GetUoid().GetObjectName());
|
|
|
|
|
|
|
|
IUpdateMatrixPaletteBoundsHack();
|
|
|
|
|
|
|
|
// Last thing here. We have a bit list of which of our spans are skinned and
|
|
|
|
// thus need to be re-blended each frame. However, we don't want to blend them
|
|
|
|
// multiple times per frame if at all possible. Since the pipeline already checks
|
|
|
|
// our bitfield, what we *really* do is make a copy and give the pipeline our copy.
|
|
|
|
// The pipeline will then clear out those bits as it blends them, and then we simply
|
|
|
|
// re-set them here, since plRenderMsg is sent once before all the rendering is done :)
|
|
|
|
fFakeBlendingSpanVector = fBlendingSpanVector;
|
|
|
|
plProfile_EndLap(PalletteHack, this->GetKey()->GetUoid().GetObjectName());
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
plDISpansMsg* diMsg = plDISpansMsg::ConvertNoRef(msg);
|
|
|
|
if( diMsg )
|
|
|
|
{
|
|
|
|
if( diMsg->fType == plDISpansMsg::kRemovingSpan )
|
|
|
|
{
|
|
|
|
// If the only set of spans we've got is about to be removed,
|
|
|
|
// (and we're not flagged to stick around) then just
|
|
|
|
// kill ourselves entirely.
|
|
|
|
if( fDIIndices.GetCount() < 2 && !(diMsg->fFlags & plDISpansMsg::kLeaveEmptyDrawable) )
|
|
|
|
{
|
|
|
|
hsAssert(diMsg->fIndex + 1 == fDIIndices.GetCount(), "Deleting the an unknown set of indices");
|
|
|
|
if( GetSceneNode() )
|
|
|
|
{
|
|
|
|
GetSceneNode()->Release(GetKey());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else /// plDrawInterface telling us to remove some spans
|
|
|
|
{
|
|
|
|
RemoveDISpans( (int32_t)diMsg->fIndex );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef HS_DEBUGGING
|
|
|
|
else if( diMsg->fType == plDISpansMsg::kAddingSpan )
|
|
|
|
{
|
|
|
|
/// plDrawInterface telling us which spans it owns
|
|
|
|
int32_t i, spanIndex = (int32_t)diMsg->fIndex;
|
|
|
|
|
|
|
|
if( spanIndex == -1 )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
for( i = 0; i < fDIIndices[ spanIndex ]->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( !fDIIndices[ spanIndex ]->IsMatrixOnly() )
|
|
|
|
fSpans[ (*fDIIndices[ spanIndex])[ i ] ]->fOwnerKey = diMsg->GetSender();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
plPipeGeoMakeMsg* make = plPipeGeoMakeMsg::ConvertNoRef(msg);
|
|
|
|
if( make )
|
|
|
|
{
|
|
|
|
fReadyToRender = false;
|
|
|
|
PrepForRender(make->Pipeline());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return plDrawable::MsgReceive(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::SetRenderLevel(const plRenderLevel& l)
|
|
|
|
{
|
|
|
|
fRenderLevel = l;
|
|
|
|
}
|
|
|
|
|
|
|
|
const plRenderLevel& plDrawableSpans::GetRenderLevel() const
|
|
|
|
{
|
|
|
|
return fRenderLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//// DoIMatch ////////////////////////////////////////////////////////////////
|
|
|
|
// Called by the sceneNode to determine if we match the criteria
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::DoIMatch( const plDrawableCriteria& crit )
|
|
|
|
{
|
|
|
|
if( crit.fCriteria ^ fCriteria )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if( crit.fLevel != fRenderLevel )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if( crit.fType != fType )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if( crit.fLoadMask != fLoadMask )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// SetCriteria /////////////////////////////////////////////////////////////
|
|
|
|
// Sets the criteria that this ice matches to. Needed since some ice will
|
|
|
|
// be static per sceneNode while others wont, etc.
|
|
|
|
|
|
|
|
void plDrawableSpans::SetCriteria( const plDrawableCriteria& crit )
|
|
|
|
{
|
|
|
|
// Very simple right now. May be more complicated later...
|
|
|
|
fCriteria = crit.fCriteria;
|
|
|
|
fRenderLevel = crit.fLevel;
|
|
|
|
fType = crit.fType;
|
|
|
|
fLoadMask = crit.fLoadMask;
|
|
|
|
|
|
|
|
if( fCriteria & kCritSortSpans )
|
|
|
|
fProps |= kPropSortSpans;
|
|
|
|
if( fCriteria & kCritSortFaces )
|
|
|
|
fProps |= kPropSortFaces;
|
|
|
|
if( fCriteria & kCritCharacter )
|
|
|
|
fProps |= kPropCharacter;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IQuickSpaceTree ///////////////////////////////////////////////////
|
|
|
|
// Creates a fast, space tree for use at run-time (like in the
|
|
|
|
// SceneViewer). Any time a space tree is requested but there isn't
|
|
|
|
// one available, this will be supplied. This is a full featured
|
|
|
|
// hierarchical bounds which is pretty fast to compute. A more highly
|
|
|
|
// optimized version may be plugged into the Optimize function at a later
|
|
|
|
// date if this one doesn't perform enough (it does so far).
|
|
|
|
|
|
|
|
void plDrawableSpans::IQuickSpaceTree( void ) const
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
// Make the space tree (hierarchical bounds).
|
|
|
|
plSpaceTreeMaker maker;
|
|
|
|
maker.Reset();
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
maker.AddLeaf( fSpans[ i ]->fWorldBounds, fSpans[ i ]->fProps & plSpan::kPropNoDraw );
|
|
|
|
}
|
|
|
|
plSpaceTree* tree = maker.MakeTree();
|
|
|
|
SetSpaceTree(tree);
|
|
|
|
}
|
|
|
|
|
|
|
|
//// SetSpaceTree ////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::SetSpaceTree( plSpaceTree *st ) const
|
|
|
|
{
|
|
|
|
delete fSpaceTree;
|
|
|
|
fSpaceTree = st;
|
|
|
|
fLastVisSet.Clear();
|
|
|
|
fLastVisNot.Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// GetDISpans //////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
plDISpanIndex& plDrawableSpans::GetDISpans( uint32_t index ) const
|
|
|
|
{
|
|
|
|
return *fDIIndices[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
//// GetVertex/IndexRef //////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsGDeviceRef *plDrawableSpans::GetVertexRef( uint32_t group, uint32_t idx )
|
|
|
|
{
|
|
|
|
return fGroups[ group ]->GetVertexBufferRef( idx );
|
|
|
|
}
|
|
|
|
|
|
|
|
hsGDeviceRef *plDrawableSpans::GetIndexRef( uint32_t group, uint32_t idx )
|
|
|
|
{
|
|
|
|
return fGroups[ group ]->GetIndexBufferRef( idx );
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::DirtyVertexBuffer(uint32_t group, uint32_t idx)
|
|
|
|
{
|
|
|
|
hsAssert(group < fGroups.GetCount(), "Dirtying vtx buffer I don't have");
|
|
|
|
GetBufferGroup(group)->DirtyVertexBuffer(idx);
|
|
|
|
|
|
|
|
SetNotReadyToRender();
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::DirtyIndexBuffer(uint32_t group, uint32_t idx)
|
|
|
|
{
|
|
|
|
hsAssert(group < fGroups.GetCount(), "Dirtying index buffer I don't have");
|
|
|
|
GetBufferGroup(group)->DirtyIndexBuffer(idx);
|
|
|
|
|
|
|
|
SetNotReadyToRender();
|
|
|
|
}
|
|
|
|
|
|
|
|
hsGMaterial* plDrawableSpans::GetSubMaterial(int index) const
|
|
|
|
{
|
|
|
|
return GetMaterial(fSpans[index]->fMaterialIdx);
|
|
|
|
}
|
|
|
|
|
|
|
|
// return true if span invisible before minDist and/or after maxDist
|
|
|
|
hsBool plDrawableSpans::GetSubVisDists(int index, float& minDist, float& maxDist) const
|
|
|
|
{
|
|
|
|
return (minDist = fSpans[index]->GetMinDist()) < (maxDist = fSpans[index]->GetMaxDist());
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
//// Runtime Dynamics ////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
//// SortSpan ////////////////////////////////////////////////////////////////
|
|
|
|
// Given the index of the span to sort, sorts the triangles of that span
|
|
|
|
// based on the sorting data. Note: the span MUST be of type plIcicle.
|
|
|
|
|
|
|
|
plProfile_CreateTimer("Face Sort", "Draw", FaceSort);
|
|
|
|
plProfile_CreateCounter("Face Sort Calls", "Draw", FaceSortCalls);
|
|
|
|
plProfile_CreateCounter("Faces Sorted", "Draw", FacesSorted);
|
|
|
|
|
|
|
|
void plDrawableSpans::SortSpan( uint32_t index, plPipeline *pipe )
|
|
|
|
{
|
|
|
|
plProfile_Inc(FaceSortCalls);
|
|
|
|
|
|
|
|
plProfile_BeginLap(FaceSort, "0");
|
|
|
|
|
|
|
|
plIcicle *span = (plIcicle *)fSpans[ index ];
|
|
|
|
plGBufferTriangle *list, temp;
|
|
|
|
uint32_t numTris;
|
|
|
|
int i;
|
|
|
|
hsMatrix44 w2cMatrix = pipe->GetWorldToCamera() * pipe->GetLocalToWorld();
|
|
|
|
float dist;
|
|
|
|
|
|
|
|
ICheckSpanForSortable(index);
|
|
|
|
|
|
|
|
static hsTArray<hsRadixSort::Elem> sortList;
|
|
|
|
static hsTArray<uint16_t> tempTriList;
|
|
|
|
hsRadixSort::Elem *elem;
|
|
|
|
|
|
|
|
|
|
|
|
/// Get some stuff
|
|
|
|
list = span->fSortData;
|
|
|
|
numTris = span->fILength / 3;
|
|
|
|
|
|
|
|
plProfile_IncCount(FacesSorted, numTris);
|
|
|
|
|
|
|
|
hsAssert( numTris > 0, "How could we start sorting no triangles??" );
|
|
|
|
|
|
|
|
/// Sort the triangles in "list"
|
|
|
|
sortList.SetCount( numTris );
|
|
|
|
tempTriList.SetCount( numTris * 3 );
|
|
|
|
elem = sortList.AcquireArray();
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "0");
|
|
|
|
plProfile_BeginLap(FaceSort, "1");
|
|
|
|
|
|
|
|
hsVector3 vec(w2cMatrix.fMap[2][0], w2cMatrix.fMap[2][1], w2cMatrix.fMap[2][2]);
|
|
|
|
float trans = w2cMatrix.fMap[2][3];
|
|
|
|
|
|
|
|
// Fill out the radix sort elements with our data
|
|
|
|
for( i = 0; i < numTris; i++ )
|
|
|
|
{
|
|
|
|
dist = vec.InnerProduct(list[ i ].fCenter) + trans;
|
|
|
|
elem[ i ].fKey.fFloat = dist;
|
|
|
|
elem[ i ].fBody = &list[ i ];
|
|
|
|
elem[ i ].fNext = elem + i + 1;
|
|
|
|
}
|
|
|
|
elem[ i - 1 ].fNext = nil;
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "1");
|
|
|
|
plProfile_BeginLap(FaceSort, "2");
|
|
|
|
|
|
|
|
// Do da sort thingy
|
|
|
|
hsRadixSort rad;
|
|
|
|
hsRadixSort::Elem *sortedList = rad.Sort( elem, 0 );
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "2");
|
|
|
|
plProfile_BeginLap(FaceSort, "3");
|
|
|
|
|
|
|
|
uint16_t* indices = tempTriList.AcquireArray();
|
|
|
|
// Stuff into the temp array
|
|
|
|
for( i = 0, elem = sortedList; i < numTris; i++ )
|
|
|
|
{
|
|
|
|
*indices++ = ((plGBufferTriangle *)elem->fBody)->fIndex1;
|
|
|
|
*indices++ = ((plGBufferTriangle *)elem->fBody)->fIndex2;
|
|
|
|
*indices++ = ((plGBufferTriangle *)elem->fBody)->fIndex3;
|
|
|
|
elem = elem->fNext;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "3");
|
|
|
|
plProfile_BeginLap(FaceSort, "4");
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
numTris, tempTriList.AcquireArray() );
|
|
|
|
|
|
|
|
/// Optional step in a way: copy back our new, sorted list to our original
|
|
|
|
/// array. This lets us do less sorting next call, since the order should
|
|
|
|
/// remain largely unchanged from the last call. If this turns out NOT to
|
|
|
|
/// be the case, or if the sorting step takes less time than this copy,
|
|
|
|
/// take this out!
|
|
|
|
// memcpy( list, tempTriList.AcquireArray(), numTris * sizeof( plGBufferTriangle ) );
|
|
|
|
|
|
|
|
/// All done! (force buffer groups to refresh during next render call)
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "4");
|
|
|
|
}
|
|
|
|
|
|
|
|
//// SortVisibleSpans ////////////////////////////////////////////////////////
|
|
|
|
// Sorts the visible spans's triangles in one big lump, for proper back-to-
|
|
|
|
// front display.
|
|
|
|
// Updated 5.14.2001 mcn - Fixed so loops don't assume spans are icicles
|
|
|
|
|
|
|
|
void plDrawableSpans::SortVisibleSpans(const hsTArray<int16_t>& visList, plPipeline* pipe)
|
|
|
|
{
|
|
|
|
#define MF_CHUNKSORT
|
|
|
|
#ifndef MF_CHUNKSORT
|
|
|
|
|
|
|
|
plProfile_Inc(FaceSortCalls);
|
|
|
|
|
|
|
|
if( !visList.GetCount() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
static hsLargeArray<hsRadixSort::Elem> sortScratch;
|
|
|
|
static hsLargeArray<uint16_t> triList;
|
|
|
|
static hsTArray<int32_t> counters;
|
|
|
|
static hsTArray<uint32_t> startIndex;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
plProfile_BeginTiming(FaceSort);
|
|
|
|
if( pipe->IsDebugFlagSet( plPipeDbg::kFlagDontSortFaces ) )
|
|
|
|
{
|
|
|
|
/// Don't sort, just send unchanged
|
|
|
|
int j, idx;
|
|
|
|
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
ICheckSpanForSortable(visList[i]);
|
|
|
|
|
|
|
|
/// Build a fake list of indices....
|
|
|
|
plGBufferTriangle* list = span->fSortData;
|
|
|
|
triList.SetCount( span->fILength );
|
|
|
|
for( j = 0, idx = 0; j < span->fILength / 3; j++, idx += 3 )
|
|
|
|
{
|
|
|
|
triList[ idx ] = list[ j ].fIndex1;
|
|
|
|
triList[ idx + 1 ] = list[ j ].fIndex2;
|
|
|
|
triList[ idx + 2 ] = list[ j ].fIndex3;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
span->fILength / 3, triList.AcquireArray() );
|
|
|
|
}
|
|
|
|
fReadyToRender = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
plProfile_EndTiming(FaceSort);
|
|
|
|
|
|
|
|
plProfile_BeginLap(FaceSort, "0");
|
|
|
|
|
|
|
|
startIndex.SetCount(fSpans.GetCount());
|
|
|
|
|
|
|
|
// First figure out the total number of tris to deal with.
|
|
|
|
int totTris = 0;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
startIndex[visList[i]] = totTris * 3;
|
|
|
|
if( span->fProps & plSpan::kPropReverseSort )
|
|
|
|
startIndex[visList[i]] += span->fILength - 3;
|
|
|
|
|
|
|
|
totTris += span->fILength / 3;
|
|
|
|
}
|
|
|
|
if( totTris == 0 )
|
|
|
|
{
|
|
|
|
plProfile_EndLap(FaceSort, "0");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_IncCount(FacesSorted, totTris);
|
|
|
|
|
|
|
|
sortScratch.SetCount(totTris);
|
|
|
|
triList.SetCount(3 * totTris);
|
|
|
|
|
|
|
|
hsRadixSort::Elem* elem = sortScratch.AcquireArray();
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "0");
|
|
|
|
plProfile_BeginLap(FaceSort, "1");
|
|
|
|
|
|
|
|
// Pack them into the sort structure. We probably want to make the
|
|
|
|
// plGBufferTriangle look like plTriSortData (just add span index)
|
|
|
|
// which would get rid of this copy and help the data alignment.
|
|
|
|
// Oops, I already did.
|
|
|
|
int cnt = 0;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
int nTris = span->fILength / 3;
|
|
|
|
|
|
|
|
hsPoint3 viewPos = span->fWorldToLocal * pipe->GetViewPositionWorld();
|
|
|
|
|
|
|
|
plGBufferTriangle* list = span->fSortData;
|
|
|
|
int j;
|
|
|
|
for( j = 0; j < nTris; j++ )
|
|
|
|
{
|
|
|
|
float dist = -(viewPos - list[j].fCenter).MagnitudeSquared();
|
|
|
|
elem[cnt].fKey.fFloat = dist;
|
|
|
|
elem[cnt].fBody = &list[j];
|
|
|
|
elem[cnt].fNext = elem + cnt + 1;
|
|
|
|
|
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elem[cnt-1].fNext = nil;
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "1");
|
|
|
|
plProfile_BeginLap(FaceSort, "2");
|
|
|
|
|
|
|
|
// Actual sort
|
|
|
|
hsRadixSort rad;
|
|
|
|
hsRadixSort::Elem* sortedList = rad.Sort( elem, 0 );
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "2");
|
|
|
|
plProfile_BeginLap(FaceSort, "3");
|
|
|
|
|
|
|
|
counters.SetCountAndZero(fSpans.GetCount());
|
|
|
|
|
|
|
|
while( sortedList )
|
|
|
|
{
|
|
|
|
plGBufferTriangle* data = (plGBufferTriangle*)sortedList->fBody;
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[data->fSpanIndex];
|
|
|
|
|
|
|
|
uint16_t* idx = &triList[startIndex[data->fSpanIndex] + counters[data->fSpanIndex]];
|
|
|
|
*idx++ = data->fIndex1;
|
|
|
|
*idx++ = data->fIndex2;
|
|
|
|
*idx++ = data->fIndex3;
|
|
|
|
if( span->fProps & plSpan::kPropReverseSort )
|
|
|
|
counters[data->fSpanIndex] -= 3;
|
|
|
|
else
|
|
|
|
counters[data->fSpanIndex] += 3;
|
|
|
|
|
|
|
|
sortedList = sortedList->fNext;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "3");
|
|
|
|
plProfile_BeginLap(FaceSort, "4");
|
|
|
|
|
|
|
|
const int kMaxBufferGroups = 20;
|
|
|
|
const int kMaxIndexBuffers = 20;
|
|
|
|
static int16_t newStarts[kMaxBufferGroups][kMaxIndexBuffers];
|
|
|
|
|
|
|
|
// Temp hack stuff so we can switch back and forth (to see if it really does any good).
|
|
|
|
static int oldWay = false;
|
|
|
|
static int lastOldWay = oldWay;
|
|
|
|
if( oldWay != lastOldWay )
|
|
|
|
{
|
|
|
|
memset(newStarts, 0, kMaxBufferGroups * kMaxIndexBuffers * sizeof(int16_t));
|
|
|
|
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[i];
|
|
|
|
|
|
|
|
span->fPackedIdx = span->fIStartIdx = newStarts[span->fGroupIdx][span->fIBufferIdx];
|
|
|
|
newStarts[span->fGroupIdx][span->fIBufferIdx] += span->fILength;
|
|
|
|
}
|
|
|
|
lastOldWay = oldWay;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( oldWay )
|
|
|
|
{
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
span->fILength / 3, triList.AcquireArray() + startIndex[visList[i]]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
memset(newStarts, 0, kMaxBufferGroups * kMaxIndexBuffers * sizeof(int16_t));
|
|
|
|
|
|
|
|
uint32_t start = 0;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
span->fPackedIdx = span->fIStartIdx = newStarts[span->fGroupIdx][span->fIBufferIdx];
|
|
|
|
newStarts[span->fGroupIdx][span->fIBufferIdx] += span->fILength;
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
span->fILength / 3, triList.AcquireArray() + startIndex[visList[i]]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "4");
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
#else // MF_CHUNKSORT
|
|
|
|
|
|
|
|
plProfile_Inc(FaceSortCalls);
|
|
|
|
|
|
|
|
if( !visList.GetCount() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
plProfile_BeginTiming(FaceSort);
|
|
|
|
|
|
|
|
static hsLargeArray<hsRadixSort::Elem> sortScratch;
|
|
|
|
static hsLargeArray<uint16_t> triList;
|
|
|
|
static hsTArray<int32_t> counters;
|
|
|
|
static hsTArray<uint32_t> startIndex;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if( pipe->IsDebugFlagSet( plPipeDbg::kFlagDontSortFaces ) )
|
|
|
|
{
|
|
|
|
/// Don't sort, just send unchanged
|
|
|
|
int j, idx;
|
|
|
|
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
ICheckSpanForSortable(visList[i]);
|
|
|
|
|
|
|
|
/// Build a fake list of indices....
|
|
|
|
plGBufferTriangle* list = span->fSortData;
|
|
|
|
triList.SetCount( span->fILength );
|
|
|
|
for( j = 0, idx = 0; j < span->fILength / 3; j++, idx += 3 )
|
|
|
|
{
|
|
|
|
triList[ idx ] = list[ j ].fIndex1;
|
|
|
|
triList[ idx + 1 ] = list[ j ].fIndex2;
|
|
|
|
triList[ idx + 2 ] = list[ j ].fIndex3;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
span->fILength / 3, triList.AcquireArray() );
|
|
|
|
}
|
|
|
|
fReadyToRender = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndTiming(FaceSort);
|
|
|
|
|
|
|
|
plProfile_BeginLap(FaceSort, "0");
|
|
|
|
|
|
|
|
startIndex.SetCount(fSpans.GetCount());
|
|
|
|
|
|
|
|
// First figure out the total number of tris to deal with.
|
|
|
|
int totTris = 0;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
ICheckSpanForSortable(visList[i]);
|
|
|
|
|
|
|
|
startIndex[visList[i]] = totTris * 3;
|
|
|
|
if( span->fProps & plSpan::kPropReverseSort )
|
|
|
|
startIndex[visList[i]] += span->fILength - 3;
|
|
|
|
|
|
|
|
|
|
|
|
totTris += span->fILength / 3;
|
|
|
|
}
|
|
|
|
if( totTris == 0 )
|
|
|
|
{
|
|
|
|
plProfile_EndLap(FaceSort, "0");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_IncCount(FacesSorted, totTris);
|
|
|
|
|
|
|
|
sortScratch.SetCount(totTris);
|
|
|
|
triList.SetCount(3 * totTris);
|
|
|
|
|
|
|
|
hsRadixSort::Elem* elem = sortScratch.AcquireArray();
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "0");
|
|
|
|
|
|
|
|
int iVis = 0;
|
|
|
|
while( iVis < visList.GetCount() )
|
|
|
|
{
|
|
|
|
plProfile_BeginLap(FaceSort, "1");
|
|
|
|
|
|
|
|
// Pack them into the sort structure. We probably want to make the
|
|
|
|
// plGBufferTriangle look like plTriSortData (just add span index)
|
|
|
|
// which would get rid of this copy and help the data alignment.
|
|
|
|
// Oops, I already did.
|
|
|
|
const int kTriCutoff = 4000;
|
|
|
|
int cnt = 0;
|
|
|
|
while( (iVis < visList.GetCount()) && (cnt < kTriCutoff) )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[iVis]];
|
|
|
|
|
|
|
|
int nTris = span->fILength / 3;
|
|
|
|
|
|
|
|
hsPoint3 viewPos = span->fWorldToLocal * pipe->GetViewPositionWorld();
|
|
|
|
|
|
|
|
plGBufferTriangle* list = span->fSortData;
|
|
|
|
int j;
|
|
|
|
for( j = 0; j < nTris; j++ )
|
|
|
|
{
|
|
|
|
float dist = -(viewPos - list[j].fCenter).MagnitudeSquared();
|
|
|
|
elem[cnt].fKey.fFloat = dist;
|
|
|
|
elem[cnt].fBody = &list[j];
|
|
|
|
elem[cnt].fNext = elem + cnt + 1;
|
|
|
|
|
|
|
|
cnt++;
|
|
|
|
}
|
|
|
|
iVis++;
|
|
|
|
}
|
|
|
|
elem[cnt-1].fNext = nil;
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "1");
|
|
|
|
plProfile_BeginLap(FaceSort, "2");
|
|
|
|
|
|
|
|
// Actual sort
|
|
|
|
hsRadixSort rad;
|
|
|
|
hsRadixSort::Elem* sortedList = rad.Sort( elem, 0 );
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "2");
|
|
|
|
plProfile_BeginLap(FaceSort, "3");
|
|
|
|
|
|
|
|
counters.SetCountAndZero(fSpans.GetCount());
|
|
|
|
|
|
|
|
while( sortedList )
|
|
|
|
{
|
|
|
|
plGBufferTriangle* data = (plGBufferTriangle*)sortedList->fBody;
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[data->fSpanIndex];
|
|
|
|
|
|
|
|
uint16_t* idx = &triList[startIndex[data->fSpanIndex] + counters[data->fSpanIndex]];
|
|
|
|
*idx++ = data->fIndex1;
|
|
|
|
*idx++ = data->fIndex2;
|
|
|
|
*idx++ = data->fIndex3;
|
|
|
|
if( span->fProps & plSpan::kPropReverseSort )
|
|
|
|
counters[data->fSpanIndex] -= 3;
|
|
|
|
else
|
|
|
|
counters[data->fSpanIndex] += 3;
|
|
|
|
|
|
|
|
sortedList = sortedList->fNext;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "3");
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_BeginLap(FaceSort, "4");
|
|
|
|
|
|
|
|
const int kMaxBufferGroups = 20;
|
|
|
|
const int kMaxIndexBuffers = 20;
|
|
|
|
static int16_t newStarts[kMaxBufferGroups][kMaxIndexBuffers];
|
|
|
|
|
|
|
|
hsAssert(kMaxBufferGroups >= GetNumBufferGroups(), "Bigger than we counted on num groups sort.");
|
|
|
|
|
|
|
|
memset(newStarts, 0, kMaxBufferGroups * kMaxIndexBuffers * sizeof(int16_t));
|
|
|
|
|
|
|
|
uint32_t start = 0;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
hsAssert(kMaxIndexBuffers > span->fIBufferIdx, "Bigger than we counted on num buffers sort.");
|
|
|
|
|
|
|
|
if( span->fProps & plSpan::kPropReverseSort )
|
|
|
|
startIndex[visList[i]] -= span->fILength - 3;
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
span->fIPackedIdx = span->fIStartIdx = newStarts[span->fGroupIdx][span->fIBufferIdx];
|
|
|
|
newStarts[span->fGroupIdx][span->fIBufferIdx] += (int16_t)(span->fILength);
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
span->fILength / 3, triList.AcquireArray() + startIndex[visList[i]]);
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "4");
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
#endif // MF_CHUNKSORT
|
|
|
|
}
|
|
|
|
|
|
|
|
struct buffTriCmpBackToFront : public std::binary_function<plGBufferTriangle, plGBufferTriangle, bool>
|
|
|
|
{
|
|
|
|
hsPoint3 fViewPos;
|
|
|
|
buffTriCmpBackToFront(const hsPoint3& p) : fViewPos(p) {}
|
|
|
|
|
|
|
|
bool operator()( const plGBufferTriangle& lhs, const plGBufferTriangle& rhs) const
|
|
|
|
{
|
|
|
|
return hsVector3(&fViewPos, &lhs.fCenter).MagnitudeSquared() > hsVector3(&fViewPos, &rhs.fCenter).MagnitudeSquared();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct buffTriCmpFrontToBack : public std::binary_function<plGBufferTriangle, plGBufferTriangle, bool>
|
|
|
|
{
|
|
|
|
hsPoint3 fViewPos;
|
|
|
|
buffTriCmpFrontToBack(const hsPoint3& p) : fViewPos(p) {}
|
|
|
|
|
|
|
|
bool operator()( const plGBufferTriangle& lhs, const plGBufferTriangle& rhs) const
|
|
|
|
{
|
|
|
|
return hsVector3(&fViewPos, &lhs.fCenter).MagnitudeSquared() < hsVector3(&fViewPos, &rhs.fCenter).MagnitudeSquared();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void plDrawableSpans::SortVisibleSpansPartial(const hsTArray<int16_t>& visList, plPipeline* pipe)
|
|
|
|
{
|
|
|
|
plProfile_Inc(FaceSortCalls);
|
|
|
|
|
|
|
|
plProfile_BeginTiming(FaceSort);
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
hsAssert(fSpans[visList[i]]->fTypeMask & plSpan::kIcicleSpan, "Unknown type for sorting faces");
|
|
|
|
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
if( span->fProps & plSpan::kPartialSort )
|
|
|
|
{
|
|
|
|
hsAssert(fGroups[span->fGroupIdx]->AreIdxVolatile(), "Badly setup buffer group - set PartialSort too late?");
|
|
|
|
|
|
|
|
ICheckSpanForSortable(visList[i]);
|
|
|
|
|
|
|
|
const hsPoint3 viewPos = span->fWorldToLocal * pipe->GetViewPositionWorld();
|
|
|
|
|
|
|
|
const int numTris = span->fILength/3;
|
|
|
|
std::sort(span->fSortData, span->fSortData+numTris, buffTriCmpBackToFront(viewPos));
|
|
|
|
|
|
|
|
uint16_t* idx = fGroups[span->fGroupIdx]->GetIndexBufferData(span->fIBufferIdx) + span->fIStartIdx;
|
|
|
|
plGBufferTriangle* iter = span->fSortData;
|
|
|
|
int j;
|
|
|
|
for( j = 0; j < numTris; j++ )
|
|
|
|
{
|
|
|
|
*idx++ = iter->fIndex1;
|
|
|
|
*idx++ = iter->fIndex2;
|
|
|
|
*idx++ = iter->fIndex3;
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
fGroups[span->fGroupIdx]->DirtyIndexBuffer(span->fIBufferIdx);
|
|
|
|
fReadyToRender = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndTiming(FaceSort);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
|
|
|
void plDrawableSpans::SortVisibleSpansUnit(const hsTArray<int16_t>& visList, plPipeline* pipe)
|
|
|
|
{
|
|
|
|
plProfile_Inc(FaceSortCalls);
|
|
|
|
|
|
|
|
if( !visList.GetCount() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
plProfile_BeginTiming(FaceSort);
|
|
|
|
|
|
|
|
static hsLargeArray<uint16_t> triList;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if( pipe->IsDebugFlagSet( plPipeDbg::kFlagDontSortFaces ) )
|
|
|
|
{
|
|
|
|
/// Don't sort, just send unchanged
|
|
|
|
int j, idx;
|
|
|
|
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
ICheckSpanForSortable(visList[i]);
|
|
|
|
|
|
|
|
/// Build a fake list of indices....
|
|
|
|
plGBufferTriangle* list = span->fSortData;
|
|
|
|
triList.SetCount( span->fILength );
|
|
|
|
for( j = 0, idx = 0; j < span->fILength / 3; j++, idx += 3 )
|
|
|
|
{
|
|
|
|
triList[ idx ] = list[ j ].fIndex1;
|
|
|
|
triList[ idx + 1 ] = list[ j ].fIndex2;
|
|
|
|
triList[ idx + 2 ] = list[ j ].fIndex3;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
span->fILength / 3, triList.AcquireArray() );
|
|
|
|
}
|
|
|
|
fReadyToRender = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndTiming(FaceSort);
|
|
|
|
|
|
|
|
plProfile_BeginLap(FaceSort, "0");
|
|
|
|
|
|
|
|
struct sortFace
|
|
|
|
{
|
|
|
|
uint16_t fIndex0;
|
|
|
|
uint16_t fIndex1;
|
|
|
|
uint16_t fIndex2;
|
|
|
|
float fDist;
|
|
|
|
};
|
|
|
|
static hsLargeArray<sortFace> sortList;
|
|
|
|
|
|
|
|
struct SelectCloserFace
|
|
|
|
{
|
|
|
|
bool operator()(const sortFace* face0, const sortFace* face1) const
|
|
|
|
{
|
|
|
|
return face0->fDist < face1->fDist;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
hsPoint3 viewPos = fSpans[visList[0]]->fWorldToLocal * pipe->GetViewPositionWorld();
|
|
|
|
|
|
|
|
float dist1 = (fViewPos - face1->fCenter).MagnitudeSquared();
|
|
|
|
|
|
|
|
// First figure out the total number of tris to deal with.
|
|
|
|
sortList.SetCount(0);
|
|
|
|
int totTris = 0;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
int nTris = span->fILength / 3;
|
|
|
|
sortList.Expand(sortList.GetCount() + nTris);
|
|
|
|
|
|
|
|
plGBufferTriangle* sortData = span->fSortData;
|
|
|
|
for( j = 0; j < nTris; j++ )
|
|
|
|
{
|
|
|
|
sortList.Append(*sortData++);
|
|
|
|
}
|
|
|
|
|
|
|
|
totTris += nTris;
|
|
|
|
}
|
|
|
|
if( totTris == 0 )
|
|
|
|
{
|
|
|
|
plProfile_EndLap(FaceSort, "0");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_IncCount(FacesSorted, totTris);
|
|
|
|
|
|
|
|
sortFace* pBegin = sortList.AcquireArray();
|
|
|
|
sortFace* pEnd = pBegin + totTris;
|
|
|
|
|
|
|
|
stl::sort(pBegin, pEnd, SelectCloserFace);
|
|
|
|
|
|
|
|
triList.SetCount(sortList.GetCount());
|
|
|
|
|
|
|
|
uint16_t* pTri = triList.AcquireArray();
|
|
|
|
|
|
|
|
while( pBegin < pEnd )
|
|
|
|
{
|
|
|
|
*pTri = pBegin->fIndex1;
|
|
|
|
pTri++;
|
|
|
|
*pTri = pBegin->fIndex2;
|
|
|
|
pTri++;
|
|
|
|
*pTri = pBegin->fIndex3;
|
|
|
|
pTri++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "0");
|
|
|
|
|
|
|
|
uint32_t start = 0;
|
|
|
|
for( i = 0; i < visList.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[visList[i]];
|
|
|
|
|
|
|
|
/// Now send them on to the buffer group
|
|
|
|
span->fIPackedIdx = span->fIStartIdx = newStarts[span->fGroupIdx][span->fIBufferIdx];
|
|
|
|
newStarts[span->fGroupIdx][span->fIBufferIdx] += span->fILength;
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffFromTriList( span->fIBufferIdx, span->fIStartIdx,
|
|
|
|
span->fILength / 3, triList.AcquireArray() + startIndex[visList[i]]);
|
|
|
|
}
|
|
|
|
|
|
|
|
plProfile_EndLap(FaceSort, "4");
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void plDrawableSpans::SetInitialBone(int i, const hsMatrix44& l2b, const hsMatrix44& b2l)
|
|
|
|
{
|
|
|
|
fLocalToBones[ i ] = l2b;
|
|
|
|
fBoneToLocals[ i ] = b2l;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// AppendDIMatrixSpans ///////////////////////////////////////////////////////////
|
|
|
|
// Adds a di span which will only reference into the matrix palette. That is,
|
|
|
|
// these indices won't index into any drawable data, they will only index into
|
|
|
|
// the same matrix list that the drawable data itself can index into. When
|
|
|
|
// an object (through its DrawInterface) sets a transform, it doesn't know
|
|
|
|
// whether it's setting the transform for some drawable data it owns, or just
|
|
|
|
// setting one of the matrices which influence the drawable data someone else
|
|
|
|
// owns.
|
|
|
|
uint32_t plDrawableSpans::AppendDIMatrixSpans(int n)
|
|
|
|
{
|
|
|
|
/// Do garbage cleanup first
|
|
|
|
if( fNeedCleanup )
|
|
|
|
IRemoveGarbage();
|
|
|
|
|
|
|
|
uint32_t baseIdx = fLocalToWorlds.GetCount();
|
|
|
|
fLocalToWorlds.Expand(baseIdx + n);
|
|
|
|
fLocalToWorlds.SetCount(baseIdx + n);
|
|
|
|
fWorldToLocals.Expand(baseIdx + n);
|
|
|
|
fWorldToLocals.SetCount(baseIdx + n);
|
|
|
|
|
|
|
|
fLocalToBones.Expand(baseIdx + n);
|
|
|
|
fLocalToBones.SetCount(baseIdx + n);
|
|
|
|
fBoneToLocals.Expand(baseIdx + n);
|
|
|
|
fBoneToLocals.SetCount(baseIdx + n);
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for( i = baseIdx; i < baseIdx + n; i++ )
|
|
|
|
{
|
|
|
|
fLocalToWorlds[i].Reset();
|
|
|
|
fWorldToLocals[i].Reset();
|
|
|
|
|
|
|
|
fLocalToBones[i].Reset();
|
|
|
|
fBoneToLocals[i].Reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
return baseIdx;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Look for a compatible set of palette matrices. Compatible means:
|
|
|
|
// a) Same number bones
|
|
|
|
// b) Same LocalToBones matrices (which implies same BoneToLocal matrices.
|
|
|
|
// Note that the LocalToBone transform is dependent on both the bone's transform
|
|
|
|
// and the transform of the object being skinned. In general, objects can only
|
|
|
|
// share a palette set if they have been flattened into world space (the object's
|
|
|
|
// transform is identity). Fortunately, this is a common case.
|
|
|
|
uint32_t plDrawableSpans::FindBoneBaseMatrix(const hsTArray<hsMatrix44>& initL2B, hsBool searchAll) const
|
|
|
|
{
|
|
|
|
if (!searchAll)
|
|
|
|
{
|
|
|
|
// Just look amongst the added spans for a matching set
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[i] && (initL2B.GetCount() == fSpans[i]->fNumMatrices) )
|
|
|
|
{
|
|
|
|
int j;
|
|
|
|
for( j = 0; j < initL2B.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( initL2B[j] != fLocalToBones[j + fSpans[i]->fBaseMatrix] )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if( initL2B.GetCount() == j )
|
|
|
|
return fSpans[i]->fBaseMatrix;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Since swappable geometry spans aren't added to the drawable until
|
|
|
|
// runtime, a sharable bone pallete won't be found by scanning fSpans.
|
|
|
|
// We have to do a larger search through all bone matrices.
|
|
|
|
int i;
|
|
|
|
for( i = 0; i + initL2B.GetCount() < fLocalToBones.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
int j;
|
|
|
|
for( j = 0; j < initL2B.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( initL2B[j] != fLocalToBones[j + i] )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if( initL2B.GetCount() == j )
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return uint32_t(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::NewDIMatrixIndex()
|
|
|
|
{
|
|
|
|
int index;
|
|
|
|
/// Do we have a free lookup entry?
|
|
|
|
for( index = 0; index < fDIIndices.GetCount(); index++ )
|
|
|
|
{
|
|
|
|
if( !fDIIndices[ index ]->GetCount() )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if( index == fDIIndices.GetCount() )
|
|
|
|
fDIIndices.Append( TRACKED_NEW plDISpanIndex );
|
|
|
|
|
|
|
|
fDIIndices[index]->Reset();
|
|
|
|
fDIIndices[index]->fFlags = plDISpanIndex::kMatrixOnly;
|
|
|
|
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IFindDIIndices //////////////////////////////////////////////////////////
|
|
|
|
// Finds the given DIIndices array and returns a pointer to it, or creates
|
|
|
|
// a new one if requested
|
|
|
|
|
|
|
|
plDISpanIndex *plDrawableSpans::IFindDIIndices( uint32_t &index )
|
|
|
|
{
|
|
|
|
plDISpanIndex *spanLookup;
|
|
|
|
|
|
|
|
|
|
|
|
/// Do we have a free lookup entry?
|
|
|
|
if( index == (uint32_t)-1 ) // new index
|
|
|
|
{
|
|
|
|
for( index = 0; index < fDIIndices.GetCount(); index++ )
|
|
|
|
{
|
|
|
|
if( fDIIndices[ index ]->GetCount() == 0 )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if( index == fDIIndices.GetCount() )
|
|
|
|
fDIIndices.Append( TRACKED_NEW plDISpanIndex );
|
|
|
|
|
|
|
|
spanLookup = fDIIndices[ index ];
|
|
|
|
spanLookup->fFlags = plDISpanIndex::kNone;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
spanLookup = fDIIndices[ index ]; // Just grab the one we requested
|
|
|
|
|
|
|
|
return spanLookup;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// AppendDISpans ///////////////////////////////////////////////////////////
|
|
|
|
// Given a set of DI spans, appends them to the current storage buffers.
|
|
|
|
// Note: AddDISpans() adds the spans to a list to be sorted, THEN put into
|
|
|
|
// the buffers; this shoves them right in, bypassing the sorting altogether.
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::AppendDISpans( hsTArray<plGeometrySpan *> &spans, uint32_t index, hsBool clearSpansAfterAdd,
|
|
|
|
hsBool doNotAddToSource, hsBool addToFront, int lod)
|
|
|
|
{
|
|
|
|
hsAssert(spans.GetCount(), "Adding no spans? Blow me.");
|
|
|
|
|
|
|
|
int i, j;
|
|
|
|
uint32_t spanIdx;
|
|
|
|
plSpan *span;
|
|
|
|
hsBounds3Ext bounds;
|
|
|
|
plDISpanIndex *spanLookup;
|
|
|
|
uint32_t numAddedIcicle = 0;
|
|
|
|
|
|
|
|
// hsAssert( fProps & kPropVolatile, "Trying to add spans on a non-volatile drawable" );
|
|
|
|
|
|
|
|
/// Do garbage cleanup first
|
|
|
|
if( fNeedCleanup )
|
|
|
|
IRemoveGarbage();
|
|
|
|
|
|
|
|
spanLookup = IFindDIIndices( index );
|
|
|
|
|
|
|
|
if( GetNativeProperty(plDrawable::kPropCharacter) )
|
|
|
|
fVisSet.SetBit(plVisMgr::kCharacter, true);
|
|
|
|
|
|
|
|
int insertionPoint = 0;
|
|
|
|
if( spans[0]->fProps & plGeometrySpan::kPartialSort )
|
|
|
|
{
|
|
|
|
insertionPoint = fSpans.GetCount();
|
|
|
|
}
|
|
|
|
else if( !addToFront )
|
|
|
|
{
|
|
|
|
int idx;
|
|
|
|
for( idx = 0; (idx < fSpans.GetCount()) && !(fSpans[idx]->fProps & plSpan::kPartialSort); idx++ )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
insertionPoint = idx;
|
|
|
|
}
|
|
|
|
hsBool inserted = insertionPoint < fSpans.GetCount();
|
|
|
|
|
|
|
|
/// Add the geometry spans to our list. Also add our internal span
|
|
|
|
/// copies
|
|
|
|
for( i = 0; i < spans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
spanIdx = fIcicles.GetCount();
|
|
|
|
fIcicles.Append( plIcicle() );
|
|
|
|
plIcicle *icicle = &fIcicles[ spanIdx ];
|
|
|
|
IConvertGeoSpanToIcicle( spans[ i ], icicle, lod );
|
|
|
|
span = (plSpan *)icicle;
|
|
|
|
numAddedIcicle++;
|
|
|
|
|
|
|
|
/// Add to our source indices
|
|
|
|
spans[ i ]->fSpanRefIndex = insertionPoint+i;
|
|
|
|
spanLookup->Append( spans[ i ]->fSpanRefIndex );
|
|
|
|
fSpans.Insert( spans[ i ]->fSpanRefIndex, span );
|
|
|
|
fSpanSourceIndices.Insert( spans[ i ]->fSpanRefIndex, spanIdx );
|
|
|
|
|
|
|
|
if( GetNativeProperty(plDrawable::kPropCharacter) )
|
|
|
|
span->SetVisBit(plVisMgr::kCharacter, true);
|
|
|
|
|
|
|
|
/// Add the material to our list if necessary
|
|
|
|
IAssignMatIdxToSpan( span, spans[ i ]->fMaterial );
|
|
|
|
|
|
|
|
if( clearSpansAfterAdd )
|
|
|
|
{
|
|
|
|
delete spans[ i ];
|
|
|
|
spans[ i ] = nil;
|
|
|
|
}
|
|
|
|
else if( !doNotAddToSource )
|
|
|
|
{
|
|
|
|
if( fSourceSpans.GetCount() < fSpans.GetCount() )
|
|
|
|
{
|
|
|
|
fSourceSpans.Expand( fSpans.GetCount() );
|
|
|
|
// Since that does not change the use count, we still have to do that ourselves. ARGH!
|
|
|
|
fSourceSpans.SetCount( fSpans.GetCount() );
|
|
|
|
}
|
|
|
|
|
|
|
|
fSourceSpans[ spans[ i ]->fSpanRefIndex ] = spans[ i ];
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
{
|
|
|
|
// Should add sort data too...
|
|
|
|
IMakeSpanSortable( fSpans.GetCount() - 1 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (inserted)
|
|
|
|
{
|
|
|
|
/// Go adjusting indices in the DI index list
|
|
|
|
for( i = 0; i < fDIIndices.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( !fDIIndices[ i ]->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
if (fDIIndices[ i ] == spanLookup)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for( j = 0; j < fDIIndices[ i ]->GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( (*fDIIndices[i])[j] >= insertionPoint )
|
|
|
|
{
|
|
|
|
(*fDIIndices[ i ])[ j ] += spans.GetCount();
|
|
|
|
hsAssert((*fDIIndices[ i ])[ j ] < fSpans.GetCount(), "Span index snafu");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update fLocalBounds, since we were updating the world bounds
|
|
|
|
fLocalBounds = fWorldBounds;
|
|
|
|
fLocalBounds.Transform( &fWorldToLocal );
|
|
|
|
fMaxWorldBounds = fWorldBounds;
|
|
|
|
|
|
|
|
/// Rebuild the pointer array
|
|
|
|
IRebuildSpanArray();
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IAddAMaterial ///////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::IAddAMaterial( hsGMaterial *material )
|
|
|
|
{
|
|
|
|
uint32_t i;
|
|
|
|
|
|
|
|
|
|
|
|
// Scan for if we already have it
|
|
|
|
for( i = 0; i < fMaterials.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fMaterials[ i ] == material )
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan again for a blank space to add into, if possible
|
|
|
|
for( i = 0; i < fMaterials.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fMaterials[ i ] == nil )
|
|
|
|
{
|
|
|
|
fMaterials[ i ] = material;
|
|
|
|
IRefMaterial( i );
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add in to the end
|
|
|
|
i = fMaterials.GetCount();
|
|
|
|
fMaterials.Append( material );
|
|
|
|
|
|
|
|
// Plus ref it
|
|
|
|
IRefMaterial( i );
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IRefMaterial ////////////////////////////////////////////////////////////
|
|
|
|
// Called if we already have a material index that we just want to use
|
|
|
|
// again...
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::IRefMaterial( uint32_t index )
|
|
|
|
{
|
|
|
|
hsGMaterial *material = fMaterials[ index ];
|
|
|
|
|
|
|
|
if( GetKey() && material != nil && material->GetKey() != nil )
|
|
|
|
hsgResMgr::ResMgr()->AddViaNotify( material->GetKey(), TRACKED_NEW plGenRefMsg( GetKey(), plRefMsg::kOnCreate, index, 0 ), plRefFlags::kActiveRef );
|
|
|
|
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ICheckToRemoveMaterial //////////////////////////////////////////////////
|
|
|
|
// Runs through the span array to see if the given material index is still
|
|
|
|
// used. If not, Release()s it and nil's it.
|
|
|
|
|
|
|
|
void plDrawableSpans::ICheckToRemoveMaterial( uint32_t materialIdx )
|
|
|
|
{
|
|
|
|
int j;
|
|
|
|
hsGMaterial *mat;
|
|
|
|
|
|
|
|
|
|
|
|
for( j = 0; j < fSpans.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( fSpans[ j ]->fMaterialIdx == materialIdx )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if( j == fSpans.GetCount() )
|
|
|
|
{
|
|
|
|
/// No longer used--Release() it
|
|
|
|
mat = fMaterials[ materialIdx ];
|
|
|
|
if( GetKey() && mat != nil && mat->GetKey() != nil )
|
|
|
|
GetKey()->Release( mat->GetKey() );
|
|
|
|
fMaterials[ materialIdx ] = nil;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IConvertGeoSpanToVertexSpan /////////////////////////////////////////////
|
|
|
|
// Helper function for the two vertex-based convert functions.
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::IConvertGeoSpanToVertexSpan( plGeometrySpan *geoSpan, plVertexSpan *span, int lod, plVertexSpan *instancedParent)
|
|
|
|
{
|
|
|
|
hsBounds3Ext bounds;
|
|
|
|
uint8_t groupIdx;
|
|
|
|
uint32_t vbIndex, cellIdx, cellOffset;
|
|
|
|
|
|
|
|
|
|
|
|
span->fLocalToWorld = geoSpan->fLocalToWorld;
|
|
|
|
span->fWorldToLocal = geoSpan->fWorldToLocal;
|
|
|
|
span->fProps |= ( geoSpan->fProps & plGeometrySpan::kPropRunTimeLight ) ? plSpan::kPropRunTimeLight : 0;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropNoShadowCast )
|
|
|
|
span->fProps |= plSpan::kPropNoShadowCast;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropNoShadow )
|
|
|
|
span->fProps |= plSpan::kPropNoShadow;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropForceShadow )
|
|
|
|
span->fProps |= plSpan::kPropForceShadow;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropReverseSort )
|
|
|
|
span->fProps |= plSpan::kPropReverseSort;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPartialSort )
|
|
|
|
span->fProps |= plSpan::kPartialSort;
|
|
|
|
switch( geoSpan->fProps & plGeometrySpan::kLiteMask )
|
|
|
|
{
|
|
|
|
case plGeometrySpan::kLiteMaterial: span->fProps |= plSpan::kLiteMaterial; break;
|
|
|
|
case plGeometrySpan::kLiteVtxPreshaded: span->fProps |= plSpan::kLiteVtxPreshaded; break;
|
|
|
|
case plGeometrySpan::kLiteVtxNonPreshaded: span->fProps |= plSpan::kLiteVtxNonPreshaded; break;
|
|
|
|
}
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kWaterHeight )
|
|
|
|
{
|
|
|
|
span->fProps |= plSpan::kWaterHeight;
|
|
|
|
span->fWaterHeight = geoSpan->fWaterHeight;
|
|
|
|
}
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kVisLOS )
|
|
|
|
{
|
|
|
|
span->fProps |= plSpan::kVisLOS;
|
|
|
|
fProps |= plDrawable::kPropHasVisLOS;
|
|
|
|
}
|
|
|
|
|
|
|
|
span->fNumMatrices = geoSpan->fNumMatrices;
|
|
|
|
span->fBaseMatrix = geoSpan->fBaseMatrix;
|
|
|
|
span->fLocalUVWChans = geoSpan->fLocalUVWChans;
|
|
|
|
span->fMaxBoneIdx = geoSpan->fMaxBoneIdx;
|
|
|
|
span->fPenBoneIdx = (uint16_t)(geoSpan->fPenBoneIdx);
|
|
|
|
span->fMinDist = geoSpan->fMinDist;
|
|
|
|
span->fMaxDist = geoSpan->fMaxDist;
|
|
|
|
|
|
|
|
bounds = geoSpan->fLocalBounds;
|
|
|
|
span->fLocalBounds = bounds;
|
|
|
|
bounds.Transform( &span->fLocalToWorld );
|
|
|
|
span->fWorldBounds = bounds;
|
|
|
|
fWorldBounds.Union( &bounds );
|
|
|
|
span->fFogEnvironment = geoSpan->fFogEnviron;
|
|
|
|
|
|
|
|
hsBool vertsVol = false;
|
|
|
|
if( fProps & kPropVolatile )
|
|
|
|
vertsVol = true;
|
|
|
|
|
|
|
|
hsBool idxVol = false;
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
idxVol = true;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPartialSort )
|
|
|
|
idxVol = true;
|
|
|
|
|
|
|
|
// Are we instanced?
|
|
|
|
if( instancedParent != nil /*&& !( fProps & kPropVolatile )*/ )
|
|
|
|
{
|
|
|
|
/// We can instance w/o vert data IF 1) we're not the first span, 2) we can fit into the same buffer group, and
|
|
|
|
/// 3) we can fit into the same vertex buffer in the buffer group
|
|
|
|
if( instancedParent != span && ( geoSpan->fProps & plGeometrySpan::kPropNoPreShade ) )
|
|
|
|
{
|
|
|
|
/// WOW! We can actually share the *exact* same data, since we don't do any preshading. Coooooool!
|
|
|
|
span->fGroupIdx = instancedParent->fGroupIdx;
|
|
|
|
span->fVBufferIdx = instancedParent->fVBufferIdx;
|
|
|
|
span->fVStartIdx = instancedParent->fVStartIdx;
|
|
|
|
cellIdx = instancedParent->fCellIdx;
|
|
|
|
cellOffset = instancedParent->fCellOffset;
|
|
|
|
}
|
|
|
|
else if( instancedParent != span &&
|
|
|
|
( IFindBufferGroup( geoSpan->fFormat, geoSpan->fNumVerts, lod, vertsVol, idxVol ) == instancedParent->fGroupIdx ) &&
|
|
|
|
fGroups[ instancedParent->fGroupIdx ]->GetNumVertsLeft( instancedParent->fVBufferIdx ) >= geoSpan->fNumVerts )
|
|
|
|
{
|
|
|
|
/// Boooring....
|
|
|
|
/// Append the colors, but the vertices themselves we don't use, rather we point
|
|
|
|
/// to our parent's verts (or rather, the cell will take care of that for us)
|
|
|
|
groupIdx = (uint8_t)(instancedParent->fGroupIdx);
|
|
|
|
fGroups[ groupIdx ]->AppendToColorStorage( geoSpan, &vbIndex, &cellIdx, &cellOffset, instancedParent->fCellIdx );
|
|
|
|
|
|
|
|
span->fGroupIdx = groupIdx;
|
|
|
|
span->fVBufferIdx = instancedParent->fVBufferIdx;
|
|
|
|
// Do a new vStart here, since it's really the colorStart, but oh well
|
|
|
|
span->fVStartIdx = fGroups[ groupIdx ]->GetVertStartFromCell( vbIndex, cellIdx, cellOffset );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// WE'RE the parent?? This means we're the first to get converted
|
|
|
|
groupIdx = IFindBufferGroup( geoSpan->fFormat, geoSpan->fNumVerts, lod, vertsVol, idxVol );
|
|
|
|
fGroups[ groupIdx ]->AppendToVertAndColorStorage( geoSpan, &vbIndex, &cellIdx, &cellOffset );
|
|
|
|
|
|
|
|
span->fGroupIdx = groupIdx;
|
|
|
|
span->fVBufferIdx = vbIndex;
|
|
|
|
span->fVStartIdx = fGroups[ groupIdx ]->GetVertStartFromCell( vbIndex, cellIdx, cellOffset );
|
|
|
|
geoSpan->fProps |= plGeometrySpan::kFirstInstance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Pack the vertices in
|
|
|
|
groupIdx = IFindBufferGroup( geoSpan->fFormat, geoSpan->fNumVerts, lod, vertsVol, idxVol );
|
|
|
|
fGroups[ groupIdx ]->AppendToVertStorage( geoSpan, &vbIndex, &cellIdx, &cellOffset );
|
|
|
|
|
|
|
|
span->fGroupIdx = groupIdx;
|
|
|
|
span->fVBufferIdx = vbIndex;
|
|
|
|
span->fVStartIdx = fGroups[ groupIdx ]->GetVertStartFromCell( vbIndex, cellIdx, cellOffset );
|
|
|
|
geoSpan->fProps |= plGeometrySpan::kFirstInstance;
|
|
|
|
}
|
|
|
|
|
|
|
|
span->fCellIdx = cellIdx;
|
|
|
|
span->fCellOffset = cellOffset;
|
|
|
|
span->fVLength = geoSpan->fNumVerts;
|
|
|
|
|
|
|
|
/// Add the material to our list if necessary
|
|
|
|
span->fMaterialIdx = IAddAMaterial( geoSpan->fMaterial );
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IConvertGeoSpanToIcicle /////////////////////////////////////////////////
|
|
|
|
|
|
|
|
hsBool plDrawableSpans::IConvertGeoSpanToIcicle(plGeometrySpan *geoSpan, plIcicle *icicle, int lod, plIcicle *instancedParent)
|
|
|
|
{
|
|
|
|
uint32_t ibIndex, ibStart;
|
|
|
|
|
|
|
|
|
|
|
|
IConvertGeoSpanToVertexSpan(geoSpan, icicle, lod, instancedParent);
|
|
|
|
|
|
|
|
/*
|
|
|
|
Disabling this 8.16.2001. Works great until you have to SORT the faces, then things blow up. We could
|
|
|
|
do this only when we won't sort faces, but it's easier right now to just never do this ever.
|
|
|
|
|
|
|
|
if( instancedParent != nil && instancedParent != icicle && ( icicle->fProps & plSpan::kPropRunTimeLight ) )
|
|
|
|
{
|
|
|
|
/// WOW! We can actually share the *exact* same data, since we don't do any preshading. Coooooool!
|
|
|
|
icicle->fIBufferIdx = instancedParent->fIBufferIdx;
|
|
|
|
icicle->fIStartIdx = instancedParent->fIStartIdx;
|
|
|
|
icicle->fILength = instancedParent->fILength;
|
|
|
|
icicle->fSortData = nil;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
// Pack the indices in (automagically offsets by the start we give it)
|
|
|
|
fGroups[ icicle->fGroupIdx ]->AppendToIndexStorage( geoSpan->fNumIndices, geoSpan->fIndexData,
|
|
|
|
icicle->fVStartIdx, &ibIndex, &ibStart );
|
|
|
|
|
|
|
|
icicle->fIBufferIdx = ibIndex;
|
|
|
|
icicle->fIPackedIdx = icicle->fIStartIdx = ibStart;
|
|
|
|
icicle->fILength = geoSpan->fNumIndices;
|
|
|
|
icicle->fSortData = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// RemoveDIMatrixSpans
|
|
|
|
// Nuke out a matrix only DI index set someone doesn't need anymore (cuz they're dead).
|
|
|
|
// Since there's no data associated, there's not much to do, except mark the slot as
|
|
|
|
// free and that some cleanup is in order some time.
|
|
|
|
void plDrawableSpans::RemoveDIMatrixSpans( uint32_t index )
|
|
|
|
{
|
|
|
|
plDISpanIndex* spanIndices = fDIIndices[ index ];
|
|
|
|
if( !spanIndices->IsMatrixOnly() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
/// Flag that we need garbage cleanup now
|
|
|
|
fNeedCleanup = true;
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
spanIndices->Reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// RemoveDISpans ///////////////////////////////////////////////////////////
|
|
|
|
// Given a drawInterface index (i.e. the index into fDIIndices), deletes
|
|
|
|
// the spans associated with that entry and clears the entry. Also packs
|
|
|
|
// the spans, deletes the verts/indices associated with the deleted spans
|
|
|
|
// and packs the vertex/index buffers as well.
|
|
|
|
|
|
|
|
void plDrawableSpans::RemoveDISpans( uint32_t index )
|
|
|
|
{
|
|
|
|
plDISpanIndex *spanIndices = fDIIndices[ index ];
|
|
|
|
|
|
|
|
if( spanIndices->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
RemoveDIMatrixSpans( index );
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t i, j, k, idxRemoving, materialIdx;
|
|
|
|
|
|
|
|
|
|
|
|
// #define MF_RENDDEP
|
|
|
|
#ifdef MF_RENDDEP
|
|
|
|
hsTArray<uint32_t> spanInverseTable;
|
|
|
|
spanInverseTable.SetCount(fSpans.GetCount());
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
spanInverseTable[i] = i;
|
|
|
|
#endif // MF_RENDDEP
|
|
|
|
|
|
|
|
// hsAssert( fProps & kPropVolatile, "Trying to remove spans on a non-volatile drawable" );
|
|
|
|
hsAssert( spanIndices->GetCount() > 0, "If there are no DI spans, why were we called?" );
|
|
|
|
|
|
|
|
/// Delete the actual spans themselves
|
|
|
|
for( i = 0; i < spanIndices->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
/// If this is the last use of this material, Release() it
|
|
|
|
materialIdx = fSpans[ (*spanIndices)[ i ] ]->fMaterialIdx;
|
|
|
|
|
|
|
|
/// Remove the source index and the object itself
|
|
|
|
if( fSourceSpans.GetCount() && !( fSpans[ (*spanIndices)[ i ] ]->fTypeMask & plSpan::kParticleSpan ) )
|
|
|
|
{
|
|
|
|
delete fSourceSpans[ (*spanIndices)[i] ];
|
|
|
|
fSourceSpans.Remove( (*spanIndices)[i] );
|
|
|
|
for( j = (*spanIndices)[ i ]; j < fSourceSpans.GetCount(); j++ )
|
|
|
|
fSourceSpans[ j ]->fSpanRefIndex = j;
|
|
|
|
}
|
|
|
|
fSpans[ (*spanIndices)[ i ] ]->Destroy();
|
|
|
|
fSpans.Remove( (*spanIndices)[ i ] );
|
|
|
|
idxRemoving = fSpanSourceIndices[ (*spanIndices)[ i ] ];
|
|
|
|
fSpanSourceIndices.Remove( (*spanIndices)[ i ] );
|
|
|
|
|
|
|
|
switch( idxRemoving & kSpanTypeMask )
|
|
|
|
{
|
|
|
|
case kSpanTypeIcicle: fIcicles.Remove( idxRemoving & kSpanIDMask ); break;
|
|
|
|
case kSpanTypeParticleSpan: fParticleSpans.Remove( idxRemoving & kSpanIDMask ); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do this AFTER the span array has been updated
|
|
|
|
ICheckToRemoveMaterial( materialIdx );
|
|
|
|
|
|
|
|
/// Go adjusting the source indices
|
|
|
|
for( j = 0; j < fSpanSourceIndices.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( ( ( fSpanSourceIndices[ j ] ^ idxRemoving ) & kSpanTypeMask ) == 0 )
|
|
|
|
{
|
|
|
|
/// Same type
|
|
|
|
k = fSpanSourceIndices[ j ] & kSpanIDMask;
|
|
|
|
if( k > ( idxRemoving & kSpanIDMask ) )
|
|
|
|
{
|
|
|
|
k--;
|
|
|
|
fSpanSourceIndices[ j ] &= kSpanTypeMask;
|
|
|
|
fSpanSourceIndices[ j ] |= k;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef MF_RENDDEP
|
|
|
|
/// Go adjusting indices in the DI index list
|
|
|
|
for( j = 0; j < fDIIndices.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( !fDIIndices[ j ]->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( k = 0; k < fDIIndices[ j ]->GetCount(); k++ )
|
|
|
|
{
|
|
|
|
if( (*fDIIndices[ j ])[ k ] > (*spanIndices)[ i ] )
|
|
|
|
(*fDIIndices[ j ])[ k ]--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else // MF_RENDDEP
|
|
|
|
spanInverseTable[(*spanIndices)[i]] = -1;
|
|
|
|
for( j = (*spanIndices)[i]; j < fSpans.GetCount(); j++ )
|
|
|
|
spanInverseTable[j]--;
|
|
|
|
#endif // MF_RENDDEP
|
|
|
|
|
|
|
|
}
|
|
|
|
#ifdef MF_RENDDEP
|
|
|
|
for( j = 0; j < fDIIndices.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( !fDIIndices[j]->IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( k = 0; k < fDIIndices[j]->GetCount(); k++ )
|
|
|
|
{
|
|
|
|
int idx = (*fDIIndices[j])[k];
|
|
|
|
hsAssert(idx >= 0, "Just deleted a span another DI was pointing at");
|
|
|
|
(*fDIIndices[j])[k] = spanInverseTable[idx];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // MF_RENDDEP
|
|
|
|
|
|
|
|
/// Rebuild the pointer array, since it's now invalid
|
|
|
|
IRebuildSpanArray();
|
|
|
|
|
|
|
|
/// Flag that we need garbage cleanup now
|
|
|
|
fNeedCleanup = true;
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
/// Clear this entry. Note: DON'T DELETE IT! Otherwise they'll be hell
|
|
|
|
/// to pay. Just clear it, so if we go looking for a free index, we just
|
|
|
|
/// look for one with zero spans.
|
|
|
|
spanIndices->Reset();
|
|
|
|
|
|
|
|
/// All done!
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IRebuildSpanArray ///////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::IRebuildSpanArray( void )
|
|
|
|
{
|
|
|
|
uint32_t j, i;
|
|
|
|
plIcicle *icicle = nil;
|
|
|
|
|
|
|
|
|
|
|
|
for( j = 0; j < fSpans.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
switch( fSpanSourceIndices[ j ] & kSpanTypeMask )
|
|
|
|
{
|
|
|
|
case kSpanTypeIcicle:
|
|
|
|
icicle = &fIcicles[ fSpanSourceIndices[ j ] & kSpanIDMask ];
|
|
|
|
fSpans[ j ] = (plSpan *)icicle;
|
|
|
|
if( icicle->fSortData != nil )
|
|
|
|
{
|
|
|
|
// GOTTA RESET THE SPAN INDICES TOO!!!
|
|
|
|
for( i = 0; i < icicle->fILength / 3; i++ )
|
|
|
|
icicle->fSortData[ i ].fSpanIndex = (uint16_t)j;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kSpanTypeParticleSpan:
|
|
|
|
fSpans[ j ] = (plSpan *)&fParticleSpans[ fSpanSourceIndices[ j ] & kSpanIDMask ];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Redid the span array, so cause the space tree to rebuild to match
|
|
|
|
SetSpaceTree( nil );
|
|
|
|
|
|
|
|
// Rebuild bit vectors for various span types
|
|
|
|
IBuildVectors();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//// ICleanupMatrices
|
|
|
|
// Find all the matrix slots in the palette that aren't being used, collapse
|
|
|
|
// them out and adjust the indices into the matrix palette (the MatrixOnly DIIndices)
|
|
|
|
void plDrawableSpans::ICleanupMatrices()
|
|
|
|
{
|
|
|
|
// We really don't want to do this at runtime. The spans as read in have their bone
|
|
|
|
// matrices fixed (via plSpan::fBaseMatrix && plSpan::fNumMatrices). We can't just
|
|
|
|
// change where in our palette (fLocalToWorlds) bones will point, because the skinned spans
|
|
|
|
// will still be looking in the same place. I'm not sure why we'd want to either.
|
|
|
|
// Commenting out for the moment (so things will work), will look into why this was
|
|
|
|
// ever here, and more mysterious, how it ever worked. mf
|
|
|
|
#if 0
|
|
|
|
hsBitVector usedMatrices;
|
|
|
|
|
|
|
|
int i, j, k;
|
|
|
|
for( i = 0; i < fDIIndices.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plDISpanIndex& indices = *fDIIndices[i];
|
|
|
|
if( indices.IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( j = 0; j < indices.GetCount(); j++ )
|
|
|
|
usedMatrices.SetBit(indices[j]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( j = 0; j < fLocalToWorlds.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( !usedMatrices.IsBitSet(j) )
|
|
|
|
{
|
|
|
|
for( i = 0; i < fDIIndices.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
plDISpanIndex& indices = *fDIIndices[i];
|
|
|
|
if( indices.IsMatrixOnly() )
|
|
|
|
{
|
|
|
|
for( k = 0; k < indices.GetCount(); k++ )
|
|
|
|
{
|
|
|
|
if( indices[k] > j )
|
|
|
|
indices[k]--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for( i = j+1; i < fLocalToWorlds.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
fLocalToWorlds[i] = fLocalToWorlds[i-1];
|
|
|
|
fWorldToLocals[i] = fWorldToLocals[i-1];
|
|
|
|
fLocalToBones[i] = fLocalToBones[i-1];
|
|
|
|
fBoneToLocals[i] = fBoneToLocals[i-1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IRemoveGarbage //////////////////////////////////////////////////////////
|
|
|
|
// Cleans out all the unused spans. Oh, joy.
|
|
|
|
|
|
|
|
void plDrawableSpans::IRemoveGarbage( void )
|
|
|
|
{
|
|
|
|
int groupIdx, ibIdx, i, j, k, count, offset;
|
|
|
|
|
|
|
|
hsTArray<hsTArray<bool> *> usedFlags;
|
|
|
|
hsTArray<hsTArray<hsTArray<bool> *> *> usedIdxFlags;
|
|
|
|
plGBufferGroup *group;
|
|
|
|
|
|
|
|
|
|
|
|
ICleanupMatrices();
|
|
|
|
|
|
|
|
/// Getting rid of verts
|
|
|
|
usedFlags.SetCount( fGroups.GetCount() );
|
|
|
|
usedIdxFlags.SetCount( fGroups.GetCount() );
|
|
|
|
for( i = 0; i < fGroups.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
hsAssert( fGroups[ i ]->GetNumVertexBuffers() == 1, "Cannot clean garbage on a non-volatile buffer group!" );
|
|
|
|
usedFlags[ i ] = TRACKED_NEW hsTArray<bool>;
|
|
|
|
usedFlags[ i ]->SetCountAndZero( fGroups[ i ]->GetVertBufferCount( 0 ) );
|
|
|
|
|
|
|
|
usedIdxFlags[ i ] = TRACKED_NEW hsTArray<hsTArray<bool> *>;
|
|
|
|
usedIdxFlags[ i ]->SetCount( fGroups[ i ]->GetNumIndexBuffers() );
|
|
|
|
for( j = 0; j < fGroups[ i ]->GetNumIndexBuffers(); j++ )
|
|
|
|
{
|
|
|
|
(*usedIdxFlags[ i ])[ j ] = TRACKED_NEW hsTArray<bool>;
|
|
|
|
(*usedIdxFlags[ i ])[ j ]->SetCountAndZero( fGroups[ i ]->GetIndexBufferCount( j ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( i = 0; i < fIcicles.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
// Set the flags so we know these verts are used
|
|
|
|
groupIdx = fIcicles[ i ].fGroupIdx;
|
|
|
|
|
|
|
|
for( j = fIcicles[ i ].fCellOffset; j < fIcicles[ i ].fCellOffset + fIcicles[ i ].fVLength; j++ )
|
|
|
|
(*usedFlags[ groupIdx ])[ j ] = true;
|
|
|
|
|
|
|
|
for( j = fIcicles[ i ].fIStartIdx; j < fIcicles[ i ].fIStartIdx + fIcicles[ i ].fILength; j++ )
|
|
|
|
(*((*usedIdxFlags[ groupIdx ])[ fIcicles[ i ].fIBufferIdx ]))[ j ] = true;
|
|
|
|
}
|
|
|
|
for( i = 0; i < fParticleSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
// Set the flags so we know these verts are used
|
|
|
|
groupIdx = fParticleSpans[ i ].fGroupIdx;
|
|
|
|
|
|
|
|
for( j = fParticleSpans[ i ].fCellOffset; j < fParticleSpans[ i ].fCellOffset + fParticleSpans[ i ].fVLength; j++ )
|
|
|
|
(*usedFlags[ groupIdx ])[ j ] = true;
|
|
|
|
|
|
|
|
for( j = fParticleSpans[ i ].fIStartIdx; j < fParticleSpans[ i ].fIStartIdx + fParticleSpans[ i ].fILength; j++ )
|
|
|
|
(*((*usedIdxFlags[ groupIdx ])[ fParticleSpans[ i ].fIBufferIdx ]))[ j ] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Now loop through and delete any unused
|
|
|
|
for( groupIdx = 0; groupIdx < fGroups.GetCount(); groupIdx++ )
|
|
|
|
{
|
|
|
|
group = fGroups[ groupIdx ];
|
|
|
|
hsTArray<bool> &uFlags = *usedFlags[ groupIdx ];
|
|
|
|
|
|
|
|
// Find groups of verts to delete!
|
|
|
|
count = uFlags.GetCount();
|
|
|
|
for( i = 0; i < count; )
|
|
|
|
{
|
|
|
|
// Skip through used verts
|
|
|
|
for( ; i < count && uFlags[ i ] == true; i++ );
|
|
|
|
|
|
|
|
// Find span of unused verts
|
|
|
|
for( j = i; j < count && uFlags[ j ] == false; j++ );
|
|
|
|
|
|
|
|
if( j > i )
|
|
|
|
{
|
|
|
|
// Delete this span of vertices
|
|
|
|
group->DeleteVertsFromStorage( 0, i, j - i );
|
|
|
|
|
|
|
|
// Adjust indices in this group
|
|
|
|
for( ibIdx = 0; ibIdx < group->GetNumIndexBuffers(); ibIdx++ )
|
|
|
|
group->AdjustIndicesInStorage( ibIdx, i, -(int16_t)( j - i ) );
|
|
|
|
|
|
|
|
// Adjust spans that use this vertex buffer
|
|
|
|
for( k = 0; k < fIcicles.GetCount(); k++ )
|
|
|
|
{
|
|
|
|
if( fIcicles[ k ].fGroupIdx == groupIdx && fIcicles[ k ].fVBufferIdx == 0 &&
|
|
|
|
fIcicles[ k ].fCellOffset >= i )
|
|
|
|
fIcicles[ k ].fCellOffset -= j - i;
|
|
|
|
|
|
|
|
// Adjust sorting data, if necessary
|
|
|
|
if( fIcicles[ k ].fSortData != nil )
|
|
|
|
IAdjustSortData( fIcicles[ k ].fSortData, fIcicles[ k ].fILength / 3, i, -(int16_t)( j - i ) );
|
|
|
|
}
|
|
|
|
for( k = 0; k < fParticleSpans.GetCount(); k++ )
|
|
|
|
{
|
|
|
|
if( fParticleSpans[ k ].fGroupIdx == groupIdx && fParticleSpans[ k ].fVBufferIdx == 0 &&
|
|
|
|
fParticleSpans[ k ].fCellOffset >= i )
|
|
|
|
fParticleSpans[ k ].fCellOffset -= j - i;
|
|
|
|
|
|
|
|
// Adjust sorting data, if necessary
|
|
|
|
if( fParticleSpans[ k ].fSortData != nil )
|
|
|
|
IAdjustSortData( fParticleSpans[ k ].fSortData, fParticleSpans[ k ].fILength / 3, i, -(int16_t)( j - i ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Move the flags too, lest we start deleting the wrong vertices
|
|
|
|
count -= j - i;
|
|
|
|
for( offset = j - i; i < count; i++ )
|
|
|
|
uFlags[ i ] = uFlags[ i + offset ];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep going
|
|
|
|
i = j;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Getting rid of indices now
|
|
|
|
/// IF WE ARE SORTING, then the fIStartIdx positions of the spans are not necessarily
|
|
|
|
/// valid (since they're refreshed every frame, and not all are guaranteed to be refreshed).
|
|
|
|
/// However, since we will be refilling the buffer next frame anyway, it doesn't really
|
|
|
|
/// matter <crosses fingers> WHAT indices we delete, so long as the total length of the
|
|
|
|
/// buffer is correct.
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
{
|
|
|
|
for( ibIdx = 0; ibIdx < group->GetNumIndexBuffers(); ibIdx++ )
|
|
|
|
{
|
|
|
|
// For this index buffer, find the total length we want
|
|
|
|
i = 0;
|
|
|
|
for( j = 0; j < fIcicles.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( fIcicles[ j ].fGroupIdx == groupIdx && fIcicles[ j ].fIBufferIdx == ibIdx )
|
|
|
|
i += fIcicles[ j ].fILength;
|
|
|
|
}
|
|
|
|
for( j = 0; j < fParticleSpans.GetCount(); j++ )
|
|
|
|
{
|
|
|
|
if( fParticleSpans[ j ].fGroupIdx == groupIdx && fParticleSpans[ j ].fIBufferIdx == ibIdx )
|
|
|
|
i += fParticleSpans[ j ].fILength;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// i is the total length
|
|
|
|
group->DeleteIndicesFromStorage( ibIdx, i, group->GetIndexBufferCount( ibIdx ) - i );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// Non-sorting, we have to figure out exactly which indices we aren't using
|
|
|
|
|
|
|
|
for( ibIdx = 0; ibIdx < group->GetNumIndexBuffers(); ibIdx++ )
|
|
|
|
{
|
|
|
|
hsTArray<bool> &uiFlags = *((*usedIdxFlags[ groupIdx ])[ ibIdx ]);
|
|
|
|
|
|
|
|
// Find groups of indices to delete!
|
|
|
|
count = uiFlags.GetCount();
|
|
|
|
for( i = 0; i < count; )
|
|
|
|
{
|
|
|
|
// Skip through used verts
|
|
|
|
for( ; i < count && uiFlags[ i ] == true; i++ );
|
|
|
|
|
|
|
|
// Find span of unused verts
|
|
|
|
for( j = i; j < count && uiFlags[ j ] == false; j++ );
|
|
|
|
|
|
|
|
if( j > i )
|
|
|
|
{
|
|
|
|
// Delete this span of indices
|
|
|
|
group->DeleteIndicesFromStorage( ibIdx, i, j - i );
|
|
|
|
|
|
|
|
// Adjust spans that use this index buffer (only icicles use them)
|
|
|
|
for( k = 0; k < fIcicles.GetCount(); k++ )
|
|
|
|
{
|
|
|
|
if( fIcicles[ k ].fGroupIdx == groupIdx && fIcicles[ k ].fIBufferIdx == ibIdx &&
|
|
|
|
fIcicles[ k ].fIStartIdx >= i )
|
|
|
|
{
|
|
|
|
fIcicles[k].fIPackedIdx = (fIcicles[ k ].fIStartIdx -= j - i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// and particle spans :)
|
|
|
|
for( k = 0; k < fParticleSpans.GetCount(); k++ )
|
|
|
|
{
|
|
|
|
if( fParticleSpans[ k ].fGroupIdx == groupIdx && fParticleSpans[ k ].fIBufferIdx == ibIdx &&
|
|
|
|
fParticleSpans[ k ].fIStartIdx >= i )
|
|
|
|
{
|
|
|
|
fParticleSpans[k].fIPackedIdx = (fParticleSpans[ k ].fIStartIdx -= j - i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Move the flags too, lest we start deleting the wrong vertices
|
|
|
|
count -= j - i;
|
|
|
|
for( offset = j - i; i < count; i++ )
|
|
|
|
uiFlags[ i ] = uiFlags[ i + offset ];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep going
|
|
|
|
i = j;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Refresh all vStartIdx entries
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[ i ]->fTypeMask & plSpan::kVertexSpan )
|
|
|
|
{
|
|
|
|
plVertexSpan *vtx = (plVertexSpan *)fSpans[ i ];
|
|
|
|
vtx->fVStartIdx = fGroups[ vtx->fGroupIdx ]->GetVertStartFromCell( vtx->fVBufferIdx, vtx->fCellIdx, vtx->fCellOffset );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Destroy!!!
|
|
|
|
for( i = 0; i < usedFlags.GetCount(); i++ )
|
|
|
|
delete usedFlags[ i ];
|
|
|
|
usedFlags.Reset();
|
|
|
|
|
|
|
|
for( i = 0; i < usedIdxFlags.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
for( j = 0; j < usedIdxFlags[ i ]->GetCount(); j++ )
|
|
|
|
delete ((*usedIdxFlags[ i ])[ j ]);
|
|
|
|
delete usedIdxFlags[ i ];
|
|
|
|
}
|
|
|
|
usedIdxFlags.Reset();
|
|
|
|
|
|
|
|
/// This will let us query the buffer group as to whether its ready to render or not.
|
|
|
|
/// If not, we'll force it then to reload from its storage
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
// Whew!
|
|
|
|
fNeedCleanup = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IAdjustSortData /////////////////////////////////////////////////////////
|
|
|
|
// Adjusts the indices in the given sort data, using the delta and threshhold
|
|
|
|
// given.
|
|
|
|
|
|
|
|
void plDrawableSpans::IAdjustSortData( plGBufferTriangle *triList, uint32_t count, uint32_t threshhold, int32_t delta )
|
|
|
|
{
|
|
|
|
uint32_t i;
|
|
|
|
|
|
|
|
|
|
|
|
for( i = 0; i < count; i++ )
|
|
|
|
{
|
|
|
|
if( triList[ i ].fIndex1 >= threshhold )
|
|
|
|
triList[ i ].fIndex1 = (uint16_t)( triList[ i ].fIndex1 + delta );
|
|
|
|
if( triList[ i ].fIndex2 >= threshhold )
|
|
|
|
triList[ i ].fIndex2 = (uint16_t)( triList[ i ].fIndex2 + delta );
|
|
|
|
if( triList[ i ].fIndex3 >= threshhold )
|
|
|
|
triList[ i ].fIndex3 = (uint16_t)( triList[ i ].fIndex3 + delta );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IMakeSpanSortable ////////////////////////////////////////////////////////
|
|
|
|
// Given an index of a span, flags it sortable and creates the sorting data
|
|
|
|
// for that span.
|
|
|
|
|
|
|
|
void plDrawableSpans::IMakeSpanSortable( uint32_t index )
|
|
|
|
{
|
|
|
|
plIcicle *span = (plIcicle *)fSpans[ index ];
|
|
|
|
plGBufferTriangle *list;
|
|
|
|
|
|
|
|
|
|
|
|
if( span->fProps & plSpan::kPropFacesSortable )
|
|
|
|
return;
|
|
|
|
|
|
|
|
/// Create data for it
|
|
|
|
list = fGroups[ span->fGroupIdx ]->ConvertToTriList( (int16_t)index, span->fIBufferIdx, span->fVBufferIdx, span->fCellIdx,
|
|
|
|
span->fIStartIdx, span->fILength / 3 );
|
|
|
|
if( list == nil )
|
|
|
|
return;
|
|
|
|
|
|
|
|
span->fSortData = list;
|
|
|
|
|
|
|
|
/// Mark as sortable
|
|
|
|
span->fProps |= plSpan::kPropFacesSortable;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::UnPackCluster(plClusterGroup* cluster)
|
|
|
|
{
|
|
|
|
const uint32_t vertsPerInst = cluster->GetTemplate()->NumVerts();
|
|
|
|
const uint32_t idxPerInst = cluster->GetTemplate()->NumIndices();
|
|
|
|
|
|
|
|
const uint32_t numClust = cluster->GetNumClusters();
|
|
|
|
|
|
|
|
fVisSet |= cluster->GetVisSet();
|
|
|
|
fVisNot |= cluster->GetVisNot();
|
|
|
|
|
|
|
|
fIcicles.SetCount(numClust);
|
|
|
|
fSpans.SetCount(numClust);
|
|
|
|
int iSpan;
|
|
|
|
for( iSpan = 0; iSpan < numClust; iSpan++ )
|
|
|
|
fSpans[iSpan] = &fIcicles[iSpan];
|
|
|
|
iSpan = 0;
|
|
|
|
|
|
|
|
uint32_t vtxFormat =
|
|
|
|
cluster->GetTemplate()->NumUVWs()
|
|
|
|
| cluster->GetTemplate()->NumWeights() << 4;
|
|
|
|
if( cluster->GetTemplate()->NumWgtIdx() )
|
|
|
|
vtxFormat |= plGBufferGroup::kSkinIndices;
|
|
|
|
|
|
|
|
const hsTArray<plLightInfo*>& lights = cluster->GetLights();
|
|
|
|
|
|
|
|
int iStart;
|
|
|
|
for( iStart = 0; iStart < cluster->GetNumClusters(); )
|
|
|
|
{
|
|
|
|
int numVerts = 0;
|
|
|
|
int numIdx = 0;
|
|
|
|
int iEnd;
|
|
|
|
for( iEnd = iStart; iEnd < cluster->GetNumClusters(); iEnd++ )
|
|
|
|
{
|
|
|
|
numVerts += vertsPerInst * cluster->GetCluster(iEnd)->NumInsts();
|
|
|
|
numIdx += idxPerInst * cluster->GetCluster(iEnd)->NumInsts();
|
|
|
|
|
|
|
|
if( (numVerts > plGBufferGroup::kMaxNumVertsPerBuffer)
|
|
|
|
||(numIdx > plGBufferGroup::kMaxNumIndicesPerBuffer) )
|
|
|
|
{
|
|
|
|
// Oops, too much.
|
|
|
|
numVerts -= vertsPerInst * cluster->GetCluster(iEnd)->NumInsts();
|
|
|
|
numIdx -= idxPerInst * cluster->GetCluster(iEnd)->NumInsts();
|
|
|
|
|
|
|
|
iEnd--;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Still in trouble here. We need to fake up that cell crap for each of
|
|
|
|
// our clusters to make a span for it. Whoo-hoo.
|
|
|
|
uint8_t grpIdx = IFindBufferGroup((uint8_t)vtxFormat, numVerts, 0, false, false);
|
|
|
|
|
|
|
|
uint32_t vbufferIdx;
|
|
|
|
uint32_t cellIdx;
|
|
|
|
uint32_t cellOffset;
|
|
|
|
fGroups[grpIdx]->ReserveVertStorage(numVerts, &vbufferIdx, &cellIdx, &cellOffset, plGBufferGroup::kReserveInterleaved | plGBufferGroup::kReserveIsolate);
|
|
|
|
hsAssert(!cellOffset, "This should be our own personal group");
|
|
|
|
hsAssert(fGroups[grpIdx]->GetVertexSize() == cluster->GetTemplate()->Stride(), "Mismatch on src and dst sizes");
|
|
|
|
|
|
|
|
uint32_t ibufferIdx;
|
|
|
|
uint32_t istartIdx;
|
|
|
|
fGroups[grpIdx]->ReserveIndexStorage(numIdx, &ibufferIdx, &istartIdx);
|
|
|
|
uint32_t iOffset = 0;
|
|
|
|
|
|
|
|
uint8_t* vData = fGroups[grpIdx]->GetVertBufferData(vbufferIdx);
|
|
|
|
uint16_t* iData = fGroups[grpIdx]->GetIndexBufferData(ibufferIdx);
|
|
|
|
uint8_t* pvData = vData;
|
|
|
|
uint16_t* piData = iData;
|
|
|
|
int i;
|
|
|
|
for( i = iStart; i < iEnd; i++ )
|
|
|
|
{
|
|
|
|
hsBounds3Ext bnd;
|
|
|
|
cluster->GetCluster(i)->UnPack(pvData, piData, cellOffset, bnd);
|
|
|
|
fIcicles[iSpan].fLocalBounds = bnd;
|
|
|
|
fIcicles[iSpan].fWorldBounds = bnd;
|
|
|
|
|
|
|
|
fIcicles[iSpan].fTypeMask = plSpan::kSpan | plSpan::kVertexSpan | plSpan::kIcicleSpan;
|
|
|
|
// STUB - need to set whether strictly runtime lit or preshaded based on cluster.
|
|
|
|
// fIcicles[iSpan].fProps = plSpan::kLiteMaterial;
|
|
|
|
fIcicles[iSpan].fProps = plSpan::kLiteVtxNonPreshaded | plSpan::kPropRunTimeLight;
|
|
|
|
if (fProps & plSpan::kPropNoDraw)
|
|
|
|
fIcicles[iSpan].fProps |= plSpan::kPropNoDraw;
|
|
|
|
fIcicles[iSpan].fMaterialIdx = 0;
|
|
|
|
fIcicles[iSpan].fLocalToWorld.Reset();
|
|
|
|
fIcicles[iSpan].fWorldToLocal.Reset();
|
|
|
|
fIcicles[iSpan].fBaseMatrix = 0;
|
|
|
|
fIcicles[iSpan].fNumMatrices = 0;
|
|
|
|
fIcicles[iSpan].fLocalUVWChans = 0;
|
|
|
|
fIcicles[iSpan].fMaxBoneIdx = 0;
|
|
|
|
fIcicles[iSpan].fPenBoneIdx = 0;
|
|
|
|
fIcicles[iSpan].fFogEnvironment = nil;
|
|
|
|
fIcicles[iSpan].fMaxDist = cluster->GetLOD().fMaxDist;
|
|
|
|
fIcicles[iSpan].fMinDist = cluster->GetLOD().fMinDist;
|
|
|
|
fIcicles[iSpan].fWaterHeight = 0;
|
|
|
|
fIcicles[iSpan].fVisSet = cluster->GetVisSet();
|
|
|
|
fIcicles[iSpan].fVisNot = cluster->GetVisNot();
|
|
|
|
|
|
|
|
if( lights.GetCount() )
|
|
|
|
{
|
|
|
|
int iLight;
|
|
|
|
for( iLight = 0; iLight < lights.GetCount(); iLight++ )
|
|
|
|
fIcicles[iSpan].AddPermaLight(lights[iLight], lights[iLight]->GetProjection() != nil);
|
|
|
|
}
|
|
|
|
|
|
|
|
fIcicles[iSpan].fGroupIdx = grpIdx;
|
|
|
|
fIcicles[iSpan].fVBufferIdx = vbufferIdx;
|
|
|
|
fIcicles[iSpan].fCellIdx = cellIdx;
|
|
|
|
fIcicles[iSpan].fCellOffset = cellOffset;
|
|
|
|
fIcicles[iSpan].fVStartIdx = cellOffset;
|
|
|
|
const uint32_t numVerts = cluster->GetCluster(i)->NumInsts() * vertsPerInst;
|
|
|
|
fIcicles[iSpan].fVLength = numVerts;
|
|
|
|
cellOffset += numVerts;
|
|
|
|
|
|
|
|
fIcicles[iSpan].fIBufferIdx = ibufferIdx;
|
|
|
|
fIcicles[iSpan].fIPackedIdx = fIcicles[iSpan].fIStartIdx = iOffset;
|
|
|
|
const uint32_t iLength = cluster->GetCluster(i)->NumInsts() * cluster->GetTemplate()->NumIndices();
|
|
|
|
fIcicles[iSpan].fILength = iLength;
|
|
|
|
iOffset += iLength;
|
|
|
|
|
|
|
|
iSpan++;
|
|
|
|
|
|
|
|
const uint32_t vSize = cluster->GetCluster(i)->NumInsts() * cluster->GetTemplate()->VertSize();
|
|
|
|
pvData += vSize;
|
|
|
|
piData += iLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
iStart = iEnd;
|
|
|
|
}
|
|
|
|
fMaterials.SetCountAndZero(1);
|
|
|
|
plGenRefMsg* refMsg = TRACKED_NEW plGenRefMsg(GetKey(), plRefMsg::kOnCreate, 0, kMsgMaterial);
|
|
|
|
hsgResMgr::ResMgr()->SendRef(cluster->GetMaterial()->GetKey(), refMsg, plRefFlags::kActiveRef);
|
|
|
|
|
|
|
|
fRenderLevel = cluster->GetRenderLevel();
|
|
|
|
GetSpaceTree();
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IFindBufferGroup ////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
uint8_t plDrawableSpans::IFindBufferGroup(uint8_t vtxFormat, uint32_t numVertsNeeded, int lod, hsBool vertVolatile, hsBool idxVolatile)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
|
|
// Scan through the buffer groups, looking for a good format. If there isn't
|
|
|
|
// one, add it
|
|
|
|
if( fProps & kPropVolatile )
|
|
|
|
vertVolatile = true;
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
idxVolatile = true;
|
|
|
|
|
|
|
|
for( i = 0; i < fGroups.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( (fGroups[ i ]->GetVertexFormat() == vtxFormat)
|
|
|
|
&& (!vertVolatile == !fGroups[i]->AreVertsVolatile())
|
|
|
|
&& (!idxVolatile == !fGroups[i]->AreIdxVolatile())
|
|
|
|
&& (lod == fGroups[i]->GetLOD()) )
|
|
|
|
{
|
|
|
|
if( fGroups[ i ]->GetNumPrimaryVertsLeft() >= numVertsNeeded )
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a new one of the right format
|
|
|
|
fGroups.Append( TRACKED_NEW plGBufferGroup(vtxFormat, vertVolatile, idxVolatile, lod) );
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// GetParticleSpanVector ///////////////////////////////////////////////////
|
|
|
|
// Get a bitVector of the spans that are particle spans
|
|
|
|
|
|
|
|
hsBitVector const &plDrawableSpans::GetParticleSpanVector( void ) const
|
|
|
|
{
|
|
|
|
return fParticleSpanVector;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// GetBlendingSpanVector ///////////////////////////////////////////////////
|
|
|
|
// Get a bitVector of the spans that are blending (i.e. numMatrices > 0)
|
|
|
|
|
|
|
|
hsBitVector const &plDrawableSpans::GetBlendingSpanVector( void ) const
|
|
|
|
{
|
|
|
|
/// See the plRenderMsg handler for why we do this
|
|
|
|
return fFakeBlendingSpanVector;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// SetBlendingSpanVectorBit ////////////////////////////////////////////////
|
|
|
|
// Could just make GetBlendingSpanVector() non-const, but this way it's harder
|
|
|
|
// for the end user, thus I'm hoping to discourage them from doing it too much.
|
|
|
|
|
|
|
|
void plDrawableSpans::SetBlendingSpanVectorBit( uint32_t bitNumber, hsBool on )
|
|
|
|
{
|
|
|
|
fFakeBlendingSpanVector.SetBit( bitNumber, on );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IBuildVectors ///////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::IBuildVectors( void )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
bool needRenderMsg = false;
|
|
|
|
|
|
|
|
|
|
|
|
fParticleSpanVector.Clear();
|
|
|
|
fBlendingSpanVector.Clear();
|
|
|
|
for( i = 0; i < fSpans.GetCount(); i++ )
|
|
|
|
{
|
|
|
|
if( fSpans[ i ]->fTypeMask & plSpan::kParticleSpan )
|
|
|
|
fParticleSpanVector.SetBit( i );
|
|
|
|
// BoneUpdate
|
|
|
|
if( ( fSpans[ i ]->fTypeMask & plSpan::kVertexSpan ) && (fSpans[ i ]->fNumMatrices > 2) )
|
|
|
|
// if( ( fSpans[ i ]->fTypeMask & plSpan::kVertexSpan ) && (fSpans[ i ]->fNumMatrices > 1) )
|
|
|
|
{
|
|
|
|
fBlendingSpanVector.SetBit( i );
|
|
|
|
needRenderMsg = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Reset this sucker too
|
|
|
|
fFakeBlendingSpanVector = fBlendingSpanVector;
|
|
|
|
|
|
|
|
if( needRenderMsg && !fRegisteredForRender )
|
|
|
|
{
|
|
|
|
// Placeholder hack - see IUpdateMatrixPaletteBoundsHack() for comments
|
|
|
|
fRegisteredForRender = true;
|
|
|
|
plgDispatch::Dispatch()->RegisterForExactType( plRenderMsg::Index(), GetKey() );
|
|
|
|
}
|
|
|
|
else if( !needRenderMsg && fRegisteredForRender )
|
|
|
|
{
|
|
|
|
plgDispatch::Dispatch()->UnRegisterForExactType( plRenderMsg::Index(), GetKey() );
|
|
|
|
fRegisteredForRender = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
//// Particle Sets ///////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
//// CreateParticleSystem ////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::CreateParticleSystem( uint32_t maxNumSpans, uint32_t maxNumParticles, hsGMaterial *material )
|
|
|
|
{
|
|
|
|
uint32_t i, numVerts, numIndices;
|
|
|
|
plParticleSet *set;
|
|
|
|
plDISpanIndex *spanLookup;
|
|
|
|
|
|
|
|
|
|
|
|
// Make a shiny new set
|
|
|
|
set = TRACKED_NEW plParticleSet;
|
|
|
|
set->fRefCount = 0;
|
|
|
|
|
|
|
|
/// Fill out info
|
|
|
|
numVerts = maxNumParticles * 4; // 4 verts per particle
|
|
|
|
numIndices = maxNumParticles * 6; // 6 indices per particle
|
|
|
|
|
|
|
|
if( material != nil && material->GetLayer( 0 ) != nil && material->GetLayer( 0 )->GetTexture() != nil )
|
|
|
|
set->fFormat = plGeometrySpan::UVCountToFormat( 1 );
|
|
|
|
else
|
|
|
|
set->fFormat = plGeometrySpan::UVCountToFormat( 0 );
|
|
|
|
|
|
|
|
// Reserve space
|
|
|
|
set->fGroupIdx = IFindBufferGroup( set->fFormat, numVerts, 0, true, false ); // Force them to volatile even if the drawable isn't
|
|
|
|
fGroups[ set->fGroupIdx ]->ReserveVertStorage( numVerts, &set->fVBufferIdx, &set->fCellIdx, &set->fCellOffset, plGBufferGroup::kReserveInterleaved | plGBufferGroup::kReserveIsolate );
|
|
|
|
fGroups[ set->fGroupIdx ]->ReserveIndexStorage( numIndices, &set->fIBufferIdx, &set->fIStartIdx );
|
|
|
|
|
|
|
|
set->fVStartIdx = fGroups[ set->fGroupIdx ]->GetVertStartFromCell( set->fVBufferIdx, set->fCellIdx, set->fCellOffset );
|
|
|
|
set->fVLength = numVerts;
|
|
|
|
set->fILength = numIndices;
|
|
|
|
|
|
|
|
uint32_t vIdx = set->fVStartIdx;
|
|
|
|
uint16_t* idx = fGroups[set->fGroupIdx]->GetIndexBufferData(set->fIBufferIdx) + set->fIStartIdx;
|
|
|
|
for( i = 0; i < maxNumParticles; i++ )
|
|
|
|
{
|
|
|
|
*idx++ = (uint16_t)vIdx;
|
|
|
|
*idx++ = (uint16_t)(vIdx + 1);
|
|
|
|
*idx++ = (uint16_t)(vIdx + 2);
|
|
|
|
|
|
|
|
*idx++ = (uint16_t)vIdx;
|
|
|
|
*idx++ = (uint16_t)(vIdx + 2);
|
|
|
|
*idx++ = (uint16_t)(vIdx + 3);
|
|
|
|
|
|
|
|
vIdx += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create us some spans
|
|
|
|
set->fDIEntry = -1;
|
|
|
|
spanLookup = IFindDIIndices( set->fDIEntry );
|
|
|
|
spanLookup->fFlags |= plDISpanIndex::kDontTransformSpans;
|
|
|
|
|
|
|
|
set->fNumSpans = maxNumSpans;
|
|
|
|
for( i = 0; i < maxNumSpans; i++ )
|
|
|
|
ICreateParticleIcicle( material, set );
|
|
|
|
|
|
|
|
set->fNextIStartIdx = set->fIStartIdx;
|
|
|
|
set->fNextVStartIdx = set->fVStartIdx;
|
|
|
|
set->fNextCellOffset = set->fCellOffset;
|
|
|
|
|
|
|
|
|
|
|
|
/// Rebuild the pointer array
|
|
|
|
IRebuildSpanArray();
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
return set->fDIEntry;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IAssignMatIdxToSpan /////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::IAssignMatIdxToSpan( plSpan *span, hsGMaterial *mtl )
|
|
|
|
{
|
|
|
|
fSettingMatIdxLock = true;
|
|
|
|
|
|
|
|
if( mtl != nil )
|
|
|
|
span->fMaterialIdx = IAddAMaterial( mtl );
|
|
|
|
|
|
|
|
if( fMaterials[ span->fMaterialIdx ] != nil )
|
|
|
|
{
|
|
|
|
if( ITestMatForSpecularity( fMaterials[ span->fMaterialIdx ] ) )
|
|
|
|
span->fProps |= plSpan::kPropMatHasSpecular;
|
|
|
|
else
|
|
|
|
span->fProps &= ~plSpan::kPropMatHasSpecular;
|
|
|
|
}
|
|
|
|
|
|
|
|
fSettingMatIdxLock = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ICreateParticleIcicle ///////////////////////////////////////////////////
|
|
|
|
// Creates a shiny new plIcicle for use with particle sets.
|
|
|
|
|
|
|
|
plParticleSpan *plDrawableSpans::ICreateParticleIcicle( hsGMaterial *material, plParticleSet *set )
|
|
|
|
{
|
|
|
|
uint32_t spanIdx;
|
|
|
|
hsMatrix44 ident;
|
|
|
|
plDISpanIndex *spanLookup;
|
|
|
|
|
|
|
|
|
|
|
|
/// Ahh, an icicle span
|
|
|
|
spanIdx = fParticleSpans.GetCount();
|
|
|
|
fParticleSpans.Append( plParticleSpan() );
|
|
|
|
plParticleSpan *icicle = &fParticleSpans[ spanIdx ];
|
|
|
|
|
|
|
|
ident.Reset();
|
|
|
|
IAssignMatIdxToSpan( icicle, material );
|
|
|
|
icicle->fMaterialIdx = IAddAMaterial( material );
|
|
|
|
icicle->fLocalToWorld = ident;
|
|
|
|
icicle->fWorldToLocal = ident;
|
|
|
|
icicle->fProps |= plSpan::kPropRunTimeLight | plSpan::kPropNoDraw | plSpan::kLiteVtxPreshaded;
|
|
|
|
if( fProps & kPropSortSpans )
|
|
|
|
icicle->fProps |= plSpan::kPropFacesSortable;
|
|
|
|
|
|
|
|
icicle->fLocalBounds.MakeEmpty();
|
|
|
|
icicle->fWorldBounds.MakeEmpty();
|
|
|
|
|
|
|
|
hsPoint3 zero(0, 0, 0);
|
|
|
|
icicle->fLocalBounds.Union( &zero );
|
|
|
|
icicle->fWorldBounds.Union( &zero );
|
|
|
|
|
|
|
|
icicle->fGroupIdx = set->fGroupIdx;
|
|
|
|
icicle->fVBufferIdx = set->fVBufferIdx;
|
|
|
|
icicle->fCellIdx = set->fCellIdx;
|
|
|
|
icicle->fCellOffset = set->fCellOffset;
|
|
|
|
icicle->fVStartIdx = set->fVStartIdx;
|
|
|
|
icicle->fVLength = 0;
|
|
|
|
|
|
|
|
icicle->fIBufferIdx = set->fIBufferIdx;
|
|
|
|
icicle->fIPackedIdx = icicle->fIStartIdx = set->fIStartIdx;
|
|
|
|
icicle->fILength = 0;
|
|
|
|
icicle->fFogEnvironment = nil;
|
|
|
|
icicle->fSortData = nil;
|
|
|
|
|
|
|
|
icicle->fSrcSpanIdx = fSpans.GetCount();
|
|
|
|
|
|
|
|
spanLookup = IFindDIIndices( set->fDIEntry );
|
|
|
|
spanLookup->Append( fSpans.GetCount() );
|
|
|
|
fSpans.Append( (plSpan *)icicle );
|
|
|
|
fSpanSourceIndices.Append( spanIdx | kSpanTypeParticleSpan );
|
|
|
|
|
|
|
|
/// Set our pointer to the set and inc it's refCount
|
|
|
|
icicle->fParentSet = set;
|
|
|
|
set->fRefCount++;
|
|
|
|
|
|
|
|
/// Flag that we need garbage cleanup now
|
|
|
|
fNeedCleanup = true;
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
// Cause us to rebuild the space tree when needed
|
|
|
|
SetSpaceTree( nil );
|
|
|
|
|
|
|
|
return icicle;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// ResetParticleSystem /////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::ResetParticleSystem( uint32_t setIndex )
|
|
|
|
{
|
|
|
|
uint32_t i;
|
|
|
|
plDISpanIndex *indices = IFindDIIndices( setIndex );
|
|
|
|
plParticleSet *set;
|
|
|
|
plParticleSpan *span;
|
|
|
|
|
|
|
|
for( i = 0; i < indices->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
span = (plParticleSpan *)fSpans[ (*indices)[ i ] ];
|
|
|
|
span->fVStartIdx = 0;
|
|
|
|
span->fCellIdx = 0;
|
|
|
|
span->fCellOffset = 0;
|
|
|
|
span->fVLength = 0;
|
|
|
|
span->fIPackedIdx = span->fIStartIdx = 0;
|
|
|
|
span->fILength = 0;
|
|
|
|
span->fSource = nil;
|
|
|
|
span->fProps |= plSpan::kPropNoDraw;
|
|
|
|
GetSpaceTree()->SetLeafFlag( (int16_t)(span->fSrcSpanIdx), plSpaceTreeNode::kDisabled );
|
|
|
|
|
|
|
|
set = span->fParentSet; // To use at the end of the loop
|
|
|
|
}
|
|
|
|
|
|
|
|
set->fNextIStartIdx = set->fIStartIdx;
|
|
|
|
set->fNextVStartIdx = set->fVStartIdx;
|
|
|
|
set->fNextCellOffset = set->fCellOffset;
|
|
|
|
|
|
|
|
fGroups[set->fGroupIdx]->SetVertBufferEnd(set->fVBufferIdx, set->fNextVStartIdx);
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
fGroups[set->fGroupIdx]->SetIndexBufferEnd(set->fIBufferIdx, set->fNextIStartIdx);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//// AssignEmitterToParticleSystem ///////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::AssignEmitterToParticleSystem( uint32_t setIndex, plParticleEmitter *emitter )
|
|
|
|
{
|
|
|
|
plDISpanIndex *indices = IFindDIIndices( setIndex );
|
|
|
|
plParticleSet *set;
|
|
|
|
plParticleSpan *icicle;
|
|
|
|
uint32_t i, index, numParticles = emitter->GetParticleCount();
|
|
|
|
plParticleCore *array = emitter->GetParticleArray();
|
|
|
|
hsMatrix44 ident;
|
|
|
|
|
|
|
|
plGBufferTriangle *sortArray;
|
|
|
|
|
|
|
|
|
|
|
|
icicle = (plParticleSpan *)fSpans[ (*indices)[ emitter->GetSpanIndex() ] ];
|
|
|
|
set = icicle->fParentSet;
|
|
|
|
|
|
|
|
icicle->fCellIdx = set->fCellIdx;
|
|
|
|
icicle->fCellOffset = set->fNextCellOffset;
|
|
|
|
icicle->fVStartIdx = set->fNextVStartIdx;
|
|
|
|
icicle->fIPackedIdx = icicle->fIStartIdx = set->fNextIStartIdx;
|
|
|
|
icicle->fVLength = numParticles * 4;
|
|
|
|
icicle->fILength = numParticles * 6;
|
|
|
|
icicle->fSource = emitter;
|
|
|
|
icicle->fProps &= ~plSpan::kPropNoDraw;
|
|
|
|
GetSpaceTree()->ClearLeafFlag( (int16_t)(icicle->fSrcSpanIdx), plSpaceTreeNode::kDisabled );
|
|
|
|
icicle->fNumParticles = numParticles;
|
|
|
|
set->fNextVStartIdx += numParticles * 4;
|
|
|
|
set->fNextIStartIdx += numParticles * 6;
|
|
|
|
set->fNextCellOffset += numParticles * 4;
|
|
|
|
|
|
|
|
fGroups[set->fGroupIdx]->SetVertBufferEnd(set->fVBufferIdx, set->fNextVStartIdx);
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
fGroups[set->fGroupIdx]->SetIndexBufferEnd(set->fIBufferIdx, set->fNextIStartIdx);
|
|
|
|
|
|
|
|
hsAssert( set->fNextVStartIdx <= set->fVStartIdx + set->fVLength, "Buffer overflow in AddParticlesToSet()" );
|
|
|
|
hsAssert( set->fNextIStartIdx <= set->fIStartIdx + set->fILength, "Buffer overflow in AddParticlesToSet()" );
|
|
|
|
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
{
|
|
|
|
// Prep sorting data
|
|
|
|
if( icicle->fSortData == nil || icicle->fSortCount < ( numParticles << 1 ) )
|
|
|
|
{
|
|
|
|
delete [] icicle->fSortData;
|
|
|
|
icicle->fSortData = sortArray = TRACKED_NEW plGBufferTriangle[ numParticles << 1 ];
|
|
|
|
icicle->fSortCount = numParticles << 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
sortArray = icicle->fSortData;
|
|
|
|
|
|
|
|
// Find our bounds and fill out sorting data
|
|
|
|
for( i = 0, index = icicle->fVStartIdx; i < numParticles; i++ )
|
|
|
|
{
|
|
|
|
sortArray->fCenter = array[ i ].fPos;
|
|
|
|
sortArray->fIndex1 = (uint16_t)index;
|
|
|
|
sortArray->fIndex2 = (uint16_t)(index + 1);
|
|
|
|
sortArray->fIndex3 = (uint16_t)(index + 2);
|
|
|
|
sortArray->fSpanIndex = (uint16_t)(icicle->fSrcSpanIdx);
|
|
|
|
sortArray++;
|
|
|
|
|
|
|
|
sortArray->fCenter = array[ i ].fPos;
|
|
|
|
sortArray->fIndex1 = (uint16_t)index;
|
|
|
|
sortArray->fIndex2 = (uint16_t)(index + 2);
|
|
|
|
sortArray->fIndex3 = (uint16_t)(index + 3);
|
|
|
|
sortArray->fSpanIndex = (uint16_t)(icicle->fSrcSpanIdx);
|
|
|
|
sortArray++;
|
|
|
|
|
|
|
|
index += 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
icicle->fLocalBounds = emitter->GetBoundingBox();
|
|
|
|
icicle->fWorldBounds = icicle->fLocalBounds;
|
|
|
|
|
|
|
|
GetSpaceTree()->MoveLeaf( (int16_t)(icicle->fSrcSpanIdx), icicle->fWorldBounds );
|
|
|
|
// Done for now, we'll fill on the pipeline side
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
//// Dynamic Functions for SceneViewer ///////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
//// RefreshDISpans //////////////////////////////////////////////////////////
|
|
|
|
// Given a set of plGeometrySpans, replaces the data in the given DI span set
|
|
|
|
// with the vertices in the plGeometrySpans. Does NOT replace the indices,
|
|
|
|
// and the count must be EXACT, or this function gets pyst.
|
|
|
|
//MFREPACK
|
|
|
|
// don't need to pass in spans, they're already in fSourceSpans.
|
|
|
|
// clearSpansAfterRefresh is ignored anyway, nuke it.
|
|
|
|
uint32_t plDrawableSpans::RefreshDISpans( uint32_t index )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
uint32_t spanIdx;
|
|
|
|
plSpan *span;
|
|
|
|
hsBounds3Ext bounds;
|
|
|
|
plDISpanIndex *spanLookup;
|
|
|
|
|
|
|
|
// hsAssert( fProps & kPropVolatile, "Trying to add spans on a non-volatile drawable" );
|
|
|
|
hsAssert( index != (uint32_t)-1, "Invalid DI span index" );
|
|
|
|
|
|
|
|
/// Do garbage cleanup first
|
|
|
|
if( fNeedCleanup )
|
|
|
|
IRemoveGarbage();
|
|
|
|
|
|
|
|
spanLookup = IFindDIIndices( index );
|
|
|
|
|
|
|
|
/// Loop through the spans and copy the vertex data over
|
|
|
|
for( i = 0; i < spanLookup->GetCount(); i++ )
|
|
|
|
{
|
|
|
|
// Main info
|
|
|
|
plGeometrySpan *geoSpan = fSourceSpans[ (*spanLookup)[i] ];
|
|
|
|
spanIdx = fSpanSourceIndices[ (*spanLookup)[ i ] ];
|
|
|
|
hsAssert( ( spanIdx & kSpanTypeMask ) == kSpanTypeIcicle, "Mismatch in span formats" );
|
|
|
|
plIcicle *icicle = &fIcicles[ spanIdx & kSpanIDMask ];
|
|
|
|
IUpdateIcicleFromGeoSpan( geoSpan, icicle );
|
|
|
|
span = (plSpan *)icicle;
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
{
|
|
|
|
// Should add sort data too...
|
|
|
|
IMakeSpanSortable( (*spanLookup)[ i ] );
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update fLocalBounds, since we were updating the world bounds
|
|
|
|
fLocalBounds = fWorldBounds;
|
|
|
|
fLocalBounds.Transform( &fWorldToLocal );
|
|
|
|
fMaxWorldBounds = fWorldBounds;
|
|
|
|
|
|
|
|
// Cause us to rebuild the space tree when needed
|
|
|
|
SetSpaceTree( nil );
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Same as above, but takes an actual span index (not a DI Index).
|
|
|
|
uint32_t plDrawableSpans::RefreshSpan( uint32_t index )
|
|
|
|
{
|
|
|
|
uint32_t spanIdx;
|
|
|
|
plSpan *span;
|
|
|
|
hsBounds3Ext bounds;
|
|
|
|
|
|
|
|
// hsAssert( fProps & kPropVolatile, "Trying to add spans on a non-volatile drawable" );
|
|
|
|
hsAssert( index < fSourceSpans.GetCount(), "Invalid span index" );
|
|
|
|
|
|
|
|
/// Do garbage cleanup first
|
|
|
|
if( fNeedCleanup )
|
|
|
|
IRemoveGarbage();
|
|
|
|
|
|
|
|
// Main info
|
|
|
|
plGeometrySpan *geoSpan = fSourceSpans[ index ];
|
|
|
|
spanIdx = fSpanSourceIndices[ index ];
|
|
|
|
hsAssert( ( spanIdx & kSpanTypeMask ) == kSpanTypeIcicle, "Mismatch in span formats" );
|
|
|
|
plIcicle *icicle = &fIcicles[ spanIdx & kSpanIDMask ];
|
|
|
|
IUpdateIcicleFromGeoSpan( geoSpan, icicle );
|
|
|
|
span = (plSpan *)icicle;
|
|
|
|
if( fProps & kPropSortFaces )
|
|
|
|
{
|
|
|
|
// Should add sort data too...
|
|
|
|
IMakeSpanSortable( index );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update fLocalBounds, since we were updating the world bounds
|
|
|
|
fLocalBounds = fWorldBounds;
|
|
|
|
fLocalBounds.Transform( &fWorldToLocal );
|
|
|
|
fMaxWorldBounds = fWorldBounds;
|
|
|
|
|
|
|
|
// Cause us to rebuild the space tree when needed
|
|
|
|
SetSpaceTree( nil );
|
|
|
|
|
|
|
|
fReadyToRender = false;
|
|
|
|
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IUpdateVertexSpanFromGeoSpan ////////////////////////////////////////////
|
|
|
|
// Common function for the two functions that follow.
|
|
|
|
|
|
|
|
void plDrawableSpans::IUpdateVertexSpanFromGeoSpan( plGeometrySpan *geoSpan, plVertexSpan *span )
|
|
|
|
{
|
|
|
|
hsBounds3Ext bounds;
|
|
|
|
|
|
|
|
// This function looks pretty dubious to me. It or's in properties instead of setting them, and
|
|
|
|
// it doesn't set all the available properties (like the skinning matrix indices). It clearly should
|
|
|
|
// be doing more or doing less, but it's unclear which. I'm guessing the latter, but until there's
|
|
|
|
// a reason to dig through this trash, here it stays. mf
|
|
|
|
|
|
|
|
hsAssert( geoSpan->fNumVerts == span->fVLength, "Vertex count mismatch in IUpdateVertexSpanFromGeoSpan()" );
|
|
|
|
|
|
|
|
span->fLocalToWorld = geoSpan->fLocalToWorld;
|
|
|
|
span->fWorldToLocal = geoSpan->fWorldToLocal;
|
|
|
|
span->fProps |= ( geoSpan->fProps & plGeometrySpan::kPropRunTimeLight ) ? plSpan::kPropRunTimeLight : 0;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropNoShadowCast )
|
|
|
|
span->fProps |= plSpan::kPropNoShadowCast;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropNoShadow )
|
|
|
|
span->fProps |= plSpan::kPropNoShadow;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropForceShadow )
|
|
|
|
span->fProps |= plSpan::kPropForceShadow;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPropReverseSort )
|
|
|
|
span->fProps |= plSpan::kPropReverseSort;
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kPartialSort )
|
|
|
|
span->fProps |= plSpan::kPartialSort;
|
|
|
|
switch( geoSpan->fProps & plGeometrySpan::kLiteMask )
|
|
|
|
{
|
|
|
|
case plGeometrySpan::kLiteMaterial: span->fProps |= plSpan::kLiteMaterial; break;
|
|
|
|
case plGeometrySpan::kLiteVtxPreshaded: span->fProps |= plSpan::kLiteVtxPreshaded; break;
|
|
|
|
case plGeometrySpan::kLiteVtxNonPreshaded: span->fProps |= plSpan::kLiteVtxNonPreshaded; break;
|
|
|
|
}
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kWaterHeight )
|
|
|
|
{
|
|
|
|
span->fProps |= plSpan::kWaterHeight;
|
|
|
|
span->fWaterHeight = geoSpan->fWaterHeight;
|
|
|
|
}
|
|
|
|
if( geoSpan->fProps & plGeometrySpan::kVisLOS )
|
|
|
|
{
|
|
|
|
span->fProps |= plSpan::kVisLOS;
|
|
|
|
fProps |= plDrawable::kPropHasVisLOS;
|
|
|
|
}
|
|
|
|
|
|
|
|
bounds = geoSpan->fLocalBounds;
|
|
|
|
span->fLocalBounds = bounds;
|
|
|
|
bounds.Transform( &span->fLocalToWorld );
|
|
|
|
span->fWorldBounds = bounds;
|
|
|
|
fWorldBounds.Union( &bounds );
|
|
|
|
|
|
|
|
// Pack the vertices in
|
|
|
|
fGroups[ span->fGroupIdx ]->StuffToVertStorage( geoSpan, span->fVBufferIdx, span->fCellIdx, span->fCellOffset,
|
|
|
|
plGBufferGroup::kReserveInterleaved );
|
|
|
|
|
|
|
|
span->fFogEnvironment = geoSpan->fFogEnviron;
|
|
|
|
|
|
|
|
/// Add the material to our list if necessary
|
|
|
|
IAssignMatIdxToSpan( span, geoSpan->fMaterial );
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IUpdateIcicleFromGeoSpan ////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void plDrawableSpans::IUpdateIcicleFromGeoSpan( plGeometrySpan *geoSpan, plIcicle *icicle )
|
|
|
|
{
|
|
|
|
IUpdateVertexSpanFromGeoSpan( geoSpan, icicle );
|
|
|
|
}
|
|
|
|
|
|
|
|
hsPoint3& plDrawableSpans::GetPosition(int spanIdx, int vtxIdx)
|
|
|
|
{
|
|
|
|
hsAssert(fSpans[spanIdx]->fTypeMask & plSpan::kVertexSpan, "Getting vertex info on non-vertex based span");
|
|
|
|
plVertexSpan* span = (plVertexSpan*)fSpans[spanIdx];
|
|
|
|
|
|
|
|
return fGroups[span->fGroupIdx]->Position(span->fVBufferIdx, span->fCellIdx, vtxIdx);
|
|
|
|
}
|
|
|
|
hsVector3& plDrawableSpans::GetNormal(int spanIdx, int vtxIdx)
|
|
|
|
{
|
|
|
|
hsAssert(fSpans[spanIdx]->fTypeMask & plSpan::kVertexSpan, "Getting vertex info on non-vertex based span");
|
|
|
|
plVertexSpan* span = (plVertexSpan*)fSpans[spanIdx];
|
|
|
|
|
|
|
|
return fGroups[span->fGroupIdx]->Normal(span->fVBufferIdx, span->fCellIdx, vtxIdx);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::GetNumTris(int spanIdx)
|
|
|
|
{
|
|
|
|
hsAssert(fSpans[spanIdx]->fTypeMask & plSpan::kIcicleSpan, "Asking for index info on non-triangle based span");
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[spanIdx];
|
|
|
|
return fGroups[span->fGroupIdx]->GetIndexBufferCount(span->fIBufferIdx) / 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t* plDrawableSpans::GetIndexList(int spanIdx)
|
|
|
|
{
|
|
|
|
hsAssert(fSpans[spanIdx]->fTypeMask & plSpan::kIcicleSpan, "Asking for index info on non-triangle based span");
|
|
|
|
plIcicle* span = (plIcicle*)fSpans[spanIdx];
|
|
|
|
return fGroups[span->fGroupIdx]->GetIndexBufferData(span->fIBufferIdx);
|
|
|
|
}
|
|
|
|
|
|
|
|
hsPoint3& plDrawableSpans::CvtGetPosition(int spanIdx, int vtxIdx)
|
|
|
|
{
|
|
|
|
return *(hsPoint3*)(fSourceSpans[spanIdx]->fVertexData + vtxIdx * plGeometrySpan::GetVertexSize(fSourceSpans[spanIdx]->fFormat));
|
|
|
|
}
|
|
|
|
|
|
|
|
hsVector3& plDrawableSpans::CvtGetNormal(int spanIdx, int vtxIdx)
|
|
|
|
{
|
|
|
|
// Normal follows the position (+ 12Bytes)
|
|
|
|
return *(hsVector3*)(fSourceSpans[spanIdx]->fVertexData + vtxIdx * plGeometrySpan::GetVertexSize(fSourceSpans[spanIdx]->fFormat) + 12);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::CvtGetNumTris(int spanIdx)
|
|
|
|
{
|
|
|
|
return fSourceSpans[spanIdx]->fNumIndices / 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t* plDrawableSpans::CvtGetIndexList(int spanIdx)
|
|
|
|
{
|
|
|
|
return fSourceSpans[spanIdx]->fIndexData;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t plDrawableSpans::CvtGetNumVerts(int spanIdx) const
|
|
|
|
{
|
|
|
|
return fSourceSpans[spanIdx]->fNumVerts;
|
|
|
|
}
|
|
|
|
|
|
|
|
plGeometrySpan* plDrawableSpans::GetGeometrySpan(int spanIdx)
|
|
|
|
{
|
|
|
|
return fSourceSpans[spanIdx];
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::GetOrigGeometrySpans( uint32_t diIndex, hsTArray<plGeometrySpan *> &arrayToFill )
|
|
|
|
{
|
|
|
|
if( diIndex >= fDIIndices.GetCount() )
|
|
|
|
{
|
|
|
|
hsAssert( false, "Invalid diIndex to GetOrigGeometrySpans()" );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
plDISpanIndex *indices = fDIIndices[ diIndex ];
|
|
|
|
uint32_t i;
|
|
|
|
|
|
|
|
|
|
|
|
arrayToFill.Reset();
|
|
|
|
for( i = 0; i < indices->GetCount(); i++ )
|
|
|
|
arrayToFill.Append( fSourceSpans[ (*indices)[ i ] ] );
|
|
|
|
|
|
|
|
// HA!
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plDrawableSpans::ClearAndSetMaterialCount(uint32_t count)
|
|
|
|
{
|
|
|
|
// Release the old materials
|
|
|
|
for (int i = 0; i < fMaterials.GetCount(); i++)
|
|
|
|
{
|
|
|
|
if ( fMaterials[ i ] != nil && fMaterials[ i ]->GetKey() != nil )
|
|
|
|
GetKey()->Release( fMaterials[ i ]->GetKey() );
|
|
|
|
}
|
|
|
|
|
|
|
|
fMaterials.SetCountAndZero(count);
|
|
|
|
}
|
|
|
|
|