diff --git a/trunk/fcl/include/fcl/collision.h b/trunk/fcl/include/fcl/collision.h
index d9f59f8d9b44209d4ec8ab66c03834a08d29d302..41413ff0aba6b619ba8d14c9290b1f6b935612c8 100644
--- a/trunk/fcl/include/fcl/collision.h
+++ b/trunk/fcl/include/fcl/collision.h
@@ -49,7 +49,7 @@ namespace fcl
 /** \brief Main collision interface: given two collision objects, and the requirements for contacts, including num of max contacts, whether perform exhaustive collision (i.e., returning 
  * returning all the contact points), whether return detailed contact information (i.e., normal, contact point, depth; otherwise only contact primitive id is returned), this function
  * performs the collision between them. 
- * Return value is the number of contacts returned 
+ * Return value is the number of contacts generated between the two objects.
  */
 
 template<typename NarrowPhaseSolver>
diff --git a/trunk/fcl/include/fcl/collision_data.h b/trunk/fcl/include/fcl/collision_data.h
index f42f76dee79907efce70e41aeb2653d97077265b..f72af25eab93362f95983b567ade2a0bfa5877b4 100644
--- a/trunk/fcl/include/fcl/collision_data.h
+++ b/trunk/fcl/include/fcl/collision_data.h
@@ -103,9 +103,10 @@ struct CollisionRequest
 
 struct CollisionResult
 {
+private:
   std::vector<Contact> contacts;
   std::vector<CostSource> cost_sources;
-
+public:
   CollisionResult()
   {
   }
@@ -135,6 +136,34 @@ struct CollisionResult
     return cost_sources.size();
   }
 
+  const Contact& getContact(size_t i) const
+  {
+    if(i < contacts.size()) 
+      return contacts[i];
+    else
+      return contacts.back();
+  }
+
+  const CostSource& getCostSource(size_t i) const
+  {
+    if(i < cost_sources.size())
+      return cost_sources[i];
+    else
+      return cost_sources.back();
+  }
+
+  void getContacts(std::vector<Contact>& contacts_)
+  {
+    contacts_.resize(contacts.size());
+    std::copy(contacts.begin(), contacts.end(), contacts_.begin());
+  }
+
+  void getCostSources(std::vector<CostSource>& cost_sources_)
+  {
+    cost_sources_.resize(cost_sources.size());
+    std::copy(cost_sources.begin(), cost_sources.end(), cost_sources_.begin());
+  }
+
   void clear()
   {
     contacts.clear();
@@ -159,24 +188,81 @@ struct CollisionData
 struct DistanceRequest
 {
   bool enable_nearest_points;
-  DistanceRequest() : enable_nearest_points(false)
+
+  DistanceRequest(bool enable_nearest_points_ = false) : enable_nearest_points(enable_nearest_points_)
   {
   }
 };
 
 struct DistanceResult
 {
+
   FCL_REAL min_distance;
 
   Vec3f nearest_points[2];
+
+  const CollisionGeometry* o1;
+  const CollisionGeometry* o2;
+  int b1;
+  int b2;
+
+  static const int NONE = -1;
   
-  DistanceResult() : min_distance(std::numeric_limits<FCL_REAL>::max())
+  DistanceResult(FCL_REAL min_distance_ = std::numeric_limits<FCL_REAL>::max()) : min_distance(min_distance_), 
+                                                                                  o1(NULL),
+                                                                                  o2(NULL),
+                                                                                  b1(-1),
+                                                                                  b2(-1)
   {
   }
 
+  void update(FCL_REAL distance, const CollisionGeometry* o1_, const CollisionGeometry* o2_, int b1_, int b2_)
+  {
+    if(min_distance > distance)
+    {
+      min_distance = distance;
+      o1 = o1_;
+      o2 = o2_;
+      b1 = b1_;
+      b2 = b2_;
+    }
+  }
+
+  void update(FCL_REAL distance, const CollisionGeometry* o1_, const CollisionGeometry* o2_, int b1_, int b2_, const Vec3f& p1, const Vec3f& p2)
+  {
+    if(min_distance > distance)
+    {
+      min_distance = distance;
+      o1 = o1_;
+      o2 = o2_;
+      b1 = b1_;
+      b2 = b2_;
+      nearest_points[0] = p1;
+      nearest_points[1] = p2;
+    }
+  }
+
+  void update(const DistanceResult& other_result)
+  {
+    if(min_distance > other_result.min_distance)
+    {
+      min_distance = other_result.min_distance;
+      o1 = other_result.o1;
+      o2 = other_result.o2;
+      b1 = other_result.b1;
+      b2 = other_result.b2;
+      nearest_points[0] = other_result.nearest_points[0];
+      nearest_points[1] = other_result.nearest_points[1];
+    }
+  }
+
   void clear()
   {
     min_distance = std::numeric_limits<FCL_REAL>::max();
+    o1 = NULL;
+    o2 = NULL;
+    b1 = -1;
+    b2 = -1;
   }
 };
 
diff --git a/trunk/fcl/include/fcl/simple_setup.h b/trunk/fcl/include/fcl/simple_setup.h
index 626ac30208a407d2f4f012ba4bdfb78fe89c84e2..d94045adcccf5d41f8f7ec9c21189da7ceaf9fdd 100644
--- a/trunk/fcl/include/fcl/simple_setup.h
+++ b/trunk/fcl/include/fcl/simple_setup.h
@@ -75,9 +75,11 @@ bool initialize(OcTreeDistanceTraversalNode<NarrowPhaseSolver>& node,
                 const OcTree& model1, const SimpleTransform& tf1,
                 const OcTree& model2, const SimpleTransform& tf2,
                 const OcTreeSolver<NarrowPhaseSolver>* otsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
   node.request = request;
+  node.result = &result;
  
   node.model1 = &model1;
   node.model2 = &model2;
@@ -139,9 +141,11 @@ bool initialize(ShapeOcTreeDistanceTraversalNode<S, NarrowPhaseSolver>& node,
                 const S& model1, const SimpleTransform& tf1,
                 const OcTree& model2, const SimpleTransform& tf2,
                 const OcTreeSolver<NarrowPhaseSolver>* otsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.model2 = &model2;
@@ -160,9 +164,11 @@ bool initialize(OcTreeShapeDistanceTraversalNode<S, NarrowPhaseSolver>& node,
                 const OcTree& model1, const SimpleTransform& tf1,
                 const S& model2, const SimpleTransform& tf2,
                 const OcTreeSolver<NarrowPhaseSolver>* otsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.model2 = &model2;
@@ -224,9 +230,11 @@ bool initialize(MeshOcTreeDistanceTraversalNode<BV, NarrowPhaseSolver>& node,
                 const BVHModel<BV>& model1, const SimpleTransform& tf1,
                 const OcTree& model2, const SimpleTransform& tf2,
                 const OcTreeSolver<NarrowPhaseSolver>* otsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.model2 = &model2;
@@ -245,9 +253,11 @@ bool initialize(OcTreeMeshDistanceTraversalNode<BV, NarrowPhaseSolver>& node,
                 const OcTree& model1, const SimpleTransform& tf1,
                 const BVHModel<BV>& model2, const SimpleTransform& tf2,
                 const OcTreeSolver<NarrowPhaseSolver>* otsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.model2 = &model2;
@@ -823,9 +833,11 @@ bool initialize(ShapeDistanceTraversalNode<S1, S2, NarrowPhaseSolver>& node,
                 const S1& shape1, const SimpleTransform& tf1,
                 const S2& shape2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
   node.request = request;
+  node.result = &result;
 
   node.model1 = &shape1;
   node.tf1 = tf1;
@@ -842,6 +854,7 @@ bool initialize(MeshDistanceTraversalNode<BV>& node,
                 BVHModel<BV>& model1, SimpleTransform& tf1,
                 BVHModel<BV>& model2, SimpleTransform& tf2,
                 const DistanceRequest& request,
+                DistanceResult& result,
                 bool use_refit = false, bool refit_bottomup = false)
 {
   if(model1.getModelType() != BVH_MODEL_TRIANGLES || model2.getModelType() != BVH_MODEL_TRIANGLES)
@@ -882,6 +895,7 @@ bool initialize(MeshDistanceTraversalNode<BV>& node,
   }
 
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.tf1 = tf1;
@@ -902,18 +916,21 @@ bool initialize(MeshDistanceTraversalNode<BV>& node,
 bool initialize(MeshDistanceTraversalNodeRSS& node,
                 const BVHModel<RSS>& model1, const SimpleTransform& tf1,
                 const BVHModel<RSS>& model2, const SimpleTransform& tf2,
-                const DistanceRequest& request);
+                const DistanceRequest& request,
+                DistanceResult& result);
 
 
 bool initialize(MeshDistanceTraversalNodekIOS& node,
                 const BVHModel<kIOS>& model1, const SimpleTransform& tf1,
                 const BVHModel<kIOS>& model2, const SimpleTransform& tf2,
-                const DistanceRequest& request);
+                const DistanceRequest& request,
+                DistanceResult& result);
 
 bool initialize(MeshDistanceTraversalNodeOBBRSS& node,
                 const BVHModel<OBBRSS>& model1, const SimpleTransform& tf1,
                 const BVHModel<OBBRSS>& model2, const SimpleTransform& tf2,
-                const DistanceRequest& request);
+                const DistanceRequest& request,
+                DistanceResult& result);
 
 
 template<typename BV, typename S, typename NarrowPhaseSolver>
@@ -922,6 +939,7 @@ bool initialize(MeshShapeDistanceTraversalNode<BV, S, NarrowPhaseSolver>& node,
                 const S& model2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
                 const DistanceRequest& request,
+                DistanceResult& result,
                 bool use_refit = false, bool refit_bottomup = false)
 {
   if(model1.getModelType() != BVH_MODEL_TRIANGLES)
@@ -945,6 +963,7 @@ bool initialize(MeshShapeDistanceTraversalNode<BV, S, NarrowPhaseSolver>& node,
   }
 
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.tf1 = tf1;
@@ -967,6 +986,7 @@ bool initialize(ShapeMeshDistanceTraversalNode<S, BV, NarrowPhaseSolver>& node,
                 BVHModel<BV>& model2, SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
                 const DistanceRequest& request,
+                DistanceResult& result,
                 bool use_refit = false, bool refit_bottomup = false)
 {
   if(model2.getModelType() != BVH_MODEL_TRIANGLES)
@@ -990,6 +1010,7 @@ bool initialize(ShapeMeshDistanceTraversalNode<S, BV, NarrowPhaseSolver>& node,
   }
 
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.tf1 = tf1;
@@ -1014,12 +1035,14 @@ static inline bool setupMeshShapeDistanceOrientedNode(OrientedNode<S, NarrowPhas
                                                       const BVHModel<BV>& model1, const SimpleTransform& tf1,
                                                       const S& model2, const SimpleTransform& tf2,
                                                       const NarrowPhaseSolver* nsolver,
-                                                      const DistanceRequest& request)
+                                                      const DistanceRequest& request,
+                                                      DistanceResult& result)
 {
   if(model1.getModelType() != BVH_MODEL_TRIANGLES)
     return false;
 
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.tf1 = tf1;
@@ -1042,9 +1065,10 @@ bool initialize(MeshShapeDistanceTraversalNodeRSS<S, NarrowPhaseSolver>& node,
                 const BVHModel<RSS>& model1, const SimpleTransform& tf1,
                 const S& model2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request, 
+                DistanceResult& result)
 {
-  return details::setupMeshShapeDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request);
+  return details::setupMeshShapeDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request, result);
 }
 
 template<typename S, typename NarrowPhaseSolver>
@@ -1052,9 +1076,10 @@ bool initialize(MeshShapeDistanceTraversalNodekIOS<S, NarrowPhaseSolver>& node,
                 const BVHModel<kIOS>& model1, const SimpleTransform& tf1,
                 const S& model2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupMeshShapeDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request);  
+  return details::setupMeshShapeDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request, result);  
 }
 
 template<typename S, typename NarrowPhaseSolver>
@@ -1062,9 +1087,10 @@ bool initialize(MeshShapeDistanceTraversalNodeOBBRSS<S, NarrowPhaseSolver>& node
                 const BVHModel<OBBRSS>& model1, const SimpleTransform& tf1,
                 const S& model2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupMeshShapeDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request);
+  return details::setupMeshShapeDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request, result);
 }
 
 
@@ -1075,12 +1101,14 @@ static inline bool setupShapeMeshDistanceOrientedNode(OrientedNode<S, NarrowPhas
                                                       const S& model1, const SimpleTransform& tf1,
                                                       const BVHModel<BV>& model2, const SimpleTransform& tf2,
                                                       const NarrowPhaseSolver* nsolver,
-                                                      const DistanceRequest& request)
+                                                      const DistanceRequest& request,
+                                                      DistanceResult& result)
 {
   if(model2.getModelType() != BVH_MODEL_TRIANGLES)
     return false;
 
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.tf1 = tf1;
@@ -1106,9 +1134,10 @@ bool initialize(ShapeMeshDistanceTraversalNodeRSS<S, NarrowPhaseSolver>& node,
                 const S& model1, const SimpleTransform& tf1,
                 const BVHModel<RSS>& model2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupShapeMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request);
+  return details::setupShapeMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request, result);
 }
 
 template<typename S, typename NarrowPhaseSolver>
