forked from RobotLocomotion/drake
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsolver_base.cc
95 lines (82 loc) · 3 KB
/
solver_base.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
#include "drake/solvers/solver_base.h"
#include <limits>
#include <utility>
#include <fmt/format.h>
#include "drake/common/drake_assert.h"
#include "drake/common/nice_type_name.h"
namespace drake {
namespace solvers {
SolverBase::SolverBase(
std::function<SolverId()> id,
std::function<bool()> available,
std::function<bool()> enabled,
std::function<bool(const MathematicalProgram&)> satisfied)
: default_id_(std::move(id)),
default_available_(std::move(available)),
default_enabled_(std::move(enabled)),
default_satisfied_(std::move(satisfied)) {}
SolverBase::~SolverBase() = default;
MathematicalProgramResult SolverBase::Solve(
const MathematicalProgram& prog,
const std::optional<Eigen::VectorXd>& initial_guess,
const std::optional<SolverOptions>& solver_options) const {
MathematicalProgramResult result;
this->Solve(prog, initial_guess, solver_options, &result);
return result;
}
void SolverBase::Solve(const MathematicalProgram& prog,
const std::optional<Eigen::VectorXd>& initial_guess,
const std::optional<SolverOptions>& solver_options,
MathematicalProgramResult* result) const {
*result = {};
if (!available()) {
throw std::invalid_argument(fmt::format(
"The {} is not available in this build", NiceTypeName::Get(*this)));
}
if (!enabled()) {
throw std::invalid_argument(fmt::format(
"{}::is_enabled() is false; see its documentation for how to enable.",
NiceTypeName::Get(*this)));
}
if (!AreProgramAttributesSatisfied(prog)) {
throw std::invalid_argument(fmt::format(
"The capabilities of {} do not meet the requirements of the "
"MathematicalProgram ({})", NiceTypeName::Get(*this),
to_string(prog.required_capabilities())));
}
result->set_solver_id(solver_id());
result->set_decision_variable_index(prog.decision_variable_index());
const Eigen::VectorXd& x_init =
initial_guess ? *initial_guess : prog.initial_guess();
if (x_init.rows() != prog.num_vars()) {
throw std::invalid_argument(
fmt::format("Solve expects initial guess of size {}, got {}.",
prog.num_vars(), x_init.rows()));
}
if (!solver_options) {
DoSolve(prog, x_init, prog.solver_options(), result);
} else {
SolverOptions merged_options = *solver_options;
merged_options.Merge(prog.solver_options());
DoSolve(prog, x_init, merged_options, result);
}
}
bool SolverBase::available() const {
DRAKE_DEMAND(default_available_ != nullptr);
return default_available_();
}
bool SolverBase::enabled() const {
DRAKE_DEMAND(default_enabled_ != nullptr);
return default_enabled_();
}
SolverId SolverBase::solver_id() const {
DRAKE_DEMAND(default_id_ != nullptr);
return default_id_();
}
bool SolverBase::AreProgramAttributesSatisfied(
const MathematicalProgram& prog) const {
DRAKE_DEMAND(default_satisfied_ != nullptr);
return default_satisfied_(prog);
}
} // namespace solvers
} // namespace drake