source: GTP/trunk/Lib/Vis/Preprocessing/src/HierarchyManager.cpp @ 1889

Revision 1889, 64.7 KB checked in by mattausch, 18 years ago (diff)
RevLine 
[1237]1#include <stack>
2#include <time.h>
3#include <iomanip>
4
5#include "ViewCell.h"
6#include "Plane3.h"
7#include "HierarchyManager.h"
8#include "Mesh.h"
9#include "common.h"
10#include "Environment.h"
11#include "Polygon3.h"
12#include "Ray.h"
13#include "AxisAlignedBox3.h"
14#include "Exporter.h"
15#include "Plane3.h"
16#include "ViewCellsManager.h"
17#include "Beam.h"
18#include "KdTree.h"
[1315]19#include "IntersectableWrapper.h"
[1237]20#include "VspTree.h"
21#include "OspTree.h"
[1259]22#include "BvHierarchy.h"
[1707]23#include "ViewCell.h"
[1237]24
25
26namespace GtpVisibilityPreprocessor {
27
28
29#define USE_FIXEDPOINT_T 0
30
[1786]31#define STUPID_METHOD 0
[1237]32
33/*******************************************************************/
34/*              class HierarchyManager implementation              */
35/*******************************************************************/
36
37
[1421]38HierarchyManager::HierarchyManager(const int objectSpaceSubdivisionType):
[1308]39mObjectSpaceSubdivisionType(objectSpaceSubdivisionType),
[1279]40mOspTree(NULL),
41mBvHierarchy(NULL)
[1237]42{
[1308]43        switch(mObjectSpaceSubdivisionType)
[1279]44        {
45        case KD_BASED_OBJ_SUBDIV:
46                mOspTree = new OspTree();
47                mOspTree->mVspTree = mVspTree;
[1379]48                mOspTree->mHierarchyManager = this;
[1279]49                break;
50        case BV_BASED_OBJ_SUBDIV:
51        mBvHierarchy = new BvHierarchy();
[1379]52                mBvHierarchy->mHierarchyManager = this;
[1279]53                break;
54        default:
55                break;
56        }
57
[1379]58        // hierarchy manager links view space partition and object space partition
[1421]59        mVspTree = new VspTree();
[1379]60        mVspTree->mHierarchyManager = this;
61       
[1580]62        mViewSpaceSubdivisionType = KD_BASED_VIEWSPACE_SUBDIV;
[1288]63        ParseEnvironment();
[1237]64}
65
66
[1421]67HierarchyManager::HierarchyManager(KdTree *kdTree):
[1308]68mObjectSpaceSubdivisionType(KD_BASED_OBJ_SUBDIV),
[1279]69mBvHierarchy(NULL)
70{
71        mOspTree = new OspTree(*kdTree);
72        mOspTree->mVspTree = mVspTree;
73
[1421]74        mVspTree = new VspTree();
[1379]75        mVspTree->mHierarchyManager = this;
[1279]76
[1580]77        mViewSpaceSubdivisionType = KD_BASED_VIEWSPACE_SUBDIV;
[1288]78        ParseEnvironment();
79}
80
81
[1723]82void HierarchySubdivisionStats::Print(ostream &app) const
83{
84        app << "#Pass\n" << 0 << endl
85                << "#Splits\n" << mNumSplits << endl
86                << "#TotalRenderCost\n" << mTotalRenderCost << endl
87                << "#TotalEntriesInPvs\n" << mEntriesInPvs << endl
88                << "#Memory\n" << mMemoryCost << endl
89                << "#StepsView\n" << mViewSpaceSplits << endl
90                << "#StepsObject\n" << mObjectSpaceSplits << endl
91                << "#VspOspRatio\n" << VspOspRatio() << endl
92                << "#FullMem\n" << mFullMemory << endl
93                << "#RenderCostDecrease\n" << mRenderCostDecrease << endl
[1744]94                << "#Priority\n" << mPriority << endl
[1723]95                << "#FpsPerMb\n" << FpsPerMb() << endl
96                << endl;
97}
98
99
[1288]100void HierarchyManager::ParseEnvironment()
101{
102        Environment::GetSingleton()->GetFloatValue(
103                "Hierarchy.Termination.minGlobalCostRatio", mTermMinGlobalCostRatio);
104        Environment::GetSingleton()->GetIntValue(
105                "Hierarchy.Termination.globalCostMissTolerance", mTermGlobalCostMissTolerance);
106
[1370]107        Environment::GetSingleton()->GetBoolValue(
108                "Hierarchy.Construction.startWithObjectSpace", mStartWithObjectSpace);
109
[1293]110        Environment::GetSingleton()->GetIntValue(
[1294]111                "Hierarchy.Termination.maxLeaves", mTermMaxLeaves);
112
113        Environment::GetSingleton()->GetIntValue(
[1293]114                "Hierarchy.Construction.type", mConstructionType);
115
[1311]116        Environment::GetSingleton()->GetIntValue(
117                "Hierarchy.Construction.minDepthForOsp", mMinDepthForObjectSpaceSubdivion);
[1370]118
119        Environment::GetSingleton()->GetIntValue(
120                "Hierarchy.Construction.minDepthForVsp", mMinDepthForViewSpaceSubdivion);
[1311]121       
[1314]122        Environment::GetSingleton()->GetBoolValue(
123                "Hierarchy.Construction.repairQueue", mRepairQueue);
124
[1449]125        Environment::GetSingleton()->GetBoolValue(
[1564]126                "Hierarchy.Construction.useMultiLevel", mUseMultiLevelConstruction);
[1449]127
[1580]128        Environment::GetSingleton()->GetIntValue(
129                "Hierarchy.Construction.levels", mNumMultiLevels);
130
[1640]131        Environment::GetSingleton()->GetIntValue(
132                "Hierarchy.Construction.minStepsOfSameType", mMinStepsOfSameType);
133       
[1684]134        Environment::GetSingleton()->GetIntValue(
135                "Hierarchy.Construction.maxStepsOfSameType", mMaxStepsOfSameType);
136
[1624]137        char subdivisionStatsLog[100];
138        Environment::GetSingleton()->GetStringValue("Hierarchy.subdivisionStats", subdivisionStatsLog);
139        mSubdivisionStats.open(subdivisionStatsLog);
140
[1633]141        Environment::GetSingleton()->GetBoolValue(
142                "Hierarchy.Construction.recomputeSplitPlaneOnRepair", mRecomputeSplitPlaneOnRepair);
[1624]143
[1662]144        Environment::GetSingleton()->GetBoolValue(
145                "Hierarchy.Construction.considerMemory", mConsiderMemory);
146
[1649]147        Environment::GetSingleton()->GetFloatValue(
148                "Hierarchy.Termination.maxMemory", mTermMaxMemory);
149
[1727]150        Environment::GetSingleton()->GetIntValue(
151                "Hierarchy.Construction.maxRepairs", mMaxRepairs);
[1736]152
[1662]153        // compare to bytes
[1653]154        mTermMaxMemory *= (1024.0f * 1024.0f);
155
[1705]156        Debug << "******** Hierarchy Manager Options ***********" << endl;
[1294]157        Debug << "max leaves: " << mTermMaxLeaves << endl;
[1288]158        Debug << "min global cost ratio: " << mTermMinGlobalCostRatio << endl;
159        Debug << "global cost miss tolerance: " << mTermGlobalCostMissTolerance << endl;
[1314]160        Debug << "min depth for object space subdivision: " << mMinDepthForObjectSpaceSubdivion << endl;
161        Debug << "repair queue: " << mRepairQueue << endl;
[1580]162        Debug << "number of multilevels: " << mNumMultiLevels << endl;
[1633]163        Debug << "recompute split plane on repair: " << mRecomputeSplitPlaneOnRepair << endl;
[1640]164        Debug << "minimal number of steps from same type: " << mMinStepsOfSameType << endl;
[1649]165        Debug << "maximal allowed memory: " << mTermMaxMemory << endl;
[1673]166        Debug << "consider memory: " << mConsiderMemory << endl;
[1684]167        Debug << "min steps of same kind: " << mMinStepsOfSameType << endl;
168        Debug << "max steps of same kind: " << mMaxStepsOfSameType << endl;
[1727]169        Debug << "max repairs: " << mMaxRepairs << endl;
[1632]170
171        switch (mConstructionType)
172        {
173        case 0:
174                Debug << "construction type: sequential" << endl;
175                break;
176        case 1:
177                Debug << "construction type: interleaved" << endl;
178                break;
179        case 2:
180                Debug << "construction type: gradient" << endl;
181                break;
182        case 3:
183                Debug << "construction type: multilevel" << endl;
184                break;
185        default:
186                Debug << "construction type " << mConstructionType << " unknown" << endl;
187                break;
188        }
189
[1625]190        //Debug << "min render cost " << mMinRenderCostDecrease << endl;
[1449]191        Debug << endl;
[1279]192}
193
194
195HierarchyManager::~HierarchyManager()
196{
197        DEL_PTR(mOspTree);
[1421]198        DEL_PTR(mVspTree);
[1279]199        DEL_PTR(mBvHierarchy);
200}
201
202
[1370]203int HierarchyManager::GetObjectSpaceSubdivisionType() const
204{
205        return mObjectSpaceSubdivisionType;
206}
207
208
209int HierarchyManager::GetViewSpaceSubdivisionType() const
210{
211        return mViewSpaceSubdivisionType;
212}
213
214
[1279]215void HierarchyManager::SetViewCellsManager(ViewCellsManager *vcm)
216{
217        mVspTree->SetViewCellsManager(vcm);
218
219        if (mOspTree)
[1576]220        {
[1279]221                mOspTree->SetViewCellsManager(vcm);
[1576]222        }
[1416]223        else if (mBvHierarchy)
[1576]224        {
[1279]225                mBvHierarchy->SetViewCellsManager(vcm);
[1576]226        }
[1279]227}
228
229
230void HierarchyManager::SetViewCellsTree(ViewCellsTree *vcTree)
231{
232        mVspTree->SetViewCellsTree(vcTree);
233}
234
235
[1379]236VspTree *HierarchyManager::GetVspTree()
237{
238        return mVspTree;
239}
240
[1563]241/*
[1379]242AxisAlignedBox3 HierarchyManager::GetViewSpaceBox() const
243{
244        return mVspTree->mBoundingBox;
[1563]245}*/
[1379]246
247
[1416]248AxisAlignedBox3 HierarchyManager::GetObjectSpaceBox() const
249{
250        switch (mObjectSpaceSubdivisionType)
251        {
252        case KD_BASED_OBJ_SUBDIV:
253                return mOspTree->mBoundingBox;
254        case BV_BASED_OBJ_SUBDIV:
255                return mBvHierarchy->mBoundingBox;
256        default:
[1576]257                // hack: empty box
[1416]258                return AxisAlignedBox3();
259        }
260}
261
262
[1625]263SubdivisionCandidate *HierarchyManager::NextSubdivisionCandidate(SplitQueue &splitQueue)
[1237]264{
[1625]265        SubdivisionCandidate *splitCandidate = splitQueue.Top();
266        splitQueue.Pop();
[1237]267
[1733]268        // split was not reevaluated before => do it now
269        if (splitCandidate->IsDirty())
270                splitCandidate->EvalCandidate();
271
[1237]272        return splitCandidate;
273}
274
275
[1830]276float HierarchyManager::EvalFullMem() const
277{
278        // question: should I also add the mem usage of the hierarchies?
279        const float objectSpaceMem = 16;//GetObjectSpaceMemUsage();
280        const float viewSpaceMem = 16;//mVspTree->GetMemUsage();
281        // HACK: the same value is used for view and object space
282        return mHierarchyStats.mMemory + mHierarchyStats.Leaves() * objectSpaceMem;
283}
284
285
[1889]286float HierarchyManager::AvgRayContribution(const float pvs, const float rays) const
287{
288        return (float)pvs / rays;
289}
290
291
[1624]292void HierarchyManager::EvalSubdivisionStats()
[1237]293{
[1830]294       
[1723]295        HierarchySubdivisionStats stats;
[1744]296
[1723]297        stats.mNumSplits = mHierarchyStats.Leaves();
298        stats.mTotalRenderCost = mHierarchyStats.mTotalCost;
299        stats.mEntriesInPvs = mHierarchyStats.mPvsEntries;
300        stats.mMemoryCost = mHierarchyStats.mMemory  / float(1024 * 1024);
[1830]301        stats.mFullMemory = EvalFullMem() / float(1024 * 1024);
[1723]302        stats.mViewSpaceSplits = mVspTree->mVspStats.Leaves();
303        stats.mObjectSpaceSplits = GetObjectSpaceSubdivisionLeaves();
[1744]304        stats.mRenderCostDecrease = mHierarchyStats.mRenderCostDecrease;
305        stats.mPriority = mPriority;
306
[1723]307        stats.Print(mSubdivisionStats);
[1237]308}
309
310
311void HierarchyManager::AddSubdivisionStats(const int splits,
312                                                                                   const float renderCostDecr,
[1576]313                                                                                   const float totalRenderCost,
[1625]314                                                                                   const int pvsEntries,
[1640]315                                                                                   const float memory,
[1654]316                                                                                   const float renderCostPerStorage,
317                                                                                   const float vspOspRatio)
[1237]318{
[1308]319        mSubdivisionStats
[1237]320                        << "#Splits\n" << splits << endl
321                        << "#RenderCostDecrease\n" << renderCostDecr << endl
[1577]322                        << "#TotalEntriesInPvs\n" << pvsEntries << endl
[1625]323                        << "#TotalRenderCost\n" << totalRenderCost << endl
324                        << "#Memory\n" << memory << endl
[1660]325                        << "#FpsPerMb\n" << renderCostPerStorage << endl
[1723]326                        << "#VspOspRatio\n" << vspOspRatio << endl
327                        << endl;
[1237]328}
329
330
331bool HierarchyManager::GlobalTerminationCriteriaMet(SubdivisionCandidate *candidate) const
332{
[1421]333        const bool terminationCriteriaMet =
334                (0
[1294]335                || (mHierarchyStats.Leaves() >= mTermMaxLeaves)
[1830]336                //|| (mHierarchyStats.mMemory >= mTermMaxMemory)
337                || (EvalFullMem() >= mTermMaxMemory)
[1649]338                || candidate->GlobalTerminationCriteriaMet()
339                //|| (mHierarchyStats.mRenderCostDecrease < mMinRenderCostDecrease)
[1633]340                //|| (mHierarchyStats.mGlobalCostMisses >= mTermGlobalCostMissTolerance)
[1288]341                );
[1421]342
[1715]343#if GTP_DEBUG
[1610]344        if (terminationCriteriaMet)
[1421]345        {
346                Debug << "hierarchy global termination criteria met:" << endl;
347                Debug << "leaves: " << mHierarchyStats.Leaves() << " " << mTermMaxLeaves << endl;
[1449]348                Debug << "cost misses: " << mHierarchyStats.mGlobalCostMisses << " " << mTermGlobalCostMissTolerance << endl;
[1649]349                Debug << "memory: " << mHierarchyStats.mMemory << " " << mTermMaxMemory << endl;
[1830]350                Debug << "full memory: " << EvalFullMem() << " " << mTermMaxMemory << endl;
[1421]351        }
[1653]352#endif
[1610]353
[1421]354        return terminationCriteriaMet;
[1237]355}
356
357
358void HierarchyManager::Construct(const VssRayContainer &sampleRays,
[1311]359                                                                 const ObjectContainer &objects,
360                                                                 AxisAlignedBox3 *forcedViewSpace)
[1237]361{
[1679]362        mTimeStamp = 1;
363
[1627]364        switch (mConstructionType)
[1449]365        {
[1627]366        case MULTILEVEL:
[1449]367                ConstructMultiLevel(sampleRays, objects, forcedViewSpace);
[1627]368                break;
369        case INTERLEAVED:
370        case SEQUENTIAL:
[1624]371                ConstructInterleaved(sampleRays, objects, forcedViewSpace);
[1627]372                break;
373        case GRADIENT:
374                ConstructInterleavedWithGradient(sampleRays, objects, forcedViewSpace);
375                break;
376        default:
377                break;
[1624]378        }
[1642]379
380        // hack: should be different parameter name
381        if (mUseMultiLevelConstruction)
382        {
383                cout << "starting optimizing multilevel ... " << endl;
384                // try to optimize on the above hierarchy
385                OptimizeMultiLevel(sampleRays, objects, forcedViewSpace);
386               
387                cout << "finished" << endl;
388        }
[1624]389}
390
391
[1686]392void HierarchyManager::ConstructInterleavedWithGradient(const VssRayContainer &sampleRays,
[1626]393                                                                                                                const ObjectContainer &objects,
394                                                                                                                AxisAlignedBox3 *forcedViewSpace)
[1624]395{
396        mHierarchyStats.Reset();
397        mHierarchyStats.Start();
[1625]398       
[1624]399        mHierarchyStats.mNodes = 2;
[1640]400
401        // create first nodes
402        mVspTree->Initialise(sampleRays, forcedViewSpace);
403        InitialiseObjectSpaceSubdivision(objects);
404
405        // hack: assume that object space can be seen from view space
[1677]406        mHierarchyStats.mTotalCost = mInitialRenderCost = (float)objects.size();
[1667]407        // only one entry for start
[1640]408        mHierarchyStats.mPvsEntries = 1;
[1667]409        mHierarchyStats.mMemory = (float)ObjectPvs::GetEntrySizeByte();
[1640]410
411        EvalSubdivisionStats();
[1624]412        Debug << "setting total cost to " << mHierarchyStats.mTotalCost << endl;
413
414        const long startTime = GetTime();
415        cout << "Constructing view space / object space tree ... \n";
416       
[1633]417        SplitQueue objectSpaceQueue;
418        SplitQueue viewSpaceQueue;
419
[1696]420        int vspSteps = 0, ospSteps = 0;
[1684]421
[1624]422        // use sah for evaluating osp tree construction
423        // in the first iteration of the subdivision
424        mSavedViewSpaceSubdivisionType = mViewSpaceSubdivisionType;
425        mViewSpaceSubdivisionType = NO_VIEWSPACE_SUBDIV;
[1662]426        mSavedObjectSpaceSubdivisionType = mObjectSpaceSubdivisionType;
[1624]427
[1632]428        // number of initial splits
[1640]429        const int minSteps = mMinStepsOfSameType;
[1684]430        const int maxSteps = mMaxStepsOfSameType;
[1625]431
[1779]432        PrepareObjectSpaceSubdivision(objectSpaceQueue, sampleRays, objects);
433       
[1633]434        /////////////////////////
435        // calulcate initial object space splits
436       
[1684]437        SubdivisionCandidateContainer dirtyList;
[1633]438
439        // subdivide object space first
440        // for first round, use sah splits. Once view space partition
441        // has started, use render cost heuristics instead
[1727]442        ospSteps = RunConstruction(objectSpaceQueue,
443                                                           dirtyList,
444                                                           NULL,
445                                                           minSteps,
446                                                           maxSteps);
447
[1640]448        cout << "\n" << ospSteps << " object space partition steps taken" << endl;
[1633]449
[1640]450        // create view space
[1779]451        PrepareViewSpaceSubdivision(viewSpaceQueue, sampleRays, objects);
[1633]452
[1684]453        dirtyList.clear();
454
[1642]455        // view space subdivision started
[1633]456        mViewSpaceSubdivisionType = mSavedViewSpaceSubdivisionType;
457
[1684]458        if (1)
459        {
460                // rather also start with 100 view space splits to avoid initial bias.
461                vspSteps = RunConstruction(viewSpaceQueue, dirtyList, NULL, minSteps, maxSteps);
462                cout << "\n" << vspSteps << " view space partition steps taken" << endl;
463               
464                /// Repair split queue
465                cout << "repairing queue ... " << endl;
466                RepairQueue(dirtyList, objectSpaceQueue, true);
467                cout << "repaired " << (int)dirtyList.size() << " candidates" << endl;
[1633]468
[1684]469                dirtyList.clear();
470        }
471        else
472        {
473                // the priorities were calculated for driving sah.
474                // => recalculate "real" priorities taking visibility into
475                // account so we can compare to view space splits
476                ResetQueue(objectSpaceQueue, false);
477        }
478
[1624]479        // This method subdivides view space / object space
480        // in order to converge to some optimal cost for this partition
481        // start with object space partiton
482        // then optimizate view space partition for the current osp
483        // and vice versa until iteration depth is reached.
[1696]484
[1684]485        bool lastSplitWasOsp = true;
486
[1633]487        while (!(viewSpaceQueue.Empty() && objectSpaceQueue.Empty()))
[1624]488        {
[1662]489                // decide upon next split type
[1784]490                const float vspPriority =
491                        viewSpaceQueue.Top() ? viewSpaceQueue.Top()->GetPriority() : -1e20f;
492                const float ospPriority =
493                        objectSpaceQueue.Top() ? objectSpaceQueue.Top()->GetPriority() : -1e20f;
[1662]494               
[1640]495                cout << "new decicion, vsp: " << vspPriority << ", osp: " << ospPriority << endl;
496
[1633]497                // should view or object space be subdivided further?
[1684]498                if (ospPriority >= vspPriority)
499                //if (!lastSplitWasOsp)
[1633]500                {
[1679]501                        lastSplitWasOsp = true;
[1673]502                        cout << "osp" << endl;
[1684]503                       
[1633]504                        // dirtied view space candidates
505                        SubdivisionCandidateContainer dirtyVspList;
[1624]506
[1727]507                        // subdivide object space first for first round,
508                        // use sah splits. Once view space partition
[1633]509                        // has started, use render cost heuristics instead
[1727]510                        const int ospSteps = RunConstruction(objectSpaceQueue,
511                                                                                                 dirtyVspList,
512                                                                                                 viewSpaceQueue.Top(),
513                                                                                                 minSteps,
514                                                                                                 maxSteps);
[1624]515
[1640]516                        cout << "\n" << ospSteps << " object space partition steps taken" << endl;
[1684]517                        Debug << "\n" << ospSteps << " object space partition steps taken" << endl;
518
[1640]519                        /// Repair split queue, i.e., affected view space candidates
[1633]520                        cout << "repairing queue ... " << endl;
[1736]521                        const int repaired = RepairQueue(dirtyVspList, viewSpaceQueue, true);
[1727]522           
[1784]523                        cout << "\nrepaired " << repaired << " candidates from "
524                                 << (int)dirtyVspList.size() << " dirtied candidates" << endl;
[1633]525                }
526                else
527                {
[1679]528                        lastSplitWasOsp = false;
[1673]529                        cout << "vsp" << endl;
530                       
[1633]531                        /////////////////
532                        // subdivide view space with respect to the objects
[1625]533
[1684]534                        // dirtied object space candidates
[1633]535                        SubdivisionCandidateContainer dirtyOspList;
[1624]536
[1633]537                        // process view space candidates
[1727]538                        const int vspSteps = RunConstruction(viewSpaceQueue,
539                                                                                                 dirtyOspList,
540                                                                                                 objectSpaceQueue.Top(),
541                                                                                                 minSteps,
542                                                                                                 maxSteps);
[1624]543
[1640]544                        cout << "\n" << vspSteps << " view space partition steps taken" << endl;
[1684]545                        Debug << "\n" << vspSteps << " view space partition steps taken" << endl;
[1625]546
[1633]547                        // view space subdivision constructed
548                        mViewSpaceSubdivisionType = mSavedViewSpaceSubdivisionType;
[1624]549
[1633]550                        /// Repair split queue
551                        cout << "repairing queue ... " << endl;
[1736]552                        const int repaired = RepairQueue(dirtyOspList, objectSpaceQueue, true);
[1727]553
[1784]554                        cout << "\nrepaired " << repaired << " candidates from "
555                                 << (int)dirtyOspList.size() << " dirtied candidates" << endl;
[1633]556                }
[1449]557        }
[1633]558
[1624]559        cout << "\nfinished in " << TimeDiff(startTime, GetTime()) * 1e-3 << " secs" << endl;
560
561        mHierarchyStats.Stop();
562        mVspTree->mVspStats.Stop();
[1625]563
[1649]564        FinishObjectSpaceSubdivision(objects, !mUseMultiLevelConstruction);
[1449]565}
566
567
568void HierarchyManager::ConstructInterleaved(const VssRayContainer &sampleRays,
569                                                                                        const ObjectContainer &objects,
570                                                                                        AxisAlignedBox3 *forcedViewSpace)
571{
[1308]572        mHierarchyStats.Reset();
573        mHierarchyStats.Start();
[1664]574
575        // two nodes for view space and object space
576        mHierarchyStats.mNodes = 2;
[1640]577        mHierarchyStats.mPvsEntries = 1;
[1667]578        mHierarchyStats.mMemory = (float)ObjectPvs::GetEntrySizeByte();
[1640]579        mHierarchyStats.mTotalCost = (float)objects.size();
[1449]580
[1624]581        mHierarchyStats.mRenderCostDecrease = 0;
[1237]582
[1624]583        EvalSubdivisionStats();
584        Debug << "setting total cost to " << mHierarchyStats.mTotalCost << endl;
585
[1311]586        const long startTime = GetTime();
[1308]587        cout << "Constructing view space / object space tree ... \n";
[1237]588       
[1640]589        // create only roots
590        mVspTree->Initialise(sampleRays, forcedViewSpace);
591        InitialiseObjectSpaceSubdivision(objects);
[1379]592
[1633]593        // use objects for evaluating vsp tree construction in the
594        // first levels of the subdivision
[1323]595        mSavedObjectSpaceSubdivisionType = mObjectSpaceSubdivisionType;
596        mObjectSpaceSubdivisionType = NO_OBJ_SUBDIV;
597
[1370]598        mSavedViewSpaceSubdivisionType = mViewSpaceSubdivisionType;
599        mViewSpaceSubdivisionType = NO_VIEWSPACE_SUBDIV;
600
[1632]601        // start view space subdivison immediately?
[1329]602        if (StartViewSpaceSubdivision())
603        {
[1624]604                // prepare vsp tree for traversal
[1640]605        mViewSpaceSubdivisionType = mSavedViewSpaceSubdivisionType;
[1779]606                PrepareViewSpaceSubdivision(mTQueue, sampleRays, objects);
[1329]607        }
608       
[1323]609        // start object space subdivision immediately?
610        if (StartObjectSpaceSubdivision())
611        {
612                mObjectSpaceSubdivisionType = mSavedObjectSpaceSubdivisionType;
[1779]613                PrepareObjectSpaceSubdivision(mTQueue, sampleRays, objects);
[1323]614        }
[1784]615       
[1624]616        // begin subdivision
[1667]617        RunConstruction(mRepairQueue, sampleRays, objects, forcedViewSpace);
[1308]618       
[1418]619        cout << "\nfinished in " << TimeDiff(startTime, GetTime()) * 1e-3 << " secs" << endl;
[1370]620
[1548]621        mObjectSpaceSubdivisionType = mSavedObjectSpaceSubdivisionType;
622        mViewSpaceSubdivisionType = mSavedViewSpaceSubdivisionType;
623
[1313]624        mHierarchyStats.Stop();
[1259]625        mVspTree->mVspStats.Stop();
[1649]626       
627        FinishObjectSpaceSubdivision(objects, !mUseMultiLevelConstruction);
[1237]628}
629
630
[1779]631void HierarchyManager::PrepareViewSpaceSubdivision(SplitQueue &tQueue,
632                                                                                                   const VssRayContainer &sampleRays,
633                                                                                                   const ObjectContainer &objects)
[1311]634{
[1632]635        cout << "\npreparing view space hierarchy construction ... " << endl;
[1625]636
[1580]637        // hack: reset global cost misses
[1548]638        mHierarchyStats.mGlobalCostMisses = 0;
[1625]639
[1311]640        RayInfoContainer *viewSpaceRays = new RayInfoContainer();
[1779]641        mVspTree->PrepareConstruction(tQueue, sampleRays, *viewSpaceRays);
[1311]642
[1661]643        /////////
644        //-- new stats
645
[1624]646        mHierarchyStats.mTotalCost = mVspTree->mTotalCost;
647        cout << "\nreseting cost for vsp, new total cost: " << mHierarchyStats.mTotalCost << endl;
[1311]648}
649
650
[1640]651float HierarchyManager::GetObjectSpaceMemUsage() const
652{
653        if (mObjectSpaceSubdivisionType == KD_BASED_OBJ_SUBDIV)
654        {
655                // TODO;
656        }
657        else if (mObjectSpaceSubdivisionType == BV_BASED_OBJ_SUBDIV)
658        {
659                return mBvHierarchy->GetMemUsage();
660        }
661
662        return -1;
663}
664
[1844]665
[1640]666void HierarchyManager::InitialiseObjectSpaceSubdivision(const ObjectContainer &objects)
667{
668        if (mObjectSpaceSubdivisionType == KD_BASED_OBJ_SUBDIV)
669        {
670                // TODO;
671        }
672        else if (mObjectSpaceSubdivisionType == BV_BASED_OBJ_SUBDIV)
673        {
674                mBvHierarchy->Initialise(objects);
675        }
676}
677
678
[1779]679void HierarchyManager::PrepareObjectSpaceSubdivision(SplitQueue &tQueue,
680                                                                                                         const VssRayContainer &sampleRays,
681                                                                                                         const ObjectContainer &objects)
[1308]682{
[1625]683        // hack: reset global cost misses
684        mHierarchyStats.mGlobalCostMisses = 0;
[1522]685
[1308]686        if (mObjectSpaceSubdivisionType == KD_BASED_OBJ_SUBDIV)
687        {
[1779]688                return PrepareOspTree(tQueue, sampleRays, objects);
[1308]689        }
690        else if (mObjectSpaceSubdivisionType == BV_BASED_OBJ_SUBDIV)
691        {
[1779]692                return PrepareBvHierarchy(tQueue, sampleRays, objects);
[1308]693        }
694}
[1294]695
[1308]696
[1779]697void HierarchyManager::PrepareBvHierarchy(SplitQueue &tQueue,
698                                                                                  const VssRayContainer &sampleRays,
699                                                                                  const ObjectContainer &objects)
[1294]700{
[1421]701        const long startTime = GetTime();
702
703        cout << "preparing bv hierarchy construction ... " << endl;
[1548]704       
[1294]705        // compute first candidate
[1779]706        mBvHierarchy->PrepareConstruction(tQueue, sampleRays, objects);
[1294]707
[1624]708        mHierarchyStats.mTotalCost = mBvHierarchy->mTotalCost;
709        Debug << "\nreseting cost, new total cost: " << mHierarchyStats.mTotalCost << endl;
[1294]710
[1548]711        cout << "finished bv hierarchy preparation in "
712                 << TimeDiff(startTime, GetTime()) * 1e-3 << " secs" << endl;
[1294]713}
714
715
[1779]716void HierarchyManager::PrepareOspTree(SplitQueue &tQueue,
717                                                                          const VssRayContainer &sampleRays,
718                                                                          const ObjectContainer &objects)
[1294]719{
[1370]720        cout << "starting osp tree construction ... " << endl;
721
[1294]722        RayInfoContainer *objectSpaceRays = new RayInfoContainer();
723
724        // start with one big kd cell - all objects can be seen from everywhere
725        // note: only true for view space = object space
726
727        // compute first candidate
[1779]728        mOspTree->PrepareConstruction(tQueue, sampleRays, objects, *objectSpaceRays);
[1294]729
[1624]730        mHierarchyStats.mTotalCost = mOspTree->mTotalCost;
[1625]731        Debug << "\nreseting cost for osp, new total cost: " << mHierarchyStats.mTotalCost << endl;
[1294]732}
733
734
[1624]735bool HierarchyManager::ApplySubdivisionCandidate(SubdivisionCandidate *sc,
[1625]736                                                                                                 SplitQueue &splitQueue,
[1624]737                                                                                                 const bool repairQueue)
[1237]738{
[1624]739        const bool terminationCriteriaMet = GlobalTerminationCriteriaMet(sc);
[1633]740        const bool success = sc->Apply(splitQueue, terminationCriteriaMet);
[1444]741
[1744]742        if (sc->IsDirty())
[1745]743                cerr << "Error: Should never come here!" << endl;
[1744]744
[1610]745        if (!success) // split was not taken
746        {
[1763]747                cout << "x";
[1610]748                return false;
749        }
750
[1744]751        //cout << "priority: " << sc->GetPriority() << " rc decr: " << sc->GetRenderCostDecrease() << " | ";
[1610]752        ///////////////
[1624]753        //-- split was successful => update stats and queue
[1610]754
755    // cost ratio of cost decrease / totalCost
[1624]756        const float costRatio = sc->GetRenderCostDecrease() / mHierarchyStats.mTotalCost;
[1744]757        //cout << "ratio: " << costRatio << " min ratio: " << mTermMinGlobalCostRatio << endl;
[1610]758       
759        if (costRatio < mTermMinGlobalCostRatio)
760        {
761                ++ mHierarchyStats.mGlobalCostMisses;
762        }
763       
[1580]764        cout << sc->Type() << " ";
765               
[1640]766        /////////////
[1580]767        // update stats
[1640]768
[1624]769        mHierarchyStats.mNodes += 2;
[1640]770        mHierarchyStats.mTotalCost -= sc->GetRenderCostDecrease();
771
772        const int pvsEntriesIncr = sc->GetPvsEntriesIncr();
773        mHierarchyStats.mPvsEntries += pvsEntriesIncr;
774        //cout << "pvs entries: " << pvsEntriesIncr << " " << mHierarchyStats.pvsEntries << endl;
775
776        // memory size in byte
[1676]777        mHierarchyStats.mMemory += (float)ObjectPvs::GetEntrySizeByte() * pvsEntriesIncr;
[1624]778        mHierarchyStats.mRenderCostDecrease = sc->GetRenderCostDecrease();
[1744]779       
780        mPriority = sc->GetPriority();
[1624]781
[1654]782        static float memoryCount = 0;
783
784        if (mHierarchyStats.mMemory > memoryCount)
785        {
786                memoryCount += 100000;
[1684]787                cout << "\nstorage cost: " << mHierarchyStats.mMemory / float(1024 * 1024)
788                         << " MB, steps: " << mHierarchyStats.Leaves() << endl;
[1654]789        }
790
[1640]791        // output stats
[1624]792        EvalSubdivisionStats();
[1580]793               
794        if (repairQueue)
[1624]795        {
796                // reevaluate candidates affected by the split for view space splits,
797                // this would be object space splits and other way round
[1625]798                vector<SubdivisionCandidate *> dirtyList;
[1633]799                sc->CollectDirtyCandidates(dirtyList, false);
[1624]800
[1633]801                RepairQueue(dirtyList, splitQueue, mRecomputeSplitPlaneOnRepair);
[1580]802        }
803
[1416]804        return true;
[1237]805}
806
807
[1370]808int HierarchyManager::GetObjectSpaceSubdivisionDepth() const
809{
810        int maxDepth = 0;
811
812        if (mObjectSpaceSubdivisionType == KD_BASED_OBJ_SUBDIV)
813        {
814                maxDepth = mOspTree->mOspStats.maxDepth;
815        }
816        else if (mObjectSpaceSubdivisionType == BV_BASED_OBJ_SUBDIV)
817        {
818                maxDepth = mBvHierarchy->mBvhStats.maxDepth;
819        }
820
821        return maxDepth;
822}
823
824
[1640]825int HierarchyManager::GetObjectSpaceSubdivisionLeaves() const
826{
827        int maxLeaves= 0;
828
829        if (mObjectSpaceSubdivisionType == KD_BASED_OBJ_SUBDIV)
830        {
831                maxLeaves = mOspTree->mOspStats.Leaves();
832        }
833        else if (mObjectSpaceSubdivisionType == BV_BASED_OBJ_SUBDIV)
834        {
835                maxLeaves = mBvHierarchy->mBvhStats.Leaves();
836        }
837
838        return maxLeaves;
839}
840
841
[1663]842int HierarchyManager::GetObjectSpaceSubdivisionNodes() const
843{
844        int maxLeaves = 0;
845
846        if (mObjectSpaceSubdivisionType == KD_BASED_OBJ_SUBDIV)
847        {
848                maxLeaves = mOspTree->mOspStats.nodes;
849        }
850        else if (mObjectSpaceSubdivisionType == BV_BASED_OBJ_SUBDIV)
851        {
852                maxLeaves = mBvHierarchy->mBvhStats.nodes;
853        }
854
855        return maxLeaves;
856}
857
[1308]858bool HierarchyManager::StartObjectSpaceSubdivision() const
[1237]859{
[1370]860        // view space construction already started
861        if (ObjectSpaceSubdivisionConstructed())
862                return false;
863
864        // start immediately with object space subdivision?
865        if (mStartWithObjectSpace)
866                return true;
867
868        // is the queue empty again?
869        if (ViewSpaceSubdivisionConstructed() && mTQueue.Empty())
870                return true;
871
872        // has the depth for subdivision been reached?
873        return
874                ((mConstructionType == INTERLEAVED) &&
[1640]875                 (mMinStepsOfSameType <= mVspTree->mVspStats.nodes));
[1308]876}
877
878
[1329]879bool HierarchyManager::StartViewSpaceSubdivision() const
880{
[1370]881        // view space construction already started
882        if (ViewSpaceSubdivisionConstructed())
883                return false;
884
885        // start immediately with view space subdivision?
886        if (!mStartWithObjectSpace)
887                return true;
888
889        // is the queue empty again?
890        if (ObjectSpaceSubdivisionConstructed() && mTQueue.Empty())
891                return true;
892
893        // has the depth for subdivision been reached?
894        return
895                ((mConstructionType == INTERLEAVED) &&
[1640]896                 (mMinStepsOfSameType <= GetObjectSpaceSubdivisionLeaves()));
[1329]897}
898
899
[1449]900void HierarchyManager::RunConstruction(const bool repairQueue,
901                                                                           const VssRayContainer &sampleRays,
[1311]902                                                                           const ObjectContainer &objects,
[1640]903                                                                           AxisAlignedBox3 *forcedViewSpace)
[1308]904{
[1313]905        while (!FinishedConstruction())
906        {
[1625]907                SubdivisionCandidate *sc = NextSubdivisionCandidate(mTQueue);   
[1632]908       
[1415]909                ///////////////////
[1237]910                //-- subdivide leaf node
[1415]911
[1625]912                ApplySubdivisionCandidate(sc, mTQueue, repairQueue);
[1580]913                               
[1313]914                // we use objects for evaluating vsp tree construction until
[1415]915                // a certain depth once a certain depth existiert ...
[1313]916                if (StartObjectSpaceSubdivision())
917                {
[1323]918                        mObjectSpaceSubdivisionType = mSavedObjectSpaceSubdivisionType;
[1314]919
[1640]920                        cout << "\nstarting object space subdivision after "
[1723]921                                 << mVspTree->mVspStats.nodes << " (" << mMinStepsOfSameType << ") steps, mem="
922                                 << mHierarchyStats.mMemory / float(1024 * 1024) << " MB" << endl;
[1329]923
[1779]924                        PrepareObjectSpaceSubdivision(mTQueue, sampleRays, objects);
[1640]925                       
[1313]926                        cout << "reseting queue ... ";
[1640]927                        ResetQueue(mTQueue, mRecomputeSplitPlaneOnRepair);
[1313]928                        cout << "finished" << endl;
929                }
930
[1329]931                if (StartViewSpaceSubdivision())
932                {
933                        mViewSpaceSubdivisionType = mSavedViewSpaceSubdivisionType;
934
[1723]935                        cout << "\nstarting view space subdivision at "
[1640]936                                 << GetObjectSpaceSubdivisionLeaves() << " ("
[1723]937                                 << mMinStepsOfSameType << ") , mem="
938                                 << mHierarchyStats.mMemory / float(1024 * 1024) << " MB" << endl;
[1329]939
[1779]940                        PrepareViewSpaceSubdivision(mTQueue, sampleRays, objects);
[1329]941
942                        cout << "reseting queue ... ";
[1640]943                        ResetQueue(mTQueue, mRecomputeSplitPlaneOnRepair);
[1329]944                        cout << "finished" << endl;
945                }
946
[1624]947                DEL_PTR(sc);
[1237]948        }
949}
950
951
[1449]952void HierarchyManager::RunConstruction(const bool repairQueue)
953{
[1580]954        // main loop
[1449]955        while (!FinishedConstruction())
956        {
[1625]957                SubdivisionCandidate *sc = NextSubdivisionCandidate(mTQueue);   
[1649]958               
[1624]959                ////////
960                //-- subdivide leaf node of either type
[1642]961        ApplySubdivisionCandidate(sc, mTQueue, repairQueue);
962               
[1624]963                DEL_PTR(sc);
[1449]964        }
965}
966
967
[1625]968int HierarchyManager::RunConstruction(SplitQueue &splitQueue,
[1633]969                                                                          SubdivisionCandidateContainer &dirtyCandidates,
[1667]970                                                                          SubdivisionCandidate *oldCandidate,
[1676]971                                                                          const int minSteps,
972                                                                          const int maxSteps)
[1624]973{
[1676]974        if (minSteps >= maxSteps)
975                cout << "error!! " << minSteps << " equal or larger maxSteps" << endl;
976
[1625]977        int steps = 0;
[1633]978        SubdivisionCandidate::NewMail();
[1625]979
[1624]980        // main loop
[1634]981        while (!splitQueue.Empty())
[1624]982        {
[1701]983                const float priority = splitQueue.Top()->GetPriority();
[1686]984                const float threshold = oldCandidate ? oldCandidate->GetPriority() : 1e20f;
[1679]985
[1701]986                // minimum slope reached
987                if ((steps >= maxSteps) || ((priority < threshold) && !(steps < minSteps)))
988                {
[1745]989                        cout << "\nbreaking on " << priority << " smaller than " << threshold << endl;
[1701]990                        break;
991                }
992               
[1624]993                ////////
994                //-- subdivide leaf node of either type
995
[1727]996                SubdivisionCandidate *sc = NextSubdivisionCandidate(splitQueue);
997                       
[1625]998                const bool repairQueue = false;
[1633]999                const bool success = ApplySubdivisionCandidate(sc, splitQueue, repairQueue);
[1625]1000
[1632]1001                if (success)
1002                {
[1697]1003                        sc->CollectDirtyCandidates(dirtyCandidates, true);
[1633]1004                        ++ steps;
[1632]1005                }
[1696]1006
1007                DEL_PTR(sc);
[1624]1008        }
[1625]1009
1010        return steps;
[1624]1011}
1012
1013
[1779]1014void HierarchyManager::ResetObjectSpaceSubdivision(SplitQueue &tQueue,
1015                                                                                                   const VssRayContainer &sampleRays,
1016                                                                                                   const ObjectContainer &objects)
[1580]1017{       
[1632]1018        SubdivisionCandidate *firstCandidate;
[1625]1019
[1580]1020        // object space partition constructed => reconstruct
[1557]1021        switch (mObjectSpaceSubdivisionType)
1022        {
1023        case BV_BASED_OBJ_SUBDIV:
[1580]1024                {
1025                        cout << "\nreseting bv hierarchy" << endl;
1026                        Debug << "old bv hierarchy:\n " << mBvHierarchy->mBvhStats << endl;
[1663]1027                               
1028                        // rather use this: remove previous nodes and add the two new ones
1029                        //mHierarchyStats.mNodes -= mBvHierarchy->mBvhStats.nodes + 1;
1030                        mHierarchyStats.mNodes = mVspTree->mVspStats.nodes;
[1649]1031                       
[1663]1032                        // create root
[1642]1033                        mBvHierarchy->Initialise(objects);
[1649]1034       
[1779]1035                        mBvHierarchy->Reset(tQueue, sampleRays, objects);
[1649]1036
[1625]1037                        mHierarchyStats.mTotalCost = mBvHierarchy->mTotalCost;
[1663]1038                       
[1662]1039                        //mHierarchyStats.mPvsEntries -= mBvHierarchy->mPvsEntries + 1;
[1649]1040                        mHierarchyStats.mPvsEntries = mBvHierarchy->CountViewCells(objects);
[1663]1041
[1667]1042                        mHierarchyStats.mMemory =
1043                                (float)mHierarchyStats.mPvsEntries * ObjectPvs::GetEntrySizeByte();
[1642]1044
[1662]1045                        mHierarchyStats.mRenderCostDecrease = 0;
1046
[1625]1047                        // evaluate stats before first subdivision
1048                        EvalSubdivisionStats();
[1649]1049                        cout << "finished bv hierarchy preparation" << endl;
[1580]1050                }
[1548]1051                break;
[1557]1052
[1548]1053        case KD_BASED_OBJ_SUBDIV:
1054                // TODO
1055        default:
1056                break;
1057        }
[1449]1058}
1059
1060
[1779]1061void HierarchyManager::ResetViewSpaceSubdivision(SplitQueue &tQueue,
1062                                                                                                 const VssRayContainer &sampleRays,
1063                                                                                                 const ObjectContainer &objects,
1064                                                                                                 AxisAlignedBox3 *forcedViewSpace)
[1557]1065{
[1649]1066        ViewCellsManager *vm = mVspTree->mViewCellsManager;
[1557]1067
[1624]1068        // HACK: rather not destroy vsp tree
[1580]1069        DEL_PTR(mVspTree);
1070        mVspTree = new VspTree();
[1649]1071
[1580]1072        mVspTree->mHierarchyManager = this;
[1649]1073        mVspTree->mViewCellsManager = vm;
[1580]1074
[1642]1075        mVspTree->Initialise(sampleRays, forcedViewSpace);
[1580]1076       
[1779]1077        //////////
[1662]1078        //-- reset stats
[1779]1079    mHierarchyStats.mNodes = GetObjectSpaceSubdivisionNodes();
1080                //-mVspTree->mVspStats.nodes + 1;
[1662]1081       
[1779]1082        PrepareViewSpaceSubdivision(mTQueue, sampleRays, objects);
[1662]1083       
1084        mHierarchyStats.mPvsEntries = mVspTree->mPvsEntries;
[1661]1085        mHierarchyStats.mRenderCostDecrease = 0;
1086
[1779]1087        mHierarchyStats.mMemory =
1088                (float)mHierarchyStats.mPvsEntries * ObjectPvs::GetEntrySizeByte();
[1662]1089
[1640]1090        // evaluate new stats before first subdivsiion
[1624]1091        EvalSubdivisionStats();
[1557]1092}
1093
1094
[1449]1095void HierarchyManager::ConstructMultiLevel(const VssRayContainer &sampleRays,                                                                                   
1096                                                                                   const ObjectContainer &objects,
1097                                                                                   AxisAlignedBox3 *forcedViewSpace)
1098{
1099        mHierarchyStats.Reset();
1100        mHierarchyStats.Start();
[1624]1101        mHierarchyStats.mNodes = 2;
[1449]1102       
[1624]1103        mHierarchyStats.mTotalCost = (float)objects.size();
1104        Debug << "setting total cost to " << mHierarchyStats.mTotalCost << endl;
[1449]1105
1106        const long startTime = GetTime();
1107        cout << "Constructing view space / object space tree ... \n";
1108       
[1640]1109        // initialise view / object space
1110        mVspTree->Initialise(sampleRays, forcedViewSpace);
1111        InitialiseObjectSpaceSubdivision(objects);
[1449]1112
[1580]1113        // use sah for evaluating osp tree construction
1114        // in the first iteration of the subdivision
[1642]1115
[1449]1116        mSavedViewSpaceSubdivisionType = mViewSpaceSubdivisionType;
1117        mViewSpaceSubdivisionType = NO_VIEWSPACE_SUBDIV;
[1640]1118
[1779]1119        PrepareObjectSpaceSubdivision(mTQueue, sampleRays, objects);
[1640]1120
[1649]1121        //////////////////////////
[1642]1122
1123
[1640]1124        const int limit = mNumMultiLevels;
1125        int i = 0;
1126
1127        // This method subdivides view space / object space
1128        // in order to converge to some optimal cost for this partition
1129        // start with object space partiton
1130        // then optimizate view space partition for the current osp
1131        // and vice versa until iteration depth is reached.
1132        while (1)
1133        {
1134                char subdivisionStatsLog[100];
1135                sprintf(subdivisionStatsLog, "tests/i3d/subdivision-%04d.log", i);
1136                mSubdivisionStats.open(subdivisionStatsLog);
1137
1138                // subdivide object space first
[1779]1139                ResetObjectSpaceSubdivision(mTQueue, sampleRays, objects);
[1640]1140
1141                // process object space candidates
1142                RunConstruction(false);
1143
1144                // object space subdivision constructed
1145                mObjectSpaceSubdivisionType = mSavedObjectSpaceSubdivisionType;
1146
1147                cout << "iteration " << i << " of " << limit << " finished" << endl;
1148                mSubdivisionStats.close();
1149
1150                if ((i ++) >= limit)
1151                        break;
1152
1153                sprintf(subdivisionStatsLog, "tests/i3d/subdivision-%04d.log", i);
1154                mSubdivisionStats.open(subdivisionStatsLog);
1155
[1649]1156
[1640]1157                /////////////////
1158                // subdivide view space with respect to the objects
1159
[1779]1160                ResetViewSpaceSubdivision(mTQueue, sampleRays, objects, forcedViewSpace);
1161               
[1640]1162                // view space subdivision constructed
1163                mViewSpaceSubdivisionType = mSavedViewSpaceSubdivisionType;
1164               
1165                // process view space candidates
1166                RunConstruction(false);
1167
1168                cout << "iteration " << i << " of " << limit << " finished" << endl;
1169                mSubdivisionStats.close();
1170
1171                if ((i ++) >= limit)
1172                        break;
1173        }
[1635]1174       
[1640]1175        cout << "\nfinished in " << TimeDiff(startTime, GetTime()) * 1e-3 << " secs" << endl;
[1580]1176
[1640]1177        mHierarchyStats.Stop();
1178        mVspTree->mVspStats.Stop();
1179        FinishObjectSpaceSubdivision(objects);
1180}
1181
1182
1183void HierarchyManager::OptimizeMultiLevel(const VssRayContainer &sampleRays,                                                                                     
1184                                                                                  const ObjectContainer &objects,
1185                                                                                  AxisAlignedBox3 *forcedViewSpace)
1186{
[1649]1187        const long startTime = GetTime();
1188        const int limit = mNumMultiLevels;
1189
1190        // open up new subdivision
1191        mSubdivisionStats.close();
1192
1193        int steps = 0;
1194
1195        int maxViewSpaceLeaves = mVspTree->mVspStats.Leaves();
1196        int maxObjectSpaceLeaves;
1197       
[1642]1198        // set the number of leaves 'evaluated' from the previous methods
[1649]1199        // we go for the same numbers, but we try to optimize both subdivisions
[1642]1200        switch (mObjectSpaceSubdivisionType)
1201        {
1202        case BV_BASED_OBJ_SUBDIV:
[1649]1203                maxObjectSpaceLeaves = mBvHierarchy->mBvhStats.Leaves();
[1642]1204                break;
1205        case KD_BASED_OBJ_SUBDIV:
[1649]1206                maxObjectSpaceLeaves = mOspTree->mOspStats.Leaves();
[1642]1207        default:
[1649]1208                maxObjectSpaceLeaves = 0;
[1642]1209                break;
1210        }
[1640]1211
[1624]1212        // This method subdivides view space / object space
1213        // in order to converge to some optimal cost for this partition
[1557]1214        // start with object space partiton
1215        // then optimizate view space partition for the current osp
1216        // and vice versa until iteration depth is reached.
1217        while (1)
[1548]1218        {
[1580]1219                char subdivisionStatsLog[100];
[1642]1220                sprintf(subdivisionStatsLog, "tests/i3d/subdivision-%04d.log", steps);
[1580]1221                mSubdivisionStats.open(subdivisionStatsLog);
1222
[1624]1223                // subdivide object space first
[1779]1224                ResetObjectSpaceSubdivision(mTQueue, sampleRays, objects);
[1649]1225       
1226                // set the number of leaves 'evaluated' from the previous methods
1227                // we go for the same numbers, but we try to optimize both subdivisions
1228                mBvHierarchy->mTermMaxLeaves = maxObjectSpaceLeaves;
[1624]1229
[1548]1230                // process object space candidates
1231                RunConstruction(false);
[1449]1232
[1642]1233                cout << "iteration " << steps << " of " << limit << " finished" << endl;
[1580]1234                mSubdivisionStats.close();
[1635]1235
[1642]1236                if ((++ steps) >= limit)
[1557]1237                        break;
1238
[1642]1239                sprintf(subdivisionStatsLog, "tests/i3d/subdivision-%04d.log", steps);
[1580]1240                mSubdivisionStats.open(subdivisionStatsLog);
1241
[1557]1242                /////////////////
[1580]1243                // subdivide view space with respect to the objects
1244
[1779]1245                ResetViewSpaceSubdivision(mTQueue, sampleRays, objects, forcedViewSpace);
[1649]1246
1247                mVspTree->mMaxViewCells = maxViewSpaceLeaves;
[1624]1248
[1548]1249                // process view space candidates
1250                RunConstruction(false);
[1624]1251
[1642]1252                cout << "iteration " << steps << " of " << limit << " finished" << endl;
[1580]1253                mSubdivisionStats.close();
1254
[1642]1255                if ((++ steps) >= limit)
[1557]1256                        break;
[1548]1257        }
1258       
[1449]1259        cout << "\nfinished in " << TimeDiff(startTime, GetTime()) * 1e-3 << " secs" << endl;
1260
1261        mHierarchyStats.Stop();
1262        mVspTree->mVspStats.Stop();
1263        FinishObjectSpaceSubdivision(objects);
1264}
1265
1266
[1640]1267
[1237]1268bool HierarchyManager::FinishedConstruction() const
1269{
1270        return mTQueue.Empty();
1271}
1272
1273
[1313]1274bool HierarchyManager::ObjectSpaceSubdivisionConstructed() const
[1311]1275{
[1642]1276        /*switch (mObjectSpaceSubdivisionType)
[1311]1277        {
1278        case KD_BASED_OBJ_SUBDIV:
1279                return mOspTree && mOspTree->GetRoot();
1280        case BV_BASED_OBJ_SUBDIV:
[1344]1281                return mBvHierarchy && mBvHierarchy->GetRoot();
[1311]1282        default:
[1580]1283                return false;
[1642]1284        }*/
1285        return mObjectSpaceSubdivisionType != NO_OBJ_SUBDIV;
[1311]1286}
1287
1288
[1329]1289bool HierarchyManager::ViewSpaceSubdivisionConstructed() const
1290{
[1635]1291        return mViewSpaceSubdivisionType != NO_VIEWSPACE_SUBDIV;
1292        //return mVspTree && mVspTree->GetRoot();
[1329]1293}
1294
1295
[1625]1296void HierarchyManager::CollectDirtyCandidates(const SubdivisionCandidateContainer &chosenCandidates,
1297                                                                                          SubdivisionCandidateContainer &dirtyList)
[1237]1298{
[1625]1299        SubdivisionCandidateContainer::const_iterator sit, sit_end = chosenCandidates.end();
[1633]1300        SubdivisionCandidate::NewMail();
[1625]1301
1302        for (sit = chosenCandidates.begin(); sit != sit_end; ++ sit)
1303        {
[1633]1304                (*sit)->CollectDirtyCandidates(dirtyList, true);
[1625]1305        }
1306}
1307
1308
[1736]1309int HierarchyManager::RepairQueue(const SubdivisionCandidateContainer &dirtyList,
1310                                                                  SplitQueue &splitQueue,
1311                                                                  const bool recomputeSplitPlaneOnRepair)
[1625]1312{
[1237]1313        // for each update of the view space partition:
1314        // the candidates from object space partition which
1315        // have been afected by the view space split (the kd split candidates
1316        // which saw the view cell which was split) must be reevaluated
1317        // (maybe not locally, just reinsert them into the queue)
1318        //
1319        // vice versa for the view cells
1320        // for each update of the object space partition
1321        // reevaluate split candidate for view cells which saw the split kd cell
1322        //
1323        // the priority queue update can be solved by implementing a binary heap
1324        // (explicit data structure, binary tree)
1325        // *) inserting and removal is efficient
1326        // *) search is not efficient => store queue position with each
1327        // split candidate
1328
[1736]1329        int repaired = 0;
1330
[1237]1331        // collect list of "dirty" candidates
[1580]1332        const long startTime = GetTime();
[1625]1333        if (0) cout << "repairing " << (int)dirtyList.size() << " candidates ... ";
[1624]1334
[1727]1335        const float prop = (float)mMaxRepairs / (float)dirtyList.size();
1336
[1684]1337        ///////////////////////////
[1305]1338        //-- reevaluate the dirty list
[1415]1339
[1313]1340        SubdivisionCandidateContainer::const_iterator sit, sit_end = dirtyList.end();
[1302]1341       
[1237]1342        for (sit = dirtyList.begin(); sit != sit_end; ++ sit)
1343        {
[1727]1344                // only repair a certain number of candidates
1345                if ((mMaxRepairs < (int)dirtyList.size()) && (Random(1.0f) >= prop))
1346                        continue;
1347
[1237]1348                SubdivisionCandidate* sc = *sit;
[1305]1349                const float rcd = sc->GetRenderCostDecrease();
[1302]1350               
[1667]1351                // erase from queue
1352                splitQueue.Erase(sc);
1353                // reevaluate candidate
1354                sc->EvalCandidate(recomputeSplitPlaneOnRepair);
1355                 // reinsert
1356                splitQueue.Push(sc);
[1736]1357               
1358                ++ repaired;
[1667]1359                cout << ".";
1360
[1715]1361#ifdef GTP_DEBUG
[1305]1362                Debug << "candidate " << sc << " reevaluated\n"
1363                          << "render cost decrease diff " <<  rcd - sc->GetRenderCostDecrease()
[1580]1364                          << " old: " << rcd << " new " << sc->GetRenderCostDecrease() << endl;
1365#endif 
[1237]1366        }
[1313]1367
[1625]1368        const long endTime = GetTime();
1369        const Real timeDiff = TimeDiff(startTime, endTime);
1370
[1624]1371        mHierarchyStats.mRepairTime += timeDiff;
[1313]1372
[1736]1373        return repaired;
[1237]1374}
1375
[1259]1376
[1640]1377void HierarchyManager::ResetQueue(SplitQueue &splitQueue, const bool recomputeSplitPlane)
[1313]1378{
1379        SubdivisionCandidateContainer mCandidateBuffer;
1380
1381        // remove from queue
[1640]1382        while (!splitQueue.Empty())
[1313]1383        {
[1640]1384                SubdivisionCandidate *candidate = NextSubdivisionCandidate(splitQueue);
[1744]1385               
1386                // reevaluate local split plane and priority
[1667]1387                candidate->EvalCandidate(recomputeSplitPlane);
[1313]1388                cout << ".";
1389                mCandidateBuffer.push_back(candidate);
1390        }
1391
1392        // put back into queue
1393        SubdivisionCandidateContainer::const_iterator sit, sit_end = mCandidateBuffer.end();
1394    for (sit = mCandidateBuffer.begin(); sit != sit_end; ++ sit)
[1580]1395        {
[1640]1396                splitQueue.Push(*sit);
[1313]1397        }
1398}
1399
1400
[1279]1401void HierarchyManager::ExportObjectSpaceHierarchy(OUT_STREAM &stream)
1402{
1403        // the type of the view cells hierarchy
[1308]1404        switch (mObjectSpaceSubdivisionType)
[1279]1405        {
1406        case KD_BASED_OBJ_SUBDIV:
1407                stream << "<ObjectSpaceHierarchy type=\"osp\">" << endl;
1408                mOspTree->Export(stream);
1409                stream << endl << "</ObjectSpaceHierarchy>" << endl;
[1305]1410                break;         
[1279]1411        case BV_BASED_OBJ_SUBDIV:
1412                stream << "<ObjectSpaceHierarchy type=\"bvh\">" << endl;
1413                mBvHierarchy->Export(stream);
1414                stream << endl << "</ObjectSpaceHierarchy>" << endl;
1415                break;
1416        }
1417}
1418
1419
[1421]1420void HierarchyManager::PrintHierarchyStatistics(ostream &stream) const
[1279]1421{
[1313]1422        stream << mHierarchyStats << endl;
1423        stream << "\nview space:" << endl << endl;
1424        stream << mVspTree->GetStatistics() << endl;
1425        stream << "\nobject space:" << endl << endl;
[1370]1426
[1308]1427        switch (mObjectSpaceSubdivisionType)
[1279]1428        {
1429        case KD_BASED_OBJ_SUBDIV:
1430                {
[1313]1431                        stream << mOspTree->GetStatistics() << endl;
[1279]1432                        break;
1433                }
1434        case BV_BASED_OBJ_SUBDIV:
1435                {
[1313]1436                        stream << mBvHierarchy->GetStatistics() << endl;
[1279]1437                        break;
1438                }
1439        default:
1440                break;
1441        }
1442}
1443
1444
[1287]1445void HierarchyManager::ExportObjectSpaceHierarchy(Exporter *exporter,
[1416]1446                                                                                                  const ObjectContainer &objects,
[1764]1447                                                                                                  AxisAlignedBox3 *bbox,
[1418]1448                                                                                                  const bool exportBounds) const
[1279]1449{
[1308]1450        switch (mObjectSpaceSubdivisionType)
[1286]1451        {
1452        case KD_BASED_OBJ_SUBDIV:
[1279]1453                {
[1287]1454                        ExportOspTree(exporter, objects);
[1286]1455                        break;
1456                }
1457        case BV_BASED_OBJ_SUBDIV:
1458                {
[1418]1459                        exporter->ExportBvHierarchy(*mBvHierarchy, 0, bbox, exportBounds);
[1286]1460                        break;
1461                }
1462        default:
1463                break;
1464        }
1465}
[1279]1466
[1286]1467
[1287]1468void HierarchyManager::ExportOspTree(Exporter *exporter,
1469                                                                         const ObjectContainer &objects) const
[1286]1470{
[1287]1471        if (0) exporter->ExportGeometry(objects);
[1279]1472                       
[1287]1473        exporter->SetWireframe();
1474        exporter->ExportOspTree(*mOspTree, 0);
[1286]1475}
1476
1477
[1743]1478Intersectable *HierarchyManager::GetIntersectable(Intersectable *obj,
1479                                                                                                  const Vector3 &point) const
1480{
[1758]1481
1482        if (!obj)
1483                return NULL;
1484
1485        switch (mObjectSpaceSubdivisionType)
[1743]1486        {
1487        case HierarchyManager::KD_BASED_OBJ_SUBDIV:
[1758]1488                {
1489                        KdLeaf *leaf = mOspTree->GetLeaf(point, NULL);
1490                        return mOspTree->GetOrCreateKdIntersectable(leaf);
1491                }
[1743]1492        case HierarchyManager::BV_BASED_OBJ_SUBDIV:
[1758]1493                {
1494                        BvhLeaf *leaf = mBvHierarchy->GetLeaf(obj);
1495                        return leaf;
1496                }
[1743]1497        default:
[1758]1498                return obj;
[1743]1499        }
1500}
1501
[1418]1502Intersectable *HierarchyManager::GetIntersectable(const VssRay &ray,
1503                                                                                                  const bool isTermination) const
1504{
[1758]1505        Intersectable *obj = NULL;
[1418]1506        Vector3 pt;
1507        KdNode *node;
1508
1509        ray.GetSampleData(isTermination, pt, &obj, &node);
[1758]1510
[1743]1511        if (!obj)
[1758]1512                return NULL;
[1418]1513
1514        switch (mObjectSpaceSubdivisionType)
1515        {
1516        case HierarchyManager::KD_BASED_OBJ_SUBDIV:
1517                {
[1758]1518                        KdLeaf *leaf = mOspTree->GetLeaf(pt, node);
[1418]1519                        return mOspTree->GetOrCreateKdIntersectable(leaf);
1520                }
1521        case HierarchyManager::BV_BASED_OBJ_SUBDIV:
1522                {
1523                        BvhLeaf *leaf = mBvHierarchy->GetLeaf(obj);
[1758]1524                        return leaf;
[1418]1525                }
1526        default:
[1758]1527                break;
[1418]1528        }
[1764]1529
[1743]1530        return obj;
[1418]1531}
1532
1533
[1313]1534void HierarchyStatistics::Print(ostream &app) const
1535{
1536        app << "=========== Hierarchy statistics ===============\n";
1537
1538        app << setprecision(4);
1539
1540        app << "#N_CTIME  ( Construction time [s] )\n" << Time() << " \n";
1541       
[1624]1542        app << "#N_RTIME  ( Repair time [s] )\n" << mRepairTime * 1e-3f << " \n";
[1313]1543
[1624]1544        app << "#N_NODES ( Number of nodes )\n" << mNodes << "\n";
[1313]1545
1546        app << "#N_INTERIORS ( Number of interior nodes )\n" << Interior() << "\n";
1547
1548        app << "#N_LEAVES ( Number of leaves )\n" << Leaves() << "\n";
1549
[1624]1550        app << "#N_PMAXDEPTH ( Maximal reached depth )\n" << mMaxDepth << endl;
[1449]1551
1552        app << "#N_GLOBALCOSTMISSES ( Global cost misses )\n" << mGlobalCostMisses << endl;
[1313]1553       
1554        app << "========== END OF Hierarchy statistics ==========\n";
1555}
1556
1557
[1418]1558static void RemoveRayRefs(const ObjectContainer &objects)
[1370]1559{
[1418]1560        ObjectContainer::const_iterator oit, oit_end = objects.end();
1561        for (oit = objects.begin(); oit != oit_end; ++ oit)
1562        {
[1696]1563                (*oit)->DelRayRefs();
[1418]1564        }
1565}
1566
1567
[1679]1568void HierarchyManager::FinishObjectSpaceSubdivision(const ObjectContainer &objects,
1569                                                                                                        const bool removeRayRefs) const
[1418]1570{
[1370]1571        switch (mObjectSpaceSubdivisionType)
1572        {
1573        case KD_BASED_OBJ_SUBDIV:
1574                {
1575                        mOspTree->mOspStats.Stop();
1576                        break;
1577                }
1578        case BV_BASED_OBJ_SUBDIV:
1579                {
1580                        mBvHierarchy->mBvhStats.Stop();
[1649]1581                        if (removeRayRefs)
1582                                RemoveRayRefs(objects);
[1370]1583                        break;
1584                }
1585        default:
1586                break;
1587        }
1588}
1589
[1614]1590
1591void HierarchyManager::ExportBoundingBoxes(OUT_STREAM &stream, const ObjectContainer &objects)
1592{
1593        stream << "<BoundingBoxes>" << endl;
1594           
1595        if (mObjectSpaceSubdivisionType == KD_BASED_OBJ_SUBDIV)
1596        {
1597                KdIntersectableMap::const_iterator kit, kit_end = mOspTree->mKdIntersectables.end();
1598
1599                int id = 0;
1600                for (kit = mOspTree->mKdIntersectables.begin(); kit != kit_end; ++ kit, ++ id)
1601                {
1602                        Intersectable *obj = (*kit).second;
1603                        const AxisAlignedBox3 box = obj->GetBox();
1604               
1605                        obj->SetId(id);
1606
1607                        stream << "<BoundingBox" << " id=\"" << id << "\""
1608                                   << " min=\"" << box.Min().x << " " << box.Min().y << " " << box.Min().z << "\""
1609                                   << " max=\"" << box.Max().x << " " << box.Max().y << " " << box.Max().z << "\" />" << endl;
1610                }
1611        }
1612        else
1613        {
1614                ObjectContainer::const_iterator oit, oit_end = objects.end();
1615
1616                for (oit = objects.begin(); oit != oit_end; ++ oit)
1617                {
1618                        const AxisAlignedBox3 box = (*oit)->GetBox();
1619               
1620                        stream << "<BoundingBox" << " id=\"" << (*oit)->GetId() << "\""
1621                                   << " min=\"" << box.Min().x << " " << box.Min().y << " " << box.Min().z << "\""
1622                                   << " max=\"" << box.Max().x << " " << box.Max().y << " " << box.Max().z << "\" />" << endl;
1623                }
1624        }
1625               
1626        stream << "</BoundingBoxes>" << endl;
1627}
1628
[1680]1629
[1706]1630class HierarchyNodeWrapper;
1631
1632
1633template <typename T> class myless
1634{
1635public:
1636        bool operator() (T v1, T v2) const
1637        {
1638                return (v1->GetMergeCost() < v2->GetMergeCost());
1639        }
1640};
1641
[1709]1642
[1706]1643typedef priority_queue<HierarchyNodeWrapper *, vector<HierarchyNodeWrapper *>,
1644                                           myless<vector<HierarchyNodeWrapper *>::value_type> > HierarchyNodeQueue;
1645
1646class HierarchyNodeWrapper
1647{
1648public:
[1707]1649        enum {VSP_NODE, BVH_NODE, VIEW_CELL};
[1706]1650
1651        virtual float GetMergeCost() const = 0;
1652        virtual int Type() const  = 0;
1653        virtual bool IsLeaf() const = 0;
1654
1655        virtual void PushChildren(HierarchyNodeQueue &tQueue) = 0;
1656};
1657
1658
1659class VspNodeWrapper: public HierarchyNodeWrapper
1660{
1661public:
1662        VspNodeWrapper(VspNode *node): mNode(node) {}
1663
1664        int Type() const { return VSP_NODE; }
1665
1666        float GetMergeCost() const { return (float) -mNode->mTimeStamp; };
1667
1668        bool IsLeaf() const { return mNode->IsLeaf(); }
1669
1670        void PushChildren(HierarchyNodeQueue &tQueue)
1671        {
1672                if (!mNode->IsLeaf())
1673                {
1674                        VspInterior *interior = dynamic_cast<VspInterior *>(mNode);
1675
1676                        tQueue.push(new VspNodeWrapper(interior->GetFront()));
1677                        tQueue.push(new VspNodeWrapper(interior->GetBack()));
1678                }
1679        }
1680
1681        VspNode *mNode;
1682};
1683
1684
1685class BvhNodeWrapper: public HierarchyNodeWrapper
1686{
1687public:
1688        BvhNodeWrapper(BvhNode *node): mNode(node) {}
[1686]1689       
[1706]1690        int Type()  const { return BVH_NODE; }
1691
[1763]1692        float GetMergeCost() const { return (float)-mNode->GetTimeStamp(); };
[1706]1693
1694        bool IsLeaf() const { return mNode->IsLeaf(); }
1695
1696        void PushChildren(HierarchyNodeQueue &tQueue)
1697        {
1698                if (!mNode->IsLeaf())
1699                {
1700                        BvhInterior *interior = dynamic_cast<BvhInterior *>(mNode);
1701
1702                        tQueue.push(new BvhNodeWrapper(interior->GetFront()));
1703                        tQueue.push(new BvhNodeWrapper(interior->GetBack()));
1704                }
1705        }
1706
1707        BvhNode *mNode;
1708};
1709
1710
[1707]1711class ViewCellWrapper: public HierarchyNodeWrapper
1712{
1713public:
1714
1715        ViewCellWrapper(ViewCell *vc): mViewCell(vc) {}
1716       
1717        int Type()  const { return VIEW_CELL; }
1718
1719        float GetMergeCost() const { return mViewCell->GetMergeCost(); };
1720
1721        bool IsLeaf() const { return mViewCell->IsLeaf(); }
1722
1723        void PushChildren(HierarchyNodeQueue &tQueue)
1724        {
1725                if (!mViewCell->IsLeaf())
1726                {
1727                        ViewCellInterior *interior = dynamic_cast<ViewCellInterior *>(mViewCell);
1728
1729                        ViewCellContainer::const_iterator it, it_end = interior->mChildren.end();
1730
1731                        for (it = interior->mChildren.begin(); it != it_end; ++ it)
1732                        {
1733                                tQueue.push(new ViewCellWrapper(*it));
1734                        }
1735                }
1736        }
1737
1738        ViewCell *mViewCell;
1739};
1740
1741
[1706]1742void HierarchyManager::CollectBestSet(const int maxSplits,
1743                                                                          const float maxMemoryCost,
[1707]1744                                                                          ViewCellContainer &viewCells,
[1706]1745                                                                          vector<BvhNode *> &bvhNodes)
1746{
1747        HierarchyNodeQueue tqueue;
[1707]1748        //tqueue.push(new VspNodeWrapper(mVspTree->GetRoot()));
1749        tqueue.push(new ViewCellWrapper(mVspTree->mViewCellsTree->GetRoot()));
[1706]1750        tqueue.push(new BvhNodeWrapper(mBvHierarchy->GetRoot()));
1751       
1752        float memCost = 0;
1753
1754        while (!tqueue.empty())
1755        {
1756                HierarchyNodeWrapper *nodeWrapper = tqueue.top();
1757                tqueue.pop();
[1713]1758                //cout << "priority: " << nodeWrapper->GetMergeCost() << endl;
[1706]1759                // save the view cells if it is a leaf or if enough view cells have already been traversed
1760                // because of the priority queue, this will be the optimal set of v
1761                if (nodeWrapper->IsLeaf() ||
[1707]1762                        ((viewCells.size() + bvhNodes.size() + tqueue.size() + 1) >= maxSplits) ||
[1706]1763                        (memCost > maxMemoryCost)
1764                        )
1765                {
[1713]1766                        if (nodeWrapper->Type() == HierarchyNodeWrapper::VIEW_CELL)
[1706]1767                        {
[1713]1768                                //cout << "1";
[1707]1769                                ViewCellWrapper *viewCellWrapper = dynamic_cast<ViewCellWrapper *>(nodeWrapper);
1770                                viewCells.push_back(viewCellWrapper->mViewCell);
[1706]1771                        }
1772                        else
1773                        {
[1713]1774                                //cout << "0";
[1706]1775                                BvhNodeWrapper *bvhNodeWrapper = dynamic_cast<BvhNodeWrapper *>(nodeWrapper);
1776                                bvhNodes.push_back(bvhNodeWrapper->mNode);
1777                        }
1778                }
1779                else
1780                {       
1781                        nodeWrapper->PushChildren(tqueue);
1782                }
1783
1784                delete nodeWrapper;
1785        }
1786}
1787
1788
[1842]1789void HierarchyManager::ComputePvs(const ObjectPvs &pvs,
1790                                                                  float &rc,
1791                                                                  int &pvsEntries)
[1745]1792{
1793        BvhNode::NewMail();
1794
1795        ObjectPvsIterator pit = pvs.GetIterator();
1796
1797        while (pit.HasMoreEntries())
1798        {
[1842]1799                const ObjectPvsEntry &entry = pit.Next();
[1745]1800
[1842]1801                if (entry.mObject->Type() != Intersectable::BVH_INTERSECTABLE)
1802                        cout << "error " << entry.mObject->Type() << endl;
1803
1804                BvhNode *intersect = dynamic_cast<BvhNode *>(entry.mObject);
[1758]1805                BvhNode *activeNode;
[1745]1806
[1758]1807                // hack for choosing which node to account for
1808                if (intersect->IsLeaf())
[1786]1809                {
[1842]1810                        activeNode =
1811                                dynamic_cast<BvhLeaf *>(intersect)->GetActiveNode();
[1786]1812                }
[1758]1813                else
[1786]1814                {
[1758]1815                        activeNode = intersect;
[1786]1816                }
[1745]1817
1818                if (!activeNode->Mailed())
1819                {
1820                        activeNode->Mail();
1821
[1786]1822#if STUPID_METHOD
1823
[1745]1824                        ObjectContainer objects;
[1786]1825            activeNode->CollectObjects(objects);
1826                        rc += mBvHierarchy->EvalAbsCost(objects);
1827#else
1828                        rc += mBvHierarchy->GetRenderCostIncrementially(activeNode);
1829#endif
[1745]1830                        ++ pvsEntries;
1831                }
1832        }
1833}
1834
1835
[1787]1836void HierarchyManager::GetPvsEfficiently(ViewCell *viewCell, ObjectPvs &pvs) const
1837{
1838        ////////////////
1839        //-- pvs is not stored with the interiors => reconstruct
1840       
1841        // add pvs from leaves
1842        stack<ViewCell *> tstack;
1843        tstack.push(viewCell);
1844
1845        Intersectable::NewMail();
1846
1847        while (!tstack.empty())
1848        {
1849                ViewCell *vc = tstack.top();
1850                tstack.pop();
1851       
1852                // add newly found pvs to merged pvs: break here even for interior
1853                if (!vc->GetPvs().Empty())
1854                {
1855                        ObjectPvsIterator pit = vc->GetPvs().GetIterator();
1856
1857                        while (pit.HasMoreEntries())
1858                        {               
1859                                const ObjectPvsEntry &entry = pit.Next();
1860
1861                                Intersectable *object = entry.mObject;
1862                                if (!object->Mailed())
1863                                {
1864                                        object->Mail();
1865                                        pvs.AddSampleDirty(object, 1.0f);
1866                                }
1867                        }
1868                }
1869                else if (!vc->IsLeaf()) // interior cells: go down to leaf level
1870                {
1871                        ViewCellInterior *interior = dynamic_cast<ViewCellInterior *>(vc);
1872                        ViewCellContainer::const_iterator it, it_end = interior->mChildren.end();
1873
1874                        for (it = interior->mChildren.begin(); it != it_end; ++ it)
1875                        {
1876                                tstack.push(*it);
1877                        }               
1878                }
1879        }
1880}
1881
1882
[1750]1883// TODO matt: implement this function for different storing methods
[1787]1884void HierarchyManager::GetPvsIncrementially(ViewCell *vc, ObjectPvs &pvs) const
[1750]1885{
1886        ////////////////
1887        //-- pvs is not stored with the interiors => reconstruct
1888       
1889        // add pvs from leaves
1890        stack<ViewCell *> tstack;
1891        tstack.push(vc);
1892
1893        while (!tstack.empty())
1894        {
1895                vc = tstack.top();
1896                tstack.pop();
1897       
1898                // add newly found pvs to merged pvs: break here even for interior
1899                if (!vc->GetPvs().Empty())
1900                {
[1787]1901                        //if (vc->IsLeaf()) cout << " l " << pvs.GetSize();
1902                        //else cout << " i " << pvs.GetSize();
[1750]1903                        pvs.MergeInPlace(vc->GetPvs());
1904                }
1905                else if (!vc->IsLeaf()) // interior cells: go down to leaf level
1906                {
[1787]1907                        //cout <<" t";
[1750]1908                        ViewCellInterior *interior = dynamic_cast<ViewCellInterior *>(vc);
1909                        ViewCellContainer::const_iterator it, it_end = interior->mChildren.end();
1910
1911                        for (it = interior->mChildren.begin(); it != it_end; ++ it)
1912                        {
1913                                tstack.push(*it);
1914                        }               
1915                }
[1786]1916                else cout <<"k";
[1750]1917        }
1918}
1919
[1786]1920
1921// TODO matt: implement this function for different storing methods
1922void HierarchyManager::PullUpPvsIncrementally(ViewCell *viewCell) const
1923{
1924        ////////////////
1925        //-- pvs is not stored with the interiors => reconstruct
1926       
1927        // early exit: pvs is already pulled up to this view cell
1928        if (!viewCell->GetPvs().Empty())
1929                return;
1930
1931        // add pvs from leaves
1932        stack<ViewCell *> tstack;
1933        tstack.push(viewCell);
1934
1935        ViewCell *vc = viewCell;
1936
1937        while (!tstack.empty())
1938        {
1939                vc = tstack.top();
1940                tstack.pop();
1941       
1942                // add newly found pvs to merged pvs: break here even for interior
1943                if (!vc->GetPvs().Empty())
1944                {
[1787]1945                        viewCell->GetPvs().MergeInPlace(vc->GetPvs());
[1786]1946                }
1947                else if (!vc->IsLeaf()) // interior cells: go down to leaf level
1948                {
1949                        //cout <<" t";
1950                        ViewCellInterior *interior = dynamic_cast<ViewCellInterior *>(vc);
1951                        ViewCellContainer::const_iterator it, it_end = interior->mChildren.end();
1952
1953                        for (it = interior->mChildren.begin(); it != it_end; ++ it)
1954                        {
1955                                tstack.push(*it);
1956                        }               
1957                }
1958        }
1959}
1960
1961
1962
1963// TODO matt: implement this function for different storing methods
1964void HierarchyManager::GetPvsRecursive(ViewCell *vc, ObjectPvs &pvs) const
1965{
1966        ////////////////
1967        //-- pvs is not stored with the interiors => reconstruct
1968        if (vc->IsLeaf() || !vc->GetPvs().Empty())
1969        {
1970                pvs = vc->GetPvs();
1971        }
1972        else
1973        {
1974                ViewCellInterior *interior = dynamic_cast<ViewCellInterior *>(vc);
1975#if 0
1976                ViewCellContainer::const_iterator it, it_end = interior->mChildren.end();
1977                const int childPvsSize = (int)interior->mChildren.size();
1978                vector<ObjectPvs> childPvs;
1979                childPvs.resize((int)interior->mChildren.size());
1980
1981                int i = 0;
1982                for (it = interior->mChildren.begin(); it != it_end; ++ it, ++ i)
1983                {
1984                        GetPvsRecursive(*it, childPvs[i]);
1985                        pvs.MergeInPlace(childPvs[i]);
1986                }
1987#else
1988
1989                ObjectPvs leftPvs, rightPvs;
1990
1991                GetPvsRecursive(interior->mChildren[0], leftPvs);
1992                GetPvsRecursive(interior->mChildren[1], rightPvs);
1993
1994                ObjectPvs::Merge(pvs, leftPvs, rightPvs);
1995#endif
1996        }
1997}
1998
1999
[1713]2000int HierarchyManager::ExtractStatistics(const int maxSplits,
2001                                                                                const float maxMemoryCost,
2002                                                                                float &renderCost,
2003                                                                                float &memory,
[1718]2004                                                                                int &pvsEntries,
2005                                                                                int &viewSpaceSplits,
[1745]2006                                                                                int &objectSpaceSplits,
2007                                                                                const bool useFilter)
[1706]2008{
[1707]2009        ViewCellContainer viewCells;
2010        vector<BvhNode *> bvhNodes;
2011
[1709]2012        // collect best set of view cells for this #splits
[1707]2013    CollectBestSet(maxSplits, maxMemoryCost, viewCells, bvhNodes);
2014        vector<BvhNode *>::const_iterator bit, bit_end = bvhNodes.end();
2015       
[1713]2016        // set new nodes to be active
[1707]2017        for (bit = bvhNodes.begin(); bit != bit_end; ++ bit)
2018        {
2019                mBvHierarchy->SetActive(*bit);
2020        }
2021
2022        ViewCellContainer::const_iterator vit, vit_end = viewCells.end();
2023
[1713]2024        pvsEntries = 0;
2025        renderCost = 0.0f;
[1707]2026
[1750]2027        ViewCell::NewMail();
2028
[1787]2029        //cout << "\nviewcells: " << viewCells.size() << endl;
[1707]2030        for (vit = viewCells.begin(); vit != vit_end; ++ vit)
2031        {
2032                ViewCell *vc = *vit;
[1750]2033                float rc = 0;
[1786]2034       
2035#if STUPID_METHOD       
2036                ObjectPvs pvs;
[1787]2037                GetPvsIncrementially(vc, pvs);
[1750]2038                vc->SetPvs(pvs);
[1787]2039               
[1786]2040#else
[1787]2041       
[1786]2042                ObjectPvs pvs;
[1789]2043                               
[1787]2044                if (vc->GetPvs().Empty())
2045                {
[1789]2046                        // warning: uses mailing, pvs not sorted!!
[1787]2047                        GetPvsEfficiently(vc, pvs);
[1789]2048                        //GetPvsRecursive(vc, pvs);
[1787]2049                        vc->SetPvs(pvs);
2050                        //cout << "q";
2051                }
2052                //else cout << "t";
[1786]2053#endif
[1750]2054
2055                vc->Mail();
2056
[1745]2057                if (useFilter)
[1707]2058                {
[1786]2059                        const long startT = GetTime();
[1745]2060                        ObjectPvs filteredPvs;
[1750]2061                        mVspTree->mViewCellsManager->ApplyFilter2(vc, false, 1.0f, filteredPvs);
[1786]2062                        const long endT = GetTime();
2063
[1787]2064                        //cout << "filter computed in " << TimeDiff(startT, endT) * 1e-3f << " secs" << endl;
[1745]2065                        ComputePvs(filteredPvs, rc, pvsEntries);
[1707]2066                }
[1745]2067                else
2068                {
[1786]2069                        ComputePvs(vc->GetPvs(), rc, pvsEntries);
[1749]2070                }
[1713]2071
2072                rc *= vc->GetVolume();
2073                renderCost += rc;
[1707]2074        }
2075
[1713]2076        renderCost /= mVspTree->mViewCellsManager->GetViewSpaceBox().GetVolume();
2077        memory = pvsEntries * ObjectPvs::GetEntrySize();
[1718]2078
2079        viewSpaceSplits = (int)viewCells.size();
2080        objectSpaceSplits = (int)bvhNodes.size();
[1713]2081        //cout << "viewCells: " << (int)viewCells.size() << " nodes: " << (int)bvhNodes.size() << " rc: " << renderCost << " entries: " << pvsEntries << endl;
2082
[1750]2083        // delete old "base" view cells if they are not leaves
2084        ViewCellContainer::const_iterator oit, oit_end = mOldViewCells.end();
2085
2086        for (oit = mOldViewCells.begin(); oit != oit_end; ++ oit)
2087        {
2088                if (!(*oit)->Mailed() && !(*oit)->IsLeaf())
2089                {
2090                        (*oit)->GetPvs().Clear();
2091                }
2092        }
2093
[1786]2094        // store current level
[1750]2095        mOldViewCells = viewCells;
2096
[1764]2097        return (int)(viewCells.size() + bvhNodes.size());
[1709]2098}
[1706]2099
2100
[1709]2101void HierarchyManager::ExportStats(ofstream &stats,
2102                                                                   SplitQueue &tQueue,
2103                                                                   const ObjectContainer &objects)
2104{
[1723]2105        HierarchySubdivisionStats subStats;
2106        subStats.Reset();
[1706]2107
2108        /////////////
[1723]2109        //-- initial situation
[1706]2110
[1723]2111        subStats.mNumSplits = 0;
2112        subStats.mTotalRenderCost = (float)objects.size();
2113        subStats.mEntriesInPvs = 1;
2114        subStats.mMemoryCost = (float)ObjectPvs::GetEntrySize();
2115        subStats.mFullMemory = subStats.mMemoryCost;
2116        subStats.mViewSpaceSplits = 0;
2117        subStats.mObjectSpaceSplits = 0;
[1744]2118        subStats.mRenderCostDecrease = 0;
[1723]2119        subStats.Print(stats);
[1706]2120
[1723]2121        cout << "exporting vsposp stats ... " << endl;
2122
[1706]2123        //-- go through tree in the order of render cost decrease
2124        //-- which is the same order as the view cells were merged
2125        //-- or the reverse order of subdivision for subdivision-only
2126        //-- view cell hierarchies.
[1789]2127
[1709]2128        while (!tQueue.Empty())
[1706]2129        {
[1709]2130                SubdivisionCandidate *nextCandidate = NextSubdivisionCandidate(tQueue);
2131                bool isLeaf;
2132                int timeStamp;
2133                float rcDecr;
2134                int entriesIncr;
[1706]2135
[1789]2136                if (nextCandidate->Type() == SubdivisionCandidate::VIEW_SPACE)
[1709]2137                {
2138                        timeStamp = (int)-nextCandidate->GetPriority();
[1706]2139
[1709]2140                        VspNode *newNode = mVspTree->SubdivideAndCopy(tQueue, nextCandidate);
2141                        VspNode *oldNode = (VspNode *)nextCandidate->mEvaluationHack;
[1789]2142
[1709]2143                        isLeaf = newNode->IsLeaf();
[1732]2144                        //subStats.mRenderCostDecrease = oldNode->mRenderCostDecr;
2145                        //entriesIncr = oldNode->mPvsEntriesIncr;
[1709]2146                }
2147                else
2148                {
2149                        timeStamp = (int)-nextCandidate->GetPriority();
[1789]2150
[1709]2151                        BvhNode *newNode = mBvHierarchy->SubdivideAndCopy(tQueue, nextCandidate);
2152                        BvhNode *oldNode = (BvhNode *)nextCandidate->mEvaluationHack;
[1789]2153
[1709]2154                        isLeaf = newNode->IsLeaf();
[1732]2155                        //subStats.mRenderCostDecrease = oldNode->mRenderCostDecr;
2156                        //entriesIncr = oldNode->mPvsEntriesIncr;
[1709]2157                }               
[1789]2158
[1709]2159                if (!isLeaf)
2160                {
[1723]2161                        subStats.mTotalRenderCost -= subStats.mRenderCostDecrease;
[1738]2162                        //subStats.mEntriesInPvs += entriesIncr;
[1723]2163
[1718]2164                        if (nextCandidate->Type() == SubdivisionCandidate::VIEW_SPACE)
2165                        {
[1723]2166                                ++ subStats.mViewSpaceSplits;
[1741]2167                                cout << "v";
2168                                //cout << "vsp t: " << timeStamp << " rc: " << rcDecr << " pvs: " << entriesIncr << endl;
[1718]2169                        }
[1709]2170                        else
[1718]2171                        {
[1723]2172                                ++ subStats.mObjectSpaceSplits;
[1741]2173                                cout << "o";
2174                                //"osp t: " << timeStamp << " rc: " << rcDecr << " pvs: " << entriesIncr << endl;
[1718]2175                        }
[1706]2176
[1723]2177                        ++ subStats.mNumSplits;
[1706]2178
[1723]2179                        if ((subStats.mNumSplits % 500) == 499)
2180                                cout << subStats.mNumSplits << " steps taken" << endl;
[1718]2181
[1723]2182                        subStats.mMemoryCost = (float)subStats.mEntriesInPvs * (float)ObjectPvs::GetEntrySize();
2183                        subStats.mFullMemory = subStats.mMemoryCost;
2184
2185                        subStats.Print(stats);
[1789]2186
[1709]2187                }
[1706]2188
[1709]2189                DEL_PTR(nextCandidate);
2190        }
[1706]2191
[1709]2192        stats.close();
2193}
[1706]2194
2195
[1741]2196void HierarchyManager::EvaluateSubdivision(const VssRayContainer &sampleRays,
[1709]2197                                                                                   const ObjectContainer &objects,
2198                                                                                   const string &filename)
2199{
[1779]2200#if 0
[1709]2201        VspTree *oldVspTree = mVspTree;
2202        ViewCellsManager *vm = mVspTree->mViewCellsManager;
2203        BvHierarchy *oldHierarchy = mBvHierarchy;
[1706]2204
[1709]2205        mBvHierarchy = new BvHierarchy();
2206        mBvHierarchy->mHierarchyManager = this;
2207        mBvHierarchy->mViewCellsManager = vm;
[1706]2208
[1709]2209        mVspTree = new VspTree();
2210        mVspTree->mHierarchyManager = this;
2211        mVspTree->mViewCellsManager = vm;
[1706]2212
[1709]2213        // create first nodes
2214        mVspTree->Initialise(sampleRays, &oldVspTree->mBoundingBox);
2215        InitialiseObjectSpaceSubdivision(objects);
[1706]2216
[1709]2217        const long startTime = GetTime();
2218        cout << "Constructing evaluation hierarchies ... \n";
2219       
2220        ofstream stats;
2221        stats.open(filename.c_str());
2222        SplitQueue tQueue;
[1706]2223
[1709]2224        BvhNode *oldBvhRoot = oldHierarchy->GetRoot();
2225        VspNode *oldVspRoot = oldVspTree->GetRoot();
[1706]2226
[1709]2227        RayInfoContainer *viewSpaceRays = new RayInfoContainer();
2228       
2229        SubdivisionCandidate *firstVsp = mVspTree->PrepareConstruction(sampleRays, *viewSpaceRays);
[1784]2230        tQueue.Push(firstVsp);
2231
[1779]2232        mBvHierarchy->PrepareConstruction(tQueue, sampleRays, objects);
[1709]2233
2234    firstVsp->mEvaluationHack = oldVspRoot;
2235        firstBvh->mEvaluationHack = oldBvhRoot;
2236
2237        firstVsp->SetPriority((float)-oldVspRoot->mTimeStamp);
[1763]2238        firstBvh->SetPriority((float)-oldBvhRoot->GetTimeStamp());
[1779]2239       
[1709]2240        ExportStats(stats, tQueue, objects);
2241
2242        cout << "\nfinished in " << TimeDiff(startTime, GetTime()) * 1e-3 << " secs" << endl;
2243        RemoveRayRefs(objects);
2244
2245        // view cells needed only for evaluation
2246        ViewCellContainer viewCells;
2247        mVspTree->CollectViewCells(viewCells, false);
2248       
2249        // helper trees can be destroyed
2250        DEL_PTR(mVspTree);
2251        DEL_PTR(mBvHierarchy);
2252
2253        CLEAR_CONTAINER(viewCells);
2254
2255        // reset hierarchies
2256        mVspTree = oldVspTree;
2257        mBvHierarchy = oldHierarchy;
2258
2259        // reinstall old bv refs
2260        vector<BvhLeaf *> leaves;
2261        mBvHierarchy->CollectLeaves(mBvHierarchy->GetRoot(), leaves);
2262        vector<BvhLeaf *>::const_iterator bit, bit_end = leaves.end();
2263
2264        for (bit = leaves.begin(); bit != bit_end; ++ bit)
2265        {
2266                mBvHierarchy->AssociateObjectsWithLeaf(*bit);
[1706]2267        }
[1779]2268#endif
[1709]2269}
[1706]2270
[1709]2271
2272void HierarchyManager::EvaluateSubdivision2(ofstream &splitsStats,
[1745]2273                                                                                        const int splitsStepSize,
2274                                                                                        const bool useFilter)
[1709]2275{
[1750]2276        vector<HierarchySubdivisionStats> subStatsContainer;
[1709]2277
[1786]2278        int splits = (1 + (mHierarchyStats.Leaves() - 1) / splitsStepSize) * splitsStepSize;
[1750]2279        cout << "splits: " << splits << endl;
2280
[1709]2281        while (1)
2282        {
[1750]2283                HierarchySubdivisionStats subStats;
[1723]2284                subStats.mNumSplits = ExtractStatistics(splits,
[1718]2285                                                                                                99999.0,
[1723]2286                                                                                                subStats.mTotalRenderCost,
2287                                                                                                subStats.mMemoryCost,
2288                                                                                                subStats.mEntriesInPvs,
2289                                                                                                subStats.mViewSpaceSplits,
[1745]2290                                                                                                subStats.mObjectSpaceSplits,
2291                                                                                                useFilter);
[1723]2292
[1713]2293               
[1723]2294                const float objectSpaceHierarchyMem = float(
[1760]2295                                                                                          subStats.mObjectSpaceSplits * mBvHierarchy->mMemoryConst//sizeof(ObjectContainer)
[1732]2296                                                                                          //+ (subStats.mObjectSpaceSplits - 1) * sizeof(BvhInterior)
2297                                                                                          //+sizeof(BvHierarchy)
[1723]2298                                                                                          ) / float(1024 * 1024);
[1718]2299
[1723]2300                       
2301                const float viewSpaceHierarchyMem = float(
[1760]2302                                                                                        subStats.mViewSpaceSplits * mVspTree->mMemoryConst//sizeof(ObjectPvs)
[1732]2303                                                                                        //+ (subStats.mViewSpaceSplits - 1) * sizeof(VspInterior)
2304                                                                                        + sizeof(ObjectPvs)
2305                                                                                        //+ sizeof(VspTree)
[1723]2306                                                                                        )  / float(1024 * 1024);
2307
2308                subStats.mFullMemory = subStats.mMemoryCost + objectSpaceHierarchyMem + viewSpaceHierarchyMem;
2309               
[1750]2310                subStatsContainer.push_back(subStats);
2311               
2312                if (splits == 0)
[1786]2313                {
[1709]2314                        break;
[1786]2315                }
[1750]2316                splits -= splitsStepSize;
2317
[1786]2318                cout << "splits: " << subStats.mNumSplits << " ";
[1709]2319        }
[1745]2320
[1750]2321        vector<HierarchySubdivisionStats>::const_reverse_iterator hit, hit_end = subStatsContainer.rend();
2322
2323        for (hit = subStatsContainer.rbegin(); hit != hit_end; ++ hit)
2324        {
2325                (*hit).Print(splitsStats);
2326        }
2327
2328        // delete old "base" view cells: only pvss in the leaves are allowed
2329        ViewCellContainer::const_iterator oit, oit_end = mOldViewCells.end();
2330        for (oit = mOldViewCells.begin(); oit != oit_end; ++ oit)
2331        {
2332                if (!(*oit)->IsLeaf())
2333                {
2334                        (*oit)->GetPvs().Clear();
2335                }
2336        }
2337
2338        mOldViewCells.clear();
2339
[1763]2340        // reset active nodes
2341        vector<BvhLeaf *> bvhLeaves;
2342
2343        mBvHierarchy->CollectLeaves(mBvHierarchy->GetRoot(), bvhLeaves);
2344
2345        vector<BvhLeaf *>::const_iterator bit, bit_end = bvhLeaves.end();
2346
2347        for (bit = bvhLeaves.begin(); bit != bit_end; ++ bit)
2348        {
2349                (*bit)->SetActiveNode(*bit);
2350        }
2351
[1732]2352        cout << endl;
[1706]2353}
[1709]2354
[1718]2355
2356void HierarchyManager::CollectObjects(const AxisAlignedBox3 &box, ObjectContainer &objects)
2357{
[1719]2358        mBvHierarchy->CollectObjects(box, objects);
[1715]2359}
[1718]2360
[1843]2361
[1845]2362int HierarchyManager::CompressObjectSpace()
[1844]2363{
2364        //mBvHierarchy->Compress();
[1845]2365        return mVspTree->CompressObjects();
2366}
[1844]2367
[1845]2368
[1843]2369void HierarchyManager::CreateUniqueObjectIds()
2370{
2371        mBvHierarchy->CreateUniqueObjectIds();
[1718]2372}
[1843]2373
2374
2375}
Note: See TracBrowser for help on using the repository browser.