1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-06 18:26:22 +03:00

0028788: Visualization, StdPrs_WFShape - Add option to compute Isolines using multiple threads

StdPrs_WFShape::Add() now accepts new argument theIsParallel (FALSE by default)
for computing Isolines using multiple threads (if there is more then 1 Face).
StdPrs_Isolines::addOnTriangulation() - changed the iteration order (cosmetics).
This commit is contained in:
kgv 2017-05-25 17:36:10 +03:00 committed by bugmaster
parent 3ae5dc8173
commit 884cafd893
5 changed files with 144 additions and 108 deletions

View File

@ -142,11 +142,10 @@ Bnd_Box Prs3d_ShapeTool::FaceBound() const
//purpose : //purpose :
//======================================================================= //=======================================================================
Standard_Boolean Prs3d_ShapeTool::IsPlanarFace() const Standard_Boolean Prs3d_ShapeTool::IsPlanarFace (const TopoDS_Face& theFace)
{ {
TopLoc_Location l; TopLoc_Location l;
const TopoDS_Face& F = TopoDS::Face(myFaceExplorer.Current()); const Handle(Geom_Surface)& S = BRep_Tool::Surface(theFace, l);
const Handle(Geom_Surface)& S = BRep_Tool::Surface(F, l);
if (S.IsNull()) if (S.IsNull())
{ {
return Standard_False; return Standard_False;

View File

@ -17,37 +17,29 @@
#ifndef _Prs3d_ShapeTool_HeaderFile #ifndef _Prs3d_ShapeTool_HeaderFile
#define _Prs3d_ShapeTool_HeaderFile #define _Prs3d_ShapeTool_HeaderFile
#include <Standard.hxx>
#include <Standard_DefineAlloc.hxx> #include <Standard_DefineAlloc.hxx>
#include <Standard_Handle.hxx> #include <TopoDS.hxx>
#include <TopoDS_Face.hxx>
#include <TopoDS_Shape.hxx> #include <TopoDS_Shape.hxx>
#include <TopExp_Explorer.hxx> #include <TopExp_Explorer.hxx>
#include <TopTools_IndexedDataMapOfShapeListOfShape.hxx> #include <TopTools_IndexedDataMapOfShapeListOfShape.hxx>
#include <TopTools_IndexedMapOfShape.hxx> #include <TopTools_IndexedMapOfShape.hxx>
#include <Standard_Integer.hxx>
#include <Standard_Boolean.hxx>
#include <TopTools_HSequenceOfShape.hxx> #include <TopTools_HSequenceOfShape.hxx>
class TopoDS_Shape;
class TopoDS_Face;
class Bnd_Box; class Bnd_Box;
class TopoDS_Edge; class TopoDS_Edge;
class TopoDS_Vertex; class TopoDS_Vertex;
class Poly_Triangulation; class Poly_Triangulation;
class TopLoc_Location;
class Poly_PolygonOnTriangulation; class Poly_PolygonOnTriangulation;
class Poly_Polygon3D; class Poly_Polygon3D;
//! describes the behaviour requested for a wireframe //! describes the behaviour requested for a wireframe
//! shape presentation. //! shape presentation.
class Prs3d_ShapeTool class Prs3d_ShapeTool
{ {
public: public:
DEFINE_STANDARD_ALLOC DEFINE_STANDARD_ALLOC
//! Constructs the tool and initializes it using theShape and theAllVertices //! Constructs the tool and initializes it using theShape and theAllVertices
//! (optional) arguments. By default, only isolated and internal vertices are considered, //! (optional) arguments. By default, only isolated and internal vertices are considered,
//! however if theAllVertices argument is equal to True, all shape's vertices are taken into account. //! however if theAllVertices argument is equal to True, all shape's vertices are taken into account.
@ -63,7 +55,11 @@ public:
Standard_EXPORT Bnd_Box FaceBound() const; Standard_EXPORT Bnd_Box FaceBound() const;
Standard_EXPORT Standard_Boolean IsPlanarFace() const; Standard_Boolean IsPlanarFace() const
{
const TopoDS_Face& aFace = TopoDS::Face (myFaceExplorer.Current());
return IsPlanarFace (aFace);
}
Standard_EXPORT void InitCurve(); Standard_EXPORT void InitCurve();
@ -97,19 +93,12 @@ public:
Standard_EXPORT Handle(Poly_Polygon3D) Polygon3D (TopLoc_Location& l) const; Standard_EXPORT Handle(Poly_Polygon3D) Polygon3D (TopLoc_Location& l) const;
public:
Standard_EXPORT static Standard_Boolean IsPlanarFace (const TopoDS_Face& theFace);
protected:
private: private:
TopoDS_Shape myShape; TopoDS_Shape myShape;
TopExp_Explorer myFaceExplorer; TopExp_Explorer myFaceExplorer;
TopTools_IndexedDataMapOfShapeListOfShape myEdgeMap; TopTools_IndexedDataMapOfShapeListOfShape myEdgeMap;
@ -117,13 +106,6 @@ private:
Standard_Integer myEdge; Standard_Integer myEdge;
Standard_Integer myVertex; Standard_Integer myVertex;
}; };
#endif // _Prs3d_ShapeTool_HeaderFile #endif // _Prs3d_ShapeTool_HeaderFile

View File

@ -231,24 +231,35 @@ void StdPrs_Isolines::addOnTriangulation (const Handle(Poly_Triangulation)& theT
Prs3d_NListOfSequenceOfPnt& theUPolylines, Prs3d_NListOfSequenceOfPnt& theUPolylines,
Prs3d_NListOfSequenceOfPnt& theVPolylines) Prs3d_NListOfSequenceOfPnt& theVPolylines)
{ {
const Standard_Integer aNbIsoU = theUIsoParams.Length();
const Standard_Integer aNbIsoV = theVIsoParams.Length();
SeqOfVecOfSegments aUPolylines, aVPolylines;
const Poly_Array1OfTriangle& aTriangles = theTriangulation->Triangles(); const Poly_Array1OfTriangle& aTriangles = theTriangulation->Triangles();
const TColgp_Array1OfPnt& aNodes = theTriangulation->Nodes(); const TColgp_Array1OfPnt& aNodes = theTriangulation->Nodes();
const TColgp_Array1OfPnt2d& aUVNodes = theTriangulation->UVNodes(); const TColgp_Array1OfPnt2d& aUVNodes = theTriangulation->UVNodes();
for (Standard_Integer anUVIter = 0; anUVIter < 2; ++anUVIter)
{
const Standard_Boolean isUIso = anUVIter == 0;
const TColStd_SequenceOfReal& anIsoParams = isUIso ? theUIsoParams : theVIsoParams;
const Standard_Integer aNbIsolines = anIsoParams.Length();
if (aNbIsolines == 0)
{
continue;
}
TColStd_Array1OfInteger aUIsoIndexes (1, aNbIsoU); SeqOfVecOfSegments aPolylines;
TColStd_Array1OfInteger aVIsoIndexes (1, aNbIsoV); TColStd_Array1OfInteger anIsoIndexes (1, aNbIsolines);
aUIsoIndexes.Init (-1); anIsoIndexes.Init (-1);
aVIsoIndexes.Init (-1); for (Standard_Integer anIsoIdx = 1; anIsoIdx <= aNbIsolines; ++anIsoIdx)
{
const gp_Lin2d anIsolineUV = isUIso ? isoU (anIsoParams.Value (anIsoIdx)) : isoV (anIsoParams.Value (anIsoIdx));
Handle(VecOfSegments) anIsoPnts;
if (anIsoIndexes.Value (anIsoIdx) != -1)
{
anIsoPnts = aPolylines.ChangeValue (anIsoIndexes.Value (anIsoIdx));
}
for (Standard_Integer anI = aTriangles.Lower(); anI <= aTriangles.Upper(); ++anI) for (Standard_Integer aTriIter = aTriangles.Lower(); aTriIter <= aTriangles.Upper(); ++aTriIter)
{ {
Standard_Integer aNodeIdxs[3]; Standard_Integer aNodeIdxs[3];
aTriangles.Value (anI).Get (aNodeIdxs[0], aNodeIdxs[1],aNodeIdxs[2]); aTriangles.Value (aTriIter).Get (aNodeIdxs[0], aNodeIdxs[1],aNodeIdxs[2]);
const gp_Pnt aNodesXYZ[3] = { aNodes.Value (aNodeIdxs[0]), const gp_Pnt aNodesXYZ[3] = { aNodes.Value (aNodeIdxs[0]),
aNodes.Value (aNodeIdxs[1]), aNodes.Value (aNodeIdxs[1]),
aNodes.Value (aNodeIdxs[2]) }; aNodes.Value (aNodeIdxs[2]) };
@ -256,51 +267,20 @@ void StdPrs_Isolines::addOnTriangulation (const Handle(Poly_Triangulation)& theT
aUVNodes.Value (aNodeIdxs[1]), aUVNodes.Value (aNodeIdxs[1]),
aUVNodes.Value (aNodeIdxs[2]) }; aUVNodes.Value (aNodeIdxs[2]) };
// Evaluate polyline points for u isolines.
for (Standard_Integer anIsoIdx = 1; anIsoIdx <= aNbIsoU; ++anIsoIdx)
{
SegOnIso aSegment;
const gp_Lin2d anIsolineUV = isoU (theUIsoParams.Value (anIsoIdx));
// Find intersections with triangle in uv space and its projection on triangulation. // Find intersections with triangle in uv space and its projection on triangulation.
if (!findSegmentOnTriangulation (theSurface, true, anIsolineUV, aNodesXYZ, aNodesUV, aSegment))
{
continue;
}
if (aUIsoIndexes.Value (anIsoIdx) == -1)
{
aUPolylines.Append (new VecOfSegments());
aUIsoIndexes.SetValue (anIsoIdx, aUPolylines.Size());
}
Handle(VecOfSegments) anIsoPnts = aUPolylines.ChangeValue (aUIsoIndexes.Value (anIsoIdx));
if (!theLocation.IsIdentity())
{
aSegment[0].Pnt.Transform (theLocation);
aSegment[1].Pnt.Transform (theLocation);
}
anIsoPnts->Append (aSegment);
}
// Evaluate polyline points for v isolines.
for (Standard_Integer anIsoIdx = 1; anIsoIdx <= aNbIsoV; ++anIsoIdx)
{
SegOnIso aSegment; SegOnIso aSegment;
const gp_Lin2d anIsolineUV = isoV (theVIsoParams.Value (anIsoIdx)); if (!findSegmentOnTriangulation (theSurface, isUIso, anIsolineUV, aNodesXYZ, aNodesUV, aSegment))
if (!findSegmentOnTriangulation (theSurface, false, anIsolineUV, aNodesXYZ, aNodesUV, aSegment))
{ {
continue; continue;
} }
if (aVIsoIndexes.Value (anIsoIdx) == -1) if (anIsoPnts.IsNull())
{ {
aVPolylines.Append (new VecOfSegments()); aPolylines.Append (new VecOfSegments());
aVIsoIndexes.SetValue (anIsoIdx, aVPolylines.Size()); anIsoIndexes.SetValue (anIsoIdx, aPolylines.Size());
anIsoPnts = aPolylines.ChangeValue (anIsoIndexes.Value (anIsoIdx));
} }
Handle(VecOfSegments) anIsoPnts = aVPolylines.ChangeValue (aVIsoIndexes.Value (anIsoIdx));
if (!theLocation.IsIdentity()) if (!theLocation.IsIdentity())
{ {
aSegment[0].Pnt.Transform (theLocation); aSegment[0].Pnt.Transform (theLocation);
@ -310,8 +290,8 @@ void StdPrs_Isolines::addOnTriangulation (const Handle(Poly_Triangulation)& theT
} }
} }
sortSegments (aUPolylines, theUPolylines); sortSegments (aPolylines, isUIso ? theUPolylines : theVPolylines);
sortSegments (aVPolylines, theVPolylines); }
} }
//================================================================== //==================================================================

View File

@ -19,6 +19,7 @@
#include <BRepAdaptor_Curve.hxx> #include <BRepAdaptor_Curve.hxx>
#include <BRepAdaptor_Surface.hxx> #include <BRepAdaptor_Surface.hxx>
#include <BRepAdaptor_HSurface.hxx> #include <BRepAdaptor_HSurface.hxx>
#include <OSD_Parallel.hxx>
#include <StdPrs_DeflectionCurve.hxx> #include <StdPrs_DeflectionCurve.hxx>
#include <StdPrs_ToolTriangulatedShape.hxx> #include <StdPrs_ToolTriangulatedShape.hxx>
#include <StdPrs_Isolines.hxx> #include <StdPrs_Isolines.hxx>
@ -41,13 +42,56 @@
#include <TopoDS.hxx> #include <TopoDS.hxx>
#include <TopTools_ListIteratorOfListOfShape.hxx> #include <TopTools_ListIteratorOfListOfShape.hxx>
//! Functor for executing StdPrs_Isolines in parallel threads.
class StdPrs_WFShape_IsoFunctor
{
public:
StdPrs_WFShape_IsoFunctor (Prs3d_NListOfSequenceOfPnt& thePolylinesU,
Prs3d_NListOfSequenceOfPnt& thePolylinesV,
const std::vector<TopoDS_Face>& theFaces,
const Handle(Prs3d_Drawer)& theDrawer,
Standard_Real theShapeDeflection)
: myPolylinesU (thePolylinesU),
myPolylinesV (thePolylinesV),
myFaces (theFaces),
myDrawer (theDrawer),
myShapeDeflection (theShapeDeflection)
{
//
}
void operator()(const Standard_Integer& theIndex) const
{
Prs3d_NListOfSequenceOfPnt aPolylinesU, aPolylinesV;
const TopoDS_Face& aFace = myFaces[theIndex];
StdPrs_Isolines::Add (aFace, myDrawer, myShapeDeflection, aPolylinesU, aPolylinesV);
{
Standard_Mutex::Sentry aLock (myMutex);
myPolylinesU.Append (aPolylinesU);
myPolylinesV.Append (aPolylinesV);
}
}
private:
StdPrs_WFShape_IsoFunctor operator= (StdPrs_WFShape_IsoFunctor& );
private:
Prs3d_NListOfSequenceOfPnt& myPolylinesU;
Prs3d_NListOfSequenceOfPnt& myPolylinesV;
const std::vector<TopoDS_Face>& myFaces;
const Handle(Prs3d_Drawer)& myDrawer;
mutable Standard_Mutex myMutex;
const Standard_Real myShapeDeflection;
};
// ========================================================================= // =========================================================================
// function : Add // function : Add
// purpose : // purpose :
// ========================================================================= // =========================================================================
void StdPrs_WFShape::Add (const Handle (Prs3d_Presentation)& thePresentation, void StdPrs_WFShape::Add (const Handle(Prs3d_Presentation)& thePresentation,
const TopoDS_Shape& theShape, const TopoDS_Shape& theShape,
const Handle (Prs3d_Drawer)& theDrawer) const Handle(Prs3d_Drawer)& theDrawer,
Standard_Boolean theIsParallel)
{ {
if (theShape.IsNull()) if (theShape.IsNull())
{ {
@ -106,14 +150,43 @@ void StdPrs_WFShape::Add (const Handle (Prs3d_Presentation)& thePresentation,
aVPolylinesPtr = &aCommonPolylines; aVPolylinesPtr = &aCommonPolylines;
} }
for (aTool.InitFace(); aTool.MoreFace(); aTool.NextFace()) bool isParallelIso = false;
if (theIsParallel)
{ {
if (aTool.IsPlanarFace() && !theDrawer->IsoOnPlane()) Standard_Integer aNbFaces = 0;
for (TopExp_Explorer aFaceExplorer (theShape, TopAbs_FACE); aFaceExplorer.More(); aFaceExplorer.Next())
{ {
continue; ++aNbFaces;
}
if (aNbFaces > 1)
{
isParallelIso = true;
std::vector<TopoDS_Face> aFaces (aNbFaces);
aNbFaces = 0;
for (TopExp_Explorer aFaceExplorer (theShape, TopAbs_FACE); aFaceExplorer.More(); aFaceExplorer.Next())
{
const TopoDS_Face& aFace = TopoDS::Face (aFaceExplorer.Current());
if (theDrawer->IsoOnPlane() || !Prs3d_ShapeTool::IsPlanarFace (aFace))
{
aFaces[aNbFaces++] = aFace;
}
} }
StdPrs_Isolines::Add (aTool.GetFace(), theDrawer, aShapeDeflection, *aUPolylinesPtr, *aVPolylinesPtr); StdPrs_WFShape_IsoFunctor anIsoFunctor (*aUPolylinesPtr, *aVPolylinesPtr, aFaces, theDrawer, aShapeDeflection);
OSD_Parallel::For (0, aNbFaces - 1, anIsoFunctor, aNbFaces < 2);
}
}
if (!isParallelIso)
{
for (TopExp_Explorer aFaceExplorer (theShape, TopAbs_FACE); aFaceExplorer.More(); aFaceExplorer.Next())
{
const TopoDS_Face& aFace = TopoDS::Face (aFaceExplorer.Current());
if (theDrawer->IsoOnPlane() || !Prs3d_ShapeTool::IsPlanarFace (aFace))
{
StdPrs_Isolines::Add (aFace, theDrawer, aShapeDeflection, *aUPolylinesPtr, *aVPolylinesPtr);
}
}
} }
Prs3d::AddPrimitivesGroup (thePresentation, anIsoAspectU, aUPolylines); Prs3d::AddPrimitivesGroup (thePresentation, anIsoAspectU, aUPolylines);

View File

@ -33,9 +33,11 @@ public:
//! @param thePresentation [in] the presentation. //! @param thePresentation [in] the presentation.
//! @param theShape [in] the shape. //! @param theShape [in] the shape.
//! @param theDrawer [in] the draw settings. //! @param theDrawer [in] the draw settings.
Standard_EXPORT static void Add (const Handle (Prs3d_Presentation)& thePresentation, //! @param theIsParallel [in] perform algorithm using multiple threads
Standard_EXPORT static void Add (const Handle(Prs3d_Presentation)& thePresentation,
const TopoDS_Shape& theShape, const TopoDS_Shape& theShape,
const Handle (Prs3d_Drawer)& theDrawer); const Handle(Prs3d_Drawer)& theDrawer,
Standard_Boolean theIsParallel = Standard_False);
//! Compute free and boundary edges on a triangulation of each face in the given shape. //! Compute free and boundary edges on a triangulation of each face in the given shape.
//! @param theShape [in] the list of triangulated faces //! @param theShape [in] the list of triangulated faces