Home | History | Annotate | Line # | Download | only in dist
      1  1.1  mrg /*
      2  1.1  mrg  * Copyright 2011      Sven Verdoolaege
      3  1.1  mrg  * Copyright 2012-2013 Ecole Normale Superieure
      4  1.1  mrg  *
      5  1.1  mrg  * Use of this software is governed by the MIT license
      6  1.1  mrg  *
      7  1.1  mrg  * Written by Sven Verdoolaege,
      8  1.1  mrg  * Ecole Normale Superieure, 45 rue d'Ulm, 75230 Paris, France
      9  1.1  mrg  */
     10  1.1  mrg 
     11  1.1  mrg #include <isl/space.h>
     12  1.1  mrg #include <isl_val_private.h>
     13  1.1  mrg 
     14  1.1  mrg #include <isl_multi_macro.h>
     15  1.1  mrg 
     16  1.1  mrg /* Add "multi2" to "multi1" and return the result.
     17  1.1  mrg  */
     18  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),add)(__isl_take MULTI(BASE) *multi1,
     19  1.1  mrg 	__isl_take MULTI(BASE) *multi2)
     20  1.1  mrg {
     21  1.1  mrg 	return FN(MULTI(BASE),bin_op)(multi1, multi2, &FN(EL,add));
     22  1.1  mrg }
     23  1.1  mrg 
     24  1.1  mrg /* Subtract "multi2" from "multi1" and return the result.
     25  1.1  mrg  */
     26  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),sub)(__isl_take MULTI(BASE) *multi1,
     27  1.1  mrg 	__isl_take MULTI(BASE) *multi2)
     28  1.1  mrg {
     29  1.1  mrg 	return FN(MULTI(BASE),bin_op)(multi1, multi2, &FN(EL,sub));
     30  1.1  mrg }
     31  1.1  mrg 
     32  1.1  mrg /* Depending on "fn", multiply or divide the elements of "multi" by "v" and
     33  1.1  mrg  * return the result.
     34  1.1  mrg  */
     35  1.1  mrg static __isl_give MULTI(BASE) *FN(MULTI(BASE),scale_val_fn)(
     36  1.1  mrg 	__isl_take MULTI(BASE) *multi, __isl_take isl_val *v,
     37  1.1  mrg 	__isl_give EL *(*fn)(__isl_take EL *el, __isl_take isl_val *v))
     38  1.1  mrg {
     39  1.1  mrg 	if (!multi || !v)
     40  1.1  mrg 		goto error;
     41  1.1  mrg 
     42  1.1  mrg 	if (isl_val_is_one(v)) {
     43  1.1  mrg 		isl_val_free(v);
     44  1.1  mrg 		return multi;
     45  1.1  mrg 	}
     46  1.1  mrg 
     47  1.1  mrg 	if (!isl_val_is_rat(v))
     48  1.1  mrg 		isl_die(isl_val_get_ctx(v), isl_error_invalid,
     49  1.1  mrg 			"expecting rational factor", goto error);
     50  1.1  mrg 
     51  1.1  mrg 	return FN(MULTI(BASE),fn_val)(multi, fn, v);
     52  1.1  mrg error:
     53  1.1  mrg 	isl_val_free(v);
     54  1.1  mrg 	return FN(MULTI(BASE),free)(multi);
     55  1.1  mrg }
     56  1.1  mrg 
     57  1.1  mrg /* Multiply the elements of "multi" by "v" and return the result.
     58  1.1  mrg  */
     59  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),scale_val)(__isl_take MULTI(BASE) *multi,
     60  1.1  mrg 	__isl_take isl_val *v)
     61  1.1  mrg {
     62  1.1  mrg 	return FN(MULTI(BASE),scale_val_fn)(multi, v, &FN(EL,scale_val));
     63  1.1  mrg }
     64  1.1  mrg 
     65  1.1  mrg /* Divide the elements of "multi" by "v" and return the result.
     66  1.1  mrg  */
     67  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),scale_down_val)(
     68  1.1  mrg 	__isl_take MULTI(BASE) *multi, __isl_take isl_val *v)
     69  1.1  mrg {
     70  1.1  mrg 	if (!v)
     71  1.1  mrg 		goto error;
     72  1.1  mrg 	if (isl_val_is_zero(v))
     73  1.1  mrg 		isl_die(isl_val_get_ctx(v), isl_error_invalid,
     74  1.1  mrg 			"cannot scale down by zero", goto error);
     75  1.1  mrg 	return FN(MULTI(BASE),scale_val_fn)(multi, v, &FN(EL,scale_down_val));
     76  1.1  mrg error:
     77  1.1  mrg 	isl_val_free(v);
     78  1.1  mrg 	return FN(MULTI(BASE),free)(multi);
     79  1.1  mrg }
     80  1.1  mrg 
     81  1.1  mrg /* Multiply the elements of "multi" by the corresponding element of "mv"
     82  1.1  mrg  * and return the result.
     83  1.1  mrg  */
     84  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),scale_multi_val)(
     85  1.1  mrg 	__isl_take MULTI(BASE) *multi, __isl_take isl_multi_val *mv)
     86  1.1  mrg {
     87  1.1  mrg 	return FN(MULTI(BASE),fn_multi_val)(multi, &FN(EL,scale_val), mv);
     88  1.1  mrg }
     89  1.1  mrg 
     90  1.1  mrg /* Divide the elements of "multi" by the corresponding element of "mv"
     91  1.1  mrg  * and return the result.
     92  1.1  mrg  */
     93  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),scale_down_multi_val)(
     94  1.1  mrg 	__isl_take MULTI(BASE) *multi, __isl_take isl_multi_val *mv)
     95  1.1  mrg {
     96  1.1  mrg 	return FN(MULTI(BASE),fn_multi_val)(multi, &FN(EL,scale_down_val), mv);
     97  1.1  mrg }
     98  1.1  mrg 
     99  1.1  mrg /* Compute the residues of the elements of "multi" modulo
    100  1.1  mrg  * the corresponding element of "mv" and return the result.
    101  1.1  mrg  */
    102  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),mod_multi_val)(
    103  1.1  mrg 	__isl_take MULTI(BASE) *multi, __isl_take isl_multi_val *mv)
    104  1.1  mrg {
    105  1.1  mrg 	return FN(MULTI(BASE),fn_multi_val)(multi, &FN(EL,mod_val), mv);
    106  1.1  mrg }
    107  1.1  mrg 
    108  1.1  mrg /* Return the opposite of "multi".
    109  1.1  mrg  */
    110  1.1  mrg __isl_give MULTI(BASE) *FN(MULTI(BASE),neg)(__isl_take MULTI(BASE) *multi)
    111  1.1  mrg {
    112  1.1  mrg 	S(MULTI(BASE),un_op_control) control = { .fn_el = &FN(EL,neg) };
    113  1.1  mrg 	return FN(MULTI(BASE),un_op)(multi, &control);
    114  1.1  mrg }
    115