mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-05 07:28:29 -04:00
AP_NavEKF2: Report position jumps due to lane switches
Also moves code required to update reset data due to lane switches into separate functionscto improve readability.
This commit is contained in:
parent
337461c16c
commit
ab55991b33
@ -657,35 +657,16 @@ void NavEKF2::UpdateFilter(void)
|
|||||||
// If the current core selected has a bad fault score or is unhealthy, switch to a healthy core with the lowest fault score
|
// If the current core selected has a bad fault score or is unhealthy, switch to a healthy core with the lowest fault score
|
||||||
if (core[primary].faultScore() > 0.0f || !core[primary].healthy()) {
|
if (core[primary].faultScore() > 0.0f || !core[primary].healthy()) {
|
||||||
float score = 1e9f;
|
float score = 1e9f;
|
||||||
bool changed = false;
|
bool has_switched = false; // true if a switch has occurred this frame
|
||||||
for (uint8_t i=0; i<num_cores; i++) {
|
for (uint8_t i=0; i<num_cores; i++) {
|
||||||
if (core[i].healthy()) {
|
if (core[i].healthy()) {
|
||||||
float tempScore = core[i].faultScore();
|
float tempScore = core[i].faultScore();
|
||||||
if (tempScore < score) {
|
if (tempScore < score) {
|
||||||
Vector3f eulers_old_primary, eulers_new_primary;
|
// update the yaw and position reset data to capture changes due to the lane switch
|
||||||
float old_yaw_delta;
|
updateLaneSwitchYawResetData(has_switched, i, primary);
|
||||||
|
updateLaneSwitchPosResetData(has_switched, i, primary);
|
||||||
|
|
||||||
// If core yaw reset data has been consumed reset delta to zero
|
has_switched = true;
|
||||||
if (!yaw_reset_data.core_changed) {
|
|
||||||
yaw_reset_data.core_delta = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If current primary has reset yaw after controller got it, add it to the delta
|
|
||||||
// Prevent adding the delta if we have already changed primary in this filter update
|
|
||||||
if (!changed && core[primary].getLastYawResetAngle(old_yaw_delta) > yaw_reset_data.last_function_call) {
|
|
||||||
yaw_reset_data.core_delta += old_yaw_delta;
|
|
||||||
}
|
|
||||||
|
|
||||||
core[primary].getEulerAngles(eulers_old_primary);
|
|
||||||
core[i].getEulerAngles(eulers_new_primary);
|
|
||||||
|
|
||||||
// Record the yaw delta between current core and new primary core and the timestamp of the core change
|
|
||||||
// Add current delta in case it hasn't been consumed yet
|
|
||||||
yaw_reset_data.core_delta = wrap_PI(eulers_new_primary.z - eulers_old_primary.z + yaw_reset_data.core_delta);
|
|
||||||
yaw_reset_data.last_primary_change = imuSampleTime_us / 1000;
|
|
||||||
yaw_reset_data.core_changed = true;
|
|
||||||
|
|
||||||
changed = true;
|
|
||||||
primary = i;
|
primary = i;
|
||||||
score = tempScore;
|
score = tempScore;
|
||||||
}
|
}
|
||||||
@ -1210,12 +1191,33 @@ uint32_t NavEKF2::getLastYawResetAngle(float &yawAngDelta)
|
|||||||
|
|
||||||
// return the amount of NE position change due to the last position reset in metres
|
// return the amount of NE position change due to the last position reset in metres
|
||||||
// returns the time of the last reset or 0 if no reset has ever occurred
|
// returns the time of the last reset or 0 if no reset has ever occurred
|
||||||
uint32_t NavEKF2::getLastPosNorthEastReset(Vector2f &pos) const
|
uint32_t NavEKF2::getLastPosNorthEastReset(Vector2f &posDelta)
|
||||||
{
|
{
|
||||||
if (!core) {
|
if (!core) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
return core[primary].getLastPosNorthEastReset(pos);
|
|
||||||
|
// Record last time controller got the position reset
|
||||||
|
pos_reset_data.last_function_call = imuSampleTime_us / 1000;
|
||||||
|
posDelta.zero();
|
||||||
|
uint32_t lastPosReset_ms = 0;
|
||||||
|
Vector2f tempPosDelta;
|
||||||
|
uint32_t lastCorePosReset_ms = core[primary].getLastPosNorthEastReset(tempPosDelta);
|
||||||
|
|
||||||
|
// If core has changed (and data not consumed yet) or if the core change was the last position reset, return its data
|
||||||
|
if (pos_reset_data.core_changed || lastCorePosReset_ms <= pos_reset_data.last_primary_change) {
|
||||||
|
posDelta = pos_reset_data.core_delta;
|
||||||
|
lastPosReset_ms = pos_reset_data.last_primary_change;
|
||||||
|
pos_reset_data.core_changed = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If current core position reset event was the last one, add it to the delta
|
||||||
|
if (lastCorePosReset_ms > lastPosReset_ms) {
|
||||||
|
posDelta = posDelta + tempPosDelta;
|
||||||
|
lastPosReset_ms = lastCorePosReset_ms;
|
||||||
|
}
|
||||||
|
|
||||||
|
return lastPosReset_ms;
|
||||||
}
|
}
|
||||||
|
|
||||||
// return the amount of NE velocity change due to the last velocity reset in metres/sec
|
// return the amount of NE velocity change due to the last velocity reset in metres/sec
|
||||||
@ -1237,4 +1239,56 @@ const char *NavEKF2::prearm_failure_reason(void) const
|
|||||||
return core[primary].prearm_failure_reason();
|
return core[primary].prearm_failure_reason();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// update the yaw reset data to capture changes due to a lane switch
|
||||||
|
void NavEKF2::updateLaneSwitchYawResetData(bool has_switched, uint8_t new_primary, uint8_t old_primary)
|
||||||
|
{
|
||||||
|
Vector3f eulers_old_primary, eulers_new_primary;
|
||||||
|
float old_yaw_delta;
|
||||||
|
|
||||||
|
// If core yaw reset data has been consumed reset delta to zero
|
||||||
|
if (!yaw_reset_data.core_changed) {
|
||||||
|
yaw_reset_data.core_delta = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If current primary has reset yaw after controller got it, add it to the delta
|
||||||
|
// Prevent adding the delta if we have already changed primary in this filter update
|
||||||
|
if (!has_switched && core[old_primary].getLastYawResetAngle(old_yaw_delta) > yaw_reset_data.last_function_call) {
|
||||||
|
yaw_reset_data.core_delta += old_yaw_delta;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Record the yaw delta between current core and new primary core and the timestamp of the core change
|
||||||
|
// Add current delta in case it hasn't been consumed yet
|
||||||
|
core[old_primary].getEulerAngles(eulers_old_primary);
|
||||||
|
core[new_primary].getEulerAngles(eulers_new_primary);
|
||||||
|
yaw_reset_data.core_delta = wrap_PI(eulers_new_primary.z - eulers_old_primary.z + yaw_reset_data.core_delta);
|
||||||
|
yaw_reset_data.last_primary_change = imuSampleTime_us / 1000;
|
||||||
|
yaw_reset_data.core_changed = true;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// update the position reset data to capture changes due to a lane switch
|
||||||
|
void NavEKF2::updateLaneSwitchPosResetData(bool has_switched, uint8_t new_primary, uint8_t old_primary)
|
||||||
|
{
|
||||||
|
Vector2f pos_old_primary, pos_new_primary, old_pos_delta;
|
||||||
|
|
||||||
|
// If core position reset data has been consumed reset delta to zero
|
||||||
|
if (!pos_reset_data.core_changed) {
|
||||||
|
pos_reset_data.core_delta.zero();
|
||||||
|
}
|
||||||
|
|
||||||
|
// If current primary has reset position after controller got it, add it to the delta
|
||||||
|
// Prevent adding the delta if we have already changed primary in this filter update
|
||||||
|
if (!has_switched && core[old_primary].getLastPosNorthEastReset(old_pos_delta) > pos_reset_data.last_function_call) {
|
||||||
|
pos_reset_data.core_delta += old_pos_delta;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Record the position delta between current core and new primary core and the timestamp of the core change
|
||||||
|
// Add current delta in case it hasn't been consumed yet
|
||||||
|
core[old_primary].getLastPosNorthEastReset(pos_old_primary);
|
||||||
|
core[new_primary].getLastPosNorthEastReset(pos_new_primary);
|
||||||
|
pos_reset_data.core_delta = pos_new_primary - pos_old_primary + pos_reset_data.core_delta;
|
||||||
|
pos_reset_data.last_primary_change = imuSampleTime_us / 1000;
|
||||||
|
pos_reset_data.core_changed = true;
|
||||||
|
}
|
||||||
|
|
||||||
#endif //HAL_CPU_CLASS
|
#endif //HAL_CPU_CLASS
|
||||||
|
@ -274,7 +274,7 @@ public:
|
|||||||
|
|
||||||
// return the amount of NE position change due to the last position reset in metres
|
// return the amount of NE position change due to the last position reset in metres
|
||||||
// returns the time of the last reset or 0 if no reset has ever occurred
|
// returns the time of the last reset or 0 if no reset has ever occurred
|
||||||
uint32_t getLastPosNorthEastReset(Vector2f &pos) const;
|
uint32_t getLastPosNorthEastReset(Vector2f &posDelta);
|
||||||
|
|
||||||
// return the amount of NE velocity change due to the last velocity reset in metres/sec
|
// return the amount of NE velocity change due to the last velocity reset in metres/sec
|
||||||
// returns the time of the last reset or 0 if no reset has ever occurred
|
// returns the time of the last reset or 0 if no reset has ever occurred
|
||||||
@ -385,6 +385,25 @@ private:
|
|||||||
uint32_t last_function_call; // last time getLastYawYawResetAngle was called
|
uint32_t last_function_call; // last time getLastYawYawResetAngle was called
|
||||||
bool core_changed; // true when a core change happened and hasn't been consumed, false otherwise
|
bool core_changed; // true when a core change happened and hasn't been consumed, false otherwise
|
||||||
uint32_t last_primary_change; // last time a primary has changed
|
uint32_t last_primary_change; // last time a primary has changed
|
||||||
float core_delta; // the ammount of yaw change between cores when a change happened
|
float core_delta; // the amount of yaw change between cores when a change happened
|
||||||
} yaw_reset_data;
|
} yaw_reset_data;
|
||||||
|
|
||||||
|
struct {
|
||||||
|
uint32_t last_function_call; // last time getLastPosNorthEastReset was called
|
||||||
|
bool core_changed; // true when a core change happened and hasn't been consumed, false otherwise
|
||||||
|
uint32_t last_primary_change; // last time a primary has changed
|
||||||
|
Vector2f core_delta; // the amount of NE position change between cores when a change happened
|
||||||
|
} pos_reset_data;
|
||||||
|
|
||||||
|
// update the yaw reset data to capture changes due to a lane switch
|
||||||
|
// has_switched - true if the primary instance has already been changed during this filter update cycle
|
||||||
|
// new_primary - index of the ekf instance that we are about to switch to as the primary
|
||||||
|
// old_primary - index of the ekf instance that we are currently using as the primary
|
||||||
|
void updateLaneSwitchYawResetData(bool has_switched, uint8_t new_primary, uint8_t old_primary);
|
||||||
|
|
||||||
|
// update the position reset data to capture changes due to a lane switch
|
||||||
|
// has_switched - true if the primary instance has already been changed during this filter update cycle
|
||||||
|
// new_primary - index of the ekf instance that we are about to switch to as the primary
|
||||||
|
// old_primary - index of the ekf instance that we are currently using as the primary
|
||||||
|
void updateLaneSwitchPosResetData(bool has_switched, uint8_t new_primary, uint8_t old_primary);
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user