39 #ifndef COAL_TRAVERSAL_NODE_OCTREE_H
40 #define COAL_TRAVERSAL_NODE_OCTREE_H
59 const GJKSolver* solver;
61 mutable const CollisionRequest* crequest;
62 mutable const DistanceRequest* drequest;
64 mutable CollisionResult* cresult;
65 mutable DistanceResult* dresult;
68 OcTreeSolver(
const GJKSolver* solver_)
76 void OcTreeIntersect(
const OcTree* tree1,
const OcTree* tree2,
77 const Transform3s& tf1,
const Transform3s& tf2,
78 const CollisionRequest& request_,
79 CollisionResult& result_)
const {
83 OcTreeIntersectRecurse(tree1, tree1->getRoot(), tree1->getRootBV(), tree2,
84 tree2->getRoot(), tree2->getRootBV(), tf1, tf2);
88 void OcTreeDistance(
const OcTree* tree1,
const OcTree* tree2,
89 const Transform3s& tf1,
const Transform3s& tf2,
90 const DistanceRequest& request_,
91 DistanceResult& result_)
const {
95 OcTreeDistanceRecurse(tree1, tree1->getRoot(), tree1->getRootBV(), tree2,
96 tree2->getRoot(), tree2->getRootBV(), tf1, tf2);
100 template <
typename BV>
101 void OcTreeMeshIntersect(
const OcTree* tree1,
const BVHModel<BV>* tree2,
102 const Transform3s& tf1,
const Transform3s& tf2,
103 const CollisionRequest& request_,
104 CollisionResult& result_)
const {
105 crequest = &request_;
108 OcTreeMeshIntersectRecurse(tree1, tree1->getRoot(), tree1->getRootBV(),
113 template <
typename BV>
114 void OcTreeMeshDistance(
const OcTree* tree1,
const BVHModel<BV>* tree2,
115 const Transform3s& tf1,
const Transform3s& tf2,
116 const DistanceRequest& request_,
117 DistanceResult& result_)
const {
118 drequest = &request_;
121 OcTreeMeshDistanceRecurse(tree1, tree1->getRoot(), tree1->getRootBV(),
126 template <
typename BV>
127 void MeshOcTreeIntersect(
const BVHModel<BV>* tree1,
const OcTree* tree2,
128 const Transform3s& tf1,
const Transform3s& tf2,
129 const CollisionRequest& request_,
130 CollisionResult& result_)
const
133 crequest = &request_;
136 OcTreeMeshIntersectRecurse(tree2, tree2->getRoot(), tree2->getRootBV(),
141 template <
typename BV>
142 void MeshOcTreeDistance(
const BVHModel<BV>* tree1,
const OcTree* tree2,
143 const Transform3s& tf1,
const Transform3s& tf2,
144 const DistanceRequest& request_,
145 DistanceResult& result_)
const {
146 drequest = &request_;
149 OcTreeMeshDistanceRecurse(tree1, 0, tree2, tree2->getRoot(),
150 tree2->getRootBV(), tf1, tf2);
153 template <
typename BV>
154 void OcTreeHeightFieldIntersect(
155 const OcTree* tree1,
const HeightField<BV>* tree2,
const Transform3s& tf1,
156 const Transform3s& tf2,
const CollisionRequest& request_,
157 CollisionResult& result_,
Scalar& sqrDistLowerBound)
const {
158 crequest = &request_;
161 OcTreeHeightFieldIntersectRecurse(tree1, tree1->getRoot(),
162 tree1->getRootBV(), tree2, 0, tf1, tf2,
166 template <
typename BV>
167 void HeightFieldOcTreeIntersect(
const HeightField<BV>* tree1,
168 const OcTree* tree2,
const Transform3s& tf1,
169 const Transform3s& tf2,
170 const CollisionRequest& request_,
171 CollisionResult& result_,
172 Scalar& sqrDistLowerBound)
const {
173 crequest = &request_;
176 HeightFieldOcTreeIntersectRecurse(tree1, 0, tree2, tree2->getRoot(),
177 tree2->getRootBV(), tf1, tf2,
182 template <
typename S>
183 void OcTreeShapeIntersect(
const OcTree* tree,
const S& s,
184 const Transform3s& tf1,
const Transform3s& tf2,
185 const CollisionRequest& request_,
186 CollisionResult& result_)
const {
187 crequest = &request_;
191 computeBV<AABB>(s, Transform3s(), bv2);
193 convertBV(bv2, tf2, obb2);
194 OcTreeShapeIntersectRecurse(tree, tree->getRoot(), tree->getRootBV(), s,
199 template <
typename S>
200 void ShapeOcTreeIntersect(
const S& s,
const OcTree* tree,
201 const Transform3s& tf1,
const Transform3s& tf2,
202 const CollisionRequest& request_,
203 CollisionResult& result_)
const {
204 crequest = &request_;
208 computeBV<AABB>(s, Transform3s(), bv1);
210 convertBV(bv1, tf1, obb1);
211 OcTreeShapeIntersectRecurse(tree, tree->getRoot(), tree->getRootBV(), s,
216 template <
typename S>
217 void OcTreeShapeDistance(
const OcTree* tree,
const S& s,
218 const Transform3s& tf1,
const Transform3s& tf2,
219 const DistanceRequest& request_,
220 DistanceResult& result_)
const {
221 drequest = &request_;
225 computeBV<AABB>(s, tf2, aabb2);
226 OcTreeShapeDistanceRecurse(tree, tree->getRoot(), tree->getRootBV(), s,
231 template <
typename S>
232 void ShapeOcTreeDistance(
const S& s,
const OcTree* tree,
233 const Transform3s& tf1,
const Transform3s& tf2,
234 const DistanceRequest& request_,
235 DistanceResult& result_)
const {
236 drequest = &request_;
240 computeBV<AABB>(s, tf1, aabb1);
241 OcTreeShapeDistanceRecurse(tree, tree->getRoot(), tree->getRootBV(), s,
246 template <
typename S>
247 bool OcTreeShapeDistanceRecurse(
const OcTree* tree1,
249 const AABB& bv1,
const S& s,
250 const AABB& aabb2,
const Transform3s& tf1,
251 const Transform3s& tf2)
const {
252 if (!tree1->nodeHasChildren(root1)) {
253 if (tree1->isNodeOccupied(root1)) {
259 box.computeLocalAABB();
262 Vec3s p1, p2, normal;
264 &box, box_tf, &s, tf2, this->solver,
265 this->drequest->enable_signed_distance, p1, p2, normal);
267 this->dresult->update(
distance, tree1, &s,
268 (
int)(root1 - tree1->getRoot()),
271 return drequest->isSatisfied(*dresult);
276 if (!tree1->isNodeOccupied(root1))
return false;
278 for (
unsigned int i = 0; i < 8; ++i) {
279 if (tree1->nodeChildExists(root1, i)) {
282 computeChildBV(bv1, i, child_bv);
285 convertBV(child_bv, tf1, aabb1);
286 Scalar d = aabb1.distance(aabb2);
287 if (d < dresult->min_distance) {
288 if (OcTreeShapeDistanceRecurse(tree1, child, child_bv, s, aabb2, tf1,
298 template <
typename S>
299 bool OcTreeShapeIntersectRecurse(
const OcTree* tree1,
301 const AABB& bv1,
const S& s,
const OBB& obb2,
302 const Transform3s& tf1,
303 const Transform3s& tf2)
const {
305 if (!root1)
return false;
311 if (tree1->isNodeFree(root1))
313 else if ((tree1->isNodeUncertain(root1) || s.isUncertain()))
317 convertBV(bv1, tf1, obb1);
319 if (!obb1.overlap(obb2, *crequest, sqrDistLowerBound)) {
326 if (!tree1->nodeHasChildren(root1)) {
327 assert(tree1->isNodeOccupied(root1));
333 box.computeLocalAABB();
336 bool contactNotAdded =
337 (cresult->numContacts() >= crequest->num_max_contacts);
338 std::size_t ncontact = ShapeShapeCollide<Box, S>(
339 &box, box_tf, &s, tf2, solver, *crequest, *cresult);
340 assert(ncontact == 0 || ncontact == 1);
341 if (!contactNotAdded && ncontact == 1) {
343 const Contact& c = cresult->getContact(cresult->numContacts() - 1);
345 cresult->numContacts() - 1,
346 Contact(tree1, c.o2,
static_cast<int>(root1 - tree1->getRoot()),
347 c.b2, c.pos, c.normal, c.penetration_depth));
352 return crequest->isSatisfied(*cresult);
355 for (
unsigned int i = 0; i < 8; ++i) {
356 if (tree1->nodeChildExists(root1, i)) {
359 computeChildBV(bv1, i, child_bv);
361 if (OcTreeShapeIntersectRecurse(tree1, child, child_bv, s, obb2, tf1,
370 template <
typename BV>
371 bool OcTreeMeshDistanceRecurse(
const OcTree* tree1,
373 const AABB& bv1,
const BVHModel<BV>* tree2,
374 unsigned int root2,
const Transform3s& tf1,
375 const Transform3s& tf2)
const {
376 if (!tree1->nodeHasChildren(root1) && tree2->getBV(root2).isLeaf()) {
377 if (tree1->isNodeOccupied(root1)) {
383 box.computeLocalAABB();
386 size_t primitive_id =
387 static_cast<size_t>(tree2->getBV(root2).primitiveId());
388 const Triangle& tri_id = (*(tree2->tri_indices))[primitive_id];
389 const TriangleP tri((*(tree2->vertices))[tri_id[0]],
390 (*(tree2->vertices))[tri_id[1]],
391 (*(tree2->vertices))[tri_id[2]]);
393 Vec3s p1, p2, normal;
394 const Scalar distance = internal::ShapeShapeDistance<Box, TriangleP>(
395 &box, box_tf, &tri, tf2, this->solver,
396 this->drequest->enable_signed_distance, p1, p2, normal);
398 this->dresult->update(
distance, tree1, tree2,
399 (
int)(root1 - tree1->getRoot()),
400 static_cast<int>(primitive_id), p1, p2, normal);
402 return this->drequest->isSatisfied(*dresult);
407 if (!tree1->isNodeOccupied(root1))
return false;
409 if (tree2->getBV(root2).isLeaf() ||
410 (tree1->nodeHasChildren(root1) &&
411 (bv1.size() > tree2->getBV(root2).bv.size()))) {
412 for (
unsigned int i = 0; i < 8; ++i) {
413 if (tree1->nodeChildExists(root1, i)) {
416 computeChildBV(bv1, i, child_bv);
420 convertBV(child_bv, tf1, aabb1);
421 convertBV(tree2->getBV(root2).bv, tf2, aabb2);
422 d = aabb1.distance(aabb2);
424 if (d < dresult->min_distance) {
425 if (OcTreeMeshDistanceRecurse(tree1, child, child_bv, tree2, root2,
434 convertBV(bv1, tf1, aabb1);
435 unsigned int child = (
unsigned int)tree2->getBV(root2).leftChild();
436 convertBV(tree2->getBV(child).bv, tf2, aabb2);
437 d = aabb1.distance(aabb2);
439 if (d < dresult->min_distance) {
440 if (OcTreeMeshDistanceRecurse(tree1, root1, bv1, tree2, child, tf1,
445 child = (
unsigned int)tree2->getBV(root2).rightChild();
446 convertBV(tree2->getBV(child).bv, tf2, aabb2);
447 d = aabb1.distance(aabb2);
449 if (d < dresult->min_distance) {
450 if (OcTreeMeshDistanceRecurse(tree1, root1, bv1, tree2, child, tf1,
460 template <
typename BV>
461 bool OcTreeMeshIntersectRecurse(
const OcTree* tree1,
463 const AABB& bv1,
const BVHModel<BV>* tree2,
464 unsigned int root2,
const Transform3s& tf1,
465 const Transform3s& tf2)
const {
471 if (!root1)
return false;
472 BVNode<BV>
const& bvn2 = tree2->getBV(root2);
478 if (tree1->isNodeFree(root1))
480 else if ((tree1->isNodeUncertain(root1) || tree2->isUncertain()))
484 convertBV(bv1, tf1, obb1);
485 convertBV(bvn2.bv, tf2, obb2);
487 if (!obb1.overlap(obb2, *crequest, sqrDistLowerBound)) {
495 if (!tree1->nodeHasChildren(root1) && bvn2.isLeaf()) {
496 assert(tree1->isNodeOccupied(root1));
501 box.computeLocalAABB();
504 size_t primitive_id =
static_cast<size_t>(bvn2.primitiveId());
505 const Triangle& tri_id = (*(tree2->tri_indices))[primitive_id];
506 const TriangleP tri((*(tree2->vertices))[tri_id[0]],
507 (*(tree2->vertices))[tri_id[1]],
508 (*(tree2->vertices))[tri_id[2]]);
515 const bool compute_penetration = this->crequest->enable_contact ||
516 (this->crequest->security_margin < 0);
517 Vec3s c1, c2, normal;
518 const Scalar distance = internal::ShapeShapeDistance<Box, TriangleP>(
519 &box, box_tf, &tri, tf2, this->solver, compute_penetration, c1, c2,
521 const Scalar distToCollision =
distance - this->crequest->security_margin;
524 *(this->crequest), *(this->cresult), distToCollision, c1, c2, normal);
526 if (cresult->numContacts() < crequest->num_max_contacts) {
527 if (distToCollision <= crequest->collision_distance_threshold) {
528 cresult->addContact(Contact(
529 tree1, tree2, (
int)(root1 - tree1->getRoot()),
530 static_cast<int>(primitive_id), c1, c2, normal,
distance));
533 return crequest->isSatisfied(*cresult);
538 (tree1->nodeHasChildren(root1) && (bv1.size() > bvn2.bv.size()))) {
539 for (
unsigned int i = 0; i < 8; ++i) {
540 if (tree1->nodeChildExists(root1, i)) {
543 computeChildBV(bv1, i, child_bv);
545 if (OcTreeMeshIntersectRecurse(tree1, child, child_bv, tree2, root2,
551 if (OcTreeMeshIntersectRecurse(tree1, root1, bv1, tree2,
552 (
unsigned int)bvn2.leftChild(), tf1, tf2))
555 if (OcTreeMeshIntersectRecurse(tree1, root1, bv1, tree2,
556 (
unsigned int)bvn2.rightChild(), tf1, tf2))
564 template <
typename BV>
565 bool OcTreeHeightFieldIntersectRecurse(
567 const HeightField<BV>* tree2,
unsigned int root2,
const Transform3s& tf1,
568 const Transform3s& tf2,
Scalar& sqrDistLowerBound)
const {
574 if (!root1)
return false;
575 HFNode<BV>
const& bvn2 = tree2->getBV(root2);
581 if (tree1->isNodeFree(root1))
583 else if ((tree1->isNodeUncertain(root1) || tree2->isUncertain()))
587 convertBV(bv1, tf1, obb1);
588 convertBV(bvn2.bv, tf2, obb2);
589 Scalar sqrDistLowerBound_;
590 if (!obb1.overlap(obb2, *crequest, sqrDistLowerBound_)) {
591 if (sqrDistLowerBound_ < sqrDistLowerBound)
592 sqrDistLowerBound = sqrDistLowerBound_;
600 if (!tree1->nodeHasChildren(root1) && bvn2.isLeaf()) {
601 assert(tree1->isNodeOccupied(root1));
606 box.computeLocalAABB();
609 typedef Convex<Triangle> ConvexTriangle;
610 ConvexTriangle convex1, convex2;
611 int convex1_active_faces, convex2_active_faces;
612 details::buildConvexTriangles(bvn2, *tree2, convex1, convex1_active_faces,
613 convex2, convex2_active_faces);
615 convex1.computeLocalAABB();
616 convex2.computeLocalAABB();
619 Vec3s c1, c2, normal, normal_top;
621 bool hfield_witness_is_on_bin_side;
623 bool collision = details::shapeDistance<Triangle, Box, 0>(
624 solver, *crequest, convex1, convex1_active_faces, convex2,
625 convex2_active_faces, tf2, box, box_tf,
distance, c2, c1, normal,
626 normal_top, hfield_witness_is_on_bin_side);
629 distance - crequest->security_margin * (normal_top.dot(normal));
631 if (distToCollision <= crequest->collision_distance_threshold) {
632 sqrDistLowerBound = 0;
633 if (crequest->num_max_contacts > cresult->numContacts()) {
634 if (normal_top.isApprox(normal) &&
635 (collision || !hfield_witness_is_on_bin_side)) {
637 Contact(tree1, tree2, (
int)(root1 - tree1->getRoot()),
642 sqrDistLowerBound = distToCollision * distToCollision;
647 *crequest, *cresult, distToCollision, c1, c2, -normal);
649 assert(cresult->isCollision() || sqrDistLowerBound > 0);
650 return crequest->isSatisfied(*cresult);
655 (tree1->nodeHasChildren(root1) && (bv1.size() > bvn2.bv.size()))) {
656 for (
unsigned int i = 0; i < 8; ++i) {
657 if (tree1->nodeChildExists(root1, i)) {
660 computeChildBV(bv1, i, child_bv);
662 if (OcTreeHeightFieldIntersectRecurse(tree1, child, child_bv, tree2,
669 if (OcTreeHeightFieldIntersectRecurse(tree1, root1, bv1, tree2,
670 (
unsigned int)bvn2.leftChild(), tf1,
671 tf2, sqrDistLowerBound))
674 if (OcTreeHeightFieldIntersectRecurse(tree1, root1, bv1, tree2,
675 (
unsigned int)bvn2.rightChild(),
676 tf1, tf2, sqrDistLowerBound))
684 template <
typename BV>
685 bool HeightFieldOcTreeIntersectRecurse(
686 const HeightField<BV>* tree1,
unsigned int root1,
const OcTree* tree2,
688 const Transform3s& tf2,
Scalar& sqrDistLowerBound)
const {
694 if (!root2)
return false;
695 HFNode<BV>
const& bvn1 = tree1->getBV(root1);
701 if (tree2->isNodeFree(root2))
703 else if ((tree2->isNodeUncertain(root2) || tree1->isUncertain()))
707 convertBV(bvn1.bv, tf1, obb1);
708 convertBV(bv2, tf2, obb2);
709 Scalar sqrDistLowerBound_;
710 if (!obb2.overlap(obb1, *crequest, sqrDistLowerBound_)) {
711 if (sqrDistLowerBound_ < sqrDistLowerBound)
712 sqrDistLowerBound = sqrDistLowerBound_;
720 if (!tree2->nodeHasChildren(root2) && bvn1.isLeaf()) {
721 assert(tree2->isNodeOccupied(root2));
726 box.computeLocalAABB();
729 typedef Convex<Triangle> ConvexTriangle;
730 ConvexTriangle convex1, convex2;
731 int convex1_active_faces, convex2_active_faces;
732 details::buildConvexTriangles(bvn1, *tree1, convex1, convex1_active_faces,
733 convex2, convex2_active_faces);
735 convex1.computeLocalAABB();
736 convex2.computeLocalAABB();
739 Vec3s c1, c2, normal, normal_top;
741 bool hfield_witness_is_on_bin_side;
743 bool collision = details::shapeDistance<Triangle, Box, 0>(
744 solver, *crequest, convex1, convex1_active_faces, convex2,
745 convex2_active_faces, tf1, box, box_tf,
distance, c1, c2, normal,
746 normal_top, hfield_witness_is_on_bin_side);
749 distance - crequest->security_margin * (normal_top.dot(normal));
751 if (distToCollision <= crequest->collision_distance_threshold) {
752 sqrDistLowerBound = 0;
753 if (crequest->num_max_contacts > cresult->numContacts()) {
754 if (normal_top.isApprox(normal) &&
755 (collision || !hfield_witness_is_on_bin_side)) {
756 cresult->addContact(Contact(tree1, tree2, (
int)
Contact::NONE,
757 (int)(root2 - tree2->getRoot()), c1, c2,
762 sqrDistLowerBound = distToCollision * distToCollision;
767 *crequest, *cresult, distToCollision, c1, c2, normal);
769 assert(cresult->isCollision() || sqrDistLowerBound > 0);
770 return crequest->isSatisfied(*cresult);
775 (tree2->nodeHasChildren(root2) && (bv2.size() > bvn1.bv.size()))) {
776 for (
unsigned int i = 0; i < 8; ++i) {
777 if (tree2->nodeChildExists(root2, i)) {
780 computeChildBV(bv2, i, child_bv);
782 if (HeightFieldOcTreeIntersectRecurse(tree1, root1, tree2, child,
789 if (HeightFieldOcTreeIntersectRecurse(
790 tree1, (
unsigned int)bvn1.leftChild(), tree2, root2, bv2, tf1,
791 tf2, sqrDistLowerBound))
794 if (HeightFieldOcTreeIntersectRecurse(
795 tree1, (
unsigned int)bvn1.rightChild(), tree2, root2, bv2, tf1,
796 tf2, sqrDistLowerBound))
803 bool OcTreeDistanceRecurse(
const OcTree* tree1,
807 const Transform3s& tf1,
808 const Transform3s& tf2)
const {
809 if (!tree1->nodeHasChildren(root1) && !tree2->nodeHasChildren(root2)) {
810 if (tree1->isNodeOccupied(root1) && tree2->isNodeOccupied(root2)) {
812 Transform3s box1_tf, box2_tf;
817 box1.computeLocalAABB();
818 box2.computeLocalAABB();
821 Vec3s p1, p2, normal;
823 &box1, box1_tf, &box2, box2_tf, this->solver,
824 this->drequest->enable_signed_distance, p1, p2, normal);
826 this->dresult->update(
distance, tree1, tree2,
827 (
int)(root1 - tree1->getRoot()),
828 (
int)(root2 - tree2->getRoot()), p1, p2, normal);
830 return drequest->isSatisfied(*dresult);
835 if (!tree1->isNodeOccupied(root1) || !tree2->isNodeOccupied(root2))
838 if (!tree2->nodeHasChildren(root2) ||
839 (tree1->nodeHasChildren(root1) && (bv1.size() > bv2.size()))) {
840 for (
unsigned int i = 0; i < 8; ++i) {
841 if (tree1->nodeChildExists(root1, i)) {
844 computeChildBV(bv1, i, child_bv);
848 convertBV(bv1, tf1, aabb1);
849 convertBV(bv2, tf2, aabb2);
850 d = aabb1.distance(aabb2);
852 if (d < dresult->min_distance) {
853 if (OcTreeDistanceRecurse(tree1, child, child_bv, tree2, root2, bv2,
860 for (
unsigned int i = 0; i < 8; ++i) {
861 if (tree2->nodeChildExists(root2, i)) {
864 computeChildBV(bv2, i, child_bv);
868 convertBV(bv1, tf1, aabb1);
869 convertBV(bv2, tf2, aabb2);
870 d = aabb1.distance(aabb2);
872 if (d < dresult->min_distance) {
873 if (OcTreeDistanceRecurse(tree1, root1, bv1, tree2, child, child_bv,
884 bool OcTreeIntersectRecurse(
const OcTree* tree1,
888 const Transform3s& tf1,
889 const Transform3s& tf2)
const {
891 if (!root1)
return false;
892 if (!root2)
return false;
898 if (tree1->isNodeFree(root1) || tree2->isNodeFree(root2))
900 else if ((tree1->isNodeUncertain(root1) || tree2->isNodeUncertain(root2)))
904 (!tree1->nodeHasChildren(root1) && !tree2->nodeHasChildren(root2));
905 if (!bothAreLeaves || !crequest->enable_contact) {
907 convertBV(bv1, tf1, obb1);
908 convertBV(bv2, tf2, obb2);
910 if (!obb1.overlap(obb2, *crequest, sqrDistLowerBound)) {
911 if (cresult->distance_lower_bound > 0 &&
913 cresult->distance_lower_bound * cresult->distance_lower_bound)
914 cresult->distance_lower_bound =
915 sqrt(sqrDistLowerBound) - crequest->security_margin;
918 if (crequest->enable_contact) {
919 if (cresult->numContacts() < crequest->num_max_contacts)
921 Contact(tree1, tree2,
static_cast<int>(root1 - tree1->getRoot()),
922 static_cast<int>(root2 - tree2->getRoot())));
923 return crequest->isSatisfied(*cresult);
929 assert(tree1->isNodeOccupied(root1) && tree2->isNodeOccupied(root2));
932 Transform3s box1_tf, box2_tf;
936 if (this->solver->gjk_initial_guess ==
938 box1.computeLocalAABB();
939 box2.computeLocalAABB();
947 const bool compute_penetration = (this->crequest->enable_contact ||
948 (this->crequest->security_margin < 0));
949 Vec3s c1, c2, normal;
951 &box1, box1_tf, &box2, box2_tf, this->solver, compute_penetration, c1,
954 const Scalar distToCollision =
distance - this->crequest->security_margin;
957 *(this->crequest), *(this->cresult), distToCollision, c1, c2, normal);
959 if (this->cresult->numContacts() < this->crequest->num_max_contacts) {
960 if (distToCollision <= this->crequest->collision_distance_threshold)
961 this->cresult->addContact(
962 Contact(tree1, tree2,
static_cast<int>(root1 - tree1->getRoot()),
963 static_cast<int>(root2 - tree2->getRoot()), c1, c2,
967 return crequest->isSatisfied(*cresult);
971 if (!tree2->nodeHasChildren(root2) ||
972 (tree1->nodeHasChildren(root1) && (bv1.size() > bv2.size()))) {
973 for (
unsigned int i = 0; i < 8; ++i) {
974 if (tree1->nodeChildExists(root1, i)) {
977 computeChildBV(bv1, i, child_bv);
979 if (OcTreeIntersectRecurse(tree1, child, child_bv, tree2, root2, bv2,
985 for (
unsigned int i = 0; i < 8; ++i) {
986 if (tree2->nodeChildExists(root2, i)) {
989 computeChildBV(bv2, i, child_bv);
991 if (OcTreeIntersectRecurse(tree1, root1, bv1, tree2, child, child_bv,
1007 :
public CollisionTraversalNodeBase {
1009 OcTreeCollisionTraversalNode(
const CollisionRequest& request)
1010 : CollisionTraversalNodeBase(request) {
1017 bool BVDisjoints(
unsigned,
unsigned,
Scalar&)
const {
return false; }
1019 void leafCollides(
unsigned,
unsigned,
Scalar& sqrDistLowerBound)
const {
1020 otsolver->OcTreeIntersect(model1, model2, tf1, tf2, request, *result);
1021 sqrDistLowerBound = std::max((
Scalar)0, result->distance_lower_bound);
1022 sqrDistLowerBound *= sqrDistLowerBound;
1025 const OcTree* model1;
1026 const OcTree* model2;
1028 Transform3s tf1, tf2;
1030 const OcTreeSolver* otsolver;
1034 template <
typename S>
1035 class COAL_DLLAPI ShapeOcTreeCollisionTraversalNode
1036 :
public CollisionTraversalNodeBase {
1038 ShapeOcTreeCollisionTraversalNode(
const CollisionRequest& request)
1039 : CollisionTraversalNodeBase(request) {
1046 bool BVDisjoints(
unsigned int,
unsigned int,
Scalar&)
const {
return false; }
1048 void leafCollides(
unsigned int,
unsigned int,
1049 Scalar& sqrDistLowerBound)
const {
1050 otsolver->OcTreeShapeIntersect(model2, *model1, tf2, tf1, request, *result);
1051 sqrDistLowerBound = std::max((
Scalar)0, result->distance_lower_bound);
1052 sqrDistLowerBound *= sqrDistLowerBound;
1056 const OcTree* model2;
1058 Transform3s tf1, tf2;
1060 const OcTreeSolver* otsolver;
1065 template <
typename S>
1066 class COAL_DLLAPI OcTreeShapeCollisionTraversalNode
1067 :
public CollisionTraversalNodeBase {
1069 OcTreeShapeCollisionTraversalNode(
const CollisionRequest& request)
1070 : CollisionTraversalNodeBase(request) {
1077 bool BVDisjoints(
unsigned int,
unsigned int,
coal::Scalar&)
const {
1081 void leafCollides(
unsigned int,
unsigned int,
1082 Scalar& sqrDistLowerBound)
const {
1083 otsolver->OcTreeShapeIntersect(model1, *model2, tf1, tf2, request, *result);
1084 sqrDistLowerBound = std::max((
Scalar)0, result->distance_lower_bound);
1085 sqrDistLowerBound *= sqrDistLowerBound;
1088 const OcTree* model1;
1091 Transform3s tf1, tf2;
1093 const OcTreeSolver* otsolver;
1097 template <
typename BV>
1098 class COAL_DLLAPI MeshOcTreeCollisionTraversalNode
1099 :
public CollisionTraversalNodeBase {
1101 MeshOcTreeCollisionTraversalNode(
const CollisionRequest& request)
1102 : CollisionTraversalNodeBase(request) {
1109 bool BVDisjoints(
unsigned int,
unsigned int,
Scalar&)
const {
return false; }
1111 void leafCollides(
unsigned int,
unsigned int,
1112 Scalar& sqrDistLowerBound)
const {
1113 otsolver->OcTreeMeshIntersect(model2, model1, tf2, tf1, request, *result);
1114 sqrDistLowerBound = std::max((
Scalar)0, result->distance_lower_bound);
1115 sqrDistLowerBound *= sqrDistLowerBound;
1118 const BVHModel<BV>* model1;
1119 const OcTree* model2;
1121 Transform3s tf1, tf2;
1123 const OcTreeSolver* otsolver;
1127 template <
typename BV>
1128 class COAL_DLLAPI OcTreeMeshCollisionTraversalNode
1129 :
public CollisionTraversalNodeBase {
1131 OcTreeMeshCollisionTraversalNode(
const CollisionRequest& request)
1132 : CollisionTraversalNodeBase(request) {
1139 bool BVDisjoints(
unsigned int,
unsigned int,
Scalar&)
const {
return false; }
1141 void leafCollides(
unsigned int,
unsigned int,
1142 Scalar& sqrDistLowerBound)
const {
1143 otsolver->OcTreeMeshIntersect(model1, model2, tf1, tf2, request, *result);
1144 sqrDistLowerBound = std::max((
Scalar)0, result->distance_lower_bound);
1145 sqrDistLowerBound *= sqrDistLowerBound;
1148 const OcTree* model1;
1149 const BVHModel<BV>* model2;
1151 Transform3s tf1, tf2;
1153 const OcTreeSolver* otsolver;
1157 template <
typename BV>
1158 class COAL_DLLAPI OcTreeHeightFieldCollisionTraversalNode
1159 :
public CollisionTraversalNodeBase {
1161 OcTreeHeightFieldCollisionTraversalNode(
const CollisionRequest& request)
1162 : CollisionTraversalNodeBase(request) {
1169 bool BVDisjoints(
unsigned int,
unsigned int,
Scalar&)
const {
return false; }
1171 void leafCollides(
unsigned int,
unsigned int,
1172 Scalar& sqrDistLowerBound)
const {
1173 otsolver->OcTreeHeightFieldIntersect(model1, model2, tf1, tf2, request,
1174 *result, sqrDistLowerBound);
1177 const OcTree* model1;
1178 const HeightField<BV>* model2;
1180 Transform3s tf1, tf2;
1182 const OcTreeSolver* otsolver;
1186 template <
typename BV>
1187 class COAL_DLLAPI HeightFieldOcTreeCollisionTraversalNode
1188 :
public CollisionTraversalNodeBase {
1190 HeightFieldOcTreeCollisionTraversalNode(
const CollisionRequest& request)
1191 : CollisionTraversalNodeBase(request) {
1198 bool BVDisjoints(
unsigned int,
unsigned int,
Scalar&)
const {
return false; }
1200 void leafCollides(
unsigned int,
unsigned int,
1201 Scalar& sqrDistLowerBound)
const {
1202 otsolver->HeightFieldOcTreeIntersect(model1, model2, tf1, tf2, request,
1203 *result, sqrDistLowerBound);
1206 const HeightField<BV>* model1;
1207 const OcTree* model2;
1209 Transform3s tf1, tf2;
1211 const OcTreeSolver* otsolver;
1221 :
public DistanceTraversalNodeBase {
1223 OcTreeDistanceTraversalNode() {
1230 Scalar BVDistanceLowerBound(
unsigned,
unsigned)
const {
return -1; }
1232 bool BVDistanceLowerBound(
unsigned,
unsigned,
Scalar&)
const {
return false; }
1234 void leafComputeDistance(
unsigned,
unsigned int)
const {
1235 otsolver->OcTreeDistance(model1, model2, tf1, tf2, request, *result);
1238 const OcTree* model1;
1239 const OcTree* model2;
1241 const OcTreeSolver* otsolver;
1245 template <
typename Shape>
1246 class COAL_DLLAPI ShapeOcTreeDistanceTraversalNode
1247 :
public DistanceTraversalNodeBase {
1249 ShapeOcTreeDistanceTraversalNode() {
1256 Scalar BVDistanceLowerBound(
unsigned int,
unsigned int)
const {
return -1; }
1258 void leafComputeDistance(
unsigned int,
unsigned int)
const {
1259 otsolver->OcTreeShapeDistance(model2, *model1, tf2, tf1, request, *result);
1262 const Shape* model1;
1263 const OcTree* model2;
1265 const OcTreeSolver* otsolver;
1269 template <
typename Shape>
1270 class COAL_DLLAPI OcTreeShapeDistanceTraversalNode
1271 :
public DistanceTraversalNodeBase {
1273 OcTreeShapeDistanceTraversalNode() {
1280 Scalar BVDistanceLowerBound(
unsigned int,
unsigned int)
const {
return -1; }
1282 void leafComputeDistance(
unsigned int,
unsigned int)
const {
1283 otsolver->OcTreeShapeDistance(model1, *model2, tf1, tf2, request, *result);
1286 const OcTree* model1;
1287 const Shape* model2;
1289 const OcTreeSolver* otsolver;
1293 template <
typename BV>
1295 :
public DistanceTraversalNodeBase {
1297 MeshOcTreeDistanceTraversalNode() {
1304 Scalar BVDistanceLowerBound(
unsigned int,
unsigned int)
const {
return -1; }
1306 void leafComputeDistance(
unsigned int,
unsigned int)
const {
1307 otsolver->OcTreeMeshDistance(model2, model1, tf2, tf1, request, *result);
1310 const BVHModel<BV>* model1;
1311 const OcTree* model2;
1313 const OcTreeSolver* otsolver;
1317 template <
typename BV>
1319 :
public DistanceTraversalNodeBase {
1321 OcTreeMeshDistanceTraversalNode() {
1328 Scalar BVDistanceLowerBound(
unsigned int,
unsigned int)
const {
return -1; }
1330 void leafComputeDistance(
unsigned int,
unsigned int)
const {
1331 otsolver->OcTreeMeshDistance(model1, model2, tf1, tf2, request, *result);
1334 const OcTree* model1;
1335 const BVHModel<BV>* model2;
1337 const OcTreeSolver* otsolver;
octomap::OcTreeNode OcTreeNode
Definition: octree.h:63
#define COAL_DLLAPI
Definition: config.hh:88
Scalar distance(const Matrix3s &R0, const Vec3s &T0, const kIOS &b1, const kIOS &b2, Vec3s *P=NULL, Vec3s *Q=NULL)
Approximate distance between two kIOS bounding volumes.
void updateDistanceLowerBoundFromBV(const CollisionRequest &, CollisionResult &res, const Scalar sqrDistLowerBound)
Definition: collision_data.h:1171
void updateDistanceLowerBoundFromLeaf(const CollisionRequest &, CollisionResult &res, const Scalar &distance, const Vec3s &p0, const Vec3s &p1, const Vec3s &normal)
Definition: collision_data.h:1180
Main namespace.
Definition: broadphase_bruteforce.h:44
void constructBox(const AABB &bv, Box &box, Transform3s &tf)
construct a box shape (with a configuration) from a given bounding volume
Eigen::Matrix< Scalar, 3, 1 > Vec3s
Definition: data_types.h:70
double Scalar
Definition: data_types.h:68
@ BoundingVolumeGuess
Definition: data_types.h:105
static const int NONE
invalid contact primitive information
Definition: collision_data.h:1085