Directory: | ./ |
---|---|
File: | src/eiquadprog-fast.cpp |
Date: | 2024-12-04 10:05:36 |
Exec | Total | Coverage | |
---|---|---|---|
Lines: | 151 | 222 | 68.0% |
Branches: | 151 | 362 | 41.7% |
Line | Branch | Exec | Source |
---|---|---|---|
1 | #include "eiquadprog/eiquadprog-fast.hpp" | ||
2 | |||
3 | #include <iostream> | ||
4 | |||
5 | namespace eiquadprog { | ||
6 | namespace solvers { | ||
7 | |||
8 |
14/28✓ Branch 2 taken 12 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 12 times.
✗ Branch 6 not taken.
✓ Branch 8 taken 12 times.
✗ Branch 9 not taken.
✓ Branch 11 taken 12 times.
✗ Branch 12 not taken.
✓ Branch 14 taken 12 times.
✗ Branch 15 not taken.
✓ Branch 17 taken 12 times.
✗ Branch 18 not taken.
✓ Branch 20 taken 12 times.
✗ Branch 21 not taken.
✓ Branch 23 taken 12 times.
✗ Branch 24 not taken.
✓ Branch 26 taken 12 times.
✗ Branch 27 not taken.
✓ Branch 29 taken 12 times.
✗ Branch 30 not taken.
✓ Branch 32 taken 12 times.
✗ Branch 33 not taken.
✓ Branch 35 taken 12 times.
✗ Branch 36 not taken.
✓ Branch 38 taken 12 times.
✗ Branch 39 not taken.
✓ Branch 41 taken 12 times.
✗ Branch 42 not taken.
|
12 | EiquadprogFast::EiquadprogFast() { |
9 | 12 | m_maxIter = DEFAULT_MAX_ITER; | |
10 | 12 | q = 0; // size of the active set A (containing the indices | |
11 | // of the active constraints) | ||
12 | 12 | is_inverse_provided_ = false; | |
13 | 12 | m_nVars = 0; | |
14 | 12 | m_nEqCon = 0; | |
15 | 12 | m_nIneqCon = 0; | |
16 | 12 | } | |
17 | |||
18 | 24 | EiquadprogFast::~EiquadprogFast() {} | |
19 | |||
20 | 12 | void EiquadprogFast::reset(size_t nVars, size_t nEqCon, size_t nIneqCon) { | |
21 | 12 | m_nVars = nVars; | |
22 | 12 | m_nEqCon = nEqCon; | |
23 | 12 | m_nIneqCon = nIneqCon; | |
24 | 12 | m_J.setZero(nVars, nVars); | |
25 | 12 | chol_.compute(m_J); | |
26 | 12 | R.resize(nVars, nVars); | |
27 | 12 | s.resize(nIneqCon); | |
28 | 12 | r.resize(nIneqCon + nEqCon); | |
29 | 12 | u.resize(nIneqCon + nEqCon); | |
30 | 12 | z.resize(nVars); | |
31 | 12 | d.resize(nVars); | |
32 | 12 | np.resize(nVars); | |
33 | 12 | A.resize(nIneqCon + nEqCon); | |
34 | 12 | iai.resize(nIneqCon); | |
35 | 12 | iaexcl.resize(nIneqCon); | |
36 | 12 | x_old.resize(nVars); | |
37 | 12 | u_old.resize(nIneqCon + nEqCon); | |
38 | 12 | A_old.resize(nIneqCon + nEqCon); | |
39 | |||
40 | #ifdef OPTIMIZE_ADD_CONSTRAINT | ||
41 | T1.resize(nVars); | ||
42 | #endif | ||
43 | 12 | } | |
44 | |||
45 | 10 | bool EiquadprogFast::add_constraint(MatrixXd &R, MatrixXd &J, VectorXd &d, | |
46 | size_t &iq, double &R_norm) { | ||
47 | 10 | size_t nVars = J.rows(); | |
48 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
49 | std::cerr << "Add constraint " << iq << '/'; | ||
50 | #endif | ||
51 | size_t j, k; | ||
52 | double cc, ss, h, t1, t2, xny; | ||
53 | |||
54 | #ifdef OPTIMIZE_ADD_CONSTRAINT | ||
55 | Eigen::Vector2d cc_ss; | ||
56 | #endif | ||
57 | |||
58 | /* we have to find the Givens rotation which will reduce the element | ||
59 | d(j) to zero. | ||
60 | if it is already zero we don't have to do anything, except of | ||
61 | decreasing j */ | ||
62 |
2/2✓ Branch 1 taken 6 times.
✓ Branch 2 taken 10 times.
|
16 | for (j = d.size() - 1; j >= iq + 1; j--) { |
63 | /* The Givens rotation is done with the matrix (cc cs, cs -cc). | ||
64 | If cc is one, then element (j) of d is zero compared with | ||
65 | element (j - 1). Hence we don't have to do anything. If cc is zero, then | ||
66 | we just have to switch column (j) and column (j - 1) of J. Since we only | ||
67 | switch columns in J, we have to be careful how we update d depending on | ||
68 | the sign of gs. Otherwise we have to apply the Givens rotation to these | ||
69 | columns. The i - 1 element of d has to be updated to h. */ | ||
70 | 6 | cc = d(j - 1); | |
71 | 6 | ss = d(j); | |
72 | 6 | h = utils::distance(cc, ss); | |
73 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | if (h == 0.0) continue; |
74 | 6 | d(j) = 0.0; | |
75 | 6 | ss = ss / h; | |
76 | 6 | cc = cc / h; | |
77 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | if (cc < 0.0) { |
78 | ✗ | cc = -cc; | |
79 | ✗ | ss = -ss; | |
80 | ✗ | d(j - 1) = -h; | |
81 | } else | ||
82 | 6 | d(j - 1) = h; | |
83 | 6 | xny = ss / (1.0 + cc); | |
84 | |||
85 | // #define OPTIMIZE_ADD_CONSTRAINT | ||
86 | #ifdef OPTIMIZE_ADD_CONSTRAINT // the optimized code is actually slower than | ||
87 | // the original | ||
88 | T1 = J.col(j - 1); | ||
89 | cc_ss(0) = cc; | ||
90 | cc_ss(1) = ss; | ||
91 | J.col(j - 1).noalias() = J.middleCols<2>(j - 1) * cc_ss; | ||
92 | J.col(j) = xny * (T1 + J.col(j - 1)) - J.col(j); | ||
93 | #else | ||
94 | // J.col(j-1) = J[:,j-1:j] * [cc; ss] | ||
95 |
2/2✓ Branch 0 taken 12 times.
✓ Branch 1 taken 6 times.
|
18 | for (k = 0; k < nVars; k++) { |
96 | 12 | t1 = J(k, j - 1); | |
97 | 12 | t2 = J(k, j); | |
98 | 12 | J(k, j - 1) = t1 * cc + t2 * ss; | |
99 | 12 | J(k, j) = xny * (t1 + J(k, j - 1)) - t2; | |
100 | } | ||
101 | #endif | ||
102 | } | ||
103 | /* update the number of constraints added*/ | ||
104 | 10 | iq++; | |
105 | /* To update R we have to put the iq components of the d vector | ||
106 | into column iq - 1 of R | ||
107 | */ | ||
108 |
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); |
109 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
110 | std::cerr << iq << std::endl; | ||
111 | #endif | ||
112 | |||
113 |
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) |
114 | // problem degenerate | ||
115 | 1 | return false; | |
116 | 9 | R_norm = std::max<double>(R_norm, std::abs(d(iq - 1))); | |
117 | 9 | return true; | |
118 | } | ||
119 | |||
120 | ✗ | void EiquadprogFast::delete_constraint(MatrixXd &R, MatrixXd &J, VectorXi &A, | |
121 | VectorXd &u, size_t nEqCon, size_t &iq, | ||
122 | size_t l) { | ||
123 | ✗ | size_t nVars = R.rows(); | |
124 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
125 | std::cerr << "Delete constraint " << l << ' ' << iq; | ||
126 | #endif | ||
127 | size_t i, j, k; | ||
128 | ✗ | size_t qq = 0; | |
129 | double cc, ss, h, xny, t1, t2; | ||
130 | |||
131 | /* Find the index qq for active constraint l to be removed */ | ||
132 | ✗ | for (i = nEqCon; i < iq; i++) | |
133 | ✗ | if (A(i) == static_cast<VectorXi::Scalar>(l)) { | |
134 | ✗ | qq = i; | |
135 | ✗ | break; | |
136 | } | ||
137 | |||
138 | /* remove the constraint from the active set and the duals */ | ||
139 | ✗ | for (i = qq; i < iq - 1; i++) { | |
140 | ✗ | A(i) = A(i + 1); | |
141 | ✗ | u(i) = u(i + 1); | |
142 | ✗ | R.col(i) = R.col(i + 1); | |
143 | } | ||
144 | |||
145 | ✗ | A(iq - 1) = A(iq); | |
146 | ✗ | u(iq - 1) = u(iq); | |
147 | ✗ | A(iq) = 0; | |
148 | ✗ | u(iq) = 0.0; | |
149 | ✗ | for (j = 0; j < iq; j++) R(j, iq - 1) = 0.0; | |
150 | /* constraint has been fully removed */ | ||
151 | ✗ | iq--; | |
152 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
153 | std::cerr << '/' << iq << std::endl; | ||
154 | #endif | ||
155 | |||
156 | ✗ | if (iq == 0) return; | |
157 | |||
158 | ✗ | for (j = qq; j < iq; j++) { | |
159 | ✗ | cc = R(j, j); | |
160 | ✗ | ss = R(j + 1, j); | |
161 | ✗ | h = utils::distance(cc, ss); | |
162 | ✗ | if (h == 0.0) continue; | |
163 | ✗ | cc = cc / h; | |
164 | ✗ | ss = ss / h; | |
165 | ✗ | R(j + 1, j) = 0.0; | |
166 | ✗ | if (cc < 0.0) { | |
167 | ✗ | R(j, j) = -h; | |
168 | ✗ | cc = -cc; | |
169 | ✗ | ss = -ss; | |
170 | } else | ||
171 | ✗ | R(j, j) = h; | |
172 | |||
173 | ✗ | xny = ss / (1.0 + cc); | |
174 | ✗ | for (k = j + 1; k < iq; k++) { | |
175 | ✗ | t1 = R(j, k); | |
176 | ✗ | t2 = R(j + 1, k); | |
177 | ✗ | R(j, k) = t1 * cc + t2 * ss; | |
178 | ✗ | R(j + 1, k) = xny * (t1 + R(j, k)) - t2; | |
179 | } | ||
180 | ✗ | for (k = 0; k < nVars; k++) { | |
181 | ✗ | t1 = J(k, j); | |
182 | ✗ | t2 = J(k, j + 1); | |
183 | ✗ | J(k, j) = t1 * cc + t2 * ss; | |
184 | ✗ | J(k, j + 1) = xny * (J(k, j) + t1) - t2; | |
185 | } | ||
186 | } | ||
187 | } | ||
188 | |||
189 | 12 | EiquadprogFast_status EiquadprogFast::solve_quadprog( | |
190 | const MatrixXd &Hess, const VectorXd &g0, const MatrixXd &CE, | ||
191 | const VectorXd &ce0, const MatrixXd &CI, const VectorXd &ci0, VectorXd &x) { | ||
192 | 12 | const size_t nVars = g0.size(); | |
193 | 12 | const size_t nEqCon = ce0.size(); | |
194 | 12 | const size_t nIneqCon = ci0.size(); | |
195 | |||
196 |
3/6✓ Branch 0 taken 12 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 12 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 12 times.
|
12 | if (nVars != m_nVars || nEqCon != m_nEqCon || nIneqCon != m_nIneqCon) |
197 | ✗ | reset(nVars, nEqCon, nIneqCon); | |
198 | |||
199 |
2/4✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 12 times.
✗ Branch 5 not taken.
|
12 | assert(static_cast<size_t>(Hess.rows()) == m_nVars && |
200 | static_cast<size_t>(Hess.cols()) == m_nVars); | ||
201 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 12 times.
|
12 | assert(static_cast<size_t>(g0.size()) == m_nVars); |
202 |
2/4✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 12 times.
✗ Branch 5 not taken.
|
12 | assert(static_cast<size_t>(CE.rows()) == m_nEqCon && |
203 | static_cast<size_t>(CE.cols()) == m_nVars); | ||
204 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 12 times.
|
12 | assert(static_cast<size_t>(ce0.size()) == m_nEqCon); |
205 |
2/4✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 12 times.
✗ Branch 5 not taken.
|
12 | assert(static_cast<size_t>(CI.rows()) == m_nIneqCon && |
206 | static_cast<size_t>(CI.cols()) == m_nVars); | ||
207 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 12 times.
|
12 | assert(static_cast<size_t>(ci0.size()) == m_nIneqCon); |
208 | |||
209 | size_t i, k, l; // indices | ||
210 | size_t ip; // index of the chosen violated constraint | ||
211 | size_t iq; // current number of active constraints | ||
212 | double psi; // current sum of constraint violations | ||
213 | double c1; // Hessian trace | ||
214 | double c2; // Hessian Cholesky factor trace | ||
215 | double ss; // largest constraint violation (negative for violation) | ||
216 | double R_norm; // norm of matrix R | ||
217 | 12 | const double inf = std::numeric_limits<double>::infinity(); | |
218 | double t, t1, t2; | ||
219 | /* t is the step length, which is the minimum of the partial step length t1 | ||
220 | * and the full step length t2 */ | ||
221 | |||
222 | 12 | iter = 0; // active-set iteration number | |
223 | |||
224 | /* | ||
225 | * Preprocessing phase | ||
226 | */ | ||
227 | /* compute the trace of the original matrix Hess */ | ||
228 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | c1 = Hess.trace(); |
229 | |||
230 | /* decompose the matrix Hess in the form LL^T */ | ||
231 |
1/2✓ Branch 0 taken 12 times.
✗ Branch 1 not taken.
|
12 | if (!is_inverse_provided_) { |
232 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_CHOLESKY_DECOMPOSITION); | ||
233 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | chol_.compute(Hess); |
234 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_CHOLESKY_DECOMPOSITION); | ||
235 | } | ||
236 | |||
237 | /* initialize the matrix R */ | ||
238 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | d.setZero(nVars); |
239 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | R.setZero(nVars, nVars); |
240 | 12 | R_norm = 1.0; | |
241 | |||
242 | /* compute the inverse of the factorized matrix Hess^-1, | ||
243 | this is the initial value for H */ | ||
244 | // m_J = L^-T | ||
245 |
1/2✓ Branch 0 taken 12 times.
✗ Branch 1 not taken.
|
12 | if (!is_inverse_provided_) { |
246 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_CHOLESKY_INVERSE); | ||
247 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | m_J.setIdentity(nVars, nVars); |
248 | #ifdef OPTIMIZE_HESSIAN_INVERSE | ||
249 |
2/4✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 12 times.
✗ Branch 5 not taken.
|
12 | chol_.matrixU().solveInPlace(m_J); |
250 | #else | ||
251 | m_J = chol_.matrixU().solve(m_J); | ||
252 | #endif | ||
253 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_CHOLESKY_INVERSE); | ||
254 | } | ||
255 | |||
256 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | c2 = m_J.trace(); |
257 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
258 | utils::print_matrix("m_J", m_J); | ||
259 | #endif | ||
260 | |||
261 | /* c1 * c2 is an estimate for cond(Hess) */ | ||
262 | |||
263 | /* | ||
264 | * Find the unconstrained minimizer of the quadratic | ||
265 | * form 0.5 * x Hess x + g0 x | ||
266 | * this is a feasible point in the dual space | ||
267 | * x = Hess^-1 * g0 | ||
268 | */ | ||
269 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_1_UNCONSTR_MINIM); | ||
270 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 12 times.
|
12 | if (is_inverse_provided_) { |
271 | ✗ | x = m_J * (m_J.transpose() * g0); | |
272 | } else { | ||
273 | #ifdef OPTIMIZE_UNCONSTR_MINIM | ||
274 |
2/4✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 12 times.
✗ Branch 5 not taken.
|
12 | x = -g0; |
275 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | chol_.solveInPlace(x); |
276 | } | ||
277 | #else | ||
278 | x = chol_.solve(g0); | ||
279 | } | ||
280 | x = -x; | ||
281 | #endif | ||
282 | /* and compute the current solution value */ | ||
283 |
1/2✓ Branch 1 taken 12 times.
✗ Branch 2 not taken.
|
12 | f_value = 0.5 * g0.dot(x); |
284 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
285 | std::cerr << "Unconstrained solution: " << f_value << std::endl; | ||
286 | utils::print_vector("x", x); | ||
287 | #endif | ||
288 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_1_UNCONSTR_MINIM); | ||
289 | |||
290 | /* Add equality constraints to the working set A */ | ||
291 | |||
292 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_ADD_EQ_CONSTR); | ||
293 | 12 | iq = 0; | |
294 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 11 times.
|
16 | for (i = 0; i < nEqCon; i++) { |
295 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_ADD_EQ_CONSTR_1); | ||
296 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | np = CE.row(i); |
297 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | compute_d(d, m_J, np); |
298 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | update_z(z, m_J, d, iq); |
299 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | update_r(R, r, d, iq); |
300 | |||
301 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
302 | utils::print_matrix("R", R); | ||
303 | utils::print_vector("z", z); | ||
304 | utils::print_vector("r", r); | ||
305 | utils::print_vector("d", d); | ||
306 | #endif | ||
307 | |||
308 | /* compute full step length t2: i.e., | ||
309 | the minimum step in primal space s.t. the contraint | ||
310 | becomes feasible */ | ||
311 | 5 | t2 = 0.0; | |
312 |
3/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 4 times.
✓ Branch 6 taken 1 times.
|
5 | if (std::abs(z.dot(z)) > std::numeric_limits<double>::epsilon()) |
313 | // i.e. z != 0 | ||
314 |
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); |
315 | |||
316 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | x += t2 * z; |
317 | |||
318 | /* set u = u+ */ | ||
319 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | u(iq) = t2; |
320 |
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); |
321 | |||
322 | /* compute the new solution value */ | ||
323 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | f_value += 0.5 * (t2 * t2) * z.dot(np); |
324 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | A(i) = static_cast<VectorXi::Scalar>(-i - 1); |
325 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_ADD_EQ_CONSTR_1); | ||
326 | |||
327 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_ADD_EQ_CONSTR_2); | ||
328 |
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, m_J, d, iq, R_norm)) { |
329 | // Equality constraints are linearly dependent | ||
330 | 1 | return EIQUADPROG_FAST_REDUNDANT_EQUALITIES; | |
331 | } | ||
332 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_ADD_EQ_CONSTR_2); | ||
333 | } | ||
334 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_ADD_EQ_CONSTR); | ||
335 | |||
336 | /* set iai = K \ A */ | ||
337 |
3/4✓ Branch 1 taken 11 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 11 times.
✓ Branch 4 taken 11 times.
|
22 | for (i = 0; i < nIneqCon; i++) iai(i) = static_cast<VectorXi::Scalar>(i); |
338 | |||
339 | #ifdef USE_WARM_START | ||
340 | // DEBUG_STREAM("Gonna warm start using previous active | ||
341 | // set:\n"<<A.transpose()<<"\n") | ||
342 | for (i = nEqCon; i < q; i++) { | ||
343 | iai(i - nEqCon) = -1; | ||
344 | ip = A(i); | ||
345 | np = CI.row(ip); | ||
346 | compute_d(d, m_J, np); | ||
347 | update_z(z, m_J, d, iq); | ||
348 | update_r(R, r, d, iq); | ||
349 | |||
350 | /* compute full step length t2: i.e., | ||
351 | the minimum step in primal space s.t. the contraint | ||
352 | becomes feasible */ | ||
353 | t2 = 0.0; | ||
354 | if (std::abs(z.dot(z)) > | ||
355 | std::numeric_limits<double>::epsilon()) // i.e. z != 0 | ||
356 | t2 = (-np.dot(x) - ci0(ip)) / z.dot(np); | ||
357 | else | ||
358 | DEBUG_STREAM("[WARM START] z=0\n") | ||
359 | |||
360 | x += t2 * z; | ||
361 | |||
362 | /* set u = u+ */ | ||
363 | u(iq) = t2; | ||
364 | u.head(iq) -= t2 * r.head(iq); | ||
365 | |||
366 | /* compute the new solution value */ | ||
367 | f_value += 0.5 * (t2 * t2) * z.dot(np); | ||
368 | |||
369 | if (!add_constraint(R, m_J, d, iq, R_norm)) { | ||
370 | // constraints are linearly dependent | ||
371 | std::cerr << "[WARM START] Constraints are linearly dependent\n"; | ||
372 | return RT_EIQUADPROG_REDUNDANT_EQUALITIES; | ||
373 | } | ||
374 | } | ||
375 | #else | ||
376 | |||
377 | #endif | ||
378 | |||
379 | 11 | l1: | |
380 | 16 | iter++; | |
381 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 16 times.
|
16 | if (iter >= m_maxIter) { |
382 | ✗ | q = iq; | |
383 | ✗ | return EIQUADPROG_FAST_MAX_ITER_REACHED; | |
384 | } | ||
385 | |||
386 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_1); | ||
387 | |||
388 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
389 | utils::print_vector("x", x); | ||
390 | #endif | ||
391 | /* step 1: choose a violated constraint */ | ||
392 |
2/2✓ Branch 0 taken 6 times.
✓ Branch 1 taken 16 times.
|
22 | for (i = nEqCon; i < iq; i++) { |
393 |
1/2✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
|
6 | ip = A(i); |
394 |
1/2✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
|
6 | iai(ip) = -1; |
395 | } | ||
396 | |||
397 | /* compute s(x) = ci^T * x + ci0 for all elements of K \ A */ | ||
398 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_1_2); | ||
399 | 16 | ss = 0.0; | |
400 | 16 | ip = 0; /* ip will be the index of the chosen violated constraint */ | |
401 | |||
402 | #ifdef OPTIMIZE_STEP_1_2 | ||
403 |
1/2✓ Branch 1 taken 16 times.
✗ Branch 2 not taken.
|
16 | s = ci0; |
404 |
3/6✓ Branch 1 taken 16 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 16 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 16 times.
✗ Branch 8 not taken.
|
16 | s.noalias() += CI * x; |
405 |
1/2✓ Branch 1 taken 16 times.
✗ Branch 2 not taken.
|
16 | iaexcl.setOnes(); |
406 |
3/6✓ Branch 1 taken 16 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 16 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 16 times.
✗ Branch 8 not taken.
|
16 | psi = (s.cwiseMin(VectorXd::Zero(nIneqCon))).sum(); |
407 | #else | ||
408 | psi = 0.0; /* this value will contain the sum of all infeasibilities */ | ||
409 | for (i = 0; i < nIneqCon; i++) { | ||
410 | iaexcl(i) = 1; | ||
411 | s(i) = CI.row(i).dot(x) + ci0(i); | ||
412 | psi += std::min(0.0, s(i)); | ||
413 | } | ||
414 | #endif | ||
415 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_1_2); | ||
416 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
417 | utils::print_vector("s", s); | ||
418 | #endif | ||
419 | |||
420 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_1); | ||
421 | |||
422 | 16 | if (std::abs(psi) <= static_cast<double>(nIneqCon) * | |
423 |
2/2✓ Branch 1 taken 9 times.
✓ Branch 2 taken 7 times.
|
16 | std::numeric_limits<double>::epsilon() * c1 * c2 * |
424 | 100.0) { | ||
425 | /* numerically there are not infeasibilities anymore */ | ||
426 | 9 | q = iq; | |
427 | // DEBUG_STREAM("Optimal active | ||
428 | // set:\n"<<A.head(iq).transpose()<<"\n\n") | ||
429 | 9 | return EIQUADPROG_FAST_OPTIMAL; | |
430 | } | ||
431 | |||
432 | /* save old values for u, x and A */ | ||
433 |
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); |
434 |
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); |
435 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | x_old = x; |
436 | |||
437 | 7 | l2: /* Step 2: check for feasibility and determine a new S-pair */ | |
438 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2); | ||
439 | // find constraint with highest violation | ||
440 | // (what about normalizing constraints?) | ||
441 |
2/2✓ Branch 0 taken 13 times.
✓ Branch 1 taken 7 times.
|
20 | for (i = 0; i < nIneqCon; i++) { |
442 |
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)) { |
443 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | ss = s(i); |
444 | 7 | ip = i; | |
445 | } | ||
446 | } | ||
447 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 7 times.
|
7 | if (ss >= 0.0) { |
448 | ✗ | q = iq; | |
449 | // DEBUG_STREAM("Optimal active set:\n"<<A.transpose()<<"\n\n") | ||
450 | ✗ | return EIQUADPROG_FAST_OPTIMAL; | |
451 | } | ||
452 | |||
453 | /* set np = n(ip) */ | ||
454 |
2/4✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 7 times.
✗ Branch 5 not taken.
|
7 | np = CI.row(ip); |
455 | /* set u = (u 0)^T */ | ||
456 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | u(iq) = 0.0; |
457 | /* add ip to the active set A */ | ||
458 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | A(iq) = static_cast<VectorXi::Scalar>(ip); |
459 | |||
460 | // DEBUG_STREAM("Add constraint "<<ip<<" to active set\n") | ||
461 | |||
462 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
463 | std::cerr << "Trying with constraint " << ip << std::endl; | ||
464 | utils::print_vector("np", np); | ||
465 | #endif | ||
466 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2); | ||
467 | |||
468 | 7 | l2a: /* Step 2a: determine step direction */ | |
469 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2A); | ||
470 | /* compute z = H np: the step direction in the primal space | ||
471 | (through m_J, see the paper) */ | ||
472 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | compute_d(d, m_J, np); |
473 | // update_z(z, m_J, d, iq); | ||
474 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 6 times.
|
7 | if (iq >= nVars) { |
475 | // throw std::runtime_error("iq >= m_J.cols()"); | ||
476 |
1/2✓ Branch 1 taken 1 times.
✗ Branch 2 not taken.
|
1 | z.setZero(); |
477 | } else { | ||
478 |
1/2✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
|
6 | update_z(z, m_J, d, iq); |
479 | } | ||
480 | /* compute N* np (if q > 0): the negative of the | ||
481 | step direction in the dual space */ | ||
482 |
1/2✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
|
7 | update_r(R, r, d, iq); |
483 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
484 | std::cerr << "Step direction z" << std::endl; | ||
485 | utils::print_vector("z", z); | ||
486 | utils::print_vector("r", r); | ||
487 | utils::print_vector("u", u); | ||
488 | utils::print_vector("d", d); | ||
489 | utils::print_vector("A", A); | ||
490 | #endif | ||
491 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2A); | ||
492 | |||
493 | /* Step 2b: compute step length */ | ||
494 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2B); | ||
495 | 7 | l = 0; | |
496 | /* Compute t1: partial step length (maximum step in dual | ||
497 | space without violating dual feasibility */ | ||
498 | 7 | t1 = inf; /* +inf */ | |
499 | /* find the index l s.t. it reaches the minimum of u+(x) / r */ | ||
500 | // l: index of constraint to drop (maybe) | ||
501 |
2/2✓ Branch 0 taken 3 times.
✓ Branch 1 taken 7 times.
|
10 | for (k = nEqCon; k < iq; k++) { |
502 | double tmp; | ||
503 |
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)) { |
504 | ✗ | t1 = tmp; | |
505 | ✗ | l = A(k); | |
506 | } | ||
507 | } | ||
508 | /* Compute t2: full step length (minimum step in primal | ||
509 | space such that the constraint ip becomes feasible */ | ||
510 |
3/4✓ Branch 1 taken 7 times.
✗ Branch 2 not taken.
✓ Branch 5 taken 5 times.
✓ Branch 6 taken 2 times.
|
7 | if (std::abs(z.dot(z)) > std::numeric_limits<double>::epsilon()) |
511 | // i.e. z != 0 | ||
512 |
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); |
513 | else | ||
514 | 2 | t2 = inf; /* +inf */ | |
515 | |||
516 | /* the step is chosen as the minimum of t1 and t2 */ | ||
517 | 7 | t = std::min(t1, t2); | |
518 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
519 | std::cerr << "Step sizes: " << t << " (t1 = " << t1 << ", t2 = " << t2 | ||
520 | << ") "; | ||
521 | #endif | ||
522 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2B); | ||
523 | |||
524 | /* Step 2c: determine new S-pair and take step: */ | ||
525 | START_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2C); | ||
526 | /* case (i): no step in primal or dual space */ | ||
527 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 5 times.
|
7 | if (t >= inf) { |
528 | /* QPP is infeasible */ | ||
529 | 2 | q = iq; | |
530 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2C); | ||
531 | 2 | return EIQUADPROG_FAST_UNBOUNDED; | |
532 | } | ||
533 | /* case (ii): step in dual space */ | ||
534 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 5 times.
|
5 | if (t2 >= inf) { |
535 | /* set u = u + t * [-r 1) and drop constraint l from the active set A */ | ||
536 | ✗ | u.head(iq) -= t * r.head(iq); | |
537 | ✗ | u(iq) += t; | |
538 | ✗ | iai(l) = static_cast<VectorXi::Scalar>(l); | |
539 | ✗ | delete_constraint(R, m_J, A, u, nEqCon, iq, l); | |
540 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
541 | std::cerr << " in dual space: " << f_value << std::endl; | ||
542 | utils::print_vector("x", x); | ||
543 | utils::print_vector("z", z); | ||
544 | utils::print_vector("A", A); | ||
545 | #endif | ||
546 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2C); | ||
547 | ✗ | goto l2a; | |
548 | } | ||
549 | |||
550 | /* case (iii): step in primal and dual space */ | ||
551 |
2/4✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
✓ Branch 4 taken 5 times.
✗ Branch 5 not taken.
|
5 | x += t * z; |
552 | /* update the solution value */ | ||
553 |
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)); |
554 | |||
555 |
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); |
556 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | u(iq) += t; |
557 | |||
558 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
559 | std::cerr << " in both spaces: " << f_value << std::endl; | ||
560 | utils::print_vector("x", x); | ||
561 | utils::print_vector("u", u); | ||
562 | utils::print_vector("r", r); | ||
563 | utils::print_vector("A", A); | ||
564 | #endif | ||
565 | |||
566 |
1/2✓ Branch 0 taken 5 times.
✗ Branch 1 not taken.
|
5 | if (t == t2) { |
567 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
568 | std::cerr << "Full step has taken " << t << std::endl; | ||
569 | utils::print_vector("x", x); | ||
570 | #endif | ||
571 | /* full step has taken */ | ||
572 | /* add constraint ip to the active set*/ | ||
573 |
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, m_J, d, iq, R_norm)) { |
574 | ✗ | iaexcl(ip) = 0; | |
575 | ✗ | delete_constraint(R, m_J, A, u, nEqCon, iq, ip); | |
576 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
577 | utils::print_matrix("R", R); | ||
578 | utils::print_vector("A", A); | ||
579 | #endif | ||
580 | ✗ | for (i = 0; i < nIneqCon; i++) iai(i) = static_cast<VectorXi::Scalar>(i); | |
581 | ✗ | for (i = 0; i < iq; i++) { | |
582 | ✗ | A(i) = A_old(i); | |
583 | ✗ | iai(A(i)) = -1; | |
584 | ✗ | u(i) = u_old(i); | |
585 | } | ||
586 | ✗ | x = x_old; | |
587 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2C); | ||
588 | ✗ | goto l2; /* go to step 2 */ | |
589 | } else | ||
590 |
1/2✓ Branch 1 taken 5 times.
✗ Branch 2 not taken.
|
5 | iai(ip) = -1; |
591 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
592 | utils::print_matrix("R", R); | ||
593 | utils::print_vector("A", A); | ||
594 | #endif | ||
595 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2C); | ||
596 | 5 | goto l1; | |
597 | } | ||
598 | |||
599 | /* a partial step has been taken => drop constraint l */ | ||
600 | ✗ | iai(l) = static_cast<VectorXi::Scalar>(l); | |
601 | ✗ | delete_constraint(R, m_J, A, u, nEqCon, iq, l); | |
602 | ✗ | s(ip) = CI.row(ip).dot(x) + ci0(ip); | |
603 | |||
604 | #ifdef EIQGUADPROG_TRACE_SOLVER | ||
605 | std::cerr << "Partial step has taken " << t << std::endl; | ||
606 | utils::print_vector("x", x); | ||
607 | utils::print_matrix("R", R); | ||
608 | utils::print_vector("A", A); | ||
609 | utils::print_vector("s", s); | ||
610 | #endif | ||
611 | STOP_PROFILER_EIQUADPROG_FAST(EIQUADPROG_FAST_STEP_2C); | ||
612 | |||
613 | ✗ | goto l2a; | |
614 | } | ||
615 | |||
616 | } /* namespace solvers */ | ||
617 | } /* namespace eiquadprog */ | ||
618 |