forked from RobotLocomotion/drake
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsymbolic_formula_cell.h
572 lines (513 loc) · 22.1 KB
/
symbolic_formula_cell.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
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
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
#pragma once
/// @file
///
/// Provides implementation-details of symbolic formulas.
///
/// It is strongly discouraged to include and use this header file outside of
/// drake/common/symbolic_* files. To include this file, you need to define
/// `DRAKE_COMMON_SYMBOLIC_DETAIL_HEADER` before. Without it, you have
/// compile-time errors.
#ifndef DRAKE_COMMON_SYMBOLIC_DETAIL_HEADER
#error Do not include this file unless you implement symbolic libraries.
#endif
#include <memory>
#include <ostream>
#include <set>
#include <string>
#include "drake/common/drake_assert.h"
#include "drake/common/hash.h"
#include "drake/common/random.h"
#include "drake/common/symbolic.h"
namespace drake {
namespace symbolic {
/** Represents an abstract class which is the base of concrete symbolic-formula
* classes (i.e. symbolic::FormulaAnd, symbolic::FormulaEq).
*
* \note It provides virtual function, FormulaCell::Display,
* because operator<< is not allowed to be a virtual function.
*/
class FormulaCell {
public:
/** Returns kind of formula. */
[[nodiscard]] FormulaKind get_kind() const { return kind_; }
/** Sends all hash-relevant bytes for this FormulaCell type into the given
* hasher, per the @ref hash_append concept -- except for get_kind(), because
* Formula already sends that.
*/
virtual void HashAppendDetail(DelegatingHasher*) const = 0;
/** Returns set of free variables in formula. */
[[nodiscard]] virtual Variables GetFreeVariables() const = 0;
/** Checks structural equality. */
[[nodiscard]] virtual bool EqualTo(const FormulaCell& c) const = 0;
/** Checks ordering. */
[[nodiscard]] virtual bool Less(const FormulaCell& c) const = 0;
/** Evaluates under a given environment. */
[[nodiscard]] virtual bool Evaluate(const Environment& env) const = 0;
/** Returns a Formula obtained by replacing all occurrences of the
* variables in @p s in the current formula cell with the corresponding
* expressions in @p s.
*/
[[nodiscard]] virtual Formula Substitute(const Substitution& s) const = 0;
/** Outputs string representation of formula into output stream @p os. */
virtual std::ostream& Display(std::ostream& os) const = 0;
/** Default constructor (deleted). */
FormulaCell() = delete;
/** Move-assign (deleted). */
FormulaCell& operator=(FormulaCell&& f) = delete;
/** Copy-assign (deleted). */
FormulaCell& operator=(const FormulaCell& f) = delete;
protected:
/** Move-construct a formula from an rvalue. */
FormulaCell(FormulaCell&& f) = default;
/** Copy-construct a formula from an lvalue. */
FormulaCell(const FormulaCell& f) = default;
/** Construct FormulaCell of kind @p k. */
explicit FormulaCell(FormulaKind k);
/** Default destructor. */
virtual ~FormulaCell() = default;
private:
const FormulaKind kind_{};
};
/** Represents the base class for relational operators (==, !=, <, <=, >, >=).
*/
class RelationalFormulaCell : public FormulaCell {
public:
/** Default constructor (deleted). */
RelationalFormulaCell() = delete;
/** Move-construct a formula from an rvalue. */
RelationalFormulaCell(RelationalFormulaCell&& f) = default;
/** Copy-construct a formula from an lvalue. */
RelationalFormulaCell(const RelationalFormulaCell& f) = default;
/** Move-assign (DELETED). */
RelationalFormulaCell& operator=(RelationalFormulaCell&& f) = delete;
/** Copy-assign (DELETED). */
RelationalFormulaCell& operator=(const RelationalFormulaCell& f) = delete;
/** Construct RelationalFormulaCell of kind @p k with @p lhs and @p rhs. */
RelationalFormulaCell(FormulaKind k, Expression lhs, Expression rhs);
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
/** Returns the expression on left-hand-side. */
[[nodiscard]] const Expression& get_lhs_expression() const { return e_lhs_; }
/** Returns the expression on right-hand-side. */
[[nodiscard]] const Expression& get_rhs_expression() const { return e_rhs_; }
private:
const Expression e_lhs_;
const Expression e_rhs_;
};
/** Represents the base class for N-ary logic operators (∧ and ∨).
*
* @note Internally this class maintains a set of symbolic formulas to avoid
* duplicated elements (i.e. f1 ∧ ... ∧ f1).
*/
class NaryFormulaCell : public FormulaCell {
public:
/** Default constructor (deleted). */
NaryFormulaCell() = delete;
/** Move-construct a formula from an rvalue. */
NaryFormulaCell(NaryFormulaCell&& f) = default;
/** Copy-construct a formula from an lvalue. */
NaryFormulaCell(const NaryFormulaCell& f) = default;
/** Move-assign (DELETED). */
NaryFormulaCell& operator=(NaryFormulaCell&& f) = delete;
/** Copy-assign (DELETED). */
NaryFormulaCell& operator=(const NaryFormulaCell& f) = delete;
/** Construct NaryFormulaCell of kind @p k with @p formulas. */
NaryFormulaCell(FormulaKind k, std::set<Formula> formulas);
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
/** Returns the formulas. */
[[nodiscard]] const std::set<Formula>& get_operands() const {
return formulas_;
}
protected:
std::ostream& DisplayWithOp(std::ostream& os, const std::string& op) const;
private:
const std::set<Formula> formulas_;
};
/** Symbolic formula representing true. */
class FormulaTrue : public FormulaCell {
public:
/** Default Constructor. */
FormulaTrue();
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing false. */
class FormulaFalse : public FormulaCell {
public:
/** Default Constructor. */
FormulaFalse();
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing a Boolean variable. */
class FormulaVar : public FormulaCell {
public:
/** Constructs a formula from @p var.
* @pre @p var is of BOOLEAN type and not a dummy variable.
*/
explicit FormulaVar(Variable v);
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& subst) const override;
std::ostream& Display(std::ostream& os) const override;
[[nodiscard]] const Variable& get_variable() const;
private:
const Variable var_;
};
/** Symbolic formula representing equality (e1 = e2). */
class FormulaEq : public RelationalFormulaCell {
public:
/** Constructs from @p e1 and @p e2. */
FormulaEq(const Expression& e1, const Expression& e2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing disequality (e1 ≠ e2). */
class FormulaNeq : public RelationalFormulaCell {
public:
/** Constructs from @p e1 and @p e2. */
FormulaNeq(const Expression& e1, const Expression& e2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing 'greater-than' (e1 > e2). */
class FormulaGt : public RelationalFormulaCell {
public:
/** Constructs from @p e1 and @p e2. */
FormulaGt(const Expression& e1, const Expression& e2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing 'greater-than-or-equal-to' (e1 ≥ e2). */
class FormulaGeq : public RelationalFormulaCell {
public:
/** Constructs from @p e1 and @p e2. */
FormulaGeq(const Expression& e1, const Expression& e2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing 'less-than' (e1 < e2). */
class FormulaLt : public RelationalFormulaCell {
public:
/** Constructs from @p e1 and @p e2. */
FormulaLt(const Expression& e1, const Expression& e2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing 'less-than-or-equal-to' (e1 ≤ e2). */
class FormulaLeq : public RelationalFormulaCell {
public:
/** Constructs from @p e1 and @p e2. */
FormulaLeq(const Expression& e1, const Expression& e2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing conjunctions (f1 ∧ ... ∧ fn). */
class FormulaAnd : public NaryFormulaCell {
public:
/** Constructs from @p formulas. */
explicit FormulaAnd(const std::set<Formula>& formulas);
/** Constructs @p f1 ∧ @p f2. */
FormulaAnd(const Formula& f1, const Formula& f2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing disjunctions (f1 ∨ ... ∨ fn). */
class FormulaOr : public NaryFormulaCell {
public:
/** Constructs from @p formulas. */
explicit FormulaOr(const std::set<Formula>& formulas);
/** Constructs @p f1 ∨ @p f2. */
FormulaOr(const Formula& f1, const Formula& f2);
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
};
/** Symbolic formula representing negations (¬f). */
class FormulaNot : public FormulaCell {
public:
/** Constructs from @p f. */
explicit FormulaNot(Formula f);
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
/** Returns the operand. */
[[nodiscard]] const Formula& get_operand() const { return f_; }
private:
const Formula f_;
};
/** Symbolic formula representing universal quantifications
* (∀ x₁, ..., * xn. F).
*/
class FormulaForall : public FormulaCell {
public:
/** Constructs from @p vars and @p f. */
FormulaForall(Variables vars, Formula f);
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
/** Returns the quantified variables. */
[[nodiscard]] const Variables& get_quantified_variables() const {
return vars_;
}
/** Returns the quantified formula. */
[[nodiscard]] const Formula& get_quantified_formula() const { return f_; }
private:
const Variables vars_; // Quantified variables.
const Formula f_; // Quantified formula.
};
/** Symbolic formula representing isnan predicate. */
class FormulaIsnan : public FormulaCell {
public:
explicit FormulaIsnan(Expression e);
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
[[nodiscard]] bool Less(const FormulaCell& f) const override;
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
private:
const Expression e_;
};
/** Symbolic formula representing positive-semidefinite (PSD) constraint. */
class FormulaPositiveSemidefinite : public FormulaCell {
public:
/** Constructs a positive-semidefinite formula from a symmetric matrix @p m.
*
* @throws std::runtime_error if @p m is not symmetric.
*
* @note This constructor checks if @p m is symmetric, which can be costly.
*/
explicit FormulaPositiveSemidefinite(
const Eigen::Ref<const MatrixX<Expression>>& m);
/** Constructs a symbolic positive-semidefinite formula from a
* lower triangular-view @p l.
*/
template <typename Derived>
explicit FormulaPositiveSemidefinite(
const Eigen::TriangularView<Derived, Eigen::Lower>& l)
: FormulaCell{FormulaKind::PositiveSemidefinite},
m_{BuildSymmetricMatrixFromLowerTringularView(l)} {}
/** Constructs a symbolic positive-semidefinite formula from an
* upper triangular-view @p u.
*/
template <typename Derived>
explicit FormulaPositiveSemidefinite(
const Eigen::TriangularView<Derived, Eigen::Upper>& u)
: FormulaCell{FormulaKind::PositiveSemidefinite},
m_{BuildSymmetricMatrixFromUpperTriangularView(u)} {}
void HashAppendDetail(DelegatingHasher*) const override;
[[nodiscard]] Variables GetFreeVariables() const override;
[[nodiscard]] bool EqualTo(const FormulaCell& f) const override;
/** Checks ordering between this PSD formula and @p f. The ordering between
* two PSD formulas `psd1` and `psd2` are determined by the ordering between
* the two matrices `m1` in `psd1` and `m2` in `psd2`.
*
* First, we compare the size of `m1` and `m2`:
*
* - If `m1` is smaller than `m2`, `psd1.Less(psd2)` is true.
* - If `m2` is smaller than `m1`, `psd1.Less(psd2)` is false.
*
* If `m1` and `m2` are of the same size, we perform element-wise comparison
* by following column-major order. See the following example:
*
* @code
* m1 << x + y, -3.14,
* -3.14, y;
* m2 << x + y, 3.14,
* 3.14, y;
* const Formula psd1{positive_semidefinite(m1)};
* const Formula psd2{positive_semidefinite(m2)};
*
* EXPECT_TRUE(psd1.Less(psd2));
* @endcode
*
* @note In the code above, `psd1.Less(psd2)` holds because we have
*
* - m1 in column-major ordering : (x + y) -3.14 -3.14 y_
* - m2 in column-major ordering : (x + y) 3.14 3.14 y_.
*/
[[nodiscard]] bool Less(const FormulaCell& f) const override;
[[nodiscard]] bool Evaluate(const Environment& env) const override;
[[nodiscard]] Formula Substitute(const Substitution& s) const override;
std::ostream& Display(std::ostream& os) const override;
/** Returns the corresponding matrix in this PSD formula. */
[[nodiscard]] const MatrixX<symbolic::Expression>& get_matrix() const {
return m_;
}
private:
// Builds a symmetric matrix from a lower triangular-view.
template <typename Derived>
static MatrixX<Expression> BuildSymmetricMatrixFromLowerTringularView(
const Eigen::TriangularView<Derived, Eigen::Lower>& l) {
MatrixX<Expression> m(l.rows(), l.cols());
m.triangularView<Eigen::Lower>() = l;
m.triangularView<Eigen::StrictlyUpper>() =
m.transpose().triangularView<Eigen::StrictlyUpper>();
return m;
}
// Builds a symmetric matrix from an upper triangular-view.
template <typename Derived>
static MatrixX<Expression> BuildSymmetricMatrixFromUpperTriangularView(
const Eigen::TriangularView<Derived, Eigen::Upper>& u) {
MatrixX<Expression> m(u.rows(), u.cols());
m.triangularView<Eigen::Upper>() = u;
m.triangularView<Eigen::StrictlyLower>() =
m.transpose().triangularView<Eigen::StrictlyLower>();
return m;
}
const MatrixX<Expression> m_;
};
/** Checks if @p f is structurally equal to False formula. */
bool is_false(const FormulaCell& f);
/** Checks if @p f is structurally equal to True formula. */
bool is_true(const FormulaCell& f);
/** Checks if @p f is a variable formula. */
bool is_variable(const FormulaCell& f);
/** Checks if @p f is a formula representing equality (==). */
bool is_equal_to(const FormulaCell& f);
/** Checks if @p f is a formula representing disequality (!=). */
bool is_not_equal_to(const FormulaCell& f);
/** Checks if @p f is a formula representing greater-than (>). */
bool is_greater_than(const FormulaCell& f);
/** Checks if @p f is a formula representing greater-than-or-equal-to (>=). */
bool is_greater_than_or_equal_to(const FormulaCell& f);
/** Checks if @p f is a formula representing less-than (<). */
bool is_less_than(const FormulaCell& f);
/** Checks if @p f is a formula representing less-than-or-equal-to (<=). */
bool is_less_than_or_equal_to(const FormulaCell& f);
/** Checks if @p f is a relational formula ({==, !=, >, >=, <, <=}). */
bool is_relational(const FormulaCell& f);
/** Checks if @p f is a conjunction (∧). */
bool is_conjunction(const FormulaCell& f);
/** Checks if @p f is a disjunction (∨). */
bool is_disjunction(const FormulaCell& f);
/** Checks if @p f is a negation (¬). */
bool is_negation(const FormulaCell& f);
/** Checks if @p f is a Forall formula (∀). */
bool is_forall(const FormulaCell& f);
/** Checks if @p f is an isnan formula. */
bool is_isnan(const FormulaCell& f);
/** Checks if @p f is a positive semidefinite formula. */
bool is_positive_semidefinite(const FormulaCell& f);
/** Casts @p f_ptr to @c shared_ptr<const FormulaFalse>.
* @pre @c is_false(*f_ptr) is true.
*/
std::shared_ptr<const FormulaFalse> to_false(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaTrue>.
* @pre @c is_true(*f_ptr) is true.
*/
std::shared_ptr<const FormulaTrue> to_true(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaVar>.
* @pre @c is_variable(*f_ptr) is true.
*/
std::shared_ptr<const FormulaVar> to_variable(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const RelationalFormulaCell>.
* @pre @c is_relational(*f_ptr) is true.
*/
std::shared_ptr<const RelationalFormulaCell> to_relational(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaEq>.
* @pre @c is_equal_to(*f_ptr) is true.
*/
std::shared_ptr<const FormulaEq> to_equal_to(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaNeq>.
* @pre @c is_not_equal_to(*f_ptr) is true.
*/
std::shared_ptr<const FormulaNeq> to_not_equal_to(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaGt>.
* @pre @c is_greater_than(*f_ptr) is true.
*/
std::shared_ptr<const FormulaGt> to_greater_than(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaGeq>.
* @pre @c is_greater_than_or_equal_to(*f_ptr) is true.
*/
std::shared_ptr<const FormulaGeq> to_greater_than_or_equal_to(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaLt>.
* @pre @c is_less_than(*f_ptr) is true.
*/
std::shared_ptr<const FormulaLt> to_less_than(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaLeq>.
* @pre @c is_less_than_or_equal_to(*f_ptr) is true.
*/
std::shared_ptr<const FormulaLeq> to_less_than_or_equal_to(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaAnd>.
* @pre @c is_conjunction(*f_ptr) is true.
*/
std::shared_ptr<const FormulaAnd> to_conjunction(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaOr>.
* @pre @c is_disjunction(*f_ptr) is true.
*/
std::shared_ptr<const FormulaOr> to_disjunction(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const NaryFormulaCell>.
* @pre @c is_nary(*f_ptr) is true.
*/
std::shared_ptr<const NaryFormulaCell> to_nary(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaNot>.
* @pre @c is_negation(*f_ptr) is true.
*/
std::shared_ptr<const FormulaNot> to_negation(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaForall>.
* @pre @c is_forall(*f_ptr) is true.
*/
std::shared_ptr<const FormulaForall> to_forall(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaIsnan>.
* @pre @c is_isnan(*f_ptr) is true.
*/
std::shared_ptr<const FormulaIsnan> to_isnan(
const std::shared_ptr<const FormulaCell>& f_ptr);
/** Casts @p f_ptr to @c shared_ptr<const FormulaPositiveSemidefinite>.
* @pre @c is_positive_semidefinite(*f_ptr) is true.
*/
std::shared_ptr<const FormulaPositiveSemidefinite> to_positive_semidefinite(
const std::shared_ptr<const FormulaCell>& f_ptr);
} // namespace symbolic
} // namespace drake