add isl_set_unshifted_simple_hull
authorSven Verdoolaege <skimo@kotnet.org>
Thu, 19 Apr 2012 14:09:42 +0000 (16:09 +0200)
committerSven Verdoolaege <skimo@kotnet.org>
Mon, 17 Sep 2012 15:15:48 +0000 (17:15 +0200)
Signed-off-by: Sven Verdoolaege <skimo@kotnet.org>
doc/user.pod
include/isl/map.h
include/isl/set.h
isl_convex_hull.c

index cd1c510..b5c98de 100644 (file)
@@ -2241,6 +2241,12 @@ variables, then the result of these operations is currently undefined.
 
 =item * Simple hull
 
+       __isl_give isl_basic_set *
+       isl_set_unshifted_simple_hull(
+               __isl_take isl_set *set);
+       __isl_give isl_basic_map *
+       isl_map_unshifted_simple_hull(
+               __isl_take isl_map *map);
        __isl_give isl_basic_set *isl_set_simple_hull(
                __isl_take isl_set *set);
        __isl_give isl_basic_map *isl_map_simple_hull(
@@ -2252,6 +2258,8 @@ These functions compute a single basic set or relation
 that contains the whole input set or relation.
 In particular, the output is described by translates
 of the constraints describing the basic sets or relations in the input.
+In case of C<isl_set_unshifted_simple_hull>, only the original
+constraints are used, without any translation.
 
 =begin latex
 
index 1b7d413..50aa1f7 100644 (file)
@@ -140,6 +140,8 @@ __isl_give isl_basic_map *isl_basic_map_remove_redundancies(
        __isl_take isl_basic_map *bmap);
 __isl_give isl_map *isl_map_remove_redundancies(__isl_take isl_map *map);
 __isl_give isl_basic_map *isl_map_simple_hull(__isl_take isl_map *map);
+__isl_give isl_basic_map *isl_map_unshifted_simple_hull(
+       __isl_take isl_map *map);
 
 __isl_export
 __isl_give isl_basic_map *isl_basic_map_intersect_domain(
index 8276f20..f38ffe0 100644 (file)
@@ -267,6 +267,8 @@ __isl_give isl_basic_set *isl_set_convex_hull(__isl_take isl_set *set);
 __isl_export
 __isl_give isl_basic_set *isl_set_polyhedral_hull(__isl_take isl_set *set);
 __isl_give isl_basic_set *isl_set_simple_hull(__isl_take isl_set *set);
+__isl_give isl_basic_set *isl_set_unshifted_simple_hull(
+       __isl_take isl_set *set);
 struct isl_basic_set *isl_set_bounded_simple_hull(struct isl_set *set);
 __isl_give isl_set *isl_set_recession_cone(__isl_take isl_set *set);
 
index 8162d4f..3853fc1 100644 (file)
@@ -2107,12 +2107,14 @@ error:
  * it can be relaxed (by increasing the constant term) to become
  * a bound for that basic set.  In the latter case, the constant
  * term is updated.
+ * Relaxation of the constant term is only allowed if "shift" is set.
+ *
  * Return 1 if "ineq" is a bound
  *       0 if "ineq" may attain arbitrarily small values on basic set "j"
  *      -1 if some error occurred
  */
 static int is_bound(struct sh_data *data, struct isl_set *set, int j,
-                       isl_int *ineq)
+       isl_int *ineq, int shift)
 {
        enum isl_lp_result res;
        isl_int opt;
@@ -2127,8 +2129,12 @@ static int is_bound(struct sh_data *data, struct isl_set *set, int j,
 
        res = isl_tab_min(data->p[j].tab, ineq, data->ctx->one,
                                &opt, NULL, 0);
-       if (res == isl_lp_ok && isl_int_is_neg(opt))
-               isl_int_sub(ineq[0], ineq[0], opt);
+       if (res == isl_lp_ok && isl_int_is_neg(opt)) {
+               if (shift)
+                       isl_int_sub(ineq[0], ineq[0], opt);
+               else
+                       res = isl_lp_unbounded;
+       }
 
        isl_int_clear(opt);
 
@@ -2136,9 +2142,9 @@ static int is_bound(struct sh_data *data, struct isl_set *set, int j,
               res == isl_lp_unbounded ? 0 : -1;
 }
 
-/* Check if inequality "ineq" from basic set "i" can be relaxed to
+/* Check if inequality "ineq" from basic set "i" is or can be relaxed to
  * become a bound on the whole set.  If so, add the (relaxed) inequality
- * to "hull".
+ * to "hull".  Relaxation is only allowed if "shift" is set.
  *
  * We first check if "hull" already contains a translate of the inequality.
  * If so, we are done.
@@ -2154,7 +2160,8 @@ static int is_bound(struct sh_data *data, struct isl_set *set, int j,
  * the inequality accordingly.
  */
 static struct isl_basic_set *add_bound(struct isl_basic_set *hull,
-       struct sh_data *data, struct isl_set *set, int i, isl_int *ineq)
+       struct sh_data *data, struct isl_set *set, int i, isl_int *ineq,
+       int shift)
 {
        uint32_t c_hash;
        struct ineq_cmp_data v;
@@ -2189,7 +2196,7 @@ static struct isl_basic_set *add_bound(struct isl_basic_set *hull,
 
        for (j = 0; j < i; ++j) {
                int bound;
-               bound = is_bound(data, set, j, hull->ineq[k]);
+               bound = is_bound(data, set, j, hull->ineq[k], shift);
                if (bound < 0)
                        goto error;
                if (!bound)
@@ -2217,7 +2224,7 @@ static struct isl_basic_set *add_bound(struct isl_basic_set *hull,
                                isl_int_neg(ineq_j[0], ineq_j[0]);
                        continue;
                }
-               bound = is_bound(data, set, j, hull->ineq[k]);
+               bound = is_bound(data, set, j, hull->ineq[k], shift);
                if (bound < 0)
                        goto error;
                if (!bound)
@@ -2240,12 +2247,12 @@ error:
        return NULL;
 }
 
-/* Check if any inequality from basic set "i" can be relaxed to
+/* Check if any inequality from basic set "i" is or can be relaxed to
  * become a bound on the whole set.  If so, add the (relaxed) inequality
- * to "hull".
+ * to "hull".  Relaxation is only allowed if "shift" is set.
  */
 static struct isl_basic_set *add_bounds(struct isl_basic_set *bset,
-       struct sh_data *data, struct isl_set *set, int i)
+       struct sh_data *data, struct isl_set *set, int i, int shift)
 {
        int j, k;
        unsigned dim = isl_basic_set_total_dim(bset);
@@ -2253,18 +2260,21 @@ static struct isl_basic_set *add_bounds(struct isl_basic_set *bset,
        for (j = 0; j < set->p[i]->n_eq; ++j) {
                for (k = 0; k < 2; ++k) {
                        isl_seq_neg(set->p[i]->eq[j], set->p[i]->eq[j], 1+dim);
-                       bset = add_bound(bset, data, set, i, set->p[i]->eq[j]);
+                       bset = add_bound(bset, data, set, i, set->p[i]->eq[j],
+                                           shift);
                }
        }
        for (j = 0; j < set->p[i]->n_ineq; ++j)
-               bset = add_bound(bset, data, set, i, set->p[i]->ineq[j]);
+               bset = add_bound(bset, data, set, i, set->p[i]->ineq[j], shift);
        return bset;
 }
 
 /* Compute a superset of the convex hull of set that is described
- * by only translates of the constraints in the constituents of set.
+ * by only (translates of) the constraints in the constituents of set.
+ * Translation is only allowed if "shift" is set.
  */
-static struct isl_basic_set *uset_simple_hull(struct isl_set *set)
+static __isl_give isl_basic_set *uset_simple_hull(__isl_take isl_set *set,
+       int shift)
 {
        struct sh_data *data = NULL;
        struct isl_basic_set *hull = NULL;
@@ -2290,7 +2300,7 @@ static struct isl_basic_set *uset_simple_hull(struct isl_set *set)
                goto error;
 
        for (i = 0; i < set->n; ++i)
-               hull = add_bounds(hull, data, set, i);
+               hull = add_bounds(hull, data, set, i, shift);
 
        sh_data_free(data);
        isl_set_free(set);
@@ -2304,9 +2314,11 @@ error:
 }
 
 /* Compute a superset of the convex hull of map that is described
- * by only translates of the constraints in the constituents of map.
+ * by only (translates of) the constraints in the constituents of map.
+ * Translation is only allowed if "shift" is set.
  */
-struct isl_basic_map *isl_map_simple_hull(struct isl_map *map)
+static __isl_give isl_basic_map *map_simple_hull(__isl_take isl_map *map,
+       int shift)
 {
        struct isl_set *set = NULL;
        struct isl_basic_map *model = NULL;
@@ -2334,7 +2346,7 @@ struct isl_basic_map *isl_map_simple_hull(struct isl_map *map)
 
        set = isl_map_underlying_set(map);
 
-       bset = uset_simple_hull(set);
+       bset = uset_simple_hull(set, shift);
 
        hull = isl_basic_map_overlying_set(bset, model);
 
@@ -2349,12 +2361,35 @@ struct isl_basic_map *isl_map_simple_hull(struct isl_map *map)
        return hull;
 }
 
+/* Compute a superset of the convex hull of map that is described
+ * by only translates of the constraints in the constituents of map.
+ */
+__isl_give isl_basic_map *isl_map_simple_hull(__isl_take isl_map *map)
+{
+       return map_simple_hull(map, 1);
+}
+
 struct isl_basic_set *isl_set_simple_hull(struct isl_set *set)
 {
        return (struct isl_basic_set *)
                isl_map_simple_hull((struct isl_map *)set);
 }
 
+/* Compute a superset of the convex hull of map that is described
+ * by only the constraints in the constituents of map.
+ */
+__isl_give isl_basic_map *isl_map_unshifted_simple_hull(
+       __isl_take isl_map *map)
+{
+       return map_simple_hull(map, 0);
+}
+
+__isl_give isl_basic_set *isl_set_unshifted_simple_hull(
+       __isl_take isl_set *set)
+{
+       return isl_map_unshifted_simple_hull(set);
+}
+
 /* Given a set "set", return parametric bounds on the dimension "dim".
  */
 static struct isl_basic_set *set_bounds(struct isl_set *set, int dim)