2 * Copyright 2011 INRIA Saclay
3 * Copyright 2011 Universiteit Leiden
5 * Use of this software is governed by the GNU LGPLv2.1 license
7 * Written by Sven Verdoolaege, INRIA Saclay - Ile-de-France,
8 * Parc Club Orsay Universite, ZAC des vignes, 4 rue Jacques Monod,
10 * and Leiden Institute of Advanced Computer Science,
11 * Universiteit Leiden, Niels Bohrweg 1, 2333 CA Leiden, The Netherlands
14 #include <isl_map_private.h>
15 #include <isl_aff_private.h>
16 #include <isl_dim_private.h>
17 #include <isl_local_space_private.h>
18 #include <isl_mat_private.h>
19 #include <isl/constraint.h>
23 __isl_give isl_aff
*isl_aff_alloc_vec(__isl_take isl_local_space
*ls
,
24 __isl_take isl_vec
*v
)
31 aff
= isl_calloc_type(v
->ctx
, struct isl_aff
);
41 isl_local_space_free(ls
);
46 __isl_give isl_aff
*isl_aff_alloc(__isl_take isl_local_space
*ls
)
55 ctx
= isl_local_space_get_ctx(ls
);
56 if (!isl_local_space_divs_known(ls
))
57 isl_die(ctx
, isl_error_invalid
, "local space has unknown divs",
60 total
= isl_local_space_dim(ls
, isl_dim_all
);
61 v
= isl_vec_alloc(ctx
, 1 + 1 + total
);
62 return isl_aff_alloc_vec(ls
, v
);
64 isl_local_space_free(ls
);
68 __isl_give isl_aff
*isl_aff_zero(__isl_take isl_local_space
*ls
)
72 aff
= isl_aff_alloc(ls
);
76 isl_int_set_si(aff
->v
->el
[0], 1);
77 isl_seq_clr(aff
->v
->el
+ 1, aff
->v
->size
- 1);
82 __isl_give isl_aff
*isl_aff_copy(__isl_keep isl_aff
*aff
)
91 __isl_give isl_aff
*isl_aff_dup(__isl_keep isl_aff
*aff
)
96 return isl_aff_alloc_vec(isl_local_space_copy(aff
->ls
),
97 isl_vec_copy(aff
->v
));
100 __isl_give isl_aff
*isl_aff_cow(__isl_take isl_aff
*aff
)
108 return isl_aff_dup(aff
);
111 void *isl_aff_free(__isl_take isl_aff
*aff
)
119 isl_local_space_free(aff
->ls
);
120 isl_vec_free(aff
->v
);
127 isl_ctx
*isl_aff_get_ctx(__isl_keep isl_aff
*aff
)
129 return aff
? isl_local_space_get_ctx(aff
->ls
) : NULL
;
132 int isl_aff_dim(__isl_keep isl_aff
*aff
, enum isl_dim_type type
)
134 return aff
? isl_local_space_dim(aff
->ls
, type
) : 0;
137 __isl_give isl_dim
*isl_aff_get_dim(__isl_keep isl_aff
*aff
)
139 return aff
? isl_local_space_get_dim(aff
->ls
) : NULL
;
142 __isl_give isl_local_space
*isl_aff_get_local_space(__isl_keep isl_aff
*aff
)
144 return aff
? isl_local_space_copy(aff
->ls
) : NULL
;
147 const char *isl_aff_get_dim_name(__isl_keep isl_aff
*aff
,
148 enum isl_dim_type type
, unsigned pos
)
150 return aff
? isl_local_space_get_dim_name(aff
->ls
, type
, pos
) : 0;
153 __isl_give isl_aff
*isl_aff_reset_dim(__isl_take isl_aff
*aff
,
154 __isl_take isl_dim
*dim
)
156 aff
= isl_aff_cow(aff
);
160 aff
->ls
= isl_local_space_reset_dim(aff
->ls
, dim
);
162 return isl_aff_free(aff
);
171 /* Reorder the coefficients of the affine expression based
172 * on the given reodering.
173 * The reordering r is assumed to have been extended with the local
176 static __isl_give isl_vec
*vec_reorder(__isl_take isl_vec
*vec
,
177 __isl_take isl_reordering
*r
, int n_div
)
185 res
= isl_vec_alloc(vec
->ctx
, 2 + isl_dim_total(r
->dim
) + n_div
);
186 isl_seq_cpy(res
->el
, vec
->el
, 2);
187 isl_seq_clr(res
->el
+ 2, res
->size
- 2);
188 for (i
= 0; i
< r
->len
; ++i
)
189 isl_int_set(res
->el
[2 + r
->pos
[i
]], vec
->el
[2 + i
]);
191 isl_reordering_free(r
);
196 isl_reordering_free(r
);
200 /* Reorder the dimensions of "aff" according to the given reordering.
202 __isl_give isl_aff
*isl_aff_realign(__isl_take isl_aff
*aff
,
203 __isl_take isl_reordering
*r
)
205 aff
= isl_aff_cow(aff
);
209 r
= isl_reordering_extend(r
, aff
->ls
->div
->n_row
);
210 aff
->v
= vec_reorder(aff
->v
, isl_reordering_copy(r
),
211 aff
->ls
->div
->n_row
);
212 aff
->ls
= isl_local_space_realign(aff
->ls
, r
);
214 if (!aff
->v
|| !aff
->ls
)
215 return isl_aff_free(aff
);
220 isl_reordering_free(r
);
224 int isl_aff_plain_is_zero(__isl_keep isl_aff
*aff
)
229 return isl_seq_first_non_zero(aff
->v
->el
+ 1, aff
->v
->size
- 1) < 0;
232 int isl_aff_plain_is_equal(__isl_keep isl_aff
*aff1
, __isl_keep isl_aff
*aff2
)
239 equal
= isl_local_space_is_equal(aff1
->ls
, aff2
->ls
);
240 if (equal
< 0 || !equal
)
243 return isl_vec_is_equal(aff1
->v
, aff2
->v
);
246 int isl_aff_get_denominator(__isl_keep isl_aff
*aff
, isl_int
*v
)
250 isl_int_set(*v
, aff
->v
->el
[0]);
254 int isl_aff_get_constant(__isl_keep isl_aff
*aff
, isl_int
*v
)
258 isl_int_set(*v
, aff
->v
->el
[1]);
262 int isl_aff_get_coefficient(__isl_keep isl_aff
*aff
,
263 enum isl_dim_type type
, int pos
, isl_int
*v
)
268 if (pos
>= isl_local_space_dim(aff
->ls
, type
))
269 isl_die(aff
->v
->ctx
, isl_error_invalid
,
270 "position out of bounds", return -1);
272 pos
+= isl_local_space_offset(aff
->ls
, type
);
273 isl_int_set(*v
, aff
->v
->el
[1 + pos
]);
278 __isl_give isl_aff
*isl_aff_set_denominator(__isl_take isl_aff
*aff
, isl_int v
)
280 aff
= isl_aff_cow(aff
);
284 aff
->v
= isl_vec_cow(aff
->v
);
286 return isl_aff_free(aff
);
288 isl_int_set(aff
->v
->el
[0], v
);
293 __isl_give isl_aff
*isl_aff_set_constant(__isl_take isl_aff
*aff
, isl_int v
)
295 aff
= isl_aff_cow(aff
);
299 aff
->v
= isl_vec_cow(aff
->v
);
301 return isl_aff_free(aff
);
303 isl_int_set(aff
->v
->el
[1], v
);
308 __isl_give isl_aff
*isl_aff_add_constant(__isl_take isl_aff
*aff
, isl_int v
)
310 if (isl_int_is_zero(v
))
313 aff
= isl_aff_cow(aff
);
317 aff
->v
= isl_vec_cow(aff
->v
);
319 return isl_aff_free(aff
);
321 isl_int_addmul(aff
->v
->el
[1], aff
->v
->el
[0], v
);
326 __isl_give isl_aff
*isl_aff_add_constant_si(__isl_take isl_aff
*aff
, int v
)
331 isl_int_set_si(t
, v
);
332 aff
= isl_aff_add_constant(aff
, t
);
338 __isl_give isl_aff
*isl_aff_set_constant_si(__isl_take isl_aff
*aff
, int v
)
340 aff
= isl_aff_cow(aff
);
344 aff
->v
= isl_vec_cow(aff
->v
);
346 return isl_aff_free(aff
);
348 isl_int_set_si(aff
->v
->el
[1], v
);
353 __isl_give isl_aff
*isl_aff_set_coefficient(__isl_take isl_aff
*aff
,
354 enum isl_dim_type type
, int pos
, isl_int v
)
359 if (pos
>= isl_local_space_dim(aff
->ls
, type
))
360 isl_die(aff
->v
->ctx
, isl_error_invalid
,
361 "position out of bounds", return isl_aff_free(aff
));
363 aff
= isl_aff_cow(aff
);
367 aff
->v
= isl_vec_cow(aff
->v
);
369 return isl_aff_free(aff
);
371 pos
+= isl_local_space_offset(aff
->ls
, type
);
372 isl_int_set(aff
->v
->el
[1 + pos
], v
);
377 __isl_give isl_aff
*isl_aff_set_coefficient_si(__isl_take isl_aff
*aff
,
378 enum isl_dim_type type
, int pos
, int v
)
383 if (pos
>= isl_local_space_dim(aff
->ls
, type
))
384 isl_die(aff
->v
->ctx
, isl_error_invalid
,
385 "position out of bounds", return isl_aff_free(aff
));
387 aff
= isl_aff_cow(aff
);
391 aff
->v
= isl_vec_cow(aff
->v
);
393 return isl_aff_free(aff
);
395 pos
+= isl_local_space_offset(aff
->ls
, type
);
396 isl_int_set_si(aff
->v
->el
[1 + pos
], v
);
401 __isl_give isl_aff
*isl_aff_add_coefficient(__isl_take isl_aff
*aff
,
402 enum isl_dim_type type
, int pos
, isl_int v
)
407 if (pos
>= isl_local_space_dim(aff
->ls
, type
))
408 isl_die(aff
->v
->ctx
, isl_error_invalid
,
409 "position out of bounds", return isl_aff_free(aff
));
411 aff
= isl_aff_cow(aff
);
415 aff
->v
= isl_vec_cow(aff
->v
);
417 return isl_aff_free(aff
);
419 pos
+= isl_local_space_offset(aff
->ls
, type
);
420 isl_int_addmul(aff
->v
->el
[1 + pos
], aff
->v
->el
[0], v
);
425 __isl_give isl_aff
*isl_aff_add_coefficient_si(__isl_take isl_aff
*aff
,
426 enum isl_dim_type type
, int pos
, int v
)
431 isl_int_set_si(t
, v
);
432 aff
= isl_aff_add_coefficient(aff
, type
, pos
, t
);
438 __isl_give isl_div
*isl_aff_get_div(__isl_keep isl_aff
*aff
, int pos
)
443 return isl_local_space_get_div(aff
->ls
, pos
);
446 __isl_give isl_aff
*isl_aff_neg(__isl_take isl_aff
*aff
)
448 aff
= isl_aff_cow(aff
);
451 aff
->v
= isl_vec_cow(aff
->v
);
453 return isl_aff_free(aff
);
455 isl_seq_neg(aff
->v
->el
+ 1, aff
->v
->el
+ 1, aff
->v
->size
- 1);
460 /* Given f, return floor(f).
461 * If f is an integer expression, then just return f.
462 * Otherwise, create a new div d = [f] and return the expression d.
464 __isl_give isl_aff
*isl_aff_floor(__isl_take isl_aff
*aff
)
472 if (isl_int_is_one(aff
->v
->el
[0]))
475 aff
= isl_aff_cow(aff
);
479 aff
->ls
= isl_local_space_add_div(aff
->ls
, isl_vec_copy(aff
->v
));
481 return isl_aff_free(aff
);
483 ctx
= isl_aff_get_ctx(aff
);
485 isl_vec_free(aff
->v
);
486 aff
->v
= isl_vec_alloc(ctx
, size
+ 1);
487 aff
->v
= isl_vec_clr(aff
->v
);
489 return isl_aff_free(aff
);
490 isl_int_set_si(aff
->v
->el
[0], 1);
491 isl_int_set_si(aff
->v
->el
[size
], 1);
496 /* Given f, return ceil(f).
497 * If f is an integer expression, then just return f.
498 * Otherwise, create a new div d = [-f] and return the expression -d.
500 __isl_give isl_aff
*isl_aff_ceil(__isl_take isl_aff
*aff
)
505 if (isl_int_is_one(aff
->v
->el
[0]))
508 aff
= isl_aff_neg(aff
);
509 aff
= isl_aff_floor(aff
);
510 aff
= isl_aff_neg(aff
);
515 /* Apply the expansion computed by isl_merge_divs.
516 * The expansion itself is given by "exp" while the resulting
517 * list of divs is given by "div".
519 __isl_give isl_aff
*isl_aff_expand_divs( __isl_take isl_aff
*aff
,
520 __isl_take isl_mat
*div
, int *exp
)
527 aff
= isl_aff_cow(aff
);
531 old_n_div
= isl_local_space_dim(aff
->ls
, isl_dim_div
);
532 new_n_div
= isl_mat_rows(div
);
533 if (new_n_div
< old_n_div
)
534 isl_die(isl_mat_get_ctx(div
), isl_error_invalid
,
535 "not an expansion", goto error
);
537 aff
->v
= isl_vec_extend(aff
->v
, aff
->v
->size
+ new_n_div
- old_n_div
);
541 offset
= 1 + isl_local_space_offset(aff
->ls
, isl_dim_div
);
543 for (i
= new_n_div
- 1; i
>= 0; --i
) {
544 if (j
>= 0 && exp
[j
] == i
) {
546 isl_int_swap(aff
->v
->el
[offset
+ i
],
547 aff
->v
->el
[offset
+ j
]);
550 isl_int_set_si(aff
->v
->el
[offset
+ i
], 0);
553 aff
->ls
= isl_local_space_replace_divs(aff
->ls
, isl_mat_copy(div
));
564 /* Add two affine expressions that live in the same local space.
566 static __isl_give isl_aff
*add_expanded(__isl_take isl_aff
*aff1
,
567 __isl_take isl_aff
*aff2
)
571 aff1
= isl_aff_cow(aff1
);
575 aff1
->v
= isl_vec_cow(aff1
->v
);
581 isl_int_gcd(gcd
, aff1
->v
->el
[0], aff2
->v
->el
[0]);
582 isl_int_divexact(f
, aff2
->v
->el
[0], gcd
);
583 isl_seq_scale(aff1
->v
->el
+ 1, aff1
->v
->el
+ 1, f
, aff1
->v
->size
- 1);
584 isl_int_divexact(f
, aff1
->v
->el
[0], gcd
);
585 isl_seq_addmul(aff1
->v
->el
+ 1, f
, aff2
->v
->el
+ 1, aff1
->v
->size
- 1);
586 isl_int_divexact(f
, aff2
->v
->el
[0], gcd
);
587 isl_int_mul(aff1
->v
->el
[0], aff1
->v
->el
[0], f
);
599 __isl_give isl_aff
*isl_aff_add(__isl_take isl_aff
*aff1
,
600 __isl_take isl_aff
*aff2
)
610 ctx
= isl_aff_get_ctx(aff1
);
611 if (!isl_dim_equal(aff1
->ls
->dim
, aff2
->ls
->dim
))
612 isl_die(ctx
, isl_error_invalid
,
613 "spaces don't match", goto error
);
615 if (aff1
->ls
->div
->n_row
== 0 && aff2
->ls
->div
->n_row
== 0)
616 return add_expanded(aff1
, aff2
);
618 exp1
= isl_alloc_array(ctx
, int, aff1
->ls
->div
->n_row
);
619 exp2
= isl_alloc_array(ctx
, int, aff2
->ls
->div
->n_row
);
623 div
= isl_merge_divs(aff1
->ls
->div
, aff2
->ls
->div
, exp1
, exp2
);
624 aff1
= isl_aff_expand_divs(aff1
, isl_mat_copy(div
), exp1
);
625 aff2
= isl_aff_expand_divs(aff2
, div
, exp2
);
629 return add_expanded(aff1
, aff2
);
638 __isl_give isl_aff
*isl_aff_sub(__isl_take isl_aff
*aff1
,
639 __isl_take isl_aff
*aff2
)
641 return isl_aff_add(aff1
, isl_aff_neg(aff2
));
644 __isl_give isl_aff
*isl_aff_scale(__isl_take isl_aff
*aff
, isl_int f
)
648 if (isl_int_is_one(f
))
651 aff
= isl_aff_cow(aff
);
654 aff
->v
= isl_vec_cow(aff
->v
);
656 return isl_aff_free(aff
);
659 isl_int_gcd(gcd
, aff
->v
->el
[0], f
);
660 isl_int_divexact(aff
->v
->el
[0], aff
->v
->el
[0], gcd
);
661 isl_int_divexact(gcd
, f
, gcd
);
662 isl_seq_scale(aff
->v
->el
+ 1, aff
->v
->el
+ 1, gcd
, aff
->v
->size
- 1);
668 __isl_give isl_aff
*isl_aff_scale_down(__isl_take isl_aff
*aff
, isl_int f
)
672 if (isl_int_is_one(f
))
675 aff
= isl_aff_cow(aff
);
678 aff
->v
= isl_vec_cow(aff
->v
);
680 return isl_aff_free(aff
);
683 isl_seq_gcd(aff
->v
->el
+ 1, aff
->v
->size
- 1, &gcd
);
684 isl_int_gcd(gcd
, gcd
, f
);
685 isl_seq_scale_down(aff
->v
->el
+ 1, aff
->v
->el
+ 1, gcd
, aff
->v
->size
- 1);
686 isl_int_divexact(gcd
, f
, gcd
);
687 isl_int_mul(aff
->v
->el
[0], aff
->v
->el
[0], gcd
);
693 __isl_give isl_aff
*isl_aff_scale_down_ui(__isl_take isl_aff
*aff
, unsigned f
)
701 isl_int_set_ui(v
, f
);
702 aff
= isl_aff_scale_down(aff
, v
);
708 __isl_give isl_aff
*isl_aff_set_dim_name(__isl_take isl_aff
*aff
,
709 enum isl_dim_type type
, unsigned pos
, const char *s
)
711 aff
= isl_aff_cow(aff
);
714 aff
->ls
= isl_local_space_set_dim_name(aff
->ls
, type
, pos
, s
);
716 return isl_aff_free(aff
);
721 /* Exploit the equalities in "eq" to simplify the affine expression
722 * and the expressions of the integer divisions in the local space.
723 * The integer divisions in this local space are assumed to appear
724 * as regular dimensions in "eq".
726 static __isl_give isl_aff
*isl_aff_substitute_equalities(
727 __isl_take isl_aff
*aff
, __isl_take isl_basic_set
*eq
)
736 isl_basic_set_free(eq
);
740 aff
= isl_aff_cow(aff
);
744 aff
->ls
= isl_local_space_substitute_equalities(aff
->ls
,
745 isl_basic_set_copy(eq
));
749 total
= 1 + isl_dim_total(eq
->dim
);
751 for (i
= 0; i
< eq
->n_eq
; ++i
) {
752 j
= isl_seq_last_non_zero(eq
->eq
[i
], total
+ n_div
);
753 if (j
< 0 || j
== 0 || j
>= total
)
756 isl_seq_elim(aff
->v
->el
+ 1, eq
->eq
[i
], j
, total
,
760 isl_basic_set_free(eq
);
763 isl_basic_set_free(eq
);
768 /* Look for equalities among the variables shared by context and aff
769 * and the integer divisions of aff, if any.
770 * The equalities are then used to eliminate coefficients and/or integer
771 * divisions from aff.
773 __isl_give isl_aff
*isl_aff_gist(__isl_take isl_aff
*aff
,
774 __isl_take isl_set
*context
)
781 n_div
= isl_local_space_dim(aff
->ls
, isl_dim_div
);
784 context
= isl_set_add_dims(context
, isl_dim_set
, n_div
);
785 bset
= isl_basic_set_from_local_space(
786 isl_aff_get_local_space(aff
));
787 bset
= isl_basic_set_lift(bset
);
788 bset
= isl_basic_set_flatten(bset
);
789 context
= isl_set_intersect(context
,
790 isl_set_from_basic_set(bset
));
793 hull
= isl_set_affine_hull(context
);
794 return isl_aff_substitute_equalities(aff
, hull
);
797 isl_set_free(context
);
801 /* Return a basic set containing those elements in the space
802 * of aff where it is non-negative.
804 __isl_give isl_basic_set
*isl_aff_nonneg_basic_set(__isl_take isl_aff
*aff
)
806 isl_constraint
*ineq
;
808 ineq
= isl_inequality_from_aff(aff
);
810 return isl_basic_set_from_constraint(ineq
);
813 /* Return a basic set containing those elements in the shared space
814 * of aff1 and aff2 where aff1 is greater than or equal to aff2.
816 __isl_give isl_basic_set
*isl_aff_ge_basic_set(__isl_take isl_aff
*aff1
,
817 __isl_take isl_aff
*aff2
)
819 aff1
= isl_aff_sub(aff1
, aff2
);
821 return isl_aff_nonneg_basic_set(aff1
);
824 __isl_give isl_aff
*isl_aff_add_on_domain(__isl_keep isl_set
*dom
,
825 __isl_take isl_aff
*aff1
, __isl_take isl_aff
*aff2
)
827 aff1
= isl_aff_add(aff1
, aff2
);
828 aff1
= isl_aff_gist(aff1
, isl_set_copy(dom
));
832 int isl_aff_is_empty(__isl_keep isl_aff
*aff
)
840 /* Set active[i] to 1 if the dimension at position i is involved
841 * in the affine expression.
843 static int set_active(__isl_keep isl_aff
*aff
, int *active
)
852 total
= aff
->v
->size
- 2;
853 for (i
= 0; i
< total
; ++i
)
854 active
[i
] = !isl_int_is_zero(aff
->v
->el
[2 + i
]);
856 offset
= isl_local_space_offset(aff
->ls
, isl_dim_div
) - 1;
857 for (i
= aff
->ls
->div
->n_row
- 1; i
>= 0; --i
) {
858 if (!active
[offset
+ i
])
860 for (j
= 0; j
< total
; ++j
)
862 !isl_int_is_zero(aff
->ls
->div
->row
[i
][2 + j
]);
868 /* Check whether the given affine expression has non-zero coefficient
869 * for any dimension in the given range or if any of these dimensions
870 * appear with non-zero coefficients in any of the integer divisions
871 * involved in the affine expression.
873 int isl_aff_involves_dims(__isl_keep isl_aff
*aff
,
874 enum isl_dim_type type
, unsigned first
, unsigned n
)
886 ctx
= isl_aff_get_ctx(aff
);
887 if (first
+ n
> isl_aff_dim(aff
, type
))
888 isl_die(ctx
, isl_error_invalid
,
889 "range out of bounds", return -1);
891 active
= isl_calloc_array(ctx
, int,
892 isl_local_space_dim(aff
->ls
, isl_dim_all
));
893 if (set_active(aff
, active
) < 0)
896 first
+= isl_local_space_offset(aff
->ls
, type
) - 1;
897 for (i
= 0; i
< n
; ++i
)
898 if (active
[first
+ i
]) {
911 __isl_give isl_aff
*isl_aff_drop_dims(__isl_take isl_aff
*aff
,
912 enum isl_dim_type type
, unsigned first
, unsigned n
)
918 if (n
== 0 && !isl_local_space_is_named_or_nested(aff
->ls
, type
))
921 ctx
= isl_aff_get_ctx(aff
);
922 if (first
+ n
> isl_aff_dim(aff
, type
))
923 isl_die(ctx
, isl_error_invalid
, "range out of bounds",
924 return isl_aff_free(aff
));
926 aff
= isl_aff_cow(aff
);
930 aff
->ls
= isl_local_space_drop_dims(aff
->ls
, type
, first
, n
);
932 return isl_aff_free(aff
);
934 first
+= 1 + isl_local_space_offset(aff
->ls
, type
);
935 aff
->v
= isl_vec_drop_els(aff
->v
, first
, n
);
937 return isl_aff_free(aff
);
942 __isl_give isl_aff
*isl_aff_insert_dims(__isl_take isl_aff
*aff
,
943 enum isl_dim_type type
, unsigned first
, unsigned n
)
949 if (n
== 0 && !isl_local_space_is_named_or_nested(aff
->ls
, type
))
952 ctx
= isl_aff_get_ctx(aff
);
953 if (first
> isl_aff_dim(aff
, type
))
954 isl_die(ctx
, isl_error_invalid
, "position out of bounds",
955 return isl_aff_free(aff
));
957 aff
= isl_aff_cow(aff
);
961 aff
->ls
= isl_local_space_insert_dims(aff
->ls
, type
, first
, n
);
963 return isl_aff_free(aff
);
965 first
+= 1 + isl_local_space_offset(aff
->ls
, type
);
966 aff
->v
= isl_vec_insert_zero_els(aff
->v
, first
, n
);
968 return isl_aff_free(aff
);
973 __isl_give isl_aff
*isl_aff_add_dims(__isl_take isl_aff
*aff
,
974 enum isl_dim_type type
, unsigned n
)
978 pos
= isl_aff_dim(aff
, type
);
980 return isl_aff_insert_dims(aff
, type
, pos
, n
);
983 __isl_give isl_pw_aff
*isl_pw_aff_add_dims(__isl_take isl_pw_aff
*pwaff
,
984 enum isl_dim_type type
, unsigned n
)
988 pos
= isl_pw_aff_dim(pwaff
, type
);
990 return isl_pw_aff_insert_dims(pwaff
, type
, pos
, n
);
994 #define PW isl_pw_aff
998 #define EL_IS_ZERO is_empty
1002 #define IS_ZERO is_empty
1008 #define NO_MOVE_DIMS
1012 #include <isl_pw_templ.c>
1014 /* Compute a piecewise quasi-affine expression with a domain that
1015 * is the union of those of pwaff1 and pwaff2 and such that on each
1016 * cell, the quasi-affine expression is the maximum of those of pwaff1
1017 * and pwaff2. If only one of pwaff1 or pwaff2 is defined on a given
1018 * cell, then the associated expression is the defined one.
1020 __isl_give isl_pw_aff
*isl_pw_aff_max(__isl_take isl_pw_aff
*pwaff1
,
1021 __isl_take isl_pw_aff
*pwaff2
)
1028 if (!pwaff1
|| !pwaff2
)
1031 ctx
= isl_dim_get_ctx(pwaff1
->dim
);
1032 if (!isl_dim_equal(pwaff1
->dim
, pwaff2
->dim
))
1033 isl_die(ctx
, isl_error_invalid
,
1034 "arguments should live in same space", goto error
);
1036 if (isl_pw_aff_is_empty(pwaff1
)) {
1037 isl_pw_aff_free(pwaff1
);
1041 if (isl_pw_aff_is_empty(pwaff2
)) {
1042 isl_pw_aff_free(pwaff2
);
1046 n
= 2 * (pwaff1
->n
+ 1) * (pwaff2
->n
+ 1);
1047 res
= isl_pw_aff_alloc_(isl_dim_copy(pwaff1
->dim
), n
);
1049 for (i
= 0; i
< pwaff1
->n
; ++i
) {
1050 set
= isl_set_copy(pwaff1
->p
[i
].set
);
1051 for (j
= 0; j
< pwaff2
->n
; ++j
) {
1052 struct isl_set
*common
;
1055 common
= isl_set_intersect(
1056 isl_set_copy(pwaff1
->p
[i
].set
),
1057 isl_set_copy(pwaff2
->p
[j
].set
));
1058 ge
= isl_set_from_basic_set(isl_aff_ge_basic_set(
1059 isl_aff_copy(pwaff2
->p
[j
].aff
),
1060 isl_aff_copy(pwaff1
->p
[i
].aff
)));
1061 ge
= isl_set_intersect(common
, ge
);
1062 if (isl_set_plain_is_empty(ge
)) {
1066 set
= isl_set_subtract(set
, isl_set_copy(ge
));
1068 res
= isl_pw_aff_add_piece(res
, ge
,
1069 isl_aff_copy(pwaff2
->p
[j
].aff
));
1071 res
= isl_pw_aff_add_piece(res
, set
,
1072 isl_aff_copy(pwaff1
->p
[i
].aff
));
1075 for (j
= 0; j
< pwaff2
->n
; ++j
) {
1076 set
= isl_set_copy(pwaff2
->p
[j
].set
);
1077 for (i
= 0; i
< pwaff1
->n
; ++i
)
1078 set
= isl_set_subtract(set
,
1079 isl_set_copy(pwaff1
->p
[i
].set
));
1080 res
= isl_pw_aff_add_piece(res
, set
,
1081 isl_aff_copy(pwaff2
->p
[j
].aff
));
1084 isl_pw_aff_free(pwaff1
);
1085 isl_pw_aff_free(pwaff2
);
1089 isl_pw_aff_free(pwaff1
);
1090 isl_pw_aff_free(pwaff2
);
1094 /* Construct a map with as domain the domain of pwaff and
1095 * one-dimensional range corresponding to the affine expressions.
1097 __isl_give isl_map
*isl_map_from_pw_aff(__isl_take isl_pw_aff
*pwaff
)
1106 dim
= isl_pw_aff_get_dim(pwaff
);
1107 dim
= isl_dim_from_domain(dim
);
1108 dim
= isl_dim_add(dim
, isl_dim_out
, 1);
1109 map
= isl_map_empty(dim
);
1111 for (i
= 0; i
< pwaff
->n
; ++i
) {
1112 isl_basic_map
*bmap
;
1115 bmap
= isl_basic_map_from_aff(isl_aff_copy(pwaff
->p
[i
].aff
));
1116 map_i
= isl_map_from_basic_map(bmap
);
1117 map_i
= isl_map_intersect_domain(map_i
,
1118 isl_set_copy(pwaff
->p
[i
].set
));
1119 map
= isl_map_union_disjoint(map
, map_i
);
1122 isl_pw_aff_free(pwaff
);
1127 /* Return a set containing those elements in the domain
1128 * of pwaff where it is non-negative.
1130 __isl_give isl_set
*isl_pw_aff_nonneg_set(__isl_take isl_pw_aff
*pwaff
)
1138 set
= isl_set_empty(isl_pw_aff_get_dim(pwaff
));
1140 for (i
= 0; i
< pwaff
->n
; ++i
) {
1141 isl_basic_set
*bset
;
1144 bset
= isl_aff_nonneg_basic_set(isl_aff_copy(pwaff
->p
[i
].aff
));
1145 set_i
= isl_set_from_basic_set(bset
);
1146 set_i
= isl_set_intersect(set_i
, isl_set_copy(pwaff
->p
[i
].set
));
1147 set
= isl_set_union_disjoint(set
, set_i
);
1150 isl_pw_aff_free(pwaff
);
1155 /* Return a set containing those elements in the shared domain
1156 * of pwaff1 and pwaff2 where pwaff1 is greater than (or equal) to pwaff2.
1158 * We compute the difference on the shared domain and then construct
1159 * the set of values where this difference is non-negative.
1160 * If strict is set, we first subtract 1 from the difference.
1162 static __isl_give isl_set
*pw_aff_gte_set(__isl_take isl_pw_aff
*pwaff1
,
1163 __isl_take isl_pw_aff
*pwaff2
, int strict
)
1165 isl_set
*set1
, *set2
;
1167 set1
= isl_pw_aff_domain(isl_pw_aff_copy(pwaff1
));
1168 set2
= isl_pw_aff_domain(isl_pw_aff_copy(pwaff2
));
1169 set1
= isl_set_intersect(set1
, set2
);
1170 pwaff1
= isl_pw_aff_intersect_domain(pwaff1
, isl_set_copy(set1
));
1171 pwaff2
= isl_pw_aff_intersect_domain(pwaff2
, isl_set_copy(set1
));
1172 pwaff1
= isl_pw_aff_add(pwaff1
, isl_pw_aff_neg(pwaff2
));
1175 isl_dim
*dim
= isl_set_get_dim(set1
);
1177 aff
= isl_aff_zero(isl_local_space_from_dim(dim
));
1178 aff
= isl_aff_add_constant_si(aff
, -1);
1179 pwaff1
= isl_pw_aff_add(pwaff1
, isl_pw_aff_alloc(set1
, aff
));
1183 return isl_pw_aff_nonneg_set(pwaff1
);
1186 /* Return a set containing those elements in the shared domain
1187 * of pwaff1 and pwaff2 where pwaff1 is greater than or equal to pwaff2.
1189 __isl_give isl_set
*isl_pw_aff_ge_set(__isl_take isl_pw_aff
*pwaff1
,
1190 __isl_take isl_pw_aff
*pwaff2
)
1192 return pw_aff_gte_set(pwaff1
, pwaff2
, 0);
1195 /* Return a set containing those elements in the shared domain
1196 * of pwaff1 and pwaff2 where pwaff1 is strictly greater than pwaff2.
1198 __isl_give isl_set
*isl_pw_aff_gt_set(__isl_take isl_pw_aff
*pwaff1
,
1199 __isl_take isl_pw_aff
*pwaff2
)
1201 return pw_aff_gte_set(pwaff1
, pwaff2
, 1);
1204 __isl_give isl_set
*isl_pw_aff_lt_set(__isl_take isl_pw_aff
*pwaff1
,
1205 __isl_take isl_pw_aff
*pwaff2
)
1207 return isl_pw_aff_gt_set(pwaff2
, pwaff1
);
1210 __isl_give isl_pw_aff
*isl_pw_aff_scale_down(__isl_take isl_pw_aff
*pwaff
,
1215 if (isl_int_is_one(v
))
1217 if (!isl_int_is_pos(v
))
1218 isl_die(isl_pw_aff_get_ctx(pwaff
), isl_error_invalid
,
1219 "factor needs to be positive",
1220 return isl_pw_aff_free(pwaff
));
1221 pwaff
= isl_pw_aff_cow(pwaff
);
1227 for (i
= 0; i
< pwaff
->n
; ++i
) {
1228 pwaff
->p
[i
].aff
= isl_aff_scale_down(pwaff
->p
[i
].aff
, v
);
1229 if (!pwaff
->p
[i
].aff
)
1230 return isl_pw_aff_free(pwaff
);