You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3962 lines
133 KiB
3962 lines
133 KiB
/*==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 "HeadSpin.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); |
|
} |
|
|
|
|