AP_GPS: added get_undulation support

This commit is contained in:
Joshua Henderson 2022-07-28 04:42:01 -04:00 committed by Andrew Tridgell
parent 7173f1abcd
commit 5f687e15ed
9 changed files with 67 additions and 6 deletions

View File

@ -1319,6 +1319,11 @@ void AP_GPS::send_mavlink_gps_raw(mavlink_channel_t chan)
float hacc = 0.0f; float hacc = 0.0f;
float vacc = 0.0f; float vacc = 0.0f;
float sacc = 0.0f; float sacc = 0.0f;
float undulation = 0.0;
int32_t height_elipsoid_mm = 0;
if (get_undulation(0, undulation)) {
height_elipsoid_mm = loc.alt*10 - undulation*1000;
}
horizontal_accuracy(0, hacc); horizontal_accuracy(0, hacc);
vertical_accuracy(0, vacc); vertical_accuracy(0, vacc);
speed_accuracy(0, sacc); speed_accuracy(0, sacc);
@ -1334,7 +1339,7 @@ void AP_GPS::send_mavlink_gps_raw(mavlink_channel_t chan)
ground_speed(0)*100, // cm/s ground_speed(0)*100, // cm/s
ground_course(0)*100, // 1/100 degrees, ground_course(0)*100, // 1/100 degrees,
num_sats(0), num_sats(0),
0, // TODO: Elipsoid height in mm height_elipsoid_mm, // Elipsoid height in mm
hacc * 1000, // one-sigma standard deviation in mm hacc * 1000, // one-sigma standard deviation in mm
vacc * 1000, // one-sigma standard deviation in mm vacc * 1000, // one-sigma standard deviation in mm
sacc * 1000, // one-sigma standard deviation in mm/s sacc * 1000, // one-sigma standard deviation in mm/s
@ -1354,6 +1359,11 @@ void AP_GPS::send_mavlink_gps2_raw(mavlink_channel_t chan)
float hacc = 0.0f; float hacc = 0.0f;
float vacc = 0.0f; float vacc = 0.0f;
float sacc = 0.0f; float sacc = 0.0f;
float undulation = 0.0;
float height_elipsoid_mm = 0;
if (get_undulation(0, undulation)) {
height_elipsoid_mm = loc.alt*10 - undulation*1000;
}
horizontal_accuracy(1, hacc); horizontal_accuracy(1, hacc);
vertical_accuracy(1, vacc); vertical_accuracy(1, vacc);
speed_accuracy(1, sacc); speed_accuracy(1, sacc);
@ -1372,7 +1382,7 @@ void AP_GPS::send_mavlink_gps2_raw(mavlink_channel_t chan)
state[1].rtk_num_sats, state[1].rtk_num_sats,
state[1].rtk_age_ms, state[1].rtk_age_ms,
gps_yaw_cdeg(1), gps_yaw_cdeg(1),
0, // TODO: Elipsoid height in mm height_elipsoid_mm, // Elipsoid height in mm
hacc * 1000, // one-sigma standard deviation in mm hacc * 1000, // one-sigma standard deviation in mm
vacc * 1000, // one-sigma standard deviation in mm vacc * 1000, // one-sigma standard deviation in mm
sacc * 1000, // one-sigma standard deviation in mm/s sacc * 1000, // one-sigma standard deviation in mm/s
@ -1808,6 +1818,16 @@ void AP_GPS::calc_blended_state(void)
timing[GPS_BLENDED_INSTANCE].last_fix_time_ms = 0; timing[GPS_BLENDED_INSTANCE].last_fix_time_ms = 0;
timing[GPS_BLENDED_INSTANCE].last_message_time_ms = 0; timing[GPS_BLENDED_INSTANCE].last_message_time_ms = 0;
if (state[0].have_undulation) {
state[GPS_BLENDED_INSTANCE].have_undulation = true;
state[GPS_BLENDED_INSTANCE].undulation = state[0].undulation;
} else if (state[1].have_undulation) {
state[GPS_BLENDED_INSTANCE].have_undulation = true;
state[GPS_BLENDED_INSTANCE].undulation = state[1].undulation;
} else {
state[GPS_BLENDED_INSTANCE].have_undulation = false;
}
// combine the states into a blended solution // combine the states into a blended solution
for (uint8_t i=0; i<GPS_MAX_RECEIVERS; i++) { for (uint8_t i=0; i<GPS_MAX_RECEIVERS; i++) {
// use the highest status // use the highest status
@ -2053,6 +2073,17 @@ bool AP_GPS::get_error_codes(uint8_t instance, uint32_t &error_codes) const
return drivers[instance]->get_error_codes(error_codes); return drivers[instance]->get_error_codes(error_codes);
} }
// get the difference between WGS84 and AMSL. A positive value means
// the AMSL height is higher than WGS84 ellipsoid height
bool AP_GPS::get_undulation(uint8_t instance, float &undulation) const
{
if (!state[instance].have_undulation) {
return false;
}
undulation = state[instance].undulation;
return true;
}
// Logging support: // Logging support:
// Write an GPS packet // Write an GPS packet
void AP_GPS::Write_GPS(uint8_t i) void AP_GPS::Write_GPS(uint8_t i)
@ -2086,9 +2117,11 @@ void AP_GPS::Write_GPS(uint8_t i)
/* write auxiliary accuracy information as well */ /* write auxiliary accuracy information as well */
float hacc = 0, vacc = 0, sacc = 0; float hacc = 0, vacc = 0, sacc = 0;
float undulation = 0;
horizontal_accuracy(i, hacc); horizontal_accuracy(i, hacc);
vertical_accuracy(i, vacc); vertical_accuracy(i, vacc);
speed_accuracy(i, sacc); speed_accuracy(i, sacc);
get_undulation(i, undulation);
struct log_GPA pkt2{ struct log_GPA pkt2{
LOG_PACKET_HEADER_INIT(LOG_GPA_MSG), LOG_PACKET_HEADER_INIT(LOG_GPA_MSG),
time_us : time_us, time_us : time_us,
@ -2100,7 +2133,8 @@ void AP_GPS::Write_GPS(uint8_t i)
yaw_accuracy : yaw_accuracy_deg, yaw_accuracy : yaw_accuracy_deg,
have_vv : (uint8_t)have_vertical_velocity(i), have_vv : (uint8_t)have_vertical_velocity(i),
sample_ms : last_message_time_ms(i), sample_ms : last_message_time_ms(i),
delta_ms : last_message_delta_time_ms(i) delta_ms : last_message_delta_time_ms(i),
undulation : undulation,
}; };
AP::logger().WriteBlock(&pkt2, sizeof(pkt2)); AP::logger().WriteBlock(&pkt2, sizeof(pkt2));
} }

