Epigraph
Convex Optimization in C++
expressions.hpp
Go to the documentation of this file.
1 
6 #pragma once
7 
8 #include "parameter.hpp"
9 #include "variable.hpp"
10 
11 #include <Eigen/Sparse>
12 
13 namespace Eigen
14 {
15 
16  template <>
17  struct NumTraits<cvx::Scalar>
18  : NumTraits<double>
19  {
20  using Real = cvx::Scalar;
21  using NonInteger = cvx::Scalar;
22  using Nested = cvx::Scalar;
23 
24  enum
25  {
26  IsComplex = 0,
27  IsInteger = 0,
28  IsSigned = 1,
29  RequireInitialization = 1,
30  ReadCost = 10,
31  AddCost = 200,
32  MulCost = 200,
33  };
34  };
35 
36 } // namespace Eigen
37 
38 namespace cvx
39 {
40  class Constraint;
41  class OptimizationProblem;
42  class Scalar;
43 
44  namespace internal
45  {
46  class Affine;
47  class SOCPWrapperBase;
48  class QPWrapperBase;
49 
50  class Term
51  {
52  public:
53  Term();
54 
55  Parameter parameter;
56  Variable variable;
57 
58  bool operator==(const Term &other) const;
59  Term &operator*=(const Parameter &param);
60  Term &operator/=(const Parameter &param);
61 
62  operator Affine() const;
63 
64  friend std::ostream &operator<<(std::ostream &os, const Term &term);
65  double evaluate() const;
66  };
67 
68  class Affine
69  {
70  public:
71  bool operator==(const Affine &other) const;
72 
73  Parameter constant = Parameter(0.);
74  std::vector<Term> terms;
75 
76  friend std::ostream &operator<<(std::ostream &os, const Affine &affine);
77  double evaluate() const;
78  Affine &operator+=(const Affine &other);
79  Affine &operator-=(const Affine &other);
80  Affine &operator*=(const Parameter &param);
81  Affine &operator/=(const Parameter &param);
82  // Affine operator+(const Affine &other) const;
83  Affine operator-(const Affine &other) const;
84  Affine operator-() const;
85 
86  void cleanUp();
87 
88  bool isZero() const;
89  bool isConstant() const;
90  bool isFirstOrder() const;
91  };
92 
93  class Product
94  {
95  public:
96  explicit Product(const Affine &term);
97  Product(const Affine &lhs, const Affine &rhs);
98  Affine &firstTerm();
99  Affine &secondTerm();
100  const Affine &firstTerm() const;
101  const Affine &secondTerm() const;
102  void toSquaredTerm();
103  double evaluate() const;
104  bool isSquare() const;
105 
106  bool operator==(const Product &other) const;
107 
108  friend std::ostream &operator<<(std::ostream &os, const Product &product);
109 
110  private:
111  std::vector<Affine> factors;
112  };
113 
114  } // namespace internal
115 
116  class Scalar
117  {
118  public:
119  Scalar() = default;
120  explicit Scalar(int x);
121  Scalar(double x);
122  explicit Scalar(double *x);
123 
124  Scalar &operator+=(const Scalar &other);
125  Scalar &operator-=(const Scalar &other);
126  Scalar &operator*=(const Scalar &other);
127  Scalar &operator/=(const Scalar &other);
128  Scalar operator-() const;
129  friend Scalar operator+(const Scalar &lhs, const Scalar &rhs);
130  friend Scalar operator-(const Scalar &lhs, const Scalar &rhs);
131  friend Scalar operator*(const Scalar &lhs, const Scalar &rhs);
132  friend Scalar operator/(const Scalar &lhs, const Scalar &rhs);
133 
134  bool operator==(const cvx::Scalar &other) const;
135 
136  double evaluate() const;
137  size_t getOrder() const;
138  bool isNorm() const;
139 
140  friend OptimizationProblem;
141  friend internal::SOCPWrapperBase;
142  friend internal::QPWrapperBase;
143 
144  explicit operator double() const;
145 
146  friend Scalar sqrt(const Scalar &scalar);
147  friend Scalar square(const Scalar &scalar);
148  friend Scalar abs2(const Scalar &scalar);
149 
150  // friend internal::Parameter::operator Scalar() const;
151  friend internal::Variable::operator Scalar() const;
152 
153  friend Constraint equalTo(const Scalar &lhs, const Scalar &rhs);
154  friend Constraint lessThan(const Scalar &lhs, const Scalar &rhs);
155  friend Constraint greaterThan(const Scalar &lhs, const Scalar &rhs);
156  friend Constraint box(const Scalar &lower, const Scalar &middle, const Scalar &upper);
157 
158  private:
159  internal::Affine affine;
160  std::vector<internal::Product> products;
161  bool norm = false;
162 
163  friend std::ostream &operator<<(std::ostream &os, const Scalar &scalar);
164  };
165 
166  using MatrixX = Eigen::Matrix<cvx::Scalar, Eigen::Dynamic, Eigen::Dynamic>;
167  using VectorX = Eigen::Matrix<cvx::Scalar, Eigen::Dynamic, 1>;
168 
175  Scalar par(double p);
176 
187  Scalar dynpar(double &p);
188 
196  template <typename Derived>
197  inline auto par(const Eigen::MatrixBase<Derived> &m)
198  {
199  return m.template cast<Scalar>().eval();
200  }
201 
209  template <typename Derived>
210  inline auto par(const Eigen::SparseMatrixBase<Derived> &m)
211  {
212  return m.template cast<Scalar>().eval();
213  }
214 
226  template <typename Derived>
227  auto dynpar(Eigen::MatrixBase<Derived> &m)
228  {
229  auto result = m.template cast<Scalar>().eval();
230 
231  for (int row = 0; row < m.rows(); row++)
232  {
233  for (int col = 0; col < m.cols(); col++)
234  {
235  result.coeffRef(row, col) = dynpar(m.coeffRef(row, col));
236  }
237  }
238 
239  return result;
240  }
241 
253  template <typename T>
254  auto dynpar(Eigen::SparseMatrix<T> &m)
255  {
256  auto result = m.template cast<Scalar>().eval();
257 
258  for (int k = 0; k < result.nonZeros(); k++)
259  {
260  result.valuePtr()[k] = dynpar(m.valuePtr()[k]);
261  }
262 
263  return result;
264  }
265 
272  double eval(const Scalar &s);
273 
281  template <typename Derived>
282  inline auto eval(const Eigen::MatrixBase<Derived> &m)
283  {
284  return m.template cast<double>();
285  }
286 
294  template <typename Derived>
295  inline auto eval(const Eigen::SparseMatrixBase<Derived> &m)
296  {
297  return m.template cast<double>();
298  }
299 
300  inline const Scalar &conj(const Scalar &x) { return x; }
301  inline const Scalar &real(const Scalar &x) { return x; }
302  inline Scalar imag(const Scalar &) { return Scalar(0.); }
303  inline Scalar square(const Scalar &x)
304  {
305  Scalar new_scalar;
306  new_scalar.products = {internal::Product(x.affine)};
307  return new_scalar;
308  }
309  inline Scalar abs2(const Scalar &x)
310  {
311  return square(x);
312  }
313 
314 } // namespace cvx
cvx::Scalar::box
friend Constraint box(const Scalar &lower, const Scalar &middle, const Scalar &upper)
Create a box constraint: lower <= middle <= upper.
cvx::par
Scalar par(double p)
Creates a constant parameter.
cvx::dynpar
Scalar dynpar(double &p)
Creates a dynamic parameter.
cvx::Scalar::lessThan
friend Constraint lessThan(const Scalar &lhs, const Scalar &rhs)
Create a less than or equal constraint: lhs <= rhs.
cvx::Scalar::equalTo
friend Constraint equalTo(const Scalar &lhs, const Scalar &rhs)
Create an equality constraint: lhs == rhs.
cvx::Constraint
Definition: constraint.hpp:47
cvx::OptimizationProblem
Definition: problem.hpp:14
cvx::eval
double eval(const Scalar &s)
Evaluates the scalar.
cvx::Scalar::greaterThan
friend Constraint greaterThan(const Scalar &lhs, const Scalar &rhs)
Create a greater than or equal constraint: lhs >= rhs.
cvx::Scalar
Definition: expressions.hpp:117