diff --git a/src/BVH/BVH.cxx b/src/BVH/BVH.cxx index efc40ffc3d..cf81cfce4a 100644 --- a/src/BVH/BVH.cxx +++ b/src/BVH/BVH.cxx @@ -15,6 +15,7 @@ #include #include +#include #include #include #include @@ -85,6 +86,12 @@ template class BVH_BinnedBuilder; template class BVH_BinnedBuilder; template class BVH_BinnedBuilder; +template class BVH_LinearBuilder; +template class BVH_LinearBuilder; + +template class BVH_LinearBuilder; +template class BVH_LinearBuilder; + template class BVH_SweepPlaneBuilder; template class BVH_SweepPlaneBuilder; template class BVH_SweepPlaneBuilder; diff --git a/src/BVH/BVH_BinnedBuilder.hxx b/src/BVH/BVH_BinnedBuilder.hxx index 1592285a88..06a32286f7 100644 --- a/src/BVH/BVH_BinnedBuilder.hxx +++ b/src/BVH/BVH_BinnedBuilder.hxx @@ -16,7 +16,7 @@ #ifndef _BVH_BinnedBuilder_Header #define _BVH_BinnedBuilder_Header -#include +#include #include @@ -34,7 +34,7 @@ struct BVH_Bin //! Performs building of BVH tree using binned SAH algorithm. //! Number of Bins controls tree's quality (greater - better) in cost of construction time. template -class BVH_BinnedBuilder : public BVH_Builder +class BVH_BinnedBuilder : public BVH_QueueBuilder { public: diff --git a/src/BVH/BVH_BinnedBuilder.lxx b/src/BVH/BVH_BinnedBuilder.lxx index c002b5680a..3ea88cd5c0 100644 --- a/src/BVH/BVH_BinnedBuilder.lxx +++ b/src/BVH/BVH_BinnedBuilder.lxx @@ -20,8 +20,8 @@ template BVH_BinnedBuilder::BVH_BinnedBuilder (const Standard_Integer theLeafNodeSize, const Standard_Integer theMaxTreeDepth) -: BVH_Builder (theLeafNodeSize, - theMaxTreeDepth) +: BVH_QueueBuilder (theLeafNodeSize, + theMaxTreeDepth) { // } @@ -36,17 +36,6 @@ BVH_BinnedBuilder::~BVH_BinnedBuilder() // } -namespace BVH -{ - template - static inline Standard_Integer IntFloor (const T theValue) - { - const Standard_Integer aRes = static_cast (theValue); - - return aRes - static_cast (aRes > theValue); - } -} - // ======================================================================= // function : GetSubVolumes // purpose : @@ -291,7 +280,7 @@ void BVH_BinnedBuilder::BuildNode (BVH_Set* theSet, if (!isLeaf) { - BVH_Builder::myTasksQueue.Append (aChildIndex); + BVH_QueueBuilder::myTasksQueue.Append (aChildIndex); } BVH_Builder::UpdateDepth (theBVH, theBVH->Level (aChildIndex)); diff --git a/src/BVH/BVH_Builder.hxx b/src/BVH/BVH_Builder.hxx index 09093bf390..15213437c1 100644 --- a/src/BVH/BVH_Builder.hxx +++ b/src/BVH/BVH_Builder.hxx @@ -19,8 +19,6 @@ #include #include -#include - namespace { //! Minimum node size to split. @@ -38,21 +36,14 @@ public: const Standard_Integer theMaxTreeDepth); //! Releases resources of BVH builder. - virtual ~BVH_Builder() = 0; + virtual ~BVH_Builder(); public: //! Builds BVH using specified algorithm. - void Build (BVH_Set* theSet, - BVH_Tree* theBVH, - const BVH_Box& theBox); - -protected: - - //! Builds BVH node for specified task info. - virtual void BuildNode (BVH_Set* theSet, - BVH_Tree* theBVH, - const Standard_Integer theTask); + virtual void Build (BVH_Set* theSet, + BVH_Tree* theBVH, + const BVH_Box& theBox) = 0; //! Updates depth of constructed BVH tree. void UpdateDepth (BVH_Tree* theBVH, @@ -68,7 +59,6 @@ protected: Standard_Integer myMaxTreeDepth; //!< Maximum depth of constructed BVH Standard_Integer myLeafNodeSize; //!< Maximum number of primitives per leaf - NCollection_Vector myTasksQueue; //!< Queue to manage BVH node building tasks }; diff --git a/src/BVH/BVH_Builder.lxx b/src/BVH/BVH_Builder.lxx index 9dd4b7aa3b..ebb173a550 100644 --- a/src/BVH/BVH_Builder.lxx +++ b/src/BVH/BVH_Builder.lxx @@ -15,7 +15,7 @@ // ======================================================================= // function : BVH_Builder -// purpose : +// purpose : Creates abstract BVH builder // ======================================================================= template BVH_Builder::BVH_Builder (const Standard_Integer theLeafNodeSize, @@ -28,57 +28,10 @@ BVH_Builder::BVH_Builder (const Standard_Integer theLeafNodeSize, // ======================================================================= // function : ~BVH_Builder -// purpose : +// purpose : Releases resources of BVH builder // ======================================================================= template BVH_Builder::~BVH_Builder() { // } - -// ======================================================================= -// function : Build -// purpose : -// ======================================================================= -template -void BVH_Builder::Build (BVH_Set* theSet, - BVH_Tree* theBVH, - const BVH_Box& theBox) -{ - if (theBVH == NULL) - { - return; - } - - theBVH->Clear(); - if (theSet->Size() == 0) - { - return; - } - - const Standard_Integer aRoot = theBVH->AddLeafNode (theBox, 0, theSet->Size() - 1); - if (theSet->Size() == 1) - { - return; - } - - myTasksQueue.Append (aRoot); - for (Standard_Integer aTask = 0; aTask < myTasksQueue.Size(); ++aTask) - { - BuildNode (theSet, theBVH, myTasksQueue.Value (aTask)); - } - - myTasksQueue.Clear(); -} - -// ======================================================================= -// function : BuildNode -// purpose : -// ======================================================================= -template -void BVH_Builder::BuildNode (BVH_Set* , - BVH_Tree* , - const Standard_Integer ) -{ - // need to disable compile warnings -} diff --git a/src/BVH/BVH_LinearBuilder.hxx b/src/BVH/BVH_LinearBuilder.hxx new file mode 100644 index 0000000000..95e2c1fa97 --- /dev/null +++ b/src/BVH/BVH_LinearBuilder.hxx @@ -0,0 +1,67 @@ +// Created on: 2014-09-11 +// Created by: Danila ULYANOV +// Copyright (c) 2013-2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BVH_LinearBuilder_Header +#define _BVH_LinearBuilder_Header + +#include + +typedef std::pair BVH_EncodedLink; + +//! Performs fast BVH construction using LBVH building approach. +//! Algorithm uses spatial Morton codes to reduce the BVH construction +//! problem to a sorting problem (radix sort -- O(N) complexity). This +//! Linear Bounding Volume Hierarchy (LBVH) builder produces BVH trees +//! of lower quality compared to SAH-based BVH builders but it is over +//! an order of magnitude faster (up to 3M triangles per second). +//! +//! For more details see: +//! C. Lauterbach, M. Garland, S. Sengupta, D. Luebke, and D. Manocha. +//! Fast BVH construction on GPUs. Eurographics, 2009. +template +class BVH_LinearBuilder : public BVH_Builder +{ +public: + + typedef typename BVH::VectorType::Type BVH_VecNt; + +public: + + //! Creates binned LBVH builder. + BVH_LinearBuilder (const Standard_Integer theLeafNodeSize = 5, + const Standard_Integer theMaxTreeDepth = 32); + + //! Releases resources of LBVH builder. + virtual ~BVH_LinearBuilder(); + + //! Builds BVH. + void Build (BVH_Set* theSet, + BVH_Tree* theBVH, + const BVH_Box& theBox); + +protected: + + //! Emits hierarchy from sorted Morton codes. + Standard_Integer EmitHierachy (BVH_Tree* theBVH, + const Standard_Integer theBit, + const Standard_Integer theShift, + std::vector::iterator theStart, + std::vector::iterator theFinal); + +}; + +#include + +#endif // _BVH_LinearBuilder_Header diff --git a/src/BVH/BVH_LinearBuilder.lxx b/src/BVH/BVH_LinearBuilder.lxx new file mode 100644 index 0000000000..26a7d0d0e4 --- /dev/null +++ b/src/BVH/BVH_LinearBuilder.lxx @@ -0,0 +1,503 @@ +// Created on: 2014-09-11 +// Created by: Danila ULYANOV +// Copyright (c) 2013-2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#include + +#include + +#ifdef HAVE_TBB + // On Windows, function TryEnterCriticalSection has appeared in Windows NT + // and is surrounded by #ifdef in MS VC++ 7.1 headers. + // Thus to use it we need to define appropriate macro saying that we will + // run on Windows NT 4.0 at least + #if defined(_WIN32) && !defined(_WIN32_WINNT) + #define _WIN32_WINNT 0x0501 + #endif + + #include +#endif + +// ======================================================================= +// function : BVH_LinearBuilder +// purpose : +// ======================================================================= +template +BVH_LinearBuilder::BVH_LinearBuilder (const Standard_Integer theLeafNodeSize, + const Standard_Integer theMaxTreeDepth) +: BVH_Builder (theLeafNodeSize, + theMaxTreeDepth) +{ + // +} + +// ======================================================================= +// function : ~BVH_LinearBuilder +// purpose : +// ======================================================================= +template +BVH_LinearBuilder::~BVH_LinearBuilder() +{ + // +} + +namespace BVH +{ + // Radix sort STL predicate for 32-bit integer. + class BitPredicate + { + Standard_Integer myBit; + + public: + + //! Creates new radix sort predicate. + BitPredicate (const Standard_Integer theBit) : myBit (theBit) + { + // + } + + //! Returns predicate value. + bool operator() (const BVH_EncodedLink theLink) const + { + const Standard_Integer aMask = 1 << myBit; + + return !(theLink.first & aMask); // 0-bit to the left side + } + }; + + //! STL compare tool used in binary search algorithm. + class BitComparator + { + Standard_Integer myBit; + + public: + + //! Creates new STL comparator. + BitComparator (const Standard_Integer theBit) : myBit (theBit) + { + // + } + + //! Checks left value for the given bit. + bool operator() (BVH_EncodedLink theLink1, BVH_EncodedLink /*theLink2*/) + { + return !(theLink1.first & (1 << myBit)); + } + }; + + //! Tool object for sorting link array using radix sort algorithm. + struct RadixSorter + { + typedef std::vector::iterator LinkIterator; + + // Performs MSD (most significant digit) radix sort. + static void Perform (LinkIterator theStart, LinkIterator theFinal, Standard_Integer theBit = 29) + { + while (theStart != theFinal && theBit >= 0) + { + LinkIterator anOffset = std::partition (theStart, theFinal, BitPredicate (theBit--)); + + Perform (theStart, anOffset, theBit); + + theStart = anOffset; + } + } + }; + + //! Calculates bounding boxes (AABBs) for the given BVH tree. + template + Standard_Integer UpdateBounds (BVH_Set* theSet, BVH_Tree* theTree, const Standard_Integer theNode = 0) + { + const BVH_Vec4i aData = theTree->NodeInfoBuffer()[theNode]; + + if (aData.x() == 0) + { + const Standard_Integer aLftChild = theTree->NodeInfoBuffer()[theNode].y(); + const Standard_Integer aRghChild = theTree->NodeInfoBuffer()[theNode].z(); + + const Standard_Integer aLftDepth = UpdateBounds (theSet, theTree, aLftChild); + const Standard_Integer aRghDepth = UpdateBounds (theSet, theTree, aRghChild); + + typename BVH_Box::BVH_VecNt aLftMinPoint = theTree->MinPointBuffer()[aLftChild]; + typename BVH_Box::BVH_VecNt aLftMaxPoint = theTree->MaxPointBuffer()[aLftChild]; + typename BVH_Box::BVH_VecNt aRghMinPoint = theTree->MinPointBuffer()[aRghChild]; + typename BVH_Box::BVH_VecNt aRghMaxPoint = theTree->MaxPointBuffer()[aRghChild]; + + BVH::BoxMinMax::CwiseMin (aLftMinPoint, aRghMinPoint); + BVH::BoxMinMax::CwiseMax (aLftMaxPoint, aRghMaxPoint); + + theTree->MinPointBuffer()[theNode] = aLftMinPoint; + theTree->MaxPointBuffer()[theNode] = aLftMaxPoint; + + return Max (aLftDepth, aRghDepth) + 1; + } + else + { + typename BVH_Box::BVH_VecNt& aMinPoint = theTree->MinPointBuffer()[theNode]; + typename BVH_Box::BVH_VecNt& aMaxPoint = theTree->MaxPointBuffer()[theNode]; + + for (Standard_Integer aPrimIdx = aData.y(); aPrimIdx <= aData.z(); ++aPrimIdx) + { + const typename BVH_Box aBox = theSet->Box (aPrimIdx); + + if (aPrimIdx == aData.y()) + { + aMinPoint = aBox.CornerMin(); + aMaxPoint = aBox.CornerMax(); + } + else + { + BVH::BoxMinMax::CwiseMin (aMinPoint, aBox.CornerMin()); + BVH::BoxMinMax::CwiseMax (aMaxPoint, aBox.CornerMax()); + } + } + } + + return 0; + } +} + +// ======================================================================= +// function : EmitHierachy +// purpose : Emits hierarchy from sorted Morton codes +// ======================================================================= +template +Standard_Integer BVH_LinearBuilder::EmitHierachy (BVH_Tree* theBVH, + const Standard_Integer theBit, + const Standard_Integer theShift, + std::vector::iterator theStart, + std::vector::iterator theFinal) +{ + if (theFinal - theStart > myLeafNodeSize && theBit >= 0) + { + std::vector::iterator aPosition = std::lower_bound ( + theStart, theFinal, BVH_EncodedLink(), BVH::BitComparator (theBit)); + + if (aPosition == theStart || aPosition == theFinal) + { + return EmitHierachy (theBVH, theBit - 1, theShift, theStart, theFinal); + } + + // Build inner node + const Standard_Integer aNode = theBVH->AddInnerNode (0, 0); + + const Standard_Integer aRghNode = theShift + static_cast (aPosition - theStart); + + const Standard_Integer aLftChild = EmitHierachy (theBVH, theBit - 1, theShift, theStart, aPosition); + const Standard_Integer aRghChild = EmitHierachy (theBVH, theBit - 1, aRghNode, aPosition, theFinal); + + theBVH->NodeInfoBuffer()[aNode].y() = aLftChild; + theBVH->NodeInfoBuffer()[aNode].z() = aRghChild; + + return aNode; + } + else + { + // Build leaf node + return theBVH->AddLeafNode (theShift, theShift + static_cast (theFinal - theStart) - 1); + } +} + +#ifdef HAVE_TBB + +namespace BVH +{ + //! TBB task for parallel radix sort. + class RadixSortTask : public tbb::task + { + typedef std::vector::iterator LinkIterator; + + private: + + //! Start range element. + LinkIterator myStart; + + //! Final range element. + LinkIterator myFinal; + + //! Bit position for range partition. + Standard_Integer myDigit; + + public: + + //! Creates new TBB radix sort task. + RadixSortTask (LinkIterator theStart, LinkIterator theFinal, Standard_Integer theDigit) + : myStart (theStart), + myFinal (theFinal), + myDigit (theDigit) + { + // + } + + //! Executes the task. + tbb::task* execute() + { + if (myDigit < 28) + { + BVH::RadixSorter::Perform (myStart, myFinal, myDigit); + } + else + { + LinkIterator anOffset = std::partition (myStart, myFinal, BitPredicate (myDigit)); + + tbb::task_list aList; + + aList.push_back (*new ( allocate_child() ) + RadixSortTask (myStart, anOffset, myDigit - 1)); + + aList.push_back (*new ( allocate_child() ) + RadixSortTask (anOffset, myFinal, myDigit - 1)); + + set_ref_count (3); // count + 1 + spawn_and_wait_for_all (aList); + } + + return NULL; + } + }; + + //! TBB task for parallel bounds updating. + template + class UpdateBoundTask: public tbb::task + { + //! Set of geometric objects. + BVH_Set* mySet; + + //! BVH tree built over the set. + BVH_Tree* myBVH; + + //! BVH node to update bounding box. + Standard_Integer myNode; + + //! Level of the processed BVH node. + Standard_Integer myLevel; + + //! Height of the processed BVH node. + Standard_Integer* myHeight; + + public: + + //! Creates new TBB parallel bound update task. + UpdateBoundTask (BVH_Set* theSet, + BVH_Tree* theBVH, + Standard_Integer theNode, + Standard_Integer theLevel, + Standard_Integer* theHeight) + : mySet (theSet), + myBVH (theBVH), + myNode (theNode), + myLevel (theLevel), + myHeight (theHeight) + { + // + } + + //! Executes the task. + tbb::task* execute() + { + if (myBVH->IsOuter (myNode) || myLevel > 2) + { + *myHeight = BVH::UpdateBounds (mySet, myBVH, myNode); + } + else + { + Standard_Integer aLftHeight = 0; + Standard_Integer aRghHeight = 0; + + tbb::task_list aList; + + const Standard_Integer aLftChild = myBVH->NodeInfoBuffer()[myNode].y(); + const Standard_Integer aRghChild = myBVH->NodeInfoBuffer()[myNode].z(); + + Standard_Integer aCount = 1; + + if (!myBVH->IsOuter (aLftChild)) + { + ++aCount; + aList.push_back (*new ( allocate_child() ) + UpdateBoundTask (mySet, myBVH, aLftChild, myLevel + 1, &aLftHeight)); + } + else + { + aLftHeight = BVH::UpdateBounds (mySet, myBVH, aLftChild); + } + + if (!myBVH->IsOuter (aRghChild)) + { + ++aCount; + aList.push_back (*new( allocate_child() ) + UpdateBoundTask (mySet, myBVH, aRghChild, myLevel + 1, &aRghHeight)); + } + else + { + aRghHeight = BVH::UpdateBounds (mySet, myBVH, aRghChild); + } + + if (aCount > 1) + { + set_ref_count (aCount); + spawn_and_wait_for_all (aList); + } + + typename BVH_Box::BVH_VecNt aLftMinPoint = myBVH->MinPointBuffer()[aLftChild]; + typename BVH_Box::BVH_VecNt aLftMaxPoint = myBVH->MaxPointBuffer()[aLftChild]; + typename BVH_Box::BVH_VecNt aRghMinPoint = myBVH->MinPointBuffer()[aRghChild]; + typename BVH_Box::BVH_VecNt aRghMaxPoint = myBVH->MaxPointBuffer()[aRghChild]; + + BVH::BoxMinMax::CwiseMin (aLftMinPoint, aRghMinPoint); + BVH::BoxMinMax::CwiseMax (aLftMaxPoint, aRghMaxPoint); + + myBVH->MinPointBuffer()[myNode] = aLftMinPoint; + myBVH->MaxPointBuffer()[myNode] = aLftMaxPoint; + + *myHeight = Max (aLftHeight, aRghHeight) + 1; + } + + return NULL; + } + }; +} + +#endif + +// ======================================================================= +// function : Build +// purpose : +// ======================================================================= +template +void BVH_LinearBuilder::Build (BVH_Set* theSet, + BVH_Tree* theBVH, + const BVH_Box& theBox) +{ + Standard_STATIC_ASSERT (N == 3 || N == 4); + + if (theBVH == NULL || theSet->Size() == 0) + { + return; + } + + theBVH->Clear(); + + const Standard_Integer aDimensionX = 1024; + const Standard_Integer aDimensionY = 1024; + const Standard_Integer aDimensionZ = 1024; + + const BVH_VecNt aSceneMin = theBox.CornerMin(); + const BVH_VecNt aSceneMax = theBox.CornerMax(); + + const BVH_VecNt aSceneSize = aSceneMax - aSceneMin; + + const T aReverseSizeX = static_cast (aDimensionX) / (aSceneMax.x() - aSceneMin.x()); + const T aReverseSizeY = static_cast (aDimensionY) / (aSceneMax.y() - aSceneMin.y()); + const T aReverseSizeZ = static_cast (aDimensionZ) / (aSceneMax.z() - aSceneMin.z()); + + std::vector anEncodedLinks (theSet->Size(), BVH_EncodedLink()); + + // Step 1 -- Assign Morton code to each primitive + for (Standard_Integer aPrimIdx = 0; aPrimIdx < theSet->Size(); ++aPrimIdx) + { + const BVH_VecNt aCenter = theSet->Box (aPrimIdx).Center(); + + Standard_Integer aVoxelX = BVH::IntFloor ((aCenter.x() - aSceneMin.x()) * aReverseSizeX); + Standard_Integer aVoxelY = BVH::IntFloor ((aCenter.y() - aSceneMin.y()) * aReverseSizeY); + Standard_Integer aVoxelZ = BVH::IntFloor ((aCenter.z() - aSceneMin.z()) * aReverseSizeZ); + + aVoxelX = Max (0, Min (aVoxelX, aDimensionX - 1)); + aVoxelY = Max (0, Min (aVoxelY, aDimensionY - 1)); + aVoxelZ = Max (0, Min (aVoxelZ, aDimensionZ - 1)); + + aVoxelX = (aVoxelX | (aVoxelX << 16)) & 0x030000FF; + aVoxelX = (aVoxelX | (aVoxelX << 8)) & 0x0300F00F; + aVoxelX = (aVoxelX | (aVoxelX << 4)) & 0x030C30C3; + aVoxelX = (aVoxelX | (aVoxelX << 2)) & 0x09249249; + + aVoxelY = (aVoxelY | (aVoxelY << 16)) & 0x030000FF; + aVoxelY = (aVoxelY | (aVoxelY << 8)) & 0x0300F00F; + aVoxelY = (aVoxelY | (aVoxelY << 4)) & 0x030C30C3; + aVoxelY = (aVoxelY | (aVoxelY << 2)) & 0x09249249; + + aVoxelZ = (aVoxelZ | (aVoxelZ << 16)) & 0x030000FF; + aVoxelZ = (aVoxelZ | (aVoxelZ << 8)) & 0x0300F00F; + aVoxelZ = (aVoxelZ | (aVoxelZ << 4)) & 0x030C30C3; + aVoxelZ = (aVoxelZ | (aVoxelZ << 2)) & 0x09249249; + + anEncodedLinks[aPrimIdx] = BVH_EncodedLink ( + aVoxelX | (aVoxelY << 1) | (aVoxelZ << 2), aPrimIdx); + } + + // Step 2 -- Sort primitives by their Morton codes using radix sort +#ifdef HAVE_TBB + + BVH::RadixSortTask& aSortTask = *new ( tbb::task::allocate_root() ) + BVH::RadixSortTask (anEncodedLinks.begin(), anEncodedLinks.end(), 29); + + tbb::task::spawn_root_and_wait (aSortTask); + +#else + + BVH::RadixSorter::Perform (anEncodedLinks.begin(), anEncodedLinks.end()); + +#endif + + // Step 3 -- Emitting BVH hierarchy from sorted Morton codes + EmitHierachy (theBVH, 29, 0, anEncodedLinks.begin(), anEncodedLinks.end()); + + Standard_Integer* aLinkMap = new Standard_Integer[theSet->Size()]; + + for (Standard_Integer aLinkIdx = 0; aLinkIdx < theSet->Size(); ++aLinkIdx) + { + aLinkMap[anEncodedLinks[aLinkIdx].second] = aLinkIdx; + } + + // Step 4 -- Rearranging primitive list according to Morton codes (in place) + Standard_Integer aPrimIdx = 0; + + while (aPrimIdx < theSet->Size()) + { + const Standard_Integer aSortIdx = aLinkMap[aPrimIdx]; + + if (aPrimIdx != aSortIdx) + { + theSet->Swap (aPrimIdx, aSortIdx); + + std::swap (aLinkMap[aPrimIdx], + aLinkMap[aSortIdx]); + } + else + { + ++aPrimIdx; + } + } + + // Step 5 -- Compute bounding boxes of BVH nodes + theBVH->MinPointBuffer().resize (theBVH->NodeInfoBuffer().size()); + theBVH->MaxPointBuffer().resize (theBVH->NodeInfoBuffer().size()); + + Standard_Integer aDepth = 0; + +#ifdef HAVE_TBB + + BVH::UpdateBoundTask& aRootTask = *new ( tbb::task::allocate_root() ) + BVH::UpdateBoundTask (theSet, theBVH, 0, 0, &aDepth); + + tbb::task::spawn_root_and_wait (aRootTask); + +#else + + aDepth = BVH::UpdateBounds (theSet, theBVH, 0); + +#endif + + BVH_Builder::UpdateDepth (theBVH, aDepth); +} diff --git a/src/BVH/BVH_QueueBuilder.hxx b/src/BVH/BVH_QueueBuilder.hxx new file mode 100644 index 0000000000..4c0d86b4f3 --- /dev/null +++ b/src/BVH/BVH_QueueBuilder.hxx @@ -0,0 +1,58 @@ +// Created on: 2014-09-15 +// Created by: Denis BOGOLEPOV +// Copyright (c) 2013-2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +#ifndef _BVH_QueueBuilder_Header +#define _BVH_QueueBuilder_Header + +#include + +#include + +//! Abstract BVH builder based on the concept of work queue. +template +class BVH_QueueBuilder : public BVH_Builder +{ +public: + + //! Creates new BVH queue based builder. + BVH_QueueBuilder (const Standard_Integer theLeafNodeSize, + const Standard_Integer theMaxTreeDepth); + + //! Releases resources of BVH queue based builder. + virtual ~BVH_QueueBuilder() = 0; + +public: + + //! Builds BVH using specific algorithm. + virtual void Build (BVH_Set* theSet, + BVH_Tree* theBVH, + const BVH_Box& theBox); + +protected: + + //! Builds BVH node for the given task info. + virtual void BuildNode (BVH_Set* theSet, + BVH_Tree* theBVH, + const Standard_Integer theTask); + +protected: + + NCollection_Vector myTasksQueue; //!< Queue to manage BVH node building tasks + +}; + +#include + +#endif // _BVH_QueueBuilder_Header diff --git a/src/BVH/BVH_QueueBuilder.lxx b/src/BVH/BVH_QueueBuilder.lxx new file mode 100644 index 0000000000..70d8bf13d5 --- /dev/null +++ b/src/BVH/BVH_QueueBuilder.lxx @@ -0,0 +1,84 @@ +// Created on: 2014-09-15 +// Created by: Denis BOGOLEPOV +// Copyright (c) 2013-2014 OPEN CASCADE SAS +// +// This file is part of Open CASCADE Technology software library. +// +// This library is free software; you can redistribute it and/or modify it under +// the terms of the GNU Lesser General Public License version 2.1 as published +// by the Free Software Foundation, with special exception defined in the file +// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT +// distribution for complete text of the license and disclaimer of any warranty. +// +// Alternatively, this file may be used under the terms of Open CASCADE +// commercial license or contractual agreement. + +// ======================================================================= +// function : BVH_QueueBuilder +// purpose : Creates new BVH queue based builder +// ======================================================================= +template +BVH_QueueBuilder::BVH_QueueBuilder (const Standard_Integer theLeafNodeSize, + const Standard_Integer theMaxTreeDepth) +: BVH_Builder (theLeafNodeSize, + theMaxTreeDepth) +{ + // +} + +// ======================================================================= +// function : ~BVH_QueueBuilder +// purpose : Releases resources of BVH queue based builder +// ======================================================================= +template +BVH_QueueBuilder::~BVH_QueueBuilder() +{ + // +} + +// ======================================================================= +// function : Build +// purpose : Builds BVH using specific algorithm +// ======================================================================= +template +void BVH_QueueBuilder::Build (BVH_Set* theSet, + BVH_Tree* theBVH, + const BVH_Box& theBox) +{ + if (theBVH == NULL) + { + return; + } + + theBVH->Clear(); + if (theSet->Size() == 0) + { + return; + } + + const Standard_Integer aRoot = theBVH->AddLeafNode (theBox, 0, theSet->Size() - 1); + if (theSet->Size() == 1) + { + return; + } + + myTasksQueue.Append (aRoot); + for (Standard_Integer aTask = 0; aTask < myTasksQueue.Size(); ++aTask) + { + BuildNode (theSet, theBVH, myTasksQueue.Value (aTask)); + } + + myTasksQueue.Clear(); +} + +// ======================================================================= +// function : BuildNode +// purpose : Builds BVH node for the given task info +// ======================================================================= +template +void BVH_QueueBuilder::BuildNode (BVH_Set* /* theSet */, + BVH_Tree* /* theBVH */, + const Standard_Integer /* theTask */) +{ + // needed to disable compile warnings +} diff --git a/src/BVH/BVH_SweepPlaneBuilder.hxx b/src/BVH/BVH_SweepPlaneBuilder.hxx index b2ff52f76b..8d05d5f206 100644 --- a/src/BVH/BVH_SweepPlaneBuilder.hxx +++ b/src/BVH/BVH_SweepPlaneBuilder.hxx @@ -16,11 +16,11 @@ #ifndef _BVH_SweepPlaneBuilder_Header #define _BVH_SweepPlaneBuilder_Header -#include +#include //! Performs building of BVH tree using sweep plane SAH algorithm. template -class BVH_SweepPlaneBuilder : public BVH_Builder +class BVH_SweepPlaneBuilder : public BVH_QueueBuilder { public: diff --git a/src/BVH/BVH_SweepPlaneBuilder.lxx b/src/BVH/BVH_SweepPlaneBuilder.lxx index cfa1255eb0..1708206f2f 100644 --- a/src/BVH/BVH_SweepPlaneBuilder.lxx +++ b/src/BVH/BVH_SweepPlaneBuilder.lxx @@ -22,8 +22,8 @@ template BVH_SweepPlaneBuilder::BVH_SweepPlaneBuilder (const Standard_Integer theLeafNodeSize, const Standard_Integer theMaxTreeDepth) -: BVH_Builder (theLeafNodeSize, - theMaxTreeDepth) +: BVH_QueueBuilder (theLeafNodeSize, + theMaxTreeDepth) { // } @@ -182,7 +182,7 @@ void BVH_SweepPlaneBuilder::BuildNode (BVH_Set* theSet, if (!isLeaf) { - BVH_Builder::myTasksQueue.Append (aChildIndex); + BVH_QueueBuilder::myTasksQueue.Append (aChildIndex); } BVH_Builder::UpdateDepth (theBVH, theBVH->Level (aChildIndex)); diff --git a/src/BVH/BVH_Tree.hxx b/src/BVH/BVH_Tree.hxx index c92cddd710..5aeecd080a 100644 --- a/src/BVH/BVH_Tree.hxx +++ b/src/BVH/BVH_Tree.hxx @@ -194,6 +194,14 @@ public: const Standard_Integer theLftChild, const Standard_Integer theRghChild); + //! Adds new leaf node to the BVH with UNINITIALIZED bounds. + Standard_Integer AddLeafNode (const Standard_Integer theBegElem, + const Standard_Integer theEndElem); + + //! Adds new inner node to the BVH with UNINITIALIZED bounds. + Standard_Integer AddInnerNode (const Standard_Integer theLftChild, + const Standard_Integer theRghChild); + //! Returns value of SAH (surface area heuristic). //! Allows to compare the quality of BVH trees constructed for //! the same sets of geometric objects with different methods. diff --git a/src/BVH/BVH_Tree.lxx b/src/BVH/BVH_Tree.lxx index 98d9b2f323..ac3d5a5dfe 100644 --- a/src/BVH/BVH_Tree.lxx +++ b/src/BVH/BVH_Tree.lxx @@ -28,6 +28,32 @@ void BVH_Tree::Clear() BVH::ArrayOp::Clear (myNodeInfoBuffer); } +// ======================================================================= +// function : AddLeafNode +// purpose : +// ======================================================================= +template +Standard_Integer BVH_Tree::AddLeafNode (const Standard_Integer theBegElem, + const Standard_Integer theEndElem) +{ + BVH::ArrayOp::Append (myNodeInfoBuffer, BVH_Vec4i (1, theBegElem, theEndElem, 0)); + + return static_cast (BVH::ArrayOp::Size (myNodeInfoBuffer) - 1); +} + +// ======================================================================= +// function : AddInnerNode +// purpose : +// ======================================================================= +template +Standard_Integer BVH_Tree::AddInnerNode (const Standard_Integer theLftChild, + const Standard_Integer theRghChild) +{ + BVH::ArrayOp::Append (myNodeInfoBuffer, BVH_Vec4i (0, theLftChild, theRghChild, 0)); + + return static_cast (BVH::ArrayOp::Size (myNodeInfoBuffer) - 1); +} + // ======================================================================= // function : AddLeafNode // purpose : diff --git a/src/BVH/BVH_Triangulation.lxx b/src/BVH/BVH_Triangulation.lxx index fb1b1801f6..5cc9d64f26 100644 --- a/src/BVH/BVH_Triangulation.lxx +++ b/src/BVH/BVH_Triangulation.lxx @@ -94,9 +94,8 @@ template void BVH_Triangulation::Swap (const Standard_Integer theIndex1, const Standard_Integer theIndex2) { - BVH_Vec4i anIndices1 = BVH::ArrayOp::Value (Elements, theIndex1); - BVH_Vec4i anIndices2 = BVH::ArrayOp::Value (Elements, theIndex2); + BVH_Vec4i& anIndices1 = BVH::ArrayOp::ChangeValue (Elements, theIndex1); + BVH_Vec4i& anIndices2 = BVH::ArrayOp::ChangeValue (Elements, theIndex2); - BVH::ArrayOp::ChangeValue (Elements, theIndex1) = anIndices2; - BVH::ArrayOp::ChangeValue (Elements, theIndex2) = anIndices1; + std::swap (anIndices1, anIndices2); } diff --git a/src/BVH/BVH_Types.hxx b/src/BVH/BVH_Types.hxx index 53e6a94dc9..d67f04fc0c 100644 --- a/src/BVH/BVH_Types.hxx +++ b/src/BVH/BVH_Types.hxx @@ -75,6 +75,14 @@ namespace BVH { typedef NCollection_Mat4 Type; }; + + template + static inline Standard_Integer IntFloor (const T theValue) + { + const Standard_Integer aRes = static_cast (theValue); + + return aRes - static_cast (aRes > theValue); + } } //! 2D vector of integers. diff --git a/src/BVH/BVH_Types.lxx b/src/BVH/BVH_Types.lxx index 9ae79c83f3..b4d5650e76 100644 --- a/src/BVH/BVH_Types.lxx +++ b/src/BVH/BVH_Types.lxx @@ -64,7 +64,7 @@ namespace BVH const Standard_Integer theIndex) { #ifdef _BVH_USE_STD_VECTOR_ - return theArray.at (theIndex); + return theArray[theIndex]; #else return theArray.Value (theIndex); #endif @@ -75,7 +75,7 @@ namespace BVH const Standard_Integer theIndex) { #ifdef _BVH_USE_STD_VECTOR_ - return theArray.at (theIndex); + return theArray[theIndex]; #else return theArray.ChangeValue (theIndex); #endif diff --git a/src/BVH/FILES b/src/BVH/FILES index 54d882bc58..c50ec7b8b5 100644 --- a/src/BVH/FILES +++ b/src/BVH/FILES @@ -7,6 +7,8 @@ BVH_Builder.hxx BVH_Builder.lxx BVH_Geometry.hxx BVH_Geometry.lxx +BVH_LinearBuilder.hxx +BVH_LinearBuilder.lxx BVH_Object.hxx BVH_Object.lxx BVH_ObjectSet.hxx @@ -30,3 +32,5 @@ BVH_Triangulation.hxx BVH_Triangulation.lxx BVH_Types.hxx BVH_Types.lxx +BVH_QueueBuilder.hxx +BVH_QueueBuilder.lxx