3 #include "conversion.h"
4 #include "evalue_convert.h"
5 #include "lattice_point.h"
17 static struct argp_option argp_options
[] = {
18 { "convert", 'c', 0, 0, "convert fractionals to periodics" },
19 { "combine", 'C', 0, 0 },
20 { "floor", 'f', 0, 0, "convert fractionals to floorings" },
21 { "list", 'l', 0, 0 },
22 { "latex", 'L', 0, 0 },
23 { "range-reduction", 'R', 0, 0 },
27 static error_t
parse_opt(int key
, char *arg
, struct argp_state
*state
)
29 struct convert_options
*options
= (struct convert_options
*)state
->input
;
61 return ARGP_ERR_UNKNOWN
;
66 struct argp convert_argp
= {
67 argp_options
, parse_opt
, 0, 0
70 static int type_offset(enode
*p
)
72 return p
->type
== fractional
? 1 :
73 p
->type
== flooring
? 1 : 0;
76 static Lattice
*extract_lattice(evalue
*e
, int nparam
)
82 /* For some mysterious reason, SolveDiophantine expects an extra
83 * [0 0 0 1] row in its input matrix.
85 Matrix
*M
= Matrix_Alloc(2, nparam
+1+1);
86 value_set_si(M
->p
[1][nparam
+1], 1);
87 evalue_extract_affine(e
, M
->p
[0], M
->p
[0]+nparam
+1, M
->p
[0]+nparam
);
89 value_set_si(M
->p
[0][nparam
+1], 0);
90 SolveDiophantine(M
, &U
, &X
);
93 L
= Matrix_Alloc(nparam
+1, nparam
+1);
94 for (i
= 0; i
< nparam
; ++i
)
95 Vector_Copy(U
->p
[i
], L
->p
[i
], nparam
);
96 value_set_si(L
->p
[nparam
][nparam
], 1);
101 /* Returns a lattice such that the quasi-polynomial e can be represented
102 * by a list of polynomials, one for each point in the fundamental
103 * parallelepiped of the lattice.
104 * If e is a polynomial, then this function returns NULL.
106 static Lattice
*extract_common_lattice(evalue
*e
, Lattice
*L
, int nparam
)
110 if (value_notzero_p(e
->d
))
113 assert(e
->x
.p
->type
!= partition
);
115 if (e
->x
.p
->type
== fractional
) {
116 Lattice
*L2
= extract_lattice(&e
->x
.p
->arr
[0], nparam
);
120 Lattice
*L3
= LatticeIntersection(L
, L2
);
127 offset
= type_offset(e
->x
.p
);
128 for (i
= e
->x
.p
->size
-1; i
>= offset
; --i
)
129 L
= extract_common_lattice(&e
->x
.p
->arr
[i
], L
, nparam
);
133 /* Construct an evalue dst from src corresponding to the coset represented
134 * by coset, a vector of size number of parameters plus one.
135 * The final value in this vector should be 1.
137 static void evalue_coset(const evalue
*src
, const Vector
*coset
, evalue
*dst
)
139 if (value_notzero_p(src
->d
)) {
140 value_assign(dst
->d
, src
->d
);
141 value_init(dst
->x
.n
);
142 value_assign(dst
->x
.n
, src
->x
.n
);
146 if (src
->x
.p
->type
== fractional
) {
149 Vector
*c
= Vector_Alloc(coset
->Size
);
152 evalue_extract_affine(&src
->x
.p
->arr
[0], c
->p
, c
->p
+c
->Size
-1, &f
.d
);
153 Inner_Product(coset
->p
, c
->p
, c
->Size
, &f
.x
.n
);
155 mpz_fdiv_r(f
.x
.n
, f
.x
.n
, f
.d
);
157 evalue_set_si(dst
, 0, 1);
158 for (int i
= src
->x
.p
->size
-1; i
>= 1; --i
) {
161 evalue_coset(&src
->x
.p
->arr
[i
], coset
, &t
);
163 free_evalue_refs(&t
);
165 free_evalue_refs(&f
);
169 if (src
->x
.p
->type
== relation
) {
170 evalue
*arg
= evalue_eval(&src
->x
.p
->arr
[0], coset
->p
);
171 if (value_zero_p(arg
->x
.n
))
172 evalue_coset(&src
->x
.p
->arr
[1], coset
, dst
);
173 else if (src
->x
.p
->size
> 2)
174 evalue_coset(&src
->x
.p
->arr
[2], coset
, dst
);
176 evalue_set_si(dst
, 0, 1);
177 free_evalue_refs(arg
);
182 assert(src
->x
.p
->type
== polynomial
);
183 value_set_si(dst
->d
, 0);
184 dst
->x
.p
= new_enode(src
->x
.p
->type
, src
->x
.p
->size
, src
->x
.p
->pos
);
185 for (int i
= 0; i
< src
->x
.p
->size
; ++i
)
186 evalue_coset(&src
->x
.p
->arr
[i
], coset
, &dst
->x
.p
->arr
[i
]);
190 static void evalue_print_list_evalue(FILE *out
, evalue
*e
, int nparam
,
193 cerr
<< "not supported" << endl
;
196 static void evalue_print_list_evalue(FILE *out
, evalue
*e
, int nparam
,
200 L
= extract_common_lattice(e
, NULL
, nparam
);
202 print_evalue(out
, e
, params
);
204 fdostream
os(dup(fileno(out
)));
205 Vector
*coset
= Vector_Alloc(nparam
+1);
206 value_set_si(coset
->p
[nparam
], 1);
209 matrix2zz(L
, RT
, nparam
, nparam
);
211 fprintf(out
, "Lattice:\n");
214 lattice_point(coset
->p
, R
, vertices
, to_ulong(abs(determinant(R
))), NULL
);
216 for (int i
= 0; i
< vertices
.NumRows(); ++i
) {
218 os
<< vertices
[i
] << endl
;
219 zz2values(vertices
[i
], coset
->p
);
221 evalue_coset(e
, coset
, &t
);
222 print_evalue(out
, &t
, params
);
223 free_evalue_refs(&t
);
230 static void evalue_print_list(FILE *out
, evalue
*e
, int nparam
, char **params
)
233 assert(value_zero_p(e
->d
));
234 assert(e
->x
.p
->type
== partition
);
236 for (i
= 0; i
< e
->x
.p
->size
/2; i
++) {
237 Print_Domain(out
, EVALUE_DOMAIN(e
->x
.p
->arr
[2*i
]), params
);
238 evalue_print_list_evalue(out
, &e
->x
.p
->arr
[2*i
+1], nparam
, params
);
242 static void print_domain_latex(std::ostream
& o
, Polyhedron
*D
, int nparam
,
246 for (int i
= 0; i
< D
->NbConstraints
; ++i
) {
247 if (First_Non_Zero(D
->Constraint
[i
]+1, D
->Dimension
) == -1)
252 for (int j
= 0; j
< D
->Dimension
; ++j
) {
253 if (value_zero_p(D
->Constraint
[i
][1+j
]))
256 if (!fc
&& value_pos_p(D
->Constraint
[i
][1+j
]))
258 if (value_mone_p(D
->Constraint
[i
][1+j
]))
260 else if (!value_one_p(D
->Constraint
[i
][1+j
]))
261 o
<< VALUE_TO_INT(D
->Constraint
[i
][1+j
]);
262 o
<< " " << params
[j
];
265 if (!fc
&& value_pos_p(D
->Constraint
[i
][1+D
->Dimension
]))
267 if (value_notzero_p(D
->Constraint
[i
][1+D
->Dimension
]))
268 o
<< VALUE_TO_INT(D
->Constraint
[i
][1+D
->Dimension
]);
269 if (value_zero_p(D
->Constraint
[i
][0]))
277 static void evalue_print_latex(std::ostream
& o
, const evalue
*e
,
278 int first
, int nested
,
279 const string
& suffix1
, const string
& suffix2
,
280 int nparam
, char **params
);
282 static void evalue_print_poly_latex1(std::ostream
& o
, const evalue
*e
,
283 int first
, int nested
, const string
& base
,
284 const string
& suffix1
, const string
& suffix2
,
285 int nparam
, char **params
)
287 int offset
= type_offset(e
->x
.p
);
288 for (int i
= e
->x
.p
->size
-1; i
>= offset
; --i
) {
289 std::ostringstream strm
;
294 strm
<< "^" << (i
-offset
);
295 evalue_print_latex(o
, &e
->x
.p
->arr
[i
], first
, nested
,
296 strm
.str(), suffix2
, nparam
, params
);
301 static void evalue_print_poly_latex2(std::ostream
& o
, const evalue
*e
,
302 int first
, int nested
, const string
& base
,
303 const string
& suffix1
, const string
& suffix2
,
304 int nparam
, char **params
)
306 int offset
= type_offset(e
->x
.p
);
307 for (int i
= e
->x
.p
->size
-1; i
>= offset
; --i
) {
308 std::ostringstream strm
;
313 strm
<< "^" << (i
-offset
);
314 evalue_print_latex(o
, &e
->x
.p
->arr
[i
], first
, nested
,
315 suffix1
, strm
.str(), nparam
, params
);
320 static void evalue_print_latex(std::ostream
& o
, const evalue
*e
,
321 int first
, int nested
,
322 const string
& suffix1
, const string
&suffix2
,
323 int nparam
, char **params
)
325 if (value_notzero_p(e
->d
)) {
326 if (value_zero_p(e
->x
.n
)) {
333 value_absolute(tmp
, e
->x
.n
);
334 if (!first
&& value_pos_p(e
->x
.n
))
336 if (value_neg_p(e
->x
.n
))
338 if (value_one_p(e
->d
)) {
339 if (!value_one_p(tmp
) ||
340 (suffix1
.length() == 0 && suffix2
.length() == 0))
341 o
<< VALUE_TO_INT(tmp
);
344 if (value_one_p(tmp
) && suffix1
.length() != 0)
347 o
<< VALUE_TO_INT(tmp
);
349 << VALUE_TO_INT(e
->d
) << "}";
351 if (!value_one_p(tmp
)) {
361 switch (e
->x
.p
->type
) {
363 o
<< "\\begin{cases}\n";
364 for (int i
= 0; i
< e
->x
.p
->size
/2; ++i
) {
367 evalue_print_latex(o
, &e
->x
.p
->arr
[2*i
+1], 1, 0,
368 suffix1
, suffix2
, nparam
, params
);
369 o
<< "& \\text{if $";
370 print_domain_latex(o
, EVALUE_DOMAIN(e
->x
.p
->arr
[2*i
]), nparam
, params
);
373 o
<< "\\end{cases}\n";
376 evalue_print_poly_latex1(o
, e
, first
, nested
, params
[e
->x
.p
->pos
-1],
377 suffix1
, suffix2
, nparam
, params
);
380 std::ostringstream strm
;
381 strm
<< "\\fractional{";
382 evalue_print_latex(strm
, &e
->x
.p
->arr
[0], 1, 1, "", "", nparam
, params
);
384 evalue_print_poly_latex2(o
, e
, first
, nested
,
385 strm
.str(), suffix1
, suffix2
, nparam
, params
);
394 static void evalue_print_latex(FILE *out
, const evalue
*e
, int nparam
,
397 cerr
<< "not supported" << endl
;
400 static void evalue_print_latex(FILE *out
, const evalue
*e
, int nparam
,
403 fdostream
os(dup(fileno(out
)));
404 evalue_print_latex(os
, e
, 1, 0, "", "", nparam
, params
);
408 int evalue_convert(evalue
*EP
, struct convert_options
*options
,
409 int verbose
, unsigned nparam
, char **params
)
412 if (options
->combine
)
415 evalue_range_reduction(EP
);
417 print_evalue(stdout
, EP
, params
);
420 if (options
->floor
) {
421 fprintf(stderr
, "WARNING: floor conversion not supported\n");
422 evalue_frac2floor2(EP
, 0);
424 print_evalue(stdout
, EP
, params
);
425 } else if (options
->list
&& params
) {
426 evalue_print_list(stdout
, EP
, nparam
, params
);
428 } else if (options
->latex
&& params
) {
429 evalue_print_latex(stdout
, EP
, nparam
, params
);
431 } else if (options
->convert
) {
432 evalue_mod2table(EP
, nparam
);
434 print_evalue(stdout
, EP
, params
);