@@ -1116,9 +1145,10 @@ bool initialize(ShapeMeshDistanceTraversalNodekIOS<S, NarrowPhaseSolver>& node,
                 const S& model1, const SimpleTransform& tf1,
                 const BVHModel<kIOS>& model2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupShapeMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request);
+  return details::setupShapeMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request, result);
 }
 
 template<typename S, typename NarrowPhaseSolver>
@@ -1126,9 +1156,10 @@ bool initialize(ShapeMeshDistanceTraversalNodeOBBRSS<S, NarrowPhaseSolver>& node
                 const S& model1, const SimpleTransform& tf1,
                 const BVHModel<OBBRSS>& model2, const SimpleTransform& tf2,
                 const NarrowPhaseSolver* nsolver,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupShapeMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request);
+  return details::setupShapeMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, nsolver, request, result);
 }
 
 
diff --git a/trunk/fcl/include/fcl/traversal_node_bvh_shape.h b/trunk/fcl/include/fcl/traversal_node_bvh_shape.h
index d91c6baf712523b4b7025046be7f6475c2fbc87f..b812b6e6ae4fcf8a1de644c23140b82216d49013 100644
--- a/trunk/fcl/include/fcl/traversal_node_bvh_shape.h
+++ b/trunk/fcl/include/fcl/traversal_node_bvh_shape.h
@@ -179,7 +179,7 @@ public:
       if(nsolver->shapeTriangleIntersect(*(this->model2), this->tf2, p1, p2, p3, NULL, NULL, NULL))
       {
         is_intersect = true;
-        this->result->contacts.push_back(Contact(this->model1, this->model2, primitive_id, Contact::NONE));
+        this->result->addContact(Contact(this->model1, this->model2, primitive_id, Contact::NONE));
       }
     }
     else
@@ -187,24 +187,24 @@ public:
       if(nsolver->shapeTriangleIntersect(*(this->model2), this->tf2, p1, p2, p3, &contactp, &penetration, &normal))
       {
         is_intersect = true;
-        this->result->contacts.push_back(Contact(this->model1, this->model2, primitive_id, Contact::NONE, contactp, -normal, penetration));
+        this->result->addContact(Contact(this->model1, this->model2, primitive_id, Contact::NONE, contactp, -normal, penetration));
       }
     }
 
-    if(is_intersect && this->request.enable_cost && (this->request.num_max_cost_sources > this->result->cost_sources.size()))
+    if(is_intersect && this->request.enable_cost && (this->request.num_max_cost_sources > this->result->numCostSources()))
     {
       AABB overlap_part;
       AABB shape_aabb;
       computeBV<AABB, S>(*(this->model2), this->tf2, shape_aabb);
       AABB(p1, p2, p3).overlap(shape_aabb, overlap_part);
-      this->result->cost_sources.push_back(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
+      this->result->addCostSource(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
     }
   }
 
   bool canStop() const
   {
-    return (this->result->contacts.size() > 0) && (!this->request.exhaustive) && (this->request.num_max_contacts <= this->result->contacts.size()) && (this->request.num_max_cost_sources <= this->result->cost_sources.size()) &&
-      (  (!this->request.enable_cost) || (this->request.num_max_cost_sources <= this->result->cost_sources.size())  );
+    return (this->result->numContacts() > 0) && (!this->request.exhaustive) && (this->request.num_max_contacts <= this->result->numContacts()) && (this->request.num_max_cost_sources <= this->result->numCostSources()) &&
+      (  (!this->request.enable_cost) || (this->request.num_max_cost_sources <= this->result->numCostSources())  );
   }
 
   Vec3f* vertices;
@@ -253,7 +253,7 @@ static inline void meshShapeCollisionOrientedNodeLeafTesting(int b1, int b2,
     if(nsolver->shapeTriangleIntersect(model2, tf2, p1, p2, p3, tf1, NULL, NULL, NULL))
     {
       is_intersect = true;
-      result.contacts.push_back(Contact(model1, &model2, primitive_id, Contact::NONE));
+      result.addContact(Contact(model1, &model2, primitive_id, Contact::NONE));
     }
   }
   else
@@ -261,17 +261,17 @@ static inline void meshShapeCollisionOrientedNodeLeafTesting(int b1, int b2,
     if(nsolver->shapeTriangleIntersect(model2, tf2, p1, p2, p3, tf1, &contactp, &penetration, &normal))
     {
       is_intersect = true;
-      result.contacts.push_back(Contact(model1, &model2, primitive_id, Contact::NONE, contactp, -normal, penetration));
+      result.addContact(Contact(model1, &model2, primitive_id, Contact::NONE, contactp, -normal, penetration));
     }
   }
 
-  if(is_intersect && request.enable_cost && (request.num_max_cost_sources > result.cost_sources.size()))
+  if(is_intersect && request.enable_cost && (request.num_max_cost_sources > result.numCostSources()))
   {
     AABB overlap_part;
     AABB shape_aabb;
     computeBV<AABB, S>(model2, tf2, shape_aabb);
     AABB(tf1.transform(p1), tf1.transform(p2), tf1.transform(p2)).overlap(shape_aabb, overlap_part);
-    result.cost_sources.push_back(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
+    result.addCostSource(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
   }
 }
 
@@ -401,7 +401,7 @@ public:
       if(nsolver->shapeTriangleIntersect(*(this->model1), this->tf1, p1, p2, p3, NULL, NULL, NULL))
       {
         is_intersect = true;
-        this->result->contacts.push_back(Contact(this->model1, this->model2, Contact::NONE, primitive_id));
+        this->result->addContact(Contact(this->model1, this->model2, Contact::NONE, primitive_id));
       }
     }
     else
@@ -409,24 +409,24 @@ public:
       if(nsolver->shapeTriangleIntersect(*(this->model1), this->tf1, p1, p2, p3, &contactp, &penetration, &normal))
       {
         is_intersect = true;
-        this->result->contacts.push_back(Contact(this->model1, this->model2, Contact::NONE, primitive_id, contactp, normal, penetration));
+        this->result->addContact(Contact(this->model1, this->model2, Contact::NONE, primitive_id, contactp, normal, penetration));
       }
     }
 
-    if(is_intersect && this->request.enable_cost && (this->request.num_max_cost_sources > this->result->cost_sources.size()))
+    if(is_intersect && this->request.enable_cost && (this->request.num_max_cost_sources > this->result->numCostSources()))
     {
       AABB overlap_part;
       AABB shape_aabb;
       computeBV<AABB, S>(*(this->model1), this->tf1, shape_aabb);
       AABB(p1, p2, p3).overlap(shape_aabb, overlap_part);
-      this->result->cost_sources.push_back(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
+      this->result->addCostSource(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
     }
   }
 
   bool canStop() const
   {
-    return (this->result->contacts.size() > 0) && (!this->request.exhaustive) && (this->request.num_max_contacts <= this->result->contacts.size()) &&
-      (  (!this->request.enable_cost) || (this->request.num_max_cost_sources <= this->result->cost_sources.size())  );
+    return (this->result->numContacts() > 0) && (!this->request.exhaustive) && (this->request.num_max_contacts <= this->result->numContacts()) &&
+      (  (!this->request.enable_cost) || (this->request.num_max_cost_sources <= this->result->numCostSources())  );
   }
 
   Vec3f* vertices;
@@ -633,13 +633,9 @@ public:
     vertices = NULL;
     tri_indices = NULL;
 
-    last_tri_id = 0;
-
     rel_err = 0;
     abs_err = 0;
 
-    min_distance = std::numeric_limits<FCL_REAL>::max();
-
     nsolver = NULL;
   }
 
@@ -660,17 +656,12 @@ public:
     FCL_REAL d;
     nsolver->shapeTriangleDistance(*(this->model2), this->tf2, p1, p2, p3, &d);
 
-    if(d < min_distance)
-    {
-      min_distance = d;
-      
-      last_tri_id = primitive_id;
-    }
+    this->result->update(d, this->model1, this->model2, primitive_id, DistanceResult::NONE);
   }
 
   bool canStop(FCL_REAL c) const
   {
-    if((c >= min_distance - abs_err) && (c * (1 + rel_err) >= min_distance))
+    if((c >= this->result->min_distance - abs_err) && (c * (1 + rel_err) >= this->result->min_distance))
       return true;
     return false;
   }
@@ -680,10 +671,7 @@ public:
 
   FCL_REAL rel_err;
   FCL_REAL abs_err;
-  
-  mutable FCL_REAL min_distance;
-  mutable int last_tri_id;
-  
+    
   const NarrowPhaseSolver* nsolver;
 };
 
@@ -699,8 +687,8 @@ void meshShapeDistanceOrientedNodeLeafTesting(int b1, int b2,
                                               const NarrowPhaseSolver* nsolver,
                                               bool enable_statistics,
                                               int & num_leaf_tests,
-                                              int& last_tri_id,
-                                              FCL_REAL& min_distance)
+                                              const DistanceRequest& request,
+                                              DistanceResult& result)
 {
   if(enable_statistics) num_leaf_tests++;
     
@@ -712,51 +700,38 @@ void meshShapeDistanceOrientedNodeLeafTesting(int b1, int b2,
   const Vec3f& p2 = vertices[tri_id[1]];
   const Vec3f& p3 = vertices[tri_id[2]];
     
-  FCL_REAL dist;
-
-  nsolver->shapeTriangleDistance(model2, tf2, p1, p2, p3, tf1, &dist);
-    
-  if(dist < min_distance)
-  {
-    min_distance = dist;
-
-    last_tri_id = primitive_id;
-  }
-}
+  FCL_REAL distance;
+  nsolver->shapeTriangleDistance(model2, tf2, p1, p2, p3, tf1, &distance);
 
+  result.update(distance, model1, &model2, primitive_id, DistanceResult::NONE);
 }
 
