Directory: | ./ |
---|---|
File: | src/eiquadprog.cpp |
Date: | 2024-12-04 10:05:36 |
Exec | Total | Coverage | |
---|---|---|---|
Lines: | 134 | 207 | 64.7% |
Branches: | 134 | 324 | 41.4% |
Line | Branch | Exec | Source |
---|---|---|---|
1 | #include <eiquadprog/eiquadprog.hpp> | ||
2 | namespace eiquadprog { | ||
3 | namespace solvers { | ||
4 | |||
5 | using namespace Eigen; | ||
6 | |||
7 | /* solve_quadprog is used for on-demand QP solving */ | ||
8 | |||
9 | 10 | double solve_quadprog(MatrixXd &G, VectorXd &g0, const MatrixXd &CE, | |
10 | const VectorXd &ce0, const MatrixXd &CI, | ||
11 | const VectorXd &ci0, VectorXd &x, VectorXi &activeSet, | ||
12 | size_t &activeSetSize) { | ||
13 | 10 | Eigen::DenseIndex p = CE.cols(); | |
14 | 10 | Eigen::DenseIndex m = CI.cols(); | |
15 | |||
16 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | VectorXd y(p + m); |
17 | |||
18 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | return solve_quadprog(G, g0, CE, ce0, CI, ci0, x, y, activeSet, |
19 | 20 | activeSetSize); | |
20 | 10 | } | |
21 | |||
22 | 10 | double solve_quadprog(MatrixXd &G, VectorXd &g0, const MatrixXd &CE, | |
23 | const VectorXd &ce0, const MatrixXd &CI, | ||
24 | const VectorXd &ci0, VectorXd &x, VectorXd &y, | ||
25 | VectorXi &activeSet, size_t &activeSetSize) { | ||
26 |
1/2✓ Branch 2 taken 10 times.
✗ Branch 3 not taken.
|
10 | LLT<MatrixXd, Lower> chol(G.cols()); |
27 | double c1; | ||
28 | /* compute the trace of the original matrix G */ | ||
29 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | c1 = G.trace(); |
30 | |||
31 | /* decompose the matrix G in the form LL^T */ | ||
32 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | chol.compute(G); |
33 | |||
34 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | return solve_quadprog(chol, c1, g0, CE, ce0, CI, ci0, x, y, activeSet, |
35 | 20 | activeSetSize); | |
36 | 10 | } | |
37 | |||
38 | ✗ | double solve_quadprog(LLT<MatrixXd, Lower> &chol, double c1, VectorXd &g0, | |
39 | const MatrixXd &CE, const VectorXd &ce0, | ||
40 | const MatrixXd &CI, const VectorXd &ci0, VectorXd &x, | ||
41 | VectorXi &activeSet, size_t &activeSetSize) { | ||
42 | ✗ | Eigen::DenseIndex p = CE.cols(); | |
43 | ✗ | Eigen::DenseIndex m = CI.cols(); | |
44 | |||
45 | ✗ | VectorXd y(p + m); | |
46 | |||
47 | ✗ | return solve_quadprog(chol, c1, g0, CE, ce0, CI, ci0, x, y, activeSet, | |
48 | ✗ | activeSetSize); | |
49 | } | ||
50 | |||
51 | /* solve_quadprog2 is used for when the Cholesky decomposition of G is | ||
52 | * pre-computed | ||
53 | * @param A Output vector containing the indexes of the active constraints. | ||
54 | * @param q Output value representing the size of the active set. | ||
55 | */ | ||
56 | 10 | double solve_quadprog(LLT<MatrixXd, Lower> &chol, double c1, VectorXd &g0, | |
57 | const MatrixXd &CE, const VectorXd &ce0, | ||
58 | const MatrixXd &CI, const VectorXd &ci0, VectorXd &x, | ||
59 | VectorXd &u, VectorXi &A, size_t &q) { | ||
60 | size_t i, k, l; /* indices */ | ||
61 | size_t ip, me, mi; | ||
62 | 10 | size_t n = g0.size(); | |
63 | 10 | size_t p = CE.cols(); | |
64 | 10 | size_t m = CI.cols(); | |
65 |
2/4✓ Branch 3 taken 10 times.
✗ Branch 4 not taken.
✓ Branch 8 taken 10 times.
✗ Branch 9 not taken.
|
10 | MatrixXd R(g0.size(), g0.size()), J(g0.size(), g0.size()); |
66 | |||
67 |
5/10✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 10 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 10 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 10 times.
✗ Branch 11 not taken.
✓ Branch 13 taken 10 times.
✗ Branch 14 not taken.
|
10 | VectorXd s(m + p), z(n), r(m + p), d(n), np(n); |
68 |
2/4✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 10 times.
✗ Branch 5 not taken.
|
10 | VectorXd x_old(n), u_old(m + p); |
69 | double f_value, psi, c2, sum, ss, R_norm; | ||
70 | 10 | const double inf = std::numeric_limits<double>::infinity(); | |
71 | double t, t1, t2; /* t is the step length, which is the minimum of the partial | ||
72 | * step length t1 and the full step length t2 */ | ||
73 | // VectorXi A(m + p); // Del Prete: active set is now an output | ||
74 | // parameter | ||
75 |
1/4✗ Branch 1 not taken.
✓ Branch 2 taken 10 times.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
|
10 | if (static_cast<size_t>(A.size()) != m + p) A.resize(m + p); |
76 |
3/6✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 10 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 10 times.
✗ Branch 8 not taken.
|
10 | VectorXi A_old(m + p), iai(m + p), iaexcl(m + p); |
77 | // int q; | ||
78 | 10 | size_t iq, iter = 0; | |
79 | |||
80 | 10 | me = p; /* number of equality constraints */ | |
81 | 10 | mi = m; /* number of inequality constraints */ | |
82 | 10 | q = 0; /* size of the active set A (containing the indices of the active | |
83 | constraints) */ | ||
84 | |||
85 | /* | ||
86 | * Preprocessing phase | ||
87 | */ | ||
88 | |||
89 | /* initialize the matrix R */ | ||
90 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | d.setZero(); |
91 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | R.setZero(); |
92 | 10 | R_norm = 1.0; /* this variable will hold the norm of the matrix R */ | |
93 | |||
94 | /* compute the inverse of the factorized matrix G^-1, this is the initial | ||
95 | * value for H */ | ||
96 | // J = L^-T | ||
97 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | J.setIdentity(); |
98 |
2/4✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 10 times.
✗ Branch 5 not taken.
|
10 | chol.matrixU().solveInPlace(J); |
99 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | c2 = J.trace(); |
100 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
101 | utils::print_matrix("J", J); | ||
102 | #endif | ||
103 | |||
104 | /* c1 * c2 is an estimate for cond(G) */ | ||
105 | |||
106 | /* | ||
107 | * Find the unconstrained minimizer of the quadratic form 0.5 * x G x + g0 x | ||
108 | * this is a feasible point in the dual space | ||
109 | * x = G^-1 * g0 | ||
110 | */ | ||
111 |
2/4✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 10 times.
✗ Branch 5 not taken.
|
10 | x = -g0; |
112 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | chol.solveInPlace(x); |
113 | /* and compute the current solution value */ | ||
114 |
1/2✓ Branch 1 taken 10 times.
✗ Branch 2 not taken.
|
10 | f_value = 0.5 * g0.dot(x); |
115 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
116 | std::cerr << "Unconstrained solution: " << f_value << std::endl; | ||
117 | utils::print_vector("x", x); | ||
118 | #endif | ||
119 | |||
120 | /* Add equality constraints to the working set A */ | ||
121 | 10 | iq = 0; | |
122 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 9 times.
|
14 | for (i = 0; i < me; i++) { |
123 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | np = CE.col(i); |
124 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | compute_d(d, J, np); |
125 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | update_z(z, J, d, iq); |
126 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | update_r(R, r, d, iq); |
127 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
128 | utils::print_matrix("R", R); | ||
129 | utils::print_vector("z", z); | ||
130 | utils::print_vector("r", r); | ||
131 | utils::print_vector("d", d); | ||
132 | #endif | ||
133 | |||
134 | /* compute full step length t2: i.e., the minimum step in primal space s.t. | ||
135 | the contraint becomes feasible */ | ||
136 | 5 | t2 = 0.0; | |
137 |
3/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 4 times.
✓ Branch 5 taken 1 times.
|
10 | if (std::abs(z.dot(z)) > |
138 | 5 | std::numeric_limits<double>::epsilon()) // i.e. z != 0 | |
139 |
3/6✓ Branch 1 taken 4 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 4 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 4 times.
✗ Branch 8 not taken.
|
4 | t2 = (-np.dot(x) - ce0(i)) / z.dot(np); |
140 | |||
141 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | x += t2 * z; |
142 | |||
143 | /* set u = u+ */ | ||
144 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | u(iq) = t2; |
145 |
4/8✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 5 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 5 times.
✗ Branch 11 not taken.
|
5 | u.head(iq) -= t2 * r.head(iq); |
146 | |||
147 | /* compute the new solution value */ | ||
148 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | f_value += 0.5 * (t2 * t2) * z.dot(np); |
149 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | A(i) = static_cast<VectorXi::Scalar>(-i - 1); |
150 | |||
151 |
3/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 1 times.
✓ Branch 4 taken 4 times.
|
5 | if (!add_constraint(R, J, d, iq, R_norm)) { |
152 | // FIXME: it should raise an error | ||
153 | // Equality constraints are linearly dependent | ||
154 | 1 | return f_value; | |
155 | } | ||
156 | } | ||
157 | |||
158 | /* set iai = K \ A */ | ||
159 |
3/4✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 11 times.
✓ Branch 4 taken 9 times.
|
20 | for (i = 0; i < mi; i++) iai(i) = static_cast<VectorXi::Scalar>(i); |
160 | |||
161 | 9 | l1: | |
162 | 14 | iter++; | |
163 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
164 | utils::print_vector("x", x); | ||
165 | #endif | ||
166 | /* step 1: choose a violated constraint */ | ||
167 |
2/2✓ Branch 0 taken 6 times.
✓ Branch 1 taken 14 times.
|
20 | for (i = me; i < iq; i++) { |
168 |
1/2✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
|
6 | ip = A(i); |
169 |
1/2✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
|
6 | iai(ip) = -1; |
170 | } | ||
171 | |||
172 | /* compute s(x) = ci^T * x + ci0 for all elements of K \ A */ | ||
173 | 14 | ss = 0.0; | |
174 | 14 | psi = 0.0; /* this value will contain the sum of all infeasibilities */ | |
175 | 14 | ip = 0; /* ip will be the index of the chosen violated constraint */ | |
176 |
2/2✓ Branch 0 taken 20 times.
✓ Branch 1 taken 14 times.
|
34 | for (i = 0; i < mi; i++) { |
177 |
1/2✓ Branch 1 taken 20 times.
✗ Branch 2 not taken.
|
20 | iaexcl(i) = 1; |
178 |
3/6✓ Branch 1 taken 20 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 20 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 20 times.
✗ Branch 8 not taken.
|
20 | sum = CI.col(i).dot(x) + ci0(i); |
179 |
1/2✓ Branch 1 taken 20 times.
✗ Branch 2 not taken.
|
20 | s(i) = sum; |
180 | 20 | psi += std::min(0.0, sum); | |
181 | } | ||
182 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
183 | utils::print_vector("s", s); | ||
184 | #endif | ||
185 | |||
186 | 14 | if (std::abs(psi) <= static_cast<double>(mi) * | |
187 |
2/2✓ Branch 1 taken 7 times.
✓ Branch 2 taken 7 times.
|
14 | std::numeric_limits<double>::epsilon() * c1 * c2 * |
188 | 100.0) { | ||
189 | /* numerically there are not infeasibilities anymore */ | ||
190 | 7 | q = iq; | |
191 | 7 | return f_value; | |
192 | } | ||
193 | |||
194 | /* save old values for u, x and A */ | ||
195 |
3/6✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 7 times.
✗ Branch 8 not taken.
|
7 | u_old.head(iq) = u.head(iq); |
196 |
3/6✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 7 times.
✗ Branch 8 not taken.
|
7 | A_old.head(iq) = A.head(iq); |
197 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | x_old = x; |
198 | |||
199 | 7 | l2: /* Step 2: check for feasibility and determine a new S-pair */ | |
200 |
2/2✓ Branch 0 taken 13 times.
✓ Branch 1 taken 7 times.
|
20 | for (i = 0; i < mi; i++) { |
201 |
9/14✓ Branch 1 taken 13 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 7 times.
✓ Branch 4 taken 6 times.
✓ Branch 6 taken 7 times.
✗ Branch 7 not taken.
✓ Branch 8 taken 7 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 7 times.
✗ Branch 12 not taken.
✓ Branch 13 taken 7 times.
✗ Branch 14 not taken.
✓ Branch 15 taken 7 times.
✓ Branch 16 taken 6 times.
|
13 | if (s(i) < ss && iai(i) != -1 && iaexcl(i)) { |
202 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | ss = s(i); |
203 | 7 | ip = i; | |
204 | } | ||
205 | } | ||
206 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 7 times.
|
7 | if (ss >= 0.0) { |
207 | ✗ | q = iq; | |
208 | ✗ | return f_value; | |
209 | } | ||
210 | |||
211 | /* set np = n(ip) */ | ||
212 |
2/4✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
|
7 | np = CI.col(ip); |
213 | /* set u = (u 0)^T */ | ||
214 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | u(iq) = 0.0; |
215 | /* add ip to the active set A */ | ||
216 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | A(iq) = static_cast<VectorXi::Scalar>(ip); |
217 | |||
218 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
219 | std::cerr << "Trying with constraint " << ip << std::endl; | ||
220 | utils::print_vector("np", np); | ||
221 | #endif | ||
222 | |||
223 | 7 | l2a: /* Step 2a: determine step direction */ | |
224 | /* compute z = H np: the step direction in the primal space (through J, see | ||
225 | * the paper) */ | ||
226 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | compute_d(d, J, np); |
227 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | update_z(z, J, d, iq); |
228 | /* compute N* np (if q > 0): the negative of the step direction in the dual | ||
229 | * space */ | ||
230 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | update_r(R, r, d, iq); |
231 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
232 | std::cerr << "Step direction z" << std::endl; | ||
233 | utils::print_vector("z", z); | ||
234 | utils::print_vector("r", r); | ||
235 | utils::print_vector("u", u); | ||
236 | utils::print_vector("d", d); | ||
237 | utils::print_vector("A", A); | ||
238 | #endif | ||
239 | |||
240 | /* Step 2b: compute step length */ | ||
241 | 7 | l = 0; | |
242 | /* Compute t1: partial step length (maximum step in dual space without | ||
243 | * violating dual feasibility */ | ||
244 | 7 | t1 = inf; /* +inf */ | |
245 | /* find the index l s.t. it reaches the minimum of u+(x) / r */ | ||
246 |
2/2✓ Branch 0 taken 3 times.
✓ Branch 1 taken 7 times.
|
10 | for (k = me; k < iq; k++) { |
247 | double tmp; | ||
248 |
3/12✓ Branch 1 taken 3 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 3 times.
✗ Branch 6 not taken.
✗ Branch 7 not taken.
✗ Branch 9 not taken.
✗ Branch 10 not taken.
✗ Branch 11 not taken.
✗ Branch 12 not taken.
✗ Branch 13 not taken.
✓ Branch 14 taken 3 times.
|
3 | if (r(k) > 0.0 && ((tmp = u(k) / r(k)) < t1)) { |
249 | ✗ | t1 = tmp; | |
250 | ✗ | l = A(k); | |
251 | } | ||
252 | } | ||
253 | /* Compute t2: full step length (minimum step in primal space such that the | ||
254 | * constraint ip becomes feasible */ | ||
255 |
3/4✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✓ Branch 5 taken 2 times.
|
14 | if (std::abs(z.dot(z)) > |
256 | 7 | std::numeric_limits<double>::epsilon()) // i.e. z != 0 | |
257 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | t2 = -s(ip) / z.dot(np); |
258 | else | ||
259 | 2 | t2 = inf; /* +inf */ | |
260 | |||
261 | /* the step is chosen as the minimum of t1 and t2 */ | ||
262 | 7 | t = std::min(t1, t2); | |
263 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
264 | std::cerr << "Step sizes: " << t << " (t1 = " << t1 << ", t2 = " << t2 | ||
265 | << ") "; | ||
266 | #endif | ||
267 | |||
268 | /* Step 2c: determine new S-pair and take step: */ | ||
269 | |||
270 | /* case (i): no step in primal or dual space */ | ||
271 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 5 times.
|
7 | if (t >= inf) { |
272 | /* QPP is infeasible */ | ||
273 | // FIXME: unbounded to raise | ||
274 | 2 | q = iq; | |
275 | 2 | return inf; | |
276 | } | ||
277 | /* case (ii): step in dual space */ | ||
278 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 5 times.
|
5 | if (t2 >= inf) { |
279 | /* set u = u + t * [-r 1) and drop constraint l from the active set A */ | ||
280 | ✗ | u.head(iq) -= t * r.head(iq); | |
281 | ✗ | u(iq) += t; | |
282 | ✗ | iai(l) = static_cast<VectorXi::Scalar>(l); | |
283 | ✗ | delete_constraint(R, J, A, u, p, iq, l); | |
284 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
285 | std::cerr << " in dual space: " << f_value << std::endl; | ||
286 | utils::print_vector("x", x); | ||
287 | utils::print_vector("z", z); | ||
288 | utils::print_vector("A", A); | ||
289 | #endif | ||
290 | ✗ | goto l2a; | |
291 | } | ||
292 | |||
293 | /* case (iii): step in primal and dual space */ | ||
294 | |||
295 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | x += t * z; |
296 | /* update the solution value */ | ||
297 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | f_value += t * z.dot(np) * (0.5 * t + u(iq)); |
298 | |||
299 |
4/8✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 5 times.
✗ Branch 8 not taken.
✓ Branch 10 taken 5 times.
✗ Branch 11 not taken.
|
5 | u.head(iq) -= t * r.head(iq); |
300 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | u(iq) += t; |
301 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
302 | std::cerr << " in both spaces: " << f_value << std::endl; | ||
303 | utils::print_vector("x", x); | ||
304 | utils::print_vector("u", u); | ||
305 | utils::print_vector("r", r); | ||
306 | utils::print_vector("A", A); | ||
307 | #endif | ||
308 | |||
309 |
1/2✓ Branch 0 taken 5 times.
✗ Branch 1 not taken.
|
5 | if (t == t2) { |
310 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
311 | std::cerr << "Full step has taken " << t << std::endl; | ||
312 | utils::print_vector("x", x); | ||
313 | #endif | ||
314 | /* full step has taken */ | ||
315 | /* add constraint ip to the active set*/ | ||
316 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
✓ Branch 4 taken 5 times.
|
5 | if (!add_constraint(R, J, d, iq, R_norm)) { |
317 | ✗ | iaexcl(ip) = 0; | |
318 | ✗ | delete_constraint(R, J, A, u, p, iq, ip); | |
319 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
320 | utils::print_matrix("R", R); | ||
321 | utils::print_vector("A", A); | ||
322 | #endif | ||
323 | ✗ | for (i = 0; i < m; i++) iai(i) = static_cast<VectorXi::Scalar>(i); | |
324 | ✗ | for (i = 0; i < iq; i++) { | |
325 | ✗ | A(i) = A_old(i); | |
326 | ✗ | iai(A(i)) = -1; | |
327 | ✗ | u(i) = u_old(i); | |
328 | } | ||
329 | ✗ | x = x_old; | |
330 | ✗ | goto l2; /* go to step 2 */ | |
331 | } else | ||
332 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | iai(ip) = -1; |
333 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
334 | utils::print_matrix("R", R); | ||
335 | utils::print_vector("A", A); | ||
336 | #endif | ||
337 | 5 | goto l1; | |
338 | } | ||
339 | |||
340 | /* a patial step has taken */ | ||
341 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
342 | std::cerr << "Partial step has taken " << t << std::endl; | ||
343 | utils::print_vector("x", x); | ||
344 | #endif | ||
345 | /* drop constraint l */ | ||
346 | ✗ | iai(l) = static_cast<VectorXi::Scalar>(l); | |
347 | ✗ | delete_constraint(R, J, A, u, p, iq, l); | |
348 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
349 | utils::print_matrix("R", R); | ||
350 | utils::print_vector("A", A); | ||
351 | #endif | ||
352 | |||
353 | ✗ | s(ip) = CI.col(ip).dot(x) + ci0(ip); | |
354 | |||
355 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
356 | utils::print_vector("s", s); | ||
357 | #endif | ||
358 | ✗ | goto l2a; | |
359 | 10 | } | |
360 | |||
361 | 10 | bool add_constraint(MatrixXd &R, MatrixXd &J, VectorXd &d, size_t &iq, | |
362 | double &R_norm) { | ||
363 | 10 | size_t n = J.rows(); | |
364 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
365 | std::cerr << "Add constraint " << iq << '/'; | ||
366 | #endif | ||
367 | size_t j, k; | ||
368 | double cc, ss, h, t1, t2, xny; | ||
369 | |||
370 | /* we have to find the Givens rotation which will reduce the element | ||
371 | d(j) to zero. | ||
372 | if it is already zero we don't have to do anything, except of | ||
373 | decreasing j */ | ||
374 |
2/2✓ Branch 0 taken 6 times.
✓ Branch 1 taken 10 times.
|
16 | for (j = n - 1; j >= iq + 1; j--) { |
375 | /* The Givens rotation is done with the matrix (cc cs, cs -cc). | ||
376 | If cc is one, then element (j) of d is zero compared with element | ||
377 | (j - 1). Hence we don't have to do anything. | ||
378 | If cc is zero, then we just have to switch column (j) and column (j - 1) | ||
379 | of J. Since we only switch columns in J, we have to be careful how we | ||
380 | update d depending on the sign of gs. | ||
381 | Otherwise we have to apply the Givens rotation to these columns. | ||
382 | The i - 1 element of d has to be updated to h. */ | ||
383 | 6 | cc = d(j - 1); | |
384 | 6 | ss = d(j); | |
385 | 6 | h = utils::distance(cc, ss); | |
386 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | if (h == 0.0) continue; |
387 | 6 | d(j) = 0.0; | |
388 | 6 | ss = ss / h; | |
389 | 6 | cc = cc / h; | |
390 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | if (cc < 0.0) { |
391 | ✗ | cc = -cc; | |
392 | ✗ | ss = -ss; | |
393 | ✗ | d(j - 1) = -h; | |
394 | } else | ||
395 | 6 | d(j - 1) = h; | |
396 | 6 | xny = ss / (1.0 + cc); | |
397 |
2/2✓ Branch 0 taken 12 times.
✓ Branch 1 taken 6 times.
|
18 | for (k = 0; k < n; k++) { |
398 | 12 | t1 = J(k, j - 1); | |
399 | 12 | t2 = J(k, j); | |
400 | 12 | J(k, j - 1) = t1 * cc + t2 * ss; | |
401 | 12 | J(k, j) = xny * (t1 + J(k, j - 1)) - t2; | |
402 | } | ||
403 | } | ||
404 | /* update the number of constraints added*/ | ||
405 | 10 | iq++; | |
406 | /* To update R we have to put the iq components of the d vector | ||
407 | into column iq - 1 of R | ||
408 | */ | ||
409 |
3/6✓ Branch 2 taken 10 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 10 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 10 times.
✗ Branch 9 not taken.
|
10 | R.col(iq - 1).head(iq) = d.head(iq); |
410 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
411 | std::cerr << iq << std::endl; | ||
412 | #endif | ||
413 | |||
414 |
2/2✓ Branch 3 taken 1 times.
✓ Branch 4 taken 9 times.
|
10 | if (std::abs(d(iq - 1)) <= std::numeric_limits<double>::epsilon() * R_norm) |
415 | // problem degenerate | ||
416 | 1 | return false; | |
417 | 9 | R_norm = std::max<double>(R_norm, std::abs(d(iq - 1))); | |
418 | 9 | return true; | |
419 | } | ||
420 | |||
421 | ✗ | void delete_constraint(MatrixXd &R, MatrixXd &J, VectorXi &A, VectorXd &u, | |
422 | size_t p, size_t &iq, size_t l) { | ||
423 | ✗ | size_t n = R.rows(); | |
424 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
425 | std::cerr << "Delete constraint " << l << ' ' << iq; | ||
426 | #endif | ||
427 | ✗ | size_t i, j, k, qq = 0; | |
428 | double cc, ss, h, xny, t1, t2; | ||
429 | |||
430 | /* Find the index qq for active constraint l to be removed */ | ||
431 | ✗ | for (i = p; i < iq; i++) | |
432 | ✗ | if (static_cast<size_t>(A(i)) == l) { | |
433 | ✗ | qq = i; | |
434 | ✗ | break; | |
435 | } | ||
436 | |||
437 | /* remove the constraint from the active set and the duals */ | ||
438 | ✗ | for (i = qq; i < iq - 1; i++) { | |
439 | ✗ | A(i) = A(i + 1); | |
440 | ✗ | u(i) = u(i + 1); | |
441 | ✗ | R.col(i) = R.col(i + 1); | |
442 | } | ||
443 | |||
444 | ✗ | A(iq - 1) = A(iq); | |
445 | ✗ | u(iq - 1) = u(iq); | |
446 | ✗ | A(iq) = 0; | |
447 | ✗ | u(iq) = 0.0; | |
448 | ✗ | for (j = 0; j < iq; j++) R(j, iq - 1) = 0.0; | |
449 | /* constraint has been fully removed */ | ||
450 | ✗ | iq--; | |
451 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
452 | std::cerr << '/' << iq << std::endl; | ||
453 | #endif | ||
454 | |||
455 | ✗ | if (iq == 0) return; | |
456 | |||
457 | ✗ | for (j = qq; j < iq; j++) { | |
458 | ✗ | cc = R(j, j); | |
459 | ✗ | ss = R(j + 1, j); | |
460 | ✗ | h = utils::distance(cc, ss); | |
461 | ✗ | if (h == 0.0) continue; | |
462 | ✗ | cc = cc / h; | |
463 | ✗ | ss = ss / h; | |
464 | ✗ | R(j + 1, j) = 0.0; | |
465 | ✗ | if (cc < 0.0) { | |
466 | ✗ | R(j, j) = -h; | |
467 | ✗ | cc = -cc; | |
468 | ✗ | ss = -ss; | |
469 | } else | ||
470 | ✗ | R(j, j) = h; | |
471 | |||
472 | ✗ | xny = ss / (1.0 + cc); | |
473 | ✗ | for (k = j + 1; k < iq; k++) { | |
474 | ✗ | t1 = R(j, k); | |
475 | ✗ | t2 = R(j + 1, k); | |
476 | ✗ | R(j, k) = t1 * cc + t2 * ss; | |
477 | ✗ | R(j + 1, k) = xny * (t1 + R(j, k)) - t2; | |
478 | } | ||
479 | ✗ | for (k = 0; k < n; k++) { | |
480 | ✗ | t1 = J(k, j); | |
481 | ✗ | t2 = J(k, j + 1); | |
482 | ✗ | J(k, j) = t1 * cc + t2 * ss; | |
483 | ✗ | J(k, j + 1) = xny * (J(k, j) + t1) - t2; | |
484 | } | ||
485 | } | ||
486 | } | ||
487 | |||
488 | } // namespace solvers | ||
489 | } // namespace eiquadprog | ||
490 |