View File

@ -193,6 +193,8 @@ public:
bool have_vertical_accuracy; ///< does GPS give vertical position accuracy? Set to true only once available. bool have_vertical_accuracy; ///< does GPS give vertical position accuracy? Set to true only once available.
bool have_gps_yaw; ///< does GPS give yaw? Set to true only once available. bool have_gps_yaw; ///< does GPS give yaw? Set to true only once available.
bool have_gps_yaw_accuracy; ///< does the GPS give a heading accuracy estimate? Set to true only once available bool have_gps_yaw_accuracy; ///< does the GPS give a heading accuracy estimate? Set to true only once available
float undulation; //<height that WGS84 is above AMSL at the current location
bool have_undulation; ///<do we have a value for the undulation
uint32_t last_gps_time_ms; ///< the system time we got the last GPS timestamp, milliseconds uint32_t last_gps_time_ms; ///< the system time we got the last GPS timestamp, milliseconds
uint64_t last_corrected_gps_time_us;///< the system time we got the last corrected GPS timestamp, microseconds uint64_t last_corrected_gps_time_us;///< the system time we got the last corrected GPS timestamp, microseconds
bool corrected_timestamp_updated; ///< true if the corrected timestamp has been updated bool corrected_timestamp_updated; ///< true if the corrected timestamp has been updated
@ -293,6 +295,16 @@ public:
return location(primary_instance); return location(primary_instance);
} }
// get the difference between WGS84 and AMSL. A positive value means
// the AMSL height is higher than WGS84 ellipsoid height
bool get_undulation(uint8_t instance, float &undulation) const;
// get the difference between WGS84 and AMSL. A positive value means
// the AMSL height is higher than WGS84 ellipsoid height
bool get_undulation(float &undulation) const {
return get_undulation(primary_instance, undulation);
}
// report speed accuracy // report speed accuracy
bool speed_accuracy(uint8_t instance, float &sacc) const; bool speed_accuracy(uint8_t instance, float &sacc) const;
bool speed_accuracy(float &sacc) const { bool speed_accuracy(float &sacc) const {

View File

@ -146,6 +146,8 @@ AP_GPS_ERB::_parse_gps(void)
state.location.lng = (int32_t)(_buffer.pos.longitude * (double)1e7); state.location.lng = (int32_t)(_buffer.pos.longitude * (double)1e7);
state.location.lat = (int32_t)(_buffer.pos.latitude * (double)1e7); state.location.lat = (int32_t)(_buffer.pos.latitude * (double)1e7);
state.location.alt = (int32_t)(_buffer.pos.altitude_msl * 100); state.location.alt = (int32_t)(_buffer.pos.altitude_msl * 100);
state.have_undulation = true;
state.undulation = _buffer.pos.altitude_msl - _buffer.pos.altitude_ellipsoid;
state.status = next_fix; state.status = next_fix;
_new_position = true; _new_position = true;
state.horizontal_accuracy = _buffer.pos.horizontal_accuracy * 1.0e-3f; state.horizontal_accuracy = _buffer.pos.horizontal_accuracy * 1.0e-3f;

View File

@ -205,6 +205,8 @@ AP_GPS_NOVA::process_message(void)
state.location.lat = (int32_t) (bestposu.lat * (double)1e7); state.location.lat = (int32_t) (bestposu.lat * (double)1e7);
state.location.lng = (int32_t) (bestposu.lng * (double)1e7); state.location.lng = (int32_t) (bestposu.lng * (double)1e7);
state.location.alt = (int32_t) (bestposu.hgt * 100); state.location.alt = (int32_t) (bestposu.hgt * 100);
state.have_undulation = true;
state.undulation = bestposu.undulation;
state.num_sats = bestposu.svsused; state.num_sats = bestposu.svsused;

View File

@ -416,6 +416,8 @@ AP_GPS_SBF::process_message(void)
state.location.lat = (int32_t)(temp.Latitude * RAD_TO_DEG_DOUBLE * (double)1e7); state.location.lat = (int32_t)(temp.Latitude * RAD_TO_DEG_DOUBLE * (double)1e7);
state.location.lng = (int32_t)(temp.Longitude * RAD_TO_DEG_DOUBLE * (double)1e7); state.location.lng = (int32_t)(temp.Longitude * RAD_TO_DEG_DOUBLE * (double)1e7);
state.location.alt = (int32_t)(((float)temp.Height - temp.Undulation) * 1e2f); state.location.alt = (int32_t)(((float)temp.Height - temp.Undulation) * 1e2f);
state.have_undulation = true;
state.undulation = temp.Undulation;
} }
if (temp.NrSV != 255) { if (temp.NrSV != 255) {

View File

@ -180,6 +180,8 @@ AP_GPS_SIRF::_parse_gps(void)
state.location.lat = swap_int32(_buffer.nav.latitude); state.location.lat = swap_int32(_buffer.nav.latitude);
state.location.lng = swap_int32(_buffer.nav.longitude); state.location.lng = swap_int32(_buffer.nav.longitude);
state.location.alt = swap_int32(_buffer.nav.altitude_msl); state.location.alt = swap_int32(_buffer.nav.altitude_msl);
state.have_undulation = true;
state.undulation = (state.location.alt - swap_int32(_buffer.nav.altitude_ellipsoid))*0.01;
state.ground_speed = swap_int32(_buffer.nav.ground_speed)*0.01f; state.ground_speed = swap_int32(_buffer.nav.ground_speed)*0.01f;
state.ground_course = wrap_360(swap_int16(_buffer.nav.ground_course)*0.01f); state.ground_course = wrap_360(swap_int16(_buffer.nav.ground_course)*0.01f);
state.num_sats = _buffer.nav.satellites; state.num_sats = _buffer.nav.satellites;

View File

@ -1216,6 +1216,9 @@ AP_GPS_UBLOX::_parse_gps(void)
state.location.lng = _buffer.posllh.longitude; state.location.lng = _buffer.posllh.longitude;
state.location.lat = _buffer.posllh.latitude; state.location.lat = _buffer.posllh.latitude;
state.location.alt = _buffer.posllh.altitude_msl / 10; state.location.alt = _buffer.posllh.altitude_msl / 10;
state.have_undulation = true;
state.undulation = (_buffer.posllh.altitude_msl - _buffer.posllh.altitude_ellipsoid) * 0.001;
state.status = next_fix; state.status = next_fix;
_new_position = true; _new_position = true;
state.horizontal_accuracy = _buffer.posllh.horizontal_accuracy*1.0e-3f; state.horizontal_accuracy = _buffer.posllh.horizontal_accuracy*1.0e-3f;
@ -1369,6 +1372,8 @@ AP_GPS_UBLOX::_parse_gps(void)
state.location.lng = _buffer.pvt.lon; state.location.lng = _buffer.pvt.lon;
state.location.lat = _buffer.pvt.lat; state.location.lat = _buffer.pvt.lat;
state.location.alt = _buffer.pvt.h_msl / 10; state.location.alt = _buffer.pvt.h_msl / 10;
state.have_undulation = true;
state.undulation = (_buffer.pvt.h_msl - _buffer.pvt.h_ellipsoid) * 0.001;
switch (_buffer.pvt.fix_type) switch (_buffer.pvt.fix_type)
{ {
case 0: case 0:

View File

@ -341,7 +341,7 @@ private:
uint8_t flags2; uint8_t flags2;
uint8_t num_sv; uint8_t num_sv;
int32_t lon, lat; int32_t lon, lat;
int32_t height, h_msl; int32_t h_ellipsoid, h_msl;
uint32_t h_acc, v_acc; uint32_t h_acc, v_acc;
int32_t velN, velE, velD, gspeed; int32_t velN, velE, velD, gspeed;
int32_t head_mot; int32_t head_mot;

View File

@ -62,6 +62,7 @@ struct PACKED log_GPS {
// @Field: VV: true if vertical velocity is available // @Field: VV: true if vertical velocity is available
// @Field: SMS: time since system startup this sample was taken // @Field: SMS: time since system startup this sample was taken
// @Field: Delta: system time delta between the last two reported positions // @Field: Delta: system time delta between the last two reported positions
// @Field: Und: Undulation
struct PACKED log_GPA { struct PACKED log_GPA {
LOG_PACKET_HEADER; LOG_PACKET_HEADER;
uint64_t time_us; uint64_t time_us;
@ -74,6 +75,7 @@ struct PACKED log_GPA {
uint8_t have_vv; uint8_t have_vv;
uint32_t sample_ms; uint32_t sample_ms;
uint16_t delta_ms; uint16_t delta_ms;
float undulation;
}; };
/* /*
@ -200,7 +202,7 @@ struct PACKED log_GPS_RAWS {
{ LOG_GPS_MSG, sizeof(log_GPS), \ { LOG_GPS_MSG, sizeof(log_GPS), \
"GPS", "QBBIHBcLLeffffB", "TimeUS,I,Status,GMS,GWk,NSats,HDop,Lat,Lng,Alt,Spd,GCrs,VZ,Yaw,U", "s#---SmDUmnhnh-", "F----0BGGB000--" , true }, \ "GPS", "QBBIHBcLLeffffB", "TimeUS,I,Status,GMS,GWk,NSats,HDop,Lat,Lng,Alt,Spd,GCrs,VZ,Yaw,U", "s#---SmDUmnhnh-", "F----0BGGB000--" , true }, \
{ LOG_GPA_MSG, sizeof(log_GPA), \ { LOG_GPA_MSG, sizeof(log_GPA), \
"GPA", "QBCCCCfBIH", "TimeUS,I,VDop,HAcc,VAcc,SAcc,YAcc,VV,SMS,Delta", "s#mmmnd-ss", "F-BBBB0-CC" , true }, \ "GPA", "QBCCCCfBIHf", "TimeUS,I,VDop,HAcc,VAcc,SAcc,YAcc,VV,SMS,Delta,Und", "s#mmmnd-ssm", "F-BBBB0-CC0" , true }, \
{ LOG_GPS_UBX1_MSG, sizeof(log_Ubx1), \ { LOG_GPS_UBX1_MSG, sizeof(log_Ubx1), \
"UBX1", "QBHBBHI", "TimeUS,Instance,noisePerMS,jamInd,aPower,agcCnt,config", "s#-----", "F------" , true }, \ "UBX1", "QBHBBHI", "TimeUS,Instance,noisePerMS,jamInd,aPower,agcCnt,config", "s#-----", "F------" , true }, \
{ LOG_GPS_UBX2_MSG, sizeof(log_Ubx2), \ { LOG_GPS_UBX2_MSG, sizeof(log_Ubx2), \