Tramway SDK v0.1.1
aabb.h
Go to the documentation of this file.
1// Tramway Drifting and Dungeon Exploration Simulator SDK Runtime
2
3#ifndef TRAM_SDK_TEMPLATES_AABB_H
4#define TRAM_SDK_TEMPLATES_AABB_H
5
6#include <framework/math.h>
7#include <framework/logging.h>
8#include <framework/core.h>
9
10#include <iostream>
11
12#include <vector>
13
14namespace tram {
15
16class AABBTree {
17public:
20 //RemoveHierarchy(root);
21 // TODO: fix
22 }
23
24 vec3 GetAABBMin() { return root->min; }
25 vec3 GetAABBMax() { return root->max; }
26
27 struct Node;
28
29 Node* InsertLeaf (uint32_t value, vec3 min, vec3 max) {
30 Node* new_node = new Node;
31
32 new_node->value = value;
33
34 new_node->min = min;
35 new_node->max = max;
36
37 if (root->left == nullptr) {
38 root->left = new_node;
39 new_node->parent = root;
40
41 if (root->right) {
43 } else {
44 root->min = new_node->min;
45 root->max = new_node->max;
46 }
47
48 return new_node;
49 }
50
51 if (root->right == nullptr) {
52 root->right = new_node;
53 new_node->parent = root;
54
55 if (root->left) {
57 } else {
58 root->min = new_node->min;
59 root->max = new_node->max;
60 }
61
62 return new_node;
63 }
64
65 Node* sibling = FindSibling(min, max, root);
66 Node* sibling_parent = sibling->parent;
67 Node* new_parent = new Node;
68
69 if (sibling_parent->left == sibling) {
70 sibling_parent->left = new_parent;
71 } else if (sibling_parent->right == sibling) {
72 sibling_parent->right = new_parent;
73 } else {
74 assert(false);
75 }
76
77 new_parent->parent = sibling_parent;
78
79 new_parent->left = new_node;
80 new_parent->right = sibling;
81
82 sibling->parent = new_parent;
83 new_node->parent = new_parent;
84
85 UpdateParentAABB(new_parent);
86
88
89 return new_node;
90 }
91
92 void RemoveLeaf(Node* node) {
93 assert(node);
94
95 Node* parent = node->parent;
96 Node* sibling = parent->left == node ? parent->right : parent->left;
97
98 if (parent->left != node && parent->right != node) {
99 assert(false);
100 }
101
102 if (parent == root) {
103 if (parent->left == node) {
104 parent->left = nullptr;
105
106 if (sibling) {
107 parent->min = sibling->min;
108 parent->max = sibling->max;
109 }
110 } else {
111 parent->right = nullptr;
112
113 if (sibling) {
114 parent->min = sibling->min;
115 parent->max = sibling->max;
116 }
117 }
118
119 delete node;
120
122
123 return;
124 }
125
126 Node* grandparent = parent->parent;
127
128 if (grandparent->left == parent) {
129 grandparent->left = sibling;
130 } else {
131 grandparent->right = sibling;
132 }
133
134 sibling->parent = grandparent;
135
136 UpdateParentAABB(grandparent);
137
138 delete node;
139 delete parent;
140
142 }
143
144 void RemoveHierarchy(Node* node) {
145 if (node->IsLeaf()) {
146 delete node;
147 } else {
148 RemoveHierarchy(node->left);
149 RemoveHierarchy(node->right);
150 delete node;
151 }
152 }
153
154 void FindIntersection(vec3 ray_pos, vec3 ray_dir, Node* node, std::vector<uint32_t>& result) const {
155 bool is_node_intersect = AABBIntersect(ray_pos, ray_dir, node->min, node->max);
156
157 if (is_node_intersect) {
158 if (node->IsLeaf() && node != root) {
159 result.push_back(node->value);
160 } else {
161 if (node->left) FindIntersection (ray_pos, ray_dir, node->left, result);
162 if (node->right) FindIntersection (ray_pos, ray_dir, node->right, result);
163 }
164 }
165 }
166
167 uint32_t FindIntersection(vec3 ray_pos, vec3 ray_dir, float distance_limit, auto filter) const {
168 bool root_intersects = AABBIntersect(ray_pos, ray_dir, root->min, root->max);
169
170 if (!root_intersects) {
171 return -1;
172 }
173
174 float nearest_dist = INFINITY;
175 uint32_t nearest_index = -1;
176
177 FindIntersectionRecursive(ray_pos, ray_dir, nearest_dist, nearest_index, distance_limit, root, filter);
178
179 return nearest_index;
180 }
181
182 // this should be marked private
183 void FindIntersectionRecursive(vec3 ray_pos, vec3 ray_dir, float& nearest_dist, uint32_t& nearest_index, float distance_limit, Node* node, auto filter) const {
184 if (node->IsLeaf() && node != root) {
185 float leaf_distance = filter(ray_pos, ray_dir, node->value);
186
187 if (leaf_distance < nearest_dist) {
188 nearest_dist = leaf_distance;
189 nearest_index = node->value;
190 }
191
192 return;
193 }
194
195 float left_distance = INFINITY;
196 float right_distance = INFINITY;
197
198 if (node->left) left_distance = AABBDistance(ray_pos, ray_dir, node->left->min, node->left->max);
199 if (node->right) right_distance = AABBDistance(ray_pos, ray_dir, node->right->min, node->right->max);
200
201 if (left_distance < right_distance) {
202
203 if (left_distance < nearest_dist && left_distance < distance_limit) {
204 FindIntersectionRecursive(ray_pos, ray_dir, nearest_dist, nearest_index, distance_limit, node->left, filter);
205 }
206
207 if (right_distance < nearest_dist && right_distance < distance_limit) {
208 FindIntersectionRecursive(ray_pos, ray_dir, nearest_dist, nearest_index, distance_limit, node->right, filter);
209 }
210
211 } else {
212
213 if (right_distance < nearest_dist && right_distance < distance_limit) {
214 FindIntersectionRecursive(ray_pos, ray_dir, nearest_dist, nearest_index, distance_limit, node->right, filter);
215 }
216
217 if (left_distance < nearest_dist && left_distance < distance_limit) {
218 FindIntersectionRecursive(ray_pos, ray_dir, nearest_dist, nearest_index, distance_limit, node->left, filter);
219 }
220
221 }
222
223 }
224
225 void FindAABBIntersection(vec3 min, vec3 max, auto callback) {
226 FindAABBIntersection(root, min, max, callback);
227 }
228
229 // should be private
230 void FindAABBIntersection(Node* node, vec3 min, vec3 max, auto callback) {
231 if (node->IsLeaf() && node != root) {
232 if (AABBOverlap(min, max, node->min, node->max)) {
233 callback(node->value);
234 }
235
236 return;
237 }
238
239 if (node->left && AABBOverlap(min, max, node->left->min, node->left->max)) {
240 FindAABBIntersection(node->left, min, max, callback);
241 }
242
243 if (node->right && AABBOverlap(min, max, node->right->min, node->right->max)) {
244 FindAABBIntersection(node->right, min, max, callback);
245 }
246
247 }
248
249//private:
250
251 void UpdateParentAABB (Node* node) {
252
253 //assert(!node->IsLeaf());
254
255 if (node->IsLeaf()) return;
256
257 Node* left_child = node->left;
258 Node* right_child = node->right;
259
260 if (!left_child || !right_child) {
261 assert(node == root);
262
263 if (!left_child && right_child) {
264 node->min = right_child->min;
265 node->max = right_child->max;
266 } else if (left_child && !right_child) {
267 node->min = left_child->min;
268 node->max = left_child->max;
269 } else {
270 node->min = {0.0f, 0.0f, 0.0f};
271 node->max = {0.0f, 0.0f, 0.0f};
272 }
273
274 return;
275 }
276
277
278 node->min = MergeAABBMin(left_child->min, right_child->min);
279 node->max = MergeAABBMax(left_child->max, right_child->max);
280
281 if (node->parent != nullptr) {
283 }
284
285 assert(node->parent != node);
286 }
287
288 // searches the children of search_node to find a sibling for target_node
289 Node* FindSibling (vec3 min, vec3 max, Node* node) {
290 assert(node);
291
292 if (node->IsLeaf()) {
293 return node;
294 }
295
296 if (node->left == nullptr) {
297 assert(false);
298 }
299 assert(node->left);
300 assert(node->right);
301
302 vec3 left_merge_min = MergeAABBMin(min, node->left->min);
303 vec3 left_merge_max = MergeAABBMax(max, node->left->max);
304
305 vec3 right_merge_min = MergeAABBMin(min, node->right->min);
306 vec3 right_merge_max = MergeAABBMax(max, node->right->max);
307
308 float left_merge_volume = AABBVolume(left_merge_min, left_merge_max);
309 float right_merge_volume = AABBVolume(right_merge_min, right_merge_max);
310
311 //float left_merge_volume = AABBSurface(left_merge_min, left_merge_max);
312 //float right_merge_volume = AABBSurface(right_merge_min, right_merge_max);
313
314 if (left_merge_volume < right_merge_volume) {
315 return FindSibling(min, max, node->left);
316 } else {
317 return FindSibling(min, max, node->right);
318 }
319 }
320
321
322 void ValidateTree (Node* node) {
323 return;
324 if (root->parent != nullptr) {
325 //if (((Node*)0)->IsLeaf()) assert(false);
326 }
327
328 return;
329 ValidateTree (node, 0);
330 }
331 void ValidateTree (Node* node, size_t num) {
332 assert(node);
333 assert((long long)node > 100);
334
335 if (num > 400) {
336 //if (((Node*)0)->IsLeaf()) assert(false);
337 }
338
339 if (node->IsLeaf() && node != root) {
340 //assert(node->value < 40000);
341 } else {
342 if (node != root || (node == root && node->left))ValidateTree(node->left, num+1);
343 if (node != root || (node == root && node->right))ValidateTree(node->right, num+1);
344 }
345 }
346
347 static vec3 MergeAABBMin (vec3 a, vec3 b) {
348 return vec3 {
349 a.x < b.x ? a.x : b.x,
350 a.y < b.y ? a.y : b.y,
351 a.z < b.z ? a.z : b.z
352 };// - vec3 {0.1f, 0.1f, 0.1f};
353 }
354
355 static vec3 MergeAABBMax (vec3 a, vec3 b) {
356 return vec3 {
357 a.x > b.x ? a.x : b.x,
358 a.y > b.y ? a.y : b.y,
359 a.z > b.z ? a.z : b.z
360 };// + vec3 {0.1f, 0.1f, 0.1f};
361 }
362
363 static bool AABBOverlap(vec3 min, vec3 max, vec3 other_min, vec3 other_max) {
364 return min.x <= other_max.x && max.x >= other_min.x &&
365 min.y <= other_max.y && max.y >= other_min.y &&
366 min.z <= other_max.z && max.z >= other_min.z;
367 }
368
369 static float AABBVolume (vec3 min, vec3 max) {
370 return (max.x - min.x) * (max.y - min.y) * (max.z - min.z);
371 }
372
373 static float AABBSurface (vec3 min, vec3 max) {
374 float x = max.x - min.x;
375 float y = max.y - min.y;
376 float z = max.z - min.z;
377
378 assert(max.x >= min.x);
379 assert(max.y >= min.y);
380 assert(max.z >= min.z);
381
382 return 2 * ((x * y) + (x * z) + (y * z));
383 }
384
385 static bool AABBIntersect (vec3 ray_pos, vec3 ray_dir, vec3 min, vec3 max) {
386 vec3 t1 = (min - ray_pos) / ray_dir; // what happens if ray_dir == 0.0f?
387 vec3 t2 = (max - ray_pos) / ray_dir; // TODO: check
388
389 vec3 t1min = glm::min(t1, t2);
390 vec3 t2max = glm::max(t1, t2);
391
392 float tnear = glm::max(glm::max(t1min.x, t1min.y), t1min.z);
393 float tfar = glm::min(glm::min(t2max.x, t2max.y), t2max.z);
394
395 return tfar >= tnear;
396 }
397
398 static float AABBDistance(vec3 ray_pos, vec3 ray_dir, vec3 min, vec3 max) {
399 vec3 t1 = (min - ray_pos) / ray_dir; // what happens if ray_dir == 0.0f?
400 vec3 t2 = (max - ray_pos) / ray_dir; // TODO: check
401
402 vec3 t1min = glm::min(t1, t2);
403 vec3 t2max = glm::max(t1, t2);
404
405 float tnear = glm::max(glm::max(t1min.x, t1min.y), t1min.z);
406 float tfar = glm::min(glm::min(t2max.x, t2max.y), t2max.z);
407
408 return tfar >= tnear ? tnear : INFINITY;
409 }
410
411 struct Node {
412 bool IsLeaf () const { return right == 0; }
413
414 void Print () const { std::cout << " l: " << left << " r: " << right << " p: " << parent << std::endl; }
415
416 union {
417 Node* left = nullptr;
418 uint32_t value;
419 };
420
421 Node* right = nullptr;
422 Node* parent = nullptr;
423
426 };
427
428 Node* root = new Node {nullptr, nullptr, nullptr, {0.0f, 0.0f, 0.0f}, {0.0f, 0.0f, 0.0f}};
429};
430
431}
432
433#endif // TRAM_SDK_TEMPLATES_AABB_H
Definition: aabb.h:16
Node * InsertLeaf(uint32_t value, vec3 min, vec3 max)
Definition: aabb.h:29
static bool AABBIntersect(vec3 ray_pos, vec3 ray_dir, vec3 min, vec3 max)
Definition: aabb.h:385
~AABBTree()
Definition: aabb.h:19
void FindIntersection(vec3 ray_pos, vec3 ray_dir, Node *node, std::vector< uint32_t > &result) const
Definition: aabb.h:154
void UpdateParentAABB(Node *node)
Definition: aabb.h:251
void ValidateTree(Node *node)
Definition: aabb.h:322
void FindAABBIntersection(Node *node, vec3 min, vec3 max, auto callback)
Definition: aabb.h:230
AABBTree()
Definition: aabb.h:18
vec3 GetAABBMax()
Definition: aabb.h:25
void FindIntersectionRecursive(vec3 ray_pos, vec3 ray_dir, float &nearest_dist, uint32_t &nearest_index, float distance_limit, Node *node, auto filter) const
Definition: aabb.h:183
Node * root
Definition: aabb.h:428
static float AABBVolume(vec3 min, vec3 max)
Definition: aabb.h:369
void FindAABBIntersection(vec3 min, vec3 max, auto callback)
Definition: aabb.h:225
static vec3 MergeAABBMax(vec3 a, vec3 b)
Definition: aabb.h:355
static float AABBDistance(vec3 ray_pos, vec3 ray_dir, vec3 min, vec3 max)
Definition: aabb.h:398
void RemoveLeaf(Node *node)
Definition: aabb.h:92
void RemoveHierarchy(Node *node)
Definition: aabb.h:144
static vec3 MergeAABBMin(vec3 a, vec3 b)
Definition: aabb.h:347
Node * FindSibling(vec3 min, vec3 max, Node *node)
Definition: aabb.h:289
vec3 GetAABBMin()
Definition: aabb.h:24
uint32_t FindIntersection(vec3 ray_pos, vec3 ray_dir, float distance_limit, auto filter) const
Definition: aabb.h:167
void ValidateTree(Node *node, size_t num)
Definition: aabb.h:331
static float AABBSurface(vec3 min, vec3 max)
Definition: aabb.h:373
static bool AABBOverlap(vec3 min, vec3 max, vec3 other_min, vec3 other_max)
Definition: aabb.h:363
Serialization, i.e.
glm::vec3 vec3
Definition: math.h:11
Definition: aabb.h:411
Node * left
Definition: aabb.h:417
Node * right
Definition: aabb.h:421
bool IsLeaf() const
Definition: aabb.h:412
Node * parent
Definition: aabb.h:422
vec3 max
Definition: aabb.h:425
vec3 min
Definition: aabb.h:424
uint32_t value
Definition: aabb.h:418
void Print() const
Definition: aabb.h:414
Definition: graph.h:16