-
Notifications
You must be signed in to change notification settings - Fork 40
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add example foir 2D triangle-point intersection
- Loading branch information
1 parent
db4d081
commit d7e7f3a
Showing
3 changed files
with
389 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
add_executable(ArborX_TriangleIntersection.exe triangle_intersection.cpp) | ||
target_link_libraries(ArborX_TriangleIntersection.exe ArborX::ArborX) | ||
add_test(NAME ArborX_TriangleIntersection_Example COMMAND ./ArborX_TriangleIntersection.exe) |
384 changes: 384 additions & 0 deletions
384
examples/triangle_intersection/triangle_intersection.cpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,384 @@ | ||
/**************************************************************************** | ||
* Copyright (c) 2017-2023 by the ArborX authors * | ||
* All rights reserved. * | ||
* * | ||
* This file is part of the ArborX library. ArborX is * | ||
* distributed under a BSD 3-clause license. For the licensing terms see * | ||
* the LICENSE file in the top-level directory. * | ||
* * | ||
* SPDX-License-Identifier: BSD-3-Clause * | ||
****************************************************************************/ | ||
|
||
#include <ArborX.hpp> | ||
#include <ArborX_HyperTriangle.hpp> | ||
|
||
#include <Kokkos_Core.hpp> | ||
|
||
// Perform intersection queries using 2D triangles on a regular mesh as | ||
// primitives and intersection with points as queries. One point per triangle. | ||
// __________ | ||
// |\x|\x|\x| | ||
// |x\|x\|x\| | ||
// __________ | ||
// |\x|\x|\x| | ||
// |x\|x\|x\| | ||
// __________ | ||
// |\x|\x|\x| | ||
// |x\|x\|x\| | ||
// __________ | ||
|
||
constexpr float Lx = 100.0; | ||
constexpr float Ly = 100.0; | ||
constexpr int nx = 101; | ||
constexpr int ny = 101; | ||
constexpr int n = nx * ny; | ||
constexpr float hx = Lx / (nx - 1); | ||
constexpr float hy = Ly / (ny - 1); | ||
|
||
// The Mapping class stores the mapping from a unit triangle to a given triangle | ||
// allowing for computing the barycentric coordinates for a given point. | ||
struct Mapping | ||
{ | ||
ArborX::ExperimentalHyperGeometry::Point<2> alpha; | ||
ArborX::ExperimentalHyperGeometry::Point<2> beta; | ||
ArborX::ExperimentalHyperGeometry::Point<2> p0; | ||
|
||
// x = a + alpha * (b - a) + beta * (c - a) | ||
// = (1-beta-alpha) * a + alpha * b + beta * c | ||
KOKKOS_FUNCTION | ||
Mapping(ArborX::ExperimentalHyperGeometry::Triangle<2> const &triangle) | ||
{ | ||
auto const &a = triangle.a; | ||
auto const &b = triangle.b; | ||
auto const &c = triangle.c; | ||
|
||
ArborX::ExperimentalHyperGeometry::Point<2> u = {b[0] - a[0], b[1] - a[1]}; | ||
ArborX::ExperimentalHyperGeometry::Point<2> v = {c[0] - a[0], c[1] - a[1]}; | ||
|
||
float const inv_det = 1. / (v[1] * u[0] - v[0] * u[1]); | ||
|
||
alpha = ArborX::ExperimentalHyperGeometry::Point<2>{v[1] * inv_det, | ||
-v[0] * inv_det}; | ||
beta = ArborX::ExperimentalHyperGeometry::Point<2>{-u[1] * inv_det, | ||
u[0] * inv_det}; | ||
p0 = a; | ||
} | ||
|
||
KOKKOS_FUNCTION ArborX::Point | ||
get_coeff(ArborX::ExperimentalHyperGeometry::Point<2> p) const | ||
{ | ||
float alpha_coeff = alpha[0] * (p[0] - p0[0]) + alpha[1] * (p[1] - p0[1]); | ||
float beta_coeff = beta[0] * (p[0] - p0[0]) + beta[1] * (p[1] - p0[1]); | ||
return {1 - alpha_coeff - beta_coeff, alpha_coeff, beta_coeff}; | ||
} | ||
|
||
#ifndef NDEBUG | ||
// Recover the triangle from the mapping. Only used for debugging. | ||
KOKKOS_FUNCTION ArborX::ExperimentalHyperGeometry::Triangle<2> | ||
get_triangle() const | ||
{ | ||
float const inv_det = 1. / (alpha[0] * beta[1] - alpha[1] * beta[0]); | ||
ArborX::ExperimentalHyperGeometry::Point<2> a = p0; | ||
ArborX::ExperimentalHyperGeometry::Point<2> b = { | ||
{p0[0] + inv_det * beta[1], p0[1] - inv_det * beta[0]}}; | ||
ArborX::ExperimentalHyperGeometry::Point<2> c = { | ||
{p0[0] - inv_det * alpha[1], p0[1] + inv_det * alpha[0]}}; | ||
return {a, b, c}; | ||
} | ||
#endif | ||
}; | ||
|
||
// Store the points that represent the queries. | ||
template <typename DeviceType> | ||
class Points | ||
{ | ||
public: | ||
Points(typename DeviceType::execution_space const &execution_space) | ||
{ | ||
initialize(execution_space); | ||
} | ||
|
||
void initialize(typename DeviceType::execution_space const &execution_space) | ||
{ | ||
points_ = Kokkos::View<ArborX::ExperimentalHyperGeometry::Point<2> *, | ||
typename DeviceType::memory_space>( | ||
Kokkos::view_alloc(Kokkos::WithoutInitializing, "points"), 2 * n); | ||
|
||
Kokkos::parallel_for( | ||
Kokkos::MDRangePolicy<Kokkos::Rank<2>, | ||
typename DeviceType::execution_space>( | ||
execution_space, {0, 0}, {nx, ny}), | ||
KOKKOS_CLASS_LAMBDA(int i, int j) { | ||
auto index = [](int i, int j) { return i + j * nx; }; | ||
|
||
points_[2 * index(i, j)] = {(i + .25f) * hx, (j + .25f) * hy}; | ||
points_[2 * index(i, j) + 1] = {(i + .75f) * hx, (j + .75f) * hy}; | ||
}); | ||
} | ||
|
||
KOKKOS_FUNCTION auto const &get_point(int i) const { return points_(i); } | ||
|
||
KOKKOS_FUNCTION auto size() const { return points_.size(); } | ||
|
||
private: | ||
Kokkos::View<ArborX::ExperimentalHyperGeometry::Point<2> *, | ||
typename DeviceType::memory_space> | ||
points_; | ||
}; | ||
|
||
template <typename DeviceType> | ||
class Triangles | ||
{ | ||
public: | ||
Triangles(typename DeviceType::execution_space const &execution_space) | ||
{ | ||
initialize(execution_space); | ||
} | ||
|
||
// Create non-intersecting triangles on a 2D cartesian grid | ||
// used for the primitives in the tree construction, and compute and store the | ||
// mappings used in the queries. | ||
void initialize(typename DeviceType::execution_space const &execution_space) | ||
{ | ||
using ExecutionSpaceType = typename DeviceType::execution_space; | ||
using MemorySpaceType = typename DeviceType::memory_space; | ||
|
||
triangles_ = Kokkos::View<ArborX::ExperimentalHyperGeometry::Triangle<2> *, | ||
MemorySpaceType>( | ||
Kokkos::view_alloc(Kokkos::WithoutInitializing, "triangles"), 2 * n); | ||
mappings_ = Kokkos::View<Mapping *, MemorySpaceType>( | ||
Kokkos::view_alloc(Kokkos::WithoutInitializing, "mappings"), 2 * n); | ||
|
||
Kokkos::parallel_for( | ||
Kokkos::MDRangePolicy<Kokkos::Rank<2>, ExecutionSpaceType>( | ||
execution_space, {0, 0}, {nx, ny}), | ||
KOKKOS_CLASS_LAMBDA(int i, int j) { | ||
ArborX::ExperimentalHyperGeometry::Point<2> bl{i * hx, j * hy}; | ||
ArborX::ExperimentalHyperGeometry::Point<2> br{(i + 1) * hx, j * hy}; | ||
ArborX::ExperimentalHyperGeometry::Point<2> tl{i * hx, (j + 1) * hy}; | ||
ArborX::ExperimentalHyperGeometry::Point<2> tr{(i + 1) * hx, | ||
(j + 1) * hy}; | ||
|
||
auto index = [](int i, int j) { return i + j * nx; }; | ||
|
||
triangles_[2 * index(i, j)] = {tl, bl, br}; | ||
mappings_[2 * index(i, j)] = Mapping(triangles_[2 * index(i, j)]); | ||
triangles_[2 * index(i, j) + 1] = {tl, br, tr}; | ||
mappings_[2 * index(i, j) + 1] = | ||
Mapping(triangles_[2 * index(i, j) + 1]); | ||
}); | ||
|
||
#ifndef NDEBUG | ||
Kokkos::parallel_for( | ||
Kokkos::RangePolicy<ExecutionSpaceType>(execution_space, 0, 2 * n), | ||
KOKKOS_CLASS_LAMBDA(int k) { | ||
ArborX::ExperimentalHyperGeometry::Triangle<2> recover_triangle = | ||
mappings_[k].get_triangle(); | ||
|
||
constexpr float eps = 1.e-3; | ||
|
||
for (unsigned int i = 0; i < 2; ++i) | ||
if (Kokkos::abs(triangles_[k].a[i] - recover_triangle.a[i]) > eps) | ||
Kokkos::abort( | ||
"Mismatch for first point when recovering triangle"); | ||
|
||
for (unsigned int i = 0; i < 2; ++i) | ||
if (Kokkos::abs(triangles_[k].b[i] - recover_triangle.b[i]) > eps) | ||
Kokkos::abort( | ||
"Mismatch for second point when recovering triangle"); | ||
|
||
for (unsigned int i = 0; i < 2; ++i) | ||
if (Kokkos::abs(triangles_[k].c[i] - recover_triangle.c[i]) > eps) | ||
Kokkos::abort( | ||
"Mismatch for third point when recoverning triangle"); | ||
|
||
auto const &coeff_a = mappings_[k].get_coeff(triangles_[k].a); | ||
if ((Kokkos::abs(coeff_a[0] - 1.) > eps) || | ||
Kokkos::abs(coeff_a[1]) > eps || Kokkos::abs(coeff_a[2]) > eps) | ||
Kokkos::abort( | ||
"Mismatch for coefficients of first point in triangle"); | ||
|
||
auto const &coeff_b = mappings_[k].get_coeff(triangles_[k].b); | ||
if ((Kokkos::abs(coeff_b[0]) > eps) || | ||
Kokkos::abs(coeff_b[1] - 1.) > eps || | ||
Kokkos::abs(coeff_b[2]) > eps) | ||
Kokkos::abort( | ||
"Mismatch for coefficients of first point in triangle"); | ||
|
||
auto const &coeff_c = mappings_[k].get_coeff(triangles_[k].c); | ||
if ((Kokkos::abs(coeff_c[0]) > eps) || | ||
Kokkos::abs(coeff_c[1]) > eps || | ||
Kokkos::abs(coeff_c[2] - 1.) > eps) | ||
Kokkos::abort( | ||
"Mismatch for coefficients of first point in triangle"); | ||
}); | ||
#endif | ||
} | ||
|
||
KOKKOS_FUNCTION int size() const { return triangles_.size(); } | ||
|
||
KOKKOS_FUNCTION ArborX::ExperimentalHyperGeometry::Triangle<2> const & | ||
get_triangle(int i) const | ||
{ | ||
return triangles_(i); | ||
} | ||
|
||
KOKKOS_FUNCTION Mapping const &get_mapping(int i) const | ||
{ | ||
return mappings_(i); | ||
} | ||
|
||
private: | ||
Kokkos::View<ArborX::ExperimentalHyperGeometry::Triangle<2> *, | ||
typename DeviceType::memory_space> | ||
triangles_; | ||
Kokkos::View<Mapping *, typename DeviceType::memory_space> mappings_; | ||
}; | ||
|
||
// For creating the bounding volume hierarchy given a Triangles object, we | ||
// need to define the memory space, how to get the total number of objects, | ||
// and how to access a specific box. Since there are corresponding functions in | ||
// the Triangles class, we just resort to them. | ||
template <typename DeviceType> | ||
struct ArborX::AccessTraits<Triangles<DeviceType>, ArborX::PrimitivesTag> | ||
{ | ||
using memory_space = typename DeviceType::memory_space; | ||
|
||
static KOKKOS_FUNCTION int size(Triangles<DeviceType> const &triangles) | ||
{ | ||
return triangles.size(); | ||
} | ||
|
||
static KOKKOS_FUNCTION auto get(Triangles<DeviceType> const &triangles, int i) | ||
{ | ||
auto const &triangle = triangles.get_triangle(i); | ||
ArborX::ExperimentalHyperGeometry::Box<2> box{}; | ||
box += triangle.a; | ||
box += triangle.b; | ||
box += triangle.c; | ||
return box; | ||
} | ||
}; | ||
|
||
template <typename DeviceType> | ||
struct ArborX::AccessTraits<Points<DeviceType>, ArborX::PredicatesTag> | ||
{ | ||
using memory_space = typename DeviceType::memory_space; | ||
static KOKKOS_FUNCTION int size(Points<DeviceType> const &points) | ||
{ | ||
return points.size(); | ||
} | ||
static KOKKOS_FUNCTION auto get(Points<DeviceType> const &points, int i) | ||
{ | ||
return ArborX::attach(ArborX::intersects(points.get_point(i)), i); | ||
} | ||
}; | ||
|
||
template <typename DeviceType> | ||
class TriangleIntersectionCallback | ||
{ | ||
public: | ||
TriangleIntersectionCallback( | ||
Triangles<DeviceType> triangles, Kokkos::View<int *, DeviceType> offsets, | ||
Kokkos::View<ArborX::Point *, DeviceType> coefficients) | ||
: triangles_(triangles) | ||
, offsets_(offsets) | ||
, coefficients_(coefficients) | ||
{} | ||
|
||
// The search tree consists entirely of boxes, although the primitives are | ||
// triangles. Thus, a detected collision doesn't mean that the point is | ||
// actuallt inside the wrapped triangle and we have to check that here. This | ||
// also gives us the opportunity to store the barycentric coordinates in case | ||
// there is an intersection. Since the triangles don't overlap in this | ||
// example, there is at most one triangle that contains a given point and we | ||
// can abort the search early when we found a match. | ||
template <typename Query, typename Primitive> | ||
KOKKOS_FUNCTION auto operator()(Query const &query, | ||
Primitive const &primitive) const | ||
{ | ||
ArborX::ExperimentalHyperGeometry::Point<2> const &point = | ||
getGeometry(getPredicate(query)); | ||
auto query_index = ArborX::getData(query); | ||
|
||
auto const coeffs = | ||
triangles_.get_mapping(primitive.index).get_coeff(point); | ||
bool intersects = coeffs[0] >= 0 && coeffs[1] >= 0 && coeffs[2] >= 0; | ||
|
||
if (intersects) | ||
{ | ||
offsets_(query_index) = primitive.index; | ||
coefficients_(query_index) = coeffs; | ||
return ArborX::CallbackTreeTraversalControl::early_exit; | ||
} | ||
return ArborX::CallbackTreeTraversalControl::normal_continuation; | ||
} | ||
|
||
private: | ||
Triangles<DeviceType> triangles_; | ||
Kokkos::View<int *, DeviceType> offsets_; | ||
Kokkos::View<ArborX::Point *, DeviceType> coefficients_; | ||
}; | ||
|
||
// Now that we have encapsulated the objects and queries to be used within the | ||
// Triangles class, we can continue with performing the actual search. | ||
int main() | ||
{ | ||
Kokkos::initialize(); | ||
{ | ||
using ExecutionSpace = Kokkos::DefaultExecutionSpace; | ||
using MemorySpace = typename ExecutionSpace::memory_space; | ||
using DeviceType = Kokkos::Device<ExecutionSpace, MemorySpace>; | ||
ExecutionSpace execution_space; | ||
|
||
std::cout << "Create grid with triangles.\n"; | ||
Triangles<DeviceType> triangles(execution_space); | ||
std::cout << "Triangles set up.\n"; | ||
|
||
std::cout << "Creating BVH tree.\n"; | ||
ArborX::BasicBoundingVolumeHierarchy< | ||
MemorySpace, ArborX::Details::PairIndexVolume< | ||
ArborX::ExperimentalHyperGeometry::Box<2>>> const | ||
tree(execution_space, triangles); | ||
std::cout << "BVH tree set up.\n"; | ||
|
||
std::cout << "Create the points used for queries.\n"; | ||
Points<DeviceType> points(execution_space); | ||
std::cout << "Points for queries set up.\n"; | ||
|
||
std::cout << "Starting the queries.\n"; | ||
int const n = points.size(); | ||
Kokkos::View<int *, MemorySpace> offsets("offsets", n); | ||
Kokkos::View<ArborX::Point *, MemorySpace> coefficients("coefficients", n); | ||
|
||
tree.query(execution_space, points, | ||
TriangleIntersectionCallback<DeviceType>{triangles, offsets, | ||
coefficients}); | ||
std::cout << "Queries done.\n"; | ||
|
||
#ifndef NDEBUG | ||
std::cout << "Starting checking results.\n"; | ||
Kokkos::parallel_for( | ||
Kokkos::RangePolicy<ExecutionSpace>(execution_space, 0, n), | ||
KOKKOS_LAMBDA(int i) { | ||
constexpr float eps = 1.e-3; | ||
|
||
if (offsets(i) != i) | ||
Kokkos::abort("Offsets are wrong"); | ||
auto const &c = coefficients(i); | ||
auto const &t = triangles.get_triangle(offsets(i)); | ||
auto const &p_h = points.get_point(i); | ||
auto const p = ArborX::ExperimentalHyperGeometry::Point<2>{ | ||
c[0] * t.a[0] + c[1] * t.b[0] + c[2] * t.c[0], | ||
c[0] * t.a[1] + c[1] * t.b[1] + c[2] * t.c[1]}; | ||
if ((Kokkos::abs(p[0] - p_h[0]) > eps) || | ||
Kokkos::abs(p[1] - p_h[1]) > eps) | ||
Kokkos::abort("Coefficients are wrong"); | ||
}); | ||
std::cout << "Checking results successful.\n"; | ||
#endif | ||
execution_space.fence(); | ||
} | ||
|
||
Kokkos::finalize(); | ||
} |