From 7e17e8f08df2f3db0424c991b990c9f5c298bda8 Mon Sep 17 00:00:00 2001 From: vpa Date: Mon, 3 Aug 2015 18:41:56 +0300 Subject: [PATCH] 0026249: Visualization, TKV3d - fix possible division by zero in SelectMgr_RectangularFrustum Added zero-length check for triangle normal in SelectMgr_RectangularFrustum::Overlaps --- .../SelectMgr_RectangularFrustum.cxx | 20 ++++++++++++++++++- 1 file changed, 19 insertions(+), 1 deletion(-) diff --git a/src/SelectMgr/SelectMgr_RectangularFrustum.cxx b/src/SelectMgr/SelectMgr_RectangularFrustum.cxx index 8cbde2a460..b98d41dec3 100644 --- a/src/SelectMgr/SelectMgr_RectangularFrustum.cxx +++ b/src/SelectMgr/SelectMgr_RectangularFrustum.cxx @@ -692,7 +692,25 @@ Standard_Boolean SelectMgr_RectangularFrustum::Overlaps (const gp_Pnt& thePnt1, SelectMgr_Vec3 aTrEdges[3] = { SelectMgr_Vec3 (thePnt2.X() - thePnt1.X(), thePnt2.Y() - thePnt1.Y(), thePnt2.Z() - thePnt1.Z()), SelectMgr_Vec3 (thePnt3.X() - thePnt2.X(), thePnt3.Y() - thePnt2.Y(), thePnt3.Z() - thePnt2.Z()), SelectMgr_Vec3 (thePnt1.X() - thePnt3.X(), thePnt1.Y() - thePnt3.Y(), thePnt1.Z() - thePnt3.Z()) }; - SelectMgr_Vec3 anEdge = (aPnt1 - myNearPickedPnt) * (1.0 / DOT (aTriangleNormal, myViewRayDir)); + + Standard_Real anAlpha = DOT (aTriangleNormal, myViewRayDir); + if (Abs (anAlpha) < gp::Resolution()) + { + // handle degenerated triangles: in this case, there is no possible way to detect overlap correctly. + if (aTriangleNormal.SquareModulus() < gp::Resolution()) + { + theDepth = std::numeric_limits::max(); + return Standard_False; + } + + // handle the case when triangle normal and selecting frustum direction are orthogonal: for this case, overlap + // is detected correctly, and distance to triangle's plane can be measured as distance to its arbitrary vertex. + const SelectMgr_Vec3 aDiff = myNearPickedPnt - aPnt1; + theDepth = DOT (aTriangleNormal, aDiff); + return Standard_True; + } + + SelectMgr_Vec3 anEdge = (aPnt1 - myNearPickedPnt) * (1.0 / anAlpha); Standard_Real aTime = DOT (aTriangleNormal, anEdge);