forked from RobotLocomotion/drake
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdrake_bool.h
121 lines (106 loc) · 4.42 KB
/
drake_bool.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
#pragma once
#include <type_traits>
#include <Eigen/Core>
#include "drake/common/double_overloads.h"
#include "drake/common/drake_throw.h"
namespace drake {
/// A traits struct that describes the return type of predicates over a scalar
/// type (named `T`). For example, a predicate that evaluates `double`s will
/// return a `bool`, but a predicate that evaluates symbolic::Expression will
/// return a symbolic::Formula. By default, the return type is inferred from
/// the type's comparison operator, but scalar types are permitted to
/// specialize this template for their needs.
template <typename T>
struct scalar_predicate {
/// The return type of predicates over T.
using type = decltype(T() < T());
/// Whether `type` is `bool`.
static constexpr bool is_bool = std::is_same_v<type, bool>;
};
/// An alias for a boolean-like value, conditioned on the scalar type `T`.
/// In many cases this will be a synonym for `bool`, e.g., when `T = double`.
/// When `T = symbolic::Expression`, this is a synonym for `symbolic::Formula`.
/// This is a convenience abbreviation for scalar_predicate<T>::type.
template <typename T>
using boolean = typename scalar_predicate<T>::type;
/// Checks truth for all elements in matrix @p m. This is identical to
/// `Eigen::DenseBase::all()`, except this function allows for lazy evaluation,
/// so works even when scalar_predicate<>::is_bool does not hold. An empty
/// matrix returns true.
template <typename Derived>
typename Derived::Scalar all(const Eigen::DenseBase<Derived>& m) {
using Boolish = typename Derived::Scalar;
if (m.rows() == 0 || m.cols() == 0) {
// `all` holds vacuously when there is nothing to check.
return Boolish{true};
}
return m.redux([](const Boolish& v1, const Boolish& v2) {
return v1 && v2;
});
}
/// Checks if unary predicate @p pred holds for all elements in the matrix @p m.
/// An empty matrix returns true.
template <typename Derived>
boolean<typename Derived::Scalar> all_of(
const Eigen::MatrixBase<Derived>& m,
const std::function<boolean<typename Derived::Scalar>(
const typename Derived::Scalar&)>& pred) {
return drake::all(m.unaryExpr(pred));
}
/// Checks truth for at least one element in matrix @p m. This is identical to
/// `Eigen::DenseBase::any()`, except this function allows for lazy evaluation,
/// so works even when scalar_predicate<>::is_bool does not hold. An empty
/// matrix returns false.
template <typename Derived>
typename Derived::Scalar any(const Eigen::DenseBase<Derived>& m) {
using Boolish = typename Derived::Scalar;
if (m.rows() == 0 || m.cols() == 0) {
// `any` is vacuously false when there is nothing to check.
return Boolish{false};
}
return m.redux([](const Boolish& v1, const Boolish& v2) {
return v1 || v2;
});
}
/// Checks if unary predicate @p pred holds for at least one element in the
/// matrix @p m. An empty matrix returns false.
template <typename Derived>
boolean<typename Derived::Scalar> any_of(
const Eigen::MatrixBase<Derived>& m,
const std::function<boolean<typename Derived::Scalar>(
const typename Derived::Scalar&)>& pred) {
return any(m.unaryExpr(pred));
}
/// Checks that no elements of @p m are true. An empty matrix returns true.
template <typename Derived>
typename Derived::Scalar none(const Eigen::MatrixBase<Derived>& m) {
using Boolish = typename Derived::Scalar;
const auto negate = [](const Boolish& v) -> Boolish {
return !v;
};
return drake::all(m.unaryExpr(negate));
}
/// Checks if unary predicate @p pred holds for no elements in the matrix @p m.
/// An empty matrix returns true.
template <typename Derived>
boolean<typename Derived::Scalar> none_of(
const Eigen::MatrixBase<Derived>& m,
const std::function<boolean<typename Derived::Scalar>(
const typename Derived::Scalar&)>& pred) {
return none(m.unaryExpr(pred));
}
/// Overloads if_then_else for Eigen vectors of `m_then` and `m_else` values
/// with with a single `f_cond` condition to toggle them all at once.
template <typename T, int Rows>
auto if_then_else(const boolean<T>& f_cond,
const Eigen::Matrix<T, Rows, 1>& m_then,
const Eigen::Matrix<T, Rows, 1>& m_else) {
DRAKE_THROW_UNLESS(m_then.rows() == m_else.rows());
const int rows = m_then.rows();
Eigen::Matrix<T, Rows, 1> result(rows);
for (int i = 0; i < rows; ++i) {
result[i] = if_then_else(f_cond, m_then[i], m_else[i]);
}
return result;
}
} // namespace drake