11#include <Eigen/SparseCore>
12#include <gch/small_vector.hpp>
14#include "sleipnir/optimization/solver/exit_status.hpp"
15#include "sleipnir/optimization/solver/iteration_info.hpp"
16#include "sleipnir/optimization/solver/options.hpp"
17#include "sleipnir/optimization/solver/sqp_matrix_callbacks.hpp"
18#include "sleipnir/optimization/solver/util/all_finite.hpp"
19#include "sleipnir/optimization/solver/util/append_as_triplets.hpp"
20#include "sleipnir/optimization/solver/util/feasibility_restoration.hpp"
21#include "sleipnir/optimization/solver/util/filter.hpp"
22#include "sleipnir/optimization/solver/util/is_locally_infeasible.hpp"
23#include "sleipnir/optimization/solver/util/kkt_error.hpp"
24#include "sleipnir/optimization/solver/util/regularized_ldlt.hpp"
25#include "sleipnir/util/assert.hpp"
26#include "sleipnir/util/print_diagnostics.hpp"
27#include "sleipnir/util/profiler.hpp"
28#include "sleipnir/util/scope_exit.hpp"
29#include "sleipnir/util/symbol_exports.hpp"
55template <
typename Scalar>
56ExitStatus sqp(
const SQPMatrixCallbacks<Scalar>& matrix_callbacks,
57 std::span<std::function<
bool(
const IterationInfo<Scalar>& info)>>
59 const Options& options,
60 Eigen::Vector<Scalar, Eigen::Dynamic>& x) {
61 using DenseVector = Eigen::Vector<Scalar, Eigen::Dynamic>;
63 DenseVector y = DenseVector::Zero(matrix_callbacks.num_equality_constraints);
65 return sqp(matrix_callbacks, iteration_callbacks, options, x, y);
90template <
typename Scalar>
91ExitStatus sqp(
const SQPMatrixCallbacks<Scalar>& matrix_callbacks,
92 std::span<std::function<
bool(
const IterationInfo<Scalar>& info)>>
94 const Options& options, Eigen::Vector<Scalar, Eigen::Dynamic>& x,
95 Eigen::Vector<Scalar, Eigen::Dynamic>& y) {
96 using DenseVector = Eigen::Vector<Scalar, Eigen::Dynamic>;
97 using SparseMatrix = Eigen::SparseMatrix<Scalar>;
98 using SparseVector = Eigen::SparseVector<Scalar>;
110 const auto solve_start_time = std::chrono::steady_clock::now();
112 gch::small_vector<SolveProfiler> solve_profilers;
113 solve_profilers.emplace_back(
"solver");
114 solve_profilers.emplace_back(
"↳ setup");
115 solve_profilers.emplace_back(
"↳ iteration");
116 solve_profilers.emplace_back(
" ↳ feasibility check");
117 solve_profilers.emplace_back(
" ↳ callbacks");
118 solve_profilers.emplace_back(
" ↳ KKT matrix build");
119 solve_profilers.emplace_back(
" ↳ KKT matrix decomp");
120 solve_profilers.emplace_back(
" ↳ KKT system solve");
121 solve_profilers.emplace_back(
" ↳ line search");
122 solve_profilers.emplace_back(
" ↳ SOC");
123 solve_profilers.emplace_back(
" ↳ feas. restoration");
124 solve_profilers.emplace_back(
" ↳ f(x)");
125 solve_profilers.emplace_back(
" ↳ ∇f(x)");
126 solve_profilers.emplace_back(
" ↳ ∇²ₓₓL");
127 solve_profilers.emplace_back(
" ↳ ∇²ₓₓL_c");
128 solve_profilers.emplace_back(
" ↳ cₑ(x)");
129 solve_profilers.emplace_back(
" ↳ ∂cₑ/∂x");
131 auto& solver_prof = solve_profilers[0];
132 auto& setup_prof = solve_profilers[1];
133 auto& inner_iter_prof = solve_profilers[2];
134 auto& feasibility_check_prof = solve_profilers[3];
135 auto& iter_callbacks_prof = solve_profilers[4];
136 auto& kkt_matrix_build_prof = solve_profilers[5];
137 auto& kkt_matrix_decomp_prof = solve_profilers[6];
138 auto& kkt_system_solve_prof = solve_profilers[7];
139 auto& line_search_prof = solve_profilers[8];
140 auto& soc_prof = solve_profilers[9];
141 auto& feasibility_restoration_prof = solve_profilers[10];
144#ifndef SLEIPNIR_DISABLE_DIAGNOSTICS
145 auto& f_prof = solve_profilers[11];
146 auto& g_prof = solve_profilers[12];
147 auto& H_prof = solve_profilers[13];
148 auto& H_c_prof = solve_profilers[14];
149 auto& c_e_prof = solve_profilers[15];
150 auto& A_e_prof = solve_profilers[16];
152 SQPMatrixCallbacks<Scalar> matrices{
153 matrix_callbacks.num_decision_variables,
154 matrix_callbacks.num_equality_constraints,
155 [&](
const DenseVector& x) -> Scalar {
156 ScopedProfiler prof{f_prof};
157 return matrix_callbacks.f(x);
159 [&](
const DenseVector& x) -> SparseVector {
160 ScopedProfiler prof{g_prof};
161 return matrix_callbacks.g(x);
163 [&](
const DenseVector& x,
const DenseVector& y) -> SparseMatrix {
164 ScopedProfiler prof{H_prof};
165 return matrix_callbacks.H(x, y);
167 [&](
const DenseVector& x,
const DenseVector& y) -> SparseMatrix {
168 ScopedProfiler prof{H_c_prof};
169 return matrix_callbacks.H_c(x, y);
171 [&](
const DenseVector& x) -> DenseVector {
172 ScopedProfiler prof{c_e_prof};
173 return matrix_callbacks.c_e(x);
175 [&](
const DenseVector& x) -> SparseMatrix {
176 ScopedProfiler prof{A_e_prof};
177 return matrix_callbacks.A_e(x);
180 const auto& matrices = matrix_callbacks;
186 Scalar f = matrices.f(x);
187 SparseVector g = matrices.g(x);
188 SparseMatrix H = matrices.H(x, y);
189 DenseVector c_e = matrices.c_e(x);
190 SparseMatrix A_e = matrices.A_e(x);
193 slp_assert(g.rows() == matrices.num_decision_variables);
194 slp_assert(H.rows() == matrices.num_decision_variables);
195 slp_assert(H.cols() == matrices.num_decision_variables);
196 slp_assert(c_e.rows() == matrices.num_equality_constraints);
197 slp_assert(A_e.rows() == matrices.num_equality_constraints);
198 slp_assert(A_e.cols() == matrices.num_decision_variables);
204 DenseVector trial_c_e;
207 if (matrices.num_equality_constraints > matrices.num_decision_variables) {
208 if (options.diagnostics) {
209 print_too_few_dofs_error(c_e);
212 return ExitStatus::TOO_FEW_DOFS;
216 if (!isfinite(f) || !all_finite(g) || !all_finite(H) || !c_e.allFinite() ||
218 return ExitStatus::NONFINITE_INITIAL_GUESS;
223 Filter<Scalar> filter{c_e.template lpNorm<1>()};
226 gch::small_vector<Eigen::Triplet<Scalar>> triplets;
228 RegularizedLDLT<Scalar> solver{matrices.num_decision_variables,
229 matrices.num_equality_constraints};
232 constexpr Scalar α_reduction_factor(0.5);
233 constexpr Scalar α_min(1e-7);
235 int full_step_rejected_counter = 0;
238 Scalar E_0 = kkt_error<Scalar, KKTErrorType::INF_NORM_SCALED>(g, A_e, c_e, y);
243 scope_exit exit{[&] {
244 if (options.diagnostics) {
246 if (iterations > 0) {
247 print_bottom_iteration_diagnostics();
249 print_solver_diagnostics(solve_profilers);
253 while (E_0 > Scalar(options.tolerance)) {
254 ScopedProfiler inner_iter_profiler{inner_iter_prof};
255 ScopedProfiler feasibility_check_profiler{feasibility_check_prof};
258 if (is_equality_locally_infeasible(A_e, c_e)) {
259 if (options.diagnostics) {
260 print_c_e_local_infeasibility_error(c_e);
263 return ExitStatus::LOCALLY_INFEASIBLE;
267 if (x.template lpNorm<Eigen::Infinity>() > Scalar(1e10) || !x.allFinite()) {
268 return ExitStatus::DIVERGING_ITERATES;
271 feasibility_check_profiler.stop();
272 ScopedProfiler iter_callbacks_profiler{iter_callbacks_prof};
275 for (
const auto& callback : iteration_callbacks) {
276 if (callback({iterations, x, {}, y, {}, g, H, A_e, {}})) {
277 return ExitStatus::CALLBACK_REQUESTED_STOP;
281 iter_callbacks_profiler.stop();
282 ScopedProfiler kkt_matrix_build_profiler{kkt_matrix_build_prof};
289 triplets.reserve(H.nonZeros() + A_e.nonZeros());
290 append_as_triplets(triplets, 0, 0, {H, A_e});
292 matrices.num_decision_variables + matrices.num_equality_constraints,
293 matrices.num_decision_variables + matrices.num_equality_constraints);
294 lhs.setFromSortedTriplets(triplets.begin(), triplets.end());
298 DenseVector rhs{x.rows() + y.rows()};
299 rhs.segment(0, x.rows()) = -g + A_e.transpose() * y;
300 rhs.segment(x.rows(), y.rows()) = -c_e;
302 kkt_matrix_build_profiler.stop();
303 ScopedProfiler kkt_matrix_decomp_profiler{kkt_matrix_decomp_prof};
306 constexpr Scalar α_max(1);
308 bool call_feasibility_restoration =
false;
314 if (solver.compute(lhs).info() != Eigen::Success) [[unlikely]] {
315 return ExitStatus::FACTORIZATION_FAILED;
318 kkt_matrix_decomp_profiler.stop();
319 ScopedProfiler kkt_system_solve_profiler{kkt_system_solve_prof};
321 auto compute_step = [&](Step& step) {
324 DenseVector p = solver.solve(rhs);
325 step.p_x = p.segment(0, x.rows());
326 step.p_y = -p.segment(x.rows(), y.rows());
330 kkt_system_solve_profiler.stop();
331 ScopedProfiler line_search_profiler{line_search_prof};
335 const FilterEntry<Scalar> current_entry{f, c_e};
339 trial_x = x + α * step.p_x;
340 trial_y = y + α * step.p_y;
342 trial_f = matrices.f(trial_x);
343 trial_c_e = matrices.c_e(trial_x);
347 if (!isfinite(trial_f) || !trial_c_e.allFinite()) {
349 α *= α_reduction_factor;
352 call_feasibility_restoration =
true;
359 FilterEntry trial_entry{trial_f, trial_c_e};
360 if (filter.try_add(current_entry, trial_entry, step.p_x, g, α)) {
365 Scalar prev_constraint_violation = c_e.template lpNorm<1>();
366 Scalar next_constraint_violation = trial_c_e.template lpNorm<1>();
373 next_constraint_violation >= prev_constraint_violation) {
375 auto soc_step = step;
378 DenseVector c_e_soc = c_e;
380 bool step_acceptable =
false;
381 for (
int soc_iteration = 0; soc_iteration < 5 && !step_acceptable;
383 ScopedProfiler soc_profiler{soc_prof};
385 scope_exit soc_exit{[&] {
388 if (options.diagnostics) {
389 print_iteration_diagnostics(
391 step_acceptable ? IterationType::ACCEPTED_SOC
392 : IterationType::REJECTED_SOC,
393 soc_profiler.current_duration(),
394 kkt_error<Scalar, KKTErrorType::INF_NORM_SCALED>(
395 g, A_e, trial_c_e, trial_y),
396 trial_f, trial_c_e.template lpNorm<1>(), Scalar(0), Scalar(0),
397 solver.hessian_regularization(), α_soc, Scalar(1),
398 α_reduction_factor, Scalar(1));
408 c_e_soc = α_soc * c_e_soc + trial_c_e;
409 rhs.bottomRows(y.rows()) = -c_e_soc;
412 compute_step(soc_step);
414 trial_x = x + α_soc * soc_step.p_x;
415 trial_y = y + α_soc * soc_step.p_y;
417 trial_f = matrices.f(trial_x);
418 trial_c_e = matrices.c_e(trial_x);
421 constexpr Scalar κ_soc(0.99);
425 next_constraint_violation = trial_c_e.template lpNorm<1>();
426 if (next_constraint_violation > κ_soc * prev_constraint_violation) {
431 FilterEntry trial_entry{trial_f, trial_c_e};
432 if (filter.try_add(current_entry, trial_entry, step.p_x, g, α)) {
435 step_acceptable =
true;
439 if (step_acceptable) {
449 ++full_step_rejected_counter;
456 if (full_step_rejected_counter >= 4 &&
457 filter.max_constraint_violation >
458 current_entry.constraint_violation / Scalar(10) &&
459 filter.last_rejection_due_to_filter()) {
460 filter.max_constraint_violation *= Scalar(0.1);
466 α *= α_reduction_factor;
471 Scalar current_kkt_error =
472 kkt_error<Scalar, KKTErrorType::ONE_NORM>(g, A_e, c_e, y);
474 trial_x = x + α_max * step.p_x;
475 trial_y = y + α_max * step.p_y;
477 trial_f = matrices.f(trial_x);
478 trial_c_e = matrices.c_e(trial_x);
480 Scalar next_kkt_error = kkt_error<Scalar, KKTErrorType::ONE_NORM>(
481 matrices.g(trial_x), matrices.A_e(trial_x), trial_c_e, trial_y);
484 if (next_kkt_error <= Scalar(0.999) * current_kkt_error) {
489 call_feasibility_restoration =
true;
494 line_search_profiler.stop();
496 if (call_feasibility_restoration) {
497 ScopedProfiler feasibility_restoration_profiler{
498 feasibility_restoration_prof};
500 FilterEntry initial_entry{matrices.f(x), c_e};
503 gch::small_vector<std::function<bool(
const IterationInfo<Scalar>& info)>>
505 for (
auto& callback : iteration_callbacks) {
506 callbacks.emplace_back(callback);
508 callbacks.emplace_back([&](
const IterationInfo<Scalar>& info) {
509 DenseVector trial_x =
510 info.x.segment(0, matrices.num_decision_variables);
512 DenseVector trial_c_e = matrices.c_e(trial_x);
514 FilterEntry trial_entry{matrices.f(trial_x), trial_c_e};
518 return trial_entry.constraint_violation <
519 Scalar(0.9) * initial_entry.constraint_violation &&
520 filter.try_add(initial_entry, trial_entry, trial_x - x, g, α);
523 feasibility_restoration<Scalar>(matrices, callbacks, options, x, y);
525 if (status != ExitStatus::SUCCESS) {
531 c_e = matrices.c_e(x);
535 full_step_rejected_counter = 0;
547 A_e = matrices.A_e(x);
549 H = matrices.H(x, y);
552 E_0 = kkt_error<Scalar, KKTErrorType::INF_NORM_SCALED>(g, A_e, c_e, y);
554 inner_iter_profiler.stop();
556 if (options.diagnostics) {
557 print_iteration_diagnostics(iterations, IterationType::NORMAL,
558 inner_iter_profiler.current_duration(), E_0,
559 f, c_e.template lpNorm<1>(), Scalar(0),
560 Scalar(0), solver.hessian_regularization(), α,
561 α_max, α_reduction_factor, α);
567 if (iterations >= options.max_iterations) {
568 return ExitStatus::MAX_ITERATIONS_EXCEEDED;
572 if (std::chrono::steady_clock::now() - solve_start_time > options.timeout) {
573 return ExitStatus::TIMEOUT;
577 return ExitStatus::SUCCESS;
580extern template SLEIPNIR_DLLEXPORT ExitStatus
581sqp(
const SQPMatrixCallbacks<double>& matrix_callbacks,
582 std::span<std::function<
bool(
const IterationInfo<double>& info)>>
584 const Options& options, Eigen::Vector<double, Eigen::Dynamic>& x);