Eigen-unsupported  5.0.1-dev+284dcc12
 
Loading...
Searching...
No Matches
BVAlgorithms.h
1// This file is part of Eigen, a lightweight C++ template library
2// for linear algebra.
3//
4// Copyright (C) 2009 Ilya Baran <ibaran@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// IWYU pragma: private
14#include "./InternalHeaderCheck.h"
15
16namespace Eigen {
17
18namespace internal {
19
20#ifndef EIGEN_PARSED_BY_DOXYGEN
21template <typename BVH, typename Intersector>
22bool intersect_helper(const BVH &tree, Intersector &intersector, typename BVH::Index root) {
23 typedef typename BVH::Index Index;
24 typedef typename BVH::VolumeIterator VolIter;
25 typedef typename BVH::ObjectIterator ObjIter;
26
27 VolIter vBegin = VolIter(), vEnd = VolIter();
28 ObjIter oBegin = ObjIter(), oEnd = ObjIter();
29
30 std::vector<Index> todo(1, root);
31
32 while (!todo.empty()) {
33 tree.getChildren(todo.back(), vBegin, vEnd, oBegin, oEnd);
34 todo.pop_back();
35
36 for (; vBegin != vEnd; ++vBegin) // go through child volumes
37 if (intersector.intersectVolume(tree.getVolume(*vBegin))) todo.push_back(*vBegin);
38
39 for (; oBegin != oEnd; ++oBegin) // go through child objects
40 if (intersector.intersectObject(*oBegin)) return true; // intersector said to stop query
41 }
42 return false;
43}
44#endif // not EIGEN_PARSED_BY_DOXYGEN
45
46template <typename Volume1, typename Object1, typename Object2, typename Intersector>
47struct intersector_helper1 {
48 intersector_helper1(const Object2 &inStored, Intersector &in) : stored(inStored), intersector(in) {}
49 bool intersectVolume(const Volume1 &vol) { return intersector.intersectVolumeObject(vol, stored); }
50 bool intersectObject(const Object1 &obj) { return intersector.intersectObjectObject(obj, stored); }
51 Object2 stored;
52 Intersector &intersector;
53
54 private:
55 intersector_helper1 &operator=(const intersector_helper1 &);
56};
57
58template <typename Volume2, typename Object2, typename Object1, typename Intersector>
59struct intersector_helper2 {
60 intersector_helper2(const Object1 &inStored, Intersector &in) : stored(inStored), intersector(in) {}
61 bool intersectVolume(const Volume2 &vol) { return intersector.intersectObjectVolume(stored, vol); }
62 bool intersectObject(const Object2 &obj) { return intersector.intersectObjectObject(stored, obj); }
63 Object1 stored;
64 Intersector &intersector;
65
66 private:
67 intersector_helper2 &operator=(const intersector_helper2 &);
68};
69
70} // end namespace internal
71
78template <typename BVH, typename Intersector>
79void BVIntersect(const BVH &tree, Intersector &intersector) {
80 internal::intersect_helper(tree, intersector, tree.getRootIndex());
81}
82
91template <typename BVH1, typename BVH2, typename Intersector>
92void BVIntersect(const BVH1 &tree1, const BVH2 &tree2,
93 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,
98 Intersector>
99 Helper1;
100 typedef internal::intersector_helper2<typename BVH2::Volume, typename BVH2::Object, typename BVH1::Object,
101 Intersector>
102 Helper2;
103 typedef typename BVH1::VolumeIterator VolIter1;
104 typedef typename BVH1::ObjectIterator ObjIter1;
105 typedef typename BVH2::VolumeIterator VolIter2;
106 typedef typename BVH2::ObjectIterator ObjIter2;
107
108 VolIter1 vBegin1 = VolIter1(), vEnd1 = VolIter1();
109 ObjIter1 oBegin1 = ObjIter1(), oEnd1 = ObjIter1();
110 VolIter2 vBegin2 = VolIter2(), vEnd2 = VolIter2(), vCur2 = VolIter2();
111 ObjIter2 oBegin2 = ObjIter2(), oEnd2 = ObjIter2(), oCur2 = ObjIter2();
112
113 std::vector<std::pair<Index1, Index2> > todo(1, std::make_pair(tree1.getRootIndex(), tree2.getRootIndex()));
114
115 while (!todo.empty()) {
116 tree1.getChildren(todo.back().first, vBegin1, vEnd1, oBegin1, oEnd1);
117 tree2.getChildren(todo.back().second, vBegin2, vEnd2, oBegin2, oEnd2);
118 todo.pop_back();
119
120 for (; vBegin1 != vEnd1; ++vBegin1) { // go through child volumes of first tree
121 const typename BVH1::Volume &vol1 = tree1.getVolume(*vBegin1);
122 for (vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { // go through child volumes of second tree
123 if (intersector.intersectVolumeVolume(vol1, tree2.getVolume(*vCur2)))
124 todo.push_back(std::make_pair(*vBegin1, *vCur2));
125 }
126
127 for (oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) { // go through child objects of second tree
128 Helper1 helper(*oCur2, intersector);
129 if (internal::intersect_helper(tree1, helper, *vBegin1)) return; // intersector said to stop query
130 }
131 }
132
133 for (; oBegin1 != oEnd1; ++oBegin1) { // go through child objects of first tree
134 for (vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { // go through child volumes of second tree
135 Helper2 helper(*oBegin1, intersector);
136 if (internal::intersect_helper(tree2, helper, *vCur2)) return; // intersector said to stop query
137 }
138
139 for (oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) { // go through child objects of second tree
140 if (intersector.intersectObjectObject(*oBegin1, *oCur2)) return; // intersector said to stop query
141 }
142 }
143 }
144}
145
146namespace internal {
147
148#ifndef EIGEN_PARSED_BY_DOXYGEN
149template <typename BVH, typename Minimizer>
150typename Minimizer::Scalar minimize_helper(const BVH &tree, Minimizer &minimizer, typename BVH::Index root,
151 typename Minimizer::Scalar minimum) {
152 typedef typename Minimizer::Scalar Scalar;
153 typedef typename BVH::Index Index;
154 typedef std::pair<Scalar, Index> QueueElement; // first element is priority
155 typedef typename BVH::VolumeIterator VolIter;
156 typedef typename BVH::ObjectIterator ObjIter;
157
158 VolIter vBegin = VolIter(), vEnd = VolIter();
159 ObjIter oBegin = ObjIter(), oEnd = ObjIter();
160 std::priority_queue<QueueElement, std::vector<QueueElement>, std::greater<QueueElement> >
161 todo; // smallest is at the top
162
163 todo.push(std::make_pair(Scalar(), root));
164
165 while (!todo.empty()) {
166 tree.getChildren(todo.top().second, vBegin, vEnd, oBegin, oEnd);
167 todo.pop();
168
169 for (; oBegin != oEnd; ++oBegin) // go through child objects
170 minimum = (std::min)(minimum, minimizer.minimumOnObject(*oBegin));
171
172 for (; vBegin != vEnd; ++vBegin) { // go through child volumes
173 Scalar val = minimizer.minimumOnVolume(tree.getVolume(*vBegin));
174 if (val < minimum) todo.push(std::make_pair(val, *vBegin));
175 }
176 }
177
178 return minimum;
179}
180#endif // not EIGEN_PARSED_BY_DOXYGEN
181
182template <typename Volume1, typename Object1, typename Object2, typename Minimizer>
183struct minimizer_helper1 {
184 typedef typename Minimizer::Scalar Scalar;
185 minimizer_helper1(const Object2 &inStored, Minimizer &m) : stored(inStored), minimizer(m) {}
186 Scalar minimumOnVolume(const Volume1 &vol) { return minimizer.minimumOnVolumeObject(vol, stored); }
187 Scalar minimumOnObject(const Object1 &obj) { return minimizer.minimumOnObjectObject(obj, stored); }
188 Object2 stored;
189 Minimizer &minimizer;
190
191 private:
192 minimizer_helper1 &operator=(const minimizer_helper1 &);
193};
194
195template <typename Volume2, typename Object2, typename Object1, typename Minimizer>
196struct minimizer_helper2 {
197 typedef typename Minimizer::Scalar Scalar;
198 minimizer_helper2(const Object1 &inStored, Minimizer &m) : stored(inStored), minimizer(m) {}
199 Scalar minimumOnVolume(const Volume2 &vol) { return minimizer.minimumOnObjectVolume(stored, vol); }
200 Scalar minimumOnObject(const Object2 &obj) { return minimizer.minimumOnObjectObject(stored, obj); }
201 Object1 stored;
202 Minimizer &minimizer;
203
204 private:
205 minimizer_helper2 &operator=(const minimizer_helper2 &);
206};
207
208} // end namespace internal
209
216template <typename BVH, typename Minimizer>
217typename Minimizer::Scalar BVMinimize(const BVH &tree, Minimizer &minimizer) {
218 return internal::minimize_helper(tree, minimizer, tree.getRootIndex(),
219 (std::numeric_limits<typename Minimizer::Scalar>::max)());
220}
221
231template <typename BVH1, typename BVH2, typename Minimizer>
232typename Minimizer::Scalar BVMinimize(const BVH1 &tree1, const BVH2 &tree2, Minimizer &minimizer) {
233 typedef typename Minimizer::Scalar Scalar;
234 typedef typename BVH1::Index Index1;
235 typedef typename BVH2::Index Index2;
236 typedef internal::minimizer_helper1<typename BVH1::Volume, typename BVH1::Object, typename BVH2::Object, Minimizer>
237 Helper1;
238 typedef internal::minimizer_helper2<typename BVH2::Volume, typename BVH2::Object, typename BVH1::Object, Minimizer>
239 Helper2;
240 typedef std::pair<Scalar, std::pair<Index1, Index2> > QueueElement; // first element is priority
241 typedef typename BVH1::VolumeIterator VolIter1;
242 typedef typename BVH1::ObjectIterator ObjIter1;
243 typedef typename BVH2::VolumeIterator VolIter2;
244 typedef typename BVH2::ObjectIterator ObjIter2;
245
246 VolIter1 vBegin1 = VolIter1(), vEnd1 = VolIter1();
247 ObjIter1 oBegin1 = ObjIter1(), oEnd1 = ObjIter1();
248 VolIter2 vBegin2 = VolIter2(), vEnd2 = VolIter2(), vCur2 = VolIter2();
249 ObjIter2 oBegin2 = ObjIter2(), oEnd2 = ObjIter2(), oCur2 = ObjIter2();
250 std::priority_queue<QueueElement, std::vector<QueueElement>, std::greater<QueueElement> >
251 todo; // smallest is at the top
252
253 Scalar minimum = (std::numeric_limits<Scalar>::max)();
254 todo.push(std::make_pair(Scalar(), std::make_pair(tree1.getRootIndex(), tree2.getRootIndex())));
255
256 while (!todo.empty()) {
257 tree1.getChildren(todo.top().second.first, vBegin1, vEnd1, oBegin1, oEnd1);
258 tree2.getChildren(todo.top().second.second, vBegin2, vEnd2, oBegin2, oEnd2);
259 todo.pop();
260
261 for (; oBegin1 != oEnd1; ++oBegin1) { // go through child objects of first tree
262 for (oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) { // go through child objects of second tree
263 minimum = (std::min)(minimum, minimizer.minimumOnObjectObject(*oBegin1, *oCur2));
264 }
265
266 for (vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { // go through child volumes of second tree
267 Helper2 helper(*oBegin1, minimizer);
268 minimum = (std::min)(minimum, internal::minimize_helper(tree2, helper, *vCur2, minimum));
269 }
270 }
271
272 for (; vBegin1 != vEnd1; ++vBegin1) { // go through child volumes of first tree
273 const typename BVH1::Volume &vol1 = tree1.getVolume(*vBegin1);
274
275 for (oCur2 = oBegin2; oCur2 != oEnd2; ++oCur2) { // go through child objects of second tree
276 Helper1 helper(*oCur2, minimizer);
277 minimum = (std::min)(minimum, internal::minimize_helper(tree1, helper, *vBegin1, minimum));
278 }
279
280 for (vCur2 = vBegin2; vCur2 != vEnd2; ++vCur2) { // go through child volumes of second tree
281 Scalar val = minimizer.minimumOnVolumeVolume(vol1, tree2.getVolume(*vCur2));
282 if (val < minimum) todo.push(std::make_pair(val, std::make_pair(*vBegin1, *vCur2)));
283 }
284 }
285 }
286 return minimum;
287}
288
289} // end namespace Eigen
290
291#endif // EIGEN_BVALGORITHMS_H
Namespace containing all symbols from the Eigen library.
void BVIntersect(const BVH &tree, Intersector &intersector)
Definition BVAlgorithms.h:79
EIGEN_DEFAULT_DENSE_INDEX_TYPE Index
Minimizer::Scalar BVMinimize(const BVH &tree, Minimizer &minimizer)
Definition BVAlgorithms.h:217