forked from RobotLocomotion/drake
-
Notifications
You must be signed in to change notification settings - Fork 0
/
roll_pitch_yaw.h
649 lines (592 loc) · 31.8 KB
/
roll_pitch_yaw.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
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
#pragma once
#include <cmath>
#include <limits>
#include <Eigen/Dense>
#include "drake/common/default_scalars.h"
#include "drake/common/drake_assert.h"
#include "drake/common/drake_copyable.h"
#include "drake/common/eigen_types.h"
namespace drake {
namespace math {
template <typename T>
class RotationMatrix;
// TODO(@mitiguy) Add Sherm/Goldstein's way to visualize rotation sequences.
/// This class represents the orientation between two arbitrary frames A and D
/// associated with a Space-fixed (extrinsic) X-Y-Z rotation by "roll-pitch-yaw"
/// angles `[r, p, y]`, which is equivalent to a Body-fixed (intrinsic) Z-Y-X
/// rotation by "yaw-pitch-roll" angles `[y, p, r]`. The rotation matrix `R_AD`
/// associated with this roll-pitch-yaw `[r, p, y]` rotation sequence is equal
/// to the matrix multiplication shown below.
/// ```
/// ⎡cos(y) -sin(y) 0⎤ ⎡ cos(p) 0 sin(p)⎤ ⎡1 0 0 ⎤
/// R_AD = ⎢sin(y) cos(y) 0⎥ * ⎢ 0 1 0 ⎥ * ⎢0 cos(r) -sin(r)⎥
/// ⎣ 0 0 1⎦ ⎣-sin(p) 0 cos(p)⎦ ⎣0 sin(r) cos(r)⎦
/// = R_AB * R_BC * R_CD
/// ```
/// @note In this discussion, A is the Space frame and D is the Body frame.
/// One way to visualize this rotation sequence is by introducing intermediate
/// frames B and C (useful constructs to understand this rotation sequence).
/// Initially, the frames are aligned so `Di = Ci = Bi = Ai (i = x, y, z)`
/// where Dx, Dy, Dz and Ax, Ay, Az are orthogonal unit vectors fixed in frames
/// D and A respectively. Similarly for Bx, By, Bz and Cx, Cy, Cz in frame B, C.
/// Then D is subjected to successive right-handed rotations relative to A.
/// @li 1st rotation R_CD: %Frame D rotates relative to frames C, B, A by a
/// roll angle `r` about `Dx = Cx`. Note: D and C are no longer aligned.
/// @li 2nd rotation R_BC: Frames D, C (collectively -- as if welded together)
/// rotate relative to frame B, A by a pitch angle `p` about `Cy = By`.
/// Note: C and B are no longer aligned.
/// @li 3rd rotation R_AB: Frames D, C, B (collectively -- as if welded)
/// rotate relative to frame A by a roll angle `y` about `Bz = Az`.
/// Note: B and A are no longer aligned.
/// The monogram notation for the rotation matrix relating A to D is `R_AD`.
///
/// @see @ref multibody_quantities for monogram notation for dynamics and
/// @ref orientation_discussion "a discussion on rotation matrices".
///
/// @note This class does not store the frames associated with this rotation
/// sequence.
///
/// @tparam T The underlying scalar type. Must be a valid Eigen scalar.
///
/// Instantiated templates for the following kinds of T's are provided:
///
/// - double
/// - AutoDiffXd
/// - symbolic::Expression
template <typename T>
class RollPitchYaw {
public:
DRAKE_DEFAULT_COPY_AND_MOVE_AND_ASSIGN(RollPitchYaw)
/// Constructs a %RollPitchYaw from a 3x1 array of angles.
/// @param[in] rpy 3x1 array with roll, pitch, yaw angles (units of radians).
/// @throws std::logic_error in debug builds if !IsValid(rpy).
explicit RollPitchYaw(const Vector3<T>& rpy) { set(rpy); }
/// Constructs a %RollPitchYaw from roll, pitch, yaw angles (radian units).
/// @param[in] roll x-directed angle in SpaceXYZ rotation sequence.
/// @param[in] pitch y-directed angle in SpaceXYZ rotation sequence.
/// @param[in] yaw z-directed angle in SpaceXYZ rotation sequence.
/// @throws std::logic_error in debug builds if
/// !IsValid(Vector3<T>(roll, pitch, yaw)).
RollPitchYaw(const T& roll, const T& pitch, const T& yaw) {
set(roll, pitch, yaw);
}
/// Uses a %RotationMatrix to construct a %RollPitchYaw with
/// roll-pitch-yaw angles `[r, p, y]` in the range
/// `-π <= r <= π`, `-π/2 <= p <= π/2, `-π <= y <= π`.
/// @param[in] R a %RotationMatrix.
/// @note This new high-accuracy algorithm avoids numerical round-off issues
/// encountered by some algorithms when pitch is within 1E-6 of π/2 or -π/2.
explicit RollPitchYaw(const RotationMatrix<T>& R) {
SetFromRotationMatrix(R);
}
/// Uses a %Quaternion to construct a %RollPitchYaw with
/// roll-pitch-yaw angles `[r, p, y]` in the range
/// `-π <= r <= π`, `-π/2 <= p <= π/2, `-π <= y <= π`.
/// @param[in] quaternion unit %Quaternion.
/// @note This new high-accuracy algorithm avoids numerical round-off issues
/// encountered by some algorithms when pitch is within 1E-6 of π/2 or -π/2.
/// @throws std::logic_error in debug builds if !IsValid(rpy).
explicit RollPitchYaw(const Eigen::Quaternion<T>& quaternion) {
SetFromQuaternion(quaternion);
}
/// Sets `this` %RollPitchYaw from a 3x1 array of angles.
/// @param[in] rpy 3x1 array with roll, pitch, yaw angles (units of radians).
/// @throws std::logic_error in debug builds if !IsValid(rpy).
RollPitchYaw<T>& set(const Vector3<T>& rpy) {
return SetOrThrowIfNotValidInDebugBuild(rpy);
}
/// Sets `this` %RollPitchYaw from roll, pitch, yaw angles (units of radians).
/// @param[in] roll x-directed angle in SpaceXYZ rotation sequence.
/// @param[in] pitch y-directed angle in SpaceXYZ rotation sequence.
/// @param[in] yaw z-directed angle in SpaceXYZ rotation sequence.
/// @throws std::logic_error in debug builds if
/// !IsValid(Vector3<T>(roll, pitch, yaw)).
RollPitchYaw<T>& set(const T& roll, const T& pitch, const T& yaw) {
return set(Vector3<T>(roll, pitch, yaw));
}
/// Uses a %Quaternion to set `this` %RollPitchYaw with
/// roll-pitch-yaw angles `[r, p, y]` in the range
/// `-π <= r <= π`, `-π/2 <= p <= π/2, `-π <= y <= π`.
/// @param[in] quaternion unit %Quaternion.
/// @note This new high-accuracy algorithm avoids numerical round-off issues
/// encountered by some algorithms when pitch is within 1E-6 of π/2 or -π/2.
/// @throws std::logic_error in debug builds if !IsValid(rpy).
void SetFromQuaternion(const Eigen::Quaternion<T>& quaternion);
/// Uses a %RotationMatrix to set `this` %RollPitchYaw with
/// roll-pitch-yaw angles `[r, p, y]` in the range
/// `-π <= r <= π`, `-π/2 <= p <= π/2, `-π <= y <= π`.
/// @param[in] R a %RotationMatrix.
/// @note This new high-accuracy algorithm avoids numerical round-off issues
/// encountered by some algorithms when pitch is within 1E-6 of π/2 or -π/2.
void SetFromRotationMatrix(const RotationMatrix<T>& R);
/// Uses a quaternion and its associated rotation matrix `R` to accurately
/// set `this` @RollPitchYaw roll-pitch-yaw angles (SpaceXYZ Euler angles),
/// even when the pitch angle is within 1E-6 of π/2 or -π/2.
/// @param[in] quaternion unit quaternion with elements `[e0, e1, e2, e3]`.
/// @param[in] R The %RotationMatrix corresponding to `quaternion`.
/// @return %RollPitchYaw containing angles `[r, p, y]` with range
/// `-π <= r <= π`, `-π/2 <= p <= π/2, `-π <= y <= π`.
/// @throws std::logic_error in debug builds if rpy fails IsValid(rpy).
/// @note Aborts in debug builds if `quaternion` does not correspond to `R`.
void SetFromQuaternionAndRotationMatrix(
const Eigen::Quaternion<T>& quaternion, const RotationMatrix<T>& R);
/// Returns the Vector3 underlying `this` %RollPitchYaw.
const Vector3<T>& vector() const { return roll_pitch_yaw_; }
/// Returns the roll-angle underlying `this` %RollPitchYaw.
const T& roll_angle() const { return roll_pitch_yaw_(0); }
/// Returns the pitch-angle underlying `this` %RollPitchYaw.
const T& pitch_angle() const { return roll_pitch_yaw_(1); }
/// Returns the yaw-angle underlying `this` %RollPitchYaw.
const T& yaw_angle() const { return roll_pitch_yaw_(2); }
/// Set the roll-angle underlying `this` %RollPitchYaw.
/// @param[in] r roll angle (in units of radians).
void set_roll_angle(const T& r) { roll_pitch_yaw_(0) = r; }
/// Set the pitch-angle underlying `this` %RollPitchYaw.
/// @param[in] p pitch angle (in units of radians).
void set_pitch_angle(const T& p) { roll_pitch_yaw_(1) = p; }
/// Set the yaw-angle underlying `this` %RollPitchYaw.
/// @param[in] y yaw angle (in units of radians).
void set_yaw_angle(const T& y) { roll_pitch_yaw_(2) = y; }
/// Returns a quaternion representation of `this` %RollPitchYaw.
Eigen::Quaternion<T> ToQuaternion() const {
using std::cos;
using std::sin;
const T q0Half = roll_pitch_yaw_(0) / 2;
const T q1Half = roll_pitch_yaw_(1) / 2;
const T q2Half = roll_pitch_yaw_(2) / 2;
const T c0 = cos(q0Half), s0 = sin(q0Half);
const T c1 = cos(q1Half), s1 = sin(q1Half);
const T c2 = cos(q2Half), s2 = sin(q2Half);
const T c1_c2 = c1 * c2, s1_c2 = s1 * c2;
const T s1_s2 = s1 * s2, c1_s2 = c1 * s2;
const T w = c0 * c1_c2 + s0 * s1_s2;
const T x = s0 * c1_c2 - c0 * s1_s2;
const T y = c0 * s1_c2 + s0 * c1_s2;
const T z = c0 * c1_s2 - s0 * s1_c2;
return Eigen::Quaternion<T>(w, x, y, z);
}
/// Returns the RotationMatrix representation of `this` %RollPitchYaw.
RotationMatrix<T> ToRotationMatrix() const;
/// Returns the 3x3 matrix representation of the %RotationMatrix that
/// corresponds to `this` %RollPitchYaw. This is a convenient "sugar" method
/// that is exactly equivalent to RotationMatrix(rpy).ToMatrix3().
Matrix3<T> ToMatrix3ViaRotationMatrix() const {
return ToRotationMatrix().matrix();
}
/// Compares each element of `this` to the corresponding element of `other`
/// to check if they are the same to within a specified `tolerance`.
/// @param[in] other %RollPitchYaw to compare to `this`.
/// @param[in] tolerance maximum allowable absolute difference between the
/// matrix elements in `this` and `other`.
/// @returns `true` if `‖this - other‖∞ <= tolerance`.
boolean<T> IsNearlyEqualTo(
const RollPitchYaw<T>& other, double tolerance) const {
const Vector3<T> difference = vector() - other.vector();
return difference.template lpNorm<Eigen::Infinity>() <= tolerance;
}
/// Compares each element of the %RotationMatrix R1 produced by `this` to the
/// corresponding element of the %RotationMatrix R2 produced by `other` to
/// check if they are the same to within a specified `tolerance`.
/// @param[in] other %RollPitchYaw to compare to `this`.
/// @param[in] tolerance maximum allowable absolute difference between R1, R2.
/// @returns `true` if `‖R1 - R2‖∞ <= tolerance`.
boolean<T> IsNearlySameOrientation(const RollPitchYaw<T>& other,
double tolerance) const;
/// Returns true if roll-pitch-yaw angles `[r, p, y]` are in the range
/// `-π <= r <= π`, `-π/2 <= p <= π/2, `-π <= y <= π`.
boolean<T> IsRollPitchYawInCanonicalRange() const {
const T& r = roll_angle();
const T& p = pitch_angle();
const T& y = yaw_angle();
return (-M_PI <= r && r <= M_PI) && (-M_PI / 2 <= p && p <= M_PI / 2) &&
(-M_PI <= y && y <= M_PI);
}
/// Returns true if the pitch-angle `p` is close to gimbal-lock, which means
/// `cos(p) ≈ 0` or `p ≈ (n*π + π/2)` where `n = 0, ±1, ±2, ...`.
/// More specifically, returns true if `abs(cos_pitch_angle)` is less than an
/// internally-defined tolerance of gimbal-lock.
/// @param[in] cos_pitch_angle cosine of the pitch angle, i.e., `cos(p)`.
/// @note Pitch-angles close to gimbal-lock can can cause problems with
/// numerical precision and numerical integration.
static boolean<T> DoesCosPitchAngleViolateGimbalLockTolerance(
const T& cos_pitch_angle) {
using std::abs;
return abs(cos_pitch_angle) < kGimbalLockToleranceCosPitchAngle;
}
/// Returns true if the pitch-angle `p` is within an internally-defined
/// tolerance of gimbal-lock. In other words, this method returns true if
/// `p ≈ (n*π + π/2)` where `n = 0, ±1, ±2, ...`.
/// @note To improve efficiency when cos(pitch_angle()) is already calculated,
/// instead use the function DoesCosPitchAngleViolateGimbalLockTolerance().
/// @see DoesCosPitchAngleViolateGimbalLockTolerance()
boolean<T> DoesPitchAngleViolateGimbalLockTolerance() const {
using std::cos;
return DoesCosPitchAngleViolateGimbalLockTolerance(cos(pitch_angle()));
}
/// Returns the internally-defined allowable closeness (in radians) of the
/// pitch angle `p` to gimbal-lock, i.e., the allowable proximity of `p` to
/// `(n*π + π/2)` where `n = 0, ±1, ±2, ...`.
static double GimbalLockPitchAngleTolerance() {
return M_PI_2 - std::acos(kGimbalLockToleranceCosPitchAngle);
}
/// Returns true if `rpy` contains valid roll, pitch, yaw angles.
/// @param[in] rpy allegedly valid roll, pitch, yaw angles.
/// @note an angle is invalid if it is NaN or infinite.
static boolean<T> IsValid(const Vector3<T>& rpy) {
using std::isfinite;
return isfinite(rpy[0]) && isfinite(rpy[1]) && isfinite(rpy[2]);
}
/// Forms Ṙ, the ordinary derivative of the %RotationMatrix `R` with respect
/// to an independent variable `t` (`t` usually denotes time) and `R` is the
/// %RotationMatrix formed by `this` %RollPitchYaw. The roll-pitch-yaw angles
/// r, p, y are regarded as functions of `t` [i.e., r(t), p(t), y(t)].
/// @param[in] rpyDt Ordinary derivative of rpy with respect to an independent
/// variable `t` (`t` usually denotes time, but not necessarily).
/// @returns Ṙ, the ordinary derivative of `R` with respect to `t`, calculated
/// as Ṙ = ∂R/∂r * ṙ + ∂R/∂p * ṗ + ∂R/∂y * ẏ. In other words, the returned
/// (i, j) element is ∂Rij/∂r * ṙ + ∂Rij/∂p * ṗ + ∂Rij/∂y * ẏ.
Matrix3<T> CalcRotationMatrixDt(const Vector3<T>& rpyDt) const {
// For the rotation matrix R generated by `this` RollPitchYaw, calculate the
// partial derivatives of R with respect to roll `r`, pitch `p` and yaw `y`.
Matrix3<T> R_r, R_p, R_y; // ∂R/∂r, ∂R/∂p, ∂R/∂y
CalcRotationMatrixDrDpDy(&R_r, &R_p, &R_y);
// When rotation matrix R is regarded as an implicit function of t as
// R(r(t), p(t), y(t)), the ordinary derivative of R with respect to t is
// Ṙ = ∂R/∂r * ṙ + ∂R/∂p * ṗ + ∂R/∂y * ẏ
const T rDt = rpyDt(0), pDt = rpyDt(1), yDt = rpyDt(2);
return R_r * rDt + R_p * pDt + R_y * yDt;
}
/// Calculates angular velocity from `this` %RollPitchYaw whose roll-pitch-yaw
/// angles `[r; p; y]` relate the orientation of two generic frames A and D.
/// @param[in] rpyDt Time-derivative of `[r; p; y]`, i.e., `[ṙ; ṗ; ẏ]`.
/// @returns w_AD_A, frame D's angular velocity in frame A, expressed in A.
// TODO(Mitiguy) Improve speed -- last column of M is (0, 0, 1).
Vector3<T> CalcAngularVelocityInParentFromRpyDt(
const Vector3<T>& rpyDt) const {
// Get the 3x3 coefficient matrix M that contains the partial derivatives of
// w_AD_A with respect to ṙ, ṗ, ẏ. In other words, `w_AD_A = M * rpyDt`.
const Matrix3<T> M = CalcMatrixRelatingAngularVelocityInParentToRpyDt();
return M * rpyDt;
}
/// Calculates angular velocity from `this` %RollPitchYaw whose roll-pitch-yaw
/// angles `[r; p; y]` relate the orientation of two generic frames A and D.
/// @param[in] rpyDt Time-derivative of `[r; p; y]`, i.e., `[ṙ; ṗ; ẏ]`.
/// @returns w_AD_D, frame D's angular velocity in frame A, expressed in D.
// TODO(Mitiguy) Improve speed -- first column of M is (1, 0, 0).
Vector3<T> CalcAngularVelocityInChildFromRpyDt(
const Vector3<T>& rpyDt) const {
// Get the 3x3 coefficient matrix M that contains the partial derivatives of
// w_AD_D with respect to ṙ, ṗ, ẏ. In other words, `w_AD_D = M * rpyDt`.
const Matrix3<T> M = CalcMatrixRelatingAngularVelocityInChildToRpyDt();
return M * rpyDt;
}
/// Uses angular velocity to compute the 1ˢᵗ time-derivative of `this`
/// %RollPitchYaw whose angles `[r; p; y]` orient two generic frames A and D.
/// @param[in] w_AD_A, frame D's angular velocity in frame A, expressed in A.
/// @returns `[ṙ; ṗ; ẏ]`, the 1ˢᵗ time-derivative of `this` %RollPitchYaw.
/// @throws std::logic_error if `cos(p) ≈ 0` (`p` is near gimbal-lock).
/// @note This method has a divide-by-zero error (singularity) when the cosine
/// of the pitch angle `p` is zero [i.e., `cos(p) = 0`]. This problem (called
/// "gimbal lock") occurs when `p = n π + π / 2`, where n is any integer.
/// There are associated precision problems (inaccuracies) in the neighborhood
/// of these pitch angles, i.e., when `cos(p) ≈ 0`.
// TODO(Mitiguy) Improve speed -- last column of M is (0, 0, 1).
// TODO(Mitiguy) Improve accuracy when `cos(p) ≈ 0`.
Vector3<T> CalcRpyDtFromAngularVelocityInParent(
const Vector3<T>& w_AD_A) const {
// Get the 3x3 M matrix that contains the partial derivatives of `[ṙ, ṗ, ẏ]`
// with respect to `[wx; wy; wz]ₐ` (which is w_AD_A expressed in A).
// In other words, `rpyDt = M * w_AD_A`.
const Matrix3<T> M = CalcMatrixRelatingRpyDtToAngularVelocityInParent(
__func__, __FILE__, __LINE__);
return M * w_AD_A;
}
/// Uses angular acceleration to compute the 2ⁿᵈ time-derivative of `this`
/// %RollPitchYaw whose angles `[r; p; y]` orient two generic frames A and D.
/// @param[in] rpyDt time-derivative of `[r; p; y]`, i.e., `[ṙ; ṗ; ẏ]`.
/// @param[in] alpha_AD_A, frame D's angular acceleration in frame A,
/// expressed in frame A.
/// @returns `[r̈, p̈, ÿ]`, the 2ⁿᵈ time-derivative of `this` %RollPitchYaw.
/// @throws std::logic_error if `cos(p) ≈ 0` (`p` is near gimbal-lock).
/// @note This method has a divide-by-zero error (singularity) when the cosine
/// of the pitch angle `p` is zero [i.e., `cos(p) = 0`]. This problem (called
/// "gimbal lock") occurs when `p = n π + π / 2`, where n is any integer.
/// There are associated precision problems (inaccuracies) in the neighborhood
/// of these pitch angles, i.e., when `cos(p) ≈ 0`.
// TODO(Mitiguy) Improve accuracy when `cos(p) ≈ 0`.
// TODO(Mitiguy) Improve speed: The last column of M is (0, 0, 1), the last
// column of MDt is (0, 0, 1) and there are repeated sine/cosine calculations.
Vector3<T> CalcRpyDDtFromRpyDtAndAngularAccelInParent(
const Vector3<T>& rpyDt, const Vector3<T>& alpha_AD_A) const {
const Matrix3<T> Minv = CalcMatrixRelatingRpyDtToAngularVelocityInParent(
__func__, __FILE__, __LINE__);
const Matrix3<T> MDt =
CalcDtMatrixRelatingAngularVelocityInParentToRpyDt(rpyDt);
return Minv * (alpha_AD_A - MDt * rpyDt);
}
/// Uses angular acceleration to compute the 2ⁿᵈ time-derivative of `this`
/// %RollPitchYaw whose angles `[r; p; y]` orient two generic frames A and D.
/// @param[in] rpyDt time-derivative of `[r; p; y]`, i.e., `[ṙ; ṗ; ẏ]`.
/// @param[in] alpha_AD_D, frame D's angular acceleration in frame A,
/// expressed in frame D.
/// @returns `[r̈, p̈, ÿ]`, the 2ⁿᵈ time-derivative of `this` %RollPitchYaw.
/// @throws std::logic_error if `cos(p) ≈ 0` (`p` is near gimbal-lock).
/// @note This method has a divide-by-zero error (singularity) when the cosine
/// of the pitch angle `p` is zero [i.e., `cos(p) = 0`]. This problem (called
/// "gimbal lock") occurs when `p = n π + π / 2`, where n is any integer.
/// There are associated precision problems (inaccuracies) in the neighborhood
/// of these pitch angles, i.e., when `cos(p) ≈ 0`.
Vector3<T> CalcRpyDDtFromAngularAccelInChild(
const Vector3<T>& rpyDt, const Vector3<T>& alpha_AD_D) const {
const T& r = roll_angle();
const T& p = pitch_angle();
using std::sin;
using std::cos;
const T sr = sin(r), cr = cos(r);
const T sp = sin(p), cp = cos(p);
if (DoesCosPitchAngleViolateGimbalLockTolerance(cp)) {
ThrowPitchAngleViolatesGimbalLockTolerance(__func__, __FILE__, __LINE__,
p);
}
const T one_over_cp = T(1)/cp;
const T cr_over_cp = cr * one_over_cp;
const T sr_over_cp = sr * one_over_cp;
// clang-format on
Matrix3<T> M;
M << T(1), sr_over_cp * sp, cr_over_cp * sp,
T(0), cr, -sr,
T(0), sr_over_cp, cr_over_cp;
// clang-format off
// Remainder terms (terms not multiplying α).
const T tanp = sp * one_over_cp;
const T rDt = rpyDt(0), pDt = rpyDt(1), yDt = rpyDt(2);
const T pDt_yDt = pDt * yDt;
const T rDt_pDt = rDt * pDt;
const Vector3<T> remainder(tanp * rDt_pDt + one_over_cp * pDt_yDt,
-cp * rDt * yDt,
tanp * pDt_yDt + one_over_cp * rDt_pDt);
// Combine terms that contains alpha with remainder terms.
// TODO(Mitiguy) M * alpha_AD_D can be calculated faster since the first
// column of M is (1, 0, 0).
return M * alpha_AD_D + remainder;
}
private:
// Throws an exception if rpy is not a valid %RollPitchYaw.
// @param[in] rpy an allegedly valid rotation matrix.
// @note If the underlying scalar type T is non-numeric (symbolic), no
// validity check is made and no assertion is thrown.
template <typename S = T>
static typename std::enable_if_t<scalar_predicate<S>::is_bool>
ThrowIfNotValid(const Vector3<T>& rpy) {
if (!RollPitchYaw<T>::IsValid(rpy)) {
throw std::logic_error(
"Error: One (or more) of the roll-pitch-yaw angles is infinity or NaN.");
}
}
template <typename S = T>
static typename std::enable_if_t<!scalar_predicate<S>::is_bool>
ThrowIfNotValid(const Vector3<S>&) {}
// Throws an exception with a message that the pitch-angle `p` violates the
// internally-defined gimbal-lock tolerance, which occurs when `cos(p) ≈ 0`,
// which means `p ≈ (n*π + π/2)` where `n = 0, ±1, ±2, ...`.
// @param[in] function_name name of the calling function/method.
// @param[in] file_name name of the file with the calling function/method.
// @param[in] line_number the line number in file_name that made the call.
// @param[in] pitch_angle pitch angle `p` (in radians).
// @throws std::logic_error with a message that `p` is too near gimbal-lock.
static void ThrowPitchAngleViolatesGimbalLockTolerance(
const char* function_name, const char* file_name, const int line_number,
const T& pitch_angle);
// For the %RotationMatrix `R` generated by `this` %RollPitchYaw, this method
// calculates the partial derivatives of `R` with respect to roll, pitch, yaw.
// @param[out] R_r ∂R/∂r Partial derivative of `R` with respect to roll `r`.
// @param[out] R_p ∂R/∂p Partial derivative of `R` with respect to pitch `p`.
// @param[out] R_y ∂R/∂y Partial derivative of `R` with respect to yaw `y`.
void CalcRotationMatrixDrDpDy(Matrix3<T>* R_r, Matrix3<T>* R_p,
Matrix3<T>* R_y) const {
DRAKE_ASSERT(R_r != nullptr && R_p != nullptr && R_y != nullptr);
const T& r = roll_angle();
const T& p = pitch_angle();
const T& y = yaw_angle();
using std::sin;
using std::cos;
const T c0 = cos(r), c1 = cos(p), c2 = cos(y);
const T s0 = sin(r), s1 = sin(p), s2 = sin(y);
const T c2_s1 = c2 * s1, s2_s1 = s2 * s1, s2_s0 = s2 * s0, s2_c0 = s2 * c0;
const T c2_c1 = c2 * c1, s2_c1 = s2 * c1, c2_s0 = c2 * s0, c2_c0 = c2 * c0;
// clang-format on
*R_r << 0, c2_s1 * c0 + s2_s0, -c2_s1 * s0 + s2_c0,
0, s2_s1 * c0 - c2_s0, -s2_s1 * s0 - c2_c0,
0, c1 * c0, -c1 * s0;
*R_p << -c2_s1, c2_c1 * s0, c2_c1 * c0,
-s2_s1, s2_c1 * s0, s2_c1 * c0,
-c1, -s1 * s0, -s1 * c0;
*R_y << -s2_c1, -s2_s1 * s0 - c2_c0, -s2_s1 * c0 + c2_s0,
c2_c1, c2_s1 * s0 - s2_c0, c2_s1 * c0 + s2_s0,
0, 0, 0;
// clang-format off
}
// For `this` %RollPitchYaw with roll-pitch-yaw angles `[r; p; y]` which
// relate the orientation of two generic frames A and D, returns the 3x3
// coefficient matrix M that contains the partial derivatives of `w_AD_A`
// (D's angular velocity in A, expressed in A) with respect to ṙ, ṗ, ẏ.
// In other words, `w_AD_A = M * rpyDt` where `rpyDt` is `[ṙ; ṗ; ẏ]`.
const Matrix3<T> CalcMatrixRelatingAngularVelocityInParentToRpyDt() const {
using std::cos;
using std::sin;
const T& p = pitch_angle();
const T& y = yaw_angle();
const T sp = sin(p), cp = cos(p);
const T sy = sin(y), cy = cos(y);
Matrix3<T> M;
// clang-format on
M << cp * cy, -sy, T(0),
cp * sy, cy, T(0),
-sp, T(0), T(1);
// clang-format off
return M;
}
// Returns the time-derivative of the 3x3 matrix returned by the `this`
// %RollPitchYaw method MatrixRelatingAngularVelocityAToRpyDt().
// @param[in] rpyDt Time-derivative of `[r; p; y]`, i.e., `[ṙ; ṗ; ẏ]`.
// @see MatrixRelatingAngularVelocityAToRpyDt()
const Matrix3<T> CalcDtMatrixRelatingAngularVelocityInParentToRpyDt(
const Vector3<T>& rpyDt) const {
using std::cos;
using std::sin;
const T& p = pitch_angle();
const T& y = yaw_angle();
const T sp = sin(p), cp = cos(p);
const T sy = sin(y), cy = cos(y);
const T pDt = rpyDt(1);
const T yDt = rpyDt(2);
const T sp_pDt = sp * pDt;
const T cp_yDt = cp * yDt;
Matrix3<T> M;
// clang-format on
M << -cy * sp_pDt - sy * cp_yDt, -cy * yDt, T(0),
-sy * sp_pDt + cy * cp_yDt, -sy * yDt, T(0),
-cp * pDt, T(0), T(0);
// clang-format off
return M;
}
// For `this` %RollPitchYaw with roll-pitch-yaw angles `[r; p; y]` which
// relate the orientation of two generic frames A and D, returns the 3x3
// coefficient matrix M that contains the partial derivatives of `w_AD_D`
// (D's angular velocity in A, expressed in D) with respect to ṙ, ṗ, ẏ.
// In other words, `w_AD_D = M * rpyDt` where `rpyDt` is `[ṙ; ṗ; ẏ]`.
const Matrix3<T> CalcMatrixRelatingAngularVelocityInChildToRpyDt() const {
using std::cos;
using std::sin;
const T& r = roll_angle();
const T& p = pitch_angle();
const T sr = sin(r), cr = cos(r);
const T sp = sin(p), cp = cos(p);
Matrix3<T> M;
// clang-format on
M << T(1), T(0), -sp,
T(0), cr, sr * cp,
T(0), -sr, cr * cp;
// clang-format off
return M;
}
// For `this` %RollPitchYaw with roll-pitch-yaw angles `[r; p; y]` which
// relate the orientation of two generic frames A and D, returns the 3x3
// matrix M that contains the partial derivatives of [ṙ, ṗ, ẏ] with respect to
// `[wx; wy; wz]ₐ` (which is w_AD_A expressed in A).
// In other words, `rpyDt = M * w_AD_A`.
// @param[in] function_name name of the calling function/method.
// @param[in] file_name name of the file with the calling function/method.
// @param[in] line_number the line number in file_name that made the call.
// @throws std::logic_error if `cos(p) ≈ 0` (`p` is near gimbal-lock).
// @note This method has a divide-by-zero error (singularity) when the cosine
// of the pitch angle `p` is zero [i.e., `cos(p) = 0`]. This problem (called
// "gimbal lock") occurs when `p = n π + π / 2`, where n is any integer.
// There are associated precision problems (inaccuracies) in the neighborhood
// of these pitch angles, i.e., when `cos(p) ≈ 0`.
// TODO(Mitiguy) Improve accuracy when `cos(p) ≈ 0`.
const Matrix3<T> CalcMatrixRelatingRpyDtToAngularVelocityInParent(
const char* function_name, const char* file_name, int line_number) const {
using std::cos;
using std::sin;
const T& p = pitch_angle();
const T& y = yaw_angle();
const T sp = sin(p), cp = cos(p);
if (DoesCosPitchAngleViolateGimbalLockTolerance(cp)) {
ThrowPitchAngleViolatesGimbalLockTolerance(function_name, file_name,
line_number, p);
}
const T one_over_cp = T(1)/cp;
const T sy = sin(y), cy = cos(y);
const T cy_over_cp = cy * one_over_cp;
const T sy_over_cp = sy * one_over_cp;
Matrix3<T> M;
// clang-format on
M << cy_over_cp, sy_over_cp, T(0),
-sy, cy, T(0),
cy_over_cp * sp, sy_over_cp * sp, T(1);
// clang-format off
return M;
}
// Sets `this` %RollPitchYaw from a Vector3.
// @param[in] rpy allegedly valid roll-pitch-yaw angles.
// @throws std::logic_error in debug builds if rpy fails IsValid(rpy).
RollPitchYaw<T>& SetOrThrowIfNotValidInDebugBuild(const Vector3<T>& rpy) {
DRAKE_ASSERT_VOID(ThrowIfNotValid(rpy));
roll_pitch_yaw_ = rpy;
return *this;
}
// Stores the underlying roll-pitch-yaw angles.
// There is no default initialization needed.
Vector3<T> roll_pitch_yaw_;
// Internally-defined value for the allowable proximity of the cosine of the
// pitch-angle `p` to gimbal-lock [the proximity of `cos(p)` to 0].
// @note For small values (<= 0.1), this value approximates the allowable
// proximity of the pitch-angle (in radians) to gimbal-lock. Example: A value
// of 0.01 corresponds to `p` within ≈ 0.01 radians (≈ 0.57°) of gimbal-lock,
// i.e., `p` is within 0.01 radians of `(n*π + π/2)`, `n = 0, ±1, ±2, ...`.
// A value 0.008 corresponds to `p` ≈ 0.008 radians (≈ 0.46°) of gimbal-lock.
// @note The conversion from angular velocity to rpyDt (the time-derivative of
// %RollPitchYaw) has a calculation that divides by `cos(p)` (the cosine of
// the pitch angle). This results in values of rpyDt that scale with angular
// velocity multiplied by `1/cos(p)`, which can cause problems with numerical
// precision and numerical integration.
// @note There is a numerical test (in roll_pitch_yaw_test.cc) that converts
// an angular velocity w(1, 1, 1) to rpyDt (time-derivative of roll-pitch-yaw)
// and then back to angular velocity. This test revealed an imprecision in
// the back-and-forth calculation near gimbal-lock by calculating max_error
// (how much the angular velocity changed in the back-and-forth calculation).
// The table below shows various values of kGimbalLockToleranceCosPitchAngle,
// it associated proximity to gimbal-lock (in degrees) and the associated
// max_error (in terms of machine kEpsilon = 1/2^52 ≈ 2.22E-16).
// ---------------------------------------------------------------------------
// kGimbalLockToleranceCosPitchAngle | max_error
// ---------------------------------------------------------------------------
// 0.001 ≈ 0.06° | (2^10 = 1024) * kEpsilon ≈ 2.274E-13
// 0.002 ≈ 0.11° | (2^9 = 512) * kEpsilon ≈ 1.137E-13
// 0.004 ≈ 0.23° | (2^8 = 256) * kEpsilon ≈ 5.684E-14
// 0.008 ≈ 0.46° | (2^7 = 128) * kEpsilon ≈ 2.842E-14
// 0.016 ≈ 0.92° | (2^6 = 64) * kEpsilon ≈ 1.421E-14
// 0.032 ≈ 1.83° | (2^5 = 32) * kEpsilon ≈ 7.105E-15
// ---------------------------------------------------------------------------
// Hence if kGimbalLockToleranceCosPitchAngle = 0.008 and the pitch angle is
// in the proximity of ≈ 0.46° of gimbal lock, there may be inaccuracies in
// 7 of the 52 bits in max_error's mantissa, which we deem acceptable.
static constexpr double kGimbalLockToleranceCosPitchAngle = 0.008;
};
// Uses a quaternion and its associated rotation matrix `R` to accurately
// construct roll-pitch-yaw angles (i.e., SpaceXYZ Euler angles) -- even
// when the pitch angle is within 1E-6 of π/2 or -π/2.
// @param[in] quaternion unit quaternion with elements `[e0, e1, e2, e3]`.
// @param[in] R The %RotationMatrix corresponding to `quaternion`.
// @return %RollPitchYaw containing angles `[r, p, y]` with range
// `-π <= r <= π`, `-π/2 <= p <= π/2, `-π <= y <= π`.
// @note The caller of this function is responsible for ensure
template <typename T>
Vector3<T> CalcRollPitchYawFromQuaternionAndRotationMatrix(
const Eigen::Quaternion<T>& quaternion, const Matrix3<T>& R);
/// Abbreviation (alias/typedef) for a RollPitchYaw double scalar type.
/// @relates RollPitchYaw
using RollPitchYawd = RollPitchYaw<double>;
} // namespace math
} // namespace drake