Coverage Report

Created: 2017-06-23 12:40

/Users/buildslave/jenkins/sharedspace/clang-stage2-coverage-R@2/llvm/tools/polly/lib/External/isl/basis_reduction_tab.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright 2008-2009 Katholieke Universiteit Leuven
3
 *
4
 * Use of this software is governed by the MIT license
5
 *
6
 * Written by Sven Verdoolaege, K.U.Leuven, Departement
7
 * Computerwetenschappen, Celestijnenlaan 200A, B-3001 Leuven, Belgium
8
 */
9
10
#include <assert.h>
11
#include <isl_map_private.h>
12
#include <isl_seq.h>
13
#include "isl_tab.h"
14
#include <isl_int.h>
15
#include <isl_config.h>
16
17
struct tab_lp {
18
  struct isl_ctx  *ctx;
19
  struct isl_vec  *row;
20
  struct isl_tab  *tab;
21
  struct isl_tab_undo **stack;
22
  isl_int   *obj;
23
  isl_int    opt;
24
  isl_int    opt_denom;
25
  isl_int    tmp;
26
  isl_int    tmp2;
27
  int          neq;
28
  unsigned   dim;
29
  /* number of constraints in initial product tableau */
30
  int    con_offset;
31
  /* objective function has fixed or no integer value */
32
  int    is_fixed;
33
};
34
35
#ifdef USE_GMP_FOR_MP
36
#define GBR_type              mpq_t
37
#define GBR_init(v)             mpq_init(v)
38
#define GBR_clear(v)              mpq_clear(v)
39
#define GBR_set(a,b)          mpq_set(a,b)
40
#define GBR_set_ui(a,b)         mpq_set_ui(a,b,1)
41
#define GBR_mul(a,b,c)          mpq_mul(a,b,c)
42
#define GBR_lt(a,b)         (mpq_cmp(a,b) < 0)
43
#define GBR_is_zero(a)          (mpq_sgn(a) == 0)
44
#define GBR_numref(a)         mpq_numref(a)
45
#define GBR_denref(a)         mpq_denref(a)
46
#define GBR_floor(a,b)          mpz_fdiv_q(a,GBR_numref(b),GBR_denref(b))
47
#define GBR_ceil(a,b)         mpz_cdiv_q(a,GBR_numref(b),GBR_denref(b))
48
#define GBR_set_num_neg(a, b)       mpz_neg(GBR_numref(*a), b);
49
#define GBR_set_den(a, b)       mpz_set(GBR_denref(*a), b);
50
#endif /* USE_GMP_FOR_MP */
51
52
#ifdef USE_IMATH_FOR_MP
53
#include <imrat.h>
54
55
26.4k
#define GBR_type              mp_rat
56
79.4k
#define GBR_init(v)             v = mp_rat_alloc()
57
79.4k
#define GBR_clear(v)              mp_rat_free(v)
58
70.3k
#define GBR_set(a,b)          mp_rat_copy(b,a)
59
45.3k
#define GBR_set_ui(a,b)         mp_rat_set_uvalue(a,b,1)
60
38.7k
#define GBR_mul(a,b,c)          mp_rat_mul(b,c,a)
61
35.3k
#define GBR_lt(a,b)         (mp_rat_compare(a,b) < 0)
62
0
#define GBR_is_zero(a)          (mp_rat_compare_zero(a) == 0)
63
#ifdef USE_SMALL_INT_OPT
64
38.7k
#define GBR_numref(a) isl_sioimath_encode_big(mp_rat_numer_ref(a))
65
38.7k
#define GBR_denref(a) isl_sioimath_encode_big(mp_rat_denom_ref(a))
66
19.3k
#define GBR_floor(a, b) 
isl_sioimath_fdiv_q((a), 19.3k
GBR_numref19.3k
(b),
GBR_denref19.3k
(b))
67
19.3k
#define GBR_ceil(a, b)  
isl_sioimath_cdiv_q((a), 19.3k
GBR_numref19.3k
(b),
GBR_denref19.3k
(b))
68
#define GBR_set_num_neg(a, b)                              \
69
94.6k
  
