17#define SWAP(x, y, T) do { T SWAP = x; x = y; y = SWAP; } while (0)
21 size_t * tgt_points,
size_t count,
23 int * interpolation_complete);
42 double * point,
size_t field_cell,
size_t cell_size,
43 size_t const * vertex_to_cell,
size_t const * vertex_to_cell_offsets,
47 if (cell_size == 0)
return 0;
58 cell_size *
sizeof(*coordinates_xyz))));
62 for (
size_t i = 0; i < cell_size; ++i)
69 size_t const * field_cell_vertices =
70 vertex_to_cell + vertex_to_cell_offsets[field_cell];
71 for (
size_t i = 0; i < cell_size; ++i)
72 memcpy(coordinates_xyz[i], field_coordinates[field_cell_vertices[i]],
73 sizeof(*coordinates_xyz));
78#define IS_GC(x) (((x) == YAC_GREAT_CIRCLE_EDGE) || ((x) == YAC_LON_CIRCLE_EDGE))
79#define IS_LON_LAT(x) (((x) == YAC_LAT_CIRCLE_EDGE) || ((x) == YAC_LON_CIRCLE_EDGE))
83 size_t const * edge_indices,
size_t num_edges) {
87 int edge_type_flag = 0;
92 {edge_types[edge_indices[0]],
93 edge_types[edge_indices[1]],
94 edge_types[edge_indices[2]],
95 edge_types[edge_indices[3]]};
98 (temp_edges[0] == temp_edges[2]) &&
99 (temp_edges[1] == temp_edges[3]) &&
100 (temp_edges[0] != temp_edges[1]))
104 for (
size_t i = 0; i < num_edges; ++i)
112 return cell_types[edge_type_flag];
119 size_t * tgt_points,
size_t count,
121 int * interpolation_complete) {
123 if (*interpolation_complete)
return 0;
130 "ERROR(do_search_avg): invalid number of source fields")
138 "ERROR(do_search_avg): unsupported source field location type")
143 interp_grid, tgt_points, count, tgt_coords);
145 size_t * size_t_buffer =
xmalloc(2 * count *
sizeof(*size_t_buffer));
146 size_t * src_field_cells = size_t_buffer;
147 size_t * reorder_idx = size_t_buffer + count;
151 interp_grid, tgt_coords, count, src_field_cells);
153 size_t const * src_field_cell_to_vertex;
154 size_t const * src_field_cell_to_vertex_offsets;
155 size_t const * src_field_cell_to_edge;
156 size_t const * src_field_cell_to_edge_offsets;
157 int const * src_field_num_vertices_per_cell;
165 interp_grid, src_field_cells, count,
166 (
size_t**)&src_field_cell_to_vertex,
167 (
size_t**)&src_field_cell_to_vertex_offsets,
168 (
int**)&src_field_num_vertices_per_cell);
179 for (
size_t i = 0; i < count; ++i) {
181 size_t curr_src_cell = src_field_cells[i];
182 if (curr_src_cell == SIZE_MAX)
continue;
184 double * curr_tgt_coord = tgt_coords[i];
185 size_t const * curr_vertices =
188 size_t curr_num_vertices =
191 size_t result_src_field_cell = SIZE_MAX;
194 for (
size_t j = 0; j < curr_num_vertices; ++j) {
196 size_t src_field_cell = curr_vertices[j];
197 size_t src_field_cell_size =
198 (size_t)src_field_num_vertices_per_cell[src_field_cell];
202 curr_tgt_coord, src_field_cell, src_field_cell_size,
203 src_field_cell_to_vertex, src_field_cell_to_vertex_offsets,
204 src_field_coordinates, &cell_buffer)) {
205 result_src_field_cell = src_field_cell;
210 src_field_cells[i] = result_src_field_cell;
214 src_edge_types = NULL;
215 src_field_cell_to_edge = NULL;
216 src_field_cell_to_edge_offsets = NULL;
235 for (
size_t i = 0; i <
count; ++i) reorder_idx[i] = i;
238 size_t result_count = 0;
239 for (result_count = 0; result_count <
count; ++result_count)
240 if (src_field_cells[result_count] == SIZE_MAX)
break;
242 size_t total_num_weights = 0;
243 size_t max_num_vertices_per_cell = 0;
244 size_t * num_weights_per_tgt =
245 xmalloc(result_count *
sizeof(*num_weights_per_tgt));
249 for (
size_t i = 0; i < result_count; ++i) {
250 size_t curr_num_vertices =
251 (size_t)(src_field_num_vertices_per_cell[src_field_cells[i]]);
252 num_weights_per_tgt[i] = curr_num_vertices;
253 total_num_weights += curr_num_vertices;
254 if (curr_num_vertices > max_num_vertices_per_cell)
255 max_num_vertices_per_cell = curr_num_vertices;
258 free((
void*)src_field_num_vertices_per_cell);
264 xmalloc(max_num_vertices_per_cell *
sizeof(*src_coord_buffer));
266 (src_field_mask != NULL)?
267 xmalloc(max_num_vertices_per_cell *
sizeof(*mask_buffer)):NULL;
268 double * w =
xmalloc(total_num_weights *
sizeof(*w));
269 size_t * src_points =
xmalloc(total_num_weights *
sizeof(*src_points));
273 total_num_weights = 0;
274 for (
size_t i = 0; i < result_count;) {
276 size_t curr_num_vertices = num_weights_per_tgt[i];
279 src_field_cell_to_vertex +
280 src_field_cell_to_vertex_offsets[src_field_cells[i]];
282 (src_edge_types != NULL)?
283 (src_field_cell_to_edge +
284 src_field_cell_to_edge_offsets[src_field_cells[i]]):NULL;
285 double * curr_weights = w + total_num_weights;
290 size_t lowest_global_id_idx = 0;
292 yac_int lowest_global_id = src_global_ids[curr_cell_to_vertex[0]];
293 for (
size_t j = 1; j < curr_num_vertices; ++j) {
294 if (src_global_ids[curr_cell_to_vertex[j]] < lowest_global_id) {
295 lowest_global_id = src_global_ids[curr_cell_to_vertex[j]];
296 lowest_global_id_idx = j;
303 for (
size_t j = 0, l = lowest_global_id_idx; j < curr_num_vertices;
304 ++j, ++total_num_weights, ++l) {
306 if (l == curr_num_vertices) l = 0;
308 size_t curr_vertex_idx = curr_cell_to_vertex[l];
309 src_points[total_num_weights] = curr_vertex_idx;
310 for (
size_t k = 0; k < 3; ++k)
311 src_coord_buffer[j][k] = src_field_coordinates[curr_vertex_idx][k];
312 if (src_field_mask != NULL)
313 mask_buffer[j] = src_field_mask[curr_vertex_idx];
318 src_edge_types, curr_cell_to_edge, curr_num_vertices);
323 mask_buffer, curr_weights, cell_type)) {
326 num_weights_per_tgt[i] = curr_num_vertices;
331 total_num_weights -= curr_num_vertices;
333 src_field_cells[i] = src_field_cells[result_count];
334 size_t temp_reorder_idx = reorder_idx[i];
335 reorder_idx[i] = reorder_idx[result_count];
336 reorder_idx[result_count] = temp_reorder_idx;
340 free((
void*)src_field_cell_to_vertex);
341 free((
void*)src_field_cell_to_vertex_offsets);
345 for (
size_t i = 0; i <
count; ++i) src_field_cells[reorder_idx[i]] = i;
350 free(src_coord_buffer);
356 interp_grid, tgt_points, result_count),
357 .count = result_count};
360 interp_grid, 0, src_points, total_num_weights);
364 weights, &tgts, num_weights_per_tgt, srcs, w);
369 free(num_weights_per_tgt);
376 double tgt_coords[3],
size_t num_vertices,
384 size_t num_unmasked_points = 0;
387 if (src_mask != NULL) {
388 for (
size_t i = 0; i < num_vertices; ++i)
389 if (src_mask[i]) num_unmasked_points++;
390 if (num_unmasked_points == 0)
return 0;
392 double weight = 1.0 / (double)num_unmasked_points;
393 for (
size_t i = 0; i < num_vertices; ++i)
394 weights[i] = (src_mask[i])?weight:0.0;
397 double weight = 1.0 / (double)num_vertices;
398 for (
size_t i = 0; i < num_vertices; ++i) weights[i] = weight;
404 double tgt_coords[3],
size_t num_vertices,
413 if (src_mask != NULL)
414 for (
size_t i = 0; i < num_vertices; ++i)
415 if (!src_mask[i])
return 0;
417 double weight = 1.0 / (double)num_vertices;
418 for (
size_t i = 0; i < num_vertices; ++i) weights[i] = weight;
424 double tgt_coords[3],
size_t num_vertices,
431 if (src_mask != NULL) {
432 int has_unmasked = 0;
433 for (
size_t i = 0; i < num_vertices; ++i) has_unmasked |= src_mask[i];
434 if (!has_unmasked)
return 0;
437 if (src_mask != NULL) {
438 for (
size_t i = 0; i < num_vertices; ++i) {
447 for (
size_t j = 0; j < num_vertices; ++j) weights[j] = 0.0;
452 weights[i] = 1.0 / distance;
458 for (
size_t i = 0; i < num_vertices; ++i) {
465 for (
size_t j = 0; j < num_vertices; ++j) weights[j] = 0.0;
470 weights[i] = 1.0 / distance;
475 double inv_distance_sum = 0.0;
476 for (
size_t i = 0; i < num_vertices; ++i)
477 inv_distance_sum += weights[i];
478 double scale = 1.0 / inv_distance_sum;
480 for (
size_t i = 0; i < num_vertices; ++i) weights[i] *= scale;
486 double tgt_coords[3],
size_t num_vertices,
492 for (
size_t i = 0; i < num_vertices; ++i) {
499 if ((src_mask != NULL) && !src_mask[i])
return 0;
500 for (
size_t j = 0; j < num_vertices; ++j) weights[j] = 0.0;
505 weights[i] = 1.0 / distance;
511 if (src_mask != NULL)
512 for (
size_t i = 0; i < num_vertices; ++i)
if(!src_mask[i])
return 0;
515 double inv_distance_sum = 0.0;
516 for (
size_t i = 0; i < num_vertices; ++i)
517 inv_distance_sum += weights[i];
518 double scale = 1.0 / inv_distance_sum;
520 for (
size_t i = 0; i < num_vertices; ++i) weights[i] *= scale;
528 double * barycentric_coords,
size_t triangle_indices[3],
530 char const * caller) {
533 lapack_int n = 3, nrhs = 1, lda = n, ldx = n, ipiv[3];
534 for (
int i = 0; i < 3; ++i)
535 memcpy(A[i], grid_coords[triangle_indices[i]],
sizeof(*grid_coords));
545 LAPACK_COL_MAJOR, n, nrhs, &A[0][0], lda, ipiv, barycentric_coords, ldx),
546 "ERROR(%s::compute_sb_coords): "
547 "internal error (could not solve linear 3x3 system)\n"
548 "(vector: (% .6e;% .6e;% .6e)\n"
549 " triangle: ((% .6e;% .6e;% .6e),\n"
550 " (% .6e;% .6e;% .6e),\n"
551 " (% .6e;% .6e;% .6e))",
552 caller, barycentric_coords[0], barycentric_coords[1], barycentric_coords[2],
553 grid_coords[triangle_indices[0]][0],
554 grid_coords[triangle_indices[0]][1],
555 grid_coords[triangle_indices[0]][2],
556 grid_coords[triangle_indices[1]][0],
557 grid_coords[triangle_indices[1]][1],
558 grid_coords[triangle_indices[1]][2],
559 grid_coords[triangle_indices[2]][0],
560 grid_coords[triangle_indices[2]][1],
561 grid_coords[triangle_indices[2]][2])
567 return (a[0] * b[1] - a[1] * b[0]) < 0.0;
572 double lon[2],
double lat[2],
int reorder[4]) {
575 int lat_neigh_offset =
576 (fabs(coords[0][2] - coords[1][2]) <
577 fabs(coords[0][2] - coords[3][2]))?1:3;
580 int closest_to_equator_idx =
581 (fabs(coords[0][2]) < fabs(coords[2][2]))?0:2;
582 int upper_edge_idx = ((coords[0][2]) > coords[2][2])?0:2;
583 int lower_edge_idx = upper_edge_idx^2;
584 int upper_edge_is_closer_to_pole =
585 closest_to_equator_idx == upper_edge_idx;
586 int closest_to_equator_edge_ordering =
588 coords[closest_to_equator_idx],
589 coords[(closest_to_equator_idx+lat_neigh_offset)%4]);
591 closest_to_equator_edge_ordering ^ upper_edge_is_closer_to_pole;
594 coords[closest_to_equator_idx], &lon[closest_to_equator_edge_ordering],
595 &lat[upper_edge_is_closer_to_pole]);
598 coords[(closest_to_equator_idx + lat_neigh_offset)%4],
599 &lon[closest_to_equator_edge_ordering^1], &dummy);
600 lat[!upper_edge_is_closer_to_pole] =
601 M_PI_2 - acos(coords[closest_to_equator_idx ^ 2][2]);
603 reorder[cell_ordering] = lower_edge_idx;
604 reorder[cell_ordering^1] = (lower_edge_idx+lat_neigh_offset)%4;
605 reorder[2+cell_ordering] = upper_edge_idx;
606 reorder[2+(cell_ordering^1)] = (upper_edge_idx+lat_neigh_offset)%4;
608 if (lon[1] < lon[0]) lon[1] += 2.0 * M_PI;
610 YAC_ASSERT_F(((lon[0] - lon[1]) != 0.0) && ((lat[0] - lat[1]) != 0.0),
611 "ERROR(get_cell_lon_lat_bounds): internal error\n"
612 "lon[0] = %e lon[1] = %e lat[0] = %e lat[1] = %e\n"
613 "cell coords: ((% .4e; % .4e; % .4e),\n"
614 " (% .4e; % .4e; % .4e),\n"
615 " (% .4e; % .4e; % .4e),\n"
616 " (% .4e; % .4e; % .4e)",
617 lon[0], lon[1], lat[0], lat[1],
618 coords[0][0], coords[0][1], coords[0][2],
619 coords[1][0], coords[1][1], coords[1][2],
620 coords[2][0], coords[2][1], coords[2][2],
621 coords[3][0], coords[3][1], coords[3][2]);
625 double point_coord[3],
double cell_lon[2],
double cell_lat[2],
626 double * point_lon,
double * point_lat) {
631 XYZtoLL(point_coord, &lon, &lat);
634 if (lon < cell_lon[0]) {
635 while (fabs(cell_lon[0] - lon) > M_PI) lon += 2.0 * M_PI;
637 while (fabs(lon - cell_lon[1]) > M_PI) lon -= 2.0 * M_PI;
645 double point_lon,
double point_lat,
double cell_lon[2],
double cell_lat[2]) {
647 return ((cell_lat[1] - cell_lat[0]) * (point_lat - cell_lat[0]) -
648 (cell_lat[1] - cell_lat[0]) * (point_lon - cell_lon[0])) > 0.0;
652 double tgt_coords[3],
656 double src_lon[2], src_lat[2], tgt_lon, tgt_lat;
667 w[0] = (tgt_lat - src_lat[1]) / (src_lat[0] - src_lat[1]);
668 w[1] = ((src_lat[1] - src_lat[0]) * (tgt_lon - src_lon[0])) /
669 ((src_lon[0] - src_lon[1]) * (src_lat[0] - src_lat[1]));
671 weights[src_reorder[0]] = w[0];
672 weights[src_reorder[2]] = w[1];
673 weights[src_reorder[3]] = 1.0 - (w[0] + w[1]);
674 weights[src_reorder[1]] = 0.0;
678 w[0] = (tgt_lon - src_lon[1]) / (src_lon[0] - src_lon[1]);
679 w[1] = ((src_lat[1] - src_lat[0]) * (tgt_lon - src_lon[1]) +
680 (src_lon[0] - src_lon[1]) * (tgt_lat - src_lat[1])) /
681 ((src_lat[0] - src_lat[1]) * (src_lon[0] - src_lon[1]));
683 weights[src_reorder[0]] = w[0];
684 weights[src_reorder[1]] = w[1];
685 weights[src_reorder[2]] = 1.0 - (w[0] + w[1]);
686 weights[src_reorder[3]] = 0.0;
691 double tgt_coords[3],
size_t num_vertices,
696 if (src_mask != NULL) {
698 for (i = 0; i < num_vertices; ++i)
if (src_mask[i] != 0)
break;
699 if (i == num_vertices)
return 0;
702 double const tol = 1e-9;
707 "ERROR(compute_weights_bary_yes): barycentric coordinates "
708 "are only supported for great circle edge cells and lon lat cells")
716 if (src_mask != NULL) {
717 double weight_sum = 1.0;
720 for (
int i = 0; i < 4; ++i)
if (!src_mask[i]) {
721 weight_sum -= weights[i];
726 if (weight_sum <
tol)
return 0;
729 if (weight_sum < 1.0) {
730 double scale = 1.0 / weight_sum;
731 for (
int i = 0; i < 4; ++i) weights[i] *= scale;
739 size_t corner_indices[num_vertices];
740 size_t triangle_indices[num_vertices-2][3];
742 for (
size_t i = 0; i < num_vertices; ++i) corner_indices[i] = i;
745 if (num_vertices > 3) {
747 corner_indices, num_vertices, 0, triangle_indices);
749 for (
size_t i = 0; i < 3; ++i) triangle_indices[0][i] = i;
753 double min_barycentric_coord = -DBL_MAX;
754 double barycentric_coords[3];
755 size_t match_index = SIZE_MAX;
757 for (
size_t i = 0; i < num_vertices - 2; ++i) {
759 double temp_barycentric_coords[3];
760 memcpy(temp_barycentric_coords, tgt_coords, 3 *
sizeof(
double));
764 temp_barycentric_coords, triangle_indices[i], src_coords,
765 "compute_weights_bary_yes");
767 double curr_min_barycentric_coord =
768 MIN(temp_barycentric_coords[0],
769 MIN(temp_barycentric_coords[1],
770 temp_barycentric_coords[2]));
772 if (curr_min_barycentric_coord > min_barycentric_coord) {
773 min_barycentric_coord = curr_min_barycentric_coord;
775 for (
int j = 0; j < 3; ++j)
776 barycentric_coords[j] =
MAX(0.0, temp_barycentric_coords[j]);
781 if (num_vertices > 3)
782 for (
size_t j = 0; j < num_vertices; ++j) weights[j] = 0.0;
785 if (src_mask != NULL)
786 for (
int j = 0; j < 3; ++j)
787 if (src_mask[triangle_indices[match_index][j]] == 0)
788 barycentric_coords[j] = 0.0;
790 double weight_sum = 0.0;
791 for (
int j = 0; j < 3; ++j) {
792 if (barycentric_coords[j] < 1e-4) barycentric_coords[j] = 0.0;
793 else weight_sum += barycentric_coords[j];
796 if (weight_sum <
tol)
return 0;
798 double scale = 1.0 / weight_sum;
800 for (
int j = 0; j < 3; ++j)
801 weights[triangle_indices[match_index][j]] =
802 barycentric_coords[j] * scale;
809 double tgt_coords[3],
size_t num_vertices,
814 if (src_mask != NULL) {
816 for (i = 0; i < num_vertices; ++i)
if (src_mask[i] != 0)
break;
817 if (i == num_vertices)
return 0;
820 double const tol = 1e-9;
824 "ERROR(compute_weights_bary_no): barycentric coordinates "
825 "are only supported for great circle edge cells and lon lat cells")
832 for (
int i = 0; i < 4; ++i)
if (weights[i] <
tol) weights[i] = 0.0;
835 if (src_mask != NULL)
836 for (
int i = 0; i < 4; ++i)
837 if (!src_mask[i] && (weights[i] != 0.0))
843 size_t corner_indices[num_vertices];
844 size_t triangle_indices[num_vertices-2][3];
846 for (
size_t i = 0; i < num_vertices; ++i) corner_indices[i] = i;
849 if (num_vertices > 3) {
851 corner_indices, num_vertices, 0, triangle_indices);
853 for (
size_t i = 0; i < 3; ++i) triangle_indices[0][i] = i;
857 double min_barycentric_coord = -DBL_MAX;
858 double barycentric_coords[3];
859 size_t match_index = SIZE_MAX;
861 for (
size_t i = 0; i < num_vertices - 2; ++i) {
863 double temp_barycentric_coords[3];
864 memcpy(temp_barycentric_coords, tgt_coords, 3 *
sizeof(
double));
868 temp_barycentric_coords, triangle_indices[i], src_coords,
869 "compute_weights_bary_no");
871 double curr_min_barycentric_coord =
872 MIN(temp_barycentric_coords[0],
873 MIN(temp_barycentric_coords[1],
874 temp_barycentric_coords[2]));
876 if (curr_min_barycentric_coord > min_barycentric_coord) {
877 min_barycentric_coord = curr_min_barycentric_coord;
879 for (
int j = 0; j < 3; ++j)
880 barycentric_coords[j] =
MAX(0.0, temp_barycentric_coords[j]);
885 if (num_vertices > 3)
886 for (
size_t j = 0; j < num_vertices; ++j) weights[j] = 0.0;
889 if (src_mask != NULL)
890 for (
int j = 0; j < 3; ++j)
891 if (!src_mask[triangle_indices[match_index][j]] &&
892 (barycentric_coords[j] >= 1e-4))
return 0;
894 double weight_sum = 0.0;
895 for (
int j = 0; j < 3; ++j) {
896 if (barycentric_coords[j] < 1e-4) barycentric_coords[j] = 0.0;
897 else weight_sum += barycentric_coords[j];
900 if (weight_sum <
tol)
return 0;
902 double scale = 1.0 / weight_sum;
904 for (
int j = 0; j < 3; ++j)
905 weights[triangle_indices[match_index][j]] =
906 barycentric_coords[j] * scale;
919 "ERROR(select_compute_weight_routine): invalid weight type")
934 int partial_coverage) {
int yac_point_in_cell(double point_coords[3], struct yac_grid_cell cell)
enum yac_interp_ncc_weight_type weight_type
int const * const_int_pointer
size_t const *const const_size_t_pointer
yac_int const * const_yac_int_pointer
void yac_triangulate_cell_indices(size_t const *corner_indices, size_t num_corners, size_t start_corner, size_t(*triangle_indices)[3])
static double get_vector_angle(double const a[3], double const b[3])
static void XYZtoLL(double const p_in[], double *lon, double *lat)
void yac_init_grid_cell(struct yac_grid_cell *cell)
void yac_free_grid_cell(struct yac_grid_cell *cell)
@ YAC_GREAT_CIRCLE_EDGE
great circle
@ YAC_LAT_CIRCLE_EDGE
latitude circle
void yac_interp_grid_do_points_search(struct yac_interp_grid *interp_grid, yac_coordinate_pointer search_coords, size_t count, size_t *src_cells)
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)
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)
struct yac_const_basic_grid_data * yac_interp_grid_get_basic_grid_data_src(struct yac_interp_grid *interp_grid)
static int compute_weights_avg_no(double tgt_coords[3], size_t num_vertices, yac_const_coordinate_pointer src_coords, int *src_mask, double *weights, enum yac_cell_type cell_type)
static int compute_weights_avg_yes(double tgt_coords[3], size_t num_vertices, yac_const_coordinate_pointer src_coords, int *src_mask, double *weights, enum yac_cell_type cell_type)
static int check_src_field_cell(double *point, size_t field_cell, size_t cell_size, size_t const *vertex_to_cell, size_t const *vertex_to_cell_offsets, yac_const_coordinate_pointer field_coordinates, struct yac_grid_cell *cell_buffer)
static struct interp_method_vtable interp_method_avg_vtable
static void get_point_lon_lat(double point_coord[3], double cell_lon[2], double cell_lat[2], double *point_lon, double *point_lat)
int(* func_compute_weights)(double[3], size_t, yac_const_coordinate_pointer, int *, double *, enum yac_cell_type cell_type)
struct interp_method * yac_interp_method_avg_new(enum yac_interp_avg_weight_type weight_type, int partial_coverage)
static func_compute_weights select_compute_weight_routine(enum yac_interp_avg_weight_type weight_type, int partial_coverage)
static int compute_weights_dist_no(double tgt_coords[3], size_t num_vertices, yac_const_coordinate_pointer src_coords, int *src_mask, double *weights, enum yac_cell_type cell_type)
static int compute_weights_bary_no(double tgt_coords[3], size_t num_vertices, yac_const_coordinate_pointer src_coords, int *src_mask, double *weights, enum yac_cell_type cell_type)
static void compute_barycentric_coords(double *barycentric_coords, size_t triangle_indices[3], yac_const_coordinate_pointer grid_coords, char const *caller)
static int determine_triangle_idx(double point_lon, double point_lat, double cell_lon[2], double cell_lat[2])
static int compute_weights_dist_yes(double tgt_coords[3], size_t num_vertices, yac_const_coordinate_pointer src_coords, int *src_mask, double *weights, enum yac_cell_type cell_type)
static void delete_avg(struct interp_method *method)
static int get_lat_edge_ordering(double const *a, double const *b)
static size_t do_search_avg(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 get_cell_lon_lat_bounds(yac_const_coordinate_pointer coords, double lon[2], double lat[2], int reorder[4])
static enum yac_cell_type determine_cell_type(enum yac_edge_type const *edge_types, size_t const *edge_indices, size_t num_edges)
static int compute_weights_bary_yes(double tgt_coords[3], size_t num_vertices, yac_const_coordinate_pointer src_coords, int *src_mask, double *weights, enum yac_cell_type cell_type)
static void compute_weights_bary_reg(double tgt_coords[3], yac_const_coordinate_pointer src_coords, double *weights)
yac_interp_avg_weight_type
@ YAC_INTERP_AVG_ARITHMETIC
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)
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)
struct interp_method_vtable * vtable
func_compute_weights compute_weights
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
const_size_t_pointer cell_to_vertex_offsets
const_size_t_pointer cell_to_vertex
const const_int_pointer num_vertices_per_cell
const_size_t_pointer cell_to_edge
const_size_t_pointer cell_to_edge_offsets
const_yac_edge_type_pointer edge_type
enum yac_edge_type * edge_type
double(* coordinates_xyz)[3]
void yac_quicksort_index_size_t_size_t(size_t *a, size_t n, size_t *idx)
#define YAC_ASSERT_F(exp, format,...)
#define YAC_ASSERT(exp, msg)
double const (* yac_const_coordinate_pointer)[3]
double(* yac_coordinate_pointer)[3]