21#ifndef HCSBB_GAUSS_NNN
22#define HCSBB_GAUSS_NNN (4)
25#define HCSBB_D_NNN (7)
28#ifndef HCSBB_GAUSS_ORDER
29#define HCSBB_GAUSS_ORDER (6)
33#if HCSBB_GAUSS_ORDER == 0
34#define HCSBB_NUM_GAUSS_POINTS (1)
35#elif HCSBB_GAUSS_ORDER == 1
36#define HCSBB_NUM_GAUSS_POINTS (1)
37#elif HCSBB_GAUSS_ORDER == 2
38#define HCSBB_NUM_GAUSS_POINTS (3)
39#elif HCSBB_GAUSS_ORDER == 3
40#define HCSBB_NUM_GAUSS_POINTS (4)
41#elif HCSBB_GAUSS_ORDER == 4
42#define HCSBB_NUM_GAUSS_POINTS (6)
43#elif HCSBB_GAUSS_ORDER == 5
44#define HCSBB_NUM_GAUSS_POINTS (7)
45#elif HCSBB_GAUSS_ORDER == 6
46#define HCSBB_NUM_GAUSS_POINTS (12)
47#elif HCSBB_GAUSS_ORDER == 7
48#define HCSBB_NUM_GAUSS_POINTS (13)
49#elif HCSBB_GAUSS_ORDER == 8
50#define HCSBB_NUM_GAUSS_POINTS (16)
51#elif HCSBB_GAUSS_ORDER == 9
52#define HCSBB_NUM_GAUSS_POINTS (19)
54#define HCSBB_NUM_GAUSS_POINTS (-1)
55#error "interpolation_method_hcsbb: the specified gauss order is currently not supported."
61#define HCSBB_NUM_SBB_COEFFS (10)
65#define SB_COORD_TOL (1.0e-6)
69 size_t * tgt_points,
size_t count,
71 int * interpolation_complete);
153 double coordinate_xyz[3];
202 double * sb_coords,
size_t num_vertices,
double triangle[3][3],
203 char const * caller) {
206 lapack_int n = 3, nrhs = (lapack_int) num_vertices, lda = n, ldx = n, ipiv[3];
207 memcpy(A, triangle,
sizeof(A));
217 LAPACK_COL_MAJOR, n, nrhs, &A[0][0], lda, ipiv, sb_coords, ldx),
218 "ERROR(%s::compute_sb_coords): "
219 "internal error (could not solve linear 3x3 system)\n"
220 "(vector: (% .6e;% .6e;% .6e)\n"
221 " triangle: ((% .6e;% .6e;% .6e),\n"
222 " (% .6e;% .6e;% .6e),\n"
223 " (% .6e;% .6e;% .6e))",
224 caller, sb_coords[0], sb_coords[1], sb_coords[2],
225 triangle[0][0], triangle[0][1], triangle[0][2],
226 triangle[1][0], triangle[1][1], triangle[1][2],
227 triangle[2][0], triangle[2][1], triangle[2][2])
232 size_t const * a_ = a, * b_ = b;
234 return (*a_ > *b_) - (*b_ > *a_);
239 size_t const * a_ = a, * b_ = b;
241 if ((ret = (a_[0] > b_[0]) - (b_[0] > a_[0])))
return ret;
242 else return (a_[1] > b_[1]) - (b_[1] > a_[1]);
247 size_t const * a_ = a, * b_ = b;
249 if ((ret = (a_[0] > b_[0]) - (b_[0] > a_[0])))
return ret;
250 if ((ret = (a_[1] > b_[1]) - (b_[1] > a_[1])))
return ret;
251 else return (a_[2] > b_[2]) - (b_[2] > a_[2]);
256 size_t const * a_ = a, * b_ = b;
259 if ((ret = (a_[i] > b_[i]) - (b_[i] > a_[i])))
270 size_t num_triangles,
271 size_t **unique_vertices,
size_t * num_unique_vertices,
272 size_t (**unique_edge_to_unique_vertex)[2],
size_t * num_unique_edges,
273 size_t (**unique_triangle_to_unique_edge)[3],
size_t * num_unique_triangles) {
275 size_t (*edge_vertices)[2] =
276 xmalloc((num_edges + 3 * num_triangles) *
sizeof(*edge_vertices));
277 size_t (*triangle_vertices)[3] =
278 xmalloc(num_triangles *
sizeof(*triangle_vertices));
281 for (
size_t i = 0; i < num_edges; ++i, ++tgt_point_data)
282 memcpy(edge_vertices[i], tgt_point_data->
src_points, 2 *
sizeof(
size_t));
283 for (
size_t i = 0; i < num_triangles; ++i, ++tgt_point_data) {
284 edge_vertices[3*i+num_edges+0][0] = tgt_point_data->
src_points[0];
285 edge_vertices[3*i+num_edges+0][1] = tgt_point_data->
src_points[1];
286 edge_vertices[3*i+num_edges+1][0] = tgt_point_data->
src_points[0];
287 edge_vertices[3*i+num_edges+1][1] = tgt_point_data->
src_points[2];
288 edge_vertices[3*i+num_edges+2][0] = tgt_point_data->
src_points[1];
289 edge_vertices[3*i+num_edges+2][1] = tgt_point_data->
src_points[2];
291 triangle_vertices[i], tgt_point_data->
src_points, 3 *
sizeof(
size_t));
295 qsort(edge_vertices, num_edges + 3 * num_triangles,
297 qsort(triangle_vertices, num_triangles,
300 *num_unique_edges = num_edges + 3 * num_triangles;
301 *num_unique_triangles = num_triangles;
308 *num_unique_vertices = 2 * (*num_unique_edges);
309 size_t * vertices =
xmalloc((*num_unique_vertices) *
sizeof(*vertices));
311 vertices, &(edge_vertices[0][0]), *num_unique_vertices *
sizeof(*vertices));
312 qsort(vertices, *num_unique_vertices,
sizeof(*vertices),
compare_size_t);
315 xrealloc(vertices, *num_unique_vertices *
sizeof(*vertices));
317 size_t (*triangle_edges)[3] =
318 xmalloc(3 * (*num_unique_triangles) *
sizeof(*triangle_edges));
319 for (
size_t i = 0; i < *num_unique_triangles; ++i) {
321 triangle_edges[3*i+0][0] = triangle_vertices[i][1];
322 triangle_edges[3*i+0][1] = triangle_vertices[i][2];
323 triangle_edges[3*i+0][2] = 3*i+0;
325 triangle_edges[3*i+1][0] = triangle_vertices[i][0];
326 triangle_edges[3*i+1][1] = triangle_vertices[i][2];
327 triangle_edges[3*i+1][2] = 3*i+1;
329 triangle_edges[3*i+2][0] = triangle_vertices[i][0];
330 triangle_edges[3*i+2][1] = triangle_vertices[i][1];
331 triangle_edges[3*i+2][2] = 3*i+2;
335 qsort(triangle_edges, 3 * (*num_unique_triangles),
339 *unique_triangle_to_unique_edge =
340 xmalloc(*num_unique_triangles *
sizeof(**unique_triangle_to_unique_edge));
341 for (
size_t i = 0, j = 0; i < 3 * (*num_unique_triangles); ++i) {
342 size_t * curr_edge = triangle_edges[i];
344 (&(*unique_triangle_to_unique_edge)[0][0])[curr_edge[2]] = j;
346 free(triangle_edges);
348 tgt_point_data -= num_edges + num_triangles;
349 for (
size_t i = 0, j = 0; i < num_edges; ++i, ++tgt_point_data) {
352 j < *num_unique_edges,
"ERROR(get_unique_interp_data): edge ids missmatch")
355 for (
size_t i = 0, j = 0; i < num_triangles; ++i, ++tgt_point_data) {
359 j < *num_unique_triangles,
360 "ERROR(get_unique_interp_data): triangle ids missmatch")
363 free(triangle_vertices);
366 *unique_edge_to_unique_vertex =
367 xmalloc(*num_unique_edges *
sizeof(**unique_edge_to_unique_vertex));
368 size_t * reorder_idx =
369 xmalloc(2 * (*num_unique_edges) *
sizeof(*reorder_idx));
370 for (
size_t i = 0; i < 2 * (*num_unique_edges); ++i) reorder_idx[i] = i;
372 &(edge_vertices[0][0]), 2 * (*num_unique_edges), reorder_idx);
373 for (
size_t i = 0, j = 0; i < 2 * (*num_unique_edges); ++i) {
374 size_t curr_vertex = (&(edge_vertices[0][0]))[i];
375 while (curr_vertex != (*unique_vertices)[j]) ++j;
376 (&(*unique_edge_to_unique_vertex[0][0]))[reorder_idx[i]] = j;
384 double (*gauss_points)[3],
size_t * nnn_search_results,
390 memcpy(src_point_buffer, nnn_search_results,
398 for (
size_t i = 0; i < num_unique_result_points; ++i)
399 global_id_buffer[i] = src_point_global_ids[src_point_buffer[i]];
401 global_id_buffer, num_unique_result_points, src_point_buffer);
404 xmalloc(num_unique_result_points *
sizeof(*src_point_buffer));
406 num_unique_result_points *
sizeof(*src_point_buffer));
411 for (
size_t i = 0; i < num_unique_result_points; ++i)
419 size_t * curr_result_points = nnn_search_results + i *
HCSBB_GAUSS_NNN;
420 double * curr_gauss_points = gauss_points[i];
421 double distance_sum = 0.0;
430 (
double*)(src_point_coords[curr_result_points[j]]));
435 double inv_distance = 1.0 / (distance * distance);
436 inv_distances[j] = inv_distance;
437 distance_sum += inv_distance;
445 curr_global_ids[k] = src_point_global_ids[curr_result_points[k]];
452 distance_sum = 1.0 / distance_sum;
454 yac_int curr_global_id = curr_global_ids[k];
455 while (curr_global_id != global_id_buffer[l]) ++l;
456 w_nnn[l][i] = inv_distances[reorder_idx[k]] * distance_sum;
461 yac_int curr_global_id = src_point_global_ids[curr_result_points[j]];
463 while (curr_global_id != global_id_buffer[l]) ++l;
474 xmalloc(num_unique_result_points *
sizeof(*weights));
476 for (
size_t i = 0; i < num_unique_result_points; ++i) {
480 accu += w_g[j][l] * w_nnn[i][l];
481 weights[i][j] = accu;
493 double bnd_triangle[3][3],
double direction[3],
double coordinate_xyz[3],
496 double sb_coords[2][3] =
497 {{coordinate_xyz[0], coordinate_xyz[1], coordinate_xyz[2]},
498 {direction[0], direction[1], direction[2]}};
503 &(sb_coords[0][0]), 2, bnd_triangle,
"compute_d_sbb_polynomials_3d");
505#define POW_0(x) (1.0)
506#define POW_1(x) ((x))
507#define POW_2(x) ((x)*(x))
508#define POW_3(x) ((x)*(x)*(x))
513#define SB_COORD_P (sb_coords[0])
514#define SB_COORD_D (sb_coords[1])
516 d_sbb_polynomials[0] =
523 d_sbb_polynomials[1] =
530 d_sbb_polynomials[2] =
537 d_sbb_polynomials[3] =
544 d_sbb_polynomials[4] =
551 d_sbb_polynomials[5] =
558 d_sbb_polynomials[6] =
565 d_sbb_polynomials[7] =
572 d_sbb_polynomials[8] =
579 d_sbb_polynomials[9] =
602 double mult_factor) {
608 d_sbb_vertex_polynomials);
614 for (
size_t i = 0, k = 0; i < num_src_points; ++i) {
618 weight += d_sbb_vertex_polynomials[j] * d_data_weights[k];
620 weights[i].
weight = weight * mult_factor;
621 weights[i].
point_id = src_points[i];
628 size_t edge_weight_vector_data_buffer_size = 0;
630 for (
size_t i = 0; i < num_edges; ++i)
631 edge_weight_vector_data_buffer_size +=
639 (edge_weight_vector_data_buffer_size > 0)?
640 xmalloc(edge_weight_vector_data_buffer_size *
641 sizeof(*curr_edge_weight_vector_data)):NULL;
643 for (
size_t i = 0; i < num_edges; ++i) {
649 double direction[3] = {
658 for (
size_t j = 0; j < 2; ++j) {
665 curr_edge_weight_vector_data, 1.0/3.0);
667 curr_edge_weight_vector_data[n-1].
weight = 1.0;
668 curr_edge_weight_vector_data[n-1].
point_id =
670 curr_edge->
c[j].
data = curr_edge_weight_vector_data;
671 curr_edge->
c[j].
n = n;
673 direction[0] *= -1, direction[1] *= -1, direction[2] *= -1;
675 curr_edge_weight_vector_data += n;
681 void const * a,
void const * b) {
691 double abs_weight_a = fabs(weight_a->
weight);
692 double abs_weight_b = fabs(weight_b->
weight);
693 ret = (abs_weight_a > abs_weight_b) - (abs_weight_a < abs_weight_b);
700 void const * a,
void const * b) {
712 void const * a,
void const * b) {
726 for (
size_t i = 0; i < n_; ++i) {
727 buffer[i].weight = weights + i;
737 for (
size_t i = 1; i < n_; ++i, ++curr_weight_data_pos) {
742 if (prev_weight_data_pos->
pos > curr_weight_data_pos->
pos)
743 prev_weight_data_pos->
pos = curr_weight_data_pos->
pos;
746 ++prev_weight_data_pos;
747 *prev_weight_data_pos = *curr_weight_data_pos;
748 prev_weight_data = prev_weight_data_pos->
weight;
752 if (n_ == new_n)
return;
756 for (
size_t i = 0; i < new_n; ++i) weights[i] = *(
buffer[i].
weight);
771 sizeof(*weight_vector_data_buffer));
775 sizeof(*weight_vector_data_pos_buffer));
778 for (
size_t i = 0; i < num_triangles; ++i, ++curr_triangle) {
810 size_t src_points[3] = {
818 {.
weight = 1.0, .point_id = src_points[0]},
819 {.weight = 1.0, .point_id = src_points[1]},
820 {.weight = 1.0, .point_id = src_points[2]}};
822 {.data = &(c_3_data[1]), .
n = 1},
823 {.data = &(c_3_data[2]), .
n = 1}};
834 double corner_coordinate_xyz[3][3] = {
848 &(curr_edges[0]->
g[0]), &(curr_edges[1]->g[0]), &(curr_edges[2]->
g[0])};
851 double b_g[3][3] = {{g[0][0], g[0][1], g[0][2]},
852 {g[1][0], g[1][1], g[1][2]},
853 {g[2][0], g[2][1], g[2][2]}};
855 &(b_g[0][0]), 3, corner_coordinate_xyz,
"compute_triangle_coefficients");
878#define COPY_D_DATA(edge_idx) \
880 get_derivative_weights( \
881 curr_edges[edge_idx]->middle_d_data.gauss_nnn_patch, \
882 curr_edges[edge_idx]->middle_d_data.coordinate_xyz, g[edge_idx], \
883 weight_vector_data_buffer + n, 1.0 / 3.0); \
884 n += curr_edges[edge_idx]->middle_d_data.gauss_nnn_patch->num_src_points; \
886#define COPY_COEFF(c_ijk, edge_idx, b_idx, bw_factor) \
888 struct weight_vector_data * curr_buffer = weight_vector_data_buffer + n; \
889 size_t curr_n = c_ijk->n; \
890 memcpy(curr_buffer, c_ijk->data, curr_n * sizeof(*curr_buffer)); \
892 double factor = - bw[edge_idx] * b_g[edge_idx][b_idx] * bw_factor; \
893 for (size_t i_ = 0; i_ < curr_n; ++i_) curr_buffer[i_].weight *= factor; \
895#define MULT_COEFF(edge_idx) \
897 double factor = 1.0 / (2.0 * bw[edge_idx] * b_g[edge_idx][edge_idx]); \
898 for (size_t i_ = 0; i_ < n; ++i_) \
899 weight_vector_data_buffer[i_].weight *= factor; \
901#define COMPACT_WEIGHTS(edge_idx) \
903 compact_weight_vector_data( \
904 weight_vector_data_buffer, &n, weight_vector_data_pos_buffer); \
905 curr_triangle->c_111_a[edge_idx].data = \
906 xmalloc(n * sizeof(*(curr_triangle->c_111_a[edge_idx].data))); \
907 memcpy(curr_triangle->c_111_a[edge_idx].data, weight_vector_data_buffer, \
908 n * sizeof(*weight_vector_data_buffer)); \
909 curr_triangle->c_111_a[edge_idx].n = n; \
966#undef COMPACT_WEIGHTS
971 free(weight_vector_data_buffer);
972 free(weight_vector_data_pos_buffer);
982 "ERROR(get_max_num_weights): invalid type_flag")
995 triangle_data->
edges[0]->
c[0].
n + triangle_data->
edges[0]->
c[1].
n +
996 triangle_data->
edges[1]->
c[0].
n + triangle_data->
edges[1]->
c[1].
n +
997 triangle_data->
edges[2]->
c[0].
n + triangle_data->
edges[2]->
c[1].
n +
1019 double * sb_coords = tgt_point_data->
sb_coords;
1025 double B_300 = sb_coords[0] * sb_coords[0] * sb_coords[0];
1026 double B_030 = sb_coords[1] * sb_coords[1] * sb_coords[1];
1027 double B_210 = 3.0 * sb_coords[0] * sb_coords[0] * sb_coords[1];
1028 double B_120 = 3.0 * sb_coords[0] * sb_coords[1] * sb_coords[1];
1037 size_t * src_points = tgt_point_data->
src_points;
1041 weights[0].
weight = B_300;
1042 weights[0].
point_id = src_points[0];
1045 weights[1].
weight = B_030;
1046 weights[1].
point_id = src_points[1];
1049 memcpy(curr_weights, c[0].
data, c[0].n *
sizeof(*curr_weights));
1050 for (
size_t i = 0; i < c[0].
n; ++i) curr_weights[i].
weight *= B_210;
1052 curr_weights += c[0].
n;
1053 memcpy(curr_weights, c[1].
data, c[1].n *
sizeof(*curr_weights));
1054 for (
size_t i = 0; i < c[1].
n; ++i) curr_weights[i].
weight *= B_120;
1056 *n = 2 + c[0].
n + c[1].
n;
1060 double * sb_coords,
double * A) {
1062 double b[3] = {sb_coords[1]*sb_coords[1] * sb_coords[2]*sb_coords[2],
1063 sb_coords[0]*sb_coords[0] * sb_coords[2]*sb_coords[2],
1064 sb_coords[0]*sb_coords[0] * sb_coords[1]*sb_coords[1]};
1065 double temp = 1.0 / (b[0] + b[1] + b[2]);
1067 for (
size_t i = 0; i < 3; ++i) {
1069 if (fabs(b[i]) < 1e-9) A[i] = 0.0;
1070 else A[i] = b[i] * temp;
1079 double * sb_coords = tgt_point_data->
sb_coords;
1092 double B_vertex[3] =
1093 {sb_coords[0] * sb_coords[0] * sb_coords[0],
1094 sb_coords[1] * sb_coords[1] * sb_coords[1],
1095 sb_coords[2] * sb_coords[2] * sb_coords[2]};
1096 double B_edge[3][2] =
1097 {{3.0 * sb_coords[1] * sb_coords[1] * sb_coords[2],
1098 3.0 * sb_coords[1] * sb_coords[2] * sb_coords[2]},
1099 {3.0 * sb_coords[0] * sb_coords[0] * sb_coords[2],
1100 3.0 * sb_coords[0] * sb_coords[2] * sb_coords[2]},
1101 {3.0 * sb_coords[0] * sb_coords[0] * sb_coords[1],
1102 3.0 * sb_coords[0] * sb_coords[1] * sb_coords[1]}};
1103 double B_111 = 6.0 * sb_coords[0] * sb_coords[1] * sb_coords[2];
1108 for (
size_t i = 0; i < 3; ++i) {
1109 weights[i].
weight = B_vertex[i];
1117 {&(curr_edges[0]->
c[0]),
1118 &(curr_edges[0]->c[1])},
1119 {&(curr_edges[1]->
c[0]),
1120 &(curr_edges[1]->c[1])},
1121 {&(curr_edges[2]->
c[0]),
1122 &(curr_edges[2]->c[1])}
1124 for (
size_t edge_idx = 0; edge_idx < 3; ++edge_idx) {
1125 for (
size_t i = 0; i < 2; ++i) {
1127 size_t curr_n = c_ijk[edge_idx][i]->
n;
1129 memcpy(curr_weights, c_ijk[edge_idx][i]->
data,
1130 c_ijk[edge_idx][i]->n *
sizeof(*curr_weights));
1131 double B = B_edge[edge_idx][i];
1132 for (
size_t j = 0; j < curr_n; ++j)
1133 curr_weights[j].
weight *= B;
1145 for (
size_t i = 0; i < 3; ++i) {
1147 size_t curr_n = c_111_a[i].
n;
1149 memcpy(curr_weights, c_111_a[i].
data, curr_n *
sizeof(*curr_weights));
1150 double B_111_x_A_i = B_111 * A[i];
1151 for (
size_t j = 0; j < curr_n; ++j)
1152 curr_weights[j].
weight *= B_111_x_A_i;
1167 "ERROR(compute_hcsbb_weights): invalid type_flag")
1191 size_t * total_num_weights) {
1205 size_t weight_vector_data_array_size = 0;
1206 size_t total_num_weights_ = 0;
1207 size_t * num_weights_per_tgt_ =
1208 xmalloc(num_tgt_points *
sizeof(*num_weights_per_tgt_));
1214 sizeof(*compact_buffer));
1220 for (
size_t i = 0; i < (num_tgt_points + 127) / 128; ++i) {
1222 size_t curr_num_tgt_points =
MIN(128, num_tgt_points - i * 128);
1224 tgt_point_data + i * 128;
1225 size_t * curr_num_weights_per_tgt = num_weights_per_tgt_ + i * 128;
1227 size_t curr_max_num_weights = 0;
1228 for (
size_t j = 0; j < curr_num_tgt_points; ++j)
1232 total_num_weights_ + curr_max_num_weights);
1234 for (
size_t j = 0; j < curr_num_tgt_points; ++j) {
1237 curr_num_weights_per_tgt + j, compact_buffer);
1238 total_num_weights_ += curr_num_weights_per_tgt[j];
1242 free(compact_buffer);
1247 *num_weights_per_tgt = num_weights_per_tgt_;
1248 *total_num_weights = total_num_weights_;
1252 const void * a,
const void * b) {
1261 for (
int i = 0; i <= (int)a_->
type_flag; ++i)
1269 double bnd_triangle[][3]) {
1274 memcpy(point_coords[i], coords[
points[i]], 3 *
sizeof(
double));
1277 &(point_coords[0][0]),
HCSBB_D_NNN, bnd_triangle, 16);
1282 double const * tri_a = a, * tri_b = b;
1285 for (
size_t i = 0; (!ret) && (i < 9); ++i) {
1286 double a_ = tri_a[i], b_ = tri_b[i];
1287 if (fabs(a_ - b_) > 1e-9) ret = (a_ > b_) - (a_ < b_);
1295 double bnd_triangle[3][3]) {
1297#if HCSBB_GAUSS_ORDER == 3
1299 {1.0-0.333333333333333-0.333333333333333,0.333333333333333,0.333333333333333},
1300 {1.0-0.200000000000000-0.200000000000000,0.200000000000000,0.200000000000000},
1301 {1.0-0.600000000000000-0.200000000000000,0.600000000000000,0.200000000000000},
1302 {1.0-0.200000000000000-0.600000000000000,0.200000000000000,0.600000000000000}
1304#elif HCSBB_GAUSS_ORDER == 4
1306 {1.0-0.091576213509771-0.091576213509771,0.091576213509771,0.091576213509771},
1307 {1.0-0.816847572980459-0.091576213509771,0.816847572980459,0.091576213509771},
1308 {1.0-0.091576213509771-0.816847572980459,0.091576213509771,0.816847572980459},
1309 {1.0-0.445948490915965-0.445948490915965,0.445948490915965,0.445948490915965},
1310 {1.0-0.108103018168070-0.445948490915965,0.108103018168070,0.445948490915965},
1311 {1.0-0.445948490915965-0.108103018168070,0.445948490915965,0.108103018168070}
1313#elif HCSBB_GAUSS_ORDER == 5
1315 {1.0-0.333333333333333-0.333333333333333,0.333333333333333,0.333333333333333},
1316 {1.0-0.101286507323456-0.101286507323456,0.101286507323456,0.101286507323456},
1317 {1.0-0.797426985353087-0.101286507323456,0.797426985353087,0.101286507323456},
1318 {1.0-0.101286507323456-0.797426985353087,0.101286507323456,0.797426985353087},
1319 {1.0-0.470142064105115-0.470142064105115,0.470142064105115,0.470142064105115},
1320 {1.0-0.059715871789770-0.470142064105115,0.059715871789770,0.470142064105115},
1321 {1.0-0.470142064105115-0.059715871789770,0.470142064105115,0.059715871789770}
1323#elif HCSBB_GAUSS_ORDER == 6
1325 {1.0-0.063089014491502-0.063089014491502,0.063089014491502,0.063089014491502},
1326 {1.0-0.873821971016996-0.063089014491502,0.873821971016996,0.063089014491502},
1327 {1.0-0.063089014491502-0.873821971016996,0.063089014491502,0.873821971016996},
1328 {1.0-0.249286745170910-0.249286745170910,0.249286745170910,0.249286745170910},
1329 {1.0-0.501426509658179-0.249286745170910,0.501426509658179,0.249286745170910},
1330 {1.0-0.249286745170910-0.501426509658179,0.249286745170910,0.501426509658179},
1331 {1.0-0.310352451033785-0.053145049844816,0.310352451033785,0.053145049844816},
1332 {1.0-0.053145049844816-0.310352451033785,0.053145049844816,0.310352451033785},
1333 {1.0-0.636502499121399-0.053145049844816,0.636502499121399,0.053145049844816},
1334 {1.0-0.053145049844816-0.636502499121399,0.053145049844816,0.636502499121399},
1335 {1.0-0.636502499121399-0.310352451033785,0.636502499121399,0.310352451033785},
1336 {1.0-0.310352451033785-0.636502499121399,0.310352451033785,0.636502499121399}
1338#elif HCSBB_GAUSS_ORDER == 7
1340 {1.0-0.333333333333333-0.333333333333333,0.333333333333333,0.333333333333333},
1341 {1.0-0.260345966079038-0.260345966079038,0.260345966079038,0.260345966079038},
1342 {1.0-0.479308067841923-0.260345966079038,0.479308067841923,0.260345966079038},
1343 {1.0-0.260345966079038-0.479308067841923,0.260345966079038,0.479308067841923},
1344 {1.0-0.065130102902216-0.065130102902216,0.065130102902216,0.065130102902216},
1345 {1.0-0.869739794195568-0.065130102902216,0.869739794195568,0.065130102902216},
1346 {1.0-0.065130102902216-0.869739794195568,0.065130102902216,0.869739794195568},
1347 {1.0-0.312865496004874-0.048690315425316,0.312865496004874,0.048690315425316},
1348 {1.0-0.048690315425316-0.312865496004874,0.048690315425316,0.312865496004874},
1349 {1.0-0.638444188569809-0.048690315425316,0.638444188569809,0.048690315425316},
1350 {1.0-0.048690315425316-0.638444188569809,0.048690315425316,0.638444188569809},
1351 {1.0-0.638444188569809-0.312865496004874,0.638444188569809,0.312865496004874},
1352 {1.0-0.312865496004874-0.638444188569809,0.312865496004874,0.638444188569809}
1354#elif HCSBB_GAUSS_ORDER == 8
1356 {1.0-0.333333333333333-0.333333333333333,0.333333333333333,0.333333333333333},
1357 {1.0-0.081414823414554-0.459292588292723,0.081414823414554,0.459292588292723},
1358 {1.0-0.459292588292723-0.081414823414554,0.459292588292723,0.081414823414554},
1359 {1.0-0.459292588292723-0.459292588292723,0.459292588292723,0.459292588292723},
1360 {1.0-0.658861384496480-0.170569307751760,0.658861384496480,0.170569307751760},
1361 {1.0-0.170569307751760-0.658861384496480,0.170569307751760,0.658861384496480},
1362 {1.0-0.170569307751760-0.170569307751760,0.170569307751760,0.170569307751760},
1363 {1.0-0.898905543365938-0.050547228317031,0.898905543365938,0.050547228317031},
1364 {1.0-0.050547228317031-0.898905543365938,0.050547228317031,0.898905543365938},
1365 {1.0-0.050547228317031-0.050547228317031,0.050547228317031,0.050547228317031},
1366 {1.0-0.008394777409958-0.263112829634638,0.008394777409958,0.263112829634638},
1367 {1.0-0.263112829634638-0.008394777409958,0.263112829634638,0.008394777409958},
1368 {1.0-0.008394777409958-0.728492392955404,0.008394777409958,0.728492392955404},
1369 {1.0-0.728492392955404-0.008394777409958,0.728492392955404,0.008394777409958},
1370 {1.0-0.263112829634638-0.728492392955404,0.263112829634638,0.728492392955404},
1371 {1.0-0.728492392955404-0.263112829634638,0.728492392955404,0.263112829634638}
1374 static double const abscissa[1][3];
1375#error "interpolation_method_hcsbb: the specified gauss order is currently not supported."
1380 gauss_vertices[i][0] = bnd_triangle[0][0] * abscissa[i][0] +
1381 bnd_triangle[1][0] * abscissa[i][1] +
1382 bnd_triangle[2][0] * abscissa[i][2];
1383 gauss_vertices[i][1] = bnd_triangle[0][1] * abscissa[i][0] +
1384 bnd_triangle[1][1] * abscissa[i][1] +
1385 bnd_triangle[2][1] * abscissa[i][2];
1386 gauss_vertices[i][2] = bnd_triangle[0][2] * abscissa[i][0] +
1387 bnd_triangle[1][2] * abscissa[i][1] +
1388 bnd_triangle[2][2] * abscissa[i][2];
1389 double scale = 1.0 / sqrt(gauss_vertices[i][0] * gauss_vertices[i][0] +
1390 gauss_vertices[i][1] * gauss_vertices[i][1] +
1391 gauss_vertices[i][2] * gauss_vertices[i][2]);
1392 gauss_vertices[i][0] *= scale;
1393 gauss_vertices[i][1] *= scale;
1394 gauss_vertices[i][2] *= scale;
1395 gauss_sb_coords[i][0] = abscissa[i][0] * scale;
1396 gauss_sb_coords[i][1] = abscissa[i][1] * scale;
1397 gauss_sb_coords[i][2] = abscissa[i][2] * scale;
1408#define POW_0(x) (1.0)
1409#define POW_1(x) ((x))
1410#define POW_2(x) ((x)*(x))
1411#define POW_3(x) ((x)*(x)*(x))
1418 sbb_polynomials[0][vertex_idx] =
1423 sbb_polynomials[1][vertex_idx] =
1428 sbb_polynomials[2][vertex_idx] =
1433 sbb_polynomials[3][vertex_idx] =
1438 sbb_polynomials[4][vertex_idx] =
1443 sbb_polynomials[5][vertex_idx] =
1448 sbb_polynomials[6][vertex_idx] =
1453 sbb_polynomials[7][vertex_idx] =
1458 sbb_polynomials[8][vertex_idx] =
1463 sbb_polynomials[9][vertex_idx] =
1481#ifdef YAC_LAPACK_NO_DSYTR
1492 !LAPACKE_dgetri_work(
1496 "internal ERROR: dgetri")
1502 !LAPACKE_dsytrf_work(
1508 !LAPACKE_dsytri_work(
1511 "internal ERROR: dsytri")
1521 double (*gauss_vertices)[3],
1527 3 *
sizeof(*bnd_triangle));
1550 accu += gauss_sbb_polynomials[i][k] * gauss_sbb_polynomials[j][k];
1564 accu += C[i][k] * gauss_sbb_polynomials[k][j];
1573 struct gauss_nnn_patch * gauss_nnn_patches,
size_t num_gauss_nnn_patches) {
1575 for (
size_t i = 0; i < num_gauss_nnn_patches; ++i) {
1578 free(gauss_nnn_patches[i].weights);
1581 free(gauss_nnn_patches);
1591 double (**unique_bnd_triangles_)[3][3],
size_t ** coord_to_triangle) {
1595 size_t * nnn_search_results =
1598 interp_grid, coords, count,
HCSBB_D_NNN, nnn_search_results, M_PI);
1605 for (
size_t i = 0, j = count - 1; i < count; ++i, --j) {
1607 size_t * curr_nnn_search_results =
1611 nnn_search_result_global_ids[k] =
1612 src_global_ids[curr_nnn_search_results[k]];
1615 nnn_search_result_global_ids,
HCSBB_D_NNN, curr_nnn_search_results);
1617 memmove(nnn_search_results + j * (
HCSBB_D_NNN + 1),
1618 curr_nnn_search_results,
1625 qsort(nnn_search_results, count,
1629 size_t * coord_to_search_results =
1630 xmalloc(count *
sizeof(*coord_to_search_results));
1631 size_t num_unique_search_results = 0;
1634 dummy_search_results[i] = SIZE_MAX;
1635 for (
size_t i = 0, * prev_search_result = dummy_search_results;
1638 size_t * curr_search_result = nnn_search_results + i * (
HCSBB_D_NNN + 1);
1639 size_t coord_idx = curr_search_result[
HCSBB_D_NNN];
1641 prev_search_result = curr_search_result;
1642 ++num_unique_search_results;
1644 coord_to_search_results[coord_idx] = num_unique_search_results - 1;
1655 xmalloc(num_unique_search_results *
sizeof(*bnd_triangles_reorder));
1656 for (
size_t i = 0, j = 0, * prev_search_result = dummy_search_results;
1659 size_t * curr_search_result = nnn_search_results + i * (
HCSBB_D_NNN + 1);
1662 curr_search_result, src_point_coords, bnd_triangles_reorder[j].
coords);
1664 prev_search_result = curr_search_result;
1668 free(nnn_search_results);
1671 qsort(bnd_triangles_reorder, num_unique_search_results,
1675 size_t * search_result_to_bnd_triangle =
1676 xmalloc(num_unique_search_results *
sizeof(*search_result_to_bnd_triangle));
1677 double (*unique_bnd_triangles)[3][3] = (double(*)[3][3])bnd_triangles_reorder;
1678 size_t num_unique_triangles = 0;
1679 double dummy_bnd_triangle[3][3] = {{-1,-1,-1},{-1,-1,-1},{-1,-1,-1}};
1680 double (*prev_bnd_triangle)[3] = dummy_bnd_triangle;
1681 for (
size_t i = 0; i < num_unique_search_results; ++i) {
1682 double (*curr_bnd_triangle)[3] = bnd_triangles_reorder[i].
coords;
1685 memmove(unique_bnd_triangles[num_unique_triangles], curr_bnd_triangle,
1686 sizeof(*unique_bnd_triangles));
1687 ++num_unique_triangles;
1688 prev_bnd_triangle = curr_bnd_triangle;
1690 search_result_to_bnd_triangle[
reorder_idx] = num_unique_triangles - 1;
1694 for (
size_t i = 0; i < count; ++i)
1695 coord_to_search_results[i] =
1696 search_result_to_bnd_triangle[coord_to_search_results[i]];
1697 free(search_result_to_bnd_triangle);
1699 *num_unique_triangles_ = num_unique_triangles;
1700 *unique_bnd_triangles_ =
1702 num_unique_triangles *
sizeof(*unique_bnd_triangles));
1703 *coord_to_triangle =
1704 xrealloc(coord_to_search_results, count *
sizeof(*coord_to_search_results));
1717 struct yac_interp_grid * interp_grid,
size_t * num_gauss_nnn_patches_,
1725 for (
size_t i = 0; i < num_vertices; ++i)
1726 memcpy(
coords[i], vertex_data[i].coordinate_xyz,
sizeof(*
coords));
1727 for (
size_t i = 0; i < num_edges; ++i)
1732 size_t num_gauss_nnn_patches;
1733 double (*bnd_triangles)[3][3];
1734 size_t * coord_to_gauss_nnn_patch;
1736 num_vertices + num_edges,
coords, interp_grid,
1737 &num_gauss_nnn_patches, &bnd_triangles, &coord_to_gauss_nnn_patch);
1741 xmalloc(num_gauss_nnn_patches *
sizeof(*gauss_nnn_patches));
1743 xmalloc(num_gauss_nnn_patches *
sizeof(*gauss_points));
1749 *num_gauss_nnn_patches_ = num_gauss_nnn_patches;
1750 *gauss_nnn_patches_ = gauss_nnn_patches;
1753 for (
size_t i = 0; i < num_gauss_nnn_patches; ++i)
1755 gauss_nnn_patches + i, bnd_triangles[i],
1756 gauss_points[i], gauss_sbb_polynomials_buffer, work_buffer);
1758 free(gauss_sbb_polynomials_buffer);
1759 free(bnd_triangles);
1761 for (
size_t i = 0; i < num_vertices; ++i)
1763 gauss_nnn_patches + coord_to_gauss_nnn_patch[i];
1764 for (
size_t i = 0; i < num_edges; ++i)
1766 gauss_nnn_patches + coord_to_gauss_nnn_patch[i + num_vertices];
1767 free(coord_to_gauss_nnn_patch);
1770 size_t * nnn_search_results =
1772 sizeof(*nnn_search_results));
1782 size_t * src_point_buffer =
1793 for (
size_t i = 0; i < num_gauss_nnn_patches; ++i)
1795 gauss_nnn_patches + i, gauss_points[i],
1797 w_nnn_buffer, src_point_buffer, global_id_buffer, src_point_coords,
1798 src_point_global_ids);
1800 free(nnn_search_results);
1801 free(global_id_buffer);
1802 free(src_point_buffer);
1808 double * check_coord,
int num_vertices,
size_t const * vertices,
1811 size_t * triangle,
double * sb_coords) {
1814 size_t num_triangles = (size_t)(num_vertices - 2);
1815 size_t triangle_indices[num_triangles][3];
1817 vertices, num_vertices, vertex_start_idx, &(triangle_indices[0]));
1820 for (
size_t i = 0; (!ret) && (i < num_triangles); ++i) {
1822 triangle[0] = triangle_indices[i][0];
1823 triangle[1] = triangle_indices[i][1];
1824 triangle[2] = triangle_indices[i][2];
1826 {global_ids[triangle[0]],
1827 global_ids[triangle[1]],
1828 global_ids[triangle[2]]};
1833 double triangle_coords[3][3];
1834 for (
int j = 0; j < 3; ++j)
1836 triangle_coords[j], point_coords[triangle[j]], 3 *
sizeof(
double));
1845 sb_coords[0] = check_coord[0];
1846 sb_coords[1] = check_coord[1];
1847 sb_coords[2] = check_coord[2];
1861 double * tgt_coord,
size_t src_cell,
1864 size_t * src_triangle_vertices,
double * sb_coords) {
1866 size_t const * src_vertices =
1874 size_t lowest_vertex_id_idx = 0;
1876 yac_int lowest_vertex_id = vertex_ids[src_vertices[0]];
1877 for (
int i = 1; i < num_src_vertices; ++i) {
1878 if (vertex_ids[src_vertices[i]] < lowest_vertex_id) {
1879 lowest_vertex_id = vertex_ids[src_vertices[i]];
1880 lowest_vertex_id_idx = i;
1887 tgt_coord, num_src_vertices, src_vertices, lowest_vertex_id_idx,
1888 vertex_ids, src_point_coords, src_triangle_vertices, sb_coords);
1892 double * tgt_coord,
size_t src_cell,
1895 size_t * vertex_to_cell,
size_t * vertex_to_cell_offsets,
1896 int * num_cells_per_vertex,
size_t * src_triangle,
double * sb_coords) {
1899 size_t const * curr_vertices =
1907 for (
size_t i = 0; (!ret) && (i < num_vertices); ++i) {
1909 size_t curr_vertex = curr_vertices[i];
1911 int curr_num_src_cells = num_cells_per_vertex[curr_vertex];
1912 if (curr_num_src_cells == 0)
continue;
1914 size_t * curr_src_cells =
1915 vertex_to_cell + vertex_to_cell_offsets[curr_vertex];
1921 tgt_coord, curr_num_src_cells, curr_src_cells, 0, cell_ids,
1922 src_point_coords, src_triangle, sb_coords);
1929 size_t num_tgt_points,
size_t * selected_tgt,
size_t * tgt_points,
1937 xmalloc(num_tgt_points *
sizeof(*tgt_point_data));
1941 size_t * vertex_to_cell = NULL;
1942 size_t * vertex_to_cell_offsets = NULL;
1943 int * num_cells_per_vertex = NULL;
1946 interp_grid, src_cells, num_tgt_points,
1947 &vertex_to_cell, &vertex_to_cell_offsets, &num_cells_per_vertex);
1959 for (
size_t i = 0; i < num_tgt_points; ++i, ++tgt_point_data) {
1961 size_t tgt_idx = selected_tgt[i];
1963 double * tgt_coord = tgt_coords[tgt_idx];
1964 tgt_point_data->
tgt_point = tgt_points[tgt_idx];
1966 tgt_point_data->
coordinate_xyz, tgt_coord, 3 *
sizeof(*tgt_coord));
1971 size_t src_triangle_vertices[3];
1972 double sb_coords[3];
1976 "ERROR(init_tgt_point_search_data): invalid source location")
1980 tgt_coord, src_cells[i], src_grid_data, src_point_coords,
1981 src_triangle_vertices, sb_coords);
1985 tgt_coord, src_cells[i], src_grid_data, src_point_coords,
1986 vertex_to_cell, vertex_to_cell_offsets, num_cells_per_vertex,
1987 src_triangle_vertices, sb_coords);
1992 int tmp_type_flag = -1;
1994 for (
size_t k = 0; k < 3; ++k) {
1997 tgt_point_data->
src_points[tmp_type_flag] = src_triangle_vertices[k];
1998 tgt_point_data->
sb_coords[tmp_type_flag] = sb_coords[k];
2000 if (src_mask != NULL) is_masked |= !src_mask[src_triangle_vertices[k]];
2009 free(num_cells_per_vertex);
2010 free(vertex_to_cell_offsets);
2011 free(vertex_to_cell);
2013 return tgt_point_data - num_tgt_points;
2017 size_t num_vertices,
size_t * vertices,
2024 xmalloc(num_vertices *
sizeof(*vertex_data));
2026 for (
size_t i = 0; i < num_vertices; ++i) {
2027 size_t vertex_idx = vertices[i];
2029 src_point_coords[vertex_idx],
sizeof(*src_point_coords));
2037 size_t num_edges,
size_t (*edge_to_vertex_d_data)[2],
2041 xmalloc(num_edges *
sizeof(*edge_data));
2043 for (
size_t i = 0; i < num_edges; ++i) {
2046 edge_data[i].
vertex_d_data[0] = vertex_data + edge_to_vertex_d_data[i][0];
2047 edge_data[i].
vertex_d_data[1] = vertex_data + edge_to_vertex_d_data[i][1];
2049 double * vertex_coordinates_xyz[2] = {
2055 middle_point[0] = vertex_coordinates_xyz[0][0] +
2056 vertex_coordinates_xyz[1][0];
2057 middle_point[1] = vertex_coordinates_xyz[0][1] +
2058 vertex_coordinates_xyz[1][1];
2059 middle_point[2] = vertex_coordinates_xyz[0][2] +
2060 vertex_coordinates_xyz[1][2];
2062 1.0 / sqrt(middle_point[0] * middle_point[0] +
2063 middle_point[1] * middle_point[1] +
2064 middle_point[2] * middle_point[2]);
2065 middle_point[0] *= sb_coord;
2066 middle_point[1] *= sb_coord;
2067 middle_point[2] *= sb_coord;
2072 crossproduct_d(vertex_coordinates_xyz[0], vertex_coordinates_xyz[1],
2073 &(edge_data[i].
g[0]));
2080 size_t num_triangles,
size_t (*triangle_to_edge_data)[3],
2084 xmalloc(num_triangles *
sizeof(*triangle_data));
2086 for (
size_t i = 0; i < num_triangles; ++i) {
2089 triangle_data[i].
edges[0] = edge_data + triangle_to_edge_data[i][0];
2090 triangle_data[i].
edges[1] = edge_data + triangle_to_edge_data[i][1];
2091 triangle_data[i].
edges[2] = edge_data + triangle_to_edge_data[i][2];
2094 return triangle_data;
2099 size_t * tgt_points,
size_t count,
2101 int * interpolation_complete) {
2103 if (*interpolation_complete)
return 0;
2109 "ERROR(do_search_hcsbb): invalid number of source fields")
2116 "ERROR(do_search_hcsbb): unsupported source field location type")
2121 interp_grid, tgt_points, count, tgt_coords);
2123 size_t * size_t_buffer =
xmalloc(2 * count *
sizeof(*size_t_buffer));
2124 size_t * src_cells = size_t_buffer;
2125 size_t * reorder_idx = size_t_buffer + count;
2130 interp_grid, tgt_coords, count, src_cells);
2134 for (
size_t i = 0; i < count; ++i) reorder_idx[i] = i;
2138 size_t temp_result_count = 0;
2139 for (temp_result_count = 0; temp_result_count < count; ++temp_result_count)
2140 if (src_cells[temp_result_count] == SIZE_MAX)
break;
2146 temp_result_count, reorder_idx, tgt_points, tgt_coords, src_cells,
2152 qsort(tgt_point_data, temp_result_count,
sizeof(*tgt_point_data),
2156 for (
size_t i = temp_result_count; i < count; ++i)
2157 reorder_idx[i] = tgt_points[reorder_idx[i]];
2158 memcpy(tgt_points + temp_result_count, reorder_idx + temp_result_count,
2159 (count - temp_result_count) *
sizeof(*tgt_points));
2162 for (
size_t i = 0; i < temp_result_count; ++i)
2163 tgt_points[i] = tgt_point_data[i].
tgt_point;
2164 free(size_t_buffer);
2167 size_t result_count = 0;
2168 size_t result_counts[3] = {0,0,0};
2169 for (result_count = 0; result_count < temp_result_count; ++result_count) {
2170 if (tgt_point_data[result_count].
is_masked)
break;
2171 else result_counts[(int)(tgt_point_data[result_count].
type_flag)]++;
2177 size_t * unique_vertices;
2178 size_t (*unique_edge_to_unique_vertex)[2];
2179 size_t (*unique_triangle_to_unique_edge)[3];
2180 size_t num_unique_vertices, num_unique_edges, num_unique_triangles;
2182 tgt_point_data + result_counts[0], result_counts[1], result_counts[2],
2183 &unique_vertices, &num_unique_vertices,
2184 &unique_edge_to_unique_vertex, &num_unique_edges,
2185 &unique_triangle_to_unique_edge, &num_unique_triangles);
2190 num_unique_vertices, unique_vertices, interp_grid);
2193 num_unique_edges, unique_edge_to_unique_vertex, vertex_data);
2196 num_unique_triangles, unique_triangle_to_unique_edge, edge_data);
2197 free(unique_vertices);
2198 free(unique_edge_to_unique_vertex);
2199 free(unique_triangle_to_unique_edge);
2202 for (
size_t i = result_counts[0];
2203 i < result_counts[0] + result_counts[1]; ++i)
2204 tgt_point_data[i].interp_data.
edge =
2206 for (
size_t i = result_counts[0] + result_counts[1];
2207 i < result_counts[0] + result_counts[1] + result_counts[2]; ++i)
2208 tgt_point_data[i].interp_data.
triangle =
2213 size_t num_gauss_nnn_patches;
2216 num_unique_vertices, vertex_data, num_unique_edges, edge_data, interp_grid,
2217 &num_gauss_nnn_patches, &gauss_nnn_patches);
2221 size_t * num_weights_per_tgt, total_num_weights;
2223 edge_data, num_unique_edges,
2224 triangle_data, num_unique_triangles,
2227 if (num_unique_edges > 0) free(edge_data->
c[0].
data);
2229 for (
size_t i = 0; i < num_unique_triangles; ++i)
2230 for (
size_t j = 0; j < 3; ++j) free(triangle_data[i].c_111_a[j].
data);
2231 free(triangle_data);
2232 free(tgt_point_data);
2237 size_t * src_points =
xmalloc(total_num_weights *
sizeof(*src_points));
2238 double * weight_data =
xmalloc(total_num_weights *
sizeof(*weight_data));
2239 for (
size_t i = 0; i < total_num_weights; ++i) {
2247 interp_grid, 0, src_points, total_num_weights);
2252 interp_grid, tgt_points, result_count),
2253 .count = result_count};
2257 weights, &tgts, num_weights_per_tgt, srcs, weight_data);
2263 free(num_weights_per_tgt);
2265 return result_count;
#define YAC_ASSERT(exp, msg)
void yac_compute_bnd_triangle(double *vertices, size_t num_vertices, double triangle[][3], size_t num_tests)
int const * const_int_pointer
yac_int const * const_yac_int_pointer
#define ENSURE_ARRAY_SIZE(arrayp, curr_array_size, req_size)
void yac_triangulate_cell_indices(size_t const *corner_indices, size_t num_corners, size_t start_corner, size_t(*triangle_indices)[3])
static int points_are_identically(double const *a, double const *b)
static void crossproduct_d(const double a[], const double b[], double cross[])
static double get_vector_angle(double const a[3], double const b[3])
size_t yac_interp_grid_get_num_src_fields(struct yac_interp_grid *interp_grid)
const_int_pointer yac_interp_grid_get_src_field_mask(struct yac_interp_grid *interp_grid, size_t src_field_idx)
void yac_interp_grid_get_aux_grid_src(struct yac_interp_grid *interp_grid, size_t *cells, size_t count, size_t **vertex_to_cell, size_t **vertex_to_cell_offsets, int **num_cells_per_vertex)
const_yac_int_pointer yac_interp_grid_get_src_field_global_ids(struct yac_interp_grid *interp_grid, size_t src_field_idx)
struct remote_point * yac_interp_grid_get_tgt_remote_points(struct yac_interp_grid *interp_grid, size_t *tgt_points, size_t count)
enum yac_location yac_interp_grid_get_src_field_location(struct yac_interp_grid *interp_grid, size_t src_field_idx)
struct remote_point * yac_interp_grid_get_src_remote_points(struct yac_interp_grid *interp_grid, size_t src_field_idx, size_t *src_points, size_t count)
void yac_interp_grid_do_nnn_search_src(struct yac_interp_grid *interp_grid, yac_coordinate_pointer search_coords, size_t count, size_t n, size_t *src_points, double max_search_distance)
yac_const_coordinate_pointer yac_interp_grid_get_src_field_coords(struct yac_interp_grid *interp_grid, size_t src_field_idx)
void yac_interp_grid_get_tgt_coordinates(struct yac_interp_grid *interp_grid, size_t *tgt_points, size_t count, yac_coordinate_pointer tgt_coordinates)
void yac_interp_grid_do_points_search_gc(struct yac_interp_grid *interp_grid, yac_coordinate_pointer search_coords, size_t count, size_t *src_cells)
struct yac_const_basic_grid_data * yac_interp_grid_get_basic_grid_data_src(struct yac_interp_grid *interp_grid)
static struct triangle_interp_data * init_triangle_interp_data(size_t num_triangles, size_t(*triangle_to_edge_data)[3], struct edge_interp_data *edge_data)
static struct edge_interp_data * init_edge_interp_data(size_t num_edges, size_t(*edge_to_vertex_d_data)[2], struct vertex_interp_data *vertex_data)
static int compare_tgt_point_search_data(const void *a, const void *b)
static void compute_weights(struct tgt_point_search_data *tgt_point_data, size_t num_tgt_points, struct edge_interp_data *edge_data, size_t num_edges, struct triangle_interp_data *triangle_data, size_t num_triangles, struct weight_vector_data **weights, size_t **num_weights_per_tgt, size_t *total_num_weights)
static void compute_gauss_nnn_patch(struct gauss_nnn_patch *gauss_nnn_patch, double(*gauss_points)[3], size_t *nnn_search_results, double w_nnn[][HCSBB_NUM_GAUSS_POINTS], size_t *src_point_buffer, yac_int *global_id_buffer, yac_const_coordinate_pointer src_point_coords, const_yac_int_pointer src_point_global_ids)
static struct vertex_interp_data * init_vertex_interp_data(size_t num_vertices, size_t *vertices, struct yac_interp_grid *interp_grid)
static int compare_HCSBB_D_NNN_size_t(const void *a, const void *b)
static int compare_size_t(const void *a, const void *b)
static void generate_d_data_triangle(size_t count, yac_coordinate_pointer coords, struct yac_interp_grid *interp_grid, size_t *num_unique_triangles_, double(**unique_bnd_triangles_)[3][3], size_t **coord_to_triangle)
static struct interp_method_vtable interp_method_hcsbb_vtable
#define HCSBB_NUM_SBB_COEFFS
#define HCSBB_NUM_GAUSS_POINTS
@ ON_EDGE
the target point is on an edge of the source point grid
static void compute_triangle_coefficients(struct triangle_interp_data *triangle_data, size_t num_triangles)
#define COMPACT_WEIGHTS(edge_idx)
#define COPY_D_DATA(edge_idx)
static void compute_sb_coords(double *sb_coords, size_t num_vertices, double triangle[3][3], char const *caller)
static void generate_derivative_data(size_t num_vertices, struct vertex_interp_data *vertex_data, size_t num_edges, struct edge_interp_data *edge_data, struct yac_interp_grid *interp_grid, size_t *num_gauss_nnn_patches_, struct gauss_nnn_patch **gauss_nnn_patches_)
static void compute_hcsbb_weights_vertex(struct tgt_point_search_data *tgt_point_data, struct weight_vector_data *weights, size_t *n)
#define MULT_COEFF(edge_idx)
static void generate_bnd_triangle(size_t *points, yac_const_coordinate_pointer coords, double bnd_triangle[][3])
static size_t do_search_hcsbb(struct interp_method *method, struct yac_interp_grid *interp_grid, size_t *tgt_points, size_t count, struct yac_interp_weights *weights, int *interpolation_complete)
static void compute_hcsbb_weights_edge(struct tgt_point_search_data *tgt_point_data, struct weight_vector_data *weights, size_t *n)
static void inverse(double A[HCSBB_NUM_SBB_COEFFS][HCSBB_NUM_SBB_COEFFS])
static void compute_hcsbb_weights(struct tgt_point_search_data *tgt_point_data, struct weight_vector_data *weights, size_t *n, struct weight_vector_data_pos *compact_buffer)
static void compact_weight_vector_data(struct weight_vector_data *weights, size_t *n, struct weight_vector_data_pos *buffer)
static void get_unique_interp_data(struct tgt_point_search_data *tgt_point_data, size_t num_edges, size_t num_triangles, size_t **unique_vertices, size_t *num_unique_vertices, size_t(**unique_edge_to_unique_vertex)[2], size_t *num_unique_edges, size_t(**unique_triangle_to_unique_edge)[3], size_t *num_unique_triangles)
static void delete_hcsbb(struct interp_method *method)
static void compute_edge_coefficients(struct edge_interp_data *edge_data, size_t num_edges)
static int compare_bnd_triangles(void const *a, void const *b)
static void compute_sbb_polynomials_gauss_3d(double sb_coords[HCSBB_NUM_GAUSS_POINTS][3], double sbb_polynomials[HCSBB_NUM_SBB_COEFFS][HCSBB_NUM_GAUSS_POINTS])
#define COPY_COEFF(c_ijk, edge_idx, b_idx, bw_factor)
static int compare_2_size_t(const void *a, const void *b)
static void init_gauss_nnn_patch(struct gauss_nnn_patch *gauss_nnn_patch, double(*bnd_triangle)[3], double(*gauss_vertices)[3], double gauss_sbb_polynomials[HCSBB_NUM_SBB_COEFFS][HCSBB_NUM_GAUSS_POINTS], double C[HCSBB_NUM_SBB_COEFFS][HCSBB_NUM_SBB_COEFFS])
static int check_polygon(double *check_coord, int num_vertices, size_t const *vertices, size_t vertex_start_idx, const_yac_int_pointer global_ids, yac_const_coordinate_pointer point_coords, size_t *triangle, double *sb_coords)
static int compare_3_size_t(const void *a, const void *b)
struct interp_method * yac_interp_method_hcsbb_new()
static int compare_weight_vector_data_point(void const *a, void const *b)
static void compute_hcsbb_weights_triangle(struct tgt_point_search_data *tgt_point_data, struct weight_vector_data *weights, size_t *n)
static void generate_gauss_legendre_points(double gauss_vertices[HCSBB_NUM_GAUSS_POINTS][3], double gauss_sb_coords[HCSBB_NUM_GAUSS_POINTS][3], double bnd_triangle[3][3])
static int compare_weight_vector_data_pos_pos(void const *a, void const *b)
static size_t get_max_num_weights(struct tgt_point_search_data *tgt_point_data)
static void evaluate_blending_function(double *sb_coords, double *A)
static int compare_weight_vector_data_pos_weight(void const *a, void const *b)
static int get_matching_aux_cell_triangle(double *tgt_coord, size_t src_cell, struct yac_const_basic_grid_data *src_grid_data, yac_const_coordinate_pointer src_point_coords, size_t *vertex_to_cell, size_t *vertex_to_cell_offsets, int *num_cells_per_vertex, size_t *src_triangle, double *sb_coords)
static void free_gauss_nnn_patches(struct gauss_nnn_patch *gauss_nnn_patches, size_t num_gauss_nnn_patches)
static int get_matching_vertex_triangle(double *tgt_coord, size_t src_cell, struct yac_const_basic_grid_data *src_grid_data, yac_const_coordinate_pointer src_point_coords, size_t *src_triangle_vertices, double *sb_coords)
static struct tgt_point_search_data * init_tgt_point_data(size_t num_tgt_points, size_t *selected_tgt, size_t *tgt_points, yac_coordinate_pointer tgt_coords, size_t *src_cells, struct yac_interp_grid *interp_grid)
static void get_derivative_weights(struct gauss_nnn_patch *gauss_nnn_patch, double coordinate_xyz[3], double direction[3], struct weight_vector_data *weights, double mult_factor)
static void compute_d_sbb_polynomials_3d(double bnd_triangle[3][3], double direction[3], double coordinate_xyz[3], double d_sbb_polynomials[HCSBB_NUM_SBB_COEFFS])
void yac_interp_weights_add_wsum(struct yac_interp_weights *weights, struct remote_points *tgts, size_t *num_src_per_tgt, struct remote_point *srcs, double *w)
#define xrealloc(ptr, size)
double coordinate_xyz[3]
point at which a derivative is to be estimated
double sb_coord_middle_point
struct weight_vector c[2]
struct edge_interp_data::@26 middle_d_data
derivative data of edge middle point
struct vertex_interp_data * vertex_d_data[2]
derivative data of both vertices
struct gauss_nnn_patch * gauss_nnn_patch
gauss integration point patch used to estimate the derivative
double bnd_triangle[3][3]
bounding triangle
struct interp_method_vtable * vtable
size_t(* do_search)(struct interp_method *method, struct yac_interp_grid *grid, size_t *tgt_points, size_t count, struct yac_interp_weights *weights, int *interpolation_complete)
information (global id and location) about a point that
structure containing the information (global id and location)
struct remote_point * data
enum interp_type_flag type_flag
depending on type flag interp_data contains a point to the required data
struct triangle_interp_data * triangle
union tgt_point_search_data::@27 interp_data
struct edge_interp_data * edge
struct weight_vector c_111_a[3]
weights for alpha values used to compute c_111
struct edge_interp_data * edges[3]
edges of the triangle (edge[0] is the on opposing corner src_points[0])
size_t src_point
index of the associacted source point
double coordinate_xyz[3]
coordinate of the source point
struct gauss_nnn_patch * gauss_nnn_patch
struct weight_vector_data * weight
struct weight_vector_data * data
const_size_t_pointer cell_to_vertex_offsets
const_size_t_pointer cell_to_vertex
const const_int_pointer num_vertices_per_cell
const const_yac_int_pointer ids[3]
void yac_quicksort_index_yac_int_size_t(yac_int *a, size_t n, size_t *idx)
static void yac_remove_duplicates_size_t_3(size_t(*array)[3], size_t *n)
static void yac_remove_duplicates_size_t(size_t *array, size_t *n)
void yac_quicksort_index_size_t_size_t(size_t *a, size_t n, size_t *idx)
static void yac_remove_duplicates_size_t_2(size_t(*array)[2], size_t *n)
static struct user_input_data_points ** points
#define YAC_ASSERT_F(exp, format,...)
double const (* yac_const_coordinate_pointer)[3]
double(* yac_coordinate_pointer)[3]