doc: document polytope_sample
[barvinok.git] / basis_reduction.c
blob8ebc38bc7c8eb60241b05db528d3201db3009e4f
1 #include <math.h>
2 #include <glpk.h>
3 #include <polylib/polylibgmp.h>
5 #define ALLOCN(type,n) (type*)malloc((n) * sizeof(type))
7 static LPX *init_lp(Polyhedron *P)
9 LPX *lp;
10 int *ind;
11 double *val;
12 int i, j, k, l;
14 ind = ALLOCN(int, 1+P->Dimension);
15 val = ALLOCN(double, 1+P->Dimension);
16 lp = lpx_create_prob();
17 lpx_set_obj_dir(lp, LPX_MAX);
18 lpx_add_rows(lp, 2*P->NbConstraints);
19 lpx_add_cols(lp, 2*P->Dimension);
20 for (i = 0; i < 2; ++i) {
21 for (j = 0; j < P->NbConstraints; ++j) {
22 for (k = 0, l = 0; k < P->Dimension; ++k) {
23 if (value_zero_p(P->Constraint[j][1+k]))
24 continue;
25 ind[1+l] = 1+i*P->Dimension+k;
26 val[1+l] = VALUE_TO_DOUBLE(P->Constraint[j][1+k]);
27 ++l;
29 lpx_set_mat_row(lp, 1+i*P->NbConstraints+j, l, ind, val);
30 lpx_set_row_bnds(lp, 1+i*P->NbConstraints+j, LPX_LO,
31 -VALUE_TO_DOUBLE(P->Constraint[j][1+P->Dimension]), 0);
33 for (k = 0, l = 0; k < P->Dimension; ++k) {
34 lpx_set_col_bnds(lp, 1+i*P->Dimension+k, LPX_FR, 0, 0);
37 free(ind);
38 free(val);
39 return lp;
42 static void set_lp_obj(LPX *lp, Value *row, int dim)
44 int j;
45 for (j = 0; j < dim; ++j) {
46 lpx_set_obj_coef(lp, 1+j, VALUE_TO_DOUBLE(row[j]));
47 lpx_set_obj_coef(lp, 1+dim+j, -VALUE_TO_DOUBLE(row[j]));
51 static int add_lp_row(LPX *lp, Value *row, int dim)
53 int j, l;
54 int nr = lpx_add_rows(lp, 1);
55 int *ind;
56 double *val;
58 ind = ALLOCN(int, 1+2*dim);
59 val = ALLOCN(double, 1+2*dim);
60 for (j = 0, l = 0; j < dim; ++j) {
61 if (value_zero_p(row[j]))
62 continue;
63 ind[1+l] = 1+j;
64 val[1+l] = VALUE_TO_DOUBLE(row[j]);
65 ind[1+l+1] = 1+dim+j;
66 val[1+l+1] = -VALUE_TO_DOUBLE(row[j]);
67 l += 2;
69 lpx_set_mat_row(lp, nr, l, ind, val);
70 lpx_set_row_bnds(lp, nr, LPX_FX, 0, 0);
71 free(ind);
72 free(val);
74 return nr;
77 static void del_lp_row(LPX *lp)
79 int rows[2];
80 rows[1] = lpx_get_num_rows(lp);
81 lpx_del_rows(lp, 1, rows);
84 static void save_alpha(LPX *lp, int first, int n, double *alpha)
86 int i;
88 for (i = 0; i < n; ++i)
89 alpha[i] = -lpx_get_row_dual(lp, first+i);
92 /* This function implements the algorithm described in
93 * "An Implementation of the Generalized Basis Reduction Algorithm
94 * for Integer Programming" of Cook el al. to compute a reduced basis.
95 * We use \epsilon = 1/4.
97 Matrix *Polyhedron_Reduced_Basis(Polyhedron *P)
99 int dim = P->Dimension;
100 int i;
101 Matrix *basis = Identity(dim);
102 LPX *lp;
103 double F_old, alpha, F_new;
104 int row;
105 Value one, tmp;
106 Vector *b_tmp;
107 double *F;
108 double *alpha_buffer[2];
109 double *alpha_saved;
110 double F_saved;
111 int use_saved = 0;
113 if (P->Dimension == 1)
114 return basis;
116 value_init(one);
117 value_init(tmp);
118 value_set_si(one, 1);
120 b_tmp = Vector_Alloc(dim);
122 F = ALLOCN(double, dim);
123 alpha_buffer[0] = ALLOCN(double, dim);
124 alpha_buffer[1] = ALLOCN(double, dim);
125 alpha_saved = alpha_buffer[0];
127 lp = init_lp(P);
129 lpx_set_int_parm(lp, LPX_K_MSGLEV, 0);
131 i = 0;
133 set_lp_obj(lp, basis->p[0], dim);
135 lpx_adv_basis(lp);
136 lpx_simplex(lp);
137 F[0] = lpx_get_obj_val(lp);
138 assert(F[0] > -1e-10);
139 if (F[0] < 0)
140 F[0] = 0;
142 do {
143 int mu[2];
145 if (use_saved) {
146 row = lpx_get_num_rows(lp)+1;
147 F_new = F_saved;
148 alpha = alpha_saved[i];
149 } else {
150 row = add_lp_row(lp, basis->p[i], dim);
151 set_lp_obj(lp, basis->p[i+1], dim);
152 lpx_adv_basis(lp);
153 lpx_simplex(lp);
154 F_new = lpx_get_obj_val(lp);
156 alpha = -lpx_get_row_dual(lp, row);
158 if (i > 0)
159 save_alpha(lp, row-i, i, alpha_saved);
161 del_lp_row(lp);
163 F[i+1] = F_new;
164 assert(F[i+1] > -1e-10);
165 if (F[i+1] < 0)
166 F[i+1] = 0;
169 mu[0] = (int)floor(alpha+1e-10);
170 mu[1] = (int)ceil(alpha-1e-10);
172 if (mu[0] == mu[1])
173 value_set_si(tmp, mu[0]);
174 else {
175 double mu_F[2];
176 int j;
178 for (j = 0; j <= 1; ++j) {
179 value_set_si(tmp, mu[j]);
180 Vector_Combine(basis->p[i+1], basis->p[i], b_tmp->p, one, tmp, dim);
181 set_lp_obj(lp, b_tmp->p, dim);
182 lpx_adv_basis(lp);
183 lpx_simplex(lp);
184 mu_F[j] = lpx_get_obj_val(lp);
185 if (i > 0)
186 save_alpha(lp, row-i, i, alpha_buffer[j]);
189 if (mu_F[0] < mu_F[1])
190 j = 0;
191 else
192 j = 1;
194 value_set_si(tmp, mu[j]);
195 F_new = mu_F[j];
196 alpha_saved = alpha_buffer[j];
198 Vector_Combine(basis->p[i+1], basis->p[i], basis->p[i+1], one, tmp, dim);
200 assert(F_new > -1e-10);
201 if (F_new < 0)
202 F_new = 0;
203 F_old = F[i];
205 use_saved = 0;
206 if (F_new < 3*F_old/4) {
207 Vector_Exchange(basis->p[i], basis->p[i+1], dim);
208 if (i > 0) {
209 use_saved = 1;
210 F_saved = F_new;
211 del_lp_row(lp);
212 --i;
213 } else
214 F[0] = F_new;
215 } else {
216 add_lp_row(lp, basis->p[i], dim);
217 ++i;
219 } while (i < dim-1);
221 Vector_Free(b_tmp);
223 value_clear(one);
224 value_clear(tmp);
225 free(F);
226 free(alpha_buffer[0]);
227 free(alpha_buffer[1]);
229 lpx_delete_prob(lp);
231 return basis;