test_bound: stop testing after predetermined number of instances
[barvinok/uuh.git] / test_bound.c
blob7bf856d5020d22579be38fc3c802f44f7864e6bd
1 #include <assert.h>
2 #include <limits.h>
3 #include <math.h>
4 #include <isl_stream.h>
5 #include <barvinok/options.h>
6 #include <barvinok/util.h>
7 #include "verify.h"
8 #include "config.h"
10 #ifdef HAVE_SYS_TIMES_H
12 #include <sys/times.h>
14 typedef clock_t my_clock_t;
16 static my_clock_t time_diff(struct tms *before, struct tms *after)
18 return after->tms_utime - before->tms_utime;
21 #else
23 typedef int my_clock_t;
25 struct tms {};
26 static void times(struct tms* time)
29 static my_clock_t time_diff(struct tms *before, struct tms *after)
31 return 0;
34 #endif
36 static struct {
37 int method;
38 } methods[] = {
39 { ISL_BOUND_BERNSTEIN },
40 { ISL_BOUND_RANGE },
43 #define nr_methods (sizeof(methods)/sizeof(*methods))
45 struct options {
46 struct verify_options *verify;
47 int quiet;
50 struct isl_arg options_arg[] = {
51 ISL_ARG_CHILD(struct options, verify, NULL, verify_options_arg, NULL)
52 ISL_ARG_BOOL(struct options, quiet, 'q', "quiet", 0, NULL)
53 ISL_ARG_END
56 ISL_ARG_DEF(options, struct options, options_arg)
58 struct result_data {
59 isl_int n;
60 double RE_sum[nr_methods];
62 my_clock_t ticks[nr_methods];
63 size_t size[nr_methods];
66 void result_data_init(struct result_data *result)
68 int i;
69 for (i = 0; i < nr_methods; ++i) {
70 result->RE_sum[i] = 0;
71 result->ticks[i] = 0;
72 result->size[i] = 0;
74 isl_int_init(result->n);
77 void result_data_clear(struct result_data *result)
79 int i;
80 isl_int_clear(result->n);
83 void result_data_print(struct result_data *result, int n)
85 int i;
87 fprintf(stderr, "%g", (double)result->ticks[0]/n);
88 for (i = 1; i < nr_methods; ++i)
89 fprintf(stderr, ", %g", (double)result->ticks[i]/n);
90 fprintf(stderr, "\n");
92 fprintf(stderr, "%zd/%d", result->size[0], n);
93 for (i = 1; i < nr_methods; ++i)
94 fprintf(stderr, ", %zd/%d", result->size[i], n);
95 fprintf(stderr, "\n");
97 fprintf(stderr, "%g\n", isl_int_get_d(result->n));
98 fprintf(stderr, "%g", result->RE_sum[0]/isl_int_get_d(result->n));
99 for (i = 1; i < nr_methods; ++i)
100 fprintf(stderr, ", %g", result->RE_sum[i]/isl_int_get_d(result->n));
101 fprintf(stderr, "\n");
104 struct verify_point_bound {
105 struct verify_point_data vpd;
106 struct result_data *result;
107 isl_pw_qpolynomial *pwqp;
108 isl_pw_qpolynomial_fold **pwf;
111 static int verify_point(__isl_take isl_point *pnt, void *user)
113 struct verify_point_bound *vpb = (struct verify_point_bound *) user;
114 const struct verify_options *options = vpb->vpd.options;
115 int i;
116 unsigned nparam;
117 isl_int max, min, exact, approx;
118 isl_int n, d;
119 isl_qpolynomial *opt;
120 isl_pw_qpolynomial *pwqp;
121 int cst;
123 vpb->vpd.n--;
125 isl_int_init(max);
126 isl_int_init(min);
127 isl_int_init(exact);
128 isl_int_init(approx);
129 isl_int_init(n);
130 isl_int_init(d);
132 pwqp = isl_pw_qpolynomial_copy(vpb->pwqp);
134 nparam = isl_pw_qpolynomial_dim(pwqp, isl_dim_param);
135 for (i = 0; i < nparam; ++i) {
136 isl_point_get_coordinate(pnt, isl_dim_param, i, &n);
137 pwqp = isl_pw_qpolynomial_fix_dim(pwqp, isl_dim_param, i, n);
140 opt = isl_pw_qpolynomial_max(isl_pw_qpolynomial_copy(pwqp));
141 cst = isl_qpolynomial_is_cst(opt, &n, &d);
142 isl_qpolynomial_free(opt);
143 assert(cst == 1);
144 isl_int_fdiv_q(max, n, d);
146 opt = isl_pw_qpolynomial_min(pwqp);
147 cst = isl_qpolynomial_is_cst(opt, &n, &d);
148 isl_qpolynomial_free(opt);
149 assert(cst == 1);
150 isl_int_cdiv_q(min, n, d);
152 isl_int_sub(exact, max, min);
153 isl_int_add_ui(exact, exact, 1);
155 if (options->print_all) {
156 fprintf(stderr, "max: "); isl_int_print(stderr, max, 0);
157 fprintf(stderr, ", min: "); isl_int_print(stderr, min, 0);
158 fprintf(stderr, ", range: "); isl_int_print(stderr, exact, 0);
161 for (i = 0; i < nr_methods; ++i) {
162 double error;
164 opt = isl_pw_qpolynomial_fold_eval(
165 isl_pw_qpolynomial_fold_copy(vpb->pwf[2 * i]),
166 isl_point_copy(pnt));
167 cst = isl_qpolynomial_is_cst(opt, &n, &d);
168 isl_qpolynomial_free(opt);
169 assert(cst == 1);
170 isl_int_fdiv_q(max, n, d);
172 opt = isl_pw_qpolynomial_fold_eval(
173 isl_pw_qpolynomial_fold_copy(vpb->pwf[2 * i + 1]),
174 isl_point_copy(pnt));
175 cst = isl_qpolynomial_is_cst(opt, &n, &d);
176 isl_qpolynomial_free(opt);
177 assert(cst == 1);
178 isl_int_cdiv_q(min, n, d);
180 isl_int_sub(approx, max, min);
181 isl_int_add_ui(approx, approx, 1);
182 if (options->print_all) {
183 fprintf(stderr, ", "); isl_int_print(stderr, approx, 0);
186 assert(isl_int_ge(approx, exact));
187 isl_int_sub(approx, approx, exact);
189 error = fabs(isl_int_get_d(approx)) / isl_int_get_d(exact);
190 if (options->print_all)
191 fprintf(stderr, " (%g)", error);
192 vpb->result->RE_sum[i] += error;
195 if (options->print_all) {
196 fprintf(stderr, "\n");
197 } else if ((vpb->vpd.n % vpb->vpd.s) == 0) {
198 printf("o");
199 fflush(stdout);
202 isl_int_clear(max);
203 isl_int_clear(min);
204 isl_int_clear(exact);
205 isl_int_clear(approx);
206 isl_int_clear(n);
207 isl_int_clear(d);
209 isl_point_free(pnt);
211 return vpb->vpd.n >= 1 ? 0 : -1;
214 static void test(__isl_keep isl_pw_qpolynomial *pwqp,
215 __isl_keep isl_pw_qpolynomial_fold **pwf, struct result_data *result,
216 struct verify_options *options)
218 struct verify_point_bound vpb = { { options }, result };
219 isl_set *context;
220 int r;
221 unsigned nvar;
223 vpb.pwf = pwf;
224 vpb.pwqp = pwqp;
225 context = isl_pw_qpolynomial_domain(isl_pw_qpolynomial_copy(vpb.pwqp));
226 nvar = isl_set_dim(context, isl_dim_set);
227 context = isl_set_remove_dims(context, isl_dim_set, 0, nvar);
228 context = verify_context_set_bounds(context, options);
230 r = verify_point_data_init(&vpb.vpd, context);
231 assert(r == 0);
232 isl_int_set_si(result->n, vpb.vpd.n);
234 isl_set_foreach_point(context, verify_point, &vpb);
235 assert(!vpb.vpd.error);
237 isl_set_free(context);
239 verify_point_data_fini(&vpb.vpd);
242 void handle(FILE *in, struct result_data *result, struct verify_options *options)
244 int i;
245 isl_ctx *ctx = isl_ctx_alloc();
246 isl_pw_qpolynomial *pwqp, *upper, *lower;
247 isl_pw_qpolynomial_fold *pwf[2*nr_methods];
248 struct isl_stream *s;
250 s = isl_stream_new_file(ctx, in);
251 pwqp = isl_stream_read_pw_qpolynomial(s);
253 upper = isl_pw_qpolynomial_to_polynomial(isl_pw_qpolynomial_copy(pwqp), 1);
254 lower = isl_pw_qpolynomial_to_polynomial(isl_pw_qpolynomial_copy(pwqp), -1);
256 for (i = 0; i < nr_methods; ++i) {
257 int j;
258 struct tms st_cpu;
259 struct tms en_cpu;
261 times(&st_cpu);
262 for (j = 0; j < 2; ++j) {
263 isl_pw_qpolynomial *poly = j == 0 ? upper : lower;
264 enum isl_fold type = j == 0 ? isl_fold_max : isl_fold_min;
265 options->barvinok->isl->bound = methods[i].method;
266 poly = isl_pw_qpolynomial_copy(poly);
267 pwf[2*i+j] = isl_pw_qpolynomial_bound(poly, type, NULL);
269 times(&en_cpu);
270 result->ticks[i] = time_diff(&en_cpu, &st_cpu);
271 result->size[i] = isl_pw_qpolynomial_fold_size(pwf[2*i]);
272 result->size[i] = isl_pw_qpolynomial_fold_size(pwf[2*i+1]);
273 if (options->barvinok->verbose) {
274 isl_printer *p = isl_printer_to_file(ctx, stdout);
275 for (j = 0; j < 2; ++j) {
276 p = isl_printer_print_pw_qpolynomial_fold(p, pwf[2*i+j]);
277 p = isl_printer_end_line(p);
279 isl_printer_free(p);
282 test(pwqp, pwf, result, options);
284 for (i = 0; i < 2*nr_methods; ++i)
285 isl_pw_qpolynomial_fold_free(pwf[i]);
286 isl_pw_qpolynomial_free(pwqp);
287 isl_pw_qpolynomial_free(lower);
288 isl_pw_qpolynomial_free(upper);
290 isl_stream_free(s);
291 isl_ctx_free(ctx);
294 int main(int argc, char **argv)
296 char path[PATH_MAX+1];
297 struct result_data all_result;
298 int n = 0;
299 struct options *options = options_new_with_defaults();
301 argc = options_parse(options, argc, argv, ISL_ARG_ALL);
303 if (options->verify->M == INT_MIN)
304 options->verify->M = 100;
305 if (options->verify->m == INT_MAX)
306 options->verify->m = -100;
308 result_data_init(&all_result);
310 while (fgets(path, sizeof(path), stdin)) {
311 struct result_data result;
312 FILE *in;
313 int i;
315 ++n;
316 result_data_init(&result);
317 fprintf(stderr, "%s", path);
318 *strchr(path, '\n') = '\0';
319 in = fopen(path, "r");
320 assert(in);
321 handle(in, &result, options->verify);
322 fclose(in);
324 if (!options->quiet)
325 result_data_print(&result, 1);
327 isl_int_add(all_result.n, all_result.n, result.n);
328 for (i = 0; i < nr_methods; ++i) {
329 all_result.RE_sum[i] += result.RE_sum[i];
330 all_result.ticks[i] += result.ticks[i];
331 all_result.size[i] += result.size[i];
334 result_data_clear(&result);
336 if (!options->quiet) {
337 fprintf(stderr, "average\n");
338 result_data_print(&all_result, n);
342 result_data_clear(&all_result);
344 options_free(options);
346 return 0;