diff --git a/src/BVH/BVH_Builder.hxx b/src/BVH/BVH_Builder.hxx index a45d5996a2..715c1298f2 100644 --- a/src/BVH/BVH_Builder.hxx +++ b/src/BVH/BVH_Builder.hxx @@ -32,19 +32,32 @@ public: //! Returns the maximum number of sub-elements in the leaf. Standard_Integer LeafNodeSize() const { return myLeafNodeSize; } + //! Returns parallel flag. + inline Standard_Boolean IsParallel() const + { + return myIsParallel; + } + + //! Set parallel flag contolling possibility of parallel execution. + inline void SetParallel(const Standard_Boolean isParallel) + { + myIsParallel = isParallel; + } + protected: //! Creates new abstract BVH builder. BVH_BuilderTransient (const Standard_Integer theLeafNodeSize, const Standard_Integer theMaxTreeDepth) : myMaxTreeDepth (theMaxTreeDepth), - myLeafNodeSize (theLeafNodeSize) {} + myLeafNodeSize (theLeafNodeSize), + myIsParallel (Standard_False) {} protected: Standard_Integer myMaxTreeDepth; //!< Maximum depth of constructed BVH Standard_Integer myLeafNodeSize; //!< Maximum number of objects per leaf - + Standard_Boolean myIsParallel; //!< Parallel execution flag. }; //! Performs construction of BVH tree using bounding diff --git a/src/BVH/BVH_DistanceField.hxx b/src/BVH/BVH_DistanceField.hxx index e17f011081..1c1f067614 100644 --- a/src/BVH/BVH_DistanceField.hxx +++ b/src/BVH/BVH_DistanceField.hxx @@ -46,6 +46,18 @@ public: //! Builds 3D distance field from BVH geometry. Standard_Boolean Build (BVH_Geometry& theGeometry); + //! Returns parallel flag. + inline Standard_Boolean IsParallel() const + { + return myIsParallel; + } + + //! Set parallel flag contolling possibility of parallel execution. + inline void SetParallel(const Standard_Boolean isParallel) + { + myIsParallel = isParallel; + } + public: //! Returns packed voxel data. @@ -141,6 +153,7 @@ protected: //! Enables/disables signing of distance field. Standard_Boolean myComputeSign; + Standard_Boolean myIsParallel; }; #include diff --git a/src/BVH/BVH_DistanceField.lxx b/src/BVH/BVH_DistanceField.lxx index a83a2a3653..5d3e12ce9a 100644 --- a/src/BVH/BVH_DistanceField.lxx +++ b/src/BVH/BVH_DistanceField.lxx @@ -24,7 +24,8 @@ template BVH_DistanceField::BVH_DistanceField (const Standard_Integer theMaximumSize, const Standard_Boolean theComputeSign) : myMaximumSize (theMaximumSize), - myComputeSign (theComputeSign) + myComputeSign (theComputeSign), + myIsParallel (Standard_False) { Standard_STATIC_ASSERT (N == 3 || N == 4); @@ -495,7 +496,7 @@ Standard_Boolean BVH_DistanceField::Build (BVH_Geometry& theGeometry myVoxelSize.y() = (myCornerMax.y() - myCornerMin.y()) / myDimensionY; myVoxelSize.z() = (myCornerMax.z() - myCornerMin.z()) / myDimensionZ; - OSD_Parallel::For (0, myDimensionZ, BVH_ParallelDistanceFieldBuilder (this, &theGeometry)); + OSD_Parallel::For (0, myDimensionZ, BVH_ParallelDistanceFieldBuilder (this, &theGeometry), !IsParallel()); return Standard_True; } diff --git a/src/BVH/BVH_LinearBuilder.hxx b/src/BVH/BVH_LinearBuilder.hxx index 9f0a2233bb..48a4623e8f 100644 --- a/src/BVH/BVH_LinearBuilder.hxx +++ b/src/BVH/BVH_LinearBuilder.hxx @@ -227,6 +227,11 @@ namespace BVH { public: + UpdateBoundTask (const Standard_Boolean isParallel) + : myIsParallel (isParallel) + { + } + //! Executes the task. void operator()(const BoundData& theData) const { @@ -266,7 +271,7 @@ namespace BVH if (!aList.empty()) { - OSD_Parallel::ForEach (aList.begin (), aList.end (), UpdateBoundTask ()); + OSD_Parallel::ForEach (aList.begin (), aList.end (), UpdateBoundTask (myIsParallel), !myIsParallel); } typename BVH_Box::BVH_VecNt aLftMinPoint = theData.myBVH->MinPointBuffer()[aLftChild]; @@ -283,6 +288,10 @@ namespace BVH *theData.myHeight = Max (aLftHeight, aRghHeight) + 1; } } + + private: + + Standard_Boolean myIsParallel; }; } @@ -306,6 +315,7 @@ void BVH_LinearBuilder::Build (BVH_Set* theSet, // Step 0 -- Initialize parameter of virtual grid BVH_RadixSorter aRadixSorter (theBox); + aRadixSorter.SetParallel (this->IsParallel()); // Step 1 - Perform radix sorting of primitive set aRadixSorter.Perform (theSet); @@ -319,7 +329,7 @@ void BVH_LinearBuilder::Build (BVH_Set* theSet, Standard_Integer aHeight = 0; BVH::BoundData aBoundData = { theSet, theBVH, 0, 0, &aHeight }; - BVH::UpdateBoundTask aBoundTask; + BVH::UpdateBoundTask aBoundTask (this->IsParallel()); aBoundTask (aBoundData); BVH_Builder::updateDepth (theBVH, aHeight); diff --git a/src/BVH/BVH_RadixSorter.hxx b/src/BVH/BVH_RadixSorter.hxx index b0a8b06950..e3b205ff89 100644 --- a/src/BVH/BVH_RadixSorter.hxx +++ b/src/BVH/BVH_RadixSorter.hxx @@ -111,18 +111,33 @@ namespace BVH }; //! Functor class to run sorting in parallel. - struct Functor + class Functor { - //! Runs sorting function for the given range. - void operator()(const SortRange& theRange) const + public: + Functor(const SortRange (&aSplits)[2], const Standard_Boolean isParallel) + : mySplits (aSplits), + myIsParallel (isParallel) { - RadixSorter::Sort (theRange.myStart, theRange.myFinal, theRange.myDigit); } + + //! Runs sorting function for the given range. + void operator()(const Standard_Integer theIndex) const + { + RadixSorter::Sort (mySplits[theIndex].myStart, mySplits[theIndex].myFinal, + mySplits[theIndex].myDigit, myIsParallel); + } + + private: + void operator=(const Functor&); + + private: + const SortRange (&mySplits)[2]; + Standard_Boolean myIsParallel; }; public: - static void Sort (LinkIterator theStart, LinkIterator theFinal, Standard_Integer theDigit) + static void Sort (LinkIterator theStart, LinkIterator theFinal, Standard_Integer theDigit, const Standard_Boolean isParallel) { if (theDigit < 24) { @@ -136,7 +151,7 @@ namespace BVH {anOffset, theFinal, theDigit - 1} }; - OSD_Parallel::ForEach (std::begin (aSplits), std::end (aSplits), Functor ()); + OSD_Parallel::For (0, 2, Functor (aSplits, isParallel), !isParallel); } } @@ -202,7 +217,7 @@ void BVH_RadixSorter::Perform (BVH_Set* theSet, const Standard_Integ } // Step 2 -- Sort primitives by their Morton codes using radix sort - BVH::RadixSorter::Sort (myEncodedLinks->begin(), myEncodedLinks->end(), 29); + BVH::RadixSorter::Sort (myEncodedLinks->begin(), myEncodedLinks->end(), 29, this->IsParallel()); NCollection_Array1 aLinkMap (theStart, theFinal); for (Standard_Integer aLinkIdx = theStart; aLinkIdx <= theFinal; ++aLinkIdx) diff --git a/src/BVH/BVH_Sorter.hxx b/src/BVH/BVH_Sorter.hxx index 59e2fbd638..30a0aa64f6 100644 --- a/src/BVH/BVH_Sorter.hxx +++ b/src/BVH/BVH_Sorter.hxx @@ -24,6 +24,11 @@ class BVH_Sorter { public: + //! Performs default initialization. + BVH_Sorter() + : myIsParallel (Standard_False) + { } + //! Releases resources of BVH sorter. virtual ~BVH_Sorter() { } @@ -33,6 +38,21 @@ public: //! Sorts the given (inclusive) range in the set. virtual void Perform (BVH_Set* theSet, const Standard_Integer theStart, const Standard_Integer theFinal) = 0; + //! Returns parallel flag. + inline Standard_Boolean IsParallel() const + { + return myIsParallel; + } + + //! Set parallel flag contolling possibility of parallel execution. + inline void SetParallel(const Standard_Boolean isParallel) + { + myIsParallel = isParallel; + } + +private: + + Standard_Boolean myIsParallel; }; #endif // _BVH_Sorter_Header