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

0023650: Slow mesher: one bspline surface, 80 seconds for 132 triangles

Corrections in GeomLib::NormEstim, added norm computing using GeomLProp_SLProps.
Corrections in GeomLib::NormEstim, normal is computed with CSLib::Normal in case of a null normal.
Modified GeomLib::NormEstim Algo in case of null normal.
Added test case bugs/mesh/bug23650
This commit is contained in:
aml 2013-10-17 12:14:52 +04:00 committed by bugmaster
parent f933f9cf88
commit 2b21c64155
2 changed files with 81 additions and 116 deletions

View File

@ -137,6 +137,9 @@
#include <GeomConvert_CompCurveToBSplineCurve.hxx>
#include <GeomConvert_ApproxSurface.hxx>
#include <CSLib.hxx>
#include <CSLib_NormalStatus.hxx>
#include <Standard_ConstructionError.hxx>
@ -2341,12 +2344,6 @@ Standard_Integer GeomLib::NormEstim(const Handle(Geom_Surface)& S,
Standard_Real MDU = DU.SquareMagnitude(), MDV = DV.SquareMagnitude();
Standard_Real h, sign;
Standard_Boolean AlongV;
Handle(Geom_Curve) Iso;
Standard_Real t, first, last, bid1, bid2;
gp_Vec Tang;
if(MDU >= aTol2 && MDV >= aTol2) {
gp_Vec Norm = DU^DV;
Standard_Real Magn = Norm.SquareMagnitude();
@ -2357,125 +2354,62 @@ Standard_Integer GeomLib::NormEstim(const Handle(Geom_Surface)& S,
return 0;
}
else if(MDU < aTol2 && MDV >= aTol2) {
AlongV = Standard_True;
Iso = S->UIso(UV.X());
t = UV.Y();
S->Bounds(bid1, bid2, first, last);
}
else if(MDU >= aTol2 && MDV < aTol2) {
AlongV = Standard_False;
Iso = S->VIso(UV.Y());
t = UV.X();
S->Bounds(first, last, bid1, bid2);
}
else {
return 3;
}
gp_Vec D2U, D2V, D2UV;
Standard_Boolean isDone;
CSLib_NormalStatus aStatus;
gp_Dir aNormal;
Standard_Real L = .001;
S->D2(UV.X(), UV.Y(), DummyPnt, DU, DV, D2U, D2V, D2UV);
CSLib::Normal(DU, DV, D2U, D2V, D2UV, Tol, isDone, aStatus, aNormal);
if(Precision::IsInfinite(Abs(first))) first = t - 1.;
if(Precision::IsInfinite(Abs(last))) last = t + 1.;
if(last - t >= t - first) {
sign = 1.;
}
else {
sign = -1.;
}
if (isDone) {
Standard_Real Umin, Umax, Vmin, Vmax;
Standard_Real step = 1.0e-5;
Standard_Real eps = 1.0e-16;
Standard_Real sign = 1;
S->Bounds(Umin, Umax, Vmin, Vmax);
// Along V
if(MDU < aTol2 && MDV >= aTol2) {
if (UV.Y() + step >= Vmax)
sign = -1.0;
S->D1(UV.X(), UV.Y() + sign * step, DummyPnt, DU, DV);
gp_Vec Norm = DU^DV;
if ((Norm.SquareMagnitude() >= eps) && (Norm.Dot(aNormal) < 0.0))
aNormal.Reverse();
Standard_Real hmax = .01*(last - first);
if(AlongV) {
h = Min(L/sqrt(MDV), hmax);
S->D1(UV.X(), UV.Y() + sign*h, DummyPnt, DU, DV);
}
else {
h = Min(L/sqrt(MDU), hmax);
S->D1(UV.X() + sign*h, UV.Y(), DummyPnt, DU, DV);
}
}
// Along U
if(MDV < aTol2 && MDU >= aTol2) {
if (UV.X() + step >= Umax)
sign = -1.0;
S->D1(UV.X() + sign * step, UV.Y(), DummyPnt, DU, DV);
gp_Vec Norm = DU^DV;
if ((Norm.SquareMagnitude() >= eps) && (Norm.Dot(aNormal) < 0.0))
aNormal.Reverse();
}
gp_Vec DD;
gp_Vec NAux = DU^DV;
Standard_Real h1 = h;
while(NAux.SquareMagnitude() < aTol2) {
h1 += h;
if(AlongV) {
Standard_Real v = UV.Y() + sign*h1;
if(v < first || v > last) return 3;
S->D1(UV.X(), v, DummyPnt, DU, DV);
// quasysingular
if ((aStatus == CSLib_D1NuIsNull) || (aStatus == CSLib_D1NvIsNull) ||
(aStatus == CSLib_D1NuIsParallelD1Nv)) {
N.SetXYZ(aNormal.XYZ());
return 1;
}
// conical
if (aStatus == CSLib_InfinityOfSolutions)
return 2;
}
// computation is impossible
else {
Standard_Real v = UV.X() + sign*h1;
if(v < first || v > last) return 3;
S->D1(v, UV.Y(), DummyPnt, DU, DV);
// conical
if (aStatus == CSLib_D1NIsNull) {
return 2;
}
return 3;
}
NAux = DU^DV;
}
Iso->D2(t, DummyPnt, Tang, DD);
if(DD.SquareMagnitude() >= aTol2) {
gp_Vec NV = DD * (Tang * Tang) - Tang * (Tang * DD);
Standard_Real MagnNV = NV.SquareMagnitude();
if(MagnNV < aTol2) return 3;
MagnNV = sqrt(MagnNV);
N.SetXYZ(NV.XYZ()/MagnNV);
Standard_Real par = .5*(bid2+bid1);
if(AlongV) {
Iso = S->UIso(par);
}
else {
Iso = S->VIso(par);
}
Iso->D2(t, DummyPnt, Tang, DD);
gp_Vec N1V = DD * (Tang * Tang) - Tang * (Tang * DD);
Standard_Real MagnN1V = N1V.SquareMagnitude();
if(MagnN1V < aTol2) return 3;
MagnN1V = sqrt(MagnN1V);
gp_Dir N1(N1V.XYZ()/MagnN1V);
Standard_Integer res = 1;
if(N*N1 < 1. - Tol) res = 2;
if(N*NAux <= 0.) N.Reverse();
return res;
}
else {
//Seems to be conical singular point
if(AlongV) {
NAux = DU^Tang;
}
else {
NAux = Tang^DV;
}
sign = NAux.Magnitude();
if(sign < Tol) return 3;
N = NAux;
return 2;
}
return 3;
}

31
tests/bugs/mesh/bug23650 Normal file
View File

@ -0,0 +1,31 @@
puts "=========="
puts "OCC23650"
puts "=========="
puts ""
##################################################################
# Slow mesher: one bspline surface, 80 seconds for 132 triangles
##################################################################
restore [locate_data_file bug23650_slowmesh.brep] result
tclean result
dchrono h reset
dchrono h start
incmesh result 0.2
dchrono h stop
set info [dchrono h show]
regexp {CPU user time: ([-0-9.+eE]+) seconds} $info full cpu_time
if { $cpu_time > 5. } {
puts "Error : meshing is slow"
} else {
puts "OK: meshing is quite fast"
}
vinit
vdisplay result
vfit
vsetdispmode 1
set only_screen 1