2 #include <barvinok/options.h>
3 #include "param_util.h"
5 #include "isl_param_util.h"
8 #define ALLOC(type) (type*)malloc(sizeof(type))
9 #define ALLOCN(type,n) (type*)malloc((n) * sizeof(type))
11 void Param_Vertex_Common_Denominator(Param_Vertices
*V
)
17 assert(V
->Vertex
->NbRows
> 0);
18 dim
= V
->Vertex
->NbColumns
-2;
22 value_assign(lcm
, V
->Vertex
->p
[0][dim
+1]);
23 for (i
= 1; i
< V
->Vertex
->NbRows
; ++i
)
24 value_lcm(lcm
, V
->Vertex
->p
[i
][dim
+1], lcm
);
26 for (i
= 0; i
< V
->Vertex
->NbRows
; ++i
) {
27 if (value_eq(V
->Vertex
->p
[i
][dim
+1], lcm
))
29 value_division(V
->Vertex
->p
[i
][dim
+1], lcm
, V
->Vertex
->p
[i
][dim
+1]);
30 Vector_Scale(V
->Vertex
->p
[i
], V
->Vertex
->p
[i
],
31 V
->Vertex
->p
[i
][dim
+1], dim
+1);
32 value_assign(V
->Vertex
->p
[i
][dim
+1], lcm
);
38 /* Plug in the parametric vertex Vertex (nvar x (nparam + 2))
39 * in the constraint constraint (1 + nvar + nparam + 1).
40 * The result is stored in row (1 + nparam + 1),
41 * with the denominator in position 0.
43 void Param_Inner_Product(Value
*constraint
, Matrix
*Vertex
, Value
*row
)
45 unsigned nparam
= Vertex
->NbColumns
- 2;
46 unsigned nvar
= Vertex
->NbRows
;
50 value_set_si(row
[0], 1);
51 Vector_Set(row
+1, 0, nparam
+1);
56 for (j
= 0 ; j
< nvar
; ++j
) {
58 value_assign(tmp2
, constraint
[1+j
]);
59 if (value_ne(row
[0], Vertex
->p
[j
][nparam
+1])) {
60 value_assign(tmp
, row
[0]);
61 value_lcm(row
[0], row
[0], Vertex
->p
[j
][nparam
+1]);
62 value_division(tmp
, row
[0], tmp
);
63 value_multiply(tmp2
, tmp2
, row
[0]);
64 value_division(tmp2
, tmp2
, Vertex
->p
[j
][nparam
+1]);
66 Vector_Combine(row
+1, Vertex
->p
[j
], row
+1, tmp
, tmp2
, nparam
+1);
69 Vector_Combine(row
+1, constraint
+1+nvar
, row
+1, tmp
, row
[0], nparam
+1);
75 static Param_Polyhedron
*PL_P2PP(Polyhedron
*Din
, Polyhedron
*Cin
,
76 struct barvinok_options
*options
)
78 unsigned MaxRays
= options
->MaxRays
;
79 if (MaxRays
& (POL_NO_DUAL
| POL_INTEGER
))
81 return Polyhedron2Param_Domain(Din
, Cin
, MaxRays
);
84 /* Compute a dummy Param_Domain that contains all vertices of Param_Domain D
85 * (which contains the vertices of P) that lie on the facet obtained by
86 * saturating constraint c of P
88 Param_Domain
*Param_Polyhedron_Facet(Param_Polyhedron
*PP
, Param_Domain
*D
,
93 unsigned nparam
= PP
->V
->Vertex
->NbColumns
-2;
94 Vector
*row
= Vector_Alloc(1+nparam
+1);
95 Param_Domain
*FD
= ALLOC(Param_Domain
);
99 nv
= (PP
->nbV
- 1)/(8*sizeof(int)) + 1;
100 FD
->F
= ALLOCN(unsigned, nv
);
101 memset(FD
->F
, 0, nv
* sizeof(unsigned));
103 FORALL_PVertex_in_ParamPolyhedron(V
, D
, PP
) /* _i, _ix, _bx internal counters */
104 Param_Inner_Product(constraint
, V
->Vertex
, row
->p
);
105 if (First_Non_Zero(row
->p
+1, nparam
+1) == -1)
107 END_FORALL_PVertex_in_ParamPolyhedron
;
114 #ifndef POINTS2TRIANGS_PATH
115 Param_Polyhedron
*TC_P2PP(Polyhedron
*P
, Polyhedron
*C
,
116 struct barvinok_options
*options
)
122 Param_Polyhedron
*Polyhedron2Param_Polyhedron(Polyhedron
*P
, Polyhedron
*C
,
123 struct barvinok_options
*options
)
125 switch(options
->chambers
) {
126 case BV_CHAMBERS_POLYLIB
:
127 return PL_P2PP(P
, C
, options
);
129 case BV_CHAMBERS_TOPCOM
:
130 return TC_P2PP(P
, C
, options
);
132 case BV_CHAMBERS_ISL
:
133 return ISL_P2PP(P
, C
, options
);
140 /* Construct a Polyhedron from the constraints of "PP".
142 * Use a copy of the constraints because Constraints2Polyhedron
143 * may modify its first argument.
145 Polyhedron
*Param_Polyhedron2Polyhedron(Param_Polyhedron
*PP
,
146 struct barvinok_options
*options
)
151 M
= Matrix_Copy(PP
->Constraints
);
152 P
= Constraints2Polyhedron(M
, options
->MaxRays
);
158 /* Is "PP" lower-dimensional?
159 * That is, is one of the constraints an equality constraint?
161 int Param_Polyhedron_Is_Lower_Dimensional(Param_Polyhedron
*PP
)
165 for (i
= 0; i
< PP
->Constraints
->NbRows
; ++i
) {
166 if (value_zero_p(PP
->Constraints
->p
[i
][0]))
173 #define INT_BITS (sizeof(unsigned) * 8)
175 /* Wegner's method for counting the number of ones in a bit vector */
176 int bit_vector_count(unsigned *F
, int F_len
)
181 for (i
= 0; i
< F_len
; ++i
) {
191 void Param_Vertex_Set_Facets(Param_Polyhedron
*PP
, Param_Vertices
*V
)
194 unsigned nparam
= V
->Vertex
->NbColumns
- 2;
195 int len
= (PP
->Constraints
->NbRows
+INT_BITS
-1)/INT_BITS
;
198 Vector
*row
= Vector_Alloc(1 + nparam
+ 1);
200 V
->Facets
= (unsigned *)calloc(len
, sizeof(unsigned));
201 for (i
= 0, ix
= 0, bx
= MSB
; i
< PP
->Constraints
->NbRows
; ++i
) {
202 Param_Inner_Product(PP
->Constraints
->p
[i
], V
->Vertex
, row
->p
);
203 if (First_Non_Zero(row
->p
+1, nparam
+1) == -1)
211 Polyhedron
*Param_Vertex_Cone(Param_Polyhedron
*PP
, Param_Vertices
*V
,
212 struct barvinok_options
*options
)
216 int len
= (PP
->Constraints
->NbRows
+INT_BITS
-1)/INT_BITS
;
220 unsigned nvar
= V
->Vertex
->NbRows
;
223 Param_Vertex_Set_Facets(PP
, V
);
224 n
= bit_vector_count(V
->Facets
, len
);
226 M
= Matrix_Alloc(n
, 1+nvar
+1);
228 for (i
= 0, j
= 0, ix
= 0, bx
= MSB
; i
< PP
->Constraints
->NbRows
; ++i
) {
229 if (V
->Facets
[ix
] & bx
)
230 Vector_Copy(PP
->Constraints
->p
[i
], M
->p
[j
++], 1+nvar
);
233 C
= Constraints2Polyhedron(M
, options
->MaxRays
);