-
-template<typename S, typename NarrowPhaseSolver>
-static inline void distance_preprocess(Vec3f* vertices, Triangle* tri_indices, int last_tri_id,
-                                       const S& s, const SimpleTransform& tf1, const SimpleTransform& tf2,
-                                       const NarrowPhaseSolver* nsolver,
-                                       FCL_REAL& min_distance)
+template<typename BV, typename S, typename NarrowPhaseSolver>
+static inline void distancePreprocessOrientedNode(const BVHModel<BV>* model1,
+                                                  Vec3f* vertices, Triangle* tri_indices, int init_tri_id,
+                                                  const S& model2, const SimpleTransform& tf1, const SimpleTransform& tf2,
+                                                  const NarrowPhaseSolver* nsolver,
+                                                  const DistanceRequest& request,
+                                                  DistanceResult& result)
 {
-  const Triangle& last_tri = tri_indices[last_tri_id];
-  
-  const Vec3f& p1 = vertices[last_tri[0]];
-  const Vec3f& p2 = vertices[last_tri[1]];
-  const Vec3f& p3 = vertices[last_tri[2]];
-
-  Vec3f last_tri_P, last_tri_Q;
+  const Triangle& init_tri = tri_indices[init_tri_id];
   
-  FCL_REAL dist;
-  nsolver->shapeTriangleDistance(s, tf2, p1, p2, p3, tf1, &dist);
+  const Vec3f& p1 = vertices[init_tri[0]];
+  const Vec3f& p2 = vertices[init_tri[1]];
+  const Vec3f& p3 = vertices[init_tri[2]];
   
-  min_distance = dist;
+  FCL_REAL distance;
+  nsolver->shapeTriangleDistance(model2, tf2, p1, p2, p3, tf1, &distance);
+
+  result.update(distance, model1, &model2, init_tri_id, DistanceResult::NONE);
 }
 
 
-static inline void distance_postprocess(const SimpleTransform& tf, Vec3f& p2)
-{
-  const SimpleQuaternion& inv_q = conj(tf.getQuatRotation());
-  p2 = inv_q.transform(p2 - tf.getTranslation());
-  // Vec3f u = p2 - T;
-  // p2 = R.transposeTimes(u);
 }
 
 
+
+
 template<typename S, typename NarrowPhaseSolver>
 class MeshShapeDistanceTraversalNodeRSS : public MeshShapeDistanceTraversalNode<RSS, S, NarrowPhaseSolver>
 {
@@ -767,12 +742,12 @@ public:
 
   void preprocess()
   {
-    distance_preprocess(this->vertices, this->tri_indices, this->last_tri_id, *(this->model2), this->tf1, this->tf2, this->nsolver, this->min_distance);
+    details::distancePreprocessOrientedNode(this->model1, this->vertices, this->tri_indices, 0, 
+                                            *(this->model2), this->tf1, this->tf2, this->nsolver, this->request, *(this->result));
   }
 
   void postprocess()
   {
-    
   }
 
   FCL_REAL BVTesting(int b1, int b2) const
@@ -784,7 +759,7 @@ public:
   void leafTesting(int b1, int b2) const
   {
     details::meshShapeDistanceOrientedNodeLeafTesting(b1, b2, this->model1, *(this->model2), this->vertices, this->tri_indices,
-                                                      this->tf1, this->tf2, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->last_tri_id, this->min_distance);
+                                                      this->tf1, this->tf2, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->request, *(this->result));
   }
 };
 
@@ -799,12 +774,12 @@ public:
 
   void preprocess()
   {
-    distance_preprocess(this->vertices, this->tri_indices, this->last_tri_id, *(this->model2), this->tf1, this->tf2, this->nsolver, this->min_distance);
+    details::distancePreprocessOrientedNode(this->model1, this->vertices, this->tri_indices, 0, 
+                                            *(this->model2), this->tf1, this->tf2, this->nsolver, this->request, *(this->result));
   }
 
   void postprocess()
-  {
-    
+  {    
   }
 
   FCL_REAL BVTesting(int b1, int b2) const
@@ -816,7 +791,7 @@ public:
   void leafTesting(int b1, int b2) const
   {
     details::meshShapeDistanceOrientedNodeLeafTesting(b1, b2, this->model1, *(this->model2), this->vertices, this->tri_indices,
-                                                      this->tf1, this->tf2, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->last_tri_id, this->min_distance);
+                                                      this->tf1, this->tf2, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->request, *(this->result));
   }
 
 };
@@ -831,7 +806,8 @@ public:
 
   void preprocess()
   {
-    distance_preprocess(this->vertices, this->tri_indices, this->last_tri_id, *(this->model2), this->tf1, this->tf2, this->nsolver, this->min_distance);
+    details::distancePreprocessOrientedNode(this->model1, this->vertices, this->tri_indices, 0, 
+                                            *(this->model2), this->tf1, this->tf2, this->nsolver, this->request, *(this->result));
   }
 
   void postprocess()
@@ -848,7 +824,7 @@ public:
   void leafTesting(int b1, int b2) const
   {
     details::meshShapeDistanceOrientedNodeLeafTesting(b1, b2, this->model1, *(this->model2), this->vertices, this->tri_indices,
-                                                      this->tf1, this->tf2, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->last_tri_id, this->min_distance);
+                                                      this->tf1, this->tf2, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->request, *(this->result));
   }
   
 };
@@ -863,13 +839,9 @@ public:
     vertices = NULL;
     tri_indices = NULL;
 
-    last_tri_id = 0;
-
     rel_err = 0;
     abs_err = 0;
 
-    min_distance = std::numeric_limits<FCL_REAL>::max();
-
     nsolver = NULL;
   }
 
@@ -887,20 +859,15 @@ public:
     const Vec3f& p2 = vertices[tri_id[1]];
     const Vec3f& p3 = vertices[tri_id[2]];
     
-    FCL_REAL d;
-    nsolver->shapeTriangleDistance(*(this->model1), this->tf1, p1, p2, p3, &d);
+    FCL_REAL distance;
+    nsolver->shapeTriangleDistance(*(this->model1), this->tf1, p1, p2, p3, &distance);
 
-    if(d < min_distance)
-    {
-      min_distance = d;
-      
-      last_tri_id = primitive_id;
-    }
+    this->result->update(distance, this->model1, this->model2, DistanceResult::NONE, primitive_id);
   }
 
   bool canStop(FCL_REAL c) const
   {
-    if((c >= min_distance - abs_err) && (c * (1 + rel_err) >= min_distance))
+    if((c >= this->result->min_distance - abs_err) && (c * (1 + rel_err) >= this->result->min_distance))
       return true;
     return false;
   }
@@ -910,10 +877,7 @@ public:
 
   FCL_REAL rel_err;
   FCL_REAL abs_err;
-  
-  mutable FCL_REAL min_distance;
-  mutable int last_tri_id;
-  
+    
   const NarrowPhaseSolver* nsolver;
 };
 
@@ -927,12 +891,12 @@ public:
 
   void preprocess()
   {
-    distance_preprocess(this->vertices, this->tri_indices, this->last_tri_id, *(this->model1), this->tf2, this->tf1, this->nsolver, this->min_distance);
+    details::distancePreprocessOrientedNode(this->model2, this->vertices, this->tri_indices, 0,
+                                            *(this->model1), this->tf2, this->tf1, this->nsolver, this->request, *(this->result));
   }
 
   void postprocess()
   {
-    
   }
 
   FCL_REAL BVTesting(int b1, int b2) const
@@ -944,7 +908,7 @@ public:
   void leafTesting(int b1, int b2) const
   {
     details::meshShapeDistanceOrientedNodeLeafTesting(b2, b1, this->model2, *(this->model1), this->vertices, this->tri_indices,
-                                                      this->tf2, this->tf1, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->last_tri_id, this->min_distance);
+                                                      this->tf2, this->tf1, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->request, *(this->result));
   }
 
 };
@@ -959,12 +923,12 @@ public:
 
   void preprocess()
   {
-    distance_preprocess(this->vertices, this->tri_indices, this->last_tri_id, *(this->model1), this->tf2, this->tf1, this->nsolver, this->min_distance);
+    details::distancePreprocessOrientedNode(this->model2, this->vertices, this->tri_indices, 0,
+                                            *(this->model1), this->tf2, this->tf1, this->nsolver, *(this->result));
   }
 
   void postprocess()
   {
-    
   }
 
   FCL_REAL BVTesting(int b1, int b2) const
@@ -976,7 +940,7 @@ public:
   void leafTesting(int b1, int b2) const
   {
     details::meshShapeDistanceOrientedNodeLeafTesting(b2, b1, this->model2, *(this->model1), this->vertices, this->tri_indices,
-                                                      this->tf2, this->tf1, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->last_tri_id, this->min_distance);
+                                                      this->tf2, this->tf1, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->request, *(this->result));
   }
   
 };
@@ -991,12 +955,12 @@ public:
 
   void preprocess()
   {
-    distance_preprocess(this->vertices, this->tri_indices, this->last_tri_id, *(this->model1), this->tf2, this->tf1, this->nsolver, this->min_distance);
+    details::distancePreprocessOrientedNode(this->model2, this->vertices, this->tri_indices, 0,
+                                            *(this->model1), this->tf2, this->tf1, this->nsolver, *(this->result));
   }
 
   void postprocess()
-  {
-    
+  {    
   }
 
   FCL_REAL BVTesting(int b1, int b2) const
@@ -1008,7 +972,7 @@ public:
   void leafTesting(int b1, int b2) const
   {
     details::meshShapeDistanceOrientedNodeLeafTesting(b2, b1, this->model2, *(this->model1), this->vertices, this->tri_indices,
-                                                      this->tf2, this->tf1, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->last_tri_id, this->min_distance);
+                                                      this->tf2, this->tf1, this->nsolver, this->enable_statistics, this->num_leaf_tests, this->request, *(this->result));
   }
   
 };
diff --git a/trunk/fcl/include/fcl/traversal_node_bvhs.h b/trunk/fcl/include/fcl/traversal_node_bvhs.h
index cb29948de1adcd7b54af64fce0dc77885fba8a36..f0bd374cd16007d6bd3f43f3ee9294b554bda708 100644
--- a/trunk/fcl/include/fcl/traversal_node_bvhs.h
+++ b/trunk/fcl/include/fcl/traversal_node_bvhs.h
@@ -187,7 +187,7 @@ public:
       if(Intersect::intersect_Triangle(p1, p2, p3, q1, q2, q3))
       {
         is_intersect = true;
-        this->result->contacts.push_back(Contact(this->model1, this->model2, primitive_id1, primitive_id2));
+        this->result->addContact(Contact(this->model1, this->model2, primitive_id1, primitive_id2));
       }
     }
     else // need compute the contact information
@@ -201,31 +201,31 @@ public:
         is_intersect = true;
         if(!this->request.exhaustive)
         {
-          if(this->request.num_max_contacts < n_contacts + this->result->contacts.size())
-            n_contacts = (this->request.num_max_contacts >= this->result->contacts.size()) ? (this->request.num_max_contacts - this->result->contacts.size()) : 0;
+          if(this->request.num_max_contacts < n_contacts + this->result->numContacts())
+            n_contacts = (this->request.num_max_contacts >= this->result->numContacts()) ? (this->request.num_max_contacts - this->result->numContacts()) : 0;
         }
     
         for(unsigned int i = 0; i < n_contacts; ++i)
         {
-          this->result->contacts.push_back(Contact(this->model1, this->model2, primitive_id1, primitive_id2, contacts[i], normal, penetration));
+          this->result->addContact(Contact(this->model1, this->model2, primitive_id1, primitive_id2, contacts[i], normal, penetration));
         }
       }
     }
 
    
