From e6ebc2b9b99e3a99cb0fc7dc42dec4c586599b98 Mon Sep 17 00:00:00 2001 From: Tomasz Wlostowski Date: Fri, 3 Jun 2022 22:18:25 +0200 Subject: [PATCH] geometry: use dedicated 64-bit integer square root for distance computations Guarantees 1 LSB error, while the C++ double type has 55 mantissa bits (meaning for sqrt(X) >~ 2^22.5) the error is not guaranteed. --- libs/kimath/src/geometry/seg.cpp | 49 +++++++++++++++++-- .../libs/kimath/geometry/test_segment.cpp | 2 +- 2 files changed, 46 insertions(+), 5 deletions(-) diff --git a/libs/kimath/src/geometry/seg.cpp b/libs/kimath/src/geometry/seg.cpp index 2f78e43c51..479317bd95 100644 --- a/libs/kimath/src/geometry/seg.cpp +++ b/libs/kimath/src/geometry/seg.cpp @@ -35,6 +35,39 @@ int sgn( T aVal ) return ( T( 0 ) < aVal ) - ( aVal < T( 0 ) ); } +template +constexpr T sqrt_helper(T x, T lo, T hi) +{ + if (lo == hi) + return lo; + + const T mid = (lo + hi + 1) / 2; + if (x / mid < mid) + return sqrt_helper(x, lo, mid - 1); + else + return sqrt_helper(x, mid, hi); +} + +template +constexpr T ct_sqrt(T x) +{ + return sqrt_helper(x, 0, x / 2 + 1); +} + +template +T isqrt(T x) +{ + T r = (T) std::sqrt((double) x); + T sqrt_max = ct_sqrt(std::numeric_limits::max()); + + while (r < sqrt_max && r * r < x) + r++; + while (r > sqrt_max || r * r > x) + r--; + + return r; +} + SEG::ecoord SEG::SquaredDistance( const SEG& aSeg ) const { @@ -207,7 +240,7 @@ bool SEG::Collide( const SEG& aSeg, int aClearance, int* aActual ) const if( dist_sq == 0 || dist_sq < (ecoord) aClearance * aClearance ) { if( aActual ) - *aActual = sqrt( dist_sq ); + *aActual = isqrt( dist_sq ); return true; } @@ -282,13 +315,13 @@ VECTOR2I SEG::LineProject( const VECTOR2I& aP ) const int SEG::Distance( const SEG& aSeg ) const { - return KiROUND( sqrt( SquaredDistance( aSeg ) ) ); + return isqrt( SquaredDistance( aSeg ) ); } int SEG::Distance( const VECTOR2I& aP ) const { - return KiROUND( sqrt( SquaredDistance( aP ) ) ); + return isqrt( SquaredDistance( aP ) ); } @@ -297,8 +330,16 @@ int SEG::LineDistance( const VECTOR2I& aP, bool aDetermineSide ) const ecoord p = ecoord{ A.y } - B.y; ecoord q = ecoord{ B.x } - A.x; ecoord r = -p * A.x - q * A.y; + ecoord l = p * p + q * q; + ecoord det = p * aP.x + q * aP.y + r; + ecoord dist_sq = 0; - ecoord dist = KiROUND( ( p * aP.x + q * aP.y + r ) / sqrt( p * p + q * q ) ); + if( l > 0 ) + { + dist_sq = rescale( det, det, l ); + } + + ecoord dist = isqrt( dist_sq ); return aDetermineSide ? dist : std::abs( dist ); } diff --git a/qa/unittests/libs/kimath/geometry/test_segment.cpp b/qa/unittests/libs/kimath/geometry/test_segment.cpp index 78a7cc1ab1..918167be21 100644 --- a/qa/unittests/libs/kimath/geometry/test_segment.cpp +++ b/qa/unittests/libs/kimath/geometry/test_segment.cpp @@ -340,7 +340,7 @@ static const std::vector seg_vec_dist_cases = { "At end (not collinear)", { { 0, 0 }, { 1000, 0 } }, { 1000 + 200, 200 }, - 283, // sqrt(200^2 + 200^2) = 282.8, rounded to nearest + 282, // sqrt(200^2 + 200^2) = 282.8, rounded to nearest }, }; // clang-format on