8 #include <NTL/mat_ZZ.h>
12 #include <polylib/polylibgmp.h>
13 #include "ev_operations.h"
28 using std::ostringstream
;
30 #define ALLOC(p) (((long *) (p))[0])
31 #define SIZE(p) (((long *) (p))[1])
32 #define DATA(p) ((mp_limb_t *) (((long *) (p)) + 2))
34 static void value2zz(Value v
, ZZ
& z
)
36 int sa
= v
[0]._mp_size
;
37 int abs_sa
= sa
< 0 ? -sa
: sa
;
39 _ntl_gsetlength(&z
.rep
, abs_sa
);
40 mp_limb_t
* adata
= DATA(z
.rep
);
41 for (int i
= 0; i
< abs_sa
; ++i
)
42 adata
[i
] = v
[0]._mp_d
[i
];
46 void zz2value(ZZ
& z
, Value
& v
)
54 int abs_sa
= sa
< 0 ? -sa
: sa
;
56 mp_limb_t
* adata
= DATA(z
.rep
);
57 _mpz_realloc(v
, abs_sa
);
58 for (int i
= 0; i
< abs_sa
; ++i
)
59 v
[0]._mp_d
[i
] = adata
[i
];
64 #define ALLOC(t,p) p = (t*)malloc(sizeof(*p))
67 * We just ignore the last column and row
68 * If the final element is not equal to one
69 * then the result will actually be a multiple of the input
71 static void matrix2zz(Matrix
*M
, mat_ZZ
& m
, unsigned nr
, unsigned nc
)
75 for (int i
= 0; i
< nr
; ++i
) {
76 // assert(value_one_p(M->p[i][M->NbColumns - 1]));
77 for (int j
= 0; j
< nc
; ++j
) {
78 value2zz(M
->p
[i
][j
], m
[i
][j
]);
83 static void values2zz(Value
*p
, vec_ZZ
& v
, int len
)
87 for (int i
= 0; i
< len
; ++i
) {
94 static void zz2values(vec_ZZ
& v
, Value
*p
)
96 for (int i
= 0; i
< v
.length(); ++i
)
100 static void rays(mat_ZZ
& r
, Polyhedron
*C
)
102 unsigned dim
= C
->NbRays
- 1; /* don't count zero vertex */
103 assert(C
->NbRays
- 1 == C
->Dimension
);
108 for (i
= 0, c
= 0; i
< dim
; ++i
)
109 if (value_zero_p(C
->Ray
[i
][dim
+1])) {
110 for (int j
= 0; j
< dim
; ++j
) {
111 value2zz(C
->Ray
[i
][j
+1], tmp
);
118 static Matrix
* rays(Polyhedron
*C
)
120 unsigned dim
= C
->NbRays
- 1; /* don't count zero vertex */
121 assert(C
->NbRays
- 1 == C
->Dimension
);
123 Matrix
*M
= Matrix_Alloc(dim
+1, dim
+1);
127 for (i
= 0, c
= 0; i
<= dim
&& c
< dim
; ++i
)
128 if (value_zero_p(C
->Ray
[i
][dim
+1])) {
129 Vector_Copy(C
->Ray
[i
] + 1, M
->p
[c
], dim
);
130 value_set_si(M
->p
[c
++][dim
], 0);
133 value_set_si(M
->p
[dim
][dim
], 1);
138 static Matrix
* rays2(Polyhedron
*C
)
140 unsigned dim
= C
->NbRays
- 1; /* don't count zero vertex */
141 assert(C
->NbRays
- 1 == C
->Dimension
);
143 Matrix
*M
= Matrix_Alloc(dim
, dim
);
147 for (i
= 0, c
= 0; i
<= dim
&& c
< dim
; ++i
)
148 if (value_zero_p(C
->Ray
[i
][dim
+1]))
149 Vector_Copy(C
->Ray
[i
] + 1, M
->p
[c
++], dim
);
156 * Returns the largest absolute value in the vector
158 static ZZ
max(vec_ZZ
& v
)
161 for (int i
= 1; i
< v
.length(); ++i
)
171 Rays
= Matrix_Copy(M
);
174 cone(Polyhedron
*C
) {
175 Cone
= Polyhedron_Copy(C
);
181 matrix2zz(Rays
, A
, Rays
->NbRows
- 1, Rays
->NbColumns
- 1);
182 det
= determinant(A
);
185 Vector
* short_vector(vec_ZZ
& lambda
) {
186 Matrix
*M
= Matrix_Copy(Rays
);
187 Matrix
*inv
= Matrix_Alloc(M
->NbRows
, M
->NbColumns
);
188 int ok
= Matrix_Inverse(M
, inv
);
195 matrix2zz(inv
, B
, inv
->NbRows
- 1, inv
->NbColumns
- 1);
196 long r
= LLL(det2
, B
, U
);
200 for (int i
= 1; i
< B
.NumRows(); ++i
) {
212 Vector
*z
= Vector_Alloc(U
[index
].length()+1);
214 zz2values(U
[index
], z
->p
);
215 value_set_si(z
->p
[U
[index
].length()], 0);
219 Polyhedron
*C
= poly();
221 for (i
= 0; i
< C
->NbConstraints
; ++i
) {
222 Inner_Product(z
->p
, C
->Constraint
[i
]+1, z
->Size
-1, &tmp
);
223 if (value_pos_p(tmp
))
226 if (i
== C
->NbConstraints
) {
227 value_set_si(tmp
, -1);
228 Vector_Scale(z
->p
, z
->p
, tmp
, z
->Size
-1);
235 Polyhedron_Free(Cone
);
241 Matrix
*M
= Matrix_Alloc(Rays
->NbRows
+1, Rays
->NbColumns
+1);
242 for (int i
= 0; i
< Rays
->NbRows
; ++i
) {
243 Vector_Copy(Rays
->p
[i
], M
->p
[i
]+1, Rays
->NbColumns
);
244 value_set_si(M
->p
[i
][0], 1);
246 Vector_Set(M
->p
[Rays
->NbRows
]+1, 0, Rays
->NbColumns
-1);
247 value_set_si(M
->p
[Rays
->NbRows
][0], 1);
248 value_set_si(M
->p
[Rays
->NbRows
][Rays
->NbColumns
], 1);
249 Cone
= Rays2Polyhedron(M
, M
->NbRows
+1);
250 assert(Cone
->NbConstraints
== Cone
->NbRays
);
264 dpoly(int d
, ZZ
& degree
, int offset
= 0) {
265 coeff
.SetLength(d
+1);
267 int min
= d
+ offset
;
268 if (degree
>= 0 && degree
< ZZ(INIT_VAL
, min
))
269 min
= to_int(degree
);
271 ZZ c
= ZZ(INIT_VAL
, 1);
274 for (int i
= 1; i
<= min
; ++i
) {
275 c
*= (degree
-i
+ 1);
280 void operator *= (dpoly
& f
) {
281 assert(coeff
.length() == f
.coeff
.length());
283 coeff
= f
.coeff
[0] * coeff
;
284 for (int i
= 1; i
< coeff
.length(); ++i
)
285 for (int j
= 0; i
+j
< coeff
.length(); ++j
)
286 coeff
[i
+j
] += f
.coeff
[i
] * old
[j
];
288 void div(dpoly
& d
, mpq_t count
, ZZ
& sign
) {
289 int len
= coeff
.length();
292 mpq_t
* c
= new mpq_t
[coeff
.length()];
295 for (int i
= 0; i
< len
; ++i
) {
297 zz2value(coeff
[i
], tmp
);
298 mpq_set_z(c
[i
], tmp
);
300 for (int j
= 1; j
<= i
; ++j
) {
301 zz2value(d
.coeff
[j
], tmp
);
302 mpq_set_z(qtmp
, tmp
);
303 mpq_mul(qtmp
, qtmp
, c
[i
-j
]);
304 mpq_sub(c
[i
], c
[i
], qtmp
);
307 zz2value(d
.coeff
[0], tmp
);
308 mpq_set_z(qtmp
, tmp
);
309 mpq_div(c
[i
], c
[i
], qtmp
);
312 mpq_sub(count
, count
, c
[len
-1]);
314 mpq_add(count
, count
, c
[len
-1]);
318 for (int i
= 0; i
< len
; ++i
)
330 dpoly_n(int d
, ZZ
& degree_0
, ZZ
& degree_1
, int offset
= 0) {
334 zz2value(degree_0
, d0
);
335 zz2value(degree_1
, d1
);
336 coeff
= Matrix_Alloc(d
+1, d
+1+1);
337 value_set_si(coeff
->p
[0][0], 1);
338 value_set_si(coeff
->p
[0][d
+1], 1);
339 for (int i
= 1; i
<= d
; ++i
) {
340 value_multiply(coeff
->p
[i
][0], coeff
->p
[i
-1][0], d0
);
341 Vector_Combine(coeff
->p
[i
-1], coeff
->p
[i
-1]+1, coeff
->p
[i
]+1,
343 value_set_si(coeff
->p
[i
][d
+1], i
);
344 value_multiply(coeff
->p
[i
][d
+1], coeff
->p
[i
][d
+1], coeff
->p
[i
-1][d
+1]);
345 value_decrement(d0
, d0
);
350 void div(dpoly
& d
, Vector
*count
, ZZ
& sign
) {
351 int len
= coeff
->NbRows
;
352 Matrix
* c
= Matrix_Alloc(coeff
->NbRows
, coeff
->NbColumns
);
355 for (int i
= 0; i
< len
; ++i
) {
356 Vector_Copy(coeff
->p
[i
], c
->p
[i
], len
+1);
357 for (int j
= 1; j
<= i
; ++j
) {
358 zz2value(d
.coeff
[j
], tmp
);
359 value_multiply(tmp
, tmp
, c
->p
[i
][len
]);
360 value_oppose(tmp
, tmp
);
361 Vector_Combine(c
->p
[i
], c
->p
[i
-j
], c
->p
[i
],
362 c
->p
[i
-j
][len
], tmp
, len
);
363 value_multiply(c
->p
[i
][len
], c
->p
[i
][len
], c
->p
[i
-j
][len
]);
365 zz2value(d
.coeff
[0], tmp
);
366 value_multiply(c
->p
[i
][len
], c
->p
[i
][len
], tmp
);
369 value_set_si(tmp
, -1);
370 Vector_Scale(c
->p
[len
-1], count
->p
, tmp
, len
);
371 value_assign(count
->p
[len
], c
->p
[len
-1][len
]);
373 Vector_Copy(c
->p
[len
-1], count
->p
, len
+1);
374 Vector_Normalize(count
->p
, len
+1);
380 struct dpoly_r_term
{
385 /* len: number of elements in c
386 * each element in c is the coefficient of a power of t
387 * in the MacLaurin expansion
390 vector
< dpoly_r_term
* > *c
;
395 void add_term(int i
, int * powers
, ZZ
& coeff
) {
396 for (int k
= 0; k
< c
[i
].size(); ++k
) {
397 if (memcmp(c
[i
][k
]->powers
, powers
, dim
* sizeof(int)) == 0) {
398 c
[i
][k
]->coeff
+= coeff
;
402 dpoly_r_term
*t
= new dpoly_r_term
;
403 t
->powers
= new int[dim
];
404 memcpy(t
->powers
, powers
, dim
* sizeof(int));
408 dpoly_r(int len
, int dim
) {
412 c
= new vector
< dpoly_r_term
* > [len
];
414 dpoly_r(dpoly
& num
, dpoly
& den
, int pos
, int sign
, int dim
) {
416 len
= num
.coeff
.length();
417 c
= new vector
< dpoly_r_term
* > [len
];
421 for (int i
= 0; i
< len
; ++i
) {
422 ZZ coeff
= num
.coeff
[i
];
423 memset(powers
, 0, dim
* sizeof(int));
426 add_term(i
, powers
, coeff
);
428 for (int j
= 1; j
<= i
; ++j
) {
429 for (int k
= 0; k
< c
[i
-j
].size(); ++k
) {
430 memcpy(powers
, c
[i
-j
][k
]->powers
, dim
*sizeof(int));
432 coeff
= -den
.coeff
[j
-1] * c
[i
-j
][k
]->coeff
;
433 add_term(i
, powers
, coeff
);
439 dpoly_r(dpoly_r
* num
, dpoly
& den
, int pos
, int sign
, int dim
) {
442 c
= new vector
< dpoly_r_term
* > [len
];
447 for (int i
= 0 ; i
< len
; ++i
) {
448 for (int k
= 0; k
< num
->c
[i
].size(); ++k
) {
449 memcpy(powers
, num
->c
[i
][k
]->powers
, dim
*sizeof(int));
451 add_term(i
, powers
, num
->c
[i
][k
]->coeff
);
454 for (int j
= 1; j
<= i
; ++j
) {
455 for (int k
= 0; k
< c
[i
-j
].size(); ++k
) {
456 memcpy(powers
, c
[i
-j
][k
]->powers
, dim
*sizeof(int));
458 coeff
= -den
.coeff
[j
-1] * c
[i
-j
][k
]->coeff
;
459 add_term(i
, powers
, coeff
);
465 for (int i
= 0 ; i
< len
; ++i
)
466 for (int k
= 0; k
< c
[i
].size(); ++k
) {
467 delete [] c
[i
][k
]->powers
;
472 dpoly_r
*div(dpoly
& d
) {
473 dpoly_r
*rc
= new dpoly_r(len
, dim
);
474 rc
->denom
= power(d
.coeff
[0], len
);
475 ZZ inv_d
= rc
->denom
/ d
.coeff
[0];
478 for (int i
= 0; i
< len
; ++i
) {
479 for (int k
= 0; k
< c
[i
].size(); ++k
) {
480 coeff
= c
[i
][k
]->coeff
* inv_d
;
481 rc
->add_term(i
, c
[i
][k
]->powers
, coeff
);
484 for (int j
= 1; j
<= i
; ++j
) {
485 for (int k
= 0; k
< rc
->c
[i
-j
].size(); ++k
) {
486 coeff
= - d
.coeff
[j
] * rc
->c
[i
-j
][k
]->coeff
/ d
.coeff
[0];
487 rc
->add_term(i
, rc
->c
[i
-j
][k
]->powers
, coeff
);
493 void div(dpoly
& d
, ZZ
& sign
, gen_fun
*gf
, mat_ZZ
& pden
, mat_ZZ
& den
,
495 dpoly_r
* rc
= div(d
);
497 int common
= pden
.NumRows();
499 vector
< dpoly_r_term
* >& final
= rc
->c
[len
-1];
501 for (int j
= 0; j
< final
.size(); ++j
) {
503 pden
.SetDims(rows
, pden
.NumCols());
504 for (int k
= 0; k
< dim
; ++k
) {
505 int n
= final
[j
]->powers
[k
];
508 int abs_n
= n
< 0 ? -n
: n
;
509 pden
.SetDims(rows
+abs_n
, pden
.NumCols());
510 for (int l
= 0; l
< abs_n
; ++l
) {
512 pden
[rows
+l
] = den
[k
];
514 pden
[rows
+l
] = -den
[k
];
518 final
[j
]->coeff
*= sign
;
519 gf
->add(final
[j
]->coeff
, rc
->denom
, num_p
, pden
);
524 for (int i
= 0; i
< len
; ++i
) {
527 cout
<< c
[i
].size() << endl
;
528 for (int j
= 0; j
< c
[i
].size(); ++j
) {
529 for (int k
= 0; k
< dim
; ++k
) {
530 cout
<< c
[i
][j
]->powers
[k
] << " ";
532 cout
<< ": " << c
[i
][j
]->coeff
<< "/" << denom
<< endl
;
540 void decompose(Polyhedron
*C
);
541 virtual void handle(Polyhedron
*P
, int sign
) = 0;
544 struct polar_decomposer
: public decomposer
{
545 void decompose(Polyhedron
*C
, unsigned MaxRays
);
546 virtual void handle(Polyhedron
*P
, int sign
);
547 virtual void handle_polar(Polyhedron
*P
, int sign
) = 0;
550 void decomposer::decompose(Polyhedron
*C
)
552 vector
<cone
*> nonuni
;
553 cone
* c
= new cone(C
);
564 while (!nonuni
.empty()) {
567 Vector
* v
= c
->short_vector(lambda
);
568 for (int i
= 0; i
< c
->Rays
->NbRows
- 1; ++i
) {
571 Matrix
* M
= Matrix_Copy(c
->Rays
);
572 Vector_Copy(v
->p
, M
->p
[i
], v
->Size
);
573 cone
* pc
= new cone(M
);
574 assert (pc
->det
!= 0);
575 if (abs(pc
->det
) > 1) {
576 assert(abs(pc
->det
) < abs(c
->det
));
577 nonuni
.push_back(pc
);
579 handle(pc
->poly(), sign(pc
->det
) * s
);
589 void polar_decomposer::decompose(Polyhedron
*cone
, unsigned MaxRays
)
591 Polyhedron_Polarize(cone
);
592 if (cone
->NbRays
- 1 != cone
->Dimension
) {
593 Polyhedron
*tmp
= cone
;
594 cone
= triangularize_cone(cone
, MaxRays
);
595 Polyhedron_Free(tmp
);
597 for (Polyhedron
*Polar
= cone
; Polar
; Polar
= Polar
->next
)
598 decomposer::decompose(Polar
);
602 void polar_decomposer::handle(Polyhedron
*P
, int sign
)
604 Polyhedron_Polarize(P
);
605 handle_polar(P
, sign
);
609 * Barvinok's Decomposition of a simplicial cone
611 * Returns two lists of polyhedra
613 void barvinok_decompose(Polyhedron
*C
, Polyhedron
**ppos
, Polyhedron
**pneg
)
615 Polyhedron
*pos
= *ppos
, *neg
= *pneg
;
616 vector
<cone
*> nonuni
;
617 cone
* c
= new cone(C
);
624 Polyhedron
*p
= Polyhedron_Copy(c
->Cone
);
630 while (!nonuni
.empty()) {
633 Vector
* v
= c
->short_vector(lambda
);
634 for (int i
= 0; i
< c
->Rays
->NbRows
- 1; ++i
) {
637 Matrix
* M
= Matrix_Copy(c
->Rays
);
638 Vector_Copy(v
->p
, M
->p
[i
], v
->Size
);
639 cone
* pc
= new cone(M
);
640 assert (pc
->det
!= 0);
641 if (abs(pc
->det
) > 1) {
642 assert(abs(pc
->det
) < abs(c
->det
));
643 nonuni
.push_back(pc
);
645 Polyhedron
*p
= pc
->poly();
647 if (sign(pc
->det
) == s
) {
666 * Returns a single list of npos "positive" cones followed by nneg
668 * The input cone is freed
670 void decompose(Polyhedron
*cone
, Polyhedron
**parts
, int *npos
, int *nneg
, unsigned MaxRays
)
672 Polyhedron_Polarize(cone
);
673 if (cone
->NbRays
- 1 != cone
->Dimension
) {
674 Polyhedron
*tmp
= cone
;
675 cone
= triangularize_cone(cone
, MaxRays
);
676 Polyhedron_Free(tmp
);
678 Polyhedron
*polpos
= NULL
, *polneg
= NULL
;
679 *npos
= 0; *nneg
= 0;
680 for (Polyhedron
*Polar
= cone
; Polar
; Polar
= Polar
->next
)
681 barvinok_decompose(Polar
, &polpos
, &polneg
);
684 for (Polyhedron
*i
= polpos
; i
; i
= i
->next
) {
685 Polyhedron_Polarize(i
);
689 for (Polyhedron
*i
= polneg
; i
; i
= i
->next
) {
690 Polyhedron_Polarize(i
);
701 const int MAX_TRY
=10;
703 * Searches for a vector that is not orthogonal to any
704 * of the rays in rays.
706 static void nonorthog(mat_ZZ
& rays
, vec_ZZ
& lambda
)
708 int dim
= rays
.NumCols();
710 lambda
.SetLength(dim
);
714 for (int i
= 2; !found
&& i
<= 50*dim
; i
+=4) {
715 for (int j
= 0; j
< MAX_TRY
; ++j
) {
716 for (int k
= 0; k
< dim
; ++k
) {
717 int r
= random_int(i
)+2;
718 int v
= (2*(r
%2)-1) * (r
>> 1);
722 for (; k
< rays
.NumRows(); ++k
)
723 if (lambda
* rays
[k
] == 0)
725 if (k
== rays
.NumRows()) {
734 static void randomvector(Polyhedron
*P
, vec_ZZ
& lambda
, int nvar
)
738 unsigned int dim
= P
->Dimension
;
741 for (int i
= 0; i
< P
->NbRays
; ++i
) {
742 for (int j
= 1; j
<= dim
; ++j
) {
743 value_absolute(tmp
, P
->Ray
[i
][j
]);
744 int t
= VALUE_TO_LONG(tmp
) * 16;
749 for (int i
= 0; i
< P
->NbConstraints
; ++i
) {
750 for (int j
= 1; j
<= dim
; ++j
) {
751 value_absolute(tmp
, P
->Constraint
[i
][j
]);
752 int t
= VALUE_TO_LONG(tmp
) * 16;
759 lambda
.SetLength(nvar
);
760 for (int k
= 0; k
< nvar
; ++k
) {
761 int r
= random_int(max
*dim
)+2;
762 int v
= (2*(r
%2)-1) * (max
/2*dim
+ (r
>> 1));
767 static void add_rays(mat_ZZ
& rays
, Polyhedron
*i
, int *r
, int nvar
= -1,
770 unsigned dim
= i
->Dimension
;
773 for (int k
= 0; k
< i
->NbRays
; ++k
) {
774 if (!value_zero_p(i
->Ray
[k
][dim
+1]))
776 if (!all
&& nvar
!= dim
&& First_Non_Zero(i
->Ray
[k
]+1, nvar
) == -1)
778 values2zz(i
->Ray
[k
]+1, rays
[(*r
)++], nvar
);
782 void lattice_point(Value
* values
, Polyhedron
*i
, vec_ZZ
& vertex
)
784 unsigned dim
= i
->Dimension
;
785 if(!value_one_p(values
[dim
])) {
786 Matrix
* Rays
= rays(i
);
787 Matrix
*inv
= Matrix_Alloc(Rays
->NbRows
, Rays
->NbColumns
);
788 int ok
= Matrix_Inverse(Rays
, inv
);
792 Vector
*lambda
= Vector_Alloc(dim
+1);
793 Vector_Matrix_Product(values
, inv
, lambda
->p
);
795 for (int j
= 0; j
< dim
; ++j
)
796 mpz_cdiv_q(lambda
->p
[j
], lambda
->p
[j
], lambda
->p
[dim
]);
797 value_set_si(lambda
->p
[dim
], 1);
798 Vector
*A
= Vector_Alloc(dim
+1);
799 Vector_Matrix_Product(lambda
->p
, Rays
, A
->p
);
802 values2zz(A
->p
, vertex
, dim
);
805 values2zz(values
, vertex
, dim
);
808 static evalue
*term(int param
, ZZ
& c
, Value
*den
= NULL
)
810 evalue
*EP
= new evalue();
812 value_set_si(EP
->d
,0);
813 EP
->x
.p
= new_enode(polynomial
, 2, param
+ 1);
814 evalue_set_si(&EP
->x
.p
->arr
[0], 0, 1);
815 value_init(EP
->x
.p
->arr
[1].x
.n
);
817 value_set_si(EP
->x
.p
->arr
[1].d
, 1);
819 value_assign(EP
->x
.p
->arr
[1].d
, *den
);
820 zz2value(c
, EP
->x
.p
->arr
[1].x
.n
);
824 static void vertex_period(
825 Polyhedron
*i
, vec_ZZ
& lambda
, Matrix
*T
,
826 Value lcm
, int p
, Vector
*val
,
827 evalue
*E
, evalue
* ev
,
830 unsigned nparam
= T
->NbRows
- 1;
831 unsigned dim
= i
->Dimension
;
838 Vector
* values
= Vector_Alloc(dim
+ 1);
839 Vector_Matrix_Product(val
->p
, T
, values
->p
);
840 value_assign(values
->p
[dim
], lcm
);
841 lattice_point(values
->p
, i
, vertex
);
842 num
= vertex
* lambda
;
847 zz2value(num
, ev
->x
.n
);
848 value_assign(ev
->d
, lcm
);
855 values2zz(T
->p
[p
], vertex
, dim
);
856 nump
= vertex
* lambda
;
857 if (First_Non_Zero(val
->p
, p
) == -1) {
858 value_assign(tmp
, lcm
);
859 evalue
*ET
= term(p
, nump
, &tmp
);
861 free_evalue_refs(ET
);
865 value_assign(tmp
, lcm
);
866 if (First_Non_Zero(T
->p
[p
], dim
) != -1)
867 Vector_Gcd(T
->p
[p
], dim
, &tmp
);
869 if (value_lt(tmp
, lcm
)) {
872 value_division(tmp
, lcm
, tmp
);
873 value_set_si(ev
->d
, 0);
874 ev
->x
.p
= new_enode(periodic
, VALUE_TO_INT(tmp
), p
+1);
875 value2zz(tmp
, count
);
877 value_decrement(tmp
, tmp
);
879 ZZ new_offset
= offset
- count
* nump
;
880 value_assign(val
->p
[p
], tmp
);
881 vertex_period(i
, lambda
, T
, lcm
, p
+1, val
, E
,
882 &ev
->x
.p
->arr
[VALUE_TO_INT(tmp
)], new_offset
);
883 } while (value_pos_p(tmp
));
885 vertex_period(i
, lambda
, T
, lcm
, p
+1, val
, E
, ev
, offset
);
889 static void mask_r(Matrix
*f
, int nr
, Vector
*lcm
, int p
, Vector
*val
, evalue
*ev
)
891 unsigned nparam
= lcm
->Size
;
894 Vector
* prod
= Vector_Alloc(f
->NbRows
);
895 Matrix_Vector_Product(f
, val
->p
, prod
->p
);
897 for (int i
= 0; i
< nr
; ++i
) {
898 value_modulus(prod
->p
[i
], prod
->p
[i
], f
->p
[i
][nparam
+1]);
899 isint
&= value_zero_p(prod
->p
[i
]);
901 value_set_si(ev
->d
, 1);
903 value_set_si(ev
->x
.n
, isint
);
910 if (value_one_p(lcm
->p
[p
]))
911 mask_r(f
, nr
, lcm
, p
+1, val
, ev
);
913 value_assign(tmp
, lcm
->p
[p
]);
914 value_set_si(ev
->d
, 0);
915 ev
->x
.p
= new_enode(periodic
, VALUE_TO_INT(tmp
), p
+1);
917 value_decrement(tmp
, tmp
);
918 value_assign(val
->p
[p
], tmp
);
919 mask_r(f
, nr
, lcm
, p
+1, val
, &ev
->x
.p
->arr
[VALUE_TO_INT(tmp
)]);
920 } while (value_pos_p(tmp
));
925 static evalue
*multi_monom(vec_ZZ
& p
)
927 evalue
*X
= new evalue();
930 unsigned nparam
= p
.length()-1;
931 zz2value(p
[nparam
], X
->x
.n
);
932 value_set_si(X
->d
, 1);
933 for (int i
= 0; i
< nparam
; ++i
) {
936 evalue
*T
= term(i
, p
[i
]);
945 * Check whether mapping polyhedron P on the affine combination
946 * num yields a range that has a fixed quotient on integer
948 * If zero is true, then we are only interested in the quotient
949 * for the cases where the remainder is zero.
950 * Returns NULL if false and a newly allocated value if true.
952 static Value
*fixed_quotient(Polyhedron
*P
, vec_ZZ
& num
, Value d
, bool zero
)
955 int len
= num
.length();
956 Matrix
*T
= Matrix_Alloc(2, len
);
957 zz2values(num
, T
->p
[0]);
958 value_set_si(T
->p
[1][len
-1], 1);
959 Polyhedron
*I
= Polyhedron_Image(P
, T
, P
->NbConstraints
);
963 for (i
= 0; i
< I
->NbRays
; ++i
)
964 if (value_zero_p(I
->Ray
[i
][2])) {
972 int bounded
= line_minmax(I
, &min
, &max
);
976 mpz_cdiv_q(min
, min
, d
);
978 mpz_fdiv_q(min
, min
, d
);
979 mpz_fdiv_q(max
, max
, d
);
981 if (value_eq(min
, max
)) {
984 value_assign(*ret
, min
);
992 * Normalize linear expression coef modulo m
993 * Removes common factor and reduces coefficients
994 * Returns index of first non-zero coefficient or len
996 static int normal_mod(Value
*coef
, int len
, Value
*m
)
1001 Vector_Gcd(coef
, len
, &gcd
);
1003 Vector_AntiScale(coef
, coef
, gcd
, len
);
1005 value_division(*m
, *m
, gcd
);
1008 if (value_one_p(*m
))
1012 for (j
= 0; j
< len
; ++j
)
1013 mpz_fdiv_r(coef
[j
], coef
[j
], *m
);
1014 for (j
= 0; j
< len
; ++j
)
1015 if (value_notzero_p(coef
[j
]))
1022 static void mask(Matrix
*f
, evalue
*factor
)
1024 int nr
= f
->NbRows
, nc
= f
->NbColumns
;
1027 for (n
= 0; n
< nr
&& value_notzero_p(f
->p
[n
][nc
-1]); ++n
)
1028 if (value_notone_p(f
->p
[n
][nc
-1]) &&
1029 value_notmone_p(f
->p
[n
][nc
-1]))
1043 value_set_si(EV
.x
.n
, 1);
1045 for (n
= 0; n
< nr
; ++n
) {
1046 value_assign(m
, f
->p
[n
][nc
-1]);
1047 if (value_one_p(m
) || value_mone_p(m
))
1050 int j
= normal_mod(f
->p
[n
], nc
-1, &m
);
1052 free_evalue_refs(factor
);
1053 value_init(factor
->d
);
1054 evalue_set_si(factor
, 0, 1);
1058 values2zz(f
->p
[n
], row
, nc
-1);
1061 if (j
< (nc
-1)-1 && row
[j
] > g
/2) {
1062 for (int k
= j
; k
< (nc
-1); ++k
)
1064 row
[k
] = g
- row
[k
];
1068 value_set_si(EP
.d
, 0);
1069 EP
.x
.p
= new_enode(relation
, 2, 0);
1070 value_clear(EP
.x
.p
->arr
[1].d
);
1071 EP
.x
.p
->arr
[1] = *factor
;
1072 evalue
*ev
= &EP
.x
.p
->arr
[0];
1073 value_set_si(ev
->d
, 0);
1074 ev
->x
.p
= new_enode(fractional
, 3, -1);
1075 evalue_set_si(&ev
->x
.p
->arr
[1], 0, 1);
1076 evalue_set_si(&ev
->x
.p
->arr
[2], 1, 1);
1077 evalue
*E
= multi_monom(row
);
1078 value_assign(EV
.d
, m
);
1080 value_clear(ev
->x
.p
->arr
[0].d
);
1081 ev
->x
.p
->arr
[0] = *E
;
1087 free_evalue_refs(&EV
);
1093 static void mask(Matrix
*f
, evalue
*factor
)
1095 int nr
= f
->NbRows
, nc
= f
->NbColumns
;
1098 for (n
= 0; n
< nr
&& value_notzero_p(f
->p
[n
][nc
-1]); ++n
)
1099 if (value_notone_p(f
->p
[n
][nc
-1]) &&
1100 value_notmone_p(f
->p
[n
][nc
-1]))
1108 unsigned np
= nc
- 2;
1109 Vector
*lcm
= Vector_Alloc(np
);
1110 Vector
*val
= Vector_Alloc(nc
);
1111 Vector_Set(val
->p
, 0, nc
);
1112 value_set_si(val
->p
[np
], 1);
1113 Vector_Set(lcm
->p
, 1, np
);
1114 for (n
= 0; n
< nr
; ++n
) {
1115 if (value_one_p(f
->p
[n
][nc
-1]) ||
1116 value_mone_p(f
->p
[n
][nc
-1]))
1118 for (int j
= 0; j
< np
; ++j
)
1119 if (value_notzero_p(f
->p
[n
][j
])) {
1120 Gcd(f
->p
[n
][j
], f
->p
[n
][nc
-1], &tmp
);
1121 value_division(tmp
, f
->p
[n
][nc
-1], tmp
);
1122 value_lcm(tmp
, lcm
->p
[j
], &lcm
->p
[j
]);
1127 mask_r(f
, nr
, lcm
, 0, val
, &EP
);
1132 free_evalue_refs(&EP
);
1143 static bool mod_needed(Polyhedron
*PD
, vec_ZZ
& num
, Value d
, evalue
*E
)
1145 Value
*q
= fixed_quotient(PD
, num
, d
, false);
1150 value_oppose(*q
, *q
);
1153 value_set_si(EV
.d
, 1);
1155 value_multiply(EV
.x
.n
, *q
, d
);
1157 free_evalue_refs(&EV
);
1163 static void ceil_mod(Value
*coef
, int len
, Value d
, ZZ
& f
, evalue
*EP
, Polyhedron
*PD
)
1167 value_set_si(m
, -1);
1169 Vector_Scale(coef
, coef
, m
, len
);
1172 int j
= normal_mod(coef
, len
, &m
);
1180 values2zz(coef
, num
, len
);
1187 evalue_set_si(&tmp
, 0, 1);
1191 while (j
< len
-1 && (num
[j
] == g
/2 || num
[j
] == 0))
1193 if ((j
< len
-1 && num
[j
] > g
/2) || (j
== len
-1 && num
[j
] >= (g
+1)/2)) {
1194 for (int k
= j
; k
< len
-1; ++k
)
1196 num
[k
] = g
- num
[k
];
1197 num
[len
-1] = g
- 1 - num
[len
-1];
1198 value_assign(tmp
.d
, m
);
1200 zz2value(t
, tmp
.x
.n
);
1206 ZZ t
= num
[len
-1] * f
;
1207 zz2value(t
, tmp
.x
.n
);
1208 value_assign(tmp
.d
, m
);
1211 evalue
*E
= multi_monom(num
);
1215 if (PD
&& !mod_needed(PD
, num
, m
, E
)) {
1217 zz2value(f
, EV
.x
.n
);
1218 value_assign(EV
.d
, m
);
1223 value_set_si(EV
.x
.n
, 1);
1224 value_assign(EV
.d
, m
);
1226 value_clear(EV
.x
.n
);
1227 value_set_si(EV
.d
, 0);
1228 EV
.x
.p
= new_enode(fractional
, 3, -1);
1229 evalue_copy(&EV
.x
.p
->arr
[0], E
);
1230 evalue_set_si(&EV
.x
.p
->arr
[1], 0, 1);
1231 value_init(EV
.x
.p
->arr
[2].x
.n
);
1232 zz2value(f
, EV
.x
.p
->arr
[2].x
.n
);
1233 value_set_si(EV
.x
.p
->arr
[2].d
, 1);
1238 free_evalue_refs(&EV
);
1239 free_evalue_refs(E
);
1243 free_evalue_refs(&tmp
);
1249 evalue
* bv_ceil3(Value
*coef
, int len
, Value d
, Polyhedron
*P
)
1251 Vector
*val
= Vector_Alloc(len
);
1255 value_set_si(t
, -1);
1256 Vector_Scale(coef
, val
->p
, t
, len
);
1257 value_absolute(t
, d
);
1260 values2zz(val
->p
, num
, len
);
1261 evalue
*EP
= multi_monom(num
);
1265 value_init(tmp
.x
.n
);
1266 value_set_si(tmp
.x
.n
, 1);
1267 value_assign(tmp
.d
, t
);
1273 ceil_mod(val
->p
, len
, t
, one
, EP
, P
);
1276 /* copy EP to malloc'ed evalue */
1282 free_evalue_refs(&tmp
);
1289 evalue
* lattice_point(
1290 Polyhedron
*i
, vec_ZZ
& lambda
, Matrix
*W
, Value lcm
, Polyhedron
*PD
)
1292 unsigned nparam
= W
->NbColumns
- 1;
1294 Matrix
* Rays
= rays2(i
);
1295 Matrix
*T
= Transpose(Rays
);
1296 Matrix
*T2
= Matrix_Copy(T
);
1297 Matrix
*inv
= Matrix_Alloc(T2
->NbRows
, T2
->NbColumns
);
1298 int ok
= Matrix_Inverse(T2
, inv
);
1303 matrix2zz(W
, vertex
, W
->NbRows
, W
->NbColumns
);
1306 num
= lambda
* vertex
;
1308 evalue
*EP
= multi_monom(num
);
1312 value_init(tmp
.x
.n
);
1313 value_set_si(tmp
.x
.n
, 1);
1314 value_assign(tmp
.d
, lcm
);
1318 Matrix
*L
= Matrix_Alloc(inv
->NbRows
, W
->NbColumns
);
1319 Matrix_Product(inv
, W
, L
);
1322 matrix2zz(T
, RT
, T
->NbRows
, T
->NbColumns
);
1325 vec_ZZ p
= lambda
* RT
;
1327 for (int i
= 0; i
< L
->NbRows
; ++i
) {
1328 ceil_mod(L
->p
[i
], nparam
+1, lcm
, p
[i
], EP
, PD
);
1334 free_evalue_refs(&tmp
);
1338 evalue
* lattice_point(
1339 Polyhedron
*i
, vec_ZZ
& lambda
, Matrix
*W
, Value lcm
, Polyhedron
*PD
)
1341 Matrix
*T
= Transpose(W
);
1342 unsigned nparam
= T
->NbRows
- 1;
1344 evalue
*EP
= new evalue();
1346 evalue_set_si(EP
, 0, 1);
1349 Vector
*val
= Vector_Alloc(nparam
+1);
1350 value_set_si(val
->p
[nparam
], 1);
1351 ZZ
offset(INIT_VAL
, 0);
1353 vertex_period(i
, lambda
, T
, lcm
, 0, val
, EP
, &ev
, offset
);
1356 free_evalue_refs(&ev
);
1367 Param_Vertices
* V
, Polyhedron
*i
, vec_ZZ
& lambda
, term_info
* term
,
1370 unsigned nparam
= V
->Vertex
->NbColumns
- 2;
1371 unsigned dim
= i
->Dimension
;
1373 vertex
.SetDims(V
->Vertex
->NbRows
, nparam
+1);
1377 value_set_si(lcm
, 1);
1378 for (int j
= 0; j
< V
->Vertex
->NbRows
; ++j
) {
1379 value_lcm(lcm
, V
->Vertex
->p
[j
][nparam
+1], &lcm
);
1381 if (value_notone_p(lcm
)) {
1382 Matrix
* mv
= Matrix_Alloc(dim
, nparam
+1);
1383 for (int j
= 0 ; j
< dim
; ++j
) {
1384 value_division(tmp
, lcm
, V
->Vertex
->p
[j
][nparam
+1]);
1385 Vector_Scale(V
->Vertex
->p
[j
], mv
->p
[j
], tmp
, nparam
+1);
1388 term
->E
= lattice_point(i
, lambda
, mv
, lcm
, PD
);
1396 for (int i
= 0; i
< V
->Vertex
->NbRows
; ++i
) {
1397 assert(value_one_p(V
->Vertex
->p
[i
][nparam
+1])); // for now
1398 values2zz(V
->Vertex
->p
[i
], vertex
[i
], nparam
+1);
1402 num
= lambda
* vertex
;
1406 for (int j
= 0; j
< nparam
; ++j
)
1412 term
->E
= multi_monom(num
);
1416 term
->constant
= num
[nparam
];
1419 term
->coeff
= num
[p
];
1426 static void normalize(ZZ
& sign
, ZZ
& num
, vec_ZZ
& den
)
1428 unsigned dim
= den
.length();
1432 for (int j
= 0; j
< den
.length(); ++j
) {
1436 den
[j
] = abs(den
[j
]);
1445 * f: the powers in the denominator for the remaining vars
1446 * each row refers to a factor
1447 * den_s: for each factor, the power of (s+1)
1449 * num_s: powers in the numerator corresponding to the summed vars
1450 * num_p: powers in the numerator corresponidng to the remaining vars
1451 * number of rays in cone: "dim" = "k"
1452 * length of each ray: "dim" = "d"
1453 * for now, it is assume: k == d
1455 * den_p: for each factor
1456 * 0: independent of remaining vars
1457 * 1: power corresponds to corresponding row in f
1458 * -1: power is inverse of corresponding row in f
1460 static void normalize(ZZ
& sign
,
1461 ZZ
& num_s
, vec_ZZ
& num_p
, vec_ZZ
& den_s
, vec_ZZ
& den_p
,
1464 unsigned dim
= f
.NumRows();
1465 unsigned nparam
= num_p
.length();
1466 unsigned nvar
= dim
- nparam
;
1470 for (int j
= 0; j
< den_s
.length(); ++j
) {
1471 if (den_s
[j
] == 0) {
1476 for (k
= 0; k
< nparam
; ++k
)
1490 den_s
[j
] = abs(den_s
[j
]);
1499 struct counter
: public polar_decomposer
{
1511 counter(Polyhedron
*P
) {
1514 randomvector(P
, lambda
, dim
);
1515 rays
.SetDims(dim
, dim
);
1520 void start(unsigned MaxRays
);
1526 virtual void handle_polar(Polyhedron
*P
, int sign
);
1529 void counter::handle_polar(Polyhedron
*C
, int s
)
1532 assert(C
->NbRays
-1 == dim
);
1533 add_rays(rays
, C
, &r
);
1534 for (int k
= 0; k
< dim
; ++k
) {
1535 assert(lambda
* rays
[k
] != 0);
1540 lattice_point(P
->Ray
[j
]+1, C
, vertex
);
1541 num
= vertex
* lambda
;
1542 den
= rays
* lambda
;
1543 normalize(sign
, num
, den
);
1546 dpoly
n(dim
, den
[0], 1);
1547 for (int k
= 1; k
< dim
; ++k
) {
1548 dpoly
fact(dim
, den
[k
], 1);
1551 d
.div(n
, count
, sign
);
1554 void counter::start(unsigned MaxRays
)
1556 for (j
= 0; j
< P
->NbRays
; ++j
) {
1557 Polyhedron
*C
= supporting_cone(P
, j
);
1558 decompose(C
, MaxRays
);
1562 // incremental counter
1563 struct icounter
: public polar_decomposer
{
1578 icounter(Polyhedron
*P
) {
1581 lambda
.SetLength(1);
1583 //den.SetLength(dim);
1591 void start(unsigned MaxRays
);
1600 virtual void handle_polar(Polyhedron
*P
, int sign
);
1601 void reduce(ZZ
& c
, ZZ
& cd
, vec_ZZ
& num
, vec_ZZ
& den_s
, mat_ZZ
& den
);
1602 void recurse(ZZ c
, ZZ cd
, vec_ZZ
& num
, mat_ZZ
& den
);
1605 void icounter::recurse(ZZ c
, ZZ cd
, vec_ZZ
& num
, mat_ZZ
& den
)
1607 unsigned d
= num
.length();
1608 unsigned len
= den
.NumRows(); // number of factors in den
1611 den_s
.SetLength(len
);
1613 den_r
.SetDims(len
, d
-1);
1616 for (r
= 0; r
< len
; ++r
) {
1617 den_s
[r
] = den
[r
][0];
1618 for (k
= 1; k
< d
; ++k
)
1619 den_r
[r
][k
-1] = den
[r
][k
];
1624 normalize(c
, num_s
, den_s
);
1626 dpoly
n(len
, num_s
);
1627 dpoly
D(len
, den_s
[0], 1);
1628 for (int k
= 1; k
< len
; ++k
) {
1629 dpoly
fact(len
, den_s
[k
], 1);
1632 mpq_set_si(tcount
, 0, 1);
1633 n
.div(D
, tcount
, one
);
1636 mpz_mul(mpq_numref(tcount
), mpq_numref(tcount
), tn
);
1637 mpz_mul(mpq_denref(tcount
), mpq_denref(tcount
), td
);
1638 mpq_canonicalize(tcount
);
1639 mpq_add(count
, count
, tcount
);
1641 reduce(c
, cd
, num
, den_s
, den_r
);
1645 void icounter::reduce(ZZ
& c
, ZZ
& cd
, vec_ZZ
& num
, vec_ZZ
& den_s
, mat_ZZ
& den
)
1647 assert(num
.length() > 1);
1648 unsigned d
= num
.length()-1;
1649 unsigned len
= den_s
.length(); // number of factors in den
1654 for (k
= 1 ; k
<= d
; ++k
)
1655 num_p
[k
-1] = num
[k
];
1658 den_p
.SetLength(len
);
1660 normalize(c
, num_s
, num_p
, den_s
, den_p
, den
);
1662 /* Since we're working incrementally, we should look
1663 * for the "easiest" parameter first.
1664 * In particular we should handle the parameters such
1665 * that no_param + only_param == len, since that allows
1666 * us to decouple the problem and the split off part
1667 * may very well be zero
1671 for (int k
= 0; k
< len
; ++k
) {
1674 else if (den_s
[k
] == 0)
1677 if (no_param
== 0) {
1678 for (int k
= 0; k
< len
; ++k
)
1681 recurse(c
, cd
, num_p
, den
);
1685 pden
.SetDims(only_param
, d
);
1687 if (no_param
+ only_param
== len
) {
1688 for (k
= 0, l
= 0; k
< len
; ++k
)
1692 for (k
= 0; k
< len
; ++k
)
1696 dpoly
n(no_param
, num_s
);
1697 dpoly
D(no_param
, den_s
[k
], 1);
1698 for ( ; ++k
< len
; )
1699 if (den_s
[k
] != 0) {
1700 dpoly
fact(no_param
, den_s
[k
], 1);
1704 mpq_set_si(tcount
, 0, 1);
1705 n
.div(D
, tcount
, one
);
1708 value2zz(mpq_numref(tcount
), qn
);
1709 value2zz(mpq_denref(tcount
), qd
);
1715 recurse(qn
, qd
, num_p
, pden
);
1719 for (k
= 0, l
= 0; k
< len
; ++k
)
1723 for (k
= 0; k
< len
; ++k
)
1727 dpoly
n(no_param
, num_s
);
1728 dpoly
D(no_param
, den_s
[k
], 1);
1729 for ( ; ++k
< len
; )
1730 if (den_p
[k
] == 0) {
1731 dpoly
fact(no_param
, den_s
[k
], 1);
1735 for (k
= 0; k
< len
; ++k
) {
1736 if (den_s
[k
] == 0 || den_p
[k
] == 0)
1739 dpoly
pd(no_param
-1, den_s
[k
], 1);
1740 int s
= den_p
[k
] < 0 ? -1 : 1;
1743 r
= new dpoly_r(n
, pd
, k
, s
, len
);
1745 dpoly_r
*nr
= new dpoly_r(r
, pd
, k
, s
, len
);
1751 dpoly_r
*rc
= r
->div(D
);
1755 int common
= pden
.NumRows();
1756 vector
< dpoly_r_term
* >& final
= rc
->c
[rc
->len
-1];
1758 for (int j
= 0; j
< final
.size(); ++j
) {
1759 if (final
[j
]->coeff
== 0)
1762 pden
.SetDims(rows
, pden
.NumCols());
1763 for (int k
= 0; k
< rc
->dim
; ++k
) {
1764 int n
= final
[j
]->powers
[k
];
1767 int abs_n
= n
< 0 ? -n
: n
;
1768 pden
.SetDims(rows
+abs_n
, pden
.NumCols());
1769 for (int l
= 0; l
< abs_n
; ++l
) {
1771 pden
[rows
+l
] = den
[k
];
1773 pden
[rows
+l
] = -den
[k
];
1777 final
[j
]->coeff
*= c
;
1778 recurse(final
[j
]->coeff
, rc
->denom
, num_p
, pden
);
1787 void icounter::handle_polar(Polyhedron
*C
, int s
)
1789 assert(C
->NbRays
-1 == dim
);
1793 lattice_point(P
->Ray
[j
]+1, C
, vertex
);
1796 den_s
.SetLength(dim
);
1798 den
.SetDims(dim
, dim
-1);
1801 for (r
= 0; r
< dim
; ++r
) {
1802 value2zz(C
->Ray
[r
][1], den_s
[r
]);
1803 values2zz(C
->Ray
[r
]+1+1, den
[r
], dim
-1);
1806 reduce(sgn
, one
, vertex
, den_s
, den
);
1809 void icounter::start(unsigned MaxRays
)
1811 for (j
= 0; j
< P
->NbRays
; ++j
) {
1812 Polyhedron
*C
= supporting_cone(P
, j
);
1813 decompose(C
, MaxRays
);
1817 typedef Polyhedron
* Polyhedron_p
;
1819 void barvinok_count(Polyhedron
*P
, Value
* result
, unsigned NbMaxCons
)
1821 Polyhedron
** vcone
;
1830 value_set_si(*result
, 0);
1834 for (; r
< P
->NbRays
; ++r
)
1835 if (value_zero_p(P
->Ray
[r
][P
->Dimension
+1]))
1837 if (P
->NbBid
!=0 || r
< P
->NbRays
) {
1838 value_set_si(*result
, -1);
1842 P
= remove_equalities(P
);
1845 value_set_si(*result
, 0);
1851 value_set_si(factor
, 1);
1852 Q
= Polyhedron_Reduce(P
, &factor
);
1859 if (P
->Dimension
== 0) {
1860 value_assign(*result
, factor
);
1863 value_clear(factor
);
1868 cnt
.start(NbMaxCons
);
1870 assert(value_one_p(&cnt
.count
[0]._mp_den
));
1871 value_multiply(*result
, &cnt
.count
[0]._mp_num
, factor
);
1875 value_clear(factor
);
1878 static void uni_polynom(int param
, Vector
*c
, evalue
*EP
)
1880 unsigned dim
= c
->Size
-2;
1882 value_set_si(EP
->d
,0);
1883 EP
->x
.p
= new_enode(polynomial
, dim
+1, param
+1);
1884 for (int j
= 0; j
<= dim
; ++j
)
1885 evalue_set(&EP
->x
.p
->arr
[j
], c
->p
[j
], c
->p
[dim
+1]);
1888 static void multi_polynom(Vector
*c
, evalue
* X
, evalue
*EP
)
1890 unsigned dim
= c
->Size
-2;
1894 evalue_set(&EC
, c
->p
[dim
], c
->p
[dim
+1]);
1897 evalue_set(EP
, c
->p
[dim
], c
->p
[dim
+1]);
1899 for (int i
= dim
-1; i
>= 0; --i
) {
1901 value_assign(EC
.x
.n
, c
->p
[i
]);
1904 free_evalue_refs(&EC
);
1907 Polyhedron
*unfringe (Polyhedron
*P
, unsigned MaxRays
)
1909 int len
= P
->Dimension
+2;
1910 Polyhedron
*T
, *R
= P
;
1913 Vector
*row
= Vector_Alloc(len
);
1914 value_set_si(row
->p
[0], 1);
1916 R
= DomainConstraintSimplify(Polyhedron_Copy(P
), MaxRays
);
1918 Matrix
*M
= Matrix_Alloc(2, len
-1);
1919 value_set_si(M
->p
[1][len
-2], 1);
1920 for (int v
= 0; v
< P
->Dimension
; ++v
) {
1921 value_set_si(M
->p
[0][v
], 1);
1922 Polyhedron
*I
= Polyhedron_Image(P
, M
, 2+1);
1923 value_set_si(M
->p
[0][v
], 0);
1924 for (int r
= 0; r
< I
->NbConstraints
; ++r
) {
1925 if (value_zero_p(I
->Constraint
[r
][0]))
1927 if (value_zero_p(I
->Constraint
[r
][1]))
1929 if (value_one_p(I
->Constraint
[r
][1]))
1931 if (value_mone_p(I
->Constraint
[r
][1]))
1933 value_absolute(g
, I
->Constraint
[r
][1]);
1934 Vector_Set(row
->p
+1, 0, len
-2);
1935 value_division(row
->p
[1+v
], I
->Constraint
[r
][1], g
);
1936 mpz_fdiv_q(row
->p
[len
-1], I
->Constraint
[r
][2], g
);
1938 R
= AddConstraints(row
->p
, 1, R
, MaxRays
);
1950 static Polyhedron
*reduce_domain(Polyhedron
*D
, Matrix
*CT
, Polyhedron
*CEq
,
1951 Polyhedron
**fVD
, int nd
, unsigned MaxRays
)
1956 Dt
= CT
? DomainPreimage(D
, CT
, MaxRays
) : D
;
1957 Polyhedron
*rVD
= DomainIntersection(Dt
, CEq
, MaxRays
);
1959 /* if rVD is empty or too small in geometric dimension */
1960 if(!rVD
|| emptyQ(rVD
) ||
1961 (rVD
->Dimension
-rVD
->NbEq
< Dt
->Dimension
-Dt
->NbEq
-CEq
->NbEq
)) {
1966 return 0; /* empty validity domain */
1972 fVD
[nd
] = Domain_Copy(rVD
);
1973 for (int i
= 0 ; i
< nd
; ++i
) {
1974 Polyhedron
*I
= DomainIntersection(fVD
[nd
], fVD
[i
], MaxRays
);
1979 Polyhedron
*F
= DomainSimplify(I
, fVD
[nd
], MaxRays
);
1981 Polyhedron
*T
= rVD
;
1982 rVD
= DomainDifference(rVD
, F
, MaxRays
);
1989 rVD
= DomainConstraintSimplify(rVD
, MaxRays
);
1991 Domain_Free(fVD
[nd
]);
1998 barvinok_count(rVD
, &c
, MaxRays
);
1999 if (value_zero_p(c
)) {
2008 static bool Polyhedron_is_infinite(Polyhedron
*P
, unsigned nparam
)
2011 for (r
= 0; r
< P
->NbRays
; ++r
)
2012 if (value_zero_p(P
->Ray
[r
][0]) ||
2013 value_zero_p(P
->Ray
[r
][P
->Dimension
+1])) {
2015 for (i
= P
->Dimension
- nparam
; i
< P
->Dimension
; ++i
)
2016 if (value_notzero_p(P
->Ray
[r
][i
+1]))
2018 if (i
>= P
->Dimension
)
2021 return r
< P
->NbRays
;
2024 /* Check whether all rays point in the positive directions
2025 * for the parameters
2027 static bool Polyhedron_has_positive_rays(Polyhedron
*P
, unsigned nparam
)
2030 for (r
= 0; r
< P
->NbRays
; ++r
)
2031 if (value_zero_p(P
->Ray
[r
][P
->Dimension
+1])) {
2033 for (i
= P
->Dimension
- nparam
; i
< P
->Dimension
; ++i
)
2034 if (value_neg_p(P
->Ray
[r
][i
+1]))
2040 typedef evalue
* evalue_p
;
2042 struct enumerator
: public polar_decomposer
{
2056 enumerator(Polyhedron
*P
, unsigned dim
, unsigned nbV
) {
2060 randomvector(P
, lambda
, dim
);
2061 rays
.SetDims(dim
, dim
);
2063 c
= Vector_Alloc(dim
+2);
2065 vE
= new evalue_p
[nbV
];
2066 for (int j
= 0; j
< nbV
; ++j
)
2072 void decompose_at(Param_Vertices
*V
, int _i
, unsigned MaxRays
) {
2073 Polyhedron
*C
= supporting_cone_p(P
, V
);
2077 vE
[_i
] = new evalue
;
2078 value_init(vE
[_i
]->d
);
2079 evalue_set_si(vE
[_i
], 0, 1);
2081 decompose(C
, MaxRays
);
2088 for (int j
= 0; j
< nbV
; ++j
)
2090 free_evalue_refs(vE
[j
]);
2096 virtual void handle_polar(Polyhedron
*P
, int sign
);
2099 void enumerator::handle_polar(Polyhedron
*C
, int s
)
2102 assert(C
->NbRays
-1 == dim
);
2103 add_rays(rays
, C
, &r
);
2104 for (int k
= 0; k
< dim
; ++k
) {
2105 assert(lambda
* rays
[k
] != 0);
2110 lattice_point(V
, C
, lambda
, &num
, 0);
2111 den
= rays
* lambda
;
2112 normalize(sign
, num
.constant
, den
);
2114 dpoly
n(dim
, den
[0], 1);
2115 for (int k
= 1; k
< dim
; ++k
) {
2116 dpoly
fact(dim
, den
[k
], 1);
2119 if (num
.E
!= NULL
) {
2120 ZZ
one(INIT_VAL
, 1);
2121 dpoly_n
d(dim
, num
.constant
, one
);
2124 multi_polynom(c
, num
.E
, &EV
);
2126 free_evalue_refs(&EV
);
2127 free_evalue_refs(num
.E
);
2129 } else if (num
.pos
!= -1) {
2130 dpoly_n
d(dim
, num
.constant
, num
.coeff
);
2133 uni_polynom(num
.pos
, c
, &EV
);
2135 free_evalue_refs(&EV
);
2137 mpq_set_si(count
, 0, 1);
2138 dpoly
d(dim
, num
.constant
);
2139 d
.div(n
, count
, sign
);
2142 evalue_set(&EV
, &count
[0]._mp_num
, &count
[0]._mp_den
);
2144 free_evalue_refs(&EV
);
2148 evalue
* barvinok_enumerate_ev(Polyhedron
*P
, Polyhedron
* C
, unsigned MaxRays
)
2150 //P = unfringe(P, MaxRays);
2151 Polyhedron
*CEq
= NULL
, *rVD
, *pVD
, *CA
;
2153 Param_Polyhedron
*PP
= NULL
;
2154 Param_Domain
*D
, *next
;
2157 unsigned nparam
= C
->Dimension
;
2159 ALLOC(evalue
, eres
);
2160 value_init(eres
->d
);
2161 value_set_si(eres
->d
, 0);
2164 value_init(factor
.d
);
2165 evalue_set_si(&factor
, 1, 1);
2167 CA
= align_context(C
, P
->Dimension
, MaxRays
);
2168 P
= DomainIntersection(P
, CA
, MaxRays
);
2169 Polyhedron_Free(CA
);
2171 if (C
->Dimension
== 0 || emptyQ(P
)) {
2173 eres
->x
.p
= new_enode(partition
, 2, C
->Dimension
);
2174 EVALUE_SET_DOMAIN(eres
->x
.p
->arr
[0],
2175 DomainConstraintSimplify(CEq
? CEq
: Polyhedron_Copy(C
), MaxRays
));
2176 value_set_si(eres
->x
.p
->arr
[1].d
, 1);
2177 value_init(eres
->x
.p
->arr
[1].x
.n
);
2179 value_set_si(eres
->x
.p
->arr
[1].x
.n
, 0);
2181 barvinok_count(P
, &eres
->x
.p
->arr
[1].x
.n
, MaxRays
);
2183 emul(&factor
, eres
);
2184 reduce_evalue(eres
);
2185 free_evalue_refs(&factor
);
2190 Param_Polyhedron_Free(PP
);
2194 if (Polyhedron_is_infinite(P
, nparam
))
2199 P
= remove_equalities_p(P
, P
->Dimension
-nparam
, &f
);
2203 if (P
->Dimension
== nparam
) {
2205 P
= Universe_Polyhedron(0);
2209 Polyhedron
*Q
= ParamPolyhedron_Reduce(P
, P
->Dimension
-nparam
, &factor
);
2212 if (Q
->Dimension
== nparam
) {
2214 P
= Universe_Polyhedron(0);
2219 Polyhedron
*oldP
= P
;
2220 PP
= Polyhedron2Param_SimplifiedDomain(&P
,C
,MaxRays
,&CEq
,&CT
);
2222 Polyhedron_Free(oldP
);
2224 if (isIdentity(CT
)) {
2228 assert(CT
->NbRows
!= CT
->NbColumns
);
2229 if (CT
->NbRows
== 1) // no more parameters
2231 nparam
= CT
->NbRows
- 1;
2234 unsigned dim
= P
->Dimension
- nparam
;
2236 enumerator
et(P
, dim
, PP
->nbV
);
2239 for (nd
= 0, D
=PP
->D
; D
; ++nd
, D
=D
->next
);
2240 struct section
{ Polyhedron
*D
; evalue E
; };
2241 section
*s
= new section
[nd
];
2242 Polyhedron
**fVD
= new Polyhedron_p
[nd
];
2244 for(nd
= 0, D
=PP
->D
; D
; D
=next
) {
2247 Polyhedron
*rVD
= reduce_domain(D
->Domain
, CT
, CEq
,
2252 pVD
= CT
? DomainImage(rVD
,CT
,MaxRays
) : rVD
;
2254 value_init(s
[nd
].E
.d
);
2255 evalue_set_si(&s
[nd
].E
, 0, 1);
2257 FORALL_PVertex_in_ParamPolyhedron(V
,D
,PP
) // _i is internal counter
2259 et
.decompose_at(V
, _i
, MaxRays
);
2260 eadd(et
.vE
[_i
] , &s
[nd
].E
);
2261 END_FORALL_PVertex_in_ParamPolyhedron
;
2262 reduce_in_domain(&s
[nd
].E
, pVD
);
2265 addeliminatedparams_evalue(&s
[nd
].E
, CT
);
2273 evalue_set_si(eres
, 0, 1);
2275 eres
->x
.p
= new_enode(partition
, 2*nd
, C
->Dimension
);
2276 for (int j
= 0; j
< nd
; ++j
) {
2277 EVALUE_SET_DOMAIN(eres
->x
.p
->arr
[2*j
], s
[j
].D
);
2278 value_clear(eres
->x
.p
->arr
[2*j
+1].d
);
2279 eres
->x
.p
->arr
[2*j
+1] = s
[j
].E
;
2280 Domain_Free(fVD
[j
]);
2288 Polyhedron_Free(CEq
);
2293 Enumeration
* barvinok_enumerate(Polyhedron
*P
, Polyhedron
* C
, unsigned MaxRays
)
2295 evalue
*EP
= barvinok_enumerate_ev(P
, C
, MaxRays
);
2297 return partition2enumeration(EP
);
2300 static void SwapColumns(Value
**V
, int n
, int i
, int j
)
2302 for (int r
= 0; r
< n
; ++r
)
2303 value_swap(V
[r
][i
], V
[r
][j
]);
2306 static void SwapColumns(Polyhedron
*P
, int i
, int j
)
2308 SwapColumns(P
->Constraint
, P
->NbConstraints
, i
, j
);
2309 SwapColumns(P
->Ray
, P
->NbRays
, i
, j
);
2312 static void negative_test_constraint(Value
*l
, Value
*u
, Value
*c
, int pos
,
2315 value_oppose(*v
, u
[pos
+1]);
2316 Vector_Combine(l
+1, u
+1, c
+1, *v
, l
[pos
+1], len
-1);
2317 value_multiply(*v
, *v
, l
[pos
+1]);
2318 value_substract(c
[len
-1], c
[len
-1], *v
);
2319 value_set_si(*v
, -1);
2320 Vector_Scale(c
+1, c
+1, *v
, len
-1);
2321 value_decrement(c
[len
-1], c
[len
-1]);
2322 ConstraintSimplify(c
, c
, len
, v
);
2325 static bool parallel_constraints(Value
*l
, Value
*u
, Value
*c
, int pos
,
2334 Vector_Gcd(&l
[1+pos
], len
, &g1
);
2335 Vector_Gcd(&u
[1+pos
], len
, &g2
);
2336 Vector_Combine(l
+1+pos
, u
+1+pos
, c
+1, g2
, g1
, len
);
2337 parallel
= First_Non_Zero(c
+1, len
) == -1;
2345 static void negative_test_constraint7(Value
*l
, Value
*u
, Value
*c
, int pos
,
2346 int exist
, int len
, Value
*v
)
2351 Vector_Gcd(&u
[1+pos
], exist
, v
);
2352 Vector_Gcd(&l
[1+pos
], exist
, &g
);
2353 Vector_Combine(l
+1, u
+1, c
+1, *v
, g
, len
-1);
2354 value_multiply(*v
, *v
, g
);
2355 value_substract(c
[len
-1], c
[len
-1], *v
);
2356 value_set_si(*v
, -1);
2357 Vector_Scale(c
+1, c
+1, *v
, len
-1);
2358 value_decrement(c
[len
-1], c
[len
-1]);
2359 ConstraintSimplify(c
, c
, len
, v
);
2364 static void oppose_constraint(Value
*c
, int len
, Value
*v
)
2366 value_set_si(*v
, -1);
2367 Vector_Scale(c
+1, c
+1, *v
, len
-1);
2368 value_decrement(c
[len
-1], c
[len
-1]);
2371 static bool SplitOnConstraint(Polyhedron
*P
, int i
, int l
, int u
,
2372 int nvar
, int len
, int exist
, int MaxRays
,
2373 Vector
*row
, Value
& f
, bool independent
,
2374 Polyhedron
**pos
, Polyhedron
**neg
)
2376 negative_test_constraint(P
->Constraint
[l
], P
->Constraint
[u
],
2377 row
->p
, nvar
+i
, len
, &f
);
2378 *neg
= AddConstraints(row
->p
, 1, P
, MaxRays
);
2380 /* We found an independent, but useless constraint
2381 * Maybe we should detect this earlier and not
2382 * mark the variable as INDEPENDENT
2384 if (emptyQ((*neg
))) {
2385 Polyhedron_Free(*neg
);
2389 oppose_constraint(row
->p
, len
, &f
);
2390 *pos
= AddConstraints(row
->p
, 1, P
, MaxRays
);
2392 if (emptyQ((*pos
))) {
2393 Polyhedron_Free(*neg
);
2394 Polyhedron_Free(*pos
);
2402 * unimodularly transform P such that constraint r is transformed
2403 * into a constraint that involves only a single (the first)
2404 * existential variable
2407 static Polyhedron
*rotate_along(Polyhedron
*P
, int r
, int nvar
, int exist
,
2413 Vector
*row
= Vector_Alloc(exist
);
2414 Vector_Copy(P
->Constraint
[r
]+1+nvar
, row
->p
, exist
);
2415 Vector_Gcd(row
->p
, exist
, &g
);
2416 if (value_notone_p(g
))
2417 Vector_AntiScale(row
->p
, row
->p
, g
, exist
);
2420 Matrix
*M
= unimodular_complete(row
);
2421 Matrix
*M2
= Matrix_Alloc(P
->Dimension
+1, P
->Dimension
+1);
2422 for (r
= 0; r
< nvar
; ++r
)
2423 value_set_si(M2
->p
[r
][r
], 1);
2424 for ( ; r
< nvar
+exist
; ++r
)
2425 Vector_Copy(M
->p
[r
-nvar
], M2
->p
[r
]+nvar
, exist
);
2426 for ( ; r
< P
->Dimension
+1; ++r
)
2427 value_set_si(M2
->p
[r
][r
], 1);
2428 Polyhedron
*T
= Polyhedron_Image(P
, M2
, MaxRays
);
2437 static bool SplitOnVar(Polyhedron
*P
, int i
,
2438 int nvar
, int len
, int exist
, int MaxRays
,
2439 Vector
*row
, Value
& f
, bool independent
,
2440 Polyhedron
**pos
, Polyhedron
**neg
)
2444 for (int l
= P
->NbEq
; l
< P
->NbConstraints
; ++l
) {
2445 if (value_negz_p(P
->Constraint
[l
][nvar
+i
+1]))
2449 for (j
= 0; j
< exist
; ++j
)
2450 if (j
!= i
&& value_notzero_p(P
->Constraint
[l
][nvar
+j
+1]))
2456 for (int u
= P
->NbEq
; u
< P
->NbConstraints
; ++u
) {
2457 if (value_posz_p(P
->Constraint
[u
][nvar
+i
+1]))
2461 for (j
= 0; j
< exist
; ++j
)
2462 if (j
!= i
&& value_notzero_p(P
->Constraint
[u
][nvar
+j
+1]))
2468 if (SplitOnConstraint(P
, i
, l
, u
,
2469 nvar
, len
, exist
, MaxRays
,
2470 row
, f
, independent
,
2474 SwapColumns(*neg
, nvar
+1, nvar
+1+i
);
2484 static bool double_bound_pair(Polyhedron
*P
, int nvar
, int exist
,
2485 int i
, int l1
, int l2
,
2486 Polyhedron
**pos
, Polyhedron
**neg
)
2490 Vector
*row
= Vector_Alloc(P
->Dimension
+2);
2491 value_set_si(row
->p
[0], 1);
2492 value_oppose(f
, P
->Constraint
[l1
][nvar
+i
+1]);
2493 Vector_Combine(P
->Constraint
[l1
]+1, P
->Constraint
[l2
]+1,
2495 P
->Constraint
[l2
][nvar
+i
+1], f
,
2497 ConstraintSimplify(row
->p
, row
->p
, P
->Dimension
+2, &f
);
2498 *pos
= AddConstraints(row
->p
, 1, P
, 0);
2499 value_set_si(f
, -1);
2500 Vector_Scale(row
->p
+1, row
->p
+1, f
, P
->Dimension
+1);
2501 value_decrement(row
->p
[P
->Dimension
+1], row
->p
[P
->Dimension
+1]);
2502 *neg
= AddConstraints(row
->p
, 1, P
, 0);
2506 return !emptyQ((*pos
)) && !emptyQ((*neg
));
2509 static bool double_bound(Polyhedron
*P
, int nvar
, int exist
,
2510 Polyhedron
**pos
, Polyhedron
**neg
)
2512 for (int i
= 0; i
< exist
; ++i
) {
2514 for (l1
= P
->NbEq
; l1
< P
->NbConstraints
; ++l1
) {
2515 if (value_negz_p(P
->Constraint
[l1
][nvar
+i
+1]))
2517 for (l2
= l1
+ 1; l2
< P
->NbConstraints
; ++l2
) {
2518 if (value_negz_p(P
->Constraint
[l2
][nvar
+i
+1]))
2520 if (double_bound_pair(P
, nvar
, exist
, i
, l1
, l2
, pos
, neg
))
2524 for (l1
= P
->NbEq
; l1
< P
->NbConstraints
; ++l1
) {
2525 if (value_posz_p(P
->Constraint
[l1
][nvar
+i
+1]))
2527 if (l1
< P
->NbConstraints
)
2528 for (l2
= l1
+ 1; l2
< P
->NbConstraints
; ++l2
) {
2529 if (value_posz_p(P
->Constraint
[l2
][nvar
+i
+1]))
2531 if (double_bound_pair(P
, nvar
, exist
, i
, l1
, l2
, pos
, neg
))
2543 INDEPENDENT
= 1 << 2,
2547 static evalue
* enumerate_or(Polyhedron
*D
,
2548 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2551 fprintf(stderr
, "\nER: Or\n");
2552 #endif /* DEBUG_ER */
2554 Polyhedron
*N
= D
->next
;
2557 barvinok_enumerate_e(D
, exist
, nparam
, MaxRays
);
2560 for (D
= N
; D
; D
= N
) {
2565 barvinok_enumerate_e(D
, exist
, nparam
, MaxRays
);
2568 free_evalue_refs(EN
);
2578 static evalue
* enumerate_sum(Polyhedron
*P
,
2579 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2581 int nvar
= P
->Dimension
- exist
- nparam
;
2582 int toswap
= nvar
< exist
? nvar
: exist
;
2583 for (int i
= 0; i
< toswap
; ++i
)
2584 SwapColumns(P
, 1 + i
, nvar
+exist
- i
);
2588 fprintf(stderr
, "\nER: Sum\n");
2589 #endif /* DEBUG_ER */
2591 evalue
*EP
= barvinok_enumerate_e(P
, exist
, nparam
, MaxRays
);
2593 for (int i
= 0; i
< /* nvar */ nparam
; ++i
) {
2594 Matrix
*C
= Matrix_Alloc(1, 1 + nparam
+ 1);
2595 value_set_si(C
->p
[0][0], 1);
2597 value_init(split
.d
);
2598 value_set_si(split
.d
, 0);
2599 split
.x
.p
= new_enode(partition
, 4, nparam
);
2600 value_set_si(C
->p
[0][1+i
], 1);
2601 Matrix
*C2
= Matrix_Copy(C
);
2602 EVALUE_SET_DOMAIN(split
.x
.p
->arr
[0],
2603 Constraints2Polyhedron(C2
, MaxRays
));
2605 evalue_set_si(&split
.x
.p
->arr
[1], 1, 1);
2606 value_set_si(C
->p
[0][1+i
], -1);
2607 value_set_si(C
->p
[0][1+nparam
], -1);
2608 EVALUE_SET_DOMAIN(split
.x
.p
->arr
[2],
2609 Constraints2Polyhedron(C
, MaxRays
));
2610 evalue_set_si(&split
.x
.p
->arr
[3], 1, 1);
2612 free_evalue_refs(&split
);
2616 evalue_range_reduction(EP
);
2618 evalue_frac2floor(EP
);
2620 evalue
*sum
= esum(EP
, nvar
);
2622 free_evalue_refs(EP
);
2626 evalue_range_reduction(EP
);
2631 static evalue
* split_sure(Polyhedron
*P
, Polyhedron
*S
,
2632 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2634 int nvar
= P
->Dimension
- exist
- nparam
;
2636 Matrix
*M
= Matrix_Alloc(exist
, S
->Dimension
+2);
2637 for (int i
= 0; i
< exist
; ++i
)
2638 value_set_si(M
->p
[i
][nvar
+i
+1], 1);
2640 S
= DomainAddRays(S
, M
, MaxRays
);
2642 Polyhedron
*F
= DomainAddRays(P
, M
, MaxRays
);
2643 Polyhedron
*D
= DomainDifference(F
, S
, MaxRays
);
2645 D
= Disjoint_Domain(D
, 0, MaxRays
);
2650 M
= Matrix_Alloc(P
->Dimension
+1-exist
, P
->Dimension
+1);
2651 for (int j
= 0; j
< nvar
; ++j
)
2652 value_set_si(M
->p
[j
][j
], 1);
2653 for (int j
= 0; j
< nparam
+1; ++j
)
2654 value_set_si(M
->p
[nvar
+j
][nvar
+exist
+j
], 1);
2655 Polyhedron
*T
= Polyhedron_Image(S
, M
, MaxRays
);
2656 evalue
*EP
= barvinok_enumerate_e(T
, 0, nparam
, MaxRays
);
2661 for (Polyhedron
*Q
= D
; Q
; Q
= Q
->next
) {
2662 Polyhedron
*N
= Q
->next
;
2664 T
= DomainIntersection(P
, Q
, MaxRays
);
2665 evalue
*E
= barvinok_enumerate_e(T
, exist
, nparam
, MaxRays
);
2667 free_evalue_refs(E
);
2676 static evalue
* enumerate_sure(Polyhedron
*P
,
2677 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2681 int nvar
= P
->Dimension
- exist
- nparam
;
2687 for (i
= 0; i
< exist
; ++i
) {
2688 Matrix
*M
= Matrix_Alloc(S
->NbConstraints
, S
->Dimension
+2);
2690 value_set_si(lcm
, 1);
2691 for (int j
= 0; j
< S
->NbConstraints
; ++j
) {
2692 if (value_negz_p(S
->Constraint
[j
][1+nvar
+i
]))
2694 if (value_one_p(S
->Constraint
[j
][1+nvar
+i
]))
2696 value_lcm(lcm
, S
->Constraint
[j
][1+nvar
+i
], &lcm
);
2699 for (int j
= 0; j
< S
->NbConstraints
; ++j
) {
2700 if (value_negz_p(S
->Constraint
[j
][1+nvar
+i
]))
2702 if (value_one_p(S
->Constraint
[j
][1+nvar
+i
]))
2704 value_division(f
, lcm
, S
->Constraint
[j
][1+nvar
+i
]);
2705 Vector_Scale(S
->Constraint
[j
], M
->p
[c
], f
, S
->Dimension
+2);
2706 value_substract(M
->p
[c
][S
->Dimension
+1],
2707 M
->p
[c
][S
->Dimension
+1],
2709 value_increment(M
->p
[c
][S
->Dimension
+1],
2710 M
->p
[c
][S
->Dimension
+1]);
2714 S
= AddConstraints(M
->p
[0], c
, S
, MaxRays
);
2729 fprintf(stderr
, "\nER: Sure\n");
2730 #endif /* DEBUG_ER */
2732 return split_sure(P
, S
, exist
, nparam
, MaxRays
);
2735 static evalue
* enumerate_sure2(Polyhedron
*P
,
2736 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2738 int nvar
= P
->Dimension
- exist
- nparam
;
2740 for (r
= 0; r
< P
->NbRays
; ++r
)
2741 if (value_one_p(P
->Ray
[r
][0]) &&
2742 value_one_p(P
->Ray
[r
][P
->Dimension
+1]))
2748 Matrix
*M
= Matrix_Alloc(nvar
+ 1 + nparam
, P
->Dimension
+2);
2749 for (int i
= 0; i
< nvar
; ++i
)
2750 value_set_si(M
->p
[i
][1+i
], 1);
2751 for (int i
= 0; i
< nparam
; ++i
)
2752 value_set_si(M
->p
[i
+nvar
][1+nvar
+exist
+i
], 1);
2753 Vector_Copy(P
->Ray
[r
]+1+nvar
, M
->p
[nvar
+nparam
]+1+nvar
, exist
);
2754 value_set_si(M
->p
[nvar
+nparam
][0], 1);
2755 value_set_si(M
->p
[nvar
+nparam
][P
->Dimension
+1], 1);
2756 Polyhedron
* F
= Rays2Polyhedron(M
, MaxRays
);
2759 Polyhedron
*I
= DomainIntersection(F
, P
, MaxRays
);
2763 fprintf(stderr
, "\nER: Sure2\n");
2764 #endif /* DEBUG_ER */
2766 return split_sure(P
, I
, exist
, nparam
, MaxRays
);
2769 static evalue
* enumerate_cyclic(Polyhedron
*P
,
2770 unsigned exist
, unsigned nparam
,
2771 evalue
* EP
, int r
, int p
, unsigned MaxRays
)
2773 int nvar
= P
->Dimension
- exist
- nparam
;
2775 /* If EP in its fractional maps only contains references
2776 * to the remainder parameter with appropriate coefficients
2777 * then we could in principle avoid adding existentially
2778 * quantified variables to the validity domains.
2779 * We'd have to replace the remainder by m { p/m }
2780 * and multiply with an appropriate factor that is one
2781 * only in the appropriate range.
2782 * This last multiplication can be avoided if EP
2783 * has a single validity domain with no (further)
2784 * constraints on the remainder parameter
2787 Matrix
*CT
= Matrix_Alloc(nparam
+1, nparam
+3);
2788 Matrix
*M
= Matrix_Alloc(1, 1+nparam
+3);
2789 for (int j
= 0; j
< nparam
; ++j
)
2791 value_set_si(CT
->p
[j
][j
], 1);
2792 value_set_si(CT
->p
[p
][nparam
+1], 1);
2793 value_set_si(CT
->p
[nparam
][nparam
+2], 1);
2794 value_set_si(M
->p
[0][1+p
], -1);
2795 value_absolute(M
->p
[0][1+nparam
], P
->Ray
[0][1+nvar
+exist
+p
]);
2796 value_set_si(M
->p
[0][1+nparam
+1], 1);
2797 Polyhedron
*CEq
= Constraints2Polyhedron(M
, 1);
2799 addeliminatedparams_enum(EP
, CT
, CEq
, MaxRays
, nparam
);
2800 Polyhedron_Free(CEq
);
2806 static void enumerate_vd_add_ray(evalue
*EP
, Matrix
*Rays
, unsigned MaxRays
)
2808 if (value_notzero_p(EP
->d
))
2811 assert(EP
->x
.p
->type
== partition
);
2812 assert(EP
->x
.p
->pos
== EVALUE_DOMAIN(EP
->x
.p
->arr
[0])->Dimension
);
2813 for (int i
= 0; i
< EP
->x
.p
->size
/2; ++i
) {
2814 Polyhedron
*D
= EVALUE_DOMAIN(EP
->x
.p
->arr
[2*i
]);
2815 Polyhedron
*N
= DomainAddRays(D
, Rays
, MaxRays
);
2816 EVALUE_SET_DOMAIN(EP
->x
.p
->arr
[2*i
], N
);
2821 static evalue
* enumerate_line(Polyhedron
*P
,
2822 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2828 fprintf(stderr
, "\nER: Line\n");
2829 #endif /* DEBUG_ER */
2831 int nvar
= P
->Dimension
- exist
- nparam
;
2833 for (i
= 0; i
< nparam
; ++i
)
2834 if (value_notzero_p(P
->Ray
[0][1+nvar
+exist
+i
]))
2837 for (j
= i
+1; j
< nparam
; ++j
)
2838 if (value_notzero_p(P
->Ray
[0][1+nvar
+exist
+i
]))
2840 assert(j
>= nparam
); // for now
2842 Matrix
*M
= Matrix_Alloc(2, P
->Dimension
+2);
2843 value_set_si(M
->p
[0][0], 1);
2844 value_set_si(M
->p
[0][1+nvar
+exist
+i
], 1);
2845 value_set_si(M
->p
[1][0], 1);
2846 value_set_si(M
->p
[1][1+nvar
+exist
+i
], -1);
2847 value_absolute(M
->p
[1][1+P
->Dimension
], P
->Ray
[0][1+nvar
+exist
+i
]);
2848 value_decrement(M
->p
[1][1+P
->Dimension
], M
->p
[1][1+P
->Dimension
]);
2849 Polyhedron
*S
= AddConstraints(M
->p
[0], 2, P
, MaxRays
);
2850 evalue
*EP
= barvinok_enumerate_e(S
, exist
, nparam
, MaxRays
);
2854 return enumerate_cyclic(P
, exist
, nparam
, EP
, 0, i
, MaxRays
);
2857 static int single_param_pos(Polyhedron
*P
, unsigned exist
, unsigned nparam
,
2860 int nvar
= P
->Dimension
- exist
- nparam
;
2861 if (First_Non_Zero(P
->Ray
[r
]+1, nvar
) != -1)
2863 int i
= First_Non_Zero(P
->Ray
[r
]+1+nvar
+exist
, nparam
);
2866 if (First_Non_Zero(P
->Ray
[r
]+1+nvar
+exist
+1, nparam
-i
-1) != -1)
2871 static evalue
* enumerate_remove_ray(Polyhedron
*P
, int r
,
2872 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2875 fprintf(stderr
, "\nER: RedundantRay\n");
2876 #endif /* DEBUG_ER */
2880 value_set_si(one
, 1);
2881 int len
= P
->NbRays
-1;
2882 Matrix
*M
= Matrix_Alloc(2 * len
, P
->Dimension
+2);
2883 Vector_Copy(P
->Ray
[0], M
->p
[0], r
* (P
->Dimension
+2));
2884 Vector_Copy(P
->Ray
[r
+1], M
->p
[r
], (len
-r
) * (P
->Dimension
+2));
2885 for (int j
= 0; j
< P
->NbRays
; ++j
) {
2888 Vector_Combine(P
->Ray
[j
], P
->Ray
[r
], M
->p
[len
+j
-(j
>r
)],
2889 one
, P
->Ray
[j
][P
->Dimension
+1], P
->Dimension
+2);
2892 P
= Rays2Polyhedron(M
, MaxRays
);
2894 evalue
*EP
= barvinok_enumerate_e(P
, exist
, nparam
, MaxRays
);
2901 static evalue
* enumerate_redundant_ray(Polyhedron
*P
,
2902 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2904 assert(P
->NbBid
== 0);
2905 int nvar
= P
->Dimension
- exist
- nparam
;
2909 for (int r
= 0; r
< P
->NbRays
; ++r
) {
2910 if (value_notzero_p(P
->Ray
[r
][P
->Dimension
+1]))
2912 int i1
= single_param_pos(P
, exist
, nparam
, r
);
2915 for (int r2
= r
+1; r2
< P
->NbRays
; ++r2
) {
2916 if (value_notzero_p(P
->Ray
[r2
][P
->Dimension
+1]))
2918 int i2
= single_param_pos(P
, exist
, nparam
, r2
);
2924 value_division(m
, P
->Ray
[r
][1+nvar
+exist
+i1
],
2925 P
->Ray
[r2
][1+nvar
+exist
+i1
]);
2926 value_multiply(m
, m
, P
->Ray
[r2
][1+nvar
+exist
+i1
]);
2927 /* r2 divides r => r redundant */
2928 if (value_eq(m
, P
->Ray
[r
][1+nvar
+exist
+i1
])) {
2930 return enumerate_remove_ray(P
, r
, exist
, nparam
, MaxRays
);
2933 value_division(m
, P
->Ray
[r2
][1+nvar
+exist
+i1
],
2934 P
->Ray
[r
][1+nvar
+exist
+i1
]);
2935 value_multiply(m
, m
, P
->Ray
[r
][1+nvar
+exist
+i1
]);
2936 /* r divides r2 => r2 redundant */
2937 if (value_eq(m
, P
->Ray
[r2
][1+nvar
+exist
+i1
])) {
2939 return enumerate_remove_ray(P
, r2
, exist
, nparam
, MaxRays
);
2947 static Polyhedron
*upper_bound(Polyhedron
*P
,
2948 int pos
, Value
*max
, Polyhedron
**R
)
2957 for (Polyhedron
*Q
= P
; Q
; Q
= N
) {
2959 for (r
= 0; r
< P
->NbRays
; ++r
) {
2960 if (value_zero_p(P
->Ray
[r
][P
->Dimension
+1]) &&
2961 value_pos_p(P
->Ray
[r
][1+pos
]))
2964 if (r
< P
->NbRays
) {
2972 for (r
= 0; r
< P
->NbRays
; ++r
) {
2973 if (value_zero_p(P
->Ray
[r
][P
->Dimension
+1]))
2975 mpz_fdiv_q(v
, P
->Ray
[r
][1+pos
], P
->Ray
[r
][1+P
->Dimension
]);
2976 if ((!Q
->next
&& r
== 0) || value_gt(v
, *max
))
2977 value_assign(*max
, v
);
2984 static evalue
* enumerate_ray(Polyhedron
*P
,
2985 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
2987 assert(P
->NbBid
== 0);
2988 int nvar
= P
->Dimension
- exist
- nparam
;
2991 for (r
= 0; r
< P
->NbRays
; ++r
)
2992 if (value_zero_p(P
->Ray
[r
][P
->Dimension
+1]))
2998 for (r2
= r
+1; r2
< P
->NbRays
; ++r2
)
2999 if (value_zero_p(P
->Ray
[r2
][P
->Dimension
+1]))
3001 if (r2
< P
->NbRays
) {
3003 return enumerate_sum(P
, exist
, nparam
, MaxRays
);
3007 fprintf(stderr
, "\nER: Ray\n");
3008 #endif /* DEBUG_ER */
3014 value_set_si(one
, 1);
3015 int i
= single_param_pos(P
, exist
, nparam
, r
);
3016 assert(i
!= -1); // for now;
3018 Matrix
*M
= Matrix_Alloc(P
->NbRays
, P
->Dimension
+2);
3019 for (int j
= 0; j
< P
->NbRays
; ++j
) {
3020 Vector_Combine(P
->Ray
[j
], P
->Ray
[r
], M
->p
[j
],
3021 one
, P
->Ray
[j
][P
->Dimension
+1], P
->Dimension
+2);
3023 Polyhedron
*S
= Rays2Polyhedron(M
, MaxRays
);
3025 Polyhedron
*D
= DomainDifference(P
, S
, MaxRays
);
3027 // Polyhedron_Print(stderr, P_VALUE_FMT, D);
3028 assert(value_pos_p(P
->Ray
[r
][1+nvar
+exist
+i
])); // for now
3030 D
= upper_bound(D
, nvar
+exist
+i
, &m
, &R
);
3034 M
= Matrix_Alloc(2, P
->Dimension
+2);
3035 value_set_si(M
->p
[0][0], 1);
3036 value_set_si(M
->p
[1][0], 1);
3037 value_set_si(M
->p
[0][1+nvar
+exist
+i
], -1);
3038 value_set_si(M
->p
[1][1+nvar
+exist
+i
], 1);
3039 value_assign(M
->p
[0][1+P
->Dimension
], m
);
3040 value_oppose(M
->p
[1][1+P
->Dimension
], m
);
3041 value_addto(M
->p
[1][1+P
->Dimension
], M
->p
[1][1+P
->Dimension
],
3042 P
->Ray
[r
][1+nvar
+exist
+i
]);
3043 value_decrement(M
->p
[1][1+P
->Dimension
], M
->p
[1][1+P
->Dimension
]);
3044 // Matrix_Print(stderr, P_VALUE_FMT, M);
3045 D
= AddConstraints(M
->p
[0], 2, P
, MaxRays
);
3046 // Polyhedron_Print(stderr, P_VALUE_FMT, D);
3047 value_substract(M
->p
[0][1+P
->Dimension
], M
->p
[0][1+P
->Dimension
],
3048 P
->Ray
[r
][1+nvar
+exist
+i
]);
3049 // Matrix_Print(stderr, P_VALUE_FMT, M);
3050 S
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3051 // Polyhedron_Print(stderr, P_VALUE_FMT, S);
3054 evalue
*EP
= barvinok_enumerate_e(D
, exist
, nparam
, MaxRays
);
3059 if (value_notone_p(P
->Ray
[r
][1+nvar
+exist
+i
]))
3060 EP
= enumerate_cyclic(P
, exist
, nparam
, EP
, r
, i
, MaxRays
);
3062 M
= Matrix_Alloc(1, nparam
+2);
3063 value_set_si(M
->p
[0][0], 1);
3064 value_set_si(M
->p
[0][1+i
], 1);
3065 enumerate_vd_add_ray(EP
, M
, MaxRays
);
3070 evalue
*E
= barvinok_enumerate_e(S
, exist
, nparam
, MaxRays
);
3072 free_evalue_refs(E
);
3079 evalue
*ER
= enumerate_or(R
, exist
, nparam
, MaxRays
);
3081 free_evalue_refs(ER
);
3088 static evalue
* new_zero_ep()
3093 evalue_set_si(EP
, 0, 1);
3097 static evalue
* enumerate_vd(Polyhedron
**PA
,
3098 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
3100 Polyhedron
*P
= *PA
;
3101 int nvar
= P
->Dimension
- exist
- nparam
;
3102 Param_Polyhedron
*PP
= NULL
;
3103 Polyhedron
*C
= Universe_Polyhedron(nparam
);
3107 PP
= Polyhedron2Param_SimplifiedDomain(&PR
,C
,MaxRays
,&CEq
,&CT
);
3111 Param_Domain
*D
, *last
;
3114 for (nd
= 0, D
=PP
->D
; D
; D
=D
->next
, ++nd
)
3117 Polyhedron
**VD
= new Polyhedron_p
[nd
];
3118 Polyhedron
**fVD
= new Polyhedron_p
[nd
];
3119 for(nd
= 0, D
=PP
->D
; D
; D
=D
->next
) {
3120 Polyhedron
*rVD
= reduce_domain(D
->Domain
, CT
, CEq
,
3134 /* This doesn't seem to have any effect */
3136 Polyhedron
*CA
= align_context(VD
[0], P
->Dimension
, MaxRays
);
3138 P
= DomainIntersection(P
, CA
, MaxRays
);
3141 Polyhedron_Free(CA
);
3146 if (!EP
&& CT
->NbColumns
!= CT
->NbRows
) {
3147 Polyhedron
*CEqr
= DomainImage(CEq
, CT
, MaxRays
);
3148 Polyhedron
*CA
= align_context(CEqr
, PR
->Dimension
, MaxRays
);
3149 Polyhedron
*I
= DomainIntersection(PR
, CA
, MaxRays
);
3150 Polyhedron_Free(CEqr
);
3151 Polyhedron_Free(CA
);
3153 fprintf(stderr
, "\nER: Eliminate\n");
3154 #endif /* DEBUG_ER */
3155 nparam
-= CT
->NbColumns
- CT
->NbRows
;
3156 EP
= barvinok_enumerate_e(I
, exist
, nparam
, MaxRays
);
3157 nparam
+= CT
->NbColumns
- CT
->NbRows
;
3158 addeliminatedparams_enum(EP
, CT
, CEq
, MaxRays
, nparam
);
3162 Polyhedron_Free(PR
);
3165 if (!EP
&& nd
> 1) {
3167 fprintf(stderr
, "\nER: VD\n");
3168 #endif /* DEBUG_ER */
3169 for (int i
= 0; i
< nd
; ++i
) {
3170 Polyhedron
*CA
= align_context(VD
[i
], P
->Dimension
, MaxRays
);
3171 Polyhedron
*I
= DomainIntersection(P
, CA
, MaxRays
);
3174 EP
= barvinok_enumerate_e(I
, exist
, nparam
, MaxRays
);
3176 evalue
*E
= barvinok_enumerate_e(I
, exist
, nparam
, MaxRays
);
3178 free_evalue_refs(E
);
3182 Polyhedron_Free(CA
);
3186 for (int i
= 0; i
< nd
; ++i
) {
3187 Polyhedron_Free(VD
[i
]);
3188 Polyhedron_Free(fVD
[i
]);
3194 if (!EP
&& nvar
== 0) {
3197 Param_Vertices
*V
, *V2
;
3198 Matrix
* M
= Matrix_Alloc(1, P
->Dimension
+2);
3200 FORALL_PVertex_in_ParamPolyhedron(V
, last
, PP
) {
3202 FORALL_PVertex_in_ParamPolyhedron(V2
, last
, PP
) {
3209 for (int i
= 0; i
< exist
; ++i
) {
3210 value_oppose(f
, V
->Vertex
->p
[i
][nparam
+1]);
3211 Vector_Combine(V
->Vertex
->p
[i
],
3213 M
->p
[0] + 1 + nvar
+ exist
,
3214 V2
->Vertex
->p
[i
][nparam
+1],
3218 for (j
= 0; j
< nparam
; ++j
)
3219 if (value_notzero_p(M
->p
[0][1+nvar
+exist
+j
]))
3223 ConstraintSimplify(M
->p
[0], M
->p
[0],
3224 P
->Dimension
+2, &f
);
3225 value_set_si(M
->p
[0][0], 0);
3226 Polyhedron
*para
= AddConstraints(M
->p
[0], 1, P
,
3229 Polyhedron_Free(para
);
3232 Polyhedron
*pos
, *neg
;
3233 value_set_si(M
->p
[0][0], 1);
3234 value_decrement(M
->p
[0][P
->Dimension
+1],
3235 M
->p
[0][P
->Dimension
+1]);
3236 neg
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3237 value_set_si(f
, -1);
3238 Vector_Scale(M
->p
[0]+1, M
->p
[0]+1, f
,
3240 value_decrement(M
->p
[0][P
->Dimension
+1],
3241 M
->p
[0][P
->Dimension
+1]);
3242 value_decrement(M
->p
[0][P
->Dimension
+1],
3243 M
->p
[0][P
->Dimension
+1]);
3244 pos
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3245 if (emptyQ(neg
) && emptyQ(pos
)) {
3246 Polyhedron_Free(para
);
3247 Polyhedron_Free(pos
);
3248 Polyhedron_Free(neg
);
3252 fprintf(stderr
, "\nER: Order\n");
3253 #endif /* DEBUG_ER */
3254 EP
= barvinok_enumerate_e(para
, exist
, nparam
, MaxRays
);
3257 E
= barvinok_enumerate_e(pos
, exist
, nparam
, MaxRays
);
3259 free_evalue_refs(E
);
3263 E
= barvinok_enumerate_e(neg
, exist
, nparam
, MaxRays
);
3265 free_evalue_refs(E
);
3268 Polyhedron_Free(para
);
3269 Polyhedron_Free(pos
);
3270 Polyhedron_Free(neg
);
3275 } END_FORALL_PVertex_in_ParamPolyhedron
;
3278 } END_FORALL_PVertex_in_ParamPolyhedron
;
3281 /* Search for vertex coordinate to split on */
3282 /* First look for one independent of the parameters */
3283 FORALL_PVertex_in_ParamPolyhedron(V
, last
, PP
) {
3284 for (int i
= 0; i
< exist
; ++i
) {
3286 for (j
= 0; j
< nparam
; ++j
)
3287 if (value_notzero_p(V
->Vertex
->p
[i
][j
]))
3291 value_set_si(M
->p
[0][0], 1);
3292 Vector_Set(M
->p
[0]+1, 0, nvar
+exist
);
3293 Vector_Copy(V
->Vertex
->p
[i
],
3294 M
->p
[0] + 1 + nvar
+ exist
, nparam
+1);
3295 value_oppose(M
->p
[0][1+nvar
+i
],
3296 V
->Vertex
->p
[i
][nparam
+1]);
3298 Polyhedron
*pos
, *neg
;
3299 value_set_si(M
->p
[0][0], 1);
3300 value_decrement(M
->p
[0][P
->Dimension
+1],
3301 M
->p
[0][P
->Dimension
+1]);
3302 neg
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3303 value_set_si(f
, -1);
3304 Vector_Scale(M
->p
[0]+1, M
->p
[0]+1, f
,
3306 value_decrement(M
->p
[0][P
->Dimension
+1],
3307 M
->p
[0][P
->Dimension
+1]);
3308 value_decrement(M
->p
[0][P
->Dimension
+1],
3309 M
->p
[0][P
->Dimension
+1]);
3310 pos
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3311 if (emptyQ(neg
) || emptyQ(pos
)) {
3312 Polyhedron_Free(pos
);
3313 Polyhedron_Free(neg
);
3316 Polyhedron_Free(pos
);
3317 value_increment(M
->p
[0][P
->Dimension
+1],
3318 M
->p
[0][P
->Dimension
+1]);
3319 pos
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3321 fprintf(stderr
, "\nER: Vertex\n");
3322 #endif /* DEBUG_ER */
3324 EP
= enumerate_or(pos
, exist
, nparam
, MaxRays
);
3329 } END_FORALL_PVertex_in_ParamPolyhedron
;
3333 /* Search for vertex coordinate to split on */
3334 /* Now look for one that depends on the parameters */
3335 FORALL_PVertex_in_ParamPolyhedron(V
, last
, PP
) {
3336 for (int i
= 0; i
< exist
; ++i
) {
3337 value_set_si(M
->p
[0][0], 1);
3338 Vector_Set(M
->p
[0]+1, 0, nvar
+exist
);
3339 Vector_Copy(V
->Vertex
->p
[i
],
3340 M
->p
[0] + 1 + nvar
+ exist
, nparam
+1);
3341 value_oppose(M
->p
[0][1+nvar
+i
],
3342 V
->Vertex
->p
[i
][nparam
+1]);
3344 Polyhedron
*pos
, *neg
;
3345 value_set_si(M
->p
[0][0], 1);
3346 value_decrement(M
->p
[0][P
->Dimension
+1],
3347 M
->p
[0][P
->Dimension
+1]);
3348 neg
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3349 value_set_si(f
, -1);
3350 Vector_Scale(M
->p
[0]+1, M
->p
[0]+1, f
,
3352 value_decrement(M
->p
[0][P
->Dimension
+1],
3353 M
->p
[0][P
->Dimension
+1]);
3354 value_decrement(M
->p
[0][P
->Dimension
+1],
3355 M
->p
[0][P
->Dimension
+1]);
3356 pos
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3357 if (emptyQ(neg
) || emptyQ(pos
)) {
3358 Polyhedron_Free(pos
);
3359 Polyhedron_Free(neg
);
3362 Polyhedron_Free(pos
);
3363 value_increment(M
->p
[0][P
->Dimension
+1],
3364 M
->p
[0][P
->Dimension
+1]);
3365 pos
= AddConstraints(M
->p
[0], 1, P
, MaxRays
);
3367 fprintf(stderr
, "\nER: ParamVertex\n");
3368 #endif /* DEBUG_ER */
3370 EP
= enumerate_or(pos
, exist
, nparam
, MaxRays
);
3375 } END_FORALL_PVertex_in_ParamPolyhedron
;
3383 Polyhedron_Free(CEq
);
3387 Param_Polyhedron_Free(PP
);
3394 evalue
*barvinok_enumerate_pip(Polyhedron
*P
,
3395 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
3400 evalue
*barvinok_enumerate_pip(Polyhedron
*P
,
3401 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
3403 int nvar
= P
->Dimension
- exist
- nparam
;
3404 evalue
*EP
= new_zero_ep();
3405 Polyhedron
*Q
, *N
, *T
= 0;
3411 fprintf(stderr
, "\nER: PIP\n");
3412 #endif /* DEBUG_ER */
3414 for (int i
= 0; i
< P
->Dimension
; ++i
) {
3417 bool posray
= false;
3418 bool negray
= false;
3419 value_set_si(min
, 0);
3420 for (int j
= 0; j
< P
->NbRays
; ++j
) {
3421 if (value_pos_p(P
->Ray
[j
][1+i
])) {
3423 if (value_zero_p(P
->Ray
[j
][1+P
->Dimension
]))
3425 } else if (value_neg_p(P
->Ray
[j
][1+i
])) {
3427 if (value_zero_p(P
->Ray
[j
][1+P
->Dimension
]))
3431 P
->Ray
[j
][1+i
], P
->Ray
[j
][1+P
->Dimension
]);
3432 if (value_lt(tmp
, min
))
3433 value_assign(min
, tmp
);
3438 assert(!(posray
&& negray
));
3439 assert(!negray
); // for now
3440 Polyhedron
*O
= T
? T
: P
;
3441 /* shift by a safe amount */
3442 Matrix
*M
= Matrix_Alloc(O
->NbRays
, O
->Dimension
+2);
3443 Vector_Copy(O
->Ray
[0], M
->p
[0], O
->NbRays
* (O
->Dimension
+2));
3444 for (int j
= 0; j
< P
->NbRays
; ++j
) {
3445 if (value_notzero_p(M
->p
[j
][1+P
->Dimension
])) {
3446 value_multiply(tmp
, min
, M
->p
[j
][1+P
->Dimension
]);
3447 value_substract(M
->p
[j
][1+i
], M
->p
[j
][1+i
], tmp
);
3452 T
= Rays2Polyhedron(M
, MaxRays
);
3455 /* negating a parameter requires that we substitute in the
3456 * sign again afterwards.
3459 assert(i
< nvar
+exist
);
3461 T
= Polyhedron_Copy(P
);
3462 for (int j
= 0; j
< T
->NbRays
; ++j
)
3463 value_oppose(T
->Ray
[j
][1+i
], T
->Ray
[j
][1+i
]);
3464 for (int j
= 0; j
< T
->NbConstraints
; ++j
)
3465 value_oppose(T
->Constraint
[j
][1+i
], T
->Constraint
[j
][1+i
]);
3471 Polyhedron
*D
= pip_lexmin(T
? T
: P
, exist
, nparam
);
3472 for (Q
= D
; Q
; Q
= N
) {
3476 exist
= Q
->Dimension
- nvar
- nparam
;
3477 E
= barvinok_enumerate_e(Q
, exist
, nparam
, MaxRays
);
3480 free_evalue_refs(E
);
3492 static bool is_single(Value
*row
, int pos
, int len
)
3494 return First_Non_Zero(row
, pos
) == -1 &&
3495 First_Non_Zero(row
+pos
+1, len
-pos
-1) == -1;
3498 static evalue
* barvinok_enumerate_e_r(Polyhedron
*P
,
3499 unsigned exist
, unsigned nparam
, unsigned MaxRays
);
3502 static int er_level
= 0;
3504 evalue
* barvinok_enumerate_e(Polyhedron
*P
,
3505 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
3507 fprintf(stderr
, "\nER: level %i\n", er_level
);
3508 int nvar
= P
->Dimension
- exist
- nparam
;
3509 fprintf(stderr
, "%d %d %d\n", nvar
, exist
, nparam
);
3511 Polyhedron_Print(stderr
, P_VALUE_FMT
, P
);
3513 P
= DomainConstraintSimplify(Polyhedron_Copy(P
), MaxRays
);
3514 evalue
*EP
= barvinok_enumerate_e_r(P
, exist
, nparam
, MaxRays
);
3520 evalue
* barvinok_enumerate_e(Polyhedron
*P
,
3521 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
3523 P
= DomainConstraintSimplify(Polyhedron_Copy(P
), MaxRays
);
3524 evalue
*EP
= barvinok_enumerate_e_r(P
, exist
, nparam
, MaxRays
);
3530 static evalue
* barvinok_enumerate_e_r(Polyhedron
*P
,
3531 unsigned exist
, unsigned nparam
, unsigned MaxRays
)
3534 Polyhedron
*U
= Universe_Polyhedron(nparam
);
3535 evalue
*EP
= barvinok_enumerate_ev(P
, U
, MaxRays
);
3536 //char *param_name[] = {"P", "Q", "R", "S", "T" };
3537 //print_evalue(stdout, EP, param_name);
3542 int nvar
= P
->Dimension
- exist
- nparam
;
3543 int len
= P
->Dimension
+ 2;
3546 return new_zero_ep();
3548 if (nvar
== 0 && nparam
== 0) {
3549 evalue
*EP
= new_zero_ep();
3550 barvinok_count(P
, &EP
->x
.n
, MaxRays
);
3551 if (value_pos_p(EP
->x
.n
))
3552 value_set_si(EP
->x
.n
, 1);
3557 for (r
= 0; r
< P
->NbRays
; ++r
)
3558 if (value_zero_p(P
->Ray
[r
][0]) ||
3559 value_zero_p(P
->Ray
[r
][P
->Dimension
+1])) {
3561 for (i
= 0; i
< nvar
; ++i
)
3562 if (value_notzero_p(P
->Ray
[r
][i
+1]))
3566 for (i
= nvar
+ exist
; i
< nvar
+ exist
+ nparam
; ++i
)
3567 if (value_notzero_p(P
->Ray
[r
][i
+1]))
3569 if (i
>= nvar
+ exist
+ nparam
)
3572 if (r
< P
->NbRays
) {
3573 evalue
*EP
= new_zero_ep();
3574 value_set_si(EP
->x
.n
, -1);
3579 for (r
= 0; r
< P
->NbEq
; ++r
)
3580 if ((first
= First_Non_Zero(P
->Constraint
[r
]+1+nvar
, exist
)) != -1)
3583 if (First_Non_Zero(P
->Constraint
[r
]+1+nvar
+first
+1,
3584 exist
-first
-1) != -1) {
3585 Polyhedron
*T
= rotate_along(P
, r
, nvar
, exist
, MaxRays
);
3587 fprintf(stderr
, "\nER: Equality\n");
3588 #endif /* DEBUG_ER */
3589 evalue
*EP
= barvinok_enumerate_e(T
, exist
-1, nparam
, MaxRays
);
3594 fprintf(stderr
, "\nER: Fixed\n");
3595 #endif /* DEBUG_ER */
3597 return barvinok_enumerate_e(P
, exist
-1, nparam
, MaxRays
);
3599 Polyhedron
*T
= Polyhedron_Copy(P
);
3600 SwapColumns(T
, nvar
+1, nvar
+1+first
);
3601 evalue
*EP
= barvinok_enumerate_e(T
, exist
-1, nparam
, MaxRays
);
3608 Vector
*row
= Vector_Alloc(len
);
3609 value_set_si(row
->p
[0], 1);
3614 enum constraint
* info
= new constraint
[exist
];
3615 for (int i
= 0; i
< exist
; ++i
) {
3617 for (int l
= P
->NbEq
; l
< P
->NbConstraints
; ++l
) {
3618 if (value_negz_p(P
->Constraint
[l
][nvar
+i
+1]))
3620 bool l_parallel
= is_single(P
->Constraint
[l
]+nvar
+1, i
, exist
);
3621 for (int u
= P
->NbEq
; u
< P
->NbConstraints
; ++u
) {
3622 if (value_posz_p(P
->Constraint
[u
][nvar
+i
+1]))
3624 bool lu_parallel
= l_parallel
||
3625 is_single(P
->Constraint
[u
]+nvar
+1, i
, exist
);
3626 value_oppose(f
, P
->Constraint
[u
][nvar
+i
+1]);
3627 Vector_Combine(P
->Constraint
[l
]+1, P
->Constraint
[u
]+1, row
->p
+1,
3628 f
, P
->Constraint
[l
][nvar
+i
+1], len
-1);
3629 if (!(info
[i
] & INDEPENDENT
)) {
3631 for (j
= 0; j
< exist
; ++j
)
3632 if (j
!= i
&& value_notzero_p(row
->p
[nvar
+j
+1]))
3635 //printf("independent: i: %d, l: %d, u: %d\n", i, l, u);
3636 info
[i
] = (constraint
)(info
[i
] | INDEPENDENT
);
3639 if (info
[i
] & ALL_POS
) {
3640 value_addto(row
->p
[len
-1], row
->p
[len
-1],
3641 P
->Constraint
[l
][nvar
+i
+1]);
3642 value_addto(row
->p
[len
-1], row
->p
[len
-1], f
);
3643 value_multiply(f
, f
, P
->Constraint
[l
][nvar
+i
+1]);
3644 value_substract(row
->p
[len
-1], row
->p
[len
-1], f
);
3645 value_decrement(row
->p
[len
-1], row
->p
[len
-1]);
3646 ConstraintSimplify(row
->p
, row
->p
, len
, &f
);
3647 value_set_si(f
, -1);
3648 Vector_Scale(row
->p
+1, row
->p
+1, f
, len
-1);
3649 value_decrement(row
->p
[len
-1], row
->p
[len
-1]);
3650 Polyhedron
*T
= AddConstraints(row
->p
, 1, P
, MaxRays
);
3652 //printf("not all_pos: i: %d, l: %d, u: %d\n", i, l, u);
3653 info
[i
] = (constraint
)(info
[i
] ^ ALL_POS
);
3655 //puts("pos remainder");
3656 //Polyhedron_Print(stdout, P_VALUE_FMT, T);
3659 if (!(info
[i
] & ONE_NEG
)) {
3661 negative_test_constraint(P
->Constraint
[l
],
3663 row
->p
, nvar
+i
, len
, &f
);
3664 oppose_constraint(row
->p
, len
, &f
);
3665 Polyhedron
*T
= AddConstraints(row
->p
, 1, P
, MaxRays
);
3667 //printf("one_neg i: %d, l: %d, u: %d\n", i, l, u);
3668 info
[i
] = (constraint
)(info
[i
] | ONE_NEG
);
3670 //puts("neg remainder");
3671 //Polyhedron_Print(stdout, P_VALUE_FMT, T);
3673 } else if (!(info
[i
] & ROT_NEG
)) {
3674 if (parallel_constraints(P
->Constraint
[l
],
3676 row
->p
, nvar
, exist
)) {
3677 negative_test_constraint7(P
->Constraint
[l
],
3679 row
->p
, nvar
, exist
,
3681 oppose_constraint(row
->p
, len
, &f
);
3682 Polyhedron
*T
= AddConstraints(row
->p
, 1, P
, MaxRays
);
3684 // printf("rot_neg i: %d, l: %d, u: %d\n", i, l, u);
3685 info
[i
] = (constraint
)(info
[i
] | ROT_NEG
);
3688 //puts("neg remainder");
3689 //Polyhedron_Print(stdout, P_VALUE_FMT, T);
3694 if (!(info
[i
] & ALL_POS
) && (info
[i
] & (ONE_NEG
| ROT_NEG
)))
3698 if (info
[i
] & ALL_POS
)
3705 for (int i = 0; i < exist; ++i)
3706 printf("%i: %i\n", i, info[i]);
3708 for (int i
= 0; i
< exist
; ++i
)
3709 if (info
[i
] & ALL_POS
) {
3711 fprintf(stderr
, "\nER: Positive\n");
3712 #endif /* DEBUG_ER */
3714 // Maybe we should chew off some of the fat here
3715 Matrix
*M
= Matrix_Alloc(P
->Dimension
, P
->Dimension
+1);
3716 for (int j
= 0; j
< P
->Dimension
; ++j
)
3717 value_set_si(M
->p
[j
][j
+ (j
>= i
+nvar
)], 1);
3718 Polyhedron
*T
= Polyhedron_Image(P
, M
, MaxRays
);
3720 evalue
*EP
= barvinok_enumerate_e(T
, exist
-1, nparam
, MaxRays
);
3727 for (int i
= 0; i
< exist
; ++i
)
3728 if (info
[i
] & ONE_NEG
) {
3730 fprintf(stderr
, "\nER: Negative\n");
3731 #endif /* DEBUG_ER */
3736 return barvinok_enumerate_e(P
, exist
-1, nparam
, MaxRays
);
3738 Polyhedron
*T
= Polyhedron_Copy(P
);
3739 SwapColumns(T
, nvar
+1, nvar
+1+i
);
3740 evalue
*EP
= barvinok_enumerate_e(T
, exist
-1, nparam
, MaxRays
);
3745 for (int i
= 0; i
< exist
; ++i
)
3746 if (info
[i
] & ROT_NEG
) {
3748 fprintf(stderr
, "\nER: Rotate\n");
3749 #endif /* DEBUG_ER */
3753 Polyhedron
*T
= rotate_along(P
, r
, nvar
, exist
, MaxRays
);
3754 evalue
*EP
= barvinok_enumerate_e(T
, exist
-1, nparam
, MaxRays
);
3758 for (int i
= 0; i
< exist
; ++i
)
3759 if (info
[i
] & INDEPENDENT
) {
3760 Polyhedron
*pos
, *neg
;
3762 /* Find constraint again and split off negative part */
3764 if (SplitOnVar(P
, i
, nvar
, len
, exist
, MaxRays
,
3765 row
, f
, true, &pos
, &neg
)) {
3767 fprintf(stderr
, "\nER: Split\n");
3768 #endif /* DEBUG_ER */
3771 barvinok_enumerate_e(neg
, exist
-1, nparam
, MaxRays
);
3773 barvinok_enumerate_e(pos
, exist
, nparam
, MaxRays
);
3775 free_evalue_refs(E
);
3777 Polyhedron_Free(neg
);
3778 Polyhedron_Free(pos
);
3792 EP
= enumerate_line(P
, exist
, nparam
, MaxRays
);
3796 EP
= barvinok_enumerate_pip(P
, exist
, nparam
, MaxRays
);
3800 EP
= enumerate_redundant_ray(P
, exist
, nparam
, MaxRays
);
3804 EP
= enumerate_sure(P
, exist
, nparam
, MaxRays
);
3808 EP
= enumerate_ray(P
, exist
, nparam
, MaxRays
);
3812 EP
= enumerate_sure2(P
, exist
, nparam
, MaxRays
);
3816 F
= unfringe(P
, MaxRays
);
3817 if (!PolyhedronIncludes(F
, P
)) {
3819 fprintf(stderr
, "\nER: Fringed\n");
3820 #endif /* DEBUG_ER */
3821 EP
= barvinok_enumerate_e(F
, exist
, nparam
, MaxRays
);
3828 EP
= enumerate_vd(&P
, exist
, nparam
, MaxRays
);
3833 EP
= enumerate_sum(P
, exist
, nparam
, MaxRays
);
3840 Polyhedron
*pos
, *neg
;
3841 for (i
= 0; i
< exist
; ++i
)
3842 if (SplitOnVar(P
, i
, nvar
, len
, exist
, MaxRays
,
3843 row
, f
, false, &pos
, &neg
))
3849 EP
= enumerate_or(pos
, exist
, nparam
, MaxRays
);
3861 gen_fun
* barvinok_series(Polyhedron
*P
, Polyhedron
* C
, unsigned MaxRays
)
3863 Polyhedron
** vcone
;
3865 unsigned nparam
= C
->Dimension
;
3869 sign
.SetLength(ncone
);
3871 CA
= align_context(C
, P
->Dimension
, MaxRays
);
3872 P
= DomainIntersection(P
, CA
, MaxRays
);
3873 Polyhedron_Free(CA
);
3875 assert(!Polyhedron_is_infinite(P
, nparam
));
3876 assert(P
->NbBid
== 0);
3877 assert(Polyhedron_has_positive_rays(P
, nparam
));
3878 assert(P
->NbEq
== 0);
3881 nvar
= dim
- nparam
;
3882 vcone
= new Polyhedron_p
[P
->NbRays
];
3884 for (int j
= 0; j
< P
->NbRays
; ++j
) {
3885 if (!value_pos_p(P
->Ray
[j
][dim
+1]))
3889 Polyhedron
*C
= supporting_cone(P
, j
);
3890 decompose(C
, &vcone
[j
], &npos
, &nneg
, MaxRays
);
3891 ncone
+= npos
+ nneg
;
3892 sign
.SetLength(ncone
);
3893 for (int k
= 0; k
< npos
; ++k
)
3894 sign
[ncone
-nneg
-k
-1] = 1;
3895 for (int k
= 0; k
< nneg
; ++k
)
3896 sign
[ncone
-k
-1] = -1;
3900 rays
.SetDims(ncone
* dim
, nvar
);
3902 for (int j
= 0; j
< P
->NbRays
; ++j
) {
3903 if (!value_pos_p(P
->Ray
[j
][dim
+1]))
3906 for (Polyhedron
*i
= vcone
[j
]; i
; i
= i
->next
) {
3907 add_rays(rays
, i
, &r
, nvar
);
3910 rays
.SetDims(r
, nvar
);
3912 nonorthog(rays
, lambda
);
3913 //randomvector(P, lambda, nvar);
3916 cout << "rays: " << rays;
3917 cout << "lambda: " << lambda;
3923 num_p
.SetLength(nparam
);
3926 den_s
.SetLength(dim
);
3928 den_p
.SetLength(dim
);
3930 den
.SetDims(dim
, nparam
);
3936 gen_fun
* gf
= new gen_fun
;
3938 rays
.SetDims(dim
, nvar
);
3940 for (int j
= 0; j
< P
->NbRays
; ++j
) {
3941 if (!value_pos_p(P
->Ray
[j
][dim
+1]))
3944 for (Polyhedron
*i
= vcone
[j
]; i
; i
= i
->next
, ++f
) {
3945 lattice_point(P
->Ray
[j
]+1, i
, vertex
);
3948 for ( ; k
< nvar
; ++k
)
3949 num_s
+= vertex
[k
] * lambda
[k
];
3950 for ( ; k
< dim
; ++k
)
3951 num_p
[k
-nvar
] = vertex
[k
];
3954 add_rays(rays
, i
, &r
, nvar
, true);
3955 for (r
= 0; r
< dim
; ++r
)
3956 values2zz(i
->Ray
[r
]+1+nvar
, den
[r
], nparam
);
3957 den_s
= rays
* lambda
;
3959 normalize(sign
[f
], num_s
, num_p
, den_s
, den_p
, den
);
3963 for (int k
= 0; k
< dim
; ++k
) {
3966 else if (den_s
[k
] == 0)
3969 if (no_param
== 0) {
3970 for (int k
= 0; k
< dim
; ++k
)
3973 gf
->add(sign
[f
], one
, num_p
, den
);
3974 } else if (no_param
+ only_param
== dim
) {
3977 pden
.SetDims(only_param
, nparam
);
3979 for (k
= 0, l
= 0; k
< dim
; ++k
)
3983 for (k
= 0; k
< dim
; ++k
)
3987 dpoly
n(no_param
, num_s
);
3988 dpoly
d(no_param
, den_s
[k
], 1);
3989 for ( ; ++k
< dim
; k
)
3990 if (den_s
[k
] != 0) {
3991 dpoly
fact(no_param
, den_s
[k
], 1);
3995 mpq_set_si(count
, 0, 1);
3996 n
.div(d
, count
, sign
[f
]);
3999 value2zz(mpq_numref(count
), qn
);
4000 value2zz(mpq_denref(count
), qd
);
4002 gf
->add(qn
, qd
, num_p
, pden
);
4007 pden
.SetDims(only_param
, nparam
);
4009 for (k
= 0, l
= 0; k
< dim
; ++k
)
4013 for (k
= 0; k
< dim
; ++k
)
4017 dpoly
n(no_param
, num_s
);
4018 dpoly
d(no_param
, den_s
[k
], 1);
4019 for ( ; ++k
< dim
; )
4020 if (den_p
[k
] == 0) {
4021 dpoly
fact(no_param
, den_s
[k
], 1);
4025 for (k
= 0; k
< dim
; ++k
) {
4026 if (den_s
[k
] == 0 || den_p
[k
] == 0)
4029 dpoly
pd(no_param
-1, den_s
[k
], 1);
4030 int s
= den_p
[k
] < 0 ? -1 : 1;
4033 r
= new dpoly_r(n
, pd
, k
, s
, dim
);
4035 assert(0); // for now
4038 r
->div(d
, sign
[f
], gf
, pden
, den
, num_p
);
4042 cout << "sign: " << sign[f];
4043 cout << "num_s: " << num_s;
4044 cout << "num_p: " << num_p;
4045 cout << "den_s: " << den_s;
4046 cout << "den_p: " << den_p;
4047 cout << "den: " << den;
4048 cout << "only_param: " << only_param;
4049 cout << "no_param: " << no_param;