-    if(is_intersect && this->request.enable_cost && (this->request.num_max_cost_sources > this->result->cost_sources.size()))
+    if(is_intersect && this->request.enable_cost && (this->request.num_max_cost_sources > this->result->numCostSources()))
     {
       AABB overlap_part;
       AABB(p1, p2, p3).overlap(AABB(q1, q2, q3), overlap_part);
-      this->result->cost_sources.push_back(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
+      this->result->addCostSource(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
     }   
   }
 
   /** \brief Whether the traversal process can stop early */
   bool canStop() const
   {
-    return (this->result->contacts.size() > 0) && (!this->request.exhaustive) && (this->request.num_max_contacts <= this->result->contacts.size()) && 
-      (  (!this->request.enable_cost) || (this->request.num_max_cost_sources <= this->result->cost_sources.size())  );
+    return (this->result->numContacts() > 0) && (!this->request.exhaustive) && (this->request.num_max_contacts <= this->result->numContacts()) && 
+      (  (!this->request.enable_cost) || (this->request.num_max_cost_sources <= this->result->numCostSources())  );
   }
 
   Vec3f* vertices1;
@@ -913,13 +913,8 @@ public:
     tri_indices1 = NULL;
     tri_indices2 = NULL;
 
-    last_tri_id1 = 0;
-    last_tri_id2 = 0;
-
     rel_err = 0;
     abs_err = 0;
-
-    min_distance = std::numeric_limits<FCL_REAL>::max();
   }
 
   void leafTesting(int b1, int b2) const
@@ -949,21 +944,19 @@ public:
     FCL_REAL d = TriangleDistance::triDistance(t11, t12, t13, t21, t22, t23,
                                                P1, P2);
 
-    if(d < min_distance)
+    if(this->request.enable_nearest_points)
     {
-      min_distance = d;
-
-      p1 = P1;
-      p2 = P2;
-
-      last_tri_id1 = primitive_id1;
-      last_tri_id2 = primitive_id2;
+      this->result->update(d, this->model1, this->model2, primitive_id1, primitive_id2, P1, P2);
     }
-  }
+    else
+    {
+      this->result->update(d, this->model1, this->model2, primitive_id1, primitive_id2);
+    }
+ }
 
   bool canStop(FCL_REAL c) const
   {
-    if((c >= min_distance - abs_err) && (c * (1 + rel_err) >= min_distance))
+    if((c >= this->result->min_distance - abs_err) && (c * (1 + rel_err) >= this->result->min_distance))
       return true;
     return false;
   }
@@ -977,17 +970,6 @@ public:
   /** \brief relative and absolute error, default value is 0.01 for both terms */
   FCL_REAL rel_err;
   FCL_REAL abs_err;
-
-  /** \brief distance and points establishing the minimum distance for the models, within the relative and absolute error bounds specified.
-   * p1 is in model1's local coordinate system while p2 is in model2's local coordinate system
-   */
-  mutable FCL_REAL min_distance;
-  mutable Vec3f p1;
-  mutable Vec3f p2;
-
-  /** \brief Remember the nearest neighbor points */
-  mutable int last_tri_id1;
-  mutable int last_tri_id2;
 };
 
 
@@ -1196,6 +1178,12 @@ public:
     }
   }
 
+  mutable FCL_REAL min_distance;
+
+  mutable Vec3f p1, p2;
+
+  mutable int last_tri_id1, last_tri_id2;
+
 
   /** \brief CA controlling variable: early stop for the early iterations of CA */
   FCL_REAL w;
diff --git a/trunk/fcl/include/fcl/traversal_node_octree.h b/trunk/fcl/include/fcl/traversal_node_octree.h
index fd2d082cfc9f4359ef19674c1bbc6da5c75eb919..2b61055754df7f348048f2f8a7271541db60a503 100644
--- a/trunk/fcl/include/fcl/traversal_node_octree.h
+++ b/trunk/fcl/include/fcl/traversal_node_octree.h
@@ -83,16 +83,17 @@ public:
   }
 
 
-  FCL_REAL OcTreeDistance(const OcTree* tree1, const OcTree* tree2,
-                          const SimpleTransform& tf1, const SimpleTransform& tf2) const
+  void OcTreeDistance(const OcTree* tree1, const OcTree* tree2,
+                      const SimpleTransform& tf1, const SimpleTransform& tf2,
+                      const DistanceRequest& request_,
+                      DistanceResult& result_) const
   {
-    FCL_REAL min_dist = std::numeric_limits<FCL_REAL>::max();
+    drequest = &request_;
+    dresult = &result_;
 
     OcTreeDistanceRecurse(tree1, tree1->getRoot(), tree1->getRootBV(), 
                           tree2, tree2->getRoot(), tree2->getRootBV(),
-                          tf1, tf2, min_dist);
-
-    return min_dist;
+                          tf1, tf2);
   }
 
   template<typename BV>
@@ -110,16 +111,17 @@ public:
   }
 
   template<typename BV>
-  FCL_REAL OcTreeMeshDistance(const OcTree* tree1, const BVHModel<BV>* tree2,
-                              const SimpleTransform& tf1, const SimpleTransform& tf2) const
+  void OcTreeMeshDistance(const OcTree* tree1, const BVHModel<BV>* tree2,
+                          const SimpleTransform& tf1, const SimpleTransform& tf2,
+                          const DistanceRequest& request_,
+                          DistanceResult& result_) const
   {
-    FCL_REAL min_dist = std::numeric_limits<FCL_REAL>::max();
+    drequest = &request_;
+    dresult = &result_;
 
     OcTreeMeshDistanceRecurse(tree1, tree1->getRoot(), tree1->getRootBV(),
                               tree2, 0,
-                              tf1, tf2, min_dist);
-    
-    return min_dist;
+                              tf1, tf2);
   }
 
 
@@ -140,16 +142,17 @@ public:
 
   
   template<typename BV>
-  FCL_REAL MeshOcTreeDistance(const BVHModel<BV>* tree1, const OcTree* tree2,
-                              const SimpleTransform& tf1, const SimpleTransform& tf2) const
+  void MeshOcTreeDistance(const BVHModel<BV>* tree1, const OcTree* tree2,
+                          const SimpleTransform& tf1, const SimpleTransform& tf2,
+                          const DistanceRequest& request_,
+                          DistanceResult& result_) const
   {
-    FCL_REAL min_dist = std::numeric_limits<FCL_REAL>::max();
+    drequest = &request_;
+    dresult = &result_;
 
     OcTreeMeshDistanceRecurse(tree1, 0,
                               tree2, tree2->getRoot(), tree2->getRootBV(),
-                              tf1, tf2, min_dist);
-    
-    return min_dist;
+                              tf1, tf2);
   }
 
   template<typename S>
@@ -190,32 +193,35 @@ public:
   }
 
   template<typename S>
-  FCL_REAL OcTreeShapeDistance(const OcTree* tree, const S& s,
-                               const SimpleTransform& tf1, const SimpleTransform& tf2) const
+  void OcTreeShapeDistance(const OcTree* tree, const S& s,
+                           const SimpleTransform& tf1, const SimpleTransform& tf2,
+                           const DistanceRequest& request_,
+                           DistanceResult& result_) const
   {
+    drequest = &request_;
+    dresult = &result_;
+
     AABB aabb2;
     computeBV<AABB>(s, tf2, aabb2);
-    FCL_REAL min_dist = std::numeric_limits<FCL_REAL>::max();
-    
     OcTreeShapeDistanceRecurse(tree, tree->getRoot(), tree->getRootBV(),
                                s, aabb2,
-                               tf1, tf2, min_dist);
-    return min_dist;
+                               tf1, tf2);
   }
 
   template<typename S>
-  FCL_REAL ShapeOcTreeDistance(const S& s, const OcTree* tree,
-                               const SimpleTransform& tf1, const SimpleTransform& tf2) const
+  void ShapeOcTreeDistance(const S& s, const OcTree* tree,
+                           const SimpleTransform& tf1, const SimpleTransform& tf2,
+                           const DistanceRequest& request_,
+                           DistanceResult& result_) const
   {
+    drequest = &request_;
+    dresult = &result_;
+
     AABB aabb1;
     computeBV<AABB>(s, tf1, aabb1);
-    FCL_REAL min_dist = std::numeric_limits<FCL_REAL>::max();
-
     OcTreeShapeDistanceRecurse(tree, tree->getRoot(), tree->getRootBV(),
                                s, aabb1,
-                               tf2, tf1, min_dist);
-
-    return min_dist;
+                               tf2, tf1);
   }
   
 
@@ -223,8 +229,7 @@ private:
   template<typename S>
   bool OcTreeShapeDistanceRecurse(const OcTree* tree1, const OcTree::OcTreeNode* root1, const AABB& bv1,
                                   const S& s, const AABB& aabb2,
-                                  const SimpleTransform& tf1, const SimpleTransform& tf2,
-                                  FCL_REAL& min_dist) const
+                                  const SimpleTransform& tf1, const SimpleTransform& tf2) const
   {
     if(!root1->hasChildren())
     {
@@ -236,9 +241,10 @@ private:
  
         FCL_REAL dist;
         solver->shapeDistance(box, box_tf, s, tf2, &dist);
-        if(dist < min_dist) min_dist = dist;
         
-        return (min_dist <= 0);
+        dresult->update(dist, tree1, &s, root1 - tree1->getRoot(), DistanceResult::NONE);
+        
+        return (dresult->min_distance <= 0);
       }
       else
         return false;
@@ -257,9 +263,9 @@ private:
         AABB aabb1;
         convertBV(child_bv, tf1, aabb1);
         FCL_REAL d = aabb1.distance(aabb2);
-        if(d < min_dist)
+        if(d < dresult->min_distance)
         {
-          if(OcTreeShapeDistanceRecurse(tree1, child, child_bv, s, aabb2, tf1, tf2, min_dist))
+          if(OcTreeShapeDistanceRecurse(tree1, child, child_bv, s, aabb2, tf1, tf2))
             return true;
         }        
       }
@@ -341,7 +347,7 @@ private:
   template<typename BV>
   bool OcTreeMeshDistanceRecurse(const OcTree* tree1, const OcTree::OcTreeNode* root1, const AABB& bv1,
                                  const BVHModel<BV>* tree2, int root2,
-                                 const SimpleTransform& tf1, const SimpleTransform& tf2, FCL_REAL& min_dist) const
+                                 const SimpleTransform& tf1, const SimpleTransform& tf2) const
   {
     if(!root1->hasChildren() && tree2->getBV(root2).isLeaf())
     {
@@ -359,8 +365,10 @@ private:
         
         FCL_REAL dist;
         solver->shapeTriangleDistance(box, box_tf, p1, p2, p3, tf2, &dist);
-        if(dist < min_dist) min_dist = dist;
-        return (min_dist <= 0);
+
+        dresult->update(dist, tree1, tree2, root1 - tree1->getRoot(), primitive_id);
+
+        return (dresult->min_distance <= 0);
       }
       else
         return false;
@@ -384,9 +392,9 @@ private:
           convertBV(tree2->getBV(root2).bv, tf2, aabb2);
           d = aabb1.distance(aabb2);
           
-          if(d < min_dist)
+          if(d < dresult->min_distance)
           {
-            if(OcTreeMeshDistanceRecurse(tree1, child, child_bv, tree2, root2, tf1, tf2, min_dist))
+            if(OcTreeMeshDistanceRecurse(tree1, child, child_bv, tree2, root2, tf1, tf2))
               return true;
           }
         }
@@ -401,9 +409,9 @@ private:
       convertBV(tree2->getBV(child).bv, tf2, aabb2);
       d = aabb1.distance(aabb2);
 
-      if(d < min_dist)
+      if(d < dresult->min_distance)
       {
-        if(OcTreeMeshDistanceRecurse(tree1, root1, bv1, tree2, child, tf1, tf2, min_dist))
+        if(OcTreeMeshDistanceRecurse(tree1, root1, bv1, tree2, child, tf1, tf2))
           return true;
       }
 
@@ -411,9 +419,9 @@ private:
       convertBV(tree2->getBV(child).bv, tf2, aabb2);
       d = aabb1.distance(aabb2);
       
-      if(d < min_dist)
+      if(d < dresult->min_distance)
       {
-        if(OcTreeMeshDistanceRecurse(tree1, root1, bv1, tree2, child, tf1, tf2, min_dist))
+        if(OcTreeMeshDistanceRecurse(tree1, root1, bv1, tree2, child, tf1, tf2))
           return true;      
       }
     }