do 94.6k
{ \94.6k
70
94.6k
    isl_sioimath_scratchspace_t scratch;       \
71
94.6k
    impz_neg(mp_rat_numer_ref(*a),             \
72
94.6k
        isl_sioimath_bigarg_src(*b, &scratch));\
73
94.6k
  } while (0)
74
#define GBR_set_den(a, b)                                  \
75
94.6k
  
do 94.6k
{ \94.6k
76
94.6k
    isl_sioimath_scratchspace_t scratch;       \
77
94.6k
    impz_set(mp_rat_denom_ref(*a),             \
78
94.6k
        isl_sioimath_bigarg_src(*b, &scratch));\
79
94.6k
  } while (0)
80
#else /* USE_SMALL_INT_OPT */
81
#define GBR_numref(a)   mp_rat_numer_ref(a)
82
#define GBR_denref(a)   mp_rat_denom_ref(a)
83
#define GBR_floor(a,b)    impz_fdiv_q(a,GBR_numref(b),GBR_denref(b))
84
#define GBR_ceil(a,b)   impz_cdiv_q(a,GBR_numref(b),GBR_denref(b))
85
#define GBR_set_num_neg(a, b) impz_neg(GBR_numref(*a), b)
86
#define GBR_set_den(a, b) impz_set(GBR_denref(*a), b)
87
#endif /* USE_SMALL_INT_OPT */
88
#endif /* USE_IMATH_FOR_MP */
89
90
static struct tab_lp *init_lp(struct isl_tab *tab);
91
static void set_lp_obj(struct tab_lp *lp, isl_int *row, int dim);
92
static int solve_lp(struct tab_lp *lp);
93
static void get_obj_val(struct tab_lp* lp, GBR_type *F);
94
static void delete_lp(struct tab_lp *lp);
95
static int add_lp_row(struct tab_lp *lp, isl_int *row, int dim);
96
static void get_alpha(struct tab_lp* lp, int row, GBR_type *alpha);
97
static int del_lp_row(struct tab_lp *lp) WARN_UNUSED;
98
static int cut_lp_to_hyperplane(struct tab_lp *lp, isl_int *row);
99
100
3.30k
#define GBR_LP                struct tab_lp
101
3.30k
#define GBR_lp_init(P)              init_lp(P)
102
32.6k
#define GBR_lp_set_obj(lp, obj, dim)      set_lp_obj(lp, obj, dim)
103
32.6k
#define GBR_lp_solve(lp)        solve_lp(lp)
104
32.6k
#define GBR_lp_get_obj_val(lp, F)     get_obj_val(lp, F)
105
3.30k
#define GBR_lp_delete(lp)       delete_lp(lp)
106
6.28k
#define GBR_lp_next_row(lp)       lp->neq
107
21.6k
#define GBR_lp_add_row(lp, row, dim)      add_lp_row(lp, row, dim)
108
61.9k
#define GBR_lp_get_alpha(lp, row, alpha)    get_alpha(lp, row, alpha)
109
19.3k
#define GBR_lp_del_row(lp)        del_lp_row(lp)
110
29.3k
#define GBR_lp_is_fixed(lp)       (lp)->is_fixed
111
0
#define GBR_lp_cut(lp, obj)           cut_lp_to_hyperplane(lp, obj)
112
#include "basis_reduction_templ.c"
113
114
/* Set up a tableau for the Cartesian product of bset with itself.
115
 * This could be optimized by first setting up a tableau for bset
116
 * and then performing the Cartesian product on the tableau.
117
 */
