Skip to content

Commit

Permalink
Merge pull request #1726 from isivigno/fixFrechetSegFault
Browse files Browse the repository at this point in the history
Fix seg fault and enable parameter error=0 in FrechetShortcut
  • Loading branch information
dcoeurjo authored Jun 10, 2024
2 parents bfdc406 + 7c78290 commit 2da53eb
Show file tree
Hide file tree
Showing 6 changed files with 7,381 additions and 52 deletions.
9 changes: 7 additions & 2 deletions ChangeLog.md
Original file line number Diff line number Diff line change
Expand Up @@ -106,15 +106,20 @@

- *Topology package*
- Fix KhalimskySpaceND to get it work with BigInteger (Tristan Roussillon,
[#1681](https://github.com/DGtal-team/DGtal/pull/1681)
[#1681](https://github.com/DGtal-team/DGtal/pull/1681))

- *Geometry package*
- Fix Issue #1676 in testStabbingCircleComputer (Tristan Roussillon,
[#1688](https://github.com/DGtal-team/DGtal/pull/1688)
- Fix BoundedLatticePolytopeCounter::countInterior method (Jacques-Olivier Lachaud,
[#1717](https://github.com/DGtal-team/DGtal/pull/1717))
- Fix const attribute that shouldn't be in FreemanChain (Colin Weill--Duflos,
[#1723](https://github.com/DGtal-team/DGtal/pull/1723))
[#1723](https://github.com/DGtal-team/DGtal/pull/1723))
- Fix seg fault due to recent compilers in FrechetShortcut (Bertrand Kerautret,
Isabelle Sivignon [#1726](https://github.com/DGtal-team/DGtal/pull/1726))
- Fix FrechetShortcut to enable the parameter error to be equal to 0 and add new
tests in testFrechetShortcut (Isabelle Sivignon, [#1726](https://github.com/DGtal-team/DGtal/pull/1726))


- *IO*
- Fix of the `getHSV` method in the `Color` class. (David Coeurjolly,
Expand Down
58 changes: 47 additions & 11 deletions src/DGtal/geometry/curves/FrechetShortcut.h
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,9 @@

#include "DGtal/geometry/curves/SegmentComputerUtils.h"


#define PRECISION 0.00000001

namespace DGtal
{

Expand Down Expand Up @@ -399,6 +402,24 @@ namespace DGtal
double *xi, double *yi,
double *xi_prime, double *yi_prime)
{
// I. Sivignon 05/2024: fix to handle the case where r0=0 or
// r1=0. Enables the case where error=0.
// Other special cases where circles are tangent are treated
// below.
if(r0==0)
{
*xi = x0; *yi = y0;
*xi_prime = x0 ; *yi_prime = y0;
return 1;
}
else
if(r1==0)
{
*xi = x1; *yi = y1;
*xi_prime = x1 ; *yi_prime = y1;
return 1;
}

double a, dx, dy, d, h, rx, ry;
double x2, y2;

Expand All @@ -407,10 +428,21 @@ namespace DGtal
*/
dx = x1 - x0;
dy = y1 - y0;

/* Determine the straight-line distance between the centers. */
//d = sqrt((dy*dy) + (dx*dx));
d = hypot(dx,dy); // Suggested by Keith Briggs

if((r0+r1)*(r0+r1)-((dy*dy)+(dx*dx)) < PRECISION)
{ // I. Sivignon 05/2024: fix to handle the case where
// circles are tangent but radii are non zero.

double alpha= r0/d;
*xi = x0 + dx*alpha;
*yi = y0 + dy*alpha;
*xi_prime = *xi; *yi_prime = *yi;
return 1;
}

/* Check for solvability. */
if (d > (r0 + r1))
Expand All @@ -424,24 +456,26 @@ namespace DGtal
/* no solution. one circle is contained in the other */
return 0;
}


// }
/* 'point 2' is the point where the line through the circle
* intersection points crosses the line between the circle
* centers.
*/

/* Determine the distance from point 0 to point 2. */
a = ((r0*r0) - (r1*r1) + (d*d)) / (2.0 * d) ;

/* Determine the coordinates of point 2. */
x2 = x0 + (dx * a/d);
y2 = y0 + (dy * a/d);



/* Determine the distance from point 2 to either of the
* intersection points.
*/
h = sqrt((r0*r0) - (a*a));

/* Now determine the offsets of the intersection points from
* point 2.
*/
Expand Down Expand Up @@ -487,6 +521,7 @@ namespace DGtal

int res =
circle_circle_intersection(x0,y0,r0,x1,y1,r1,xi,yi,xi_prime,yi_prime);


return res;

Expand Down Expand Up @@ -522,10 +557,13 @@ namespace DGtal
}
else
{
if(y>0)
return M_PI_2;
else
return 3*M_PI_2;
if(y==0)
return 0;
else
if(y>0)
return M_PI_2;
else
return 3*M_PI_2;
}
return -1;
}
Expand Down Expand Up @@ -797,8 +835,6 @@ namespace DGtal
ConstIterator end() const;




public:

/**
Expand Down
55 changes: 30 additions & 25 deletions src/DGtal/geometry/curves/FrechetShortcut.ih
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@
// Class backpath
////////////////////////////////////////////////////////////////

#define PRECISION 0.00001
//#define PRECISION 0.00001


//creation of a backPath
Expand Down Expand Up @@ -146,6 +146,7 @@ void DGtal::FrechetShortcut<TIterator,TInteger>::Backpath::updateOcculters()
double angle_min=0;
double angle_max=M_PI_4;
bool occ = false;
bool ok = true;

IntegerComputer<TInteger> ic;

Expand All @@ -157,16 +158,14 @@ void DGtal::FrechetShortcut<TIterator,TInteger>::Backpath::updateOcculters()
}
else
{
typename occulter_list::iterator iter, next;
bool ok = true;
typename occulter_list::iterator iter=myOcculters.begin();

iter = myOcculters.begin();
while(iter!=myOcculters.end() && ok)
for(typename occulter_list::size_type i=0; i < myOcculters.size() && ok ; ++i)
{
pi = Point(*(iter->first));
v = p-pi;

next = iter;
next++;
// pi is after p for all directions -> p is not an occulter
if(ic.dotProduct(v,u1) < 0 && ic.dotProduct(v,u2) <0)
{
Expand All @@ -178,7 +177,7 @@ void DGtal::FrechetShortcut<TIterator,TInteger>::Backpath::updateOcculters()
// anymore, p is a new occulter.
if(ic.dotProduct(v,u1) > 0 && ic.dotProduct(v,u2) > 0)
{
myOcculters.erase(iter);
iter = myOcculters.erase(iter);
occ = true;
angle_min = 0;
angle_max = M_PI_4;
Expand All @@ -198,12 +197,13 @@ void DGtal::FrechetShortcut<TIterator,TInteger>::Backpath::updateOcculters()
angle_max = alpha;
// pi's angle_min is updated
iter->second.angle_min = alpha;
iter++;
}
else
if(alpha > iter->second.angle_max)
{
//pi is not an occulter anymore
myOcculters.erase(iter);
iter = myOcculters.erase(iter);
occ=true;
angle_min = 0;
angle_max = M_PI_4;
Expand All @@ -225,21 +225,23 @@ void DGtal::FrechetShortcut<TIterator,TInteger>::Backpath::updateOcculters()
angle_max = M_PI_4;
// pi's angle_max is updated
iter->second.angle_max = alpha;
iter++;
}
else
if(alpha < iter->second.angle_min)
{
//pi is not an occulter anymore
myOcculters.erase(iter);
iter = myOcculters.erase(iter);
occ=true;
angle_min = 0;
angle_max = M_PI_4;
}
else
iter++;
// if(alpha > iter->second.angle_max), pi does not
// change, p may be an occulter -> do nothing

}
iter = next;
}
}

Expand Down Expand Up @@ -369,7 +371,7 @@ DGtal::FrechetShortcut<TIterator,TInteger>::Cone::Cone()
{
myInf = true;
myMin = 0;
myMax = 0;
myMax = 2*M_PI;
}


Expand Down Expand Up @@ -441,7 +443,8 @@ bool DGtal::FrechetShortcut<TIterator,TInteger>::Cone::isEmpty() const
if(myInf)
return false;
else
if(myMin==myMax)
// Fix 05/2024 to enable error = 0: a cone may be defined by two values myMin=myMax --> check for empty cone by setting myMin=myMax= -1 instead
if(myMin==-1) // and then myMax = -1 too: way to represent the empty intersection of two cones.
return true;
else
return false;
Expand Down Expand Up @@ -509,7 +512,7 @@ typename DGtal::FrechetShortcut<TIterator,TInteger>::Cone DGtal::FrechetShortcut
// first possibility: the cones are disjoint
if(!Tools::isBetween(myMin, c.myMin, c.myMax, 2*M_PI) && !Tools::isBetween(myMax, c.myMin,
c.myMax, 2*M_PI))
res = Cone(0,0);
res = Cone(-1,-1); // empty cone: both angles are set to -1
else
// or the new cone includes the old one, nothing changes, the cone remains the same.
res = *this;
Expand All @@ -523,6 +526,7 @@ typename DGtal::FrechetShortcut<TIterator,TInteger>::Cone DGtal::FrechetShortcut
res = Cone(c.myMin, myMax);
else
res = Cone(myMin,c.myMax);


return res;
}
Expand Down Expand Up @@ -568,6 +572,7 @@ template <typename TIterator, typename TInteger>
inline
DGtal::FrechetShortcut<TIterator,TInteger>::FrechetShortcut(double error)
{

myError = error;
myCone = Cone();

Expand Down Expand Up @@ -688,14 +693,18 @@ DGtal::FrechetShortcut<TIterator,TInteger>::computeNewCone()
Point firstP = Point(*myBegin);
Point newP = Point(*(myEnd+1));


Cone newCone=myCone;

if(firstP == newP)
return newCone;

// compute the tangent points defined by the first point and the
// circle C(newP,error)


bool intersect = Tools::circleTangentPoints(firstP[0],firstP[1], newP[0], newP[1], myError/(sqrt(2.0F)), &x0, &y0,
&x1, &y1);

if(intersect)
{
// define a cone according to the new tangent points
Expand All @@ -704,20 +713,16 @@ DGtal::FrechetShortcut<TIterator,TInteger>::computeNewCone()
if(fabs(x0-x1) < PRECISION && fabs(y0-y1) < PRECISION)
{
double angle = Tools::computeAngle(firstP[0],firstP[1],newP[0],newP[1]);
assert(angle != -1);
double angle0 = angle - M_PI_2;
if(angle0<0)
angle0 = angle0+2*M_PI;
double angle1 = angle + M_PI_2;
if(angle1>2*M_PI)
angle1 = angle1-2*M_PI;
c = Cone(angle0,angle1);

// the cone is reduced to a line
c = Cone(angle,angle);
}
else
c = Cone(firstP[0],firstP[1],x0,y0,x1,y1);

newCone.intersectCones(c);
}


return newCone;

Expand Down Expand Up @@ -873,7 +878,7 @@ inline
void DGtal::FrechetShortcut<TIterator,TInteger>::resetCone()
{
myCone.myMin = 0;
myCone.myMax = 0;
myCone.myMax = 2*M_PI; // default cone is the whole space
myCone.myInf = true;
}

Expand Down
Loading

0 comments on commit 2da53eb

Please sign in to comment.