@@ -506,8 +514,7 @@ private:
 
   bool OcTreeDistanceRecurse(const OcTree* tree1, const OcTree::OcTreeNode* root1, const AABB& bv1,
                              const OcTree* tree2, const OcTree::OcTreeNode* root2, const AABB& bv2,
-                             const SimpleTransform& tf1, const SimpleTransform& tf2,
-                             FCL_REAL& min_dist) const
+                             const SimpleTransform& tf1, const SimpleTransform& tf2) const
   {
     if(!root1->hasChildren() && !root2->hasChildren())
     {
@@ -520,9 +527,10 @@ private:
 
         FCL_REAL dist;
         solver->shapeDistance(box1, box1_tf, box2, box2_tf, &dist);
-        if(dist < min_dist) min_dist = dist;
+
+        dresult->update(dist, tree1, tree2, root1 - tree1->getRoot(), root2 - tree2->getRoot());
         
-        return (min_dist <= 0);
+        return (dresult->min_distance <= 0);
       }
       else
         return false;
@@ -546,10 +554,10 @@ private:
           convertBV(bv2, tf2, aabb2);
           d = aabb1.distance(aabb2);
 
-          if(d < min_dist)
+          if(d < dresult->min_distance)
           {
           
-            if(OcTreeDistanceRecurse(tree1, child, child_bv, tree2, root2, bv2, tf1, tf2, min_dist))
+            if(OcTreeDistanceRecurse(tree1, child, child_bv, tree2, root2, bv2, tf1, tf2))
               return true;
           }
         }
@@ -571,9 +579,9 @@ private:
           convertBV(bv2, tf2, aabb2);
           d = aabb1.distance(aabb2);
 
-          if(d < min_dist)
+          if(d < dresult->min_distance)
           {
-            if(OcTreeDistanceRecurse(tree1, root1, bv1, tree2, child, child_bv, tf1, tf2, min_dist))
+            if(OcTreeDistanceRecurse(tree1, root1, bv1, tree2, child, child_bv, tf1, tf2))
               return true;
           }
         }
@@ -722,13 +730,11 @@ public:
 
   void leafTesting(int, int) const
   {
-    min_distance = otsolver->OcTreeDistance(model1, model2, tf1, tf2);
+    otsolver->OcTreeDistance(model1, model2, tf1, tf2, request, *result);
   }
 
   const OcTree* model1;
   const OcTree* model2;
-  
-  mutable FCL_REAL min_distance;
 
   const OcTreeSolver<NarrowPhaseSolver>* otsolver;
 };
@@ -812,13 +818,11 @@ public:
 
   void leafTesting(int, int) const
   {
-    min_distance = otsolver->OcTreeShapeDistance(model2, *model1, tf2, tf1);
+    otsolver->OcTreeShapeDistance(model2, *model1, tf2, tf1, request, *result);
   }
 
   const S* model1;
   const OcTree* model2;
-  
-  mutable FCL_REAL min_distance;
 
   const OcTreeSolver<NarrowPhaseSolver>* otsolver;
 };
@@ -842,13 +846,11 @@ public:
 
   void leafTesting(int, int) const
   {
-    min_distance = otsolver->OcTreeShapeDistance(model1, *model2, tf1, tf2);
+    otsolver->OcTreeShapeDistance(model1, *model2, tf1, tf2, request, *result);
   }
 
   const OcTree* model1;
   const S* model2;
-  
-  mutable FCL_REAL min_distance;
 
   const OcTreeSolver<NarrowPhaseSolver>* otsolver;
 };
@@ -934,13 +936,11 @@ public:
 
   void leafTesting(int, int) const
   {
-    min_distance = otsolver->OcTreeMeshDistance(model2, model1, tf2, tf1);
+    otsolver->OcTreeMeshDistance(model2, model1, tf2, tf1, request, *result);
   }
 
   const BVHModel<BV>* model1;
   const OcTree* model2;
-  
-  mutable FCL_REAL min_distance;
 
   const OcTreeSolver<NarrowPhaseSolver>* otsolver;
 
@@ -965,13 +965,11 @@ public:
 
   void leafTesting(int, int) const
   {
-    min_distance = otsolver->OcTreeMeshDistance(model1, model2, tf1, tf2);
+    otsolver->OcTreeMeshDistance(model1, model2, tf1, tf2, request, *result);
   }
 
   const OcTree* model1;
   const BVHModel<BV>* model2;
-  
-  mutable FCL_REAL min_distance;
 
   const OcTreeSolver<NarrowPhaseSolver>* otsolver;
 
diff --git a/trunk/fcl/include/fcl/traversal_node_shapes.h b/trunk/fcl/include/fcl/traversal_node_shapes.h
index 9f8065f9e30f33e15c545768b7dbb27c24b14cbd..2c2da37921ef0e960620a27b267433e715178e30 100644
--- a/trunk/fcl/include/fcl/traversal_node_shapes.h
+++ b/trunk/fcl/include/fcl/traversal_node_shapes.h
@@ -73,7 +73,7 @@ public:
       if(nsolver->shapeIntersect(*model1, tf1, *model2, tf2, &contact_point, &penetration_depth, &normal))
       {
         is_collision = true;
-        result->contacts.push_back(Contact(model1, model2, Contact::NONE, Contact::NONE, contact_point, normal, penetration_depth));
+        result->addContact(Contact(model1, model2, Contact::NONE, Contact::NONE, contact_point, normal, penetration_depth));
       }
     }
     else
@@ -81,7 +81,7 @@ public:
       if(nsolver->shapeIntersect(*model1, tf1, *model2, tf2, NULL, NULL, NULL))
       {
         is_collision = true;
-        result->contacts.push_back(Contact(model1, model2, Contact::NONE, Contact::NONE));
+        result->addContact(Contact(model1, model2, Contact::NONE, Contact::NONE));
       }
     }
 
@@ -92,7 +92,7 @@ public:
       computeBV<AABB, S2>(*model2, tf2, aabb2);
       AABB overlap_part;
       aabb1.overlap(aabb2, overlap_part);
-      result->cost_sources.push_back(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
+      result->addCostSource(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
     }
   }
 
@@ -123,18 +123,14 @@ public:
 
   void leafTesting(int, int) const
   {
-    is_collision = !nsolver->shapeDistance(*model1, tf1, *model2, tf2, &min_distance);
+    FCL_REAL distance;
+    !nsolver->shapeDistance(*model1, tf1, *model2, tf2, &distance);
+    result->update(distance, model1, model2, DistanceResult::NONE, DistanceResult::NONE);
   }
 
   const S1* model1;
   const S2* model2;
 
-  mutable FCL_REAL min_distance;
-  mutable Vec3f p1;
-  mutable Vec3f p2;
-
-  mutable bool is_collision;
-
   const NarrowPhaseSolver* nsolver;
 };
 
diff --git a/trunk/fcl/src/broad_phase_collision.cpp b/trunk/fcl/src/broad_phase_collision.cpp
index 3145339624175af33712f988fc8bbe40227594c3..cab7b423a375c881ae26a34600e2e0eec9a07960 100644
--- a/trunk/fcl/src/broad_phase_collision.cpp
+++ b/trunk/fcl/src/broad_phase_collision.cpp
@@ -57,18 +57,8 @@ bool defaultCollisionFunction(CollisionObject* o1, CollisionObject* o2, void* cd
 
   if(cdata->done) return true;
 
-  CollisionResult local_result;
-  int num_contacts = collide(o1, o2, request, local_result);
+  collide(o1, o2, request, result);
 
-  std::vector<Contact>& out_contacts = result.contacts;
-  const std::vector<Contact>& in_contacts = local_result.contacts;
-  for(int i = 0; i < num_contacts; ++i)
-  {
-    out_contacts.push_back(in_contacts[i]);
-    // result.contacts.push_back(local_result.contacts[i]);
-  }
-
-  // set done flag
   if( (!request.exhaustive) && (result.isCollision()) && (result.numContacts() >= request.num_max_contacts))
     cdata->done = true;
 
diff --git a/trunk/fcl/src/collision_func_matrix.cpp b/trunk/fcl/src/collision_func_matrix.cpp
index 74b9bd18df05fdbd141eb8811bfae8058db40ad6..5b88cea63c144e61445774a0253bcccb30048a20 100644
--- a/trunk/fcl/src/collision_func_matrix.cpp
+++ b/trunk/fcl/src/collision_func_matrix.cpp
@@ -45,39 +45,23 @@
 namespace fcl
 {
 
-template<typename T_SH>
-static inline int OcTreeShapeContactCollection(const std::vector<Contact>& pairs, const OcTree* obj1, const T_SH* obj2,
-                                               const CollisionRequest& request, CollisionResult& result)
-{
-  int num_contacts = pairs.size();
-  if(num_contacts > 0)
-  {
-    if((!request.exhaustive) && (num_contacts > request.num_max_contacts)) num_contacts = request.num_max_contacts;
-    std::vector<Contact>& contacts = result.contacts;
-    contacts.resize(num_contacts);
-    for(int i = 0; i < num_contacts; ++i)
-      contacts[i] = pairs[i];
-  }
-  
-  return num_contacts;
-}
-
 template<typename T_SH, typename NarrowPhaseSolver>
 int ShapeOcTreeCollide(const CollisionGeometry* o1, const SimpleTransform& tf1, const CollisionGeometry* o2, const SimpleTransform& tf2,
                        const NarrowPhaseSolver* nsolver,
                        const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   ShapeOcTreeCollisionTraversalNode<T_SH, NarrowPhaseSolver> node;
   const T_SH* obj1 = static_cast<const T_SH*>(o1);
   const OcTree* obj2 = static_cast<const OcTree*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, result);
   collide(&node);
-  int num_contacts = OcTreeShapeContactCollection(local_result.contacts, obj2, obj1, request, result);
 
-  return num_contacts;
+  return result.numContacts() - num_contacts_old;
 }
 
 template<typename T_SH, typename NarrowPhaseSolver>
@@ -85,33 +69,18 @@ int OcTreeShapeCollide(const CollisionGeometry* o1, const SimpleTransform& tf1,
                        const NarrowPhaseSolver* nsolver,
                        const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   OcTreeShapeCollisionTraversalNode<T_SH, NarrowPhaseSolver> node;
   const OcTree* obj1 = static_cast<const OcTree*>(o1);
   const T_SH* obj2 = static_cast<const T_SH*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, result);
   collide(&node);
-  int num_contacts = OcTreeShapeContactCollection(local_result.contacts, obj1, obj2, request, result);
-
-  return num_contacts;
-}
 
-static inline int OcTreeContactCollection(const std::vector<Contact>& pairs, const OcTree* obj1, const OcTree* obj2,
-                                          const CollisionRequest& request, CollisionResult& result)
-{
-  int num_contacts = pairs.size();
-  if(num_contacts > 0)
-  {
-    if((!request.exhaustive) && (num_contacts > request.num_max_contacts)) num_contacts = request.num_max_contacts;
-    std::vector<Contact>& contacts = result.contacts;
-    contacts.resize(num_contacts);
-    for(int i = 0; i < num_contacts; ++i)
-      contacts[i] = pairs[i];
-  }
-  
-  return num_contacts;
+  return result.numContacts() - num_contacts_old;
 }
 
 template<typename NarrowPhaseSolver>
@@ -119,34 +88,18 @@ int OcTreeCollide(const CollisionGeometry* o1, const SimpleTransform& tf1, const
                   const NarrowPhaseSolver* nsolver,
                   const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   OcTreeCollisionTraversalNode<NarrowPhaseSolver> node;
   const OcTree* obj1 = static_cast<const OcTree*>(o1);
   const OcTree* obj2 = static_cast<const OcTree*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, result);
   collide(&node);
-  int num_contacts = OcTreeContactCollection(local_result.contacts, obj1, obj2, request, result);
-  return num_contacts;
-}
 
