LCOV - code coverage report
Current view: top level - elsa/projectors - TraverseAABB.cpp (source / functions) Hit Total Coverage
Test: coverage-all.lcov Lines: 92 92 100.0 %
Date: 2025-01-22 07:37:33 Functions: 30 30 100.0 %

          Line data    Source code
       1             : #include "TraverseAABB.h"
       2             : #include "Intersection.h"
       3             : 
       4             : namespace elsa
       5             : {
       6             :     template <int dim>
       7             :     TraverseAABB<dim>::TraverseAABB(const BoundingBox& aabb, const RealRay_t& r,
       8             :                                     IndexArray_t<dim> productOfCoefficientsPerDimension)
       9             :         : _productOfCoefficientsPerDimension{std::move(productOfCoefficientsPerDimension)}
      10       61917 :     {
      11       61917 :         static_assert(dim == 2 || dim == 3);
      12       61917 :         _aabbMin = aabb.min();
      13       61917 :         _aabbMax = aabb.max();
      14             : 
      15             :         // compute the first intersection
      16       61917 :         const RealArray_t<dim> entryPoint = calculateAABBIntersections(r, aabb);
      17       61917 :         if (!isInBoundingBox()) // early abort if necessary
      18        5855 :             return;
      19             : 
      20             :         // constant array containing epsilon
      21       56062 :         const RealArray_t<dim> EPS{
      22       56062 :             RealArray_t<dim>().setConstant(std::numeric_limits<real_t>::epsilon())};
      23             : 
      24             :         // constant vector containing the maximum number
      25       56062 :         const RealArray_t<dim> MAX{
      26       56062 :             RealArray_t<dim>().setConstant(std::numeric_limits<real_t>::max())};
      27             : 
      28             :         // determine whether we go up/down or left/right
      29       56062 :         initStepDirection(r.direction());
      30             : 
      31             :         // select the initial voxel (closest to the entry point)
      32       56062 :         selectClosestVoxel(entryPoint);
      33             : 
      34             :         // initialize the step sizes for the step parameter
      35       56062 :         initDelta(r.direction(), EPS, MAX);
      36             : 
      37             :         // initialize the maximum step parameters
      38       56062 :         initT(r.direction(), EPS, MAX, entryPoint);
      39       56062 :         initCurrentIndex();
      40       56062 :     }
      41             : 
      42             :     template <int dim>
      43             :     void TraverseAABB<dim>::updateTraverse()
      44     2173558 :     {
      45             :         // --> calculate the mask that masks out all but the lowest t values
      46     2173558 :         calcMask();
      47             : 
      48             :         // --> step into the current direction
      49     2173558 :         _currentPos += _mask.select(_stepDirection.template cast<real_t>(), 0);
      50             : 
      51             :         // --> update the T for next iteration
      52     2173558 :         _T += _mask.select(_tDelta, 0);
      53             : 
      54             :         // --> check if we are still in bounding box
      55     2173558 :         _isInAABB = isCurrentPositionInAABB();
      56     2173558 :     }
      57             : 
      58             :     template <int dim>
      59             :     void TraverseAABB<dim>::calcMask()
      60     2081566 :     {
      61     2081566 :         if constexpr (dim == 2) {
      62       62841 :             _mask[0] = (_T[0] <= _T[1]);
      63       62841 :             _mask[1] = (_T[1] <= _T[0]);
      64       62841 :         } else if constexpr (dim == 3) {
      65       62559 :             _mask[0] = ((_T[0] <= _T[1]) && (_T[0] <= _T[2]));
      66       62559 :             _mask[1] = ((_T[1] <= _T[0]) && (_T[1] <= _T[2]));
      67       62559 :             _mask[2] = ((_T[2] <= _T[0]) && (_T[2] <= _T[1]));
      68       62559 :         }
      69     2081566 :     }
      70             : 
      71             :     template <int dim>
      72             :     real_t TraverseAABB<dim>::updateTraverseAndGetDistance()
      73     1701334 :     {
      74             :         // --> compute the distance
      75     1701334 :         real_t tEntry = _tExit;
      76     1701334 :         _tExit = _T.minCoeff();
      77             : 
      78             :         // --> do the update
      79     1701334 :         updateTraverse();
      80     1701334 :         updateCurrentIndex();
      81             : 
      82     1701334 :         return (_tExit - tEntry);
      83     1701334 :     }
      84             : 
      85             :     template <int dim>
      86             :     bool TraverseAABB<dim>::isInBoundingBox() const
      87     2396963 :     {
      88     2396963 :         return _isInAABB;
      89     2396963 :     }
      90             : 
      91             :     template <int dim>
      92             :     IndexArray_t<dim> TraverseAABB<dim>::getCurrentVoxel() const
      93      598641 :     {
      94      598641 :         return _currentPos.template cast<index_t>();
      95      598641 :     }
      96             : 
      97             :     template <int dim>
      98             :     index_t TraverseAABB<dim>::getCurrentIndex() const
      99     1854459 :     {
     100     1854459 :         return _currentIndex;
     101     1854459 :     }
     102             : 
     103             :     template <int dim>
     104             :     RealArray_t<dim> TraverseAABB<dim>::calculateAABBIntersections(const RealRay_t& r,
     105             :                                                                    const BoundingBox& aabb)
     106       61790 :     {
     107       61790 :         RealArray_t<dim> entryPoint;
     108             :         // entry and exit point parameters
     109       61790 :         real_t tmin;
     110             : 
     111             :         // --> calculate intersection parameter and if the volume is hit
     112       61790 :         auto opt = intersectRay(aabb, r);
     113             : 
     114       61790 :         if (opt) { // hit!
     115       56515 :             _isInAABB = true;
     116       56515 :             tmin = opt->_tmin;
     117             : 
     118             :             // --> get points at which they intersect
     119       56515 :             entryPoint = r.pointAt(tmin);
     120             : 
     121             :             // --> because of floating point error it can happen, that values are out of
     122             :             // the bounding box, this can lead to errors
     123       56515 :             entryPoint = (entryPoint < _aabbMin).select(_aabbMin, entryPoint);
     124       56515 :             entryPoint = (entryPoint > _aabbMax).select(_aabbMax, entryPoint);
     125       56515 :         }
     126       61790 :         return entryPoint;
     127       61790 :     }
     128             : 
     129             :     template <int dim>
     130             :     void TraverseAABB<dim>::initStepDirection(const RealArray_t<dim>& rd)
     131       56139 :     {
     132       56139 :         _stepDirection = rd.sign().template cast<index_t>();
     133       56139 :     }
     134             : 
     135             :     template <int dim>
     136             :     void TraverseAABB<dim>::selectClosestVoxel(const RealArray_t<dim>& entryPoint)
     137       56211 :     {
     138       56211 :         RealArray_t<dim> lowerCorner = entryPoint.floor();
     139       56211 :         lowerCorner = ((lowerCorner == entryPoint) && (_stepDirection < 0.0))
     140       56211 :                           .select(lowerCorner - 1, lowerCorner);
     141             : 
     142       56211 :         _currentPos = lowerCorner;
     143             : 
     144             :         // check if we are still inside the aabb
     145       56321 :         if ((_currentPos >= _aabbMax).any() || (_currentPos < _aabbMin).any())
     146          12 :             _isInAABB = false;
     147       56211 :     }
     148             : 
     149             :     template <int dim>
     150             :     void TraverseAABB<dim>::initDelta(const RealArray_t<dim>& rd, const RealArray_t<dim>& EPS,
     151             :                                       const RealArray_t<dim>& MAX)
     152       56154 :     {
     153       56154 :         RealArray_t<dim> tdelta = _stepDirection.template cast<real_t>() / rd;
     154             : 
     155       56154 :         _tDelta = (Eigen::abs(rd) > EPS).select(tdelta, MAX);
     156       56154 :     }
     157             : 
     158             :     template <int dim>
     159             :     void TraverseAABB<dim>::initT(const RealArray_t<dim>& rd, const RealArray_t<dim>& EPS,
     160             :                                   const RealArray_t<dim>& MAX, const RealArray_t<dim>& entryPoint)
     161       56204 :     {
     162       56204 :         RealArray_t<dim> T =
     163       56204 :             (((rd > 0.0f).select(_currentPos + 1., _currentPos)) - entryPoint) / rd;
     164             : 
     165       56204 :         _T = (Eigen::abs(rd) > EPS).select(T, MAX);
     166       56204 :     }
     167             : 
     168             :     template <int dim>
     169             :     bool TraverseAABB<dim>::isCurrentPositionInAABB() const
     170     2082585 :     {
     171     2195372 :         return (_currentPos < _aabbMax).all() && (_currentPos >= _aabbMin).all();
     172     2082585 :     }
     173             : 
     174             :     template <int dim>
     175             :     void TraverseAABB<dim>::updateCurrentIndex()
     176     1693201 :     {
     177     1693201 :         _currentIndex +=
     178     1693201 :             (_stepDirection * _mask.select(_productOfCoefficientsPerDimension, 0)).sum();
     179     1693201 :     }
     180             :     template <int dim>
     181             :     void TraverseAABB<dim>::initCurrentIndex()
     182       56224 :     {
     183       56224 :         _currentIndex = (_productOfCoefficientsPerDimension * getCurrentVoxel()).sum();
     184       56224 :     }
     185             : 
     186             :     template class TraverseAABB<2>;
     187             :     template class TraverseAABB<3>;
     188             : } // namespace elsa

Generated by: LCOV version 1.14