118
static struct isl_tab *gbr_tab(struct isl_tab *tab, struct isl_vec *row)
119
3.30k
{
120
3.30k
  unsigned dim;
121
3.30k
  struct isl_tab *prod;
122
3.30k
123
3.30k
  if (
!tab || 3.30k
!row3.30k
)
124
0
    return NULL;
125
3.30k
126
3.30k
  dim = tab->n_var;
127
3.30k
  prod = isl_tab_product(tab, tab);
128
3.30k
  if (
isl_tab_extend_cons(prod, 3 * dim + 1) < 03.30k
)
{0
129
0
    isl_tab_free(prod);
130
0
    return NULL;
131
0
  }
132
3.30k
  return prod;
133
3.30k
}
134
135
static struct tab_lp *init_lp(struct isl_tab *tab)
136
3.30k
{
137
3.30k
  struct tab_lp *lp = NULL;
138
3.30k
139
3.30k
  if (!tab)
140
0
    return NULL;
141
3.30k
142
3.30k
  
lp = 3.30k
isl_calloc_type3.30k
(tab->mat->ctx, struct tab_lp);
143
3.30k
  if (!lp)
144
0
    return NULL;
145
3.30k
146
3.30k
  
isl_int_init3.30k
(lp->opt);3.30k
147
3.30k
  isl_int_init(lp->opt_denom);
148
3.30k
  isl_int_init(lp->tmp);
149
3.30k
  isl_int_init(lp->tmp2);
150
3.30k
151
3.30k
  lp->dim = tab->n_var;
152
3.30k
153
3.30k
  lp->ctx = tab->mat->ctx;
154
3.30k
  isl_ctx_ref(lp->ctx);
155
3.30k
156
3.30k
  lp->stack = isl_alloc_array(lp->ctx, struct isl_tab_undo *, lp->dim);
157
3.30k
158
3.30k
  lp->row = isl_vec_alloc(lp->ctx, 1 + 2 * lp->dim);
159
3.30k
  if (!lp->row)
160
0
    goto error;
161
3.30k
  lp->tab = gbr_tab(tab, lp->row);
162
3.30k
  if (!lp->tab)
163
0
    goto error;
164
3.30k
  lp->con_offset = lp->tab->n_con;
165
3.30k
  lp->obj = NULL;
166
3.30k
  lp->neq = 0;
167
3.30k
168
3.30k
  return lp;
169
0
error:
170
0
  delete_lp(lp);
171
0
  return NULL;
172
3.30k
}
173
174
static void set_lp_obj(struct tab_lp *lp, isl_int *row, int dim)
175
32.6k
{
176
32.6k
  lp->obj = row;
177
32.6k
}
178
179
static int solve_lp(struct tab_lp *lp)
180
32.6k
{
181
32.6k
  enum isl_lp_result res;
182
32.6k
  unsigned flags = 0;
183
32.6k
184
32.6k
  lp->is_fixed = 0;
185
32.6k
186
32.6k
  isl_int_set_si(lp->row->el[0], 0);
187
32.6k
  isl_seq_cpy(lp->row->el + 1, lp->obj, lp->dim);
188
32.6k
  isl_seq_neg(lp->row->el + 1 + lp->dim, lp->obj, lp->dim);
189
32.6k
  if (lp->neq)
190
21.6k
    
flags = 21.6k
ISL_TAB_SAVE_DUAL21.6k
;
191
32.6k
  res = isl_tab_min(lp->tab, lp->row->el, lp->ctx->one,
192
32.6k
        &lp->opt, &lp->opt_denom, flags);
193
32.6k
  isl_int_mul_ui(lp->opt_denom, lp->opt_denom, 2);
194
32.6k
  if (
isl_int_abs_lt32.6k
(lp->opt, lp->opt_denom))
{12.6k
195
12.6k
    struct isl_vec *sample = isl_tab_get_sample_value(lp->tab);
196
12.6k
    if (!sample)
197
0
      return -1;
198
12.6k
    isl_seq_inner_product(lp->obj, sample->el + 1, lp->dim, &lp->tmp);
199
12.6k
    isl_seq_inner_product(lp->obj, sample->el + 1 + lp->dim, lp->dim, &lp->tmp2);
200
12.6k
    isl_int_cdiv_q(lp->tmp, lp->tmp, sample->el[0]);
201
12.6k
    isl_int_fdiv_q(lp->tmp2, lp->tmp2, sample->el[0]);
202
12.6k
    if (isl_int_ge(lp->tmp, lp->tmp2))
203
9.23k
      lp->is_fixed = 1;
204
12.6k
    isl_vec_free(sample);
205
12.6k
  }
206
32.6k
  
isl_int_divexact_ui32.6k
(lp->opt_denom, lp->opt_denom, 2);32.6k
207
32.6k
  if (res < 0)
208
0
    return -1;
209
32.6k
  
if (32.6k
res != isl_lp_ok32.6k
)
210
0
    isl_die(lp->ctx, isl_error_internal,
211
32.6k
      "unexpected missing (bounded) solution", return -1);
212
32.6k
  return 0;
213
32.6k
}
214
215
/* The current objective function has a fixed (or no) integer value.
216
 * Cut the tableau to the hyperplane that fixes this value in
217
 * both halves of the tableau.
218
 * Return 1 if the resulting tableau is empty.
219
 */
