From 1df482ba4301e9a9de64c31042cdb475f35f9fcb Mon Sep 17 00:00:00 2001
From: Florent Lamiraux <florent@laas.fr>
Date: Tue, 25 Dec 2018 11:24:17 +0100
Subject: [PATCH] [GJK] Slightly modify function getSupport for cylinders.

  - if dir [2] == 0, return a point with z coordinate equal to 0,
  - add a test between box and cylinder that did make GJK algorithm fail.
---
 src/narrowphase/gjk.cpp            | 11 ++++--
 test/test_fcl_geometric_shapes.cpp | 56 ++++++++++++++++++++++++++++++
 2 files changed, 65 insertions(+), 2 deletions(-)

diff --git a/src/narrowphase/gjk.cpp b/src/narrowphase/gjk.cpp
index 9862fdbb..e45cd21a 100644
--- a/src/narrowphase/gjk.cpp
+++ b/src/narrowphase/gjk.cpp
@@ -46,6 +46,7 @@ namespace details
 
 Vec3f getSupport(const ShapeBase* shape, const Vec3f& dir)
 {
+  FCL_REAL eps (sqrt(std::numeric_limits<FCL_REAL>::epsilon()));
   switch(shape->getNodeType())
   {
   case GEOM_TRIANGLE:
@@ -126,15 +127,21 @@ Vec3f getSupport(const ShapeBase* shape, const Vec3f& dir)
       const Cylinder* cylinder = static_cast<const Cylinder*>(shape);
       FCL_REAL zdist = std::sqrt(dir[0] * dir[0] + dir[1] * dir[1]);
       FCL_REAL half_h = cylinder->lz * 0.5;
+      Vec3f res;
       if(zdist == 0.0)
       {
-        return Vec3f(0, 0, (dir[2]>0)? half_h:-half_h);
+        res =  Vec3f(0, 0, (dir[2]>0)? half_h:-half_h);
       }
       else
       {
         FCL_REAL d = cylinder->radius / zdist;
-        return Vec3f(d * dir[0], d * dir[1], (dir[2]>0)?half_h:-half_h);
+        FCL_REAL z (0.);
+        if (dir [2] > eps) z = half_h;
+        else if (dir [2] < -eps) z = -half_h;
+        res =  Vec3f(d * dir[0], d * dir[1], z);
       }
+      assert (fabs (res [0] * dir [1] - res [1] * dir [0]) < eps);
+      return res;
     }
     break;
   case GEOM_CONVEX:
diff --git a/test/test_fcl_geometric_shapes.cpp b/test/test_fcl_geometric_shapes.cpp
index 5e05a114..808e0b22 100644
--- a/test/test_fcl_geometric_shapes.cpp
+++ b/test/test_fcl_geometric_shapes.cpp
@@ -188,6 +188,62 @@ void testShapeIntersection(const S1& s1, const Transform3f& tf1,
   }
 }
 
+BOOST_AUTO_TEST_CASE (shapeIntersection_cylinderbox)
+{
+  Cylinder s1 (0.029, 0.1);
+  Box s2 (1.6, 0.6, 0.025);
+
+  Transform3f tf1
+    (Quaternion3f (0.5279170511703305, -0.50981118132505521,
+                   -0.67596178682051911, 0.0668715876735793),
+     Vec3f (0.041218354748013122, 1.2022554710435607, 0.77338855025700015));
+
+  Transform3f tf2
+    (Quaternion3f (0.70738826916719977, 0, 0, 0.70682518110536596),
+     Vec3f (-0.29936284351096382, 0.80023864435868775, 0.71750000000000003));
+
+  GJKSolver_indep solver;
+  FCL_REAL distance;
+  Vec3f p1, p2, normal;
+  bool res = solver.shapeDistance (s1, tf1, s2, tf2, distance, p1, p2, normal);
+  BOOST_CHECK ((res && distance > 0) || (!res && distance <= 0));
+  // If objects are not colliding, p2 should be outside the cylinder and
+  // p1 should be outside the box
+  Vec3f p2Loc (inverse (tf1).transform (p2));
+  bool p2_in_cylinder ((fabs (p2Loc [2]) <= .5*s1.lz) &&
+                       (p2Loc [0] * p2Loc [0] + p2Loc [1] * p2Loc [1]
+                        <= s1.radius));
+  Vec3f p1Loc (inverse (tf2).transform (p1));
+  bool p1_in_box ((fabs (p1Loc [0]) <= .5 * s2.side [0]) &&
+                  (fabs (p1Loc [1]) <= .5 * s2.side [1]) &&
+                  (fabs (p1Loc [2]) <= .5 * s2.side [2]));
+  std::cout << "p2 in cylinder = (" << p2Loc.transpose () << ")" << std::endl;
+  std::cout << "p1 in box = (" << p1Loc.transpose () << ")" << std::endl;
+
+  BOOST_CHECK ((res && !p2_in_cylinder && !p1_in_box) ||
+               (!res && p2_in_cylinder && p1_in_box));
+
+  res = solver.shapeDistance (s2, tf2, s1, tf1, distance, p2, p1, normal);
+  BOOST_CHECK ((res && distance > 0) || (!res && distance <= 0));
+  // If objects are not colliding, p2 should be outside the cylinder and
+  // p1 should be outside the box
+
+  p2Loc = inverse (tf1).transform (p2);
+  p2_in_cylinder = (fabs (p2Loc [2]) <= .5*s1.lz) &&
+    (p2Loc [0] * p2Loc [0] + p2Loc [1] * p2Loc [1]
+     <= s1.radius);
+  p1Loc = inverse (tf2).transform (p1);
+  p1_in_box = (fabs (p1Loc [0]) <= .5 * s2.side [0]) &&
+    (fabs (p1Loc [1]) <= .5 * s2.side [1]) &&
+    (fabs (p1Loc [2]) <= .5 * s2.side [2]);
+
+  std::cout << "p2 in cylinder = (" << p2Loc.transpose () << ")" << std::endl;
+  std::cout << "p1 in box = (" << p1.transpose () << ")" << std::endl;
+
+  BOOST_CHECK ((res && !p2_in_cylinder && !p1_in_box) ||
+               (!res && p2_in_cylinder && p1_in_box));
+}
+
 BOOST_AUTO_TEST_CASE(shapeIntersection_spheresphere)
 {
   Sphere s1(20);
-- 
GitLab