2 * Copyright 2012-2013 Ecole Normale Superieure
4 * Use of this software is governed by the MIT license
6 * Written by Sven Verdoolaege,
7 * Ecole Normale Superieure, 45 rue d'Ulm, 75230 Paris, France
11 #include <isl_map_private.h>
12 #include <isl_aff_private.h>
13 #include <isl/constraint.h>
16 /* Stride information about a specific set dimension.
17 * The values of the set dimension are equal to
18 * "offset" plus a multiple of "stride".
20 struct isl_stride_info
{
25 /* Return the ctx to which "si" belongs.
27 isl_ctx
*isl_stride_info_get_ctx(__isl_keep isl_stride_info
*si
)
32 return isl_val_get_ctx(si
->stride
);
35 /* Free "si" and return NULL.
37 __isl_null isl_stride_info
*isl_stride_info_free(
38 __isl_take isl_stride_info
*si
)
42 isl_val_free(si
->stride
);
43 isl_aff_free(si
->offset
);
48 /* Construct an isl_stride_info object with given offset and stride.
50 __isl_give isl_stride_info
*isl_stride_info_alloc(
51 __isl_take isl_val
*stride
, __isl_take isl_aff
*offset
)
53 struct isl_stride_info
*si
;
55 if (!stride
|| !offset
)
57 si
= isl_alloc_type(isl_val_get_ctx(stride
), struct isl_stride_info
);
69 /* Make a copy of "si" and return it.
71 __isl_give isl_stride_info
*isl_stride_info_copy(
72 __isl_keep isl_stride_info
*si
)
77 return isl_stride_info_alloc(isl_val_copy(si
->stride
),
78 isl_aff_copy(si
->offset
));
81 /* Return the stride of "si".
83 __isl_give isl_val
*isl_stride_info_get_stride(__isl_keep isl_stride_info
*si
)
87 return isl_val_copy(si
->stride
);
90 /* Return the offset of "si".
92 __isl_give isl_aff
*isl_stride_info_get_offset(__isl_keep isl_stride_info
*si
)
96 return isl_aff_copy(si
->offset
);
99 /* Information used inside detect_stride.
101 * "pos" is the set dimension at which the stride is being determined.
102 * "want_offset" is set if the offset should be computed.
103 * "found" is set if some stride was found already.
104 * "stride" and "offset" contain the (combined) stride and offset
105 * found so far and are NULL when "found" is not set.
106 * If "want_offset" is not set, then "offset" remains NULL.
108 struct isl_detect_stride_data
{
116 /* Set the stride and offset of data->pos to the given
117 * value and expression.
119 * If we had already found a stride before, then the two strides
120 * are combined into a single stride.
122 * In particular, if the new stride information is of the form
126 * and the old stride information is of the form
130 * then we compute the extended gcd of s and s2
134 * with g = gcd(s,s2), multiply the first equation with t1 = b s2/g
135 * and the second with t2 = a s1/g.
138 * i = (b s2 + a s1)/g i = t1 f + t2 f2 + (s s2)/g (...)
140 * so that t1 f + t2 f2 is the combined offset and (s s2)/g = lcm(s,s2)
141 * is the combined stride.
143 static isl_stat
set_stride(struct isl_detect_stride_data
*data
,
144 __isl_take isl_val
*stride
, __isl_take isl_aff
*offset
)
148 if (!stride
|| !offset
)
154 isl_val
*stride2
, *a
, *b
, *g
;
157 stride2
= data
->stride
;
158 g
= isl_val_gcdext(isl_val_copy(stride
), isl_val_copy(stride2
),
160 a
= isl_val_mul(a
, isl_val_copy(stride
));
161 a
= isl_val_div(a
, isl_val_copy(g
));
162 stride2
= isl_val_div(stride2
, g
);
163 b
= isl_val_mul(b
, isl_val_copy(stride2
));
164 stride
= isl_val_mul(stride
, stride2
);
166 if (!data
->want_offset
) {
170 offset2
= data
->offset
;
171 offset2
= isl_aff_scale_val(offset2
, a
);
172 offset
= isl_aff_scale_val(offset
, b
);
173 offset
= isl_aff_add(offset
, offset2
);
178 data
->stride
= stride
;
179 if (data
->want_offset
)
180 data
->offset
= offset
;
182 isl_aff_free(offset
);
183 if (!data
->stride
|| (data
->want_offset
&& !data
->offset
))
184 return isl_stat_error
;
188 isl_val_free(stride
);
189 isl_aff_free(offset
);
190 return isl_stat_error
;
193 /* Check if constraint "c" imposes any stride on dimension data->pos
194 * and, if so, update the stride information in "data".
196 * In order to impose a stride on the dimension, "c" needs to be an equality
197 * and it needs to involve the dimension. Note that "c" may also be
198 * a div constraint and thus an inequality that we cannot use.
200 * Let c be of the form
202 * h(p) + g * v * i + g * stride * f(alpha) = 0
204 * with h(p) an expression in terms of the parameters and other dimensions
205 * and f(alpha) an expression in terms of the existentially quantified
208 * If "stride" is not zero and not one, then it represents a non-trivial stride
209 * on "i". We compute a and b such that
215 * g v i = -h(p) + g stride f(alpha)
217 * a g v i = -a h(p) + g stride f(alpha)
219 * a g v i + b g stride i = -a h(p) + g stride * (...)
221 * g i = -a h(p) + g stride * (...)
223 * i = -a h(p)/g + stride * (...)
225 * The expression "-a h(p)/g" can therefore be used as offset.
227 static isl_stat
detect_stride(__isl_take isl_constraint
*c
, void *user
)
229 struct isl_detect_stride_data
*data
= user
;
232 isl_stat r
= isl_stat_ok
;
233 isl_val
*v
, *stride
, *m
;
234 isl_bool is_eq
, relevant
, has_stride
;
236 is_eq
= isl_constraint_is_equality(c
);
237 relevant
= isl_constraint_involves_dims(c
, isl_dim_set
, data
->pos
, 1);
238 if (is_eq
< 0 || relevant
< 0)
240 if (!is_eq
|| !relevant
) {
241 isl_constraint_free(c
);
245 ctx
= isl_constraint_get_ctx(c
);
246 stride
= isl_val_zero(ctx
);
247 n_div
= isl_constraint_dim(c
, isl_dim_div
);
248 for (i
= 0; i
< n_div
; ++i
) {
249 v
= isl_constraint_get_coefficient_val(c
, isl_dim_div
, i
);
250 stride
= isl_val_gcd(stride
, v
);
253 v
= isl_constraint_get_coefficient_val(c
, isl_dim_set
, data
->pos
);
254 m
= isl_val_gcd(isl_val_copy(stride
), isl_val_copy(v
));
255 stride
= isl_val_div(stride
, isl_val_copy(m
));
256 v
= isl_val_div(v
, isl_val_copy(m
));
258 has_stride
= isl_val_gt_si(stride
, 1);
259 if (has_stride
>= 0 && has_stride
) {
261 isl_val
*gcd
, *a
, *b
;
263 gcd
= isl_val_gcdext(v
, isl_val_copy(stride
), &a
, &b
);
267 aff
= isl_constraint_get_aff(c
);
268 for (i
= 0; i
< n_div
; ++i
)
269 aff
= isl_aff_set_coefficient_si(aff
,
271 aff
= isl_aff_set_coefficient_si(aff
, isl_dim_in
, data
->pos
, 0);
272 aff
= isl_aff_remove_unused_divs(aff
);
274 aff
= isl_aff_scale_val(aff
, a
);
275 aff
= isl_aff_scale_down_val(aff
, m
);
276 r
= set_stride(data
, stride
, aff
);
278 isl_val_free(stride
);
283 isl_constraint_free(c
);
285 return isl_stat_error
;
288 isl_constraint_free(c
);
289 return isl_stat_error
;
292 /* Check if the constraints in "set" imply any stride on set dimension "pos" and
293 * store the results in data->stride and data->offset.
295 * In particular, compute the affine hull and then check if
296 * any of the constraints in the hull impose any stride on the dimension.
297 * If no such constraint can be found, then the offset is taken
298 * to be the zero expression and the stride is taken to be one.
300 static void set_detect_stride(__isl_keep isl_set
*set
, int pos
,
301 struct isl_detect_stride_data
*data
)
305 hull
= isl_set_affine_hull(isl_set_copy(set
));
311 if (isl_basic_set_foreach_constraint(hull
, &detect_stride
, data
) < 0)
315 data
->stride
= isl_val_one(isl_set_get_ctx(set
));
316 if (data
->want_offset
) {
320 space
= isl_set_get_space(set
);
321 ls
= isl_local_space_from_space(space
);
322 data
->offset
= isl_aff_zero_on_domain(ls
);
325 isl_basic_set_free(hull
);
328 isl_basic_set_free(hull
);
329 data
->stride
= isl_val_free(data
->stride
);
330 data
->offset
= isl_aff_free(data
->offset
);
333 /* Check if the constraints in "set" imply any stride on set dimension "pos" and
334 * return the results in the form of an offset and a stride.
336 __isl_give isl_stride_info
*isl_set_get_stride_info(__isl_keep isl_set
*set
,
339 struct isl_detect_stride_data data
;
341 data
.want_offset
= 1;
342 set_detect_stride(set
, pos
, &data
);
344 return isl_stride_info_alloc(data
.stride
, data
.offset
);
347 /* Check if the constraints in "set" imply any stride on set dimension "pos" and
348 * return this stride.
350 __isl_give isl_val
*isl_set_get_stride(__isl_keep isl_set
*set
, int pos
)
352 struct isl_detect_stride_data data
;
354 data
.want_offset
= 0;
355 set_detect_stride(set
, pos
, &data
);
360 /* Check if the constraints in "map" imply any stride on output dimension "pos",
361 * independently of any other output dimensions, and
362 * return the results in the form of an offset and a stride.
364 * Convert the input to a set with only the input dimensions and
365 * the single output dimension such that it be passed to
366 * isl_set_get_stride_info and convert the result back to
367 * an expression defined over the domain of "map".
369 __isl_give isl_stride_info
*isl_map_get_range_stride_info(
370 __isl_keep isl_map
*map
, int pos
)
375 map
= isl_map_copy(map
);
376 map
= isl_map_project_onto(map
, isl_dim_out
, pos
, 1);
377 pos
= isl_map_dim(map
, isl_dim_in
);
378 set
= isl_map_wrap(map
);
379 si
= isl_set_get_stride_info(set
, pos
);
383 si
->offset
= isl_aff_domain_factor_domain(si
->offset
);
385 return isl_stride_info_free(si
);