forked from google/or-tools
-
Notifications
You must be signed in to change notification settings - Fork 9
/
quadratic_program.h
224 lines (203 loc) · 10.8 KB
/
quadratic_program.h
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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
// Copyright 2010-2024 Google LLC
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#ifndef PDLP_QUADRATIC_PROGRAM_H_
#define PDLP_QUADRATIC_PROGRAM_H_
#include <cstdint>
#include <limits>
#include <optional>
#include <string>
#include <utility>
#include <vector>
#include "Eigen/Core"
#include "Eigen/SparseCore"
#include "absl/status/status.h"
#include "absl/status/statusor.h"
#include "ortools/linear_solver/linear_solver.pb.h"
namespace operations_research::pdlp {
// Represents the quadratic program (QP):
// min_x (`objective_vector`^T x + (1/2) x^T `objective_matrix` x) s.t.
// `constraint_lower_bounds` <= `constraint_matrix` x
// <= `constraint_upper_bounds`
// `variable_lower_bounds` <= x <= `variable_upper_bounds`
//
// `constraint_lower_bounds` and `variable_lower_bounds` may include negative
// infinities. `constraint_upper_bounds` and `variable_upper_bounds` may
// contain positive infinities. Other than that all entries of all fields must
// be finite. The `objective_matrix` must be diagonal and non-negative.
//
// `variable_lower_bounds`, `variable_upper_bounds`, `objective_vector`,
// `objective_matrix` (if it has a value), and `variable_names` (if it has a
// value) must all have the same size as the number of columns in
// `constraint_matrix`. `constraint_lower_bounds`, `constraint_upper_bounds`,
// and `constraint_names` (if it has a value) must all have the same size as the
// number of rows in `constraint_matrix`. Consistency of these values is checked
// by `ValidateQuadraticProgramDimensions()`.
//
// For convenience, the struct also stores `scaling_factor` and
// `objective_offset`. These factors can be used to transform objective values
// based on the problem definition above into objective values that are
// meaningful for the user. See `ApplyObjectiveScalingAndOffset`.
//
// This struct is also intended for use with linear programs (LPs), which are
// QPs with a zero `objective_matrix`.
//
// The dual is documented at
// https://developers.google.com/optimization/lp/pdlp_math.
struct QuadraticProgram {
QuadraticProgram(int64_t num_variables, int64_t num_constraints) {
ResizeAndInitialize(num_variables, num_constraints);
}
QuadraticProgram() : QuadraticProgram(0, 0) {}
// `QuadraticProgram` may be copied or moved. `Eigen::SparseMatrix` doesn't
// have move operations so we use custom implementations based on swap.
QuadraticProgram(const QuadraticProgram& other) = default;
QuadraticProgram(QuadraticProgram&& other) noexcept
: objective_vector(std::move(other.objective_vector)),
objective_matrix(std::move(other.objective_matrix)),
constraint_lower_bounds(std::move(other.constraint_lower_bounds)),
constraint_upper_bounds(std::move(other.constraint_upper_bounds)),
variable_lower_bounds(std::move(other.variable_lower_bounds)),
variable_upper_bounds(std::move(other.variable_upper_bounds)),
problem_name(std::move(other.problem_name)),
variable_names(std::move(other.variable_names)),
constraint_names(std::move(other.constraint_names)),
objective_offset(other.objective_offset),
objective_scaling_factor(other.objective_scaling_factor) {
constraint_matrix.swap(other.constraint_matrix);
}
QuadraticProgram& operator=(const QuadraticProgram& other) = default;
QuadraticProgram& operator=(QuadraticProgram&& other) {
objective_vector = std::move(other.objective_vector);
objective_matrix = std::move(other.objective_matrix);
constraint_matrix.swap(other.constraint_matrix);
constraint_lower_bounds = std::move(other.constraint_lower_bounds);
constraint_upper_bounds = std::move(other.constraint_upper_bounds);
variable_lower_bounds = std::move(other.variable_lower_bounds);
variable_upper_bounds = std::move(other.variable_upper_bounds);
problem_name = std::move(other.problem_name);
variable_names = std::move(other.variable_names);
constraint_names = std::move(other.constraint_names);
objective_offset = other.objective_offset;
objective_scaling_factor = other.objective_scaling_factor;
return *this;
}
// Initializes the quadratic program with `num_variables` variables and
// `num_constraints` constraints. Lower and upper bounds are set to negative
// and positive infinity, respectively. `objective_matrix` is cleared. All
// other matrices and vectors are set to zero. Resets the optional names
// (`program_name`, `variable_names`, and `constraint_names`).
// `objective_offset` is set to 0 and `objective_scaling_factor` is set to 1.
void ResizeAndInitialize(int64_t num_variables, int64_t num_constraints) {
constexpr double kInfinity = std::numeric_limits<double>::infinity();
objective_vector = Eigen::VectorXd::Zero(num_variables);
objective_matrix.reset();
constraint_matrix.resize(num_constraints, num_variables);
constraint_lower_bounds =
Eigen::VectorXd::Constant(num_constraints, -kInfinity);
constraint_upper_bounds =
Eigen::VectorXd::Constant(num_constraints, kInfinity);
variable_lower_bounds =
Eigen::VectorXd::Constant(num_variables, -kInfinity);
variable_upper_bounds = Eigen::VectorXd::Constant(num_variables, kInfinity);
problem_name.reset();
variable_names.reset();
constraint_names.reset();
objective_offset = 0.0;
objective_scaling_factor = 1.0;
}
// Returns `objective_scaling_factor * (objective + objective_offset)`.
// `objective_scaling_factor` is useful for modeling maximization problems.
// For example, max c^T x = -1 * min (-c)^T x. `objective_offset` can be a
// by-product of presolve transformations that eliminate variables.
double ApplyObjectiveScalingAndOffset(double objective) const {
return objective_scaling_factor * (objective + objective_offset);
}
Eigen::VectorXd objective_vector;
// If this field isn't set, the `objective matrix` is interpreted to be zero,
// i.e., this is a linear programming problem.
std::optional<Eigen::DiagonalMatrix<double, Eigen::Dynamic>> objective_matrix;
Eigen::SparseMatrix<double, Eigen::ColMajor, int64_t> constraint_matrix;
Eigen::VectorXd constraint_lower_bounds, constraint_upper_bounds;
Eigen::VectorXd variable_lower_bounds, variable_upper_bounds;
std::optional<std::string> problem_name;
std::optional<std::vector<std::string>> variable_names;
std::optional<std::vector<std::string>> constraint_names;
// These fields are provided for convenience; they don't change the
// mathematical definition of the problem, but they change the objective
// values reported to the user.
double objective_offset;
double objective_scaling_factor;
};
// Returns `InvalidArgumentError` if vector or matrix dimensions are
// inconsistent. Returns `OkStatus` otherwise.
absl::Status ValidateQuadraticProgramDimensions(const QuadraticProgram& qp);
inline bool IsLinearProgram(const QuadraticProgram& qp) {
return !qp.objective_matrix.has_value();
}
// Converts an `MPModelProto` into a `QuadraticProgram`.
// Returns an error if general constraints are present.
// If `relax_integer_variables` is true integer variables are relaxed to
// continuous; otherwise integer variables are an error.
// If `include_names` is true (the default is false), the problem, constraint,
// and variable names are included in the `QuadraticProgram`; otherwise they are
// left empty.
// Maximization problems are converted to minimization by negating the
// objective and setting `objective_scaling_factor` to -1, which preserves the
// reported objective values.
absl::StatusOr<QuadraticProgram> QpFromMpModelProto(
const MPModelProto& proto, bool relax_integer_variables,
bool include_names = false);
// Returns `InvalidArgumentError` if `qp` is too large to convert to
// `MPModelProto` and `OkStatus` otherwise.
absl::Status CanFitInMpModelProto(const QuadraticProgram& qp);
// Converts a `QuadraticProgram` into an `MPModelProto`. To preserve objective
// values in the conversion, `objective_vector`, `objective_matrix`, and
// `objective_offset` are scaled by `objective_scaling_factor`, and if
// `objective_scaling_factor` is negative, then the proto is a maximization
// problem (otherwise it's a minimization problem). Returns
// `InvalidArgumentError` if `objective_scaling_factor` is zero or if
// `CanFitInMpModelProto()` fails.
absl::StatusOr<MPModelProto> QpToMpModelProto(const QuadraticProgram& qp);
// Returns a "pretty" version of `qp`, truncating to at most `max_size`
// characters. This is for debugging purposes only - the format may change
// without notice. Although this output is vaguely similar to "LP format", it is
// not actually compatible with "LP format".
std::string ToString(const QuadraticProgram& qp, int64_t max_size = 1'000'000);
// Like `matrix.setFromTriplets(triplets)`, except that `setFromTriplets`
// results in having three copies of the nonzeros in memory at the same time,
// because it first fills one matrix from triplets, and then transposes it into
// another. This avoids having the third copy in memory by sorting the triplets,
// reserving space in the matrix, and then inserting in sorted order.
// Compresses the matrix (`SparseMatrix.makeCompressed()`) after loading it.
// NOTE: This intentionally passes `triplets` by value, because it modifies
// them. To avoid the copy, pass a move reference.
void SetEigenMatrixFromTriplets(
std::vector<Eigen::Triplet<double, int64_t>> triplets,
Eigen::SparseMatrix<double, Eigen::ColMajor, int64_t>& matrix);
// Utility functions for internal use only.
namespace internal {
// Like `CanFitInMpModelProto()` but has an extra argument for the largest
// number of variables, constraints, or objective non-zeros that should be
// counted as convertible. `CanFitInMpModelProto()` passes 2^31 - 1 for this
// argument and unit tests pass small values.
absl::Status TestableCanFitInMpModelProto(const QuadraticProgram& qp,
int64_t largest_ok_size);
// Modifies `triplets` in place, combining consecutive entries with the same row
// and column, summing their values. This is most effective if `triplets` are
// sorted by row and column, so that multiple entries for the same entry will be
// consecutive.
void CombineRepeatedTripletsInPlace(
std::vector<Eigen::Triplet<double, int64_t>>& triplets);
} // namespace internal
} // namespace operations_research::pdlp
#endif // PDLP_QUADRATIC_PROGRAM_H_