pose_instability_detector#
The pose_instability_detector
is a node designed to monitor the stability of /localization/kinematic_state
, which is an output topic of the Extended Kalman Filter (EKF).
This node triggers periodic timer callbacks to compare two poses:
- The pose calculated by dead reckoning starting from the pose of
/localization/kinematic_state
obtainedtimer_period
seconds ago. - The latest pose from
/localization/kinematic_state
.
The results of this comparison are then output to the /diagnostics
topic.
If this node outputs WARN messages to /diagnostics
, it means that the EKF output is significantly different from the integrated twist values.
In other words, WARN outputs indicate that the vehicle has moved to a place outside the expected range based on the twist values.
This discrepancy suggests that there may be an issue with either the estimated pose or the input twist.
The following diagram provides an overview of how the procedure looks like:
Dead reckoning algorithm#
Dead reckoning is a method of estimating the position of a vehicle based on its previous position and velocity. The procedure for dead reckoning is as follows:
- Capture the necessary twist values from the
/input/twist
topic. - Integrate the twist values to calculate the pose transition.
- Apply the pose transition to the previous pose to obtain the current pose.
Collecting twist values#
The pose_instability_detector
node collects the twist values from the ~/input/twist
topic to perform dead reckoning.
Ideally, pose_instability_detector
needs the twist values between the previous pose and the current pose.
Therefore, pose_instability_detector
snips the twist buffer and apply interpolations and extrapolations to obtain the twist values at the desired time.
Linear transition and angular transition#
After the twist values are collected, the node calculates the linear transition and angular transition based on the twist values and add them to the previous pose.
Threshold definition#
The pose_instability_detector
node compares the pose calculated by dead reckoning with the latest pose from the EKF output.
These two pose are ideally the same, but in reality, they are not due to the error in the twist values the pose observation.
If these two poses are significantly different so that the absolute value exceeds the threshold, the node outputs a WARN message to the /diagnostics
topic.
There are six thresholds (x, y, z, roll, pitch, and yaw) to determine whether the poses are significantly different, and these thresholds are determined by the following subsections.
diff_position_x
#
This threshold examines the difference in the longitudinal axis between the two poses, and check whether the vehicle goes beyond the expected error. This threshold is a sum of "maximum longitudinal error due to velocity scale factor error" and "pose estimation error tolerance".
Symbol | Description | Unit |
---|---|---|
\(\tau_x\) | Threshold for the difference in the longitudinal axis | \(m\) |
\(v_{\rm max}\) | Maximum velocity | \(m/s\) |
\(\beta_v\) | Scale factor tolerance for the maximum velocity | \(\%\) |
\(\Delta t\) | Time interval | \(s\) |
\(\epsilon_x\) | Pose estimator (e. g. ndt_scan_matcher) error tolerance in the longitudinal axis | \(m\) |
diff_position_y
and diff_position_z
#
These thresholds examine the difference in the lateral and vertical axes between the two poses, and check whether the vehicle goes beyond the expected error.
The pose_instability_detector
calculates the possible range where the vehicle goes, and get the maximum difference between the nominal dead reckoning pose and the maximum limit pose.
Addition to this, the pose_instability_detector
node considers the pose estimation error tolerance to determine the threshold.
Symbol | Description | Unit |
---|---|---|
\(\tau_y\) | Threshold for the difference in the lateral axis | \(m\) |
\(l\) | Maximum lateral distance described in the image above (See the appendix how this is calculated) | \(m\) |
\(\epsilon_y\) | Pose estimator (e. g. ndt_scan_matcher) error tolerance in the lateral axis | \(m\) |
Note that pose_instability_detector
sets the threshold for the vertical axis as the same as the lateral axis. Only the pose estimator error tolerance is different.
diff_angle_x
, diff_angle_y
, and diff_angle_z
#
These thresholds examine the difference in the roll, pitch, and yaw angles between the two poses. This threshold is a sum of "maximum angular error due to velocity scale factor error and bias error" and "pose estimation error tolerance".
Symbol | Description | Unit |
---|---|---|
\(\tau_\phi\) | Threshold for the difference in the roll angle | \({\rm rad}\) |
\(\tau_\theta\) | Threshold for the difference in the pitch angle | \({\rm rad}\) |
\(\tau_\psi\) | Threshold for the difference in the yaw angle | \({\rm rad}\) |
\(\omega_{\rm max}\) | Maximum angular velocity | \({\rm rad}/s\) |
\(\beta_\omega\) | Scale factor tolerance for the maximum angular velocity | \(\%\) |
\(b\) | Bias tolerance of the angular velocity | \({\rm rad}/s\) |
\(\Delta t\) | Time interval | \(s\) |
\(\epsilon_\psi\) | Pose estimator (e. g. ndt_scan_matcher) error tolerance in the yaw angle | \({\rm rad}\) |
Parameters#
Name | Type | Description | Default | Range |
---|---|---|---|---|
timer_period | float | The period of timer_callback (sec). | 0.5 | >0 |
heading_velocity_maximum | float | The maximum of heading velocity (m/s). | 16.667 | ≥0.0 |
heading_velocity_scale_factor_tolerance | float | The tolerance of heading velocity scale factor (%). | 3 | ≥0.0 |
angular_velocity_maximum | float | The maximum of angular velocity (rad/s). | 0.523 | ≥0.0 |
angular_velocity_scale_factor_tolerance | float | The tolerance of angular velocity scale factor (%). | 0.2 | ≥0.0 |
angular_velocity_bias_tolerance | float | The tolerance of angular velocity bias (rad/s). | 0.00698 | ≥0.0 |
pose_estimator_longitudinal_tolerance | float | The tolerance of longitudinal position of pose estimator (m). | 0.11 | ≥0.0 |
pose_estimator_lateral_tolerance | float | The tolerance of lateral position of pose estimator (m). | 0.11 | ≥0.0 |
pose_estimator_vertical_tolerance | float | The tolerance of vertical position of pose estimator (m). | 0.11 | ≥0.0 |
pose_estimator_angular_tolerance | float | The tolerance of roll angle of pose estimator (rad). | 0.0175 | ≥0.0 |
Input#
Name | Type | Description |
---|---|---|
~/input/odometry |
nav_msgs::msg::Odometry | Pose estimated by EKF |
~/input/twist |
geometry_msgs::msg::TwistWithCovarianceStamped | Twist |
Output#
Name | Type | Description |
---|---|---|
~/debug/diff_pose |
geometry_msgs::msg::PoseStamped | diff_pose |
/diagnostics |
diagnostic_msgs::msg::DiagnosticArray | Diagnostics |
Appendix#
On calculating the maximum lateral distance \(l\), the pose_instability_detector
node will estimate the following poses.
Pose | heading velocity \(v\) | angular velocity \(\omega\) |
---|---|---|
Nominal dead reckoning pose | \(v_{\rm max}\) | \(\omega_{\rm max}\) |
Dead reckoning pose of corner A | \(\left(1+\frac{\beta_v}{100}\right) v_{\rm max}\) | \(\left(1+\frac{\beta_\omega}{100}\right) \omega_{\rm max} + b\) |
Dead reckoning pose of corner B | \(\left(1-\frac{\beta_v}{100}\right) v_{\rm max}\) | \(\left(1+\frac{\beta_\omega}{100}\right) \omega_{\rm max} + b\) |
Dead reckoning pose of corner C | \(\left(1-\frac{\beta_v}{100}\right) v_{\rm max}\) | \(\left(1-\frac{\beta_\omega}{100}\right) \omega_{\rm max} - b\) |
Dead reckoning pose of corner D | \(\left(1+\frac{\beta_v}{100}\right) v_{\rm max}\) | \(\left(1-\frac{\beta_\omega}{100}\right) \omega_{\rm max} - b\) |
Given a heading velocity \(v\) and \(\omega\), the 2D theoretical variation seen from the previous pose is calculated as follows:
We calculate this variation for each corner and get the maximum value of the lateral distance \(l\) by comparing the distance between the nominal dead reckoning pose and the corner poses.