220
static int cut_lp_to_hyperplane(struct tab_lp *lp, isl_int *row)
221
0
{
222
0
  enum isl_lp_result res;
223
0
224
0
  isl_int_set_si(lp->row->el[0], 0);
225
0
  isl_seq_cpy(lp->row->el + 1, row, lp->dim);
226
0
  isl_seq_clr(lp->row->el + 1 + lp->dim, lp->dim);
227
0
  res = isl_tab_min(lp->tab, lp->row->el, lp->ctx->one,
228
0
        &lp->tmp, NULL, 0);
229
0
  if (res != isl_lp_ok)
230
0
    return -1;
231
0
232
0
  
isl_int_neg0
(lp->row->el[0], lp->tmp);0
233
0
  if (isl_tab_add_eq(lp->tab, lp->row->el) < 0)
234
0
    return -1;
235
0
236
0
  isl_seq_cpy(lp->row->el + 1 + lp->dim, row, lp->dim);
237
0
  isl_seq_clr(lp->row->el + 1, lp->dim);
238
0
  if (isl_tab_add_eq(lp->tab, lp->row->el) < 0)
239
0
    return -1;
240
0
241
0
  lp->con_offset += 2;
242
0
243
0
  return lp->tab->empty;
244
0
}
245
246
static void get_obj_val(struct tab_lp* lp, GBR_type *F)
247
32.6k
{
248
32.6k
  GBR_set_num_neg(F, lp->opt);
249
32.6k
  GBR_set_den(F, lp->opt_denom);
250
32.6k
}
251
252
static void delete_lp(struct tab_lp *lp)
253
3.30k
{
254
3.30k
  if (!lp)
255
0
    return;
256
3.30k
257
3.30k
  
isl_int_clear3.30k
(lp->opt);3.30k
258
3.30k
  isl_int_clear(lp->opt_denom);
259
3.30k
  isl_int_clear(lp->tmp);
260
3.30k
  isl_int_clear(lp->tmp2);
261
3.30k
  isl_vec_free(lp->row);
262
3.30k
  free(lp->stack);
263
3.30k
  isl_tab_free(lp->tab);
264
3.30k
  isl_ctx_deref(lp->ctx);
265
3.30k
  free(lp);
266
3.30k
}
267
268
static int add_lp_row(struct tab_lp *lp, isl_int *row, int dim)
269
21.6k
{
270
21.6k
  lp->stack[lp->neq] = isl_tab_snap(lp->tab);
271
21.6k
272
21.6k
  isl_int_set_si(lp->row->el[0], 0);
273
21.6k
  isl_seq_cpy(lp->row->el + 1, row, lp->dim);
274
21.6k
  isl_seq_neg(lp->row->el + 1 + lp->dim, row, lp->dim);
275
21.6k
276
21.6k
  if (isl_tab_add_valid_eq(lp->tab, lp->row->el) < 0)
277
0
    return -1;
278
21.6k
279
21.6k
  return lp->neq++;
280
21.6k
}
281
282
static void get_alpha(struct tab_lp* lp, int row, GBR_type *alpha)
283
61.9k
{
284
61.9k
  row += lp->con_offset;
285
61.9k
  GBR_set_num_neg(alpha, lp->tab->dual->el[1 + row]);
286
61.9k
  GBR_set_den(alpha, lp->tab->dual->el[0]);
287
61.9k
}
288
289
static int del_lp_row(struct tab_lp *lp)
290
19.3k
{
291
19.3k
  lp->neq--;
292
19.3k
  return isl_tab_rollback(lp->tab, lp->stack[lp->neq]);
293
19.3k
}