-
-template<typename T_BVH>
-static inline int OcTreeBVHContactCollection(const std::vector<Contact>& pairs, const OcTree* obj1, const BVHModel<T_BVH>* obj2,
-                                             const CollisionRequest& request, CollisionResult& result)
-{
-  int num_contacts = pairs.size();
-  if(num_contacts > 0)
-  {
-    if((!request.exhaustive) && (num_contacts > request.num_max_contacts)) num_contacts = request.num_max_contacts;
-    std::vector<Contact>& contacts = result.contacts;
-    contacts.resize(num_contacts);
-    for(int i = 0; i < num_contacts; ++i)
-      contacts[i] = pairs[i];
-  }
-  
-  return num_contacts;
+  return result.numContacts() - num_contacts_old;
 }
 
 template<typename T_BVH, typename NarrowPhaseSolver>
@@ -154,16 +107,18 @@ int OcTreeBVHCollide(const CollisionGeometry* o1, const SimpleTransform& tf1, co
                      const NarrowPhaseSolver* nsolver,
                      const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   OcTreeMeshCollisionTraversalNode<T_BVH, NarrowPhaseSolver> node;
   const OcTree* obj1 = static_cast<const OcTree*>(o1);
   const BVHModel<T_BVH>* obj2 = static_cast<const BVHModel<T_BVH>*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, result);
   collide(&node);
-  int num_contacts = OcTreeBVHContactCollection(local_result.contacts, obj1, obj2, request, result);
-  return num_contacts;
+  
+  return result.numContacts() - num_contacts_old;
 }
 
 template<typename T_BVH, typename NarrowPhaseSolver>
@@ -171,16 +126,18 @@ int BVHOcTreeCollide(const CollisionGeometry* o1, const SimpleTransform& tf1, co
                      const NarrowPhaseSolver* nsolver,
                      const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   MeshOcTreeCollisionTraversalNode<T_BVH, NarrowPhaseSolver> node;
   const BVHModel<T_BVH>* obj1 = static_cast<const BVHModel<T_BVH>*>(o1);
   const OcTree* obj2 = static_cast<const OcTree*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, result);
   collide(&node);
-  int num_contacts = OcTreeBVHContactCollection(local_result.contacts, obj2, obj1, request, result);
-  return num_contacts;
+  
+  return result.numContacts() - num_contacts_old;
 }
 
 
@@ -189,39 +146,19 @@ int ShapeShapeCollide(const CollisionGeometry* o1, const SimpleTransform& tf1, c
                       const NarrowPhaseSolver* nsolver,
                       const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   ShapeCollisionTraversalNode<T_SH1, T_SH2, NarrowPhaseSolver> node;
   const T_SH1* obj1 = static_cast<const T_SH1*>(o1);
   const T_SH2* obj2 = static_cast<const T_SH2*>(o2);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, result);
   collide(&node);
-  if(local_result.contacts.size() == 0) return 0;
-  result.contacts.resize(1);                                                   
-  result.contacts[0] = local_result.contacts[0];
-  return 1;
-}
-
-
-
-template<typename T_BVH, typename T_SH>
-static inline int BVHShapeContactCollection(const std::vector<Contact>& pairs, const BVHModel<T_BVH>* obj1, const T_SH* obj2,
-                                            const CollisionRequest& request, CollisionResult& result)
-{
-  int num_contacts = pairs.size();
-  if(num_contacts > 0)
-  {
-    if((!request.exhaustive) && (num_contacts > request.num_max_contacts)) num_contacts = request.num_max_contacts;
-    std::vector<Contact>& contacts = result.contacts;
-    contacts.resize(num_contacts);
-    for(int i = 0; i < num_contacts; ++i)
-      contacts[i] = pairs[i];
-  }
 
-  return num_contacts;
+  return result.numContacts() - num_contacts_old;
 }
 
-
 template<typename T_BVH, typename T_SH, typename NarrowPhaseSolver>
 struct BVHShapeCollider
 {
@@ -229,20 +166,20 @@ struct BVHShapeCollider
                      const NarrowPhaseSolver* nsolver,
                      const CollisionRequest& request, CollisionResult& result)
   {
+    if(request.num_max_contacts <= result.numContacts()) return 0;
+    size_t num_contacts_old = result.numContacts();
+
     MeshShapeCollisionTraversalNode<T_BVH, T_SH, NarrowPhaseSolver> node;
     const BVHModel<T_BVH>* obj1 = static_cast<const BVHModel<T_BVH>* >(o1);
     BVHModel<T_BVH>* obj1_tmp = new BVHModel<T_BVH>(*obj1);
     SimpleTransform tf1_tmp = tf1;
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    CollisionResult local_result;
-    initialize(node, *obj1_tmp, tf1_tmp, *obj2, tf2, nsolver, request, local_result);
+    initialize(node, *obj1_tmp, tf1_tmp, *obj2, tf2, nsolver, request, result);
     fcl::collide(&node);
 
-    int num_contacts = BVHShapeContactCollection(local_result.contacts, obj1, obj2, request, result);
-
     delete obj1_tmp;
-    return num_contacts;
+    return result.numContacts() - num_contacts_old;
   }
 };
 
@@ -254,15 +191,17 @@ struct BVHShapeCollider<OBB, T_SH, NarrowPhaseSolver>
                      const NarrowPhaseSolver* nsolver,
                      const CollisionRequest& request, CollisionResult& result)
   {
+    if(request.num_max_contacts <= result.numContacts()) return 0;
+    size_t num_contacts_old = result.numContacts();
+
     MeshShapeCollisionTraversalNodeOBB<T_SH, NarrowPhaseSolver> node;
     const BVHModel<OBB>* obj1 = static_cast<const BVHModel<OBB>* >(o1);
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    CollisionResult local_result;
-    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
+    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, result);
     fcl::collide(&node);
 
-    return BVHShapeContactCollection(local_result.contacts, obj1, obj2, request, result);
+    return result.numContacts() - num_contacts_old;
   } 
 };
 
@@ -274,15 +213,17 @@ struct BVHShapeCollider<RSS, T_SH, NarrowPhaseSolver>
                      const NarrowPhaseSolver* nsolver,
                      const CollisionRequest& request, CollisionResult& result)
   {
+    if(request.num_max_contacts <= result.numContacts()) return 0;
+    size_t num_contacts_old = result.numContacts();
+
     MeshShapeCollisionTraversalNodeRSS<T_SH, NarrowPhaseSolver> node;
     const BVHModel<RSS>* obj1 = static_cast<const BVHModel<RSS>* >(o1);
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    CollisionResult local_result;
-    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
+    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, result);
     fcl::collide(&node);
 
-    return BVHShapeContactCollection(local_result.contacts, obj1, obj2, request, result);
+    return result.numContacts() - num_contacts_old;
   } 
 };
 
@@ -294,15 +235,17 @@ struct BVHShapeCollider<kIOS, T_SH, NarrowPhaseSolver>
                      const NarrowPhaseSolver* nsolver,
                      const CollisionRequest& request, CollisionResult& result)
   {
+    if(request.num_max_contacts <= result.numContacts()) return 0;
+    size_t num_contacts_old = result.numContacts();
+
     MeshShapeCollisionTraversalNodekIOS<T_SH, NarrowPhaseSolver> node;
     const BVHModel<kIOS>* obj1 = static_cast<const BVHModel<kIOS>* >(o1);
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    CollisionResult local_result;
-    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
+    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, result);
     fcl::collide(&node);
 
-    return BVHShapeContactCollection(local_result.contacts, obj1, obj2, request, result);
+    return result.numContacts() - num_contacts_old;
   } 
 };
 
@@ -314,39 +257,27 @@ struct BVHShapeCollider<OBBRSS, T_SH, NarrowPhaseSolver>
                      const NarrowPhaseSolver* nsolver,
                      const CollisionRequest& request, CollisionResult& result)
   {
+    if(request.num_max_contacts <= result.numContacts()) return 0;
+    size_t num_contacts_old = result.numContacts();
+
     MeshShapeCollisionTraversalNodeOBBRSS<T_SH, NarrowPhaseSolver> node;
     const BVHModel<OBBRSS>* obj1 = static_cast<const BVHModel<OBBRSS>* >(o1);
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    CollisionResult local_result;
-    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
+    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, result);
     fcl::collide(&node);
 
-    return BVHShapeContactCollection(local_result.contacts, obj1, obj2, request, result);
+    return result.numContacts() - num_contacts_old;
   } 
 };
 
 
-template<typename T_BVH>
-static inline int BVHContactCollection(const std::vector<Contact>& pairs, const BVHModel<T_BVH>* obj1, const BVHModel<T_BVH>* obj2, const CollisionRequest& request, CollisionResult& result)
-{
-  int num_contacts = pairs.size();
-  if(num_contacts > 0)
-  {
-    if((!request.exhaustive) && (num_contacts > request.num_max_contacts)) num_contacts = request.num_max_contacts;
-    std::vector<Contact>& contacts = result.contacts;
-    contacts.resize(num_contacts);
-    for(int i = 0; i < num_contacts; ++i)
-      contacts[i] = pairs[i];
-  }
-
-  return num_contacts;
-}
-
-
 template<typename T_BVH>
 int BVHCollide(const CollisionGeometry* o1, const SimpleTransform& tf1, const CollisionGeometry* o2, const SimpleTransform& tf2, const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+  
   MeshCollisionTraversalNode<T_BVH> node;
   const BVHModel<T_BVH>* obj1 = static_cast<const BVHModel<T_BVH>* >(o1);
   const BVHModel<T_BVH>* obj2 = static_cast<const BVHModel<T_BVH>* >(o2);
@@ -355,57 +286,62 @@ int BVHCollide(const CollisionGeometry* o1, const SimpleTransform& tf1, const Co
   BVHModel<T_BVH>* obj2_tmp = new BVHModel<T_BVH>(*obj2);
   SimpleTransform tf2_tmp = tf2;
   
-  CollisionResult local_result;
-  initialize(node, *obj1_tmp, tf1_tmp, *obj2_tmp, tf2_tmp, request, local_result);
+  initialize(node, *obj1_tmp, tf1_tmp, *obj2_tmp, tf2_tmp, request, result);
   collide(&node);
-  int num_contacts = BVHContactCollection(local_result.contacts, obj1, obj2, request, result);
 
   delete obj1_tmp;
   delete obj2_tmp;
-  return num_contacts;
+
+  return result.numContacts() - num_contacts_old;
 }
 
 template<>
 int BVHCollide<OBB>(const CollisionGeometry* o1, const SimpleTransform& tf1, const CollisionGeometry* o2, const SimpleTransform& tf2, const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   MeshCollisionTraversalNodeOBB node;
   const BVHModel<OBB>* obj1 = static_cast<const BVHModel<OBB>* >(o1);
   const BVHModel<OBB>* obj2 = static_cast<const BVHModel<OBB>* >(o2);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, request, result);
   collide(&node);
 
-  return BVHContactCollection(local_result.contacts, obj1, obj2, request, result);
+  return result.numContacts() - num_contacts_old;
 }
 
 template<>
 int BVHCollide<OBBRSS>(const CollisionGeometry* o1, const SimpleTransform& tf1, const CollisionGeometry* o2, const SimpleTransform& tf2, const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   MeshCollisionTraversalNodeOBBRSS node;
   const BVHModel<OBBRSS>* obj1 = static_cast<const BVHModel<OBBRSS>* >(o1);
   const BVHModel<OBBRSS>* obj2 = static_cast<const BVHModel<OBBRSS>* >(o2);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, request, result);
   collide(&node);
 
