2021-07-27 19:48:53 -03:00
/*
Copyright ( C ) 2021 Kraus Hamdani Aerospace Inc . All rights reserved .
This program is free software : you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation , either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
Author : Tom Pittenger
*/
# include "AP_ADSB_uAvionix_UCP.h"
2021-12-16 17:28:24 -04:00
// This driver implements the UCP protocol from uAvionix which is a variant of the GDL90 protocol by Garmin
// https://uavionix.com/downloads/ping200X/uAvionix-UCP-Transponder-ICD-Rev-Q.pdf
2021-07-27 19:48:53 -03:00
# if HAL_ADSB_UCP_ENABLED
# include <AP_SerialManager/AP_SerialManager.h>
# include <GCS_MAVLink/GCS.h>
# include <AP_HAL/AP_HAL.h>
# include <AP_Math/AP_Math.h>
# include <AP_Math/crc.h>
# include <ctype.h>
# include <AP_Notify/AP_Notify.h>
2024-08-30 23:01:02 -03:00
# include <AP_GPS/AP_GPS.h>
2021-07-27 19:48:53 -03:00
extern const AP_HAL : : HAL & hal ;
2021-09-21 16:12:43 -03:00
# define AP_ADSB_UAVIONIX_HEALTH_TIMEOUT_MS (5000UL)
2021-07-27 19:48:53 -03:00
# define AP_ADSB_UAVIONIX_GCS_LOST_COMMS_LONG_TIMEOUT_MINUTES (15UL)
# define AP_ADSB_UAVIONIX_GCS_LOST_COMMS_LONG_TIMEOUT_MS (1000UL * 60UL * AP_ADSB_UAVIONIX_GCS_LOST_COMMS_LONG_TIMEOUT_MINUTES)
# define AP_ADSB_UAVIONIX_DETECT_GROUNDSTATE 0
# define AP_ADSB_UAVIONIX_EMERGENCY_STATUS_ON_LOST_LINK 0
// detect if any port is configured as uAvionix_UCP
bool AP_ADSB_uAvionix_UCP : : detect ( )
{
2021-12-20 00:35:42 -04:00
return AP : : serialmanager ( ) . have_serial ( AP_SerialManager : : SerialProtocol_ADSB , 0 ) ;
2021-07-27 19:48:53 -03:00
}
// Init, called once after class is constructed
bool AP_ADSB_uAvionix_UCP : : init ( )
{
_port = AP : : serialmanager ( ) . find_serial ( AP_SerialManager : : SerialProtocol_ADSB , 0 ) ;
if ( _port = = nullptr ) {
return false ;
}
2024-10-09 15:31:38 -03:00
_frontend . out_state . ctrl . squawkCode = 1200 ;
_frontend . out_state . tx_status . squawk = 1200 ;
_frontend . out_state . tx_status . fault | = UAVIONIX_ADSB_OUT_STATUS_FAULT_STATUS_MESSAGE_UNAVAIL ;
2021-07-27 19:48:53 -03:00
return true ;
}
void AP_ADSB_uAvionix_UCP : : update ( )
{
if ( _port = = nullptr ) {
return ;
}
const uint32_t now_ms = AP_HAL : : millis ( ) ;
// -----------------------------
// read any available data on serial port
// -----------------------------
uint32_t nbytes = MIN ( _port - > available ( ) , 10UL * GDL90_RX_MAX_PACKET_LENGTH ) ;
while ( nbytes - - > 0 ) {
2023-07-11 16:30:23 -03:00
uint8_t data ;
if ( ! _port - > read ( data ) ) {
2021-07-27 19:48:53 -03:00
break ;
}
2023-07-11 16:30:23 -03:00
if ( parseByte ( data , rx . msg , rx . status ) ) {
2021-07-27 19:48:53 -03:00
rx . last_msg_ms = now_ms ;
handle_msg ( rx . msg ) ;
}
} // while nbytes
2024-10-17 11:52:25 -03:00
if ( run_state . last_packet_Transponder_Id_ms = = 0 & & run_state . request_Transponder_Id_tries < 5 ) {
2024-10-09 15:31:38 -03:00
if ( now_ms - run_state . last_packet_Request_Transponder_Id_ms > = 1000 )
{
request_msg ( GDL90_ID_IDENTIFICATION ) ;
run_state . request_Transponder_Id_tries + + ;
}
}
2024-10-17 11:52:25 -03:00
if ( run_state . last_packet_Transponder_Config_ms = = 0 & & run_state . request_Transponder_Config_tries < 5 ) {
2024-10-09 15:31:38 -03:00
if ( now_ms - run_state . last_packet_Request_Transponder_Config_ms > = 1000 )
{
request_msg ( GDL90_ID_TRANSPONDER_CONFIG ) ;
run_state . request_Transponder_Config_tries + + ;
}
}
2021-07-27 19:48:53 -03:00
if ( now_ms - run_state . last_packet_Transponder_Control_ms > = 1000 ) {
run_state . last_packet_Transponder_Control_ms = now_ms ;
2024-10-09 15:31:38 -03:00
// We want to use the defaults stored on the ping200X, if possible.
// Until we get the config message (or we've tried requesting it several times),
2024-10-17 11:52:25 -03:00
// don't send the control message.
if ( run_state . last_packet_Transponder_Config_ms ! = 0 | | run_state . request_Transponder_Config_tries > = 5 ) {
send_Transponder_Control ( ) ;
}
2021-07-27 19:48:53 -03:00
}
if ( ( now_ms - run_state . last_packet_GPS_ms > = 200 ) & & ( _frontend . _options & uint32_t ( AP_ADSB : : AdsbOption : : Ping200X_Send_GPS ) ) ! = 0 ) {
run_state . last_packet_GPS_ms = now_ms ;
send_GPS_Data ( ) ;
}
2021-09-21 16:12:43 -03:00
2024-10-17 11:52:25 -03:00
// if the transponder has stopped giving us the data needed to
2024-10-09 15:31:38 -03:00
// fill the transponder status mavlink message, indicate status unavailable.
if ( ( now_ms - run_state . last_packet_Transponder_Status_ms > = 10000 )
& & ( now_ms - run_state . last_packet_Transponder_Heartbeat_ms > = 10000 )
2024-10-17 11:52:25 -03:00
& & ( now_ms - run_state . last_packet_Transponder_Ownship_ms > = 10000 ) ) {
2021-10-13 20:31:14 -03:00
_frontend . out_state . tx_status . fault | = UAVIONIX_ADSB_OUT_STATUS_FAULT_STATUS_MESSAGE_UNAVAIL ;
2021-09-21 16:12:43 -03:00
}
2021-07-27 19:48:53 -03:00
}
void AP_ADSB_uAvionix_UCP : : handle_msg ( const GDL90_RX_MESSAGE & msg )
{
switch ( msg . messageId ) {
2021-10-13 20:31:14 -03:00
case GDL90_ID_HEARTBEAT : {
2021-07-27 19:48:53 -03:00
// The Heartbeat message provides real-time indications of the status and operation of the
// transponder. The message will be transmitted with a period of one second for the UCP
// protocol.
memcpy ( & rx . decoded . heartbeat , msg . raw , sizeof ( rx . decoded . heartbeat ) ) ;
2021-10-13 20:31:14 -03:00
run_state . last_packet_Transponder_Heartbeat_ms = AP_HAL : : millis ( ) ;
2024-10-09 15:31:38 -03:00
_frontend . out_state . tx_status . fault & = ~ UAVIONIX_ADSB_OUT_STATUS_FAULT_STATUS_MESSAGE_UNAVAIL ;
2021-10-13 20:31:14 -03:00
if ( rx . decoded . heartbeat . status . one . maintenanceRequired ) {
_frontend . out_state . tx_status . fault | = UAVIONIX_ADSB_OUT_STATUS_FAULT_MAINT_REQ ;
} else {
_frontend . out_state . tx_status . fault & = ~ UAVIONIX_ADSB_OUT_STATUS_FAULT_MAINT_REQ ;
}
if ( rx . decoded . heartbeat . status . two . functionFailureGnssUnavailable ) {
_frontend . out_state . tx_status . fault | = UAVIONIX_ADSB_OUT_STATUS_FAULT_GPS_UNAVAIL ;
} else {
_frontend . out_state . tx_status . fault & = ~ UAVIONIX_ADSB_OUT_STATUS_FAULT_GPS_UNAVAIL ;
}
if ( rx . decoded . heartbeat . status . two . functionFailureGnssNo3dFix ) {
_frontend . out_state . tx_status . fault | = UAVIONIX_ADSB_OUT_STATUS_FAULT_GPS_NO_POS ;
} else {
_frontend . out_state . tx_status . fault & = ~ UAVIONIX_ADSB_OUT_STATUS_FAULT_GPS_NO_POS ;
}
if ( rx . decoded . heartbeat . status . two . functionFailureTransmitSystem ) {
_frontend . out_state . tx_status . fault | = UAVIONIX_ADSB_OUT_STATUS_FAULT_TX_SYSTEM_FAIL ;
} else {
_frontend . out_state . tx_status . fault & = ~ UAVIONIX_ADSB_OUT_STATUS_FAULT_TX_SYSTEM_FAIL ;
}
2021-07-27 19:48:53 -03:00
}
break ;
case GDL90_ID_IDENTIFICATION :
2024-10-09 15:31:38 -03:00
run_state . last_packet_Transponder_Id_ms = AP_HAL : : millis ( ) ;
2021-07-27 19:48:53 -03:00
// The Identification message contains information used to identify the connected device. The
// Identification message will be transmitted with a period of one second regardless of data status
// or update for the UCP protocol and will be transmitted upon request for the UCP-HD protocol.
if ( memcmp ( & rx . decoded . identification , msg . raw , sizeof ( rx . decoded . identification ) ) ! = 0 ) {
memcpy ( & rx . decoded . identification , msg . raw , sizeof ( rx . decoded . identification ) ) ;
// Firmware Part Number (not null terminated, but null padded if part number is less than 15 characters).
// Copy into a temporary string that is 1 char longer so we ensure it's null terminated
const uint8_t str_len = sizeof ( rx . decoded . identification . primaryFwPartNumber ) ;
char primaryFwPartNumber [ str_len + 1 ] ;
memcpy ( & primaryFwPartNumber , rx . decoded . identification . primaryFwPartNumber , str_len ) ;
primaryFwPartNumber [ str_len ] = 0 ;
2024-10-17 11:52:25 -03:00
2021-07-27 19:48:53 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_DEBUG , " ADSB:Detected %s v%u.%u.%u SN:%u %s " ,
get_hardware_name ( rx . decoded . identification . primary . hwId ) ,
( unsigned ) rx . decoded . identification . primary . fwMajorVersion ,
( unsigned ) rx . decoded . identification . primary . fwMinorVersion ,
( unsigned ) rx . decoded . identification . primary . fwBuildVersion ,
( unsigned ) rx . decoded . identification . primary . serialNumber ,
primaryFwPartNumber ) ;
}
break ;
case GDL90_ID_TRANSPONDER_CONFIG :
2024-10-09 15:31:38 -03:00
run_state . last_packet_Transponder_Config_ms = AP_HAL : : millis ( ) ;
2021-07-27 19:48:53 -03:00
memcpy ( & rx . decoded . transponder_config , msg . raw , sizeof ( rx . decoded . transponder_config ) ) ;
break ;
# if AP_ADSB_UAVIONIX_UCP_CAPTURE_ALL_RX_PACKETS
case GDL90_ID_OWNSHIP_REPORT :
2024-10-09 15:31:38 -03:00
_frontend . out_state . tx_status . fault & = ~ UAVIONIX_ADSB_OUT_STATUS_FAULT_STATUS_MESSAGE_UNAVAIL ;
2021-07-27 19:48:53 -03:00
// The Ownship message contains information on the GNSS position. If the Ownship GNSS
// position fix is invalid, the Latitude, Longitude, and NIC fields will all have the ZERO value. The
// Ownship message will be transmitted with a period of one second regardless of data status or
// update for the UCP protocol. All fields in the ownship message are transmitted MSB first
memcpy ( & rx . decoded . ownship_report , msg . raw , sizeof ( rx . decoded . ownship_report ) ) ;
2021-10-13 20:31:14 -03:00
run_state . last_packet_Transponder_Ownship_ms = AP_HAL : : millis ( ) ;
_frontend . out_state . tx_status . NIC_NACp = rx . decoded . ownship_report . report . NIC | ( rx . decoded . ownship_report . report . NACp < < 4 ) ;
memcpy ( _frontend . out_state . tx_status . flight_id , rx . decoded . ownship_report . report . callsign , sizeof ( _frontend . out_state . tx_status . flight_id ) ) ;
2021-07-27 19:48:53 -03:00
break ;
case GDL90_ID_OWNSHIP_GEOMETRIC_ALTITUDE :
// An Ownship Geometric Altitude message will be transmitted with a period of one second when
// the GNSS fix is valid for the UCP protocol. All fields in the Geometric Ownship Altitude
// message are transmitted MSB first.
memcpy ( & rx . decoded . ownship_geometric_altitude , msg . raw , sizeof ( rx . decoded . ownship_geometric_altitude ) ) ;
break ;
case GDL90_ID_SENSOR_MESSAGE :
memcpy ( & rx . decoded . sensor_message , msg . raw , sizeof ( rx . decoded . sensor_message ) ) ;
break ;
case GDL90_ID_TRANSPONDER_STATUS :
2024-10-09 15:31:38 -03:00
{
2021-10-13 20:31:14 -03:00
_frontend . out_state . tx_status . fault & = ~ UAVIONIX_ADSB_OUT_STATUS_FAULT_STATUS_MESSAGE_UNAVAIL ;
2024-10-17 11:52:25 -03:00
switch ( msg . payload [ 0 ] ) {
case 1 : {
2024-10-09 15:31:38 -03:00
// version 1 of the transponder status message is sent at 1 Hz (if UCP protocol out is enabled on the transponder)
memcpy ( & rx . decoded . transponder_status , msg . raw , sizeof ( rx . decoded . transponder_status ) ) ;
if ( rx . decoded . transponder_status . identActive ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_IDENT_ACTIVE ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_IDENT_ACTIVE ;
}
if ( rx . decoded . transponder_status . modeAEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_A_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_A_ENABLED ;
}
if ( rx . decoded . transponder_status . modeCEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_C_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_C_ENABLED ;
}
if ( rx . decoded . transponder_status . modeSEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_S_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_S_ENABLED ;
}
if ( rx . decoded . transponder_status . es1090TxEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_1090ES_TX_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_1090ES_TX_ENABLED ;
}
if ( rx . decoded . transponder_status . x_bit ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_XBIT_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_XBIT_ENABLED ;
}
_frontend . out_state . tx_status . squawk = rx . decoded . transponder_status . squawkCode ;
if ( run_state . last_packet_Transponder_Status_ms = = 0 & & run_state . last_packet_Transponder_Config_ms = = 0 ) {
// If this is the first time we've seen a status message,
// and we haven't initialized the control message from the config message,
// set initial control message contents to match transponder's current behavior.
_frontend . out_state . ctrl . modeAEnabled = rx . decoded . transponder_status . modeAEnabled ;
_frontend . out_state . ctrl . modeCEnabled = rx . decoded . transponder_status . modeCEnabled ;
_frontend . out_state . ctrl . modeSEnabled = rx . decoded . transponder_status . modeSEnabled ;
_frontend . out_state . ctrl . es1090TxEnabled = rx . decoded . transponder_status . es1090TxEnabled ;
_frontend . out_state . ctrl . squawkCode = rx . decoded . transponder_status . squawkCode ;
_frontend . out_state . ctrl . x_bit = rx . decoded . transponder_status . x_bit ;
}
run_state . last_packet_Transponder_Status_ms = AP_HAL : : millis ( ) ;
# if AP_MAVLINK_MSG_UAVIONIX_ADSB_OUT_STATUS_ENABLED
GCS_SEND_MESSAGE ( MSG_UAVIONIX_ADSB_OUT_STATUS ) ;
run_state . last_gcs_send_message_Transponder_Status_ms = AP_HAL : : millis ( ) ;
# endif
break ;
2021-10-13 20:31:14 -03:00
}
2024-10-09 15:31:38 -03:00
case 2 :
// deprecated
break ;
2024-10-17 11:52:25 -03:00
case 3 : {
2024-10-09 15:31:38 -03:00
// Version 3 of the transponder status message is sent in response to the transponder control message (if UCP-HD protocol out is enabled on the transponder)
memcpy ( & rx . decoded . transponder_status_v3 , msg . raw , sizeof ( rx . decoded . transponder_status_v3 ) ) ;
if ( rx . decoded . transponder_status_v3 . indicatingOnGround ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_ON_GROUND ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_ON_GROUND ;
}
if ( rx . decoded . transponder_status_v3 . fault ) {
// unsure what fault is indicated, query heartbeat for more info
request_msg ( GDL90_ID_HEARTBEAT ) ;
}
if ( rx . decoded . transponder_status_v3 . identActive ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_IDENT_ACTIVE ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_IDENT_ACTIVE ;
}
if ( rx . decoded . transponder_status_v3 . modeAEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_A_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_A_ENABLED ;
}
if ( rx . decoded . transponder_status_v3 . modeCEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_C_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_C_ENABLED ;
}
if ( rx . decoded . transponder_status_v3 . modeSEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_S_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_MODE_S_ENABLED ;
}
if ( rx . decoded . transponder_status_v3 . es1090TxEnabled ) {
_frontend . out_state . tx_status . state | = UAVIONIX_ADSB_OUT_STATUS_STATE_1090ES_TX_ENABLED ;
} else {
_frontend . out_state . tx_status . state & = ~ UAVIONIX_ADSB_OUT_STATUS_STATE_1090ES_TX_ENABLED ;
}
_frontend . out_state . tx_status . squawk = rx . decoded . transponder_status_v3 . squawkCode ;
_frontend . out_state . tx_status . NIC_NACp = rx . decoded . transponder_status_v3 . NIC | ( rx . decoded . transponder_status_v3 . NACp < < 4 ) ;
_frontend . out_state . tx_status . boardTemp = rx . decoded . transponder_status_v3 . temperature ;
if ( run_state . last_packet_Transponder_Status_ms = = 0 & & run_state . last_packet_Transponder_Config_ms = = 0 ) {
// If this is the first time we've seen a status message,
// and we haven't initialized the control message from the config message,
// set initial control message contents to match transponder's current behavior.
_frontend . out_state . ctrl . modeAEnabled = rx . decoded . transponder_status_v3 . modeAEnabled ;
_frontend . out_state . ctrl . modeCEnabled = rx . decoded . transponder_status_v3 . modeCEnabled ;
_frontend . out_state . ctrl . modeSEnabled = rx . decoded . transponder_status_v3 . modeSEnabled ;
_frontend . out_state . ctrl . es1090TxEnabled = rx . decoded . transponder_status_v3 . es1090TxEnabled ;
_frontend . out_state . ctrl . squawkCode = rx . decoded . transponder_status_v3 . squawkCode ;
}
run_state . last_packet_Transponder_Status_ms = AP_HAL : : millis ( ) ;
2023-11-07 19:36:58 -04:00
# if AP_MAVLINK_MSG_UAVIONIX_ADSB_OUT_STATUS_ENABLED
2024-10-09 15:31:38 -03:00
GCS_SEND_MESSAGE ( MSG_UAVIONIX_ADSB_OUT_STATUS ) ;
run_state . last_gcs_send_message_Transponder_Status_ms = AP_HAL : : millis ( ) ;
2023-11-07 19:36:58 -04:00
# endif
2024-10-09 15:31:38 -03:00
break ;
}
default :
break ;
}
2021-07-27 19:48:53 -03:00
break ;
2024-10-09 15:31:38 -03:00
}
2021-10-13 20:31:14 -03:00
# endif // AP_ADSB_UAVIONIX_UCP_CAPTURE_ALL_RX_PACKETS
2021-07-27 19:48:53 -03:00
case GDL90_ID_TRANSPONDER_CONTROL :
case GDL90_ID_GPS_DATA :
case GDL90_ID_MESSAGE_REQUEST :
// not handled, outbound only
break ;
default :
//GCS_SEND_TEXT(MAV_SEVERITY_DEBUG,"ADSB:Unknown msg %d", (int)msg.messageId);
break ;
}
}
const char * AP_ADSB_uAvionix_UCP : : get_hardware_name ( const uint8_t hwId )
{
switch ( hwId ) {
case 0x09 : return " Ping200s " ;
case 0x0A : return " Ping20s " ;
case 0x18 : return " Ping200C " ;
case 0x27 : return " Ping20Z " ;
case 0x2D : return " SkyBeaconX " ; // (certified)
case 0x26 : //return "Ping200Z/Ping200X"; // (uncertified). Let's fallthrough and use Ping200X
case 0x2F : return " Ping200X " ; // (certified)
case 0x30 : return " TailBeaconX " ; // (certified)
} // switch hwId
return " Unknown HW " ;
}
void AP_ADSB_uAvionix_UCP : : send_Transponder_Control ( )
{
GDL90_TRANSPONDER_CONTROL_MSG msg { } ;
msg . messageId = GDL90_ID_TRANSPONDER_CONTROL ;
2021-10-13 20:31:14 -03:00
msg . version = GDL90_TRANSPONDER_CONTROL_VERSION ;
2021-07-27 19:48:53 -03:00
# if CONFIG_HAL_BOARD == HAL_BOARD_SITL
// when using the simulator, always declare we're on the ground to help
// inhibit chaos if this ias actually being broadcasted on real hardware
msg . airGroundState = ADSB_ON_GROUND ;
# elif AP_ADSB_UAVIONIX_DETECT_GROUNDSTATE
msg . airGroundState = _frontend . out_state . is_flying ? ADSB_AIRBORNE_SUBSONIC : ADSB_ON_GROUND ;
# else
msg . airGroundState = ADSB_AIRBORNE_SUBSONIC ;
# endif
msg . baroCrossChecked = ADSB_NIC_BARO_UNVERIFIED ;
2021-10-13 20:31:14 -03:00
msg . identActive = _frontend . out_state . ctrl . identActive ;
_frontend . out_state . ctrl . identActive = false ; // only send identButtonActive once per request
msg . modeAEnabled = _frontend . out_state . ctrl . modeAEnabled ;
msg . modeCEnabled = _frontend . out_state . ctrl . modeCEnabled ;
2024-11-27 20:42:02 -04:00
msg . modeSEnabled = ( _frontend . _options & uint32_t ( AP_ADSB : : AdsbOption : : Mode3_Only ) ) ? 0 : _frontend . out_state . ctrl . modeSEnabled ;
msg . es1090TxEnabled = ( _frontend . _options & uint32_t ( AP_ADSB : : AdsbOption : : Mode3_Only ) ) ? 0 : _frontend . out_state . ctrl . es1090TxEnabled ;
2021-07-27 19:48:53 -03:00
// if enabled via param ADSB_OPTIONS, use squawk 7400 while in any Loss-Comms related failsafe
// https://www.faa.gov/documentLibrary/media/Notice/N_JO_7110.724_5-2-9_UAS_Lost_Link_2.pdf
const AP_Notify & notify = AP : : notify ( ) ;
if ( ( ( _frontend . _options & uint32_t ( AP_ADSB : : AdsbOption : : Squawk_7400_FS_RC ) ) & & notify . flags . failsafe_radio ) | |
( ( _frontend . _options & uint32_t ( AP_ADSB : : AdsbOption : : Squawk_7400_FS_GCS ) ) & & notify . flags . failsafe_gcs ) ) {
msg . squawkCode = 7400 ;
} else {
2021-10-13 20:31:14 -03:00
msg . squawkCode = _frontend . out_state . ctrl . squawkCode ;
2021-07-27 19:48:53 -03:00
}
# if AP_ADSB_UAVIONIX_EMERGENCY_STATUS_ON_LOST_LINK
const uint32_t last_gcs_ms = gcs ( ) . sysid_myggcs_last_seen_time_ms ( ) ;
const bool gcs_lost_comms = ( last_gcs_ms ! = 0 ) & & ( AP_HAL : : millis ( ) - last_gcs_ms > AP_ADSB_UAVIONIX_GCS_LOST_COMMS_LONG_TIMEOUT_MS ) ;
msg . emergencyState = gcs_lost_comms ? ADSB_EMERGENCY_STATUS : : ADSB_EMERGENCY_UAS_LOST_LINK : ADSB_EMERGENCY_STATUS : : ADSB_EMERGENCY_NONE ;
# else
msg . emergencyState = ADSB_EMERGENCY_STATUS : : ADSB_EMERGENCY_NONE ;
# endif
2021-10-13 20:31:14 -03:00
# if GDL90_TRANSPONDER_CONTROL_VERSION == 2
msg . x_bit = 0 ;
# endif
memcpy ( msg . callsign , _frontend . out_state . ctrl . callsign , sizeof ( msg . callsign ) ) ;
2021-07-27 19:48:53 -03:00
gdl90Transmit ( ( GDL90_TX_MESSAGE & ) msg , sizeof ( msg ) ) ;
}
void AP_ADSB_uAvionix_UCP : : send_GPS_Data ( )
{
GDL90_GPS_DATA_V2 msg { } ;
msg . messageId = GDL90_ID_GPS_DATA ;
msg . version = 2 ;
2023-10-08 03:31:11 -03:00
const AP_ADSB : : Loc & gps { _frontend . _my_loc } ;
2021-07-27 19:48:53 -03:00
const GPS_FIX fix = ( GPS_FIX ) gps . status ( ) ;
const bool fix_is_good = ( fix > = GPS_FIX_3D ) ;
const Vector3f velocity = fix_is_good ? gps . velocity ( ) : Vector3f ( ) ;
msg . utcTime_s = gps . time_epoch_usec ( ) * 1E-6 ;
msg . latitude_ddE7 = fix_is_good ? _frontend . _my_loc . lat : INT32_MAX ;
msg . longitude_ddE7 = fix_is_good ? _frontend . _my_loc . lng : INT32_MAX ;
2021-09-21 16:12:43 -03:00
msg . altitudeGnss_mm = fix_is_good ? ( _frontend . _my_loc . alt * 10 ) : INT32_MAX ;
2021-07-27 19:48:53 -03:00
// Protection Limits. FD or SBAS-based depending on state bits
2024-10-09 15:31:38 -03:00
// Estimate HPL based on horizontal accuracy/HFOM to a probability of 10^-7:
// Using the central limit theorem for a Gaussian distribution,
// this is 5.326724*stdDev.
// Conservatively round up to 6 as a scaling factor,
// and asssume HFOM of 95% was calculated as 2*stdDev*HDOP.
// This yields a factor of 3 to estimate HPL from horizontal accuracy.
float accHoriz ;
bool gotAccHoriz = gps . horizontal_accuracy ( accHoriz ) ;
msg . HPL_mm = gotAccHoriz ? 3 * accHoriz * 1E3 : UINT32_MAX ; // required to calculate NIC
msg . VPL_cm = UINT32_MAX ; // unused by ping200X
2021-07-27 19:48:53 -03:00
// Figure of Merits
2024-10-09 15:31:38 -03:00
msg . horizontalFOM_mm = gotAccHoriz ? accHoriz * 1E3 : UINT32_MAX ;
2021-07-27 19:48:53 -03:00
float accVert ;
msg . verticalFOM_cm = gps . vertical_accuracy ( accVert ) ? accVert * 1E2 : UINT16_MAX ;
float accVel ;
msg . horizontalVelocityFOM_mmps = gps . speed_accuracy ( accVel ) ? accVel * 1E3 : UINT16_MAX ;
msg . verticalVelocityFOM_mmps = msg . horizontalVelocityFOM_mmps ;
// Velocities
2021-10-27 17:44:09 -03:00
msg . verticalVelocity_cmps = fix_is_good ? - 1.0f * velocity . z * 1E2 : INT16_MAX ;
2021-07-27 19:48:53 -03:00
msg . northVelocity_mmps = fix_is_good ? velocity . x * 1E3 : INT32_MAX ;
msg . eastVelocity_mmps = fix_is_good ? velocity . y * 1E3 : INT32_MAX ;
// State
msg . fixType = fix ;
GDL90_GPS_NAV_STATE nav_state { } ;
nav_state . HPLfdeActive = 1 ;
nav_state . fault = 0 ;
nav_state . HrdMagNorth = 0 ; // 1 means "north" is magnetic north
msg . navState = nav_state ;
2023-10-08 03:31:11 -03:00
msg . satsUsed = gps . num_sats ( ) ;
2021-07-27 19:48:53 -03:00
gdl90Transmit ( ( GDL90_TX_MESSAGE & ) msg , sizeof ( msg ) ) ;
}
bool AP_ADSB_uAvionix_UCP : : hostTransmit ( uint8_t * buffer , uint16_t length )
{
if ( _port = = nullptr | | _port - > txspace ( ) < length ) {
return false ;
}
_port - > write ( buffer , length ) ;
return true ;
}
bool AP_ADSB_uAvionix_UCP : : request_msg ( const GDL90_MESSAGE_ID msg_id )
{
2021-10-13 20:31:14 -03:00
const GDL90_TRANSPONDER_MESSAGE_REQUEST_V2 msg = {
2021-07-27 19:48:53 -03:00
messageId : GDL90_ID_MESSAGE_REQUEST ,
version : 2 ,
reqMsgId : msg_id
} ;
return gdl90Transmit ( ( GDL90_TX_MESSAGE & ) msg , sizeof ( msg ) ) ! = 0 ;
}
uint16_t AP_ADSB_uAvionix_UCP : : gdl90Transmit ( GDL90_TX_MESSAGE & message , const uint16_t length )
{
uint8_t gdl90FrameBuffer [ GDL90_TX_MAX_FRAME_LENGTH ] { } ;
const uint16_t frameCrc = crc16_ccitt_GDL90 ( ( uint8_t * ) & message . raw , length , 0 ) ;
// Set flag byte in frame buffer
gdl90FrameBuffer [ 0 ] = GDL90_FLAG_BYTE ;
uint16_t frameIndex = 1 ;
2024-10-17 11:52:25 -03:00
2021-07-27 19:48:53 -03:00
// Copy and stuff all payload bytes into frame buffer
for ( uint16_t i = 0 ; i < length + 2 ; i + + ) {
// Check for overflow of frame buffer
if ( frameIndex > = GDL90_TX_MAX_FRAME_LENGTH ) {
return 0 ;
}
2024-10-17 11:52:25 -03:00
2021-07-27 19:48:53 -03:00
uint8_t data ;
// Append CRC to payload
if ( i = = length ) {
data = LOWBYTE ( frameCrc ) ;
} else if ( i = = length + 1 ) {
data = HIGHBYTE ( frameCrc ) ;
} else {
2024-10-17 11:52:25 -03:00
data = message . raw [ i ] ;
2021-07-27 19:48:53 -03:00
}
if ( data = = GDL90_FLAG_BYTE | | data = = GDL90_CONTROL_ESCAPE_BYTE ) {
// Check for frame buffer overflow on stuffed byte
if ( frameIndex + 2 > GDL90_TX_MAX_FRAME_LENGTH ) {
return 0 ;
}
2024-10-17 11:52:25 -03:00
2021-07-27 19:48:53 -03:00
// Set control break and stuff this byte
gdl90FrameBuffer [ frameIndex + + ] = GDL90_CONTROL_ESCAPE_BYTE ;
gdl90FrameBuffer [ frameIndex + + ] = data ^ GDL90_STUFF_BYTE ;
} else {
gdl90FrameBuffer [ frameIndex + + ] = data ;
}
}
2024-10-17 11:52:25 -03:00
2021-07-27 19:48:53 -03:00
// Add end of frame indication
gdl90FrameBuffer [ frameIndex + + ] = GDL90_FLAG_BYTE ;
// Push packet to UART
if ( hostTransmit ( gdl90FrameBuffer , frameIndex ) ) {
return frameIndex ;
}
2024-10-17 11:52:25 -03:00
2021-07-27 19:48:53 -03:00
return 0 ;
}
bool AP_ADSB_uAvionix_UCP : : parseByte ( const uint8_t data , GDL90_RX_MESSAGE & msg , GDL90_RX_STATUS & status )
{
switch ( status . state )
{
case GDL90_RX_IDLE :
if ( data = = GDL90_FLAG_BYTE & & status . prev_data = = GDL90_FLAG_BYTE ) {
status . length = 0 ;
status . state = GDL90_RX_IN_PACKET ;
}
break ;
case GDL90_RX_IN_PACKET :
if ( data = = GDL90_CONTROL_ESCAPE_BYTE ) {
status . state = GDL90_RX_UNSTUFF ;
} else if ( data = = GDL90_FLAG_BYTE ) {
// packet complete! Check CRC and restart packet cycle on all pass or fail scenarios
status . state = GDL90_RX_IDLE ;
if ( status . length < GDL90_OVERHEAD_LENGTH ) {
// something is wrong, there's no actual data
return false ;
}
const uint8_t crc_LSB = msg . raw [ status . length - 2 ] ;
const uint8_t crc_MSB = msg . raw [ status . length - 1 ] ;
// NOTE: status.length contains messageId, payload and CRC16. So status.length-3 is effective payload length
msg . crc = ( uint16_t ) crc_LSB | ( ( uint16_t ) crc_MSB < < 8 ) ;
const uint16_t crc = crc16_ccitt_GDL90 ( ( uint8_t * ) & msg . raw , status . length - 2 , 0 ) ;
if ( crc = = msg . crc ) {
status . prev_data = data ;
// NOTE: this is the only path that returns true
return true ;
}
} else if ( status . length < GDL90_RX_MAX_PACKET_LENGTH ) {
msg . raw [ status . length + + ] = data ;
} else {
status . state = GDL90_RX_IDLE ;
}
break ;
case GDL90_RX_UNSTUFF :
msg . raw [ status . length + + ] = data ^ GDL90_STUFF_BYTE ;
status . state = GDL90_RX_IN_PACKET ;
break ;
}
status . prev_data = data ;
return false ;
}
# endif // HAL_ADSB_UCP_ENABLED