11#include <boost/multiprecision/cpp_bin_float.hpp>
34inline std::array<T, 4> det4(
const std::array<T, 12>& s)
36 std::span<const T, 3> s0(s.begin(), 3);
37 std::span<const T, 3> s1(s.begin() + 3, 3);
38 std::span<const T, 3> s2(s.begin() + 6, 3);
39 std::span<const T, 3> s3(s.begin() + 9, 3);
42 T c0 = s2[1] * s3[2] - s2[2] * s3[1];
43 T c1 = s2[0] * s3[2] - s2[2] * s3[0];
44 T c2 = s2[0] * s3[1] - s2[1] * s3[0];
45 w[2] = -s0[0] * c0 + s0[1] * c1 - s0[2] * c2;
46 w[3] = s1[0] * c0 - s1[1] * c1 + s1[2] * c2;
48 c0 = s0[1] * s1[2] - s0[2] * s1[1];
49 c1 = s0[0] * s1[2] - s0[2] * s1[0];
50 c2 = s0[0] * s1[1] - s0[1] * s1[0];
51 w[0] = -s2[0] * c0 + s2[1] * c1 - s2[2] * c2;
52 w[1] = s3[0] * c0 - s3[1] * c1 + s3[2] * c2;
61template <
typename Vec>
62inline Vec::value_type dot3(
const Vec& a,
const Vec& b)
64 return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
75template <
typename T, std::
size_t simplex_size>
76void nearest_simplex(
const std::array<T, 12>& s, std::array<T, 4>& coordinates)
79 SPDLOG_DEBUG(
"GJK: nearest_simplex({})", simplex_size);
81 if constexpr (simplex_size == 2)
85 std::span<const T, 3> s0(s.data(), 3);
86 std::span<const T, 3> s1(s.data() + 3, 3);
88 T lm = dot3(s0, s0) - dot3(s0, s1);
91 SPDLOG_DEBUG(
"GJK: line point A");
97 T mu = dot3(s1, s1) - dot3(s1, s0);
100 SPDLOG_DEBUG(
"GJK: line point B");
101 coordinates[0] = 0.0;
102 coordinates[1] = 1.0;
106 SPDLOG_DEBUG(
"GJK line: AB");
107 T f1 = 1.0 / (lm + mu);
108 coordinates[0] = mu * f1;
109 coordinates[1] = lm * f1;
112 else if constexpr (simplex_size == 3)
116 std::span<const T, 3> a(s.data(), 3);
117 std::span<const T, 3> b(s.data() + 3, 3);
118 std::span<const T, 3> c(s.data() + 6, 3);
125 if (d1 < 0.0 and d2 < 0.0)
127 SPDLOG_DEBUG(
"GJK: Point A");
128 coordinates[0] = 1.0;
129 coordinates[1] = 0.0;
130 coordinates[2] = 0.0;
138 if (d3 < 0.0 and d4 < 0.0)
140 SPDLOG_DEBUG(
"GJK: Point B");
141 coordinates[0] = 0.0;
142 coordinates[1] = 1.0;
143 coordinates[2] = 0.0;
150 if (d5 < 0.0 and d6 < 0.0)
152 SPDLOG_DEBUG(
"GJK: Point C");
153 coordinates[0] = 0.0;
154 coordinates[1] = 0.0;
155 coordinates[2] = 1.0;
159 T vc = d4 * d1 - d1 * d3 + d3 * d2;
160 if (vc < 0.0 and d1 > 0.0 and d3 > 0.0)
162 SPDLOG_DEBUG(
"GJK: edge AB");
163 T f1 = 1.0 / (d1 + d3);
168 coordinates[2] = 0.0;
171 T vb = d1 * d5 - d5 * d2 + d2 * d6;
172 if (vb < 0.0 and d2 > 0.0 and d5 > 0.0)
174 SPDLOG_DEBUG(
"GJK: edge AC");
175 T f1 = 1.0 / (d2 + d5);
179 coordinates[1] = 0.0;
183 T va = d3 * d6 - d6 * d4 + d4 * d5;
184 if (va < 0.0 and d4 > 0.0 and d6 > 0.0)
186 SPDLOG_DEBUG(
"GJK: edge BC");
187 T f1 = 1.0 / (d4 + d6);
190 coordinates[0] = 0.0;
196 SPDLOG_DEBUG(
"GJK: triangle ABC");
197 T f1 = 1.0 / (va + vb + vc);
198 coordinates[0] = va * f1;
199 coordinates[1] = vb * f1;
200 coordinates[2] = vc * f1;
203 else if constexpr (simplex_size == 4)
207 std::ranges::fill(coordinates, 0.0);
210 for (
int i = 0; i < 4; ++i)
213 std::span<const T, 3> si(s.begin() + i * 3, 3);
214 T sii = dot3(si, si);
216 for (
int j = 0; j < 4; ++j)
218 std::span<const T, 3> sj(s.begin() + j * 3, 3);
220 d[i][j] = (sii - dot3(si, sj));
221 SPDLOG_DEBUG(
"d[{}][{}] = {}", i, j,
static_cast<double>(d[i][j]));
228 coordinates[i] = 1.0;
233 SPDLOG_DEBUG(
"Check for edges");
237 int edges[6][2] = {{2, 3}, {1, 3}, {1, 2}, {0, 3}, {0, 2}, {0, 1}};
238 for (
int i = 0; i < 6; ++i)
242 int j0 = edges[i][0];
243 int j1 = edges[i][1];
244 int j2 = edges[5 - i][0];
245 int j3 = edges[5 - i][1];
246 v[i][0] = d[j1][j2] * d[j0][j1] - d[j0][j1] * d[j1][j0]
247 + d[j1][j0] * d[j0][j2];
248 v[i][1] = d[j1][j3] * d[j0][j1] - d[j0][j1] * d[j1][j0]
249 + d[j1][j0] * d[j0][j3];
251 SPDLOG_DEBUG(
"v[{}] = {},{}", i, (
double)v[i][0], (
double)v[i][1]);
252 if (v[i][0] <= 0.0 and v[i][1] <= 0.0 and d[j0][j1] >= 0.0
253 and d[j1][j0] >= 0.0)
256 T f1 = 1.0 / (d[j0][j1] + d[j1][j0]);
257 coordinates[j0] = f1 * d[j1][j0];
258 coordinates[j1] = f1 * d[j0][j1];
264 std::array<T, 4> w = det4(s);
265 T wsum = w[0] + w[1] + w[2] + w[3];
275 if (w[0] < 0.0 and v[2][0] > 0.0 and v[4][0] > 0.0 and v[5][0] > 0.0)
277 T f1 = 1.0 / (v[2][0] + v[4][0] + v[5][0]);
278 coordinates[0] = v[2][0] * f1;
279 coordinates[1] = v[4][0] * f1;
280 coordinates[2] = v[5][0] * f1;
281 coordinates[3] = 0.0;
285 if (w[1] < 0.0 and v[1][0] > 0.0 and v[3][0] > 0.0 and v[5][1] > 0.0)
287 T f1 = 1.0 / (v[1][0] + v[3][0] + v[5][1]);
288 coordinates[0] = v[1][0] * f1;
289 coordinates[1] = v[3][0] * f1;
290 coordinates[2] = 0.0;
291 coordinates[3] = v[5][1] * f1;
295 if (w[2] < 0.0 and v[0][0] > 0.0 and v[3][1] > 0 and v[4][1] > 0.0)
297 T f1 = 1.0 / (v[0][0] + v[3][1] + v[4][1]);
298 coordinates[0] = v[0][0] * f1;
299 coordinates[1] = 0.0;
300 coordinates[2] = v[3][1] * f1;
301 coordinates[3] = v[4][1] * f1;
305 if (w[3] < 0.0 and v[0][1] > 0.0 and v[1][1] > 0.0 and v[2][1] > 0.0)
307 T f1 = 1.0 / (v[0][1] + v[1][1] + v[2][1]);
308 coordinates[0] = 0.0;
309 coordinates[1] = v[0][1] * f1;
310 coordinates[2] = v[1][1] * f1;
311 coordinates[3] = v[2][1] * f1;
316 coordinates[0] = w[3] / wsum;
317 coordinates[1] = w[2] / wsum;
318 coordinates[2] = w[1] / wsum;
319 coordinates[3] = w[0] / wsum;
325 static_assert(simplex_size >= 2 && simplex_size <= 4,
326 "Number of rows defining simplex not supported.");
335inline int support(std::span<const T> bd,
const std::array<T, 3>& v)
338 T qmax = bd[0] * v[0] + bd[1] * v[1] + bd[2] * v[2];
339 for (std::size_t m = 1; m < bd.size() / 3; ++m)
341 T q = bd[3 * m] * v[0] + bd[3 * m + 1] * v[1] + bd[3 * m + 2] * v[2];
366template <std::floating_point T,
367 typename U = boost::multiprecision::cpp_bin_float_double_extended>
369 std::span<const T> q0)
371 assert(p0.size() % 3 == 0);
372 assert(q0.size() % 3 == 0);
374 constexpr int maxk = 15;
375 const U eps = 1000 * std::numeric_limits<U>::epsilon();
378 std::array<U, 3> x_k = {
static_cast<U
>(p0[0]) -
static_cast<U
>(q0[0]),
379 static_cast<U
>(p0[1]) -
static_cast<U
>(q0[1]),
380 static_cast<U
>(p0[2]) -
static_cast<U
>(q0[2])};
382 std::array<U, 12> s = {0};
386 std::array<U, 4> lmn = {0};
388 std::size_t simplex_size = 1;
391 for (k = 0; k < maxk; ++k)
396 const U x_norm2 = impl_gjk::dot3(x_k, x_k);
397 std::array<U, 3> x_k_normalized = x_k;
398 if (x_norm2 > eps * eps)
404 U inv_norm = U(1.0) / sqrt(x_norm2);
405 x_k_normalized[0] *= inv_norm;
406 x_k_normalized[1] *= inv_norm;
407 x_k_normalized[2] *= inv_norm;
410 std::array<T, 3> dir_p = {
static_cast<T
>(-x_k_normalized[0]),
411 static_cast<T
>(-x_k_normalized[1]),
412 static_cast<T
>(-x_k_normalized[2])};
413 std::array<T, 3> dir_q
414 = {
static_cast<T
>(x_k_normalized[0]),
static_cast<T
>(x_k_normalized[1]),
415 static_cast<T
>(x_k_normalized[2])};
416 int ip = impl_gjk::support(p0, dir_p);
417 int iq = impl_gjk::support(q0, dir_q);
421 = {
static_cast<U
>(p0[ip * 3]) -
static_cast<U
>(q0[iq * 3]),
422 static_cast<U
>(p0[ip * 3 + 1]) -
static_cast<U
>(q0[iq * 3 + 1]),
423 static_cast<U
>(p0[ip * 3 + 2]) -
static_cast<U
>(q0[iq * 3 + 2])};
427 for (m = 0; m < simplex_size; ++m)
429 auto it = std::next(s.begin(), 3 * m);
430 if (std::equal(it, std::next(it, 3), s_k.begin(), s_k.end()))
434 if (m != simplex_size)
438 const U xs_diff = x_norm2 - impl_gjk::dot3(x_k, s_k);
439 if (xs_diff < (eps * x_norm2) or xs_diff < eps)
442 SPDLOG_DEBUG(
"GJK: xs_diff={}/{}",
static_cast<double>(xs_diff),
443 static_cast<double>(eps));
446 std::ranges::copy(s_k, s.begin() + 3 * simplex_size);
450 switch (simplex_size)
453 impl_gjk::nearest_simplex<U, 2>(s, lmn);
456 impl_gjk::nearest_simplex<U, 3>(s, lmn);
459 impl_gjk::nearest_simplex<U, 4>(s, lmn);
462 throw std::runtime_error(
"Invalid simplex size");
467 x_k = {0.0, 0.0, 0.0};
468 for (std::size_t i = 0; i < simplex_size; ++i)
470 std::span<const U> sc(std::next(s.begin(), 3 * i), 3);
473 x_k[0] += lmn[i] * sc[0];
474 x_k[1] += lmn[i] * sc[1];
475 x_k[2] += lmn[i] * sc[2];
477 std::ranges::copy(sc, std::next(s.begin(), 3 * j));
484 const U x_next_norm2 = impl_gjk::dot3(x_k, x_k);
485 if (x_norm2 <= x_next_norm2)
489 if (x_next_norm2 < eps * eps)
494 throw std::runtime_error(
"GJK error - max iteration limit reached");
495 return {
static_cast<T
>(x_k[0]),
static_cast<T
>(x_k[1]),
496 static_cast<T
>(x_k[2])};
516template <std::floating_point T,
517 typename U = boost::multiprecision::cpp_bin_float_double_extended>
520 std::span<const T> q,
size_t num_threads)
522 size_t total_size = bodies.size();
523 num_threads = std::min(num_threads, total_size);
525 std::vector<T> results(total_size * 3);
527 = [&results, &bodies](
size_t c0,
size_t c1, std::span<const T> q_ref)
529 for (
size_t i = c0; i < c1; ++i)
533 results[3 * i + 0] = dist[0];
534 results[3 * i + 1] = dist[1];
535 results[3 * i + 2] = dist[2];
539 if (num_threads <= 1)
541 compute_chunk(0, total_size, q);
545 std::vector<std::jthread> threads(num_threads);
546 for (
size_t i = 0; i < num_threads; ++i)
549 threads[i] = std::jthread(compute_chunk, c0, c1, q);
constexpr std::array< std::int64_t, 2 > local_range(int rank, std::int64_t N, int size)
Return local range for the calling process, partitioning the global [0, N - 1] range across all ranks...
Definition MPI.h:89
Geometry data structures and algorithms.
Definition BoundingBoxTree.h:22
std::vector< T > compute_distances_gjk(const std::vector< std::span< const T > > &bodies, std::span< const T > q, size_t num_threads)
Compute the distance between a sequence of convex bodies p0, ..., pN and q, each defined by a set of ...
Definition gjk.h:519
std::array< T, 3 > compute_distance_gjk(std::span< const T > p0, std::span< const T > q0)
Compute the distance between two convex bodies p0 and q0, each defined by a set of points.
Definition gjk.h:368