doc: document polytope_sample
[barvinok.git] / decomposer.cc
blob24931c10fe4ece34d7a288de21be73aa6f6972ef
1 #include <vector>
2 #include <gmp.h>
3 #include <NTL/mat_ZZ.h>
4 #include <NTL/LLL.h>
5 extern "C" {
6 #include <polylib/polylibgmp.h>
8 #include <barvinok/barvinok.h>
9 #include <barvinok/util.h>
10 #include "conversion.h"
11 #include "decomposer.h"
13 #ifdef NTL_STD_CXX
14 using namespace NTL;
15 #endif
16 using std::vector;
19 * Returns the largest absolute value in the vector
21 static ZZ max(vec_ZZ& v)
23 ZZ max = abs(v[0]);
24 for (int i = 1; i < v.length(); ++i)
25 if (abs(v[i]) > max)
26 max = abs(v[i]);
27 return max;
30 class cone {
31 public:
32 cone(Matrix *M) {
33 Cone = 0;
34 Rays = Matrix_Copy(M);
35 set_det();
37 cone(Polyhedron *C) {
38 Cone = Polyhedron_Copy(C);
39 Rays = rays(C);
40 set_det();
42 void set_det() {
43 mat_ZZ A;
44 matrix2zz(Rays, A, Rays->NbRows - 1, Rays->NbColumns - 1);
45 det = determinant(A);
48 Vector* short_vector(vec_ZZ& lambda, barvinok_options *options) {
49 Matrix *M = Matrix_Copy(Rays);
50 Matrix *inv = Matrix_Alloc(M->NbRows, M->NbColumns);
51 int ok = Matrix_Inverse(M, inv);
52 assert(ok);
53 Matrix_Free(M);
55 ZZ det2;
56 mat_ZZ B;
57 mat_ZZ U;
58 matrix2zz(inv, B, inv->NbRows - 1, inv->NbColumns - 1);
59 long r = LLL(det2, B, U, options->LLL_a, options->LLL_b);
61 ZZ min = max(B[0]);
62 int index = 0;
63 for (int i = 1; i < B.NumRows(); ++i) {
64 ZZ tmp = max(B[i]);
65 if (tmp < min) {
66 min = tmp;
67 index = i;
71 Matrix_Free(inv);
73 lambda = B[index];
75 Vector *z = Vector_Alloc(U[index].length()+1);
76 assert(z);
77 zz2values(U[index], z->p);
78 value_set_si(z->p[U[index].length()], 0);
80 Polyhedron *C = poly();
81 int i;
82 for (i = 0; i < lambda.length(); ++i)
83 if (lambda[i] > 0)
84 break;
85 if (i == lambda.length()) {
86 Value tmp;
87 value_init(tmp);
88 value_set_si(tmp, -1);
89 Vector_Scale(z->p, z->p, tmp, z->Size-1);
90 value_clear(tmp);
92 return z;
95 ~cone() {
96 Polyhedron_Free(Cone);
97 Matrix_Free(Rays);
100 Polyhedron *poly() {
101 if (!Cone) {
102 Matrix *M = Matrix_Alloc(Rays->NbRows+1, Rays->NbColumns+1);
103 for (int i = 0; i < Rays->NbRows; ++i) {
104 Vector_Copy(Rays->p[i], M->p[i]+1, Rays->NbColumns);
105 value_set_si(M->p[i][0], 1);
107 Vector_Set(M->p[Rays->NbRows]+1, 0, Rays->NbColumns-1);
108 value_set_si(M->p[Rays->NbRows][0], 1);
109 value_set_si(M->p[Rays->NbRows][Rays->NbColumns], 1);
110 Cone = Rays2Polyhedron(M, M->NbRows+1);
111 assert(Cone->NbConstraints == Cone->NbRays);
112 Matrix_Free(M);
114 return Cone;
117 ZZ det;
118 Polyhedron *Cone;
119 Matrix *Rays;
122 void decomposer::decompose(Polyhedron *C, barvinok_options *options)
124 vector<cone *> nonuni;
125 cone * c = new cone(C);
126 ZZ det = c->det;
127 int s = sign(det);
128 assert(det != 0);
129 if (abs(det) > 1) {
130 nonuni.push_back(c);
131 } else {
132 try {
133 handle(C, 1);
134 delete c;
135 } catch (...) {
136 delete c;
137 throw;
140 vec_ZZ lambda;
141 while (!nonuni.empty()) {
142 c = nonuni.back();
143 nonuni.pop_back();
144 Vector* v = c->short_vector(lambda, options);
145 for (int i = 0; i < c->Rays->NbRows - 1; ++i) {
146 if (lambda[i] == 0)
147 continue;
148 Matrix* M = Matrix_Copy(c->Rays);
149 Vector_Copy(v->p, M->p[i], v->Size);
150 cone * pc = new cone(M);
151 assert (pc->det != 0);
152 if (abs(pc->det) > 1) {
153 assert(abs(pc->det) < abs(c->det));
154 nonuni.push_back(pc);
155 } else {
156 try {
157 handle(pc->poly(), sign(pc->det) * s);
158 delete pc;
159 } catch (...) {
160 delete c;
161 delete pc;
162 while (!nonuni.empty()) {
163 c = nonuni.back();
164 nonuni.pop_back();
165 delete c;
167 Matrix_Free(M);
168 Vector_Free(v);
169 throw;
172 Matrix_Free(M);
174 Vector_Free(v);
175 delete c;
179 void polar_decomposer::decompose(Polyhedron *cone, barvinok_options *options)
181 POL_ENSURE_VERTICES(cone);
182 Polyhedron_Polarize(cone);
183 if (cone->NbRays - 1 != cone->Dimension) {
184 Polyhedron *tmp = cone;
185 cone = triangulate_cone(cone, options->MaxRays);
186 Polyhedron_Free(tmp);
188 try {
189 for (Polyhedron *Polar = cone; Polar; Polar = Polar->next)
190 decomposer::decompose(Polar, options);
191 Domain_Free(cone);
192 } catch (...) {
193 Domain_Free(cone);
194 throw;
198 void polar_decomposer::handle(Polyhedron *P, int sign)
200 Polyhedron_Polarize(P);
201 handle_polar(P, sign);
204 void vertex_decomposer::decompose_at_vertex(Param_Vertices *V, int _i,
205 barvinok_options *options)
207 Polyhedron *C = supporting_cone_p(P, V);
208 vert = _i;
209 this->V = V;
211 pd->decompose(C, options);
215 * Barvinok's Decomposition of a simplicial cone
217 * Returns two lists of polyhedra
219 void barvinok_decompose(Polyhedron *C, Polyhedron **ppos, Polyhedron **pneg)
221 barvinok_options *options = barvinok_options_new_with_defaults();
222 Polyhedron *pos = *ppos, *neg = *pneg;
223 vector<cone *> nonuni;
224 cone * c = new cone(C);
225 ZZ det = c->det;
226 int s = sign(det);
227 assert(det != 0);
228 if (abs(det) > 1) {
229 nonuni.push_back(c);
230 } else {
231 Polyhedron *p = Polyhedron_Copy(c->Cone);
232 p->next = pos;
233 pos = p;
234 delete c;
236 vec_ZZ lambda;
237 while (!nonuni.empty()) {
238 c = nonuni.back();
239 nonuni.pop_back();
240 Vector* v = c->short_vector(lambda, options);
241 for (int i = 0; i < c->Rays->NbRows - 1; ++i) {
242 if (lambda[i] == 0)
243 continue;
244 Matrix* M = Matrix_Copy(c->Rays);
245 Vector_Copy(v->p, M->p[i], v->Size);
246 cone * pc = new cone(M);
247 assert (pc->det != 0);
248 if (abs(pc->det) > 1) {
249 assert(abs(pc->det) < abs(c->det));
250 nonuni.push_back(pc);
251 } else {
252 Polyhedron *p = pc->poly();
253 pc->Cone = 0;
254 if (sign(pc->det) == s) {
255 p->next = pos;
256 pos = p;
257 } else {
258 p->next = neg;
259 neg = p;
261 delete pc;
263 Matrix_Free(M);
265 Vector_Free(v);
266 delete c;
268 *ppos = pos;
269 *pneg = neg;
270 free(options);