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 /* Free "si" and return NULL.
27 __isl_null isl_stride_info
*isl_stride_info_free(
28 __isl_take isl_stride_info
*si
)
32 isl_val_free(si
->stride
);
33 isl_aff_free(si
->offset
);
38 /* Construct an isl_stride_info object with given offset and stride.
40 __isl_give isl_stride_info
*isl_stride_info_alloc(
41 __isl_take isl_val
*stride
, __isl_take isl_aff
*offset
)
43 struct isl_stride_info
*si
;
45 if (!stride
|| !offset
)
47 si
= isl_alloc_type(isl_val_get_ctx(stride
), struct isl_stride_info
);
59 /* Return the stride of "si".
61 __isl_give isl_val
*isl_stride_info_get_stride(__isl_keep isl_stride_info
*si
)
65 return isl_val_copy(si
->stride
);
68 /* Return the offset of "si".
70 __isl_give isl_aff
*isl_stride_info_get_offset(__isl_keep isl_stride_info
*si
)
74 return isl_aff_copy(si
->offset
);
77 /* Information used inside detect_stride.
79 * "pos" is the set dimension at which the stride is being determined.
80 * "want_offset" is set if the offset should be computed.
81 * "found" is set if some stride was found already.
82 * "stride" and "offset" contain the (combined) stride and offset
83 * found so far and are NULL when "found" is not set.
84 * If "want_offset" is not set, then "offset" remains NULL.
86 struct isl_detect_stride_data
{
94 /* Set the stride and offset of data->pos to the given
95 * value and expression.
97 * If we had already found a stride before, then the two strides
98 * are combined into a single stride.
100 * In particular, if the new stride information is of the form
104 * and the old stride information is of the form
108 * then we compute the extended gcd of s and s2
112 * with g = gcd(s,s2), multiply the first equation with t1 = b s2/g
113 * and the second with t2 = a s1/g.
116 * i = (b s2 + a s1)/g i = t1 f + t2 f2 + (s s2)/g (...)
118 * so that t1 f + t2 f2 is the combined offset and (s s2)/g = lcm(s,s2)
119 * is the combined stride.
121 static isl_stat
set_stride(struct isl_detect_stride_data
*data
,
122 __isl_take isl_val
*stride
, __isl_take isl_aff
*offset
)
126 if (!stride
|| !offset
)
132 isl_val
*stride2
, *a
, *b
, *g
;
135 stride2
= data
->stride
;
136 g
= isl_val_gcdext(isl_val_copy(stride
), isl_val_copy(stride2
),
138 a
= isl_val_mul(a
, isl_val_copy(stride
));
139 a
= isl_val_div(a
, isl_val_copy(g
));
140 stride2
= isl_val_div(stride2
, g
);
141 b
= isl_val_mul(b
, isl_val_copy(stride2
));
142 stride
= isl_val_mul(stride
, stride2
);
144 if (!data
->want_offset
) {
148 offset2
= data
->offset
;
149 offset2
= isl_aff_scale_val(offset2
, a
);
150 offset
= isl_aff_scale_val(offset
, b
);
151 offset
= isl_aff_add(offset
, offset2
);
156 data
->stride
= stride
;
157 if (data
->want_offset
)
158 data
->offset
= offset
;
160 isl_aff_free(offset
);
161 if (!data
->stride
|| (data
->want_offset
&& !data
->offset
))
162 return isl_stat_error
;
166 isl_val_free(stride
);
167 isl_aff_free(offset
);
168 return isl_stat_error
;
171 /* Check if constraint "c" imposes any stride on dimension data->pos
172 * and, if so, update the stride information in "data".
174 * In order to impose a stride on the dimension, "c" needs to be an equality
175 * and it needs to involve the dimension. Note that "c" may also be
176 * a div constraint and thus an inequality that we cannot use.
178 * Let c be of the form
180 * h(p) + g * v * i + g * stride * f(alpha) = 0
182 * with h(p) an expression in terms of the parameters and other dimensions
183 * and f(alpha) an expression in terms of the existentially quantified
186 * If "stride" is not zero and not one, then it represents a non-trivial stride
187 * on "i". We compute a and b such that
193 * g v i = -h(p) + g stride f(alpha)
195 * a g v i = -a h(p) + g stride f(alpha)
197 * a g v i + b g stride i = -a h(p) + g stride * (...)
199 * g i = -a h(p) + g stride * (...)
201 * i = -a h(p)/g + stride * (...)
203 * The expression "-a h(p)/g" can therefore be used as offset.
205 static isl_stat
detect_stride(__isl_take isl_constraint
*c
, void *user
)
207 struct isl_detect_stride_data
*data
= user
;
210 isl_stat r
= isl_stat_ok
;
211 isl_val
*v
, *stride
, *m
;
212 isl_bool is_eq
, relevant
, has_stride
;
214 is_eq
= isl_constraint_is_equality(c
);
215 relevant
= isl_constraint_involves_dims(c
, isl_dim_set
, data
->pos
, 1);
216 if (is_eq
< 0 || relevant
< 0)
218 if (!is_eq
|| !relevant
) {
219 isl_constraint_free(c
);
223 ctx
= isl_constraint_get_ctx(c
);
224 stride
= isl_val_zero(ctx
);
225 n_div
= isl_constraint_dim(c
, isl_dim_div
);
226 for (i
= 0; i
< n_div
; ++i
) {
227 v
= isl_constraint_get_coefficient_val(c
, isl_dim_div
, i
);
228 stride
= isl_val_gcd(stride
, v
);
231 v
= isl_constraint_get_coefficient_val(c
, isl_dim_set
, data
->pos
);
232 m
= isl_val_gcd(isl_val_copy(stride
), isl_val_copy(v
));
233 stride
= isl_val_div(stride
, isl_val_copy(m
));
234 v
= isl_val_div(v
, isl_val_copy(m
));
236 has_stride
= isl_val_gt_si(stride
, 1);
237 if (has_stride
>= 0 && has_stride
) {
239 isl_val
*gcd
, *a
, *b
;
241 gcd
= isl_val_gcdext(v
, isl_val_copy(stride
), &a
, &b
);
245 aff
= isl_constraint_get_aff(c
);
246 for (i
= 0; i
< n_div
; ++i
)
247 aff
= isl_aff_set_coefficient_si(aff
,
249 aff
= isl_aff_set_coefficient_si(aff
, isl_dim_in
, data
->pos
, 0);
251 aff
= isl_aff_scale_val(aff
, a
);
252 aff
= isl_aff_scale_down_val(aff
, m
);
253 r
= set_stride(data
, stride
, aff
);
255 isl_val_free(stride
);
260 isl_constraint_free(c
);
262 return isl_stat_error
;
265 isl_constraint_free(c
);
266 return isl_stat_error
;
269 /* Check if the constraints in "set" imply any stride on set dimension "pos" and
270 * store the results in data->stride and data->offset.
272 * In particular, compute the affine hull and then check if
273 * any of the constraints in the hull impose any stride on the dimension.
274 * If no such constraint can be found, then the offset is taken
275 * to be the zero expression and the stride is taken to be one.
277 static void set_detect_stride(__isl_keep isl_set
*set
, int pos
,
278 struct isl_detect_stride_data
*data
)
282 hull
= isl_set_affine_hull(isl_set_copy(set
));
288 if (isl_basic_set_foreach_constraint(hull
, &detect_stride
, data
) < 0)
292 data
->stride
= isl_val_one(isl_set_get_ctx(set
));
293 if (data
->want_offset
) {
297 space
= isl_set_get_space(set
);
298 ls
= isl_local_space_from_space(space
);
299 data
->offset
= isl_aff_zero_on_domain(ls
);
302 isl_basic_set_free(hull
);
305 isl_basic_set_free(hull
);
306 data
->stride
= isl_val_free(data
->stride
);
307 data
->offset
= isl_aff_free(data
->offset
);
310 /* Check if the constraints in "set" imply any stride on set dimension "pos" and
311 * return the results in the form of an offset and a stride.
313 __isl_give isl_stride_info
*isl_set_get_stride_info(__isl_keep isl_set
*set
,
316 struct isl_detect_stride_data data
;
318 data
.want_offset
= 1;
319 set_detect_stride(set
, pos
, &data
);
321 return isl_stride_info_alloc(data
.stride
, data
.offset
);
324 /* Check if the constraints in "set" imply any stride on set dimension "pos" and
325 * return this stride.
327 __isl_give isl_val
*isl_set_get_stride(__isl_keep isl_set
*set
, int pos
)
329 struct isl_detect_stride_data data
;
331 data
.want_offset
= 0;
332 set_detect_stride(set
, pos
, &data
);
337 /* Check if the constraints in "map" imply any stride on output dimension "pos",
338 * independently of any other output dimensions, and
339 * return the results in the form of an offset and a stride.
341 * Convert the input to a set with only the input dimensions and
342 * the single output dimension such that it be passed to
343 * isl_set_get_stride_info and convert the result back to
344 * an expression defined over the domain of "map".
346 __isl_give isl_stride_info
*isl_map_get_range_stride_info(
347 __isl_keep isl_map
*map
, int pos
)
352 map
= isl_map_copy(map
);
353 map
= isl_map_project_onto(map
, isl_dim_out
, pos
, 1);
354 pos
= isl_map_dim(map
, isl_dim_in
);
355 set
= isl_map_wrap(map
);
356 si
= isl_set_get_stride_info(set
, pos
);
360 si
->offset
= isl_aff_domain_factor_domain(si
->offset
);
362 return isl_stride_info_free(si
);