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

523 lines
12 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
*
* 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;
}
__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 - 1; i >= pos; --i)
list->p[i + 1] = list->p[i];
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 (!list)
return NULL;
if (index < 0 || index >= list->n)
isl_die(list->ctx, isl_error_invalid,
"index out of bounds", 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 (index < 0 || index >= list->n)
isl_die(list->ctx, isl_error_invalid,
"index out of bounds", 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;
}
int FN(LIST(EL),foreach)(__isl_keep LIST(EL) *list,
int (*fn)(__isl_take EL *el, void *user), void *user)
{
int i;
if (!list)
return -1;
for (i = 0; i < list->n; ++i) {
EL *el = FN(EL,copy(list->p[i]));
if (!el)
return -1;
if (fn(el, user) < 0)
return -1;
}
return 0;
}
/* 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;
int (*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 int 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 int FN(LIST(EL),call_on_scc)(__isl_keep LIST(EL) *list, int *pos, int n,
int (*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.
*/
int FN(LIST(EL),foreach_scc)(__isl_keep LIST(EL) *list,
int (*follows)(__isl_keep EL *a, __isl_keep EL *b, void *user),
void *follows_user,
int (*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 -1;
if (list->n == 0)
return 0;
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 -1;
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 ? -1 : 0;
}
__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;
}
__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;
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);
}