2 #include <barvinok/polylib.h>
3 #include <barvinok/barvinok.h>
4 #include <barvinok/evalue.h>
5 #include <barvinok/options.h>
6 #include <barvinok/util.h>
7 #include "reduce_domain.h"
8 #include "param_util.h"
12 #define ALLOC(type) (type*)malloc(sizeof(type))
13 #define ALLOCN(type,n) (type*)malloc((n) * sizeof(type))
15 /* Computes an evalue representation of a coordinate
18 static evalue
*vertex2evalue(Value
*vertex
, int nparam
)
20 return affine2evalue(vertex
, vertex
[nparam
+1], nparam
);
23 static void matrix_print(evalue
***matrix
, int dim
, int *cols
,
24 const char **param_names
) __attribute__((unused
));
25 static void matrix_print(evalue
***matrix
, int dim
, int *cols
,
26 const char **param_names
)
30 for (i
= 0; i
< dim
; ++i
)
31 for (j
= 0; j
< dim
; ++j
) {
32 int k
= cols
? cols
[j
] : j
;
33 fprintf(stderr
, "%d %d: ", i
, j
);
34 print_evalue(stderr
, matrix
[i
][k
], param_names
);
35 fprintf(stderr
, "\n");
39 /* Compute determinant using Laplace's formula.
40 * In particular, the determinant is expanded along the last row.
41 * The cols array is a list of columns that remain in the currect submatrix.
43 static evalue
*determinant_cols(evalue
***matrix
, int dim
, int *cols
)
53 evalue_copy(det
, matrix
[0][cols
[0]]);
58 evalue_set_si(&mone
, -1, 1);
60 newcols
= ALLOCN(int, dim
-1);
61 for (j
= 1; j
< dim
; ++j
)
62 newcols
[j
-1] = cols
[j
];
63 for (j
= 0; j
< dim
; ++j
) {
65 newcols
[j
-1] = cols
[j
-1];
66 tmp
= determinant_cols(matrix
, dim
-1, newcols
);
67 emul(matrix
[dim
-1][cols
[j
]], tmp
);
78 free_evalue_refs(&mone
);
83 static evalue
*determinant(evalue
***matrix
, int dim
)
86 int *cols
= ALLOCN(int, dim
);
89 for (i
= 0; i
< dim
; ++i
)
92 det
= determinant_cols(matrix
, dim
, cols
);
99 /* Compute the facet of P that saturates constraint c.
101 static Polyhedron
*facet(Polyhedron
*P
, int c
, unsigned MaxRays
)
104 Vector
*row
= Vector_Alloc(1+P
->Dimension
+1);
105 Vector_Copy(P
->Constraint
[c
]+1, row
->p
+1, P
->Dimension
+1);
106 F
= AddConstraints(row
->p
, 1, P
, MaxRays
);
111 /* Substitute parameters by the corresponding element in subs
113 static evalue
*evalue_substitute_new(evalue
*e
, evalue
**subs
)
119 if (value_notzero_p(e
->d
)) {
125 assert(e
->x
.p
->type
== polynomial
);
128 for (i
= e
->x
.p
->size
-1; i
> 0; --i
) {
129 c
= evalue_substitute_new(&e
->x
.p
->arr
[i
], subs
);
132 emul(subs
[e
->x
.p
->pos
-1], res
);
134 c
= evalue_substitute_new(&e
->x
.p
->arr
[0], subs
);
141 struct parameter_point
{
146 struct parameter_point
*parameter_point_new(unsigned nparam
)
148 struct parameter_point
*point
= ALLOC(struct parameter_point
);
149 point
->coord
= Vector_Alloc(nparam
+1);
154 evalue
**parameter_point_evalue(struct parameter_point
*point
)
157 unsigned nparam
= point
->coord
->Size
-1;
162 point
->e
= ALLOCN(evalue
*, nparam
);
163 for (j
= 0; j
< nparam
; ++j
) {
164 point
->e
[j
] = ALLOC(evalue
);
165 value_init(point
->e
[j
]->d
);
166 evalue_set(point
->e
[j
], point
->coord
->p
[j
], point
->coord
->p
[nparam
]);
172 void parameter_point_free(struct parameter_point
*point
)
175 unsigned nparam
= point
->coord
->Size
-1;
177 Vector_Free(point
->coord
);
180 for (i
= 0; i
< nparam
; ++i
)
181 evalue_free(point
->e
[i
]);
187 /* Computes point in pameter space where polyhedron is non-empty.
189 static struct parameter_point
*non_empty_point(Param_Domain
*D
)
191 unsigned nparam
= D
->Domain
->Dimension
;
192 struct parameter_point
*point
;
195 v
= inner_point(D
->Domain
);
196 point
= parameter_point_new(nparam
);
197 Vector_Copy(v
->p
+1, point
->coord
->p
, nparam
+1);
203 static Matrix
*barycenter(Param_Polyhedron
*PP
, Param_Domain
*D
)
206 Matrix
*center
= NULL
;
216 FORALL_PVertex_in_ParamPolyhedron(V
, D
, PP
)
219 center
= Matrix_Copy(V
->Vertex
);
220 nparam
= center
->NbColumns
- 2;
222 for (i
= 0; i
< center
->NbRows
; ++i
) {
223 value_assign(fc
, center
->p
[i
][nparam
+1]);
224 value_lcm(center
->p
[i
][nparam
+1],
225 fc
, V
->Vertex
->p
[i
][nparam
+1]);
226 value_division(fc
, center
->p
[i
][nparam
+1], fc
);
227 value_division(fv
, center
->p
[i
][nparam
+1],
228 V
->Vertex
->p
[i
][nparam
+1]);
229 Vector_Combine(center
->p
[i
], V
->Vertex
->p
[i
], center
->p
[i
],
233 END_FORALL_PVertex_in_ParamPolyhedron
;
238 value_set_si(denom
, nbV
);
239 for (i
= 0; i
< center
->NbRows
; ++i
) {
240 value_multiply(center
->p
[i
][nparam
+1], center
->p
[i
][nparam
+1], denom
);
241 Vector_Normalize(center
->p
[i
], nparam
+2);
248 static Matrix
*triangulation_vertex(Param_Polyhedron
*PP
, Param_Domain
*D
,
253 FORALL_PVertex_in_ParamPolyhedron(V
, D
, PP
)
255 END_FORALL_PVertex_in_ParamPolyhedron
;
261 /* Compute dim! times the volume of polyhedron F in Param_Domain D.
262 * If F is a simplex, then the volume is computed of a recursive pyramid
263 * over F with the points already in matrix.
264 * Otherwise, the barycenter of F is added to matrix and the function
265 * is called recursively on the facets of F.
267 * The first row of matrix contain the _negative_ of the first point.
268 * The remaining rows of matrix contain the distance of the corresponding
269 * point to the first point.
271 static evalue
*volume_in_domain(Param_Polyhedron
*PP
, Param_Domain
*D
,
272 unsigned dim
, evalue
***matrix
,
273 struct parameter_point
*point
,
274 int row
, Polyhedron
*F
,
275 struct barvinok_options
*options
);
277 static evalue
*volume_triangulate(Param_Polyhedron
*PP
, Param_Domain
*D
,
278 unsigned dim
, evalue
***matrix
,
279 struct parameter_point
*point
,
280 int row
, Polyhedron
*F
,
281 struct barvinok_options
*options
)
288 unsigned cut_MaxRays
= options
->MaxRays
;
289 unsigned nparam
= PP
->V
->Vertex
->NbColumns
-2;
292 POL_UNSET(cut_MaxRays
, POL_INTEGER
);
295 evalue_set_si(&mone
, -1, 1);
297 if (options
->approx
->volume_triangulate
== BV_VOL_BARYCENTER
)
298 center
= barycenter(PP
, D
);
300 center
= triangulation_vertex(PP
, D
, F
);
301 for (j
= 0; j
< dim
; ++j
)
302 matrix
[row
][j
] = vertex2evalue(center
->p
[j
], center
->NbColumns
- 2);
303 if (options
->approx
->volume_triangulate
== BV_VOL_BARYCENTER
)
306 v
= Vector_Alloc(1+nparam
+1);
309 for (j
= 0; j
< dim
; ++j
)
310 emul(&mone
, matrix
[row
][j
]);
312 for (j
= 0; j
< dim
; ++j
)
313 eadd(matrix
[0][j
], matrix
[row
][j
]);
317 POL_ENSURE_FACETS(F
);
318 for (j
= F
->NbEq
; j
< F
->NbConstraints
; ++j
) {
321 if (First_Non_Zero(F
->Constraint
[j
]+1, dim
) == -1)
323 if (options
->approx
->volume_triangulate
!= BV_VOL_BARYCENTER
) {
324 Param_Inner_Product(F
->Constraint
[j
], center
, v
->p
);
325 if (First_Non_Zero(v
->p
+1, nparam
+1) == -1)
328 FF
= facet(F
, j
, options
->MaxRays
);
329 FD
= Param_Polyhedron_Facet(PP
, D
, F
->Constraint
[j
]);
330 tmp
= volume_in_domain(PP
, FD
, dim
, matrix
, point
,
339 Param_Domain_Free(FD
);
342 if (options
->approx
->volume_triangulate
!= BV_VOL_BARYCENTER
)
345 for (j
= 0; j
< dim
; ++j
)
346 evalue_free(matrix
[row
][j
]);
348 free_evalue_refs(&mone
);
352 static evalue
*volume_simplex(Param_Polyhedron
*PP
, Param_Domain
*D
,
353 unsigned dim
, evalue
***matrix
,
354 struct parameter_point
*point
,
355 int row
, struct barvinok_options
*options
)
362 options
->stats
->volume_simplices
++;
365 evalue_set_si(&mone
, -1, 1);
368 FORALL_PVertex_in_ParamPolyhedron(V
, D
, PP
) /* _ix, _bx internal counters */
369 for (j
= 0; j
< dim
; ++j
) {
370 matrix
[i
][j
] = vertex2evalue(V
->Vertex
->p
[j
],
371 V
->Vertex
->NbColumns
- 2);
373 emul(&mone
, matrix
[i
][j
]);
375 eadd(matrix
[0][j
], matrix
[i
][j
]);
378 END_FORALL_PVertex_in_ParamPolyhedron
;
380 vol
= determinant(matrix
+1, dim
);
382 val
= evalue_substitute_new(vol
, parameter_point_evalue(point
));
384 assert(value_notzero_p(val
->d
));
385 assert(value_notzero_p(val
->x
.n
));
386 if (value_neg_p(val
->x
.n
))
391 for (i
= row
; i
< dim
+1; ++i
)
392 for (j
= 0; j
< dim
; ++j
)
393 evalue_free(matrix
[i
][j
]);
395 free_evalue_refs(&mone
);
400 static evalue
*volume_triangulate_lift(Param_Polyhedron
*PP
, Param_Domain
*D
,
401 unsigned dim
, evalue
***matrix
,
402 struct parameter_point
*point
,
403 int row
, struct barvinok_options
*options
)
405 const static int MAX_TRY
=10;
410 Matrix
*FixedRays
, *M
;
418 nv
= (PP
->nbV
- 1)/(8*sizeof(int)) + 1;
419 SD
.F
= ALLOCN(unsigned, nv
);
421 FixedRays
= Matrix_Alloc(PP
->nbV
+1, 1+dim
+2);
423 FORALL_PVertex_in_ParamPolyhedron(V
, D
, PP
)
424 unsigned nparam
= V
->Vertex
->NbColumns
- 2;
425 Param_Vertex_Common_Denominator(V
);
426 for (i
= 0; i
< V
->Vertex
->NbRows
; ++i
) {
427 value_multiply(FixedRays
->p
[nbV
][1+i
], V
->Vertex
->p
[i
][nparam
],
428 point
->coord
->p
[nparam
]);
429 Inner_Product(V
->Vertex
->p
[i
], point
->coord
->p
, nparam
,
430 &FixedRays
->p
[nbV
][1+dim
]);
431 value_addto(FixedRays
->p
[nbV
][1+i
], FixedRays
->p
[nbV
][1+i
],
432 FixedRays
->p
[nbV
][1+dim
]);
434 value_multiply(FixedRays
->p
[nbV
][1+dim
+1], V
->Vertex
->p
[0][nparam
+1],
435 point
->coord
->p
[nparam
]);
436 value_set_si(FixedRays
->p
[nbV
][0], 1);
438 END_FORALL_PVertex_in_ParamPolyhedron
;
439 value_set_si(FixedRays
->p
[nbV
][0], 1);
440 value_set_si(FixedRays
->p
[nbV
][1+dim
], 1);
441 FixedRays
->NbRows
= nbV
+1;
446 /* Usually vol should still be NULL */
452 assert(t
<= MAX_TRY
);
455 for (i
= 0; i
< nbV
; ++i
)
456 value_set_si(FixedRays
->p
[i
][1+dim
], random_int((t
+1)*dim
*nbV
)+1);
458 M
= Matrix_Copy(FixedRays
);
459 L
= Rays2Polyhedron(M
, options
->MaxRays
);
462 POL_ENSURE_FACETS(L
);
463 for (i
= 0; i
< L
->NbConstraints
; ++i
) {
465 /* Ignore perpendicular facets, i.e., facets with 0 z-coordinate */
466 if (value_negz_p(L
->Constraint
[i
][1+dim
]))
469 memset(SD
.F
, 0, nv
* sizeof(unsigned));
472 FORALL_PVertex_in_ParamPolyhedron(V
, D
, PP
) /* _ix, _bx internal */
473 Inner_Product(FixedRays
->p
[nbV
]+1, L
->Constraint
[i
]+1, dim
+2, &tmp
);
474 if (value_zero_p(tmp
)) {
481 END_FORALL_PVertex_in_ParamPolyhedron
;
482 assert(r
== (dim
-row
)+1);
484 s
= volume_simplex(PP
, &SD
, dim
, matrix
, point
, row
, options
);
493 Matrix_Free(FixedRays
);
500 static evalue
*volume_in_domain(Param_Polyhedron
*PP
, Param_Domain
*D
,
501 unsigned dim
, evalue
***matrix
,
502 struct parameter_point
*point
,
503 int row
, Polyhedron
*F
,
504 struct barvinok_options
*options
)
513 FORALL_PVertex_in_ParamPolyhedron(V
, D
, PP
)
515 END_FORALL_PVertex_in_ParamPolyhedron
;
517 if (nbV
> (dim
-row
) + 1) {
518 if (options
->approx
->volume_triangulate
== BV_VOL_LIFT
)
519 vol
= volume_triangulate_lift(PP
, D
, dim
, matrix
, point
,
522 vol
= volume_triangulate(PP
, D
, dim
, matrix
, point
,
525 assert(nbV
== (dim
-row
) + 1);
526 vol
= volume_simplex(PP
, D
, dim
, matrix
, point
, row
, options
);
532 /* Compute the volume of the parametric polytope "PP" with context "C".
534 static evalue
*PP_Volume(Param_Polyhedron
*PP
, Polyhedron
* C
,
535 struct barvinok_options
*options
)
544 struct evalue_section
*s
;
549 P
= Param_Polyhedron2Polyhedron(PP
, options
);
551 nvar
= P
->Dimension
- C
->Dimension
;
553 TC
= true_context(P
, C
, options
->MaxRays
);
555 MaxRays
= options
->MaxRays
;
556 POL_UNSET(options
->MaxRays
, POL_INTEGER
);
559 Factorial(nvar
, &fact
);
561 for (nd
= 0, D
= PP
->D
; D
; ++nd
, D
= D
->next
);
562 s
= ALLOCN(struct evalue_section
, nd
);
564 matrix
= ALLOCN(evalue
**, nvar
+1);
565 for (i
= 0; i
< nvar
+1; ++i
)
566 matrix
[i
] = ALLOCN(evalue
*, nvar
);
568 FORALL_REDUCED_DOMAIN(PP
, TC
, nd
, options
, i
, D
, rVD
)
570 struct parameter_point
*point
;
572 CA
= align_context(D
->Domain
, P
->Dimension
, MaxRays
);
573 F
= DomainIntersection(P
, CA
, options
->MaxRays
);
576 point
= non_empty_point(D
);
578 s
[i
].E
= volume_in_domain(PP
, D
, nvar
, matrix
, point
, 0, F
, options
);
580 parameter_point_free(point
);
581 evalue_div(s
[i
].E
, fact
);
582 END_FORALL_REDUCED_DOMAIN
583 options
->MaxRays
= MaxRays
;
586 vol
= evalue_from_section_array(s
, nd
);
589 for (i
= 0; i
< nvar
+1; ++i
)
598 evalue
* Param_Polyhedron_Volume(Polyhedron
*P
, Polyhedron
* C
,
599 struct barvinok_options
*options
)
601 unsigned nparam
= C
->Dimension
;
602 Param_Polyhedron
*PP
;
605 if (options
->approx
->approximation
== BV_APPROX_SIGN_NONE
)
608 if (options
->approx
->approximation
!= BV_APPROX_SIGN_APPROX
) {
609 int pa
= options
->approx
->approximation
;
610 assert(pa
== BV_APPROX_SIGN_UPPER
|| pa
== BV_APPROX_SIGN_LOWER
);
612 P
= Polyhedron_Flate(P
, nparam
, pa
== BV_APPROX_SIGN_UPPER
,
615 /* Don't deflate/inflate again (on this polytope) */
616 options
->approx
->approximation
= BV_APPROX_SIGN_APPROX
;
617 vol
= barvinok_enumerate_with_options(P
, C
, options
);
618 options
->approx
->approximation
= pa
;
624 PP
= Polyhedron2Param_Polyhedron(P
, C
, options
);
625 vol
= PP_Volume(PP
, C
, options
);
626 Param_Polyhedron_Free(PP
);