-  return BVHContactCollection(local_result.contacts, obj1, obj2, request, result);
+  return result.numContacts() - num_contacts_old;
 }
 
 
 template<>
 int BVHCollide<kIOS>(const CollisionGeometry* o1, const SimpleTransform& tf1, const CollisionGeometry* o2, const SimpleTransform& tf2, const CollisionRequest& request, CollisionResult& result)
 {
+  if(request.num_max_contacts <= result.numContacts()) return 0;
+  size_t num_contacts_old = result.numContacts();
+
   MeshCollisionTraversalNodekIOS node;
   const BVHModel<kIOS>* obj1 = static_cast<const BVHModel<kIOS>* >(o1);
   const BVHModel<kIOS>* obj2 = static_cast<const BVHModel<kIOS>* >(o2);
 
-  CollisionResult local_result;
-  initialize(node, *obj1, tf1, *obj2, tf2, request, local_result);
+  initialize(node, *obj1, tf1, *obj2, tf2, request, result);
   collide(&node);
 
-  return BVHContactCollection(local_result.contacts, obj1, obj2, request, result);
+  return result.numContacts() - num_contacts_old;
 }
 
 
diff --git a/trunk/fcl/src/conservative_advancement.cpp b/trunk/fcl/src/conservative_advancement.cpp
index d4bd095196770ff09cf89abd3e2d591b9c597ba5..dcad4af3dacc05f8321a6d07a3fcab3c6c258ffa 100644
--- a/trunk/fcl/src/conservative_advancement.cpp
+++ b/trunk/fcl/src/conservative_advancement.cpp
@@ -93,7 +93,7 @@ int conservativeAdvancement(const CollisionGeometry* o1,
 
   collide(&cnode);
 
-  int b = result.contacts.size();
+  int b = result.numContacts();
 
   if(b > 0)
   {
diff --git a/trunk/fcl/src/distance_func_matrix.cpp b/trunk/fcl/src/distance_func_matrix.cpp
index 4a4429fc5280d125c278cc2e78d565efb74e3078..66c363ae819b59df20e9e513c36b1906ebd6d344 100644
--- a/trunk/fcl/src/distance_func_matrix.cpp
+++ b/trunk/fcl/src/distance_func_matrix.cpp
@@ -51,11 +51,13 @@ FCL_REAL ShapeOcTreeDistance(const CollisionGeometry* o1, const SimpleTransform&
   const T_SH* obj1 = static_cast<const T_SH*>(o1);
   const OcTree* obj2 = static_cast<const OcTree*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request);
+
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
   distance(&node);
   
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 template<typename T_SH, typename NarrowPhaseSolver>
@@ -66,11 +68,13 @@ FCL_REAL OcTreeShapeDistance(const CollisionGeometry* o1, const SimpleTransform&
   const OcTree* obj1 = static_cast<const OcTree*>(o1);
   const T_SH* obj2 = static_cast<const T_SH*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request);
+
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
   distance(&node);
   
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 template<typename NarrowPhaseSolver>
@@ -81,11 +85,13 @@ FCL_REAL OcTreeDistance(const CollisionGeometry* o1, const SimpleTransform& tf1,
   const OcTree* obj1 = static_cast<const OcTree*>(o1);
   const OcTree* obj2 = static_cast<const OcTree*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request);
+
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
   distance(&node);
 
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 template<typename T_BVH, typename NarrowPhaseSolver>
@@ -96,11 +102,13 @@ FCL_REAL BVHOcTreeDistance(const CollisionGeometry* o1, const SimpleTransform& t
   const BVHModel<T_BVH>* obj1 = static_cast<const BVHModel<T_BVH>*>(o1);
   const OcTree* obj2 = static_cast<const OcTree*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request);
+
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
   distance(&node);
 
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 template<typename T_BVH, typename NarrowPhaseSolver>
@@ -111,11 +119,13 @@ FCL_REAL OcTreeBVHDistance(const CollisionGeometry* o1, const SimpleTransform& t
   const OcTree* obj1 = static_cast<const OcTree*>(o1);
   const BVHModel<T_BVH>* obj2 = static_cast<const BVHModel<T_BVH>*>(o2);
   OcTreeSolver<NarrowPhaseSolver> otsolver(nsolver);
-  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request);
+
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, &otsolver, request, local_result);
   distance(&node);
 
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 
@@ -127,11 +137,13 @@ FCL_REAL ShapeShapeDistance(const CollisionGeometry* o1, const SimpleTransform&
   ShapeDistanceTraversalNode<T_SH1, T_SH2, NarrowPhaseSolver> node;
   const T_SH1* obj1 = static_cast<const T_SH1*>(o1);
   const T_SH2* obj2 = static_cast<const T_SH2*>(o2);
-  initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request);
+
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
   distance(&node);
 
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 template<typename T_BVH, typename T_SH, typename NarrowPhaseSolver>
@@ -146,12 +158,13 @@ struct BVHShapeDistancer
     SimpleTransform tf1_tmp = tf1;
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    initialize(node, *obj1_tmp, tf1_tmp, *obj2, tf2, nsolver, request);
+    DistanceResult local_result;
+    initialize(node, *obj1_tmp, tf1_tmp, *obj2, tf2, nsolver, request, local_result);
     fcl::distance(&node);
     
     delete obj1_tmp;
-    result.min_distance = node.min_distance;
-    return node.min_distance;
+    result.update(local_result);
+    return local_result.min_distance;
   }
 };
 
@@ -165,11 +178,12 @@ struct BVHShapeDistancer<RSS, T_SH, NarrowPhaseSolver>
     const BVHModel<RSS>* obj1 = static_cast<const BVHModel<RSS>* >(o1);
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request);
+    DistanceResult local_result;
+    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
     fcl::distance(&node);
 
-    result.min_distance = node.min_distance;
-    return node.min_distance;
+    result.update(local_result);
+    return local_result.min_distance;
   }
 };
 
@@ -184,11 +198,12 @@ struct BVHShapeDistancer<kIOS, T_SH, NarrowPhaseSolver>
     const BVHModel<kIOS>* obj1 = static_cast<const BVHModel<kIOS>* >(o1);
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request);
+    DistanceResult local_result;
+    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
     fcl::distance(&node);
 
-    result.min_distance = node.min_distance;
-    return node.min_distance;
+    result.update(local_result);
+    return local_result.min_distance;
   }
 };
 
@@ -202,11 +217,12 @@ struct BVHShapeDistancer<OBBRSS, T_SH, NarrowPhaseSolver>
     const BVHModel<OBBRSS>* obj1 = static_cast<const BVHModel<OBBRSS>* >(o1);
     const T_SH* obj2 = static_cast<const T_SH*>(o2);
 
-    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request);
+    DistanceResult local_result;
+    initialize(node, *obj1, tf1, *obj2, tf2, nsolver, request, local_result);
     fcl::distance(&node);
 
-    result.min_distance = node.min_distance;
-    return node.min_distance;
+    result.update(local_result);
+    return local_result.min_distance;
   }
 };
 
@@ -223,11 +239,12 @@ FCL_REAL BVHDistance(const CollisionGeometry* o1, const SimpleTransform& tf1, co
   BVHModel<T_BVH>* obj2_tmp = new BVHModel<T_BVH>(*obj2);
   SimpleTransform tf2_tmp = tf2;
 
-  initialize(node, *obj1_tmp, tf1_tmp, *obj2_tmp, tf2_tmp, request);
+  DistanceResult local_result;
+  initialize(node, *obj1_tmp, tf1_tmp, *obj2_tmp, tf2_tmp, request, local_result);
   distance(&node);
   
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 template<>
@@ -238,11 +255,12 @@ FCL_REAL BVHDistance<RSS>(const CollisionGeometry* o1, const SimpleTransform& tf
   const BVHModel<RSS>* obj1 = static_cast<const BVHModel<RSS>* >(o1);
   const BVHModel<RSS>* obj2 = static_cast<const BVHModel<RSS>* >(o2);
 
-  initialize(node, *obj1, tf1, *obj2, tf2, request);
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, request, local_result);
   distance(&node);
 
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 template<>
@@ -253,11 +271,12 @@ FCL_REAL BVHDistance<kIOS>(const CollisionGeometry* o1, const SimpleTransform& t
   const BVHModel<kIOS>* obj1 = static_cast<const BVHModel<kIOS>* >(o1);
   const BVHModel<kIOS>* obj2 = static_cast<const BVHModel<kIOS>* >(o2);
 
-  initialize(node, *obj1, tf1, *obj2, tf2, request);
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, request, local_result);
   distance(&node);
 
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 
@@ -269,11 +288,12 @@ FCL_REAL BVHDistance<OBBRSS>(const CollisionGeometry* o1, const SimpleTransform&
   const BVHModel<OBBRSS>* obj1 = static_cast<const BVHModel<OBBRSS>* >(o1);
   const BVHModel<OBBRSS>* obj2 = static_cast<const BVHModel<OBBRSS>* >(o2);
 
-  initialize(node, *obj1, tf1, *obj2, tf2, request);
+  DistanceResult local_result;
+  initialize(node, *obj1, tf1, *obj2, tf2, request, local_result);
   distance(&node);
 
-  result.min_distance = node.min_distance;
-  return node.min_distance;
+  result.update(local_result);
+  return local_result.min_distance;
 }
 
 
diff --git a/trunk/fcl/src/simple_setup.cpp b/trunk/fcl/src/simple_setup.cpp
index a9d9c21188bc7de8af2b64f33c8fe9c93785b253..53d472a685d0c6eb32ea4001363e6cd645265056 100644
--- a/trunk/fcl/src/simple_setup.cpp
+++ b/trunk/fcl/src/simple_setup.cpp
@@ -255,12 +255,14 @@ template<typename BV, typename OrientedNode>
 static inline bool setupMeshDistanceOrientedNode(OrientedNode& node,
                                                  const BVHModel<BV>& model1, const SimpleTransform& tf1,
                                                  const BVHModel<BV>& model2, const SimpleTransform& tf2,
-                                                 const DistanceRequest& request)
+                                                 const DistanceRequest& request,
+                                                 DistanceResult& result)
 {
   if(model1.getModelType() != BVH_MODEL_TRIANGLES || model2.getModelType() != BVH_MODEL_TRIANGLES)
     return false;
 
   node.request = request;
+  node.result = &result;
 
   node.model1 = &model1;
   node.tf1 = tf1;
@@ -284,26 +286,29 @@ static inline bool setupMeshDistanceOrientedNode(OrientedNode& node,
 bool initialize(MeshDistanceTraversalNodeRSS& node,
                 const BVHModel<RSS>& model1, const SimpleTransform& tf1,
                 const BVHModel<RSS>& model2, const SimpleTransform& tf2,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, request);
+  return details::setupMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, request, result);
 }
 
 
 bool initialize(MeshDistanceTraversalNodekIOS& node,
                 const BVHModel<kIOS>& model1, const SimpleTransform& tf1,
                 const BVHModel<kIOS>& model2, const SimpleTransform& tf2,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, request);
+  return details::setupMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, request, result);
 }
 
 bool initialize(MeshDistanceTraversalNodeOBBRSS& node,
                 const BVHModel<OBBRSS>& model1, const SimpleTransform& tf1,
                 const BVHModel<OBBRSS>& model2, const SimpleTransform& tf2,
-                const DistanceRequest& request)
+                const DistanceRequest& request,
+                DistanceResult& result)
 {
-  return details::setupMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, request);
+  return details::setupMeshDistanceOrientedNode(node, model1, tf1, model2, tf2, request, result);
 }
 
 
diff --git a/trunk/fcl/src/traversal_node_bvhs.cpp b/trunk/fcl/src/traversal_node_bvhs.cpp
index e9a667fabedcf7d5b62d1b0a29986c7f039ebb16..5c487612230b4b56f8c1565080e445a560fb6b88 100644
--- a/trunk/fcl/src/traversal_node_bvhs.cpp
+++ b/trunk/fcl/src/traversal_node_bvhs.cpp
@@ -85,7 +85,7 @@ static inline void meshCollisionOrientedNodeLeafTesting(int b1, int b2,
     if(Intersect::intersect_Triangle(p1, p2, p3, q1, q2, q3, R, T))
     {
       is_intersect = true;
-      result.contacts.push_back(Contact(model1, model2, primitive_id1, primitive_id2));
+      result.addContact(Contact(model1, model2, primitive_id1, primitive_id2));
     }
   }
   else // need compute the contact information
@@ -100,22 +100,22 @@ static inline void meshCollisionOrientedNodeLeafTesting(int b1, int b2,
       is_intersect = true;
       if(!request.exhaustive)
       {
-        if(request.num_max_contacts < result.contacts.size() + n_contacts)
-          n_contacts = (request.num_max_contacts > result.contacts.size()) ? (request.num_max_contacts - result.contacts.size()) : 0;
+        if(request.num_max_contacts < result.numContacts() + n_contacts)
+          n_contacts = (request.num_max_contacts > result.numContacts()) ? (request.num_max_contacts - result.numContacts()) : 0;
       }
 
       for(unsigned int i = 0; i < n_contacts; ++i)
       {
-        result.contacts.push_back(Contact(model1, model2, primitive_id1, primitive_id2, tf1.transform(contacts[i]), tf1.getQuatRotation().transform(normal), penetration));
+        result.addContact(Contact(model1, model2, primitive_id1, primitive_id2, tf1.transform(contacts[i]), tf1.getQuatRotation().transform(normal), penetration));
       }
     }
   }
   
-  if(is_intersect && request.enable_cost && (request.num_max_cost_sources > result.cost_sources.size()))
+  if(is_intersect && request.enable_cost && (request.num_max_cost_sources > result.numCostSources()))
   {
     AABB overlap_part;
     AABB(tf1.transform(p1), tf1.transform(p2), tf1.transform(p3)).overlap(AABB(tf2.transform(q1), tf2.transform(q2), tf2.transform(q3)), overlap_part);
-    result.cost_sources.push_back(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
+    result.addCostSource(CostSource(overlap_part.min_, overlap_part.max_, cost_density));
   }
 }
 
@@ -130,11 +130,8 @@ static inline void meshDistanceOrientedNodeLeafTesting(int b1, int b2,
                                                        const Matrix3f& R, const Vec3f& T,
                                                        bool enable_statistics,
                                                        int& num_leaf_tests,
-                                                       Vec3f& p1,
-                                                       Vec3f& p2,
-                                                       int& last_tri_id1,
-                                                       int& last_tri_id2,
-                                                       FCL_REAL& min_distance)
+                                                       const DistanceRequest& request,
+                                                       DistanceResult& result)
 {
   if(enable_statistics) num_leaf_tests++;
 
@@ -162,16 +159,10 @@ static inline void meshDistanceOrientedNodeLeafTesting(int b1, int b2,
                                              R, T,
                                              P1, P2);
 
-  if(d < min_distance)
-  {
-    min_distance = d;
-
-    p1 = P1;
-    p2 = P2;
-
-    last_tri_id1 = primitive_id1;
-    last_tri_id2 = primitive_id2;
-  }
+  if(request.enable_nearest_points)
+    result.update(d, model1, model2, primitive_id1, primitive_id2, P1, P2);
+  else
+    result.update(d, model1, model2, primitive_id1, primitive_id2);
 }
 
 }
@@ -472,63 +463,70 @@ void PointCloudMeshCollisionTraversalNodeRSS::leafTesting(int b1, int b2) const
 
 #endif
 
-
-
-
-static inline void distance_preprocess(Vec3f* vertices1, Vec3f* vertices2,
-                                       Triangle* tri_indices1, Triangle* tri_indices2,
-                                       int last_tri_id1, int last_tri_id2,
-                                       const Matrix3f& R, const Vec3f& T,
-                                       FCL_REAL& min_distance,
-                                       Vec3f& p1,
-                                       Vec3f& p2)
+namespace details
 {
-  const Triangle& last_tri1 = tri_indices1[last_tri_id1];
-  const Triangle& last_tri2 = tri_indices2[last_tri_id2];
 
-  Vec3f last_tri1_points[3];
-  Vec3f last_tri2_points[3];
-
-  last_tri1_points[0] = vertices1[last_tri1[0]];
-  last_tri1_points[1] = vertices1[last_tri1[1]];
-  last_tri1_points[2] = vertices1[last_tri1[2]];
-
-  last_tri2_points[0] = vertices2[last_tri2[0]];
-  last_tri2_points[1] = vertices2[last_tri2[1]];
-  last_tri2_points[2] = vertices2[last_tri2[2]];
-
-  Vec3f last_tri_P, last_tri_Q;
-
-  FCL_REAL init_dist = TriangleDistance::triDistance(last_tri1_points[0], last_tri1_points[1], last_tri1_points[2],
-                                                     last_tri2_points[0], last_tri2_points[1], last_tri2_points[2],
-                                                     R, T, last_tri_P, last_tri_Q);
-  p1 = last_tri_P;
-  p2 = R.transposeTimes(last_tri_Q - T);
-
-  // if(min_distance > init_dist) min_distance = init_dist;
+template<typename BV>
+static inline void distancePreprocessOrientedNode(const BVHModel<BV>* model1, const BVHModel<BV>* model2,
+                                                  const Vec3f* vertices1, Vec3f* vertices2,
+                                                  Triangle* tri_indices1, Triangle* tri_indices2,
+                                                  int init_tri_id1, int init_tri_id2,
+                                                  const Matrix3f& R, const Vec3f& T,
+                                                  const DistanceRequest& request,
+                                                  DistanceResult& result)
+{
+  const Triangle& init_tri1 = tri_indices1[init_tri_id1];
+  const Triangle& init_tri2 = tri_indices2[init_tri_id2];
+
+  Vec3f init_tri1_points[3];
+  Vec3f init_tri2_points[3];
+
+  init_tri1_points[0] = vertices1[init_tri1[0]];
+  init_tri1_points[1] = vertices1[init_tri1[1]];
+  init_tri1_points[2] = vertices1[init_tri1[2]];
+
+  init_tri2_points[0] = vertices2[init_tri2[0]];
+  init_tri2_points[1] = vertices2[init_tri2[1]];
+  init_tri2_points[2] = vertices2[init_tri2[2]];
+
+  Vec3f p1, p2;
+  FCL_REAL distance = TriangleDistance::triDistance(init_tri1_points[0], init_tri1_points[1], init_tri1_points[2],
+                                                    init_tri2_points[0], init_tri2_points[1], init_tri2_points[2],
+                                                    R, T, p1, p2);
+
+  if(request.enable_nearest_points)
+    result.update(distance, model1, model2, init_tri_id1, init_tri_id2, p1, p2);
+  else
+    result.update(distance, model1, model2, init_tri_id1, init_tri_id2);
 }
 
-static inline void distance_postprocess(const Matrix3f& R, const Vec3f& T, Vec3f& p2)
+template<typename BV>
+static inline void distancePostprocessOrientedNode(const BVHModel<BV>* model1, const BVHModel<BV>* model2,
+                                                   const SimpleTransform& tf1, const DistanceRequest& request, DistanceResult& result)
 {
-  Vec3f u = p2 - T;
-  p2 = R.transposeTimes(u);
+  /// the points obtained by triDistance are not in world space: both are in object1's local coordinate system, so we need to convert them into the world space.
+  if(request.enable_nearest_points && (result.o1 == model1) && (result.o2 == model2))
+  {
+    result.nearest_points[0] = tf1.transform(result.nearest_points[0]);
+    result.nearest_points[1] = tf1.transform(result.nearest_points[1]);
+  }
 }
 
+}
 
 MeshDistanceTraversalNodeRSS::MeshDistanceTraversalNodeRSS() : MeshDistanceTraversalNode<RSS>()
 {
   R.setIdentity();
-  // default T is 0
 }
 
 void MeshDistanceTraversalNodeRSS::preprocess()
 {
-  distance_preprocess(vertices1, vertices2, tri_indices1, tri_indices2, last_tri_id1, last_tri_id2, R, T, min_distance, p1, p2);
+  details::distancePreprocessOrientedNode(model1, model2, vertices1, vertices2, tri_indices1, tri_indices2, 0, 0, R, T, request, *result);
 }
 
 void MeshDistanceTraversalNodeRSS::postprocess()
 {
-  distance_postprocess(R, T, p2);
+  details::distancePostprocessOrientedNode(model1, model2, tf1, request, *result);
 }
 
 FCL_REAL MeshDistanceTraversalNodeRSS::BVTesting(int b1, int b2) const
@@ -541,23 +539,22 @@ void MeshDistanceTraversalNodeRSS::leafTesting(int b1, int b2) const
 {
   details::meshDistanceOrientedNodeLeafTesting(b1, b2, model1, model2, vertices1, vertices2, tri_indices1, tri_indices2, 
                                                R, T, enable_statistics, num_leaf_tests, 
-                                               p1, p2, last_tri_id1, last_tri_id2, min_distance);
+                                               request, *result);
 }
 
 MeshDistanceTraversalNodekIOS::MeshDistanceTraversalNodekIOS() : MeshDistanceTraversalNode<kIOS>()
 {
   R.setIdentity();
-  // default T is 0
 }
 
 void MeshDistanceTraversalNodekIOS::preprocess()
 {
-  distance_preprocess(vertices1, vertices2, tri_indices1, tri_indices2, last_tri_id1, last_tri_id2, R, T, min_distance, p1, p2);
+  details::distancePreprocessOrientedNode(model1, model2, vertices1, vertices2, tri_indices1, tri_indices2, 0, 0, R, T, request, *result);
 }
 
 void MeshDistanceTraversalNodekIOS::postprocess()
 {
-  distance_postprocess(R, T, p2);
+  details::distancePostprocessOrientedNode(model1, model2, tf1, request, *result);
 }
 
 FCL_REAL MeshDistanceTraversalNodekIOS::BVTesting(int b1, int b2) const
@@ -570,23 +567,22 @@ void MeshDistanceTraversalNodekIOS::leafTesting(int b1, int b2) const
 {
   details::meshDistanceOrientedNodeLeafTesting(b1, b2, model1, model2, vertices1, vertices2, tri_indices1, tri_indices2, 
                                                R, T, enable_statistics, num_leaf_tests, 
-                                               p1, p2, last_tri_id1, last_tri_id2, min_distance);
+                                               request, *result);
 }
 
 MeshDistanceTraversalNodeOBBRSS::MeshDistanceTraversalNodeOBBRSS() : MeshDistanceTraversalNode<OBBRSS>()
 {
   R.setIdentity();
-  // default T is 0
 }
 
 void MeshDistanceTraversalNodeOBBRSS::preprocess()
 {
-  distance_preprocess(vertices1, vertices2, tri_indices1, tri_indices2, last_tri_id1, last_tri_id2, R, T, min_distance, p1, p2);
+  details::distancePreprocessOrientedNode(model1, model2, vertices1, vertices2, tri_indices1, tri_indices2, 0, 0, R, T, request, *result);
 }
 
 void MeshDistanceTraversalNodeOBBRSS::postprocess()
 {
-  distance_postprocess(R, T, p2);
+  details::distancePostprocessOrientedNode(model1, model2, tf1, request, *result);
 }
 
 FCL_REAL MeshDistanceTraversalNodeOBBRSS::BVTesting(int b1, int b2) const
@@ -599,7 +595,7 @@ void MeshDistanceTraversalNodeOBBRSS::leafTesting(int b1, int b2) const
 {
   details::meshDistanceOrientedNodeLeafTesting(b1, b2, model1, model2, vertices1, vertices2, tri_indices1, tri_indices2, 
                                                R, T, enable_statistics, num_leaf_tests, 
-                                               p1, p2, last_tri_id1, last_tri_id2, min_distance);
+                                               request, *result);
 }