diff --git a/src/BSplCLib/BSplCLib.cxx b/src/BSplCLib/BSplCLib.cxx index 00f1e1815b..340c30d5fc 100644 --- a/src/BSplCLib/BSplCLib.cxx +++ b/src/BSplCLib/BSplCLib.cxx @@ -73,31 +73,40 @@ public: //purpose : //======================================================================= -void BSplCLib::Hunt (const Array1OfReal& XX, - const Standard_Real X, - Standard_Integer& Ilc) +void BSplCLib::Hunt (const TColStd_Array1OfReal& theArray, + const Standard_Real theX, + Standard_Integer& theXPos) { // replaced by simple dichotomy (RLE) - Ilc = XX.Lower(); - if (XX.Length() <= 1) return; - const Standard_Real *px = &XX(Ilc); - px -= Ilc; - - if (X < px[Ilc]) { - Ilc--; + if (theArray.First() > theX) + { + theXPos = theArray.Lower() - 1; return; } - Standard_Integer Ihi = XX.Upper(); - if (X > px[Ihi]) { - Ilc = Ihi + 1; + else if (theArray.Last() < theX) + { + theXPos = theArray.Upper() + 1; return; } - Standard_Integer Im; - while (Ihi - Ilc != 1) { - Im = (Ihi + Ilc) >> 1; - if (X > px[Im]) Ilc = Im; - else Ihi = Im; + theXPos = theArray.Lower(); + if (theArray.Length() <= 1) + { + return; + } + + Standard_Integer aHi = theArray.Upper(); + while (aHi - theXPos != 1) + { + const Standard_Integer aMid = (aHi + theXPos) / 2; + if (theArray.Value (aMid) < theX) + { + theXPos = aMid; + } + else + { + aHi = aMid; + } } } diff --git a/src/BSplCLib/BSplCLib.hxx b/src/BSplCLib/BSplCLib.hxx index 78701cc706..946ac3ca43 100644 --- a/src/BSplCLib/BSplCLib.hxx +++ b/src/BSplCLib/BSplCLib.hxx @@ -124,23 +124,16 @@ public: DEFINE_STANDARD_ALLOC - //! This routine searches the position of the real - //! value X in the ordered set of real values XX. + //! This routine searches the position of the real value theX + //! in the monotonically increasing set of real values theArray using bisection algorithm. //! - //! The elements in the table XX are either - //! monotonically increasing or monotonically - //! decreasing. + //! If the given value is out of range or array values, algorithm returns either + //! theArray.Lower()-1 or theArray.Upper()+1 depending on theX position in the ordered set. //! - //! The input value Iloc is used to initialize the - //! algorithm : if Iloc is outside of the bounds - //! [XX.Lower(), -- XX.Upper()] the bisection algorithm - //! is used else the routine searches from a previous - //! known position by increasing steps then converges - //! by bisection. - //! - //! This routine is used to locate a knot value in a - //! set of knots. - Standard_EXPORT static void Hunt (const TColStd_Array1OfReal& XX, const Standard_Real X, Standard_Integer& Iloc); + //! This routine is used to locate a knot value in a set of knots. + Standard_EXPORT static void Hunt (const TColStd_Array1OfReal& theArray, + const Standard_Real theX, + Standard_Integer& theXPos); //! Computes the index of the knots value which gives //! the start point of the curve. diff --git a/src/GCPnts/GCPnts_QuasiUniformDeflection.pxx b/src/GCPnts/GCPnts_QuasiUniformDeflection.pxx index 91ee68b511..aff5f87a79 100644 --- a/src/GCPnts/GCPnts_QuasiUniformDeflection.pxx +++ b/src/GCPnts/GCPnts_QuasiUniformDeflection.pxx @@ -215,7 +215,9 @@ static Standard_Boolean PerformComposite (TColStd_SequenceOfReal& Parameters, Standard_Real Ua = U1; for (Standard_Integer Index = PIndex;;) { - Standard_Real Ub = Min (U2, TI (Index + 1)); + Standard_Real Ub = Index + 1 <= TI.Upper() + ? Min (U2, TI (Index + 1)) + : U2; if (!PerformCurve (Parameters, Points, C, Deflection, Ua, Ub, EPSILON, Continuity)) return Standard_False; diff --git a/src/GCPnts/GCPnts_UniformDeflection.pxx b/src/GCPnts/GCPnts_UniformDeflection.pxx index ab582b53ef..f50fb5c57c 100644 --- a/src/GCPnts/GCPnts_UniformDeflection.pxx +++ b/src/GCPnts/GCPnts_UniformDeflection.pxx @@ -174,7 +174,9 @@ static Standard_Boolean PerformComposite (TColStd_SequenceOfReal& Parameters, Standard_Real Ua = U1; for (Standard_Integer Index = PIndex;;) { - Standard_Real Ub = Min (U2, TI (Index + 1)); + Standard_Real Ub = Index + 1 <= TI.Upper() + ? Min (U2, TI (Index + 1)) + : U2; if (!PerformCurve (Parameters, Points, C, Deflection, Ua, Ub, EPSILON, WithControl)) { diff --git a/src/Geom/Geom_BSplineCurve_1.cxx b/src/Geom/Geom_BSplineCurve_1.cxx index 0354196746..303374e934 100644 --- a/src/Geom/Geom_BSplineCurve_1.cxx +++ b/src/Geom/Geom_BSplineCurve_1.cxx @@ -14,14 +14,6 @@ // Alternatively, this file may be used under the terms of Open CASCADE // commercial license or contractual agreement. -// 03-02-97 : pmn ->LocateU sur Periodic (PRO6963), -// bon appel a LocateParameter (PRO6973) et mise en conformite avec -// le cdl de LocateU, lorsque U est un noeud (PRO6988) - -#define No_Standard_OutOfRange -#define No_Standard_DimensionError - - #include #include #include @@ -809,7 +801,12 @@ void Geom_BSplineCurve::LocateU else { I1 = 1; BSplCLib::Hunt (CKnots, NewU, I1); - while ( Abs( CKnots(I1+1) - NewU) <= PParametricTolerance) I1++; + I1 = Max (Min (I1, CKnots.Upper()), CKnots.Lower()); + while (I1 + 1 <= CKnots.Upper() + && Abs (CKnots (I1 + 1) - NewU) <= PParametricTolerance) + { + I1++; + } if ( Abs( CKnots(I1) - NewU) <= PParametricTolerance) { I2 = I1; } diff --git a/src/Geom/Geom_BSplineSurface_1.cxx b/src/Geom/Geom_BSplineSurface_1.cxx index db99bfcdd8..02610081d3 100644 --- a/src/Geom/Geom_BSplineSurface_1.cxx +++ b/src/Geom/Geom_BSplineSurface_1.cxx @@ -23,10 +23,6 @@ // + bon appel a LocateParameter (PRO6973). // RBD : 15/10/98 ; Le cache est desormais defini sur [-1,1] (pro15537). -#define No_Standard_OutOfRange -#define No_Standard_DimensionError - - #include #include #include @@ -1360,7 +1356,12 @@ void Geom_BSplineSurface::LocateU else { I1 = 1; BSplCLib::Hunt (Knots, NewU, I1); - while ( Abs( Knots(I1+1) - NewU) <= PParametricTolerance) I1++; + I1 = Max (Min (I1, Knots.Upper()), Knots.Lower()); + while (I1 + 1 <= Knots.Upper() + && Abs (Knots (I1 + 1) - NewU) <= PParametricTolerance) + { + I1++; + } if ( Abs( Knots(I1) - NewU) <= PParametricTolerance) { I2 = I1; } @@ -1408,7 +1409,12 @@ void Geom_BSplineSurface::LocateV else { I1 = 1; BSplCLib::Hunt (Knots, NewV, I1); - while ( Abs( Knots(I1+1) - NewV) <= PParametricTolerance) I1++; + I1 = Max (Min (I1, Knots.Upper()), Knots.Lower()); + while (I1 + 1 <= Knots.Upper() + && Abs (Knots (I1 + 1) - NewV) <= PParametricTolerance) + { + I1++; + } if ( Abs( Knots(I1) - NewV) <= PParametricTolerance) { I2 = I1; } diff --git a/src/Geom2d/Geom2d_BSplineCurve_1.cxx b/src/Geom2d/Geom2d_BSplineCurve_1.cxx index a4142a5b2f..d6b231ec17 100644 --- a/src/Geom2d/Geom2d_BSplineCurve_1.cxx +++ b/src/Geom2d/Geom2d_BSplineCurve_1.cxx @@ -14,14 +14,6 @@ // Alternatively, this file may be used under the terms of Open CASCADE // commercial license or contractual agreement. -// 03-02-97 : pmn ->LocateU sur Periodic (PRO6963), -// bon appel a LocateParameter (PRO6973) et mise en conformite avec -// le cdl de LocateU, lorsque U est un noeud (PRO6988) - -#define No_Standard_OutOfRange -#define No_Standard_DimensionError - - #include #include #include @@ -820,7 +812,12 @@ void Geom2d_BSplineCurve::LocateU else { I1 = 1; BSplCLib::Hunt (CKnots, NewU, I1); - while ( Abs( CKnots(I1+1) - NewU) <= PParametricTolerance) I1++; + I1 = Max (Min (I1, CKnots.Upper()), CKnots.Lower()); + while (I1 + 1 <= CKnots.Upper() + && Abs (CKnots (I1 + 1) - NewU) <= PParametricTolerance) + { + I1++; + } if ( Abs( CKnots(I1) - NewU) <= PParametricTolerance) { I2 = I1; } diff --git a/src/Law/Law_BSpline.cxx b/src/Law/Law_BSpline.cxx index 1d19cf83e8..c4eaeb3e3b 100644 --- a/src/Law/Law_BSpline.cxx +++ b/src/Law/Law_BSpline.cxx @@ -1943,7 +1943,12 @@ void Law_BSpline::LocateU else { I1 = 1; BSplCLib::Hunt (CKnots, NewU, I1); - while ( Abs( CKnots(I1+1) - NewU) <= Abs(ParametricTolerance)) I1++; + I1 = Max (Min (I1, CKnots.Upper()), CKnots.Lower()); + while (I1 + 1 <= CKnots.Upper() + && Abs (CKnots (I1 + 1) - NewU) <= Abs(ParametricTolerance)) + { + I1++; + } if ( Abs( CKnots(I1) - NewU) <= Abs(ParametricTolerance)) { I2 = I1; }