mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-08 17:08:28 -04:00
AP_GPS: support KSXT message for Unicore NMEA
this gives both yaw and 3D velocity
This commit is contained in:
parent
c565b8a84e
commit
27cec48b82
@ -210,20 +210,21 @@ bool AP_GPS_NMEA::_have_new_message()
|
||||
}
|
||||
|
||||
/*
|
||||
if we have seen the $PHD messages then wait for them again. This
|
||||
is important as the have_vertical_velocity field will be
|
||||
overwritten by fill_3d_velocity()
|
||||
if we have seen a message with 3D velocity data messages then
|
||||
wait for them again. This is important as the
|
||||
have_vertical_velocity field will be overwritten by
|
||||
fill_3d_velocity()
|
||||
*/
|
||||
if (_last_PHD_12_ms != 0 &&
|
||||
now - _last_PHD_12_ms > 150 &&
|
||||
now - _last_PHD_12_ms < 1000) {
|
||||
// waiting on PHD_12
|
||||
if (_last_vvelocity_ms != 0 &&
|
||||
now - _last_vvelocity_ms > 150 &&
|
||||
now - _last_vvelocity_ms < 1000) {
|
||||
// waiting on a message with velocity
|
||||
return false;
|
||||
}
|
||||
if (_last_PHD_26_ms != 0 &&
|
||||
now - _last_PHD_26_ms > 150 &&
|
||||
now - _last_PHD_26_ms < 1000) {
|
||||
// waiting on PHD_26
|
||||
if (_last_vaccuracy_ms != 0 &&
|
||||
now - _last_vaccuracy_ms > 150 &&
|
||||
now - _last_vaccuracy_ms < 1000) {
|
||||
// waiting on a message with velocity accuracy
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -232,7 +233,7 @@ bool AP_GPS_NMEA::_have_new_message()
|
||||
_last_VTG_ms = 1;
|
||||
}
|
||||
|
||||
if (now - _last_HDT_THS_ms > 300) {
|
||||
if (now - _last_yaw_ms > 300) {
|
||||
// we have lost GPS yaw
|
||||
state.have_gps_yaw = false;
|
||||
}
|
||||
@ -276,13 +277,16 @@ bool AP_GPS_NMEA::_term_complete()
|
||||
//date = _new_date;
|
||||
state.location.lat = _new_latitude;
|
||||
state.location.lng = _new_longitude;
|
||||
state.ground_speed = _new_speed*0.01f;
|
||||
state.ground_course = wrap_360(_new_course*0.01f);
|
||||
if (_last_3D_velocity_ms == 0 ||
|
||||
now - _last_3D_velocity_ms > 1000) {
|
||||
state.ground_speed = _new_speed*0.01f;
|
||||
state.ground_course = wrap_360(_new_course*0.01f);
|
||||
}
|
||||
make_gps_time(_new_date, _new_time * 10);
|
||||
set_uart_timestamp(_sentence_length);
|
||||
state.last_gps_time_ms = now;
|
||||
if (_last_PHD_12_ms == 0 ||
|
||||
now - _last_PHD_12_ms > 1000) {
|
||||
if (_last_vvelocity_ms == 0 ||
|
||||
now - _last_vvelocity_ms > 1000) {
|
||||
fill_3d_velocity();
|
||||
}
|
||||
break;
|
||||
@ -322,17 +326,20 @@ bool AP_GPS_NMEA::_term_complete()
|
||||
break;
|
||||
case _GPS_SENTENCE_VTG:
|
||||
_last_VTG_ms = now;
|
||||
state.ground_speed = _new_speed*0.01f;
|
||||
state.ground_course = wrap_360(_new_course*0.01f);
|
||||
if (_last_PHD_12_ms == 0 ||
|
||||
now - _last_PHD_12_ms > 1000) {
|
||||
fill_3d_velocity();
|
||||
if (_last_3D_velocity_ms == 0 ||
|
||||
now - _last_3D_velocity_ms > 1000) {
|
||||
state.ground_speed = _new_speed*0.01f;
|
||||
state.ground_course = wrap_360(_new_course*0.01f);
|
||||
if (_last_vvelocity_ms == 0 ||
|
||||
now - _last_vvelocity_ms > 1000) {
|
||||
fill_3d_velocity();
|
||||
}
|
||||
}
|
||||
// VTG has no fix indicator, can't change fix status
|
||||
break;
|
||||
case _GPS_SENTENCE_HDT:
|
||||
case _GPS_SENTENCE_THS:
|
||||
_last_HDT_THS_ms = now;
|
||||
_last_yaw_ms = now;
|
||||
state.gps_yaw = wrap_360(_new_gps_yaw*0.01f);
|
||||
state.have_gps_yaw = true;
|
||||
state.gps_yaw_time_ms = AP_HAL::millis();
|
||||
@ -348,7 +355,11 @@ bool AP_GPS_NMEA::_term_complete()
|
||||
state.velocity.y = _phd.fields[1] * 0.01;
|
||||
state.velocity.z = _phd.fields[2] * 0.01;
|
||||
state.have_vertical_velocity = true;
|
||||
_last_PHD_12_ms = now;
|
||||
_last_vvelocity_ms = now;
|
||||
// we prefer a true 3D velocity when available
|
||||
state.ground_course = wrap_360(degrees(atan2f(state.velocity.y, state.velocity.x)));
|
||||
state.ground_speed = state.velocity.xy().length();
|
||||
_last_3D_velocity_ms = now;
|
||||
} else if (_phd.msg_id == 26) {
|
||||
state.horizontal_accuracy = MAX(_phd.fields[0],_phd.fields[1]) * 0.001;
|
||||
state.have_horizontal_accuracy = true;
|
||||
@ -356,8 +367,32 @@ bool AP_GPS_NMEA::_term_complete()
|
||||
state.have_vertical_accuracy = true;
|
||||
state.speed_accuracy = MAX(_phd.fields[3],_phd.fields[4]) * 0.001;
|
||||
state.have_speed_accuracy = true;
|
||||
_last_PHD_26_ms = now;
|
||||
_last_vaccuracy_ms = now;
|
||||
}
|
||||
break;
|
||||
case _GPS_SENTENCE_KSXT:
|
||||
if (_ksxt.fields[9] >= 1) {
|
||||
// we have 3D fix
|
||||
constexpr float kmh_to_mps = 1.0 / 3.6;
|
||||
state.velocity.y = _ksxt.fields[16] * kmh_to_mps;
|
||||
state.velocity.x = _ksxt.fields[17] * kmh_to_mps;
|
||||
state.velocity.z = _ksxt.fields[18] * -kmh_to_mps;
|
||||
state.have_vertical_velocity = true;
|
||||
_last_vvelocity_ms = now;
|
||||
// we prefer a true 3D velocity when available
|
||||
state.ground_course = wrap_360(degrees(atan2f(state.velocity.y, state.velocity.x)));
|
||||
state.ground_speed = state.velocity.xy().length();
|
||||
_last_3D_velocity_ms = now;
|
||||
}
|
||||
if (is_equal(3.0f, _ksxt.fields[10])) {
|
||||
// have good yaw (from RTK fixed moving baseline solution)
|
||||
_last_yaw_ms = now;
|
||||
state.gps_yaw = _ksxt.fields[4];
|
||||
state.have_gps_yaw = true;
|
||||
state.gps_yaw_time_ms = AP_HAL::millis();
|
||||
state.gps_yaw_configured = true;
|
||||
}
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
switch (_sentence_type) {
|
||||
@ -388,6 +423,11 @@ bool AP_GPS_NMEA::_term_complete()
|
||||
_sentence_type = _GPS_SENTENCE_PHD;
|
||||
return false;
|
||||
}
|
||||
if (strcmp(_term, "KSXT") == 0) {
|
||||
_sentence_type = _GPS_SENTENCE_KSXT;
|
||||
_gps_data_good = true;
|
||||
return false;
|
||||
}
|
||||
/*
|
||||
The first two letters of the NMEA term are the talker
|
||||
ID. The most common is 'GP' but there are a bunch of others
|
||||
@ -512,6 +552,9 @@ bool AP_GPS_NMEA::_term_complete()
|
||||
case _GPS_SENTENCE_PHD + 6 ... _GPS_SENTENCE_PHD + 11: // PHD message, fields
|
||||
_phd.fields[_term_number-6] = atol(_term);
|
||||
break;
|
||||
case _GPS_SENTENCE_KSXT + 1 ... _GPS_SENTENCE_KSXT + 22: // PHD message, fields
|
||||
_ksxt.fields[_term_number-1] = atof(_term);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -74,6 +74,7 @@ private:
|
||||
_GPS_SENTENCE_HDT = 128,
|
||||
_GPS_SENTENCE_PHD = 138, // extension for AllyStar GPS modules
|
||||
_GPS_SENTENCE_THS = 160, // True heading with quality indicator, available on Trimble MB-Two
|
||||
_GPS_SENTENCE_KSXT = 170, // extension for Unicore, 21 fields
|
||||
_GPS_SENTENCE_OTHER = 0
|
||||
};
|
||||
|
||||
@ -144,9 +145,10 @@ private:
|
||||
uint32_t _last_RMC_ms;
|
||||
uint32_t _last_GGA_ms;
|
||||
uint32_t _last_VTG_ms;
|
||||
uint32_t _last_HDT_THS_ms;
|
||||
uint32_t _last_PHD_12_ms;
|
||||
uint32_t _last_PHD_26_ms;
|
||||
uint32_t _last_yaw_ms;
|
||||
uint32_t _last_vvelocity_ms;
|
||||
uint32_t _last_vaccuracy_ms;
|
||||
uint32_t _last_3D_velocity_ms;
|
||||
uint32_t _last_fix_ms;
|
||||
|
||||
/// @name Init strings
|
||||
@ -178,6 +180,15 @@ private:
|
||||
uint32_t itow;
|
||||
int32_t fields[8];
|
||||
} _phd;
|
||||
|
||||
/*
|
||||
The KSXT message is an extension from Unicore that gives 3D velocity and yaw
|
||||
example: $KSXT,20211016083433.00,116.31296102,39.95817066,49.4911,223.57,-11.32,330.19,0.024,,1,3,28,27,,,,-0.012,0.021,0.020,,*2D
|
||||
*/
|
||||
struct {
|
||||
float fields[21];
|
||||
} _ksxt;
|
||||
|
||||
};
|
||||
|
||||
#define AP_GPS_NMEA_HEMISPHERE_INIT_STRING \
|
||||
|
Loading…
Reference in New Issue
Block a user