forked from RobotLocomotion/drake
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsymbolic_chebyshev_basis_element.cc
251 lines (231 loc) · 9.68 KB
/
symbolic_chebyshev_basis_element.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
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
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
// NOLINTNEXTLINE(build/include): Its header file is included in symbolic.h.
#include <cmath>
#include <vector>
#include "drake/common/symbolic.h"
#define DRAKE_COMMON_SYMBOLIC_DETAIL_HEADER
#include "drake/common/symbolic_expression_cell.h"
#undef DRAKE_COMMON_SYMBOLIC_DETAIL_HEADER
namespace drake {
namespace symbolic {
ChebyshevBasisElement::ChebyshevBasisElement() : PolynomialBasisElement() {}
ChebyshevBasisElement::ChebyshevBasisElement(
const std::map<Variable, int>& var_to_degree_map)
: PolynomialBasisElement(var_to_degree_map) {}
ChebyshevBasisElement::ChebyshevBasisElement(const Variable& var)
: ChebyshevBasisElement({{var, 1}}) {}
ChebyshevBasisElement::ChebyshevBasisElement(const Variable& var, int degree)
: ChebyshevBasisElement({{var, degree}}) {}
ChebyshevBasisElement::ChebyshevBasisElement(std::nullptr_t)
: ChebyshevBasisElement() {}
ChebyshevBasisElement::ChebyshevBasisElement(
const Eigen::Ref<const VectorX<Variable>>& vars,
const Eigen::Ref<const Eigen::VectorXi>& degrees)
: PolynomialBasisElement(vars, degrees) {}
bool ChebyshevBasisElement::operator<(
const ChebyshevBasisElement& other) const {
return this->lexicographical_compare(other);
}
std::map<ChebyshevBasisElement, double> ChebyshevBasisElement::Differentiate(
const Variable& var) const {
if (var_to_degree_map().count(var) == 0) {
// Return an empty map (the differentiation result is 0) when @p var is not
// a variable in @p this.
return {};
}
std::map<ChebyshevBasisElement, double> result;
std::map<Variable, int> var_to_degree_map = this->var_to_degree_map();
auto it = var_to_degree_map.find(var);
const int degree = it->second;
const int start_degree = degree % 2 == 0 ? 1 : 2;
for (int i = start_degree; i < degree; i += 2) {
it->second = i;
result.emplace(ChebyshevBasisElement(var_to_degree_map), 2 * degree);
}
// Lastly, add the term for T0(x) if the degree is odd. The coefficient for
// T0(x) is degree instead of 2 * degree.
if (degree % 2 == 1) {
it->second = 0;
result.emplace(ChebyshevBasisElement(var_to_degree_map), degree);
}
return result;
}
std::map<ChebyshevBasisElement, double> ChebyshevBasisElement::Integrate(
const Variable& var) const {
auto var_to_degree_map = this->var_to_degree_map();
auto it = var_to_degree_map.find(var);
if (it == var_to_degree_map.end()) {
// var is not a variable in this Chebyshev basis element.
// Append T1(var) to the var_to_degree_map.
var_to_degree_map.emplace_hint(it, var, 1);
return {{ChebyshevBasisElement(var_to_degree_map), 1}};
}
const int degree = it->second;
std::map<ChebyshevBasisElement, double> result;
it->second = degree + 1;
result.emplace(ChebyshevBasisElement(var_to_degree_map),
1.0 / (2 * degree + 2));
it->second = degree - 1;
result.emplace(ChebyshevBasisElement(var_to_degree_map),
-1.0 / (2 * degree - 2));
return result;
}
void ChebyshevBasisElement::MergeBasisElementInPlace(
const ChebyshevBasisElement& other) {
this->DoMergeBasisElementInPlace(other);
}
std::pair<double, ChebyshevBasisElement> ChebyshevBasisElement::EvaluatePartial(
const Environment& env) const {
double coeff;
std::map<Variable, int> new_basis_element;
DoEvaluatePartial(env, &coeff, &new_basis_element);
return std::make_pair(coeff, ChebyshevBasisElement(new_basis_element));
}
double ChebyshevBasisElement::DoEvaluate(double variable_val,
int degree) const {
return EvaluateChebyshevPolynomial(variable_val, degree);
}
Expression ChebyshevBasisElement::DoToExpression() const {
std::map<Expression, Expression> base_to_exponent_map;
for (const auto& [var, degree] : var_to_degree_map()) {
base_to_exponent_map.emplace(
ChebyshevPolynomial(var, degree).ToPolynomial().ToExpression(), 1);
}
return ExpressionMulFactory{1.0, base_to_exponent_map}.GetExpression();
}
namespace {
// This function is used in operator*, it appends the pair (var, degree) to each
// entry of chebyshev_basis_all.
void AppendVariableAndDegree(
const Variable& var, int degree,
std::vector<std::map<Variable, int>>* chebyshev_basis_all) {
for (auto& cheby_basis : *chebyshev_basis_all) {
cheby_basis.emplace(var, degree);
}
}
int power_of_2(int degree) {
if (degree < 0) {
throw std::invalid_argument("power of 2 underflow");
}
if (degree > static_cast<int>((sizeof(int) * CHAR_BIT - 2))) {
throw std::invalid_argument("power of 2 overflow");
}
return 1 << degree;
}
} // namespace
std::map<ChebyshevBasisElement, double> operator*(
const ChebyshevBasisElement& a, const ChebyshevBasisElement& b) {
// If variable x shows up in both ChebyshevBasisElement a and b, we know that
// Tₘ(x) * Tₙ(x) = 0.5 (Tₘ₊ₙ(x) + Tₘ₋ₙ(x)), namely we will expand the product
// to the sum of two new Chebyshev polynomials. Hence the number of terms in
// the product of a * b is 2 to the power of # common variables showing up in
// both a and b.
// Number of variables that show up in both ChebyshevBasisElement a and b.
// I first count the nummber of common variables, so as to do memory
// allocation for the product result.
int num_common_variables = 0;
auto it_a = a.var_to_degree_map().begin();
auto it_b = b.var_to_degree_map().begin();
// Since the keys in var_to_degree_map are sorted, we can loop through
// a.var_to_degree_map and b.var_to_degree_map jointly to find the common
// variables.
while (it_a != a.var_to_degree_map().end() &&
it_b != b.var_to_degree_map().end()) {
const Variable& var_a = it_a->first;
const Variable& var_b = it_b->first;
if (var_a.less(var_b)) {
// var_a is less than var_b, and hence var_a less than all variables in b
// after var_b. We can increment it_a.
it_a++;
} else if (var_b.less(var_a)) {
it_b++;
} else {
num_common_variables++;
it_a++;
it_b++;
}
}
// The number of ChebyshevBasisElement in the product result is
// 2^num_common_variables.
std::vector<std::map<Variable, int>> chebyshev_basis_all(
power_of_2(num_common_variables));
// I will go through the (variable, degree) pair of both a and b. If the
// variable shows up in only a or b, then each term in the product a * b
// contains that variable and its degree. If a has term Tₘ(x) and b has term
// Tₙ(x), where x is the common variable, then half of the basis in a * b
// contains term Tₘ₊ₙ(x), and the other half contains Tₘ₋ₙ(x).
int common_variables_count = 0;
it_a = a.var_to_degree_map().begin();
it_b = b.var_to_degree_map().begin();
// Note that var_to_degree_map() has its keys sorted in an increasing order.
while (it_a != a.var_to_degree_map().end() &&
it_b != b.var_to_degree_map().end()) {
const Variable& var_a = it_a->first;
const int degree_a = it_a->second;
const Variable& var_b = it_b->first;
const int degree_b = it_b->second;
if (var_a.less(var_b)) {
// var_a is not a common variable, add (var_a, degree_a) to each element
// in chebyshev_basis_all.
AppendVariableAndDegree(var_a, degree_a, &chebyshev_basis_all);
it_a++;
} else if (var_b.less(var_a)) {
// var_b is not a common variable, add (var_b, degree_b) to each element
// in chebyshev_basis_all.
AppendVariableAndDegree(var_b, degree_b, &chebyshev_basis_all);
it_b++;
} else {
// Add (var_common, degree_a + degree_b) to half of the elements in
// chebyshev_basis_all, and (var_common, |degree_a - degree_b| to the
// other half of the elements in chebyshev_basis_all.
// If we denote 2 ^ common_variables_count = n, then we add (var_common,
// degree_a + degree_b) to the chebyshev_basis_all[i], if i is in the
// interval of [j * 2n, j * 2n + n) for some j, and we add (var_common,
// |degree_a - degree_b|) to chebyshev_basis_all[i] if i is in the
// interval [j * 2n + n, (j+1) * 2n).
const int degree_sum = degree_a + degree_b;
const int degree_diff = std::abs(degree_a - degree_b);
const int n = power_of_2(common_variables_count);
for (int j = 0;
j < power_of_2(num_common_variables - common_variables_count - 1);
++j) {
for (int i = j * 2 * n; i < j * 2 * n + n; ++i) {
chebyshev_basis_all[i].emplace(var_a, degree_sum);
}
for (int i = j * 2 * n + n; i < (j + 1) * 2 * n; ++i) {
chebyshev_basis_all[i].emplace(var_a, degree_diff);
}
}
it_a++;
it_b++;
common_variables_count++;
}
}
// Now either or both it_a or it_b is at the end of var_to_degree_map.end(),
// if it_a != a.var_to_degree_map().end(), then the remaining variables after
// it_a in a.var_to_degree_map cannot be a common variable. Append the rest of
// the (variable, degree) to the Chebyshev basis.
for (; it_a != a.var_to_degree_map().end(); ++it_a) {
AppendVariableAndDegree(it_a->first, it_a->second, &chebyshev_basis_all);
}
for (; it_b != b.var_to_degree_map().end(); ++it_b) {
AppendVariableAndDegree(it_b->first, it_b->second, &chebyshev_basis_all);
}
const double coeff = 1.0 / power_of_2(num_common_variables);
std::map<ChebyshevBasisElement, double> result;
for (const auto& var_to_degree_map : chebyshev_basis_all) {
result.emplace(ChebyshevBasisElement(var_to_degree_map), coeff);
}
return result;
}
std::ostream& operator<<(std::ostream& out, const ChebyshevBasisElement& m) {
if (m.var_to_degree_map().empty()) {
out << "T0()";
} else {
for (const auto& [var, degree] : m.var_to_degree_map()) {
out << ChebyshevPolynomial(var, degree);
}
}
return out;
}
} // namespace symbolic
} // namespace drake