llvm-project/polly/lib/External/isl/isl_list_templ.c

612 lines
14 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright 2008-2009 Katholieke Universiteit Leuven
* Copyright 2011 INRIA Saclay
* Copyright 2012-2013 Ecole Normale Superieure
* Copyright 2017 Sven Verdoolaege
*
* Use of this software is governed by the MIT license
*
* Written by Sven Verdoolaege, K.U.Leuven, Departement
* Computerwetenschappen, Celestijnenlaan 200A, B-3001 Leuven, Belgium
* and 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 dUlm, 75230 Paris, France
*/
#include <isl_sort.h>
#include <isl_tarjan.h>
#define xCAT(A,B) A ## B
#define CAT(A,B) xCAT(A,B)
#undef EL
#define EL CAT(isl_,BASE)
#define xFN(TYPE,NAME) TYPE ## _ ## NAME
#define FN(TYPE,NAME) xFN(TYPE,NAME)
#define xLIST(EL) EL ## _list
#define LIST(EL) xLIST(EL)
#define xS(TYPE,NAME) struct TYPE ## _ ## NAME
#define S(TYPE,NAME) xS(TYPE,NAME)
isl_ctx *FN(LIST(EL),get_ctx)(__isl_keep LIST(EL) *list)
{
return list ? list->ctx : NULL;
}
__isl_give LIST(EL) *FN(LIST(EL),alloc)(isl_ctx *ctx, int n)
{
LIST(EL) *list;
if (n < 0)
isl_die(ctx, isl_error_invalid,
"cannot create list of negative length",
return NULL);
list = isl_alloc(ctx, LIST(EL),
sizeof(LIST(EL)) + (n - 1) * sizeof(struct EL *));
if (!list)
return NULL;
list->ctx = ctx;
isl_ctx_ref(ctx);
list->ref = 1;
list->size = n;
list->n = 0;
return list;
}
__isl_give LIST(EL) *FN(LIST(EL),copy)(__isl_keep LIST(EL) *list)
{
if (!list)
return NULL;
list->ref++;
return list;
}
__isl_give LIST(EL) *FN(LIST(EL),dup)(__isl_keep LIST(EL) *list)
{
int i;
LIST(EL) *dup;
if (!list)
return NULL;
dup = FN(LIST(EL),alloc)(FN(LIST(EL),get_ctx)(list), list->n);
if (!dup)
return NULL;
for (i = 0; i < list->n; ++i)
dup = FN(LIST(EL),add)(dup, FN(EL,copy)(list->p[i]));
return dup;
}
__isl_give LIST(EL) *FN(LIST(EL),cow)(__isl_take LIST(EL) *list)
{
if (!list)
return NULL;
if (list->ref == 1)
return list;
list->ref--;
return FN(LIST(EL),dup)(list);
}
/* Make sure "list" has room for at least "n" more pieces.
* Always return a list with a single reference.
*
* If there is only one reference to list, we extend it in place.
* Otherwise, we create a new LIST(EL) and copy the elements.
*/
static __isl_give LIST(EL) *FN(LIST(EL),grow)(__isl_take LIST(EL) *list, int n)
{
isl_ctx *ctx;
int i, new_size;
LIST(EL) *res;
if (!list)
return NULL;
if (list->ref == 1 && list->n + n <= list->size)
return list;
ctx = FN(LIST(EL),get_ctx)(list);
new_size = ((list->n + n + 1) * 3) / 2;
if (list->ref == 1) {
res = isl_realloc(ctx, list, LIST(EL),
sizeof(LIST(EL)) + (new_size - 1) * sizeof(EL *));
if (!res)
return FN(LIST(EL),free)(list);
res->size = new_size;
return res;
}
if (list->n + n <= list->size && list->size < new_size)
new_size = list->size;
res = FN(LIST(EL),alloc)(ctx, new_size);
if (!res)
return FN(LIST(EL),free)(list);
for (i = 0; i < list->n; ++i)
res = FN(LIST(EL),add)(res, FN(EL,copy)(list->p[i]));
FN(LIST(EL),free)(list);
return res;
}
/* Check that "index" is a valid position in "list".
*/
static isl_stat FN(LIST(EL),check_index)(__isl_keep LIST(EL) *list, int index)
{
if (!list)
return isl_stat_error;
if (index < 0 || index >= list->n)
isl_die(FN(LIST(EL),get_ctx)(list), isl_error_invalid,
"index out of bounds", return isl_stat_error);
return isl_stat_ok;
}
__isl_give LIST(EL) *FN(LIST(EL),add)(__isl_take LIST(EL) *list,
__isl_take struct EL *el)
{
list = FN(LIST(EL),grow)(list, 1);
if (!list || !el)
goto error;
list->p[list->n] = el;
list->n++;
return list;
error:
FN(EL,free)(el);
FN(LIST(EL),free)(list);
return NULL;
}
/* Remove the "n" elements starting at "first" from "list".
*/
__isl_give LIST(EL) *FN(LIST(EL),drop)(__isl_take LIST(EL) *list,
unsigned first, unsigned n)
{
int i;
if (!list)
return NULL;
if (first + n > list->n || first + n < first)
isl_die(list->ctx, isl_error_invalid,
"index out of bounds", return FN(LIST(EL),free)(list));
if (n == 0)
return list;
list = FN(LIST(EL),cow)(list);
if (!list)
return NULL;
for (i = 0; i < n; ++i)
FN(EL,free)(list->p[first + i]);
for (i = first; i + n < list->n; ++i)
list->p[i] = list->p[i + n];
list->n -= n;
return list;
}
/* Insert "el" at position "pos" in "list".
*
* If there is only one reference to "list" and if it already has space
* for one extra element, we insert it directly into "list".
* Otherwise, we create a new list consisting of "el" and copied
* elements from "list".
*/
__isl_give LIST(EL) *FN(LIST(EL),insert)(__isl_take LIST(EL) *list,
unsigned pos, __isl_take struct EL *el)
{
int i;
isl_ctx *ctx;
LIST(EL) *res;
if (!list || !el)
goto error;
ctx = FN(LIST(EL),get_ctx)(list);
if (pos > list->n)
isl_die(ctx, isl_error_invalid,
"index out of bounds", goto error);
if (list->ref == 1 && list->size > list->n) {
for (i = list->n; i > pos; --i)
list->p[i] = list->p[i - 1];
list->n++;
list->p[pos] = el;
return list;
}
res = FN(LIST(EL),alloc)(ctx, list->n + 1);
for (i = 0; i < pos; ++i)
res = FN(LIST(EL),add)(res, FN(EL,copy)(list->p[i]));
res = FN(LIST(EL),add)(res, el);
for (i = pos; i < list->n; ++i)
res = FN(LIST(EL),add)(res, FN(EL,copy)(list->p[i]));
FN(LIST(EL),free)(list);
return res;
error:
FN(EL,free)(el);
FN(LIST(EL),free)(list);
return NULL;
}
__isl_null LIST(EL) *FN(LIST(EL),free)(__isl_take LIST(EL) *list)
{
int i;
if (!list)
return NULL;
if (--list->ref > 0)
return NULL;
isl_ctx_deref(list->ctx);
for (i = 0; i < list->n; ++i)
FN(EL,free)(list->p[i]);
free(list);
return NULL;
}
int FN(FN(LIST(EL),n),BASE)(__isl_keep LIST(EL) *list)
{
return list ? list->n : 0;
}
__isl_give EL *FN(FN(LIST(EL),get),BASE)(__isl_keep LIST(EL) *list, int index)
{
if (FN(LIST(EL),check_index)(list, index) < 0)
return NULL;
return FN(EL,copy)(list->p[index]);
}
/* Replace the element at position "index" in "list" by "el".
*/
__isl_give LIST(EL) *FN(FN(LIST(EL),set),BASE)(__isl_take LIST(EL) *list,
int index, __isl_take EL *el)
{
if (!list || !el)
goto error;
if (FN(LIST(EL),check_index)(list, index) < 0)
goto error;
if (list->p[index] == el) {
FN(EL,free)(el);
return list;
}
list = FN(LIST(EL),cow)(list);
if (!list)
goto error;
FN(EL,free)(list->p[index]);
list->p[index] = el;
return list;
error:
FN(EL,free)(el);
FN(LIST(EL),free)(list);
return NULL;
}
/* Return the element at position "index" of "list".
* This may be either a copy or the element itself
* if there is only one reference to "list".
* This allows the element to be modified inplace
* if both the list and the element have only a single reference.
* The caller is not allowed to modify "list" between
* this call to isl_list_*_take_* and a subsequent call
* to isl_list_*_restore_*.
* The only exception is that isl_list_*_free can be called instead.
*/
static __isl_give EL *FN(FN(LIST(EL),take),BASE)(__isl_keep LIST(EL) *list,
int index)
{
EL *el;
if (FN(LIST(EL),check_index)(list, index) < 0)
return NULL;
if (list->ref != 1)
return FN(FN(LIST(EL),get),BASE)(list, index);
el = list->p[index];
list->p[index] = NULL;
return el;
}
/* Set the element at position "index" of "list" to "el",
* where the position may be empty due to a previous call
* to isl_list_*_take_*.
*/
static __isl_give LIST(EL) *FN(FN(LIST(EL),restore),BASE)(
__isl_take LIST(EL) *list, int index, __isl_take EL *el)
{
return FN(FN(LIST(EL),set),BASE)(list, index, el);
}
isl_stat FN(LIST(EL),foreach)(__isl_keep LIST(EL) *list,
isl_stat (*fn)(__isl_take EL *el, void *user), void *user)
{
int i;
if (!list)
return isl_stat_error;
for (i = 0; i < list->n; ++i) {
EL *el = FN(EL,copy)(list->p[i]);
if (!el)
return isl_stat_error;
if (fn(el, user) < 0)
return isl_stat_error;
}
return isl_stat_ok;
}
/* Replace each element in "list" by the result of calling "fn"
* on the element.
*/
__isl_give LIST(EL) *FN(LIST(EL),map)(__isl_keep LIST(EL) *list,
__isl_give EL *(*fn)(__isl_take EL *el, void *user), void *user)
{
int i, n;
if (!list)
return NULL;
n = list->n;
for (i = 0; i < n; ++i) {
EL *el = FN(FN(LIST(EL),take),BASE)(list, i);
if (!el)
return FN(LIST(EL),free)(list);
el = fn(el, user);
list = FN(FN(LIST(EL),restore),BASE)(list, i, el);
}
return list;
}
/* Internal data structure for isl_*_list_sort.
*
* "cmp" is the original comparison function.
* "user" is a user provided pointer that should be passed to "cmp".
*/
S(LIST(EL),sort_data) {
int (*cmp)(__isl_keep EL *a, __isl_keep EL *b, void *user);
void *user;
};
/* Compare two entries of an isl_*_list based on the user provided
* comparison function on pairs of isl_* objects.
*/
static int FN(LIST(EL),cmp)(const void *a, const void *b, void *user)
{
S(LIST(EL),sort_data) *data = user;
EL * const *el1 = a;
EL * const *el2 = b;
return data->cmp(*el1, *el2, data->user);
}
/* Sort the elements of "list" in ascending order according to
* comparison function "cmp".
*/
__isl_give LIST(EL) *FN(LIST(EL),sort)(__isl_take LIST(EL) *list,
int (*cmp)(__isl_keep EL *a, __isl_keep EL *b, void *user), void *user)
{
S(LIST(EL),sort_data) data = { cmp, user };
if (!list)
return NULL;
if (list->n <= 1)
return list;
list = FN(LIST(EL),cow)(list);
if (!list)
return NULL;
if (isl_sort(list->p, list->n, sizeof(list->p[0]),
&FN(LIST(EL),cmp), &data) < 0)
return FN(LIST(EL),free)(list);
return list;
}
/* Internal data structure for isl_*_list_foreach_scc.
*
* "list" is the original list.
* "follows" is the user provided callback that defines the edges of the graph.
*/
S(LIST(EL),foreach_scc_data) {
LIST(EL) *list;
isl_bool (*follows)(__isl_keep EL *a, __isl_keep EL *b, void *user);
void *follows_user;
};
/* Does element i of data->list follow element j?
*
* Use the user provided callback to find out.
*/
static isl_bool FN(LIST(EL),follows)(int i, int j, void *user)
{
S(LIST(EL),foreach_scc_data) *data = user;
return data->follows(data->list->p[i], data->list->p[j],
data->follows_user);
}
/* Call "fn" on the sublist of "list" that consists of the elements
* with indices specified by the "n" elements of "pos".
*/
static isl_stat FN(LIST(EL),call_on_scc)(__isl_keep LIST(EL) *list, int *pos,
int n, isl_stat (*fn)(__isl_take LIST(EL) *scc, void *user), void *user)
{
int i;
isl_ctx *ctx;
LIST(EL) *slice;
ctx = FN(LIST(EL),get_ctx)(list);
slice = FN(LIST(EL),alloc)(ctx, n);
for (i = 0; i < n; ++i) {
EL *el;
el = FN(EL,copy)(list->p[pos[i]]);
slice = FN(LIST(EL),add)(slice, el);
}
return fn(slice, user);
}
/* Call "fn" on each of the strongly connected components (SCCs) of
* the graph with as vertices the elements of "list" and
* a directed edge from node b to node a iff follows(a, b)
* returns 1. follows should return -1 on error.
*
* If SCC a contains a node i that follows a node j in another SCC b
* (i.e., follows(i, j, user) returns 1), then fn will be called on SCC a
* after being called on SCC b.
*
* We simply call isl_tarjan_graph_init, extract the SCCs from the result and
* call fn on each of them.
*/
isl_stat FN(LIST(EL),foreach_scc)(__isl_keep LIST(EL) *list,
isl_bool (*follows)(__isl_keep EL *a, __isl_keep EL *b, void *user),
void *follows_user,
isl_stat (*fn)(__isl_take LIST(EL) *scc, void *user), void *fn_user)
{
S(LIST(EL),foreach_scc_data) data = { list, follows, follows_user };
int i, n;
isl_ctx *ctx;
struct isl_tarjan_graph *g;
if (!list)
return isl_stat_error;
if (list->n == 0)
return isl_stat_ok;
if (list->n == 1)
return fn(FN(LIST(EL),copy)(list), fn_user);
ctx = FN(LIST(EL),get_ctx)(list);
n = list->n;
g = isl_tarjan_graph_init(ctx, n, &FN(LIST(EL),follows), &data);
if (!g)
return isl_stat_error;
i = 0;
do {
int first;
if (g->order[i] == -1)
isl_die(ctx, isl_error_internal, "cannot happen",
break);
first = i;
while (g->order[i] != -1) {
++i; --n;
}
if (first == 0 && n == 0) {
isl_tarjan_graph_free(g);
return fn(FN(LIST(EL),copy)(list), fn_user);
}
if (FN(LIST(EL),call_on_scc)(list, g->order + first, i - first,
fn, fn_user) < 0)
break;
++i;
} while (n);
isl_tarjan_graph_free(g);
return n > 0 ? isl_stat_error : isl_stat_ok;
}
__isl_give LIST(EL) *FN(FN(LIST(EL),from),BASE)(__isl_take EL *el)
{
isl_ctx *ctx;
LIST(EL) *list;
if (!el)
return NULL;
ctx = FN(EL,get_ctx)(el);
list = FN(LIST(EL),alloc)(ctx, 1);
if (!list)
goto error;
list = FN(LIST(EL),add)(list, el);
return list;
error:
FN(EL,free)(el);
return NULL;
}
/* Append the elements of "list2" to "list1", where "list1" is known
* to have only a single reference and enough room to hold
* the extra elements.
*/
static __isl_give LIST(EL) *FN(LIST(EL),concat_inplace)(
__isl_take LIST(EL) *list1, __isl_take LIST(EL) *list2)
{
int i;
for (i = 0; i < list2->n; ++i)
list1 = FN(LIST(EL),add)(list1, FN(EL,copy)(list2->p[i]));
FN(LIST(EL),free)(list2);
return list1;
}
/* Concatenate "list1" and "list2".
* If "list1" has only one reference and has enough room
* for the elements of "list2", the add the elements to "list1" itself.
* Otherwise, create a new list to store the result.
*/
__isl_give LIST(EL) *FN(LIST(EL),concat)(__isl_take LIST(EL) *list1,
__isl_take LIST(EL) *list2)
{
int i;
isl_ctx *ctx;
LIST(EL) *res;
if (!list1 || !list2)
goto error;
if (list1->ref == 1 && list1->n + list2->n <= list1->size)
return FN(LIST(EL),concat_inplace)(list1, list2);
ctx = FN(LIST(EL),get_ctx)(list1);
res = FN(LIST(EL),alloc)(ctx, list1->n + list2->n);
for (i = 0; i < list1->n; ++i)
res = FN(LIST(EL),add)(res, FN(EL,copy)(list1->p[i]));
for (i = 0; i < list2->n; ++i)
res = FN(LIST(EL),add)(res, FN(EL,copy)(list2->p[i]));
FN(LIST(EL),free)(list1);
FN(LIST(EL),free)(list2);
return res;
error:
FN(LIST(EL),free)(list1);
FN(LIST(EL),free)(list2);
return NULL;
}
__isl_give isl_printer *CAT(isl_printer_print_,LIST(BASE))(
__isl_take isl_printer *p, __isl_keep LIST(EL) *list)
{
int i;
if (!p || !list)
goto error;
p = isl_printer_print_str(p, "(");
for (i = 0; i < list->n; ++i) {
if (i)
p = isl_printer_print_str(p, ",");
p = CAT(isl_printer_print_,BASE)(p, list->p[i]);
}
p = isl_printer_print_str(p, ")");
return p;
error:
isl_printer_free(p);
return NULL;
}
void FN(LIST(EL),dump)(__isl_keep LIST(EL) *list)
{
isl_printer *printer;
if (!list)
return;
printer = isl_printer_to_file(FN(LIST(EL),get_ctx)(list), stderr);
printer = CAT(isl_printer_print_,LIST(BASE))(printer, list);
printer = isl_printer_end_line(printer);
isl_printer_free(printer);
}