11#include <boost/multiprecision/cpp_bin_float.hpp>
33inline std::array<T, 4> det4(
const std::array<T, 12>& s)
35 std::span<const T, 3> s0(s.begin(), 3);
36 std::span<const T, 3> s1(s.begin() + 3, 3);
37 std::span<const T, 3> s2(s.begin() + 6, 3);
38 std::span<const T, 3> s3(s.begin() + 9, 3);
41 T c0 = s2[1] * s3[2] - s2[2] * s3[1];
42 T c1 = s2[0] * s3[2] - s2[2] * s3[0];
43 T c2 = s2[0] * s3[1] - s2[1] * s3[0];
44 w[2] = -s0[0] * c0 + s0[1] * c1 - s0[2] * c2;
45 w[3] = s1[0] * c0 - s1[1] * c1 + s1[2] * c2;
47 c0 = s0[1] * s1[2] - s0[2] * s1[1];
48 c1 = s0[0] * s1[2] - s0[2] * s1[0];
49 c2 = s0[0] * s1[1] - s0[1] * s1[0];
50 w[0] = -s2[0] * c0 + s2[1] * c1 - s2[2] * c2;
51 w[1] = s3[0] * c0 - s3[1] * c1 + s3[2] * c2;
60template <
typename Vec>
61inline Vec::value_type dot3(
const Vec& a,
const Vec& b)
63 return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
74template <
typename T, std::
size_t simplex_size>
75void nearest_simplex(
const std::array<T, 12>& s, std::array<T, 4>& coordinates)
78 SPDLOG_DEBUG(
"GJK: nearest_simplex({})", simplex_size);
80 if constexpr (simplex_size == 2)
84 std::span<const T, 3> s0(s.data(), 3);
85 std::span<const T, 3> s1(s.data() + 3, 3);
87 T lm = dot3(s0, s0) - dot3(s0, s1);
90 SPDLOG_DEBUG(
"GJK: line point A");
96 T mu = dot3(s1, s1) - dot3(s1, s0);
99 SPDLOG_DEBUG(
"GJK: line point B");
100 coordinates[0] = 0.0;
101 coordinates[1] = 1.0;
105 SPDLOG_DEBUG(
"GJK line: AB");
106 T f1 = 1.0 / (lm + mu);
107 coordinates[0] = mu * f1;
108 coordinates[1] = lm * f1;
111 else if constexpr (simplex_size == 3)
115 std::span<const T, 3> a(s.data(), 3);
116 std::span<const T, 3> b(s.data() + 3, 3);
117 std::span<const T, 3> c(s.data() + 6, 3);
124 if (d1 < 0.0 and d2 < 0.0)
126 SPDLOG_DEBUG(
"GJK: Point A");
127 coordinates[0] = 1.0;
128 coordinates[1] = 0.0;
129 coordinates[2] = 0.0;
137 if (d3 < 0.0 and d4 < 0.0)
139 SPDLOG_DEBUG(
"GJK: Point B");
140 coordinates[0] = 0.0;
141 coordinates[1] = 1.0;
142 coordinates[2] = 0.0;
149 if (d5 < 0.0 and d6 < 0.0)
151 SPDLOG_DEBUG(
"GJK: Point C");
152 coordinates[0] = 0.0;
153 coordinates[1] = 0.0;
154 coordinates[2] = 1.0;
158 T vc = d4 * d1 - d1 * d3 + d3 * d2;
159 if (vc < 0.0 and d1 > 0.0 and d3 > 0.0)
161 SPDLOG_DEBUG(
"GJK: edge AB");
162 T f1 = 1.0 / (d1 + d3);
167 coordinates[2] = 0.0;
170 T vb = d1 * d5 - d5 * d2 + d2 * d6;
171 if (vb < 0.0 and d2 > 0.0 and d5 > 0.0)
173 SPDLOG_DEBUG(
"GJK: edge AC");
174 T f1 = 1.0 / (d2 + d5);
178 coordinates[1] = 0.0;
182 T va = d3 * d6 - d6 * d4 + d4 * d5;
183 if (va < 0.0 and d4 > 0.0 and d6 > 0.0)
185 SPDLOG_DEBUG(
"GJK: edge BC");
186 T f1 = 1.0 / (d4 + d6);
189 coordinates[0] = 0.0;
195 SPDLOG_DEBUG(
"GJK: triangle ABC");
196 T f1 = 1.0 / (va + vb + vc);
197 coordinates[0] = va * f1;
198 coordinates[1] = vb * f1;
199 coordinates[2] = vc * f1;
202 else if constexpr (simplex_size == 4)
206 std::ranges::fill(coordinates, 0.0);
209 for (
int i = 0; i < 4; ++i)
212 std::span<const T, 3> si(s.begin() + i * 3, 3);
213 T sii = dot3(si, si);
215 for (
int j = 0; j < 4; ++j)
217 std::span<const T, 3> sj(s.begin() + j * 3, 3);
219 d[i][j] = (sii - dot3(si, sj));
220 SPDLOG_DEBUG(
"d[{}][{}] = {}", i, j,
static_cast<double>(d[i][j]));
227 coordinates[i] = 1.0;
232 SPDLOG_DEBUG(
"Check for edges");
236 int edges[6][2] = {{2, 3}, {1, 3}, {1, 2}, {0, 3}, {0, 2}, {0, 1}};
237 for (
int i = 0; i < 6; ++i)
241 int j0 = edges[i][0];
242 int j1 = edges[i][1];
243 int j2 = edges[5 - i][0];
244 int j3 = edges[5 - i][1];
245 v[i][0] = d[j1][j2] * d[j0][j1] - d[j0][j1] * d[j1][j0]
246 + d[j1][j0] * d[j0][j2];
247 v[i][1] = d[j1][j3] * d[j0][j1] - d[j0][j1] * d[j1][j0]
248 + d[j1][j0] * d[j0][j3];
250 SPDLOG_DEBUG(
"v[{}] = {},{}", i, (
double)v[i][0], (
double)v[i][1]);
251 if (v[i][0] <= 0.0 and v[i][1] <= 0.0 and d[j0][j1] >= 0.0
252 and d[j1][j0] >= 0.0)
255 T f1 = 1.0 / (d[j0][j1] + d[j1][j0]);
256 coordinates[j0] = f1 * d[j1][j0];
257 coordinates[j1] = f1 * d[j0][j1];
263 std::array<T, 4> w = det4(s);
264 T wsum = w[0] + w[1] + w[2] + w[3];
274 if (w[0] < 0.0 and v[2][0] > 0.0 and v[4][0] > 0.0 and v[5][0] > 0.0)
276 T f1 = 1.0 / (v[2][0] + v[4][0] + v[5][0]);
277 coordinates[0] = v[2][0] * f1;
278 coordinates[1] = v[4][0] * f1;
279 coordinates[2] = v[5][0] * f1;
280 coordinates[3] = 0.0;
284 if (w[1] < 0.0 and v[1][0] > 0.0 and v[3][0] > 0.0 and v[5][1] > 0.0)
286 T f1 = 1.0 / (v[1][0] + v[3][0] + v[5][1]);
287 coordinates[0] = v[1][0] * f1;
288 coordinates[1] = v[3][0] * f1;
289 coordinates[2] = 0.0;
290 coordinates[3] = v[5][1] * f1;
294 if (w[2] < 0.0 and v[0][0] > 0.0 and v[3][1] > 0 and v[4][1] > 0.0)
296 T f1 = 1.0 / (v[0][0] + v[3][1] + v[4][1]);
297 coordinates[0] = v[0][0] * f1;
298 coordinates[1] = 0.0;
299 coordinates[2] = v[3][1] * f1;
300 coordinates[3] = v[4][1] * f1;
304 if (w[3] < 0.0 and v[0][1] > 0.0 and v[1][1] > 0.0 and v[2][1] > 0.0)
306 T f1 = 1.0 / (v[0][1] + v[1][1] + v[2][1]);
307 coordinates[0] = 0.0;
308 coordinates[1] = v[0][1] * f1;
309 coordinates[2] = v[1][1] * f1;
310 coordinates[3] = v[2][1] * f1;
315 coordinates[0] = w[3] / wsum;
316 coordinates[1] = w[2] / wsum;
317 coordinates[2] = w[1] / wsum;
318 coordinates[3] = w[0] / wsum;
324 static_assert(simplex_size >= 2 && simplex_size <= 4,
325 "Number of rows defining simplex not supported.");
334inline int support(std::span<const T> bd,
const std::array<T, 3>& v)
337 T qmax = bd[0] * v[0] + bd[1] * v[1] + bd[2] * v[2];
338 for (std::size_t m = 1; m < bd.size() / 3; ++m)
340 T q = bd[3 * m] * v[0] + bd[3 * m + 1] * v[1] + bd[3 * m + 2] * v[2];
365template <std::floating_point T,
366 typename U = boost::multiprecision::cpp_bin_float_double_extended>
368 std::span<const T> q0)
370 assert(p0.size() % 3 == 0);
371 assert(q0.size() % 3 == 0);
373 constexpr int maxk = 15;
374 const U eps = 1000 * std::numeric_limits<U>::epsilon();
377 std::array<U, 3> x_k = {
static_cast<U
>(p0[0]) -
static_cast<U
>(q0[0]),
378 static_cast<U
>(p0[1]) -
static_cast<U
>(q0[1]),
379 static_cast<U
>(p0[2]) -
static_cast<U
>(q0[2])};
381 std::array<U, 12> s = {0};
385 std::array<U, 4> lmn = {0};
387 std::size_t simplex_size = 1;
390 for (k = 0; k < maxk; ++k)
395 const U x_norm2 = impl_gjk::dot3(x_k, x_k);
396 std::array<U, 3> x_k_normalized = x_k;
397 if (x_norm2 > eps * eps)
403 U inv_norm = U(1.0) / sqrt(x_norm2);
404 x_k_normalized[0] *= inv_norm;
405 x_k_normalized[1] *= inv_norm;
406 x_k_normalized[2] *= inv_norm;
409 std::array<T, 3> dir_p = {
static_cast<T
>(-x_k_normalized[0]),
410 static_cast<T
>(-x_k_normalized[1]),
411 static_cast<T
>(-x_k_normalized[2])};
412 std::array<T, 3> dir_q
413 = {
static_cast<T
>(x_k_normalized[0]),
static_cast<T
>(x_k_normalized[1]),
414 static_cast<T
>(x_k_normalized[2])};
415 int ip = impl_gjk::support(p0, dir_p);
416 int iq = impl_gjk::support(q0, dir_q);
420 = {
static_cast<U
>(p0[ip * 3]) -
static_cast<U
>(q0[iq * 3]),
421 static_cast<U
>(p0[ip * 3 + 1]) -
static_cast<U
>(q0[iq * 3 + 1]),
422 static_cast<U
>(p0[ip * 3 + 2]) -
static_cast<U
>(q0[iq * 3 + 2])};
426 for (m = 0; m < simplex_size; ++m)
428 auto it = std::next(s.begin(), 3 * m);
429 if (std::equal(it, std::next(it, 3), s_k.begin(), s_k.end()))
433 if (m != simplex_size)
437 const U xs_diff = x_norm2 - impl_gjk::dot3(x_k, s_k);
438 if (xs_diff < (eps * x_norm2) or xs_diff < eps)
441 SPDLOG_DEBUG(
"GJK: xs_diff={}/{}",
static_cast<double>(xs_diff),
442 static_cast<double>(eps));
445 std::ranges::copy(s_k, s.begin() + 3 * simplex_size);
449 switch (simplex_size)
452 impl_gjk::nearest_simplex<U, 2>(s, lmn);
455 impl_gjk::nearest_simplex<U, 3>(s, lmn);
458 impl_gjk::nearest_simplex<U, 4>(s, lmn);
461 throw std::runtime_error(
"Invalid simplex size");
466 x_k = {0.0, 0.0, 0.0};
467 for (std::size_t i = 0; i < simplex_size; ++i)
469 std::span<const U> sc(std::next(s.begin(), 3 * i), 3);
472 x_k[0] += lmn[i] * sc[0];
473 x_k[1] += lmn[i] * sc[1];
474 x_k[2] += lmn[i] * sc[2];
476 std::ranges::copy(sc, std::next(s.begin(), 3 * j));
483 const U x_next_norm2 = impl_gjk::dot3(x_k, x_k);
484 if (x_norm2 <= x_next_norm2)
488 if (x_next_norm2 < eps * eps)
493 throw std::runtime_error(
"GJK error - max iteration limit reached");
494 return {
static_cast<T
>(x_k[0]),
static_cast<T
>(x_k[1]),
495 static_cast<T
>(x_k[2])};
515template <std::floating_point T,
516 typename U = boost::multiprecision::cpp_bin_float_double_extended>
519 std::span<const T> q,
size_t num_threads)
521 size_t total_size = bodies.size();
522 num_threads = std::min(num_threads, total_size);
524 std::vector<T> results(total_size * 3);
526 = [&results, &bodies](
size_t c0,
size_t c1, std::span<const T> q_ref)
528 for (
size_t i = c0; i < c1; ++i)
532 results[3 * i + 0] = dist[0];
533 results[3 * i + 1] = dist[1];
534 results[3 * i + 2] = dist[2];
538 if (num_threads <= 1)
540 compute_chunk(0, total_size, q);
544 std::vector<std::jthread> threads(num_threads);
545 for (
size_t i = 0; i < num_threads; ++i)
548 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:518
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:367