Home | History | Annotate | Download | only in BVH
      1 // This file is part of Eigen, a lightweight C++ template library
      2 // for linear algebra.
      3 //
      4 // Copyright (C) 2009 Ilya Baran <ibaran (at) mit.edu>
      5 //
      6 // This Source Code Form is subject to the terms of the Mozilla
      7 // Public License v. 2.0. If a copy of the MPL was not distributed
      8 // with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
      9 
     10 #ifndef EIGEN_BVALGORITHMS_H
     11 #define EIGEN_BVALGORITHMS_H
     12 
     13 namespace Eigen {
     14 
     15 namespace internal {
     16 
     17 #ifndef EIGEN_PARSED_BY_DOXYGEN
     18 template<typename BVH, typename Intersector>
     19 bool intersect_helper(const BVH &tree, Intersector &intersector, typename BVH::Index root)
     20 {
     21   typedef typename BVH::Index Index;
     22   typedef typename BVH::VolumeIterator VolIter;
     23   typedef typename BVH::ObjectIterator ObjIter;
     24 
     25   VolIter vBegin = VolIter(), vEnd = VolIter();
     26   ObjIter oBegin = ObjIter(), oEnd = ObjIter();
     27 
     28   std::vector<Index> todo(1, root);
     29 
     30   while(!todo.empty()) {
     31     tree.getChildren(todo.back(), vBegin, vEnd, oBegin, oEnd);
     32     todo.pop_back();
     33 
     34     for(; vBegin != vEnd; ++vBegin) //go through child volumes
     35       if(intersector.intersectVolume(tree.getVolume(*vBegin)))
     36         todo.push_back(*vBegin);
     37 
     38     for(; oBegin != oEnd; ++oBegin) //go through child objects
     39       if(intersector.intersectObject(*oBegin))
     40         return true; //intersector said to stop query
     41   }
     42   return false;
     43 }
     44 #endif //not EIGEN_PARSED_BY_DOXYGEN
     45 
     46 template<typename Volume1, typename Object1, typename Object2, typename Intersector>
     47 struct intersector_helper1
     48 {
     49   intersector_helper1(const Object2 &inStored, Intersector &in) : stored(inStored), intersector(in) {}
     50   bool intersectVolume(const Volume1 &vol) { return intersector.intersectVolumeObject(vol, stored); }
     51   bool intersectObject(const Object1 &obj) { return intersector.intersectObjectObject(obj, stored); }
     52   Object2 stored;
     53   Intersector &intersector;
     54 private:
     55   intersector_helper1& operator=(const intersector_helper1&);
     56 };
     57 
     58 template<typename Volume2, typename Object2, typename Object1, typename Intersector>
     59 struct intersector_helper2
     60 {
     61   intersector_helper2(const Object1 &inStored, Intersector &in) : stored(inStored), intersector(in) {}
     62   bool intersectVolume(const Volume2 &vol) { return intersector.intersectObjectVolume(stored, vol); }
     63   bool intersectObject(const Object2 &obj) { return intersector.intersectObjectObject(stored, obj); }
     64   Object1 stored;
     65   Intersector &intersector;
     66 private:
     67   intersector_helper2& operator=(const intersector_helper2&);
     68 };
     69 
     70 } // end namespace internal
     71 
     72 /**  Given a BVH, runs the query encapsulated by \a intersector.
     73   *  The Intersector type must provide the following members: \code
     74      bool intersectVolume(const BVH::Volume &volume) //returns true if volume intersects the query
     75      bool intersectObject(const BVH::Object &object) //returns true if the search should terminate immediately
     76   \endcode
     77   */
     78 template<typename BVH, typename Intersector>
     79 void BVIntersect(const BVH &tree, Intersector &intersector)
     80 {
     81   internal::intersect_helper(tree, intersector, tree.getRootIndex());
     82 }
     83 
     84 /**  Given two BVH's, runs the query on their Cartesian product encapsulated by \a intersector.
     85   *  The Intersector type must provide the following members: \code
     86      bool intersectVolumeVolume(const BVH1::Volume &v1, const BVH2::Volume &v2) //returns true if product of volumes intersects the query
     87      bool intersectVolumeObject(const BVH1::Volume &v1, const BVH2::Object &o2) //returns true if the volume-object product intersects the query
     88      bool intersectObjectVolume(const BVH1::Object &o1, const BVH2::Volume &v2) //returns true if the volume-object product intersects the query
     89      bool intersectObjectObject(const BVH1::Object &o1, const BVH2::Object &o2) //returns true if the search should terminate immediately
     90   \endcode
     91   */
     92 template<typename BVH1, typename BVH2, typename Intersector>
     93 void BVIntersect(const BVH1 &tree1, const BVH2 &tree2, Intersector &intersector) //TODO: tandem descent when it makes sense
     94 {
     95   typedef typename BVH1::Index Index1;
     96   typedef typename BVH2::Index Index2;
     97   typedef internal::intersector_helper1<typename BVH1::Volume, typename BVH1::Object, typename BVH2::Object, Intersector> Helper1;
     98   typedef internal::intersector_helper2<typename BVH2::Volume, typename BVH2::Object, typename BVH1::Object, Intersector> Helper2;
     99   typedef typename BVH1::VolumeIterator VolIter1;
    100   typedef typename BVH1::ObjectIterator ObjIter1;
    101   typedef typename BVH2::VolumeIterator VolIter2;
    102   typedef typename BVH2::ObjectIterator ObjIter2;
    103 
    104   VolIter1 vBegin1 = VolIter1(), vEnd1 = VolIter1();
    105   ObjIter1 oBegin1 = ObjIter1(), oEnd1 = ObjIter1();
    106   VolIter2 vBegin2 = VolIter2(), vEnd2 = VolIter2(), vCur2 = VolIter2();
    107   ObjIter2 oBegin2 = ObjIter2(), oEnd2 = ObjIter2(), oCur2 = ObjIter2();
    108 
    109   std::vector<std::pair<Index1, Index2> > todo(1, std::make_pair(tree1.getRootIndex(), tree2.getRootIndex()));
    110 
    111   while(!todo.empty()) {
    112     tree1.getChildren(todo.back().first, vBegin1, vEnd1, oBegin1, oEnd1);
    113     tree2.getChildren(todo.back().second, vBegin2, vEnd2, oBegin2, oEnd2);
    114     todo.pop_back();
    115 
    116     for(; vBegin1 != vEnd1; ++vBegin1) { //go through child volumes of first tree
    117       const typename BVH1::Volume &vol1 = tree1.getVolume(*vBegin1);
    118       for(vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { //go through child volumes of second tree
    119         if(intersector.intersectVolumeVolume(vol1, tree2.getVolume(*vCur2)))
    120           todo.push_back(std::make_pair(*vBegin1, *vCur2));
    121       }
    122 
    123       for(oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) {//go through child objects of second tree
    124         Helper1 helper(*oCur2, intersector);
    125         if(internal::intersect_helper(tree1, helper, *vBegin1))
    126           return; //intersector said to stop query
    127       }
    128     }
    129 
    130     for(; oBegin1 != oEnd1; ++oBegin1) { //go through child objects of first tree
    131       for(vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { //go through child volumes of second tree
    132         Helper2 helper(*oBegin1, intersector);
    133         if(internal::intersect_helper(tree2, helper, *vCur2))
    134           return; //intersector said to stop query
    135       }
    136 
    137       for(oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) {//go through child objects of second tree
    138         if(intersector.intersectObjectObject(*oBegin1, *oCur2))
    139           return; //intersector said to stop query
    140       }
    141     }
    142   }
    143 }
    144 
    145 namespace internal {
    146 
    147 #ifndef EIGEN_PARSED_BY_DOXYGEN
    148 template<typename BVH, typename Minimizer>
    149 typename Minimizer::Scalar minimize_helper(const BVH &tree, Minimizer &minimizer, typename BVH::Index root, typename Minimizer::Scalar minimum)
    150 {
    151   typedef typename Minimizer::Scalar Scalar;
    152   typedef typename BVH::Index Index;
    153   typedef std::pair<Scalar, Index> QueueElement; //first element is priority
    154   typedef typename BVH::VolumeIterator VolIter;
    155   typedef typename BVH::ObjectIterator ObjIter;
    156 
    157   VolIter vBegin = VolIter(), vEnd = VolIter();
    158   ObjIter oBegin = ObjIter(), oEnd = ObjIter();
    159   std::priority_queue<QueueElement, std::vector<QueueElement>, std::greater<QueueElement> > todo; //smallest is at the top
    160 
    161   todo.push(std::make_pair(Scalar(), root));
    162 
    163   while(!todo.empty()) {
    164     tree.getChildren(todo.top().second, vBegin, vEnd, oBegin, oEnd);
    165     todo.pop();
    166 
    167     for(; oBegin != oEnd; ++oBegin) //go through child objects
    168       minimum = (std::min)(minimum, minimizer.minimumOnObject(*oBegin));
    169 
    170     for(; vBegin != vEnd; ++vBegin) { //go through child volumes
    171       Scalar val = minimizer.minimumOnVolume(tree.getVolume(*vBegin));
    172       if(val < minimum)
    173         todo.push(std::make_pair(val, *vBegin));
    174     }
    175   }
    176 
    177   return minimum;
    178 }
    179 #endif //not EIGEN_PARSED_BY_DOXYGEN
    180 
    181 
    182 template<typename Volume1, typename Object1, typename Object2, typename Minimizer>
    183 struct minimizer_helper1
    184 {
    185   typedef typename Minimizer::Scalar Scalar;
    186   minimizer_helper1(const Object2 &inStored, Minimizer &m) : stored(inStored), minimizer(m) {}
    187   Scalar minimumOnVolume(const Volume1 &vol) { return minimizer.minimumOnVolumeObject(vol, stored); }
    188   Scalar minimumOnObject(const Object1 &obj) { return minimizer.minimumOnObjectObject(obj, stored); }
    189   Object2 stored;
    190   Minimizer &minimizer;
    191 private:
    192   minimizer_helper1& operator=(const minimizer_helper1&) {}
    193 };
    194 
    195 template<typename Volume2, typename Object2, typename Object1, typename Minimizer>
    196 struct minimizer_helper2
    197 {
    198   typedef typename Minimizer::Scalar Scalar;
    199   minimizer_helper2(const Object1 &inStored, Minimizer &m) : stored(inStored), minimizer(m) {}
    200   Scalar minimumOnVolume(const Volume2 &vol) { return minimizer.minimumOnObjectVolume(stored, vol); }
    201   Scalar minimumOnObject(const Object2 &obj) { return minimizer.minimumOnObjectObject(stored, obj); }
    202   Object1 stored;
    203   Minimizer &minimizer;
    204 private:
    205   minimizer_helper2& operator=(const minimizer_helper2&);
    206 };
    207 
    208 } // end namespace internal
    209 
    210 /**  Given a BVH, runs the query encapsulated by \a minimizer.
    211   *  \returns the minimum value.
    212   *  The Minimizer type must provide the following members: \code
    213      typedef Scalar //the numeric type of what is being minimized--not necessarily the Scalar type of the BVH (if it has one)
    214      Scalar minimumOnVolume(const BVH::Volume &volume)
    215      Scalar minimumOnObject(const BVH::Object &object)
    216   \endcode
    217   */
    218 template<typename BVH, typename Minimizer>
    219 typename Minimizer::Scalar BVMinimize(const BVH &tree, Minimizer &minimizer)
    220 {
    221   return internal::minimize_helper(tree, minimizer, tree.getRootIndex(), (std::numeric_limits<typename Minimizer::Scalar>::max)());
    222 }
    223 
    224 /**  Given two BVH's, runs the query on their cartesian product encapsulated by \a minimizer.
    225   *  \returns the minimum value.
    226   *  The Minimizer type must provide the following members: \code
    227      typedef Scalar //the numeric type of what is being minimized--not necessarily the Scalar type of the BVH (if it has one)
    228      Scalar minimumOnVolumeVolume(const BVH1::Volume &v1, const BVH2::Volume &v2)
    229      Scalar minimumOnVolumeObject(const BVH1::Volume &v1, const BVH2::Object &o2)
    230      Scalar minimumOnObjectVolume(const BVH1::Object &o1, const BVH2::Volume &v2)
    231      Scalar minimumOnObjectObject(const BVH1::Object &o1, const BVH2::Object &o2)
    232   \endcode
    233   */
    234 template<typename BVH1, typename BVH2, typename Minimizer>
    235 typename Minimizer::Scalar BVMinimize(const BVH1 &tree1, const BVH2 &tree2, Minimizer &minimizer)
    236 {
    237   typedef typename Minimizer::Scalar Scalar;
    238   typedef typename BVH1::Index Index1;
    239   typedef typename BVH2::Index Index2;
    240   typedef internal::minimizer_helper1<typename BVH1::Volume, typename BVH1::Object, typename BVH2::Object, Minimizer> Helper1;
    241   typedef internal::minimizer_helper2<typename BVH2::Volume, typename BVH2::Object, typename BVH1::Object, Minimizer> Helper2;
    242   typedef std::pair<Scalar, std::pair<Index1, Index2> > QueueElement; //first element is priority
    243   typedef typename BVH1::VolumeIterator VolIter1;
    244   typedef typename BVH1::ObjectIterator ObjIter1;
    245   typedef typename BVH2::VolumeIterator VolIter2;
    246   typedef typename BVH2::ObjectIterator ObjIter2;
    247 
    248   VolIter1 vBegin1 = VolIter1(), vEnd1 = VolIter1();
    249   ObjIter1 oBegin1 = ObjIter1(), oEnd1 = ObjIter1();
    250   VolIter2 vBegin2 = VolIter2(), vEnd2 = VolIter2(), vCur2 = VolIter2();
    251   ObjIter2 oBegin2 = ObjIter2(), oEnd2 = ObjIter2(), oCur2 = ObjIter2();
    252   std::priority_queue<QueueElement, std::vector<QueueElement>, std::greater<QueueElement> > todo; //smallest is at the top
    253 
    254   Scalar minimum = (std::numeric_limits<Scalar>::max)();
    255   todo.push(std::make_pair(Scalar(), std::make_pair(tree1.getRootIndex(), tree2.getRootIndex())));
    256 
    257   while(!todo.empty()) {
    258     tree1.getChildren(todo.top().second.first, vBegin1, vEnd1, oBegin1, oEnd1);
    259     tree2.getChildren(todo.top().second.second, vBegin2, vEnd2, oBegin2, oEnd2);
    260     todo.pop();
    261 
    262     for(; oBegin1 != oEnd1; ++oBegin1) { //go through child objects of first tree
    263       for(oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) {//go through child objects of second tree
    264         minimum = (std::min)(minimum, minimizer.minimumOnObjectObject(*oBegin1, *oCur2));
    265       }
    266 
    267       for(vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { //go through child volumes of second tree
    268         Helper2 helper(*oBegin1, minimizer);
    269         minimum = (std::min)(minimum, internal::minimize_helper(tree2, helper, *vCur2, minimum));
    270       }
    271     }
    272 
    273     for(; vBegin1 != vEnd1; ++vBegin1) { //go through child volumes of first tree
    274       const typename BVH1::Volume &vol1 = tree1.getVolume(*vBegin1);
    275 
    276       for(oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) {//go through child objects of second tree
    277         Helper1 helper(*oCur2, minimizer);
    278         minimum = (std::min)(minimum, internal::minimize_helper(tree1, helper, *vBegin1, minimum));
    279       }
    280 
    281       for(vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { //go through child volumes of second tree
    282         Scalar val = minimizer.minimumOnVolumeVolume(vol1, tree2.getVolume(*vCur2));
    283         if(val < minimum)
    284           todo.push(std::make_pair(val, std::make_pair(*vBegin1, *vCur2)));
    285       }
    286     }
    287   }
    288   return minimum;
    289 }
    290 
    291 } // end namespace Eigen
    292 
    293 #endif // EIGEN_BVALGORITHMS_H
    294