2014-01-27 21:38:34 +01:00
|
|
|
|
/*
|
|
|
|
|
* Copyright 2010 INRIA Saclay
|
|
|
|
|
* Copyright 2012 Ecole Normale Superieure
|
|
|
|
|
*
|
|
|
|
|
* Use of this software is governed by the MIT license
|
|
|
|
|
*
|
|
|
|
|
* Written by Sven Verdoolaege,
|
|
|
|
|
* INRIA Saclay - Ile-de-France, Parc Club Orsay Universite,
|
|
|
|
|
* ZAC des vignes, 4 rue Jacques Monod, 91893 Orsay, France
|
|
|
|
|
* and Ecole Normale Superieure, 45 rue d’Ulm, 75230 Paris, France
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* Function for computing the lexicographic optimum of a map
|
|
|
|
|
* in the form of either an isl_map or an isl_pw_multi_aff.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#define xSF(TYPE,SUFFIX) TYPE ## SUFFIX
|
|
|
|
|
#define SF(TYPE,SUFFIX) xSF(TYPE,SUFFIX)
|
|
|
|
|
|
2018-07-03 21:52:50 +02:00
|
|
|
|
/* Compute the lexicographic minimum (or maximum if "flags" includes
|
|
|
|
|
* ISL_OPT_MAX) of "bmap" over the domain "dom" and return the result.
|
|
|
|
|
* If "empty" is not NULL, then *empty is assigned a set that
|
|
|
|
|
* contains those parts of the domain where there is no solution.
|
|
|
|
|
* If "flags" includes ISL_OPT_FULL, then "dom" is NULL and the optimum
|
|
|
|
|
* should be computed over the domain of "bmap". "empty" is also NULL
|
|
|
|
|
* in this case.
|
|
|
|
|
* If "bmap" is marked as rational (ISL_BASIC_MAP_RATIONAL),
|
|
|
|
|
* then the rational optimum is computed. Otherwise, the integral optimum
|
|
|
|
|
* is computed.
|
|
|
|
|
*/
|
|
|
|
|
static __isl_give TYPE *SF(isl_basic_map_partial_lexopt,SUFFIX)(
|
|
|
|
|
__isl_take isl_basic_map *bmap, __isl_take isl_basic_set *dom,
|
|
|
|
|
__isl_give isl_set **empty, unsigned flags)
|
|
|
|
|
{
|
|
|
|
|
return SF(isl_tab_basic_map_partial_lexopt,SUFFIX)(bmap, dom, empty,
|
|
|
|
|
flags);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_basic_map_partial_lexmax,SUFFIX)(
|
|
|
|
|
__isl_take isl_basic_map *bmap, __isl_take isl_basic_set *dom,
|
|
|
|
|
__isl_give isl_set **empty)
|
|
|
|
|
{
|
|
|
|
|
unsigned flags = ISL_OPT_MAX;
|
|
|
|
|
return SF(isl_basic_map_partial_lexopt,SUFFIX)(bmap, dom, empty, flags);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_basic_map_partial_lexmin,SUFFIX)(
|
|
|
|
|
__isl_take isl_basic_map *bmap, __isl_take isl_basic_set *dom,
|
|
|
|
|
__isl_give isl_set **empty)
|
|
|
|
|
{
|
|
|
|
|
unsigned flags = 0;
|
|
|
|
|
return SF(isl_basic_map_partial_lexopt,SUFFIX)(bmap, dom, empty, flags);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_basic_set_partial_lexmin,SUFFIX)(
|
|
|
|
|
__isl_take isl_basic_set *bset, __isl_take isl_basic_set *dom,
|
|
|
|
|
__isl_give isl_set **empty)
|
|
|
|
|
{
|
|
|
|
|
return SF(isl_basic_map_partial_lexmin,SUFFIX)(bset, dom, empty);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_basic_set_partial_lexmax,SUFFIX)(
|
|
|
|
|
__isl_take isl_basic_set *bset, __isl_take isl_basic_set *dom,
|
|
|
|
|
__isl_give isl_set **empty)
|
|
|
|
|
{
|
|
|
|
|
return SF(isl_basic_map_partial_lexmax,SUFFIX)(bset, dom, empty);
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-27 21:38:34 +01:00
|
|
|
|
/* Given a basic map "bmap", compute the lexicographically minimal
|
|
|
|
|
* (or maximal) image element for each domain element in dom.
|
2018-07-03 21:52:50 +02:00
|
|
|
|
* If empty is not NULL, then set *empty to those elements in dom
|
|
|
|
|
* that do not have an image element.
|
|
|
|
|
* If "flags" includes ISL_OPT_FULL, then "dom" is NULL and the optimum
|
|
|
|
|
* should be computed over the domain of "bmap". "empty" is also NULL
|
|
|
|
|
* in this case.
|
2014-01-27 21:38:34 +01:00
|
|
|
|
*
|
|
|
|
|
* We first make sure the basic sets in dom are disjoint and then
|
|
|
|
|
* simply collect the results over each of the basic sets separately.
|
|
|
|
|
* We could probably improve the efficiency a bit by moving the union
|
|
|
|
|
* domain down into the parametric integer programming.
|
2018-07-03 21:52:50 +02:00
|
|
|
|
*
|
|
|
|
|
* If a full optimum is being computed (i.e., "flags" includes ISL_OPT_FULL),
|
|
|
|
|
* then no domain is given and there is then also no need to consider
|
|
|
|
|
* the disjuncts of the domain.
|
2014-01-27 21:38:34 +01:00
|
|
|
|
*/
|
|
|
|
|
static __isl_give TYPE *SF(basic_map_partial_lexopt,SUFFIX)(
|
|
|
|
|
__isl_take isl_basic_map *bmap, __isl_take isl_set *dom,
|
2018-07-03 21:52:50 +02:00
|
|
|
|
__isl_give isl_set **empty, unsigned flags)
|
2014-01-27 21:38:34 +01:00
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
TYPE *res;
|
2018-07-03 21:52:50 +02:00
|
|
|
|
isl_set *all_empty;
|
|
|
|
|
|
|
|
|
|
if (ISL_FL_ISSET(flags, ISL_OPT_FULL))
|
|
|
|
|
return SF(isl_basic_map_partial_lexopt,SUFFIX)(bmap, NULL,
|
|
|
|
|
empty, flags);
|
2014-01-27 21:38:34 +01:00
|
|
|
|
|
|
|
|
|
dom = isl_set_make_disjoint(dom);
|
|
|
|
|
if (!dom)
|
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
|
|
if (isl_set_plain_is_empty(dom)) {
|
|
|
|
|
isl_space *space = isl_basic_map_get_space(bmap);
|
|
|
|
|
if (empty)
|
|
|
|
|
*empty = dom;
|
|
|
|
|
else
|
|
|
|
|
isl_set_free(dom);
|
|
|
|
|
isl_basic_map_free(bmap);
|
|
|
|
|
return EMPTY(space);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
res = SF(isl_basic_map_partial_lexopt,SUFFIX)(isl_basic_map_copy(bmap),
|
2018-07-03 21:52:50 +02:00
|
|
|
|
isl_basic_set_copy(dom->p[0]), empty, flags);
|
|
|
|
|
|
|
|
|
|
if (empty)
|
|
|
|
|
all_empty = *empty;
|
2014-01-27 21:38:34 +01:00
|
|
|
|
for (i = 1; i < dom->n; ++i) {
|
|
|
|
|
TYPE *res_i;
|
|
|
|
|
|
|
|
|
|
res_i = SF(isl_basic_map_partial_lexopt,SUFFIX)(
|
|
|
|
|
isl_basic_map_copy(bmap),
|
2018-07-03 21:52:50 +02:00
|
|
|
|
isl_basic_set_copy(dom->p[i]), empty, flags);
|
2014-01-27 21:38:34 +01:00
|
|
|
|
|
|
|
|
|
res = ADD(res, res_i);
|
2018-07-03 21:52:50 +02:00
|
|
|
|
if (empty)
|
|
|
|
|
all_empty = isl_set_union_disjoint(all_empty, *empty);
|
2014-01-27 21:38:34 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-07-03 21:52:50 +02:00
|
|
|
|
if (empty)
|
|
|
|
|
*empty = all_empty;
|
2014-01-27 21:38:34 +01:00
|
|
|
|
isl_set_free(dom);
|
|
|
|
|
isl_basic_map_free(bmap);
|
|
|
|
|
return res;
|
|
|
|
|
error:
|
2018-07-03 21:52:50 +02:00
|
|
|
|
if (empty)
|
|
|
|
|
*empty = NULL;
|
2014-01-27 21:38:34 +01:00
|
|
|
|
isl_set_free(dom);
|
|
|
|
|
isl_basic_map_free(bmap);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-03 21:52:50 +02:00
|
|
|
|
/* Compute the lexicographic minimum (or maximum if "flags" includes
|
|
|
|
|
* ISL_OPT_MAX) of "bmap" over its domain.
|
|
|
|
|
*/
|
|
|
|
|
__isl_give TYPE *SF(isl_basic_map_lexopt,SUFFIX)(
|
|
|
|
|
__isl_take isl_basic_map *bmap, unsigned flags)
|
|
|
|
|
{
|
|
|
|
|
ISL_FL_SET(flags, ISL_OPT_FULL);
|
|
|
|
|
return SF(isl_basic_map_partial_lexopt,SUFFIX)(bmap, NULL, NULL, flags);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_basic_map_lexmin,SUFFIX)(__isl_take isl_basic_map *bmap)
|
|
|
|
|
{
|
|
|
|
|
return SF(isl_basic_map_lexopt,SUFFIX)(bmap, 0);
|
|
|
|
|
}
|
|
|
|
|
|
2014-01-27 21:38:34 +01:00
|
|
|
|
static __isl_give TYPE *SF(isl_map_partial_lexopt_aligned,SUFFIX)(
|
|
|
|
|
__isl_take isl_map *map, __isl_take isl_set *dom,
|
2018-07-03 21:52:50 +02:00
|
|
|
|
__isl_give isl_set **empty, unsigned flags);
|
|
|
|
|
/* This function is currently only used when TYPE is defined as isl_map. */
|
|
|
|
|
static __isl_give TYPE *SF(isl_map_partial_lexopt,SUFFIX)(
|
|
|
|
|
__isl_take isl_map *map, __isl_take isl_set *dom,
|
|
|
|
|
__isl_give isl_set **empty, unsigned flags)
|
|
|
|
|
__attribute__ ((unused));
|
2014-01-27 21:38:34 +01:00
|
|
|
|
|
|
|
|
|
/* Given a map "map", compute the lexicographically minimal
|
|
|
|
|
* (or maximal) image element for each domain element in dom.
|
|
|
|
|
* Set *empty to those elements in dom that do not have an image element.
|
|
|
|
|
*
|
|
|
|
|
* Align parameters if needed and then call isl_map_partial_lexopt_aligned.
|
|
|
|
|
*/
|
|
|
|
|
static __isl_give TYPE *SF(isl_map_partial_lexopt,SUFFIX)(
|
|
|
|
|
__isl_take isl_map *map, __isl_take isl_set *dom,
|
2018-07-03 21:52:50 +02:00
|
|
|
|
__isl_give isl_set **empty, unsigned flags)
|
2014-01-27 21:38:34 +01:00
|
|
|
|
{
|
2018-07-03 21:52:50 +02:00
|
|
|
|
isl_bool aligned;
|
|
|
|
|
|
|
|
|
|
aligned = isl_map_set_has_equal_params(map, dom);
|
|
|
|
|
if (aligned < 0)
|
2014-01-27 21:38:34 +01:00
|
|
|
|
goto error;
|
2018-07-03 21:52:50 +02:00
|
|
|
|
if (aligned)
|
2014-01-27 21:38:34 +01:00
|
|
|
|
return SF(isl_map_partial_lexopt_aligned,SUFFIX)(map, dom,
|
2018-07-03 21:52:50 +02:00
|
|
|
|
empty, flags);
|
2014-01-27 21:38:34 +01:00
|
|
|
|
if (!isl_space_has_named_params(map->dim) ||
|
|
|
|
|
!isl_space_has_named_params(dom->dim))
|
|
|
|
|
isl_die(map->ctx, isl_error_invalid,
|
|
|
|
|
"unaligned unnamed parameters", goto error);
|
|
|
|
|
map = isl_map_align_params(map, isl_map_get_space(dom));
|
|
|
|
|
dom = isl_map_align_params(dom, isl_map_get_space(map));
|
2018-07-03 21:52:50 +02:00
|
|
|
|
return SF(isl_map_partial_lexopt_aligned,SUFFIX)(map, dom, empty,
|
|
|
|
|
flags);
|
2014-01-27 21:38:34 +01:00
|
|
|
|
error:
|
|
|
|
|
if (empty)
|
|
|
|
|
*empty = NULL;
|
|
|
|
|
isl_set_free(dom);
|
|
|
|
|
isl_map_free(map);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-03 21:52:50 +02:00
|
|
|
|
/* Compute the lexicographic minimum (or maximum if "flags" includes
|
|
|
|
|
* ISL_OPT_MAX) of "map" over its domain.
|
|
|
|
|
*/
|
|
|
|
|
__isl_give TYPE *SF(isl_map_lexopt,SUFFIX)(__isl_take isl_map *map,
|
|
|
|
|
unsigned flags)
|
2014-01-27 21:38:34 +01:00
|
|
|
|
{
|
2018-07-03 21:52:50 +02:00
|
|
|
|
ISL_FL_SET(flags, ISL_OPT_FULL);
|
|
|
|
|
return SF(isl_map_partial_lexopt_aligned,SUFFIX)(map, NULL, NULL,
|
|
|
|
|
flags);
|
2014-01-27 21:38:34 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_map_lexmin,SUFFIX)(__isl_take isl_map *map)
|
|
|
|
|
{
|
|
|
|
|
return SF(isl_map_lexopt,SUFFIX)(map, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_map_lexmax,SUFFIX)(__isl_take isl_map *map)
|
|
|
|
|
{
|
2018-07-03 21:52:50 +02:00
|
|
|
|
return SF(isl_map_lexopt,SUFFIX)(map, ISL_OPT_MAX);
|
2014-01-27 21:38:34 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_set_lexmin,SUFFIX)(__isl_take isl_set *set)
|
|
|
|
|
{
|
|
|
|
|
return SF(isl_map_lexmin,SUFFIX)(set);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__isl_give TYPE *SF(isl_set_lexmax,SUFFIX)(__isl_take isl_set *set)
|
|
|
|
|
{
|
|
|
|
|
return SF(isl_map_lexmax,SUFFIX)(set);
|
|
|
|
|
}
|