1#define TINYSPLINE_EXPORT
4#if !defined(TINYSPLINE_NO_SERIALIZATION)
18#pragma warning(disable:4232)
20#pragma warning(disable:4710)
22#pragma warning(disable:4820)
24#pragma warning(disable:4996)
26#pragma warning(disable:5045)
29#define INIT_OUT_BSPLINE(in, out) \
31 ts_int_bspline_init(out);
74ts_int_bspline_sof_state(
const tsBSpline *spline)
77 ts_bspline_sof_control_points(spline) +
78 ts_bspline_sof_knots(spline);
82ts_int_bspline_access_ctrlp(
const tsBSpline *spline)
88ts_int_bspline_access_knots(
const tsBSpline *spline)
90 return ts_int_bspline_access_ctrlp(spline) +
91 ts_bspline_len_control_points(spline);
95ts_int_bspline_access_ctrlp_at(
const tsBSpline *spline,
100 const size_t num = ts_bspline_num_control_points(spline);
103 "index (%lu) >= num(control_points) (%lu)",
104 (
unsigned long) index,
107 *ctrlp = ts_int_bspline_access_ctrlp(spline) +
108 index * ts_bspline_dimension(spline);
109 TS_RETURN_SUCCESS(status)
113ts_int_bspline_access_knot_at(
const tsBSpline *spline,
118 const size_t num = ts_bspline_num_knots(spline);
121 "index (%lu) >= num(knots) (%lu)",
122 (
unsigned long) index,
125 *knot = ts_int_bspline_access_knots(spline)[index];
126 TS_RETURN_SUCCESS(status)
139 ts_deboornet_sof_points(net) +
140 ts_deboornet_sof_result(net);
144ts_int_deboornet_access_points(
const tsDeBoorNet *net)
150ts_int_deboornet_access_result(
const tsDeBoorNet *net)
152 if (ts_deboornet_num_result(net) == 2) {
153 return ts_int_deboornet_access_points(net);
155 return ts_int_deboornet_access_points(net) +
157 (ts_deboornet_len_points(net) -
158 ts_deboornet_dimension(net));
178 return ts_bspline_degree(spline) + 1;
190 return ts_bspline_num_control_points(spline) *
191 ts_bspline_dimension(spline);
203 return ts_bspline_len_control_points(spline) *
sizeof(
tsReal);
209 return ts_int_bspline_access_ctrlp(spline);
217 const size_t size = ts_bspline_sof_control_points(spline);
218 *ctrlp = (
tsReal*) malloc(size);
219 if (!*ctrlp) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
220 memcpy(*ctrlp, ts_int_bspline_access_ctrlp(spline), size);
221 TS_RETURN_SUCCESS(status)
232 TS_TRY(
try, err, status)
233 TS_CALL(
try, err, ts_int_bspline_access_ctrlp_at(
234 spline, index, &vals, status))
238 TS_END_TRY_RETURN(err)
246 const size_t size = ts_bspline_sof_control_points(spline);
247 memmove(ts_int_bspline_access_ctrlp(spline), ctrlp, size);
248 TS_RETURN_SUCCESS(status)
260 TS_TRY(
try, err, status)
261 TS_CALL(
try, err, ts_int_bspline_access_ctrlp_at(
262 spline, index, &to, status))
263 size = ts_bspline_dimension(spline) *
sizeof(
tsReal);
264 memcpy(to, ctrlp, size);
265 TS_END_TRY_RETURN(err)
277 return ts_bspline_num_knots(spline) *
sizeof(
tsReal);
283 return ts_int_bspline_access_knots(spline);
291 const size_t size = ts_bspline_sof_knots(spline);
292 *knots = (
tsReal*) malloc(size);
293 if (!*knots) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
294 memcpy(*knots, ts_int_bspline_access_knots(spline), size);
295 TS_RETURN_SUCCESS(status)
304 return ts_int_bspline_access_knot_at(spline, index, knot, status);
312 const size_t size = ts_bspline_sof_knots(spline);
313 const size_t num_knots = ts_bspline_num_knots(spline);
314 const size_t order = ts_bspline_order(spline);
319 for (idx = 1; idx < num_knots; idx++) {
321 if (ts_knots_equal(lst_knot, knot)) {
323 }
else if (lst_knot > knot) {
325 "decreasing knot vector at index: %lu",
332 "mult(%f) (%lu) > order (%lu)",
333 knot, (
unsigned long) mult,
334 (
unsigned long) order)
338 memmove(ts_int_bspline_access_knots(spline), knots, size);
339 TS_RETURN_SUCCESS(status)
354 TS_TRY(
try, err, status)
355 TS_CALL(
try, err, ts_bspline_knots(
356 spline, &values, status))
359 values[1] = (
tsReal) knot1;
360 va_start(argp, knot1);
361 for (idx = 2; idx < ts_bspline_num_knots(spline); idx++)
362 values[idx] = (
tsReal) va_arg(argp,
double);
365 TS_CALL(
try, err, ts_bspline_set_knots(
366 spline, values, status))
368 if (values) free(values);
369 TS_END_TRY_RETURN(err)
381 tsReal oldKnot = ts_int_bspline_access_knots(spline)[0];
382 TS_TRY(
try, err, status)
383 TS_CALL(
try, err, ts_int_bspline_access_knot_at(
384 spline, index, &oldKnot, status))
388 knots = ts_int_bspline_access_knots(spline);
390 TS_CALL(
try, err, ts_bspline_set_knots(
391 spline, knots, status))
394 if (knots) knots[index] = oldKnot;
395 TS_END_TRY_RETURN(err)
409 ts_int_bspline_init(&spline);
414ts_int_bspline_generate_knots(
const tsBSpline *spline,
418 const size_t n_knots = ts_bspline_num_knots(spline);
419 const size_t deg = ts_bspline_degree(spline);
420 const size_t order = ts_bspline_order(spline);
426 if (type ==
TS_BEZIERS && n_knots % order != 0) {
428 "num(knots) (%lu) %% order (%lu) != 0",
429 (
unsigned long) n_knots, (
unsigned long) order)
432 knots = ts_int_bspline_access_knots(spline);
438 for (i = 1; i < n_knots-1; i++)
444 / (n_knots - 2*deg - 1);
446 for (i = order ;i < n_knots-order; i++)
452 / (n_knots/order - 1);
454 for (i = order; i < n_knots-order; i += order)
455 ts_arr_fill(knots + i,
460 TS_RETURN_SUCCESS(status)
464ts_bspline_new(
size_t num_control_points,
471 const size_t order = degree + 1;
472 const size_t num_knots = num_control_points + order;
473 const size_t len_ctrlp = num_control_points * dimension;
475 const size_t sof_real =
sizeof(
tsReal);
477 const size_t sof_ctrlp_vec = len_ctrlp * sof_real;
478 const size_t sof_knots_vec = num_knots * sof_real;
479 const size_t sof_spline = sof_impl + sof_ctrlp_vec + sof_knots_vec;
482 ts_int_bspline_init(spline);
485 TS_RETURN_0(status,
TS_DIM_ZERO,
"unsupported dimension: 0")
489 "unsupported number of knots: %lu > %i",
492 if (degree >= num_control_points) {
494 "degree (%lu) >= num(control_points) (%lu)",
495 (
unsigned long) degree,
496 (
unsigned long) num_control_points)
500 if (!spline->
pImpl) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
507 TS_TRY(
try, err, status)
508 TS_CALL(
try, err, ts_int_bspline_generate_knots(
509 spline, type, status))
511 ts_bspline_free(spline);
512 TS_END_TRY_RETURN(err)
516ts_bspline_new_with_control_points(
size_t num_control_points,
530 TS_TRY(
try, err, status)
531 TS_CALL(
try, err, ts_bspline_new(
532 num_control_points, dimension,
533 degree, type, spline, status))
535 ts_bspline_free(spline);
537 ctrlp = ts_int_bspline_access_ctrlp(spline);
539 ctrlp[0] = (
tsReal) first;
540 va_start(argp, first);
541 for (i = 1; i < ts_bspline_len_control_points(spline); i++)
542 ctrlp[i] = (
tsReal) va_arg(argp,
double);
545 TS_RETURN_SUCCESS(status)
554 if (src == dest) TS_RETURN_SUCCESS(status)
555 ts_int_bspline_init(dest);
556 size = ts_int_bspline_sof_state(src);
558 if (!dest->pImpl) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
559 memcpy(dest->pImpl, src->
pImpl, size);
560 TS_RETURN_SUCCESS(status)
567 if (src == dest)
return;
568 dest->pImpl = src->
pImpl;
569 ts_int_bspline_init(src);
576 ts_int_bspline_init(spline);
619 return ts_deboornet_num_points(net) *
620 ts_deboornet_dimension(net);
626 return net->
pImpl->n_points;
632 return ts_deboornet_len_points(net) *
sizeof(
tsReal);
638 return ts_int_deboornet_access_points(net);
646 const size_t size = ts_deboornet_sof_points(net);
647 *points = (
tsReal*) malloc(size);
648 if (!*points) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
649 memcpy(*points, ts_int_deboornet_access_points(net), size);
650 TS_RETURN_SUCCESS(status)
656 return ts_deboornet_num_result(net) *
657 ts_deboornet_dimension(net);
663 return ts_deboornet_num_points(net) == 2 ? 2 : 1;
669 return ts_deboornet_len_result(net) *
sizeof(
tsReal);
675 return ts_int_deboornet_access_result(net);
683 const size_t size = ts_deboornet_sof_result(net);
684 *result = (
tsReal*) malloc(size);
685 if (!*result) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
686 memcpy(*result, ts_int_deboornet_access_result(net), size);
687 TS_RETURN_SUCCESS(status)
698ts_deboornet_init(
void)
701 ts_int_deboornet_init(&net);
706ts_int_deboornet_new(
const tsBSpline *spline,
710 const size_t dim = ts_bspline_dimension(spline);
711 const size_t deg = ts_bspline_degree(spline);
712 const size_t order = ts_bspline_order(spline);
713 const size_t num_points = (size_t)(order * (order+1) * 0.5f);
715 const size_t fixed_num_points = num_points < 2 ? 2 : num_points;
717 const size_t sof_real =
sizeof(
tsReal);
719 const size_t sof_points_vec = fixed_num_points *
dim * sof_real;
720 const size_t sof_net = sof_impl + sof_points_vec;
730 net->pImpl->n_points = fixed_num_points;
731 TS_RETURN_SUCCESS(status)
737 if (net->pImpl) free(net->pImpl);
738 ts_int_deboornet_init(net);
747 if (src == dest) TS_RETURN_SUCCESS(status)
748 ts_int_deboornet_init(dest);
749 size = ts_int_deboornet_sof_state(src);
751 if (!dest->pImpl) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
752 memcpy(dest->pImpl, src->
pImpl, size);
753 TS_RETURN_SUCCESS(status)
760 if (src == dest)
return;
761 dest->pImpl = src->
pImpl;
762 ts_int_deboornet_init(src);
773ts_int_cubic_point(
const tsReal *point,
778 const size_t size =
dim *
sizeof(
tsReal);
782 TS_CALL_ROE(err, ts_bspline_new(
785 ctrlp = ts_int_bspline_access_ctrlp(spline);
786 for (i = 0; i < 4; i++) {
787 memcpy(ctrlp + i*
dim,
791 TS_RETURN_SUCCESS(status)
795ts_int_thomas_algorithm(
const tsReal *a,
809 "unsupported dimension: 0")
813 "num(points) (%lu) <= 1",
817 if (!cc) TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
819 TS_TRY(try, err, status)
821 if (fabs(b[0]) <= fabs(c[0])) {
823 "error: |%f| <= |%f|", b[0], c[0])
828 for (i = 0; i <
dim; i++)
830 for (i = 1; i < num; i++) {
831 if (fabs(b[i]) <= fabs(a[i]) + fabs(c[i])) {
833 "error: |%f| <= |%f| + |%f|",
839 m = 1.f / (b[i] - a[i] * cc[i - 1]);
849 for (j = 0; j <
dim; j++) {
852 d[
k] = (d[
k] - a[i] * d[l]) * m;
857 for (i = num-1; i > 0; i--) {
858 for (j = 0; j <
dim; j++) {
861 d[
k] -= cc[i-1] * d[l];
866 TS_END_TRY_RETURN(err)
870ts_int_relaxed_uniform_cubic_bspline(
const tsReal *points,
876 const size_t order = 4;
877 const tsReal as = 1.f/6.f;
878 const tsReal at = 1.f/3.f;
879 const tsReal tt = 2.f/3.f;
890 TS_RETURN_0(status,
TS_DIM_ZERO,
"unsupported dimension: 0")
893 "num(points) (%lu) <= 1",
901 TS_TRY(
try, err, status)
903 TS_CALL(
try, err, ts_bspline_new(
904 (n-1) * 4,
dim, order - 1,
906 ctrlp = ts_int_bspline_access_ctrlp(spline);
908 s = (
tsReal*) malloc(n * sof_ctrlp);
915 memcpy(
s, b, sof_ctrlp);
916 memcpy(
s + (n-1)*
dim, b + (n-1)*
dim, sof_ctrlp);
919 for (i = 1; i < n-1; i++) {
920 for (d = 0; d <
dim; d++) {
931 for (i = 0; i < n-1; i++) {
932 for (d = 0; d <
dim; d++) {
937 ctrlp[
k+
dim] = tt*b[j] + at*b[l];
938 ctrlp[
k+2*
dim] = at*b[j] + tt*b[l];
939 ctrlp[
k+3*
dim] =
s[l];
943 ts_bspline_free(spline);
947 TS_END_TRY_RETURN(err)
951ts_bspline_interpolate_cubic_natural(
const tsReal *points,
957 const size_t sof_ctrlp = dimension *
sizeof(
tsReal);
958 const size_t len_points = num_points * dimension;
959 const size_t num_int_points = num_points - 2;
960 const size_t len_int_points = num_int_points * dimension;
961 tsReal *buffer, *a, *b, *c, *d;
965 ts_int_bspline_init(spline);
968 if (num_points == 1) {
969 TS_CALL_ROE(err, ts_int_cubic_point(
970 points, dimension, spline, status))
971 TS_RETURN_SUCCESS(status)
973 if (num_points == 2) {
974 return ts_int_relaxed_uniform_cubic_bspline(
975 points, num_points, dimension, spline, status);
979 TS_TRY(
try, err, status)
980 buffer = (
tsReal *) malloc(
982 2 * num_int_points *
sizeof(
tsReal) +
986 num_points * dimension *
sizeof(
tsReal));
995 ts_arr_fill(a, num_int_points, 1);
996 b = a + num_int_points;
997 ts_arr_fill(b, num_int_points, 4);
998 d = b + num_int_points
1001 for (i = 0; i < num_int_points; i++) {
1002 for (j = 0; j < dimension; j++) {
1003 k = i * dimension + j;
1004 l = (i+1) * dimension + j;
1005 d[
k] = 6 * points[l];
1008 for (i = 0; i < dimension; i++) {
1012 k = len_int_points - (i+1);
1013 l = len_points - (i+1);
1019 if (num_int_points == 1) {
1020 for (i = 0; i < dimension; i++)
1023 TS_CALL(
try, err, ts_int_thomas_algorithm(
1024 a, b, c, num_int_points, dimension, d,
1027 memcpy(d - dimension, points, sof_ctrlp);
1028 memcpy(d + num_int_points * dimension,
1029 points + (num_points-1) * dimension,
1031 TS_CALL(
try, err, ts_int_relaxed_uniform_cubic_bspline(
1032 d - dimension, num_points, dimension, spline, status))
1034 ts_bspline_free(spline);
1036 if (buffer) free(buffer);
1037 TS_END_TRY_RETURN(err)
1041ts_bspline_interpolate_catmull_rom(
const tsReal *points,
1051 const size_t sof_real =
sizeof(
tsReal);
1052 const size_t sof_ctrlp = dimension * sof_real;
1063 tsReal c1, c2, d1, d2, m1, m2;
1064 tsReal *p0, *p1, *p2, *p3;
1066 ts_int_bspline_init(spline);
1068 TS_RETURN_0(status,
TS_DIM_ZERO,
"unsupported dimension: 0")
1069 if (num_points == 0)
1075 cr_ctrlp = (
tsReal *) malloc((num_points + 2) * sof_ctrlp);
1077 TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
1078 memcpy(cr_ctrlp + dimension, points, num_points * sof_ctrlp);
1084 p0 = cr_ctrlp + (i * dimension);
1085 p1 = p0 + dimension;
1086 if (ts_distance(p0, p1, dimension) <= eps) {
1089 if (i < num_points - 1) {
1090 memmove(p1, p1 + dimension,
1091 (num_points - (i + 1)) * sof_ctrlp);
1099 if (num_points == 1) {
1101 TS_CALL_ROE(err, ts_int_cubic_point(
1102 points, dimension, spline, status))
1103 TS_RETURN_SUCCESS(status)
1107 p0 = cr_ctrlp + dimension;
1108 if (first && ts_distance(first, p0, dimension) > eps) {
1109 memcpy(cr_ctrlp, first, sof_ctrlp);
1111 p1 = p0 + dimension;
1112 for (d = 0; d < dimension; d++)
1113 cr_ctrlp[d] = p0[d] + (p0[d] - p1[d]);
1115 p1 = cr_ctrlp + (num_points * dimension);
1116 if (last && ts_distance(p1, last, dimension) > eps) {
1117 memcpy(cr_ctrlp + ((num_points + 1) * dimension),
1120 p0 = p1 - dimension;
1121 for (d = 0; d < dimension; d++) {
1122 cr_ctrlp[((num_points + 1) * dimension) + d] =
1123 p1[d] + (p1[d] - p0[d]);
1126 num_points = num_points + 2;
1130 TS_TRY(
try, err, status)
1131 TS_CALL(
try, err, ts_bspline_new(
1132 (num_points - 3) * 4, dimension, 3,
1134 bs_ctrlp = ts_int_bspline_access_ctrlp(spline);
1138 for (i = 0; i < ts_bspline_num_control_points(spline) / 4; i++) {
1139 p0 = cr_ctrlp + ((i+0) * dimension);
1140 p1 = cr_ctrlp + ((i+1) * dimension);
1141 p2 = cr_ctrlp + ((i+2) * dimension);
1142 p3 = cr_ctrlp + ((i+3) * dimension);
1145 t1 = t0 + (
tsReal) pow(ts_distance(p0, p1, dimension), alpha);
1146 t2 = t1 + (
tsReal) pow(ts_distance(p1, p2, dimension), alpha);
1147 t3 = t2 + (
tsReal) pow(ts_distance(p2, p3, dimension), alpha);
1149 c1 = (t2-t1) / (t2-t0);
1150 c2 = (t1-t0) / (t2-t0);
1151 d1 = (t3-t2) / (t3-t1);
1152 d2 = (t2-t1) / (t3-t1);
1154 for (d = 0; d < dimension; d++) {
1155 m1 = (t2-t1)*(c1*(p1[d]-p0[d])/(t1-t0)
1156 + c2*(p2[d]-p1[d])/(t2-t1));
1157 m2 = (t2-t1)*(d1*(p2[d]-p1[d])/(t2-t1)
1158 + d2*(p3[d]-p2[d])/(t3-t2));
1159 bs_ctrlp[((i*4 + 0) * dimension) + d] = p1[d];
1160 bs_ctrlp[((i*4 + 1) * dimension) + d] = p1[d] + m1/3;
1161 bs_ctrlp[((i*4 + 2) * dimension) + d] = p2[d] - m2/3;
1162 bs_ctrlp[((i*4 + 3) * dimension) + d] = p2[d];
1166 TS_RETURN_SUCCESS(status)
1177ts_int_bspline_find_knot(
const tsBSpline *spline,
1183 const size_t deg = ts_bspline_degree(spline);
1184 const size_t num_knots = ts_bspline_num_knots(spline);
1185 const tsReal *knots = ts_int_bspline_access_knots(spline);
1189 ts_bspline_domain(spline, &min, &max);
1192 if (ts_knots_equal(*knot, min)) *knot =
min;
1195 "knot (%f) < min(domain) (%f)",
1199 else if (*knot > max && !ts_knots_equal(*knot, max)) {
1201 "knot (%f) > max(domain) (%f)",
1206 if (ts_knots_equal(*knot, knots[num_knots - 1])) {
1207 *idx = num_knots - 1;
1210 high = num_knots - 1;
1211 *idx = (low+high) / 2;
1212 while (*knot < knots[*idx] || *knot >= knots[*idx + 1]) {
1213 if (*knot < knots[*idx])
1217 *idx = (low+high) / 2;
1222 while (*idx < num_knots - 1 &&
1223 ts_knots_equal(*knot, knots[*idx + 1])) {
1226 if (ts_knots_equal(*knot, knots[*idx]))
1227 *knot = knots[*idx];
1230 for (*mult = deg + 1; *mult > 0 ; (*mult)--) {
1231 if (ts_knots_equal(*knot, knots[*idx - (*mult-1)]))
1235 TS_RETURN_SUCCESS(status)
1239ts_int_bspline_eval_woa(
const tsBSpline *spline,
1244 const size_t deg = ts_bspline_degree(spline);
1245 const size_t order = ts_bspline_order(spline);
1246 const size_t dim = ts_bspline_dimension(spline);
1247 const size_t num_knots = ts_bspline_num_knots(spline);
1248 const size_t sof_ctrlp =
dim *
sizeof(
tsReal);
1250 const tsReal *ctrlp = ts_int_bspline_access_ctrlp(spline);
1251 const tsReal *knots = ts_int_bspline_access_knots(spline);
1272 points = ts_int_deboornet_access_points(net);
1280 TS_CALL_ROE(err, ts_int_bspline_find_knot(
1281 spline, &
u, &
k, &
s, status))
1287 net->
pImpl->
h = deg <
s ? 0 : deg-
s;
1298 k == num_knots - 1) {
1299 from =
k == deg ? 0 : (
k-
s) *
dim;
1300 net->
pImpl->n_points = 1;
1301 memcpy(points, ctrlp + from, sof_ctrlp);
1304 net->
pImpl->n_points = 2;
1305 memcpy(points, ctrlp + from, 2 * sof_ctrlp);
1312 net->
pImpl->n_points = (size_t)(N * (N+1) * 0.5f);
1315 memcpy(points, ctrlp + fst*
dim, N * sof_ctrlp);
1321 for (;r <= ts_deboornet_num_insertions(net); r++) {
1323 for (; i <= lst; i++) {
1325 a = (ts_deboornet_knot(net) - ui) /
1326 (knots[i+deg-r+1] - ui);
1329 for (d = 0; d <
dim; d++) {
1331 a_hat * points[lidx++] +
1339 TS_RETURN_SUCCESS(status)
1349 ts_int_deboornet_init(net);
1350 TS_TRY(
try, err, status)
1351 TS_CALL(
try, err, ts_int_deboornet_new(
1352 spline, net, status))
1353 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1354 spline, knot, net, status))
1356 ts_deboornet_free(net);
1357 TS_END_TRY_RETURN(err)
1367 const size_t dim = ts_bspline_dimension(spline);
1368 const size_t sof_point =
dim *
sizeof(
tsReal);
1369 const size_t sof_points = num * sof_point;
1374 TS_TRY(
try, err, status)
1375 *points = (
tsReal *) malloc(sof_points);
1380 TS_CALL(
try, err, ts_int_deboornet_new(
1381 spline,&net, status))
1382 for (i = 0; i < num; i++) {
1383 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1384 spline, knots[i], &net, status))
1385 result = ts_int_deboornet_access_result(&net);
1386 memcpy((*points) + i *
dim, result, sof_point);
1393 ts_deboornet_free(&net);
1394 TS_END_TRY_RETURN(err)
1407 num = num == 0 ? 100 : num;
1412 TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
1414 ts_bspline_uniform_knot_seq(spline, num, knots);
1415 TS_TRY(
try, err, status)
1416 TS_CALL(
try, err, ts_bspline_eval_all(
1417 spline, knots, num, points, status))
1420 TS_END_TRY_RETURN(err)
1435 const size_t dim = ts_bspline_dimension(spline);
1442 ts_int_deboornet_init(net);
1446 "dimension (%lu) <= index (%lu)",
1447 (
unsigned long)
dim,
1448 (
unsigned long) index)
1453 ts_bspline_domain(spline, &min, &max);
1454 TS_TRY(
try, err, status)
1455 TS_CALL(
try, err, ts_int_deboornet_new(
1456 spline, net, status))
1458 mid = (
tsReal) ((min + max) / 2.0);
1459 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1460 spline, mid, net, status))
1461 P = ts_int_deboornet_access_result(net);
1462 dist = ts_distance(&P[index], &value, 1);
1464 TS_RETURN_SUCCESS(status)
1466 if (P[index] < value)
1471 if (P[index] < value)
1476 }
while (i++ < max_iter);
1479 "maximum iterations (%lu) exceeded",
1480 (
unsigned long) max_iter)
1483 ts_deboornet_free(net);
1484 TS_END_TRY_RETURN(err)
1491 *min = ts_int_bspline_access_knots(spline)
1492 [ts_bspline_degree(spline)];
1493 *max = ts_int_bspline_access_knots(spline)
1494 [ts_bspline_num_knots(spline) - ts_bspline_order(spline)];
1503 const size_t deg = ts_bspline_degree(spline);
1504 const size_t dim = ts_bspline_dimension(spline);
1511 ts_int_bspline_init(&derivative);
1512 ts_int_deboornet_init(&first);
1513 ts_int_deboornet_init(&last);
1515 TS_TRY(
try, err, status)
1516 for (i = 0; i < deg; i++) {
1517 TS_CALL(
try, err, ts_bspline_derive(
1518 spline, i, -1.f, &derivative, status))
1519 ts_bspline_domain(&derivative, &min, &max);
1520 TS_CALL(
try, err, ts_bspline_eval(
1521 &derivative, min, &first, status))
1522 TS_CALL(
try, err, ts_bspline_eval(
1523 &derivative, max, &last, status))
1524 *closed = ts_distance(
1525 ts_int_deboornet_access_result(&first),
1526 ts_int_deboornet_access_result(&last),
1527 dim) <= epsilon ? 1 : 0;
1528 ts_bspline_free(&derivative);
1529 ts_deboornet_free(&first);
1530 ts_deboornet_free(&last);
1532 TS_RETURN_SUCCESS(status)
1535 ts_bspline_free(&derivative);
1536 ts_deboornet_free(&first);
1537 ts_deboornet_free(&last);
1538 TS_END_TRY_RETURN(err)
1545 int has_first_normal,
1552 tsReal xc[3], xn[3], v1[3], c1, v2[3], c2, rL[3], tL[3];
1558 TS_RETURN_SUCCESS(status);
1560 TS_TRY(
try, err, status)
1561 TS_CALL(
try, err, ts_int_deboornet_new(
1562 spline, &curr, status))
1563 TS_CALL(
try, err, ts_int_deboornet_new(
1564 spline, &next, status))
1565 TS_CALL(
try, err, ts_bspline_derive(
1566 spline, 1, (
tsReal) -1.0, &deriv, status))
1569 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1570 spline, knots[0], &curr, status))
1571 ts_vec3_set(frames[0].position,
1572 ts_int_deboornet_access_result(&curr),
1573 ts_bspline_dimension(spline));
1575 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1576 &deriv, knots[0], &curr, status))
1577 ts_vec3_set(frames[0].tangent,
1578 ts_int_deboornet_access_result(&curr),
1579 ts_bspline_dimension(&deriv));
1580 ts_vec_norm(frames[0].tangent, 3, frames[0].tangent);
1582 if (!has_first_normal) {
1583 fx = (
tsReal) fabs(frames[0].tangent[0]);
1584 fy = (
tsReal) fabs(frames[0].tangent[1]);
1585 fz = (
tsReal) fabs(frames[0].tangent[2]);
1587 ts_vec3_init(frames[0].normal,
1593 ts_vec3_init(frames[0].normal,
1599 ts_vec3_init(frames[0].normal,
1604 ts_vec3_cross(frames[0].tangent,
1607 ts_vec_norm(frames[0].normal, 3, frames[0].normal);
1608 if (ts_bspline_dimension(spline) >= 3) {
1613 ts_vec3_cross(frames[0].tangent,
1619 ts_vec_norm(frames[0].normal, 3, frames[0].normal);
1622 ts_vec3_cross(frames[0].tangent,
1624 frames[0].binormal);
1626 for (i = 0; i < num - 1; i++) {
1628 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1629 spline, knots[i], &curr, status))
1630 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1631 spline, knots[i+1], &next, status))
1633 ts_int_deboornet_access_result(&curr),
1634 ts_bspline_dimension(spline));
1636 ts_int_deboornet_access_result(&next),
1637 ts_bspline_dimension(spline));
1640 ts_vec3_set(frames[i+1].position, xn, 3);
1643 ts_vec_sub(xn, xc, 3, v1);
1644 c1 = ts_vec_dot(v1, v1, 3);
1647 rL[0] = (
tsReal) 2.0 / c1;
1648 rL[1] = ts_vec_dot(v1, frames[i].normal, 3);
1649 rL[2] = rL[0] * rL[1];
1650 ts_vec_mul(v1, 3, rL[2], rL);
1651 ts_vec_sub(frames[i].normal, rL, 3, rL);
1654 tL[0] = (
tsReal) 2.0 / c1;
1655 tL[1] = ts_vec_dot(v1, frames[i].tangent, 3);
1656 tL[2] = tL[0] * tL[1];
1657 ts_vec_mul(v1, 3, tL[2], tL);
1658 ts_vec_sub(frames[i].tangent, tL, 3, tL);
1661 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1662 &deriv, knots[i+1], &next, status))
1664 ts_int_deboornet_access_result(&next),
1665 ts_bspline_dimension(&deriv));
1666 ts_vec_norm(xn, 3, xn);
1667 ts_vec_sub(xn, tL, 3, v2);
1668 c2 = ts_vec_dot(v2, v2, 3);
1672 frames[i+1].normal, 3);
1673 xc[0] = (
tsReal) 2.0 / c2;
1674 xc[1] = ts_vec_dot(v2, rL, 3);
1675 xc[2] = xc[0] * xc[1];
1676 ts_vec_mul(v2, 3, xc[2], xc);
1677 ts_vec_sub(rL, xc, 3, xc);
1678 ts_vec_norm(xc, 3, xc);
1681 ts_vec3_cross(xn, xc, frames[i+1].binormal);
1684 ts_vec3_set(frames[i+1].tangent, xn, 3);
1685 ts_vec3_set(frames[i+1].normal, xc, 3);
1688 ts_bspline_free(&deriv);
1689 ts_deboornet_free(&curr);
1690 ts_deboornet_free(&next);
1691 TS_END_TRY_RETURN(err)
1703 tsReal dist, lst_knot, cur_knot;
1704 size_t i,
dim = ts_bspline_dimension(spline);
1709 if (num == 0) TS_RETURN_SUCCESS(status);
1711 TS_TRY(
try, err, status)
1712 TS_CALL(
try, err, ts_int_deboornet_new(
1713 spline, &lst, status))
1714 TS_CALL(
try, err, ts_int_deboornet_new(
1715 spline, &cur, status))
1718 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1719 spline, knots[0], &lst, status));
1720 lengths[0] = (
tsReal) 0.0;
1722 for (i = 1; i < num; i++) {
1723 TS_CALL(
try, err, ts_int_bspline_eval_woa(
1724 spline, knots[i], &cur, status));
1725 lst_knot = ts_deboornet_knot(&lst);
1726 cur_knot = ts_deboornet_knot(&cur);
1727 if (cur_knot < lst_knot) {
1729 "decreasing knot at index: %lu",
1732 dist = ts_distance(ts_deboornet_result_ptr(&lst),
1733 ts_deboornet_result_ptr(&cur),
1735 lengths[i] = lengths[i-1] + dist;
1736 ts_deboornet_move(&lst, &tmp);
1737 ts_deboornet_move(&cur, &lst);
1738 ts_deboornet_move(&tmp, &cur);
1741 ts_deboornet_free(&lst);
1742 ts_deboornet_free(&cur);
1743 TS_END_TRY_RETURN(err)
1757 size_t dim, deg, order;
1771 ts_int_bspline_init(&worker);
1772 INIT_OUT_BSPLINE(spline, sub)
1774 ts_bspline_domain(spline, &min, &max);
1775 dim = ts_bspline_dimension(spline);
1776 deg = ts_bspline_degree(spline);
1777 order = ts_bspline_order(spline);
1780 if (ts_knots_equal(knot0, knot1)) {
1789 reverse = knot0 > knot1;
1792 if (!tmp) TS_RETURN_0(status,
TS_MALLOC,
"out of memory");
1798 TS_TRY(
try, err, status)
1799 if (!ts_knots_equal(knot0 , min)) {
1800 TS_CALL(
try , err, ts_bspline_split(
1801 spline, knot0, &worker, &k0, status))
1802 }
else { k0 = deg; }
1803 if (!ts_knots_equal(knot1, max)) {
1804 TS_CALL(
try , err, ts_bspline_split(
1807 !worker.
pImpl ? spline : &worker,
1808 knot1, &worker, &k1, status))
1810 k1 = ts_bspline_num_knots(
1813 !worker.
pImpl ? spline : &worker) - 1;
1817 if (!worker.
pImpl) {
1818 TS_CALL(
try, err, ts_bspline_copy(
1819 spline, &worker, status))
1821 ctrlp = ts_int_bspline_access_ctrlp(&worker);
1822 knots = ts_int_bspline_access_knots(&worker);
1823 nc = ts_bspline_num_control_points(&worker);
1825 c0 = (k0-deg) *
dim;
1826 c1 = (k1-order) *
dim;
1827 nc = ((c1-c0) /
dim) + 1;
1828 nk = (k1-k0) + order;
1831 ctrlp = ts_int_bspline_access_ctrlp(&worker);
1832 knots = ts_int_bspline_access_knots(&worker);
1839 memmove(ctrlp + nc *
dim,
1847 i = ts_int_bspline_sof_state(&worker);
1849 if (worker.
pImpl == NULL) {
1857 for (i = 0; i < nc / 2; i++) {
1861 memmove(ctrlp + i *
dim,
1862 ctrlp + (nc-1 - i) *
dim,
1864 memcpy(ctrlp + (nc-1 - i) *
dim,
1871 if (spline == sub) ts_bspline_free(sub);
1872 ts_bspline_move(&worker, sub);
1874 ts_bspline_free(&worker);
1877 TS_END_TRY_RETURN(err)
1887 if (num == 0)
return;
1888 ts_bspline_domain(spline, &min, &max);
1889 for (i = 0; i < num; i++) {
1890 knots[i] = max - min;
1891 knots[i] *= (
tsReal) i / (num - 1);
1896 knots[num - 1] = max;
1908 tsReal *samples = NULL, *lengths = NULL;
1910 if (num == 0) TS_RETURN_SUCCESS(status);
1911 if (num_samples == 0) num_samples = 200;
1913 samples = (
tsReal *) malloc(2 * num_samples *
sizeof(
tsReal));
1914 if (!samples) TS_RETURN_0(status,
TS_MALLOC,
"out of memory");
1915 ts_bspline_uniform_knot_seq(spline, num_samples, samples);
1916 lengths = samples + num_samples;
1917 TS_TRY(
try, err, status)
1918 TS_CALL(
try, err, ts_bspline_chord_lengths(
1919 spline, samples, num_samples, lengths, status))
1920 TS_CALL(
try, err, ts_chord_lengths_equidistant_knot_seq(
1921 samples, lengths, num_samples, num, knots, status))
1925 TS_END_TRY_RETURN(err)
1936ts_int_bspline_resize(
const tsBSpline *spline,
1942 const size_t deg = ts_bspline_degree(spline);
1943 const size_t dim = ts_bspline_dimension(spline);
1944 const size_t sof_real =
sizeof(
tsReal);
1946 const size_t num_ctrlp = ts_bspline_num_control_points(spline);
1947 const size_t num_knots = ts_bspline_num_knots(spline);
1948 const size_t nnum_ctrlp = num_ctrlp + n;
1949 const size_t nnum_knots = num_knots + n;
1950 const size_t min_num_ctrlp_vec = n < 0 ? nnum_ctrlp : num_ctrlp;
1951 const size_t min_num_knots_vec = n < 0 ? nnum_knots : num_knots;
1953 const size_t sof_min_num_ctrlp = min_num_ctrlp_vec *
dim * sof_real;
1954 const size_t sof_min_num_knots = min_num_knots_vec * sof_real;
1957 const tsReal* from_ctrlp = ts_int_bspline_access_ctrlp(spline);
1958 const tsReal* from_knots = ts_int_bspline_access_knots(spline);
1964 if (n == 0)
return ts_bspline_copy(spline, resized, status);
1966 INIT_OUT_BSPLINE(spline, resized)
1967 TS_CALL_ROE(err, ts_bspline_new(
1970 to_ctrlp = ts_int_bspline_access_ctrlp(&tmp);
1971 to_knots = ts_int_bspline_access_knots(&tmp);
1974 if (!back && n < 0) {
1975 memcpy(to_ctrlp, from_ctrlp - n*
dim, sof_min_num_ctrlp);
1976 memcpy(to_knots, from_knots - n , sof_min_num_knots);
1977 }
else if (!back && n > 0) {
1978 memcpy(to_ctrlp + n*
dim, from_ctrlp, sof_min_num_ctrlp);
1979 memcpy(to_knots + n , from_knots, sof_min_num_knots);
1982 memcpy(to_ctrlp, from_ctrlp, sof_min_num_ctrlp);
1983 memcpy(to_knots, from_knots, sof_min_num_knots);
1986 if (spline == resized)
1987 ts_bspline_free(resized);
1988 ts_bspline_move(&tmp, resized);
1989 TS_RETURN_SUCCESS(status)
1999 const size_t sof_real =
sizeof(
tsReal);
2000 const size_t dim = ts_bspline_dimension(spline);
2001 const size_t sof_ctrlp =
dim * sof_real;
2002 size_t deg = ts_bspline_degree(spline);
2003 size_t num_ctrlp = ts_bspline_num_control_points(spline);
2004 size_t num_knots = ts_bspline_num_knots(spline);
2010 size_t m, i, j,
k, l;
2019 INIT_OUT_BSPLINE(spline, deriv)
2020 TS_CALL_ROE(err, ts_bspline_copy(spline, &worker, status))
2021 ctrlp = ts_int_bspline_access_ctrlp(&worker);
2022 knots = ts_int_bspline_access_knots(&worker);
2024 TS_TRY(
try, err, status)
2025 for (m = 1; m <= n; m++) {
2027 ts_arr_fill(ctrlp,
dim, 0.f);
2028 ts_bspline_domain(spline,
2036 for (i = 2*deg + 1; i < num_knots - (deg+1); i++) {
2037 if (!ts_knots_equal(knots[i], knots[i-deg]))
2039 fst = ctrlp + (i - (deg+1)) *
dim;
2041 dist = ts_distance(fst, snd,
dim);
2042 if (epsilon >= 0.f && dist > epsilon) {
2043 TS_THROW_1(
try, err, status,
2045 "discontinuity at knot: %f",
2050 (num_ctrlp - (i+1-deg)) * sof_ctrlp);
2053 (num_knots - (i+1)) * sof_real);
2059 for (i = 0; i < num_ctrlp-1; i++) {
2060 for (j = 0; j <
dim; j++) {
2063 ctrlp[
k] = ctrlp[l] - ctrlp[
k];
2064 kid1 = knots[i+deg+1];
2078 TS_CALL(
try, err, ts_bspline_new(
2081 memcpy(ts_int_bspline_access_ctrlp(&swap),
2083 num_ctrlp * sof_ctrlp);
2084 memcpy(ts_int_bspline_access_knots(&swap),
2086 num_knots * sof_real);
2087 if (spline == deriv)
2088 ts_bspline_free(deriv);
2089 ts_bspline_move(&swap, deriv);
2091 ts_bspline_free(&worker);
2092 TS_END_TRY_RETURN(err)
2096ts_int_bspline_insert_knot(
const tsBSpline *spline,
2102 const size_t deg = ts_bspline_degree(spline);
2103 const size_t order = ts_bspline_order(spline);
2104 const size_t dim = ts_bspline_dimension(spline);
2105 const tsReal knot = ts_deboornet_knot(net);
2106 const size_t k = ts_deboornet_index(net);
2107 const size_t mult = ts_deboornet_multiplicity(net);
2108 const size_t sof_real =
sizeof(
tsReal);
2109 const size_t sof_ctrlp =
dim * sof_real;
2117 tsReal *ctrlp_spline, *ctrlp_result;
2118 tsReal *knots_spline, *knots_result;
2119 size_t num_ctrlp_result;
2120 size_t num_knots_result;
2124 INIT_OUT_BSPLINE(spline, result)
2126 return ts_bspline_copy(spline, result, status);
2127 if (mult + n > order) {
2129 "multiplicity(%f) (%lu) + %lu > order (%lu)",
2130 knot, (
unsigned long) mult, (
unsigned long) n,
2131 (
unsigned long) order)
2134 TS_CALL_ROE(err, ts_int_bspline_resize(
2135 spline, (
int)n, 1, result, status))
2136 ctrlp_spline = ts_int_bspline_access_ctrlp(spline);
2137 knots_spline = ts_int_bspline_access_knots(spline);
2138 ctrlp_result = ts_int_bspline_access_ctrlp(result);
2139 knots_result = ts_int_bspline_access_knots(result);
2140 num_ctrlp_result = ts_bspline_num_control_points(result);
2141 num_knots_result = ts_bspline_num_knots(result);
2147 N = ts_deboornet_num_insertions(net) + 1;
2159 memmove(ctrlp_result, ctrlp_spline, (
k-deg) * sof_ctrlp);
2160 from = (
tsReal *) ctrlp_spline +
dim*(
k-deg+N);
2162 to = ctrlp_result +
dim*(
k-deg+N+n);
2163 memmove(to, from, (num_ctrlp_result-n-(
k-deg+N)) * sof_ctrlp);
2165 memmove(knots_result, knots_spline, (
k+1) * sof_real);
2166 from = (
tsReal *) knots_spline +
k+1;
2168 to = knots_result +
k+1+n;
2169 memmove(to, from, (num_knots_result-n-(
k+1)) * sof_real);
2177 from = ts_int_deboornet_access_points(net);
2178 to = ctrlp_result + (
k-deg)*
dim;
2179 stride = (int)(N*
dim);
2182 for (i = 0; i < n; i++) {
2183 memcpy(to, from, sof_ctrlp);
2188 memcpy(to, from, (N-n) * sof_ctrlp);
2198 stride = -(int)(N-n+1) * (int)
dim;
2200 for (i = 0; i < n; i++) {
2201 memcpy(to, from, sof_ctrlp);
2207 to = knots_result +
k+1;
2208 for (i = 0; i < n; i++) {
2212 TS_RETURN_SUCCESS(status)
2225 INIT_OUT_BSPLINE(spline, result)
2226 ts_int_deboornet_init(&net);
2227 TS_TRY(
try, err, status)
2228 TS_CALL(
try, err, ts_bspline_eval(
2229 spline, knot, &net, status))
2230 TS_CALL(
try, err, ts_int_bspline_insert_knot(
2231 spline, &net, num, result, status))
2232 ts_deboornet_free(&net);
2233 TS_CALL(
try, err, ts_bspline_eval(
2234 result, knot, &net, status))
2235 *
k = ts_deboornet_index(&net);
2239 ts_deboornet_free(&net);
2240 TS_END_TRY_RETURN(err)
2252 INIT_OUT_BSPLINE(spline, split)
2253 ts_int_deboornet_init(&net);
2254 TS_TRY(
try, err, status)
2255 TS_CALL(
try, err, ts_bspline_eval(
2256 spline, knot, &net, status))
2257 if (ts_deboornet_multiplicity(&net)
2258 == ts_bspline_order(spline)) {
2259 TS_CALL(
try, err, ts_bspline_copy(
2260 spline, split, status))
2261 *
k = ts_deboornet_index(&net);
2263 TS_CALL(
try, err, ts_int_bspline_insert_knot(
2265 ts_deboornet_num_insertions(&net) + 1,
2267 *
k = ts_deboornet_index(&net) +
2268 ts_deboornet_num_insertions(&net) + 1;
2273 ts_deboornet_free(&net);
2274 TS_END_TRY_RETURN(err)
2283 const size_t dim = ts_bspline_dimension(spline);
2284 const size_t N = ts_bspline_num_control_points(spline);
2285 const tsReal* p0 = ts_int_bspline_access_ctrlp(spline);
2294 TS_CALL_ROE(err, ts_bspline_copy(spline, out, status))
2295 ctrlp = ts_int_bspline_access_ctrlp(out);
2300 for (i = 0; i < N; i++) {
2301 for (d = 0; d <
dim; d++) {
2302 vec = ((
tsReal)i / (N-1)) * (pn_1[d] - p0[d]);
2303 ctrlp[i*
dim + d] = beta * ctrlp[i*
dim + d] +
2307 TS_RETURN_SUCCESS(status)
2315 const size_t deg = ts_bspline_degree(spline);
2316 const size_t order = ts_bspline_order(spline);
2329 INIT_OUT_BSPLINE(spline, beziers)
2330 TS_CALL_ROE(err, ts_bspline_copy(spline, &tmp, status))
2331 knots = ts_int_bspline_access_knots(&tmp);
2332 num_knots = ts_bspline_num_knots(&tmp);
2334 TS_TRY(
try, err, status)
2340 if (!ts_knots_equal(knots[0], u_min)) {
2341 TS_CALL(
try, err, ts_bspline_split(
2342 &tmp, u_min, &tmp, &
k, status))
2343 resize = (int)(-1*deg + (deg*2 -
k));
2344 TS_CALL(
try, err, ts_int_bspline_resize(
2345 &tmp, resize, 0, &tmp, status))
2346 knots = ts_int_bspline_access_knots(&tmp);
2347 num_knots = ts_bspline_num_knots(&tmp);
2351 u_max = knots[num_knots - order];
2352 if (!ts_knots_equal(knots[num_knots - 1], u_max)) {
2353 TS_CALL(
try, err, ts_bspline_split(
2354 &tmp, u_max, &tmp, &
k, status))
2355 num_knots = ts_bspline_num_knots(&tmp);
2356 resize = (int)(-1*deg + (
k - (num_knots - order)));
2357 TS_CALL(
try, err, ts_int_bspline_resize(
2358 &tmp, resize, 1, &tmp, status))
2359 knots = ts_int_bspline_access_knots(&tmp);
2360 num_knots = ts_bspline_num_knots(&tmp);
2365 while (
k < num_knots - order) {
2366 TS_CALL(
try, err, ts_bspline_split(
2367 &tmp, knots[
k], &tmp, &
k, status))
2368 knots = ts_int_bspline_access_knots(&tmp);
2369 num_knots = ts_bspline_num_knots(&tmp);
2373 if (spline == beziers)
2374 ts_bspline_free(beziers);
2375 ts_bspline_move(&tmp, beziers);
2377 ts_bspline_free(&tmp);
2378 TS_END_TRY_RETURN(err)
2391 size_t num_beziers, i, a, c, d, offset, idx;
2392 tsReal f, f_hat, *first, *last;
2397 return ts_bspline_copy(spline, elevated, status);
2401 INIT_OUT_BSPLINE(spline, elevated);
2402 worker = ts_bspline_init();
2403 TS_TRY(
try, err, status)
2407 TS_CALL(
try, err, ts_bspline_to_beziers(
2408 spline, &worker, status));
2409 num_beziers = ts_bspline_num_control_points(&worker) /
2410 ts_bspline_order(&worker);
2411 TS_CALL(
try, err, ts_int_bspline_resize(
2416 &worker, (
int) ((num_beziers+1) * amount), 1, &worker,
2418 dim = ts_bspline_dimension(&worker);
2419 order = ts_bspline_order(&worker);
2420 ctrlp = ts_int_bspline_access_ctrlp(&worker);
2421 knots = ts_int_bspline_access_knots(&worker);
2428 for (i = num_beziers - 1; i > 0; i--) {
2433 offset = i * order *
dim;
2438 memmove(ctrlp + offset + (i * amount *
dim),
2448 for (i = num_beziers; i > 0; i--) {
2460 memmove(knots + offset + (i * amount),
2468 for (a = 0; a < amount; a++) {
2470 for (i = 0; i < num_beziers; i++) {
2498 offset = (i * order + i * (amount - a)) *
dim;
2501 memmove(ctrlp + offset + ((order) *
dim),
2502 ctrlp + offset + ((order-1) *
dim),
2508 for (c = order - 1; c > 0; c--) {
2511 idx = offset + c *
dim;
2514 for (d = 0; d <
dim; d++) {
2521 f * ctrlp[idx -
dim] +
2562 offset = i * order + i * (amount - a);
2564 knots[offset + order] = knots[offset];
2571 offset = num_beziers * order +
2572 num_beziers * (amount - a);
2573 knots[offset + order] = knots[offset];
2581 for (i = 0; i < num_beziers - 1; i++) {
2590 if (ts_distance(last, first,
dim) <= epsilon) {
2592 memmove(last, first, (num_beziers - 1 - i) *
2602 last = knots + i * order - d + (2 * order - 1);
2604 memmove(last, first, (num_beziers - 1 - i) *
2615 worker.
pImpl->
n_ctrlp = ts_bspline_num_knots(&worker) - order;
2616 memmove(ts_int_bspline_access_knots(&worker),
2617 knots, ts_bspline_sof_knots(&worker));
2619 ts_int_bspline_sof_state(&worker));
2620 if (worker.
pImpl == NULL) {
2626 if (spline == elevated)
2627 ts_bspline_free(elevated);
2628 ts_bspline_move(&worker, elevated);
2630 ts_bspline_free(&worker);
2631 TS_END_TRY_RETURN(err)
2642 tsBSpline s1_worker, s2_worker, *smaller, *larger;
2644 size_t i, missing, remaining;
2645 tsReal min, max, shift, nextKnot;
2648 INIT_OUT_BSPLINE(s1, s1_out)
2649 INIT_OUT_BSPLINE(s2, s2_out)
2650 s1_worker = ts_bspline_init();
2651 s2_worker = ts_bspline_init();
2652 smaller = larger = NULL;
2653 TS_TRY(
try, err, status)
2657 if (ts_bspline_degree(s1) < ts_bspline_degree(s2)) {
2658 TS_CALL(
try, err, ts_bspline_elevate_degree(s1,
2659 ts_bspline_degree(s2) - ts_bspline_degree(s1),
2660 epsilon, &s1_worker, status))
2661 TS_CALL(
try, err, ts_bspline_copy(
2662 s2, &s2_worker, status))
2663 }
else if (ts_bspline_degree(s2) < ts_bspline_degree(s1)) {
2664 TS_CALL(
try, err, ts_bspline_elevate_degree(s2,
2665 ts_bspline_degree(s1) - ts_bspline_degree(s2),
2666 epsilon, &s2_worker, status))
2667 TS_CALL(
try, err, ts_bspline_copy(
2668 s1, &s1_worker, status))
2670 TS_CALL(
try, err, ts_bspline_copy(
2671 s1, &s1_worker, status))
2672 TS_CALL(
try, err, ts_bspline_copy(
2673 s2, &s2_worker, status))
2677 if (ts_bspline_num_knots(&s1_worker) <
2678 ts_bspline_num_knots(&s2_worker)) {
2679 smaller = &s1_worker;
2680 larger = &s2_worker;
2682 smaller = &s2_worker;
2683 larger = &s1_worker;
2685 TS_CALL(
try, err, ts_int_deboornet_new(
2686 smaller, &net, status))
2691 ts_bspline_domain(smaller, &min, &max);
2692 missing = remaining = ts_bspline_num_knots(larger) -
2693 ts_bspline_num_knots(smaller);
2697 for (i = 0; remaining > 0; i++, remaining--) {
2698 nextKnot = (max - min) * ((
tsReal)i / missing) + min;
2700 TS_CALL(
try, err, ts_int_bspline_eval_woa(
2701 smaller, nextKnot, &net, status))
2702 while (!ts_deboornet_num_insertions(&net)) {
2705 if (nextKnot > max) {
2706 TS_THROW_0(
try, err, status,
2708 "no more knots for insertion")
2710 TS_CALL(
try, err, ts_int_bspline_eval_woa(
2711 smaller, nextKnot, &net, status))
2713 TS_CALL(
try, err, ts_int_bspline_insert_knot(
2714 smaller, &net, 1, smaller, status))
2718 ts_bspline_free(s1_out);
2720 ts_bspline_free(s2_out);
2721 ts_bspline_move(&s1_worker, s1_out);
2725 if (s1_out != s2_out)
2726 ts_bspline_move(&s2_worker, s2_out);
2728 ts_bspline_free(&s1_worker);
2729 ts_bspline_free(&s2_worker);
2730 ts_deboornet_free(&net);
2731 TS_END_TRY_RETURN(err)
2743 tsReal *origin_al_c, *origin_al_k;
2744 tsReal *target_al_c, *target_al_k;
2747 size_t deg,
dim, num_ctrlp, num_knots;
2752 size_t i, offset, d;
2755 origin_al = ts_bspline_init();
2756 target_al = ts_bspline_init();
2757 TS_TRY(
try, err, status)
2761 t_hat = (
tsReal) 1.0 - t;
2765 if (ts_bspline_degree(origin) != ts_bspline_degree(target) ||
2767 ts_bspline_num_knots(origin) != ts_bspline_num_knots(target)) {
2768 TS_CALL(
try, err, ts_bspline_align(
2769 origin, target, epsilon, &origin_al, &target_al,
2773 origin_al = *origin;
2774 target_al = *target;
2776 origin_al_c = ts_int_bspline_access_ctrlp(&origin_al);
2777 origin_al_k = ts_int_bspline_access_knots(&origin_al);
2778 target_al_c = ts_int_bspline_access_ctrlp(&target_al);
2779 target_al_k = ts_int_bspline_access_knots(&target_al);
2782 deg = ts_bspline_degree(&origin_al);
2783 num_ctrlp = ts_bspline_num_control_points(&origin_al);
2784 dim = ts_bspline_dimension(&origin_al);
2785 if (ts_bspline_dimension(&target_al) <
dim)
2786 dim = ts_bspline_dimension(&target_al);
2787 if (out->
pImpl == NULL) {
2788 TS_CALL(
try, err, ts_bspline_new(num_ctrlp,
dim, deg,
2790 }
else if (ts_bspline_degree(out) != deg ||
2791 ts_bspline_num_control_points(out) != num_ctrlp ||
2792 ts_bspline_dimension(out) !=
dim) {
2793 TS_CALL(
try, err, ts_bspline_new(num_ctrlp,
dim, deg,
2795 ts_bspline_free(out);
2796 ts_bspline_move(&tmp, out);
2798 num_knots = ts_bspline_num_knots(out);
2799 ctrlp = ts_int_bspline_access_ctrlp(out);
2800 knots = ts_int_bspline_access_knots(out);
2803 for (i = 0; i < num_ctrlp; i++) {
2804 for (d = 0; d <
dim; d++) {
2805 offset = i *
dim + d;
2806 ctrlp[offset] = t * target_al_c[offset] +
2807 t_hat * origin_al_c[offset];
2812 for (i = 0; i < num_knots; i++) {
2813 knots[i] = t * target_al_k[i] +
2814 t_hat * origin_al_k[i];
2818 ts_bspline_free(&origin_al);
2820 ts_bspline_free(&target_al);
2821 TS_END_TRY_RETURN(err)
2826#if !defined(TINYSPLINE_NO_SERIALIZATION)
2832ts_int_bspline_to_json(
const tsBSpline *spline,
2836 const size_t deg = ts_bspline_degree(spline);
2837 const size_t dim = ts_bspline_dimension(spline);
2838 const size_t len_ctrlp = ts_bspline_len_control_points(spline);
2839 const size_t len_knots = ts_bspline_num_knots(spline);
2840 const tsReal *ctrlp = ts_int_bspline_access_ctrlp(spline);
2841 const tsReal *knots = ts_int_bspline_access_knots(spline);
2846 JSON_Value *ctrlp_value;
2847 JSON_Value *knots_value;
2848 JSON_Object *spline_object;
2849 JSON_Array *ctrlp_array;
2850 JSON_Array *knots_array;
2852 *value = ctrlp_value = knots_value = NULL;
2853 TS_TRY(values, err, status)
2855 *value = json_value_init_object();
2857 TS_THROW_0(values, err, status,
TS_MALLOC,
2860 ctrlp_value = json_value_init_array();
2862 TS_THROW_0(values, err, status,
TS_MALLOC,
2865 knots_value = json_value_init_array();
2867 TS_THROW_0(values, err, status,
TS_MALLOC,
2876 spline_object = json_value_get_object(*value);
2877 if (!spline_object) {
2878 TS_THROW_0(values, err, status,
TS_MALLOC,
2883 if (JSONSuccess != json_object_set_number(spline_object,
2886 TS_THROW_0(values, err, status,
TS_MALLOC,
2889 if (JSONSuccess != json_object_set_number(spline_object,
2892 TS_THROW_0(values, err, status,
TS_MALLOC,
2897 if (JSONSuccess != json_object_set_value(spline_object,
2900 TS_THROW_0(values, err, status,
TS_MALLOC,
2903 ctrlp_array = json_array(ctrlp_value);
2905 TS_THROW_0(values, err, status,
TS_MALLOC,
2908 for (i = 0; i < len_ctrlp; i++) {
2909 if (JSONSuccess != json_array_append_number(
2910 ctrlp_array, (
double) ctrlp[i])) {
2911 TS_THROW_0(values, err, status,
TS_MALLOC,
2917 if (JSONSuccess != json_object_set_value(spline_object,
2920 TS_THROW_0(values, err, status,
TS_MALLOC,
2923 knots_array = json_array(knots_value);
2925 TS_THROW_0(values, err, status,
TS_MALLOC,
2928 for (i = 0; i < len_knots; i++) {
2929 if (JSONSuccess != json_array_append_number(
2930 knots_array, (
double) knots[i])) {
2931 TS_THROW_0(values, err, status,
TS_MALLOC,
2937 json_value_free(*value);
2938 if (ctrlp_value && !json_value_get_parent(ctrlp_value))
2939 json_value_free(ctrlp_value);
2940 if (knots_value && !json_value_get_parent(knots_value))
2941 json_value_free(knots_value);
2943 TS_END_TRY_RETURN(err)
2947ts_int_bspline_parse_json(
const JSON_Value *spline_value,
2951 size_t deg,
dim, len_ctrlp, num_knots;
2954 JSON_Object *spline_object;
2955 JSON_Value *deg_value;
2956 JSON_Value *dim_value;
2957 JSON_Value *ctrlp_value;
2958 JSON_Array *ctrlp_array;
2959 JSON_Value *knots_value;
2960 JSON_Array *knots_array;
2961 JSON_Value *real_value;
2965 ts_int_bspline_init(spline);
2968 if (json_value_get_type(spline_value) != JSONObject)
2970 spline_object = json_value_get_object(spline_value);
2975 deg_value = json_object_get_value(spline_object, "degree");
2976 if (json_value_get_type(deg_value) != JSONNumber)
2978 if (json_value_get_number(deg_value) < -0.01f) {
2981 json_value_get_number(deg_value))
2983 deg = (size_t) json_value_get_number(deg_value);
2986 dim_value = json_object_get_value(spline_object,
"dimension");
2987 if (json_value_get_type(dim_value) != JSONNumber) {
2989 "dimension is not a number")
2991 if (json_value_get_number(dim_value) < 0.99f) {
2993 "dimension (%f) < 1",
2994 json_value_get_number(deg_value))
2996 dim = (size_t) json_value_get_number(dim_value);
2999 ctrlp_value = json_object_get_value(spline_object,
"control_points");
3000 if (json_value_get_type(ctrlp_value) != JSONArray) {
3002 "control_points is not an array")
3004 ctrlp_array = json_value_get_array(ctrlp_value);
3005 len_ctrlp = json_array_get_count(ctrlp_array);
3006 if (len_ctrlp %
dim != 0) {
3008 "len(control_points) (%lu) %% dimension (%lu) != 0",
3009 (
unsigned long) len_ctrlp, (
unsigned long)
dim)
3013 knots_value = json_object_get_value(spline_object,
"knots");
3014 if (json_value_get_type(knots_value) != JSONArray) {
3016 "knots is not an array")
3018 knots_array = json_value_get_array(knots_value);
3019 num_knots = json_array_get_count(knots_array);
3022 TS_TRY(
try, err, status)
3023 TS_CALL(
try, err, ts_bspline_new(
3026 if (num_knots != ts_bspline_num_knots(spline))
3028 "unexpected num(knots): (%lu) != (%lu)",
3029 (
unsigned long) num_knots,
3030 (
unsigned long) ts_bspline_num_knots(spline))
3033 ctrlp = ts_int_bspline_access_ctrlp(spline);
3034 for (i = 0; i < len_ctrlp; i++) {
3035 real_value = json_array_get_value(ctrlp_array, i);
3036 if (json_value_get_type(real_value) != JSONNumber)
3038 "control_points: value at index %lu is not a number",
3040 ctrlp[i] = (
tsReal) json_value_get_number(real_value);
3042 TS_CALL(
try, err, ts_bspline_set_control_points(
3043 spline, ctrlp, status))
3046 knots = ts_int_bspline_access_knots(spline);
3047 for (i = 0; i < num_knots; i++) {
3048 real_value = json_array_get_value(knots_array, i);
3049 if (json_value_get_type(real_value) != JSONNumber)
3051 "knots: value at index %lu is not a number",
3053 knots[i] = (
tsReal) json_value_get_number(real_value);
3055 TS_CALL(
try, err, ts_bspline_set_knots(
3056 spline, knots, status))
3058 ts_bspline_free(spline);
3059 TS_END_TRY_RETURN(err)
3068 JSON_Value *value = NULL;
3070 TS_CALL_ROE(err, ts_int_bspline_to_json(spline, &value, status))
3071 *json = json_serialize_to_string_pretty(value);
3072 json_value_free(value);
3074 TS_RETURN_0(status,
TS_MALLOC,
"out of memory")
3075 TS_RETURN_SUCCESS(status)
3079ts_bspline_parse_json(
const char *json,
3084 JSON_Value *value = NULL;
3085 ts_int_bspline_init(spline);
3086 TS_TRY(
try, err, status)
3087 value = json_parse_string(json);
3090 "invalid json input")
3092 TS_CALL(
try, err, ts_int_bspline_parse_json(
3093 value, spline, status))
3096 json_value_free(value);
3097 TS_END_TRY_RETURN(err)
3106 JSON_Status json_status;
3107 JSON_Value *value = NULL;
3108 TS_CALL_ROE(err, ts_int_bspline_to_json(spline, &value, status))
3109 json_status = json_serialize_to_file_pretty(value, path);
3110 json_value_free(value);
3111 if (json_status != JSONSuccess)
3112 TS_RETURN_0(status,
TS_IO_ERROR,
"unexpected io error")
3113 TS_RETURN_SUCCESS(status)
3117ts_bspline_load(
const char *path,
3123 JSON_Value *value = NULL;
3124 ts_int_bspline_init(spline);
3125 TS_TRY(
try, err, status)
3126 file = fopen(path,
"r");
3129 "unable to open file")
3131 value = json_parse_file(path);
3134 "invalid json input")
3136 TS_CALL(
try, err, ts_int_bspline_parse_json(
3137 value, spline, status))
3139 if (file) fclose(file);
3140 if (value) json_value_free(value);
3142 ts_bspline_free(spline);
3143 TS_END_TRY_RETURN(err)
3190 const size_t n =
dim > 2 ? 2 :
dim;
3191 memmove(out, x, n *
sizeof(
tsReal));
3201 const size_t n =
dim > 3 ? 3 :
dim;
3202 memmove(out, x, n *
sizeof(
tsReal));
3212 const size_t n =
dim > 4 ? 4 :
dim;
3213 memmove(out, x, n *
sizeof(
tsReal));
3225 for (i = 0; i <
dim; i++)
3226 out[i] = x[i] + y[i];
3241 for (i = 0; i <
dim; i++)
3242 out[i] = x[i] - y[i];
3252 for (i = 0; i <
dim; i++)
3263 const tsReal *x_norm, *y_norm;
3265 ts_vec_norm(x,
dim, buf);
3266 ts_vec_norm(y,
dim, buf +
dim);
3275 acos(ts_vec_dot(x_norm,
3288 a = x[1] * y[2] - x[2] * y[1];
3289 b = x[2] * y[0] - x[0] * y[2];
3290 c = x[0] * y[1] - x[1] * y[0];
3307 for (i = 0; i <
dim; i++)
3317 for (i = 0; i <
dim; i++)
3318 sum += (x[i] * x[i]);
3319 return (
tsReal) sqrt(sum);
3329 for (i = 0; i <
dim; i++)
3330 out[i] = x[i] * val;
3341ts_chord_lengths_length_to_knot(
const tsReal *knots,
3348 tsReal numer, denom, r, r_hat;
3349 size_t idx, low, high;
3353 TS_RETURN_0(status,
TS_NO_RESULT,
"empty chord lengths")
3357 TS_RETURN_SUCCESS(status)
3361 TS_RETURN_SUCCESS(status)
3363 if (len <= lengths[0]) {
3365 TS_RETURN_SUCCESS(status)
3367 if (len >= lengths[num - 1]) {
3368 *knot = knots[num - 1];
3369 TS_RETURN_SUCCESS(status)
3383 idx = (low+high) / 2;
3384 while (len < lengths[idx] || len >= lengths[(idx) + 1]) {
3385 if (len < lengths[idx]) high = idx;
3387 idx = (low+high) / 2;
3391 denom = lengths[(idx) + 1] - lengths[idx];
3394 TS_RETURN_SUCCESS(status)
3396 numer = len - lengths[idx];
3398 r_hat = (
tsReal) 1.0 - r;
3399 *knot = r * knots[(idx) + 1] + r_hat * knots[idx];
3400 TS_RETURN_SUCCESS(status)
3404ts_chord_lengths_t_to_knot(
const tsReal *knots,
3413 tsReal len = num == 0 ? 0 : t * lengths[num - 1];
3414 return ts_chord_lengths_length_to_knot(knots,
3423ts_chord_lengths_equidistant_knot_seq(
const tsReal *knots,
3426 size_t num_knot_seq,
3433 if (num_knot_seq == 0) TS_RETURN_SUCCESS(status)
3434 TS_TRY(
try, err, status)
3435 for (i = 0; i < num_knot_seq; i++) {
3436 t = (
tsReal) i / (num_knot_seq - 1);
3437 TS_CALL(
try, err, ts_chord_lengths_t_to_knot(
3438 knots, lengths, num, t, &knot, status))
3444 knot_seq[num_knot_seq - 1] = knots[num - 1];
3445 knot_seq[0] = knots[0];
3446 TS_END_TRY_RETURN(err)
3467 for (i = 0; i < num; i++)
3477 for (i = 0; i <
dim; i++)
3478 sum += (x[i] - y[i]) * (x[i] - y[i]);
3479 return (
tsReal) sqrt(sum);
OSSIA_INLINE constexpr auto min(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
min function tailored for values
Definition math.hpp:125
OSSIA_INLINE constexpr auto max(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
max function tailored for values
Definition math.hpp:96
Definition tinyspline.h:718
struct tsBSplineImpl * pImpl
Definition tinyspline.h:719
Definition tinyspline.c:45
size_t deg
Definition tinyspline.c:46
size_t n_knots
Definition tinyspline.c:49
size_t dim
Definition tinyspline.c:47
size_t n_ctrlp
Definition tinyspline.c:48
Definition tinyspline.h:1308
struct tsDeBoorNetImpl * pImpl
Definition tinyspline.h:1309
Definition tinyspline.c:57
tsReal u
Definition tinyspline.c:58
size_t s
Definition tinyspline.c:60
size_t k
Definition tinyspline.c:59
size_t h
Definition tinyspline.c:61
size_t dim
Definition tinyspline.c:62
Definition tinyspline.h:665
Definition tinyspline.h:477
#define TS_MAX_NUM_KNOTS
Definition tinyspline.h:121
#define TS_PI
Definition tinyspline.h:108
#define TS_LENGTH_ZERO
Definition tinyspline.h:179
tsError
Definition tinyspline.h:426
@ TS_DEG_GE_NCTRLP
Definition tinyspline.h:437
@ TS_NO_RESULT
Definition tinyspline.h:467
@ TS_MULTIPLICITY
Definition tinyspline.h:443
@ TS_UNDERIVABLE
Definition tinyspline.h:452
@ TS_KNOTS_DECR
Definition tinyspline.h:446
@ TS_IO_ERROR
Definition tinyspline.h:458
@ TS_DIM_ZERO
Definition tinyspline.h:434
@ TS_NUM_KNOTS
Definition tinyspline.h:449
@ TS_NUM_POINTS
Definition tinyspline.h:470
@ TS_U_UNDEFINED
Definition tinyspline.h:440
@ TS_PARSE_ERROR
Definition tinyspline.h:461
@ TS_MALLOC
Definition tinyspline.h:431
@ TS_INDEX_ERROR
Definition tinyspline.h:464
#define TS_DOMAIN_DEFAULT_MAX
Definition tinyspline.h:135
#define TS_KNOT_EPSILON
Definition tinyspline.h:156
#define TS_DOMAIN_DEFAULT_MIN
Definition tinyspline.h:128
double tsReal
Definition tinyspline.h:213
tsBSplineType
Definition tinyspline.h:642
@ TS_CLAMPED
Definition tinyspline.h:647
@ TS_BEZIERS
Definition tinyspline.h:653
@ TS_OPENED
Definition tinyspline.h:644