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