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