2022-07-07 03:18:47 -03:00
/*
* This file 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 file 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/>.
*/
/*
*
* original code by :
* BlueMark Innovations BV , Roel Schiphorst
* Contributors : Tom Pittenger , Josh Henderson , Andrew Tridgell
* Parts of this code are based on / copied from the Open Drone ID project https : //github.com/opendroneid/opendroneid-core-c
*
* The code has been tested with the BlueMark DroneBeacon MAVLink transponder running this command in the ArduPlane folder :
* sim_vehicle . py - - console - - map - A - - serial1 = uart : / dev / ttyUSB1 : 9600
* ( and a DroneBeacon MAVLink transponder connected to ttyUSB1 )
*
* See https : //github.com/ArduPilot/ArduRemoteID for an open implementation of a transmitter module on serial
* and DroneCAN
*/
# include "AP_OpenDroneID.h"
# if AP_OPENDRONEID_ENABLED
# include <AP_HAL/AP_HAL.h>
# include <GCS_MAVLink/GCS.h>
# include <GCS_MAVLink/GCS_MAVLink.h>
# include <GCS_MAVLink/GCS.h>
# include <AP_GPS/AP_GPS.h>
# include <AP_Baro/AP_Baro.h>
# include <AP_AHRS/AP_AHRS.h>
extern const AP_HAL : : HAL & hal ;
const AP_Param : : GroupInfo AP_OpenDroneID : : var_info [ ] = {
// @Param: ENABLE
// @DisplayName: Enable ODID subsystem
// @Description: Enable ODID subsystem
// @Values: 0:Disabled,1:Enabled
AP_GROUPINFO_FLAGS ( " ENABLE " , 1 , AP_OpenDroneID , _enable , 0 , AP_PARAM_FLAG_ENABLE ) ,
// @Param: MAVPORT
// @DisplayName: MAVLink serial port
// @Description: Serial port number to send OpenDroneID MAVLink messages to. Can be -1 if using DroneCAN.
// @Values: -1:Disabled,0:Serial0,1:Serial1,2:Serial2,3:Serial3,4:Serial4,5:Serial5,6:Serial6
AP_GROUPINFO ( " MAVPORT " , 2 , AP_OpenDroneID , _mav_port , - 1 ) ,
// @Param: CANDRIVER
// @DisplayName: DroneCAN driver number
// @Description: DroneCAN driver index, 0 to disable DroneCAN
// @Values: 0:Disabled,1:Driver1,2:Driver2
AP_GROUPINFO ( " CANDRIVER " , 3 , AP_OpenDroneID , _can_driver , 0 ) ,
// @Param: OPTIONS
// @DisplayName: OpenDroneID options
// @Description: Options for OpenDroneID subsystem. Bit 0 means to enforce arming checks
// @Bitmask: 0:EnforceArming
AP_GROUPINFO ( " OPTIONS " , 4 , AP_OpenDroneID , _options , 0 ) ,
// @Param: BARO_ACC
// @DisplayName: Barometer vertical accuraacy
// @Description: Barometer Vertical Accuracy when installed in the vehicle. Note this is dependent upon installation conditions and thus disabled by default
// @Units: m
// @User: Advanced
AP_GROUPINFO ( " BARO_ACC " , 5 , AP_OpenDroneID , _baro_accuracy , - 1.0 ) ,
AP_GROUPEND
} ;
// constructor
AP_OpenDroneID : : AP_OpenDroneID ( )
{
# if CONFIG_HAL_BOARD == HAL_BOARD_SITL
if ( _singleton ! = nullptr ) {
AP_HAL : : panic ( " OpenDroneID must be singleton " ) ;
}
# endif
_singleton = this ;
AP_Param : : setup_object_defaults ( this , var_info ) ;
}
void AP_OpenDroneID : : init ( )
{
if ( _enable = = 0 ) {
return ;
}
_chan = mavlink_channel_t ( gcs ( ) . get_channel_from_port_number ( _mav_port ) ) ;
}
// Perform the pre-arm checks and prevent arming if they are not satisifed
// Except in the case of an in-flight reboot
bool AP_OpenDroneID : : pre_arm_check ( char * failmsg , uint8_t failmsg_len )
{
WITH_SEMAPHORE ( _sem ) ;
if ( ! option_enabled ( Options : : EnforceArming ) ) {
return true ;
}
if ( pkt_basic_id . id_type = = MAV_ODID_ID_TYPE_NONE ) {
strncpy ( failmsg , " UA_TYPE required in BasicID " , failmsg_len ) ;
return false ;
}
if ( pkt_system . operator_latitude = = 0 & & pkt_system . operator_longitude = = 0 ) {
strncpy ( failmsg , " operator location must be set " , failmsg_len ) ;
return false ;
}
const uint32_t max_age_ms = 3000 ;
const uint32_t now_ms = AP_HAL : : millis ( ) ;
if ( last_arm_status_ms = = 0 | | now_ms - last_arm_status_ms > max_age_ms ) {
strncpy ( failmsg , " ARM_STATUS not available " , failmsg_len ) ;
return false ;
}
if ( last_system_ms = = 0 | | now_ms - last_system_ms > max_age_ms ) {
strncpy ( failmsg , " SYSTEM not available " , failmsg_len ) ;
return false ;
}
if ( arm_status . status ! = MAV_ODID_GOOD_TO_ARM ) {
strncpy ( failmsg , arm_status . error , failmsg_len ) ;
return false ;
}
return true ;
}
void AP_OpenDroneID : : update ( )
{
if ( _enable = = 0 ) {
return ;
}
const uint32_t now = AP_HAL : : millis ( ) ;
const bool armed = hal . util - > get_soft_armed ( ) ;
if ( armed & & ! _was_armed ) {
// use arm location as takeoff location
AP : : ahrs ( ) . get_location ( _takeoff_location ) ;
}
_was_armed = armed ;
if ( now - _last_send_dynamic_messages_ms > = _mavlink_dynamic_period_ms ) {
_last_send_dynamic_messages_ms = now ;
send_dynamic_out ( ) ;
}
send_static_out ( ) ;
}
void AP_OpenDroneID : : send_dynamic_out ( )
{
send_location_message ( ) ;
}
void AP_OpenDroneID : : send_static_out ( )
{
const uint32_t now_ms = AP_HAL : : millis ( ) ;
// we need to notify user if we lost the transmitter
2022-08-09 01:58:47 -03:00
if ( now_ms - last_arm_status_ms > 5000 & & now_ms - last_lost_tx_ms > 5000 ) {
last_lost_tx_ms = now_ms ;
2022-07-07 03:18:47 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_WARNING , " ODID: lost transmitter " ) ;
}
const uint32_t msg_spacing_ms = _mavlink_dynamic_period_ms / 4 ;
if ( now_ms - last_msg_send_ms > = msg_spacing_ms ) {
// allow update of channel during setup, this makes it easy to debug with a GCS
_chan = mavlink_channel_t ( gcs ( ) . get_channel_from_port_number ( _mav_port ) ) ;
last_msg_send_ms = now_ms ;
switch ( next_msg_to_send ) {
case NEXT_MSG_BASIC_ID :
send_basic_id_message ( ) ;
break ;
case NEXT_MSG_SYSTEM :
send_system_message ( ) ;
break ;
case NEXT_MSG_SELF_ID :
send_self_id_message ( ) ;
break ;
case NEXT_MSG_OPERATOR_ID :
send_operator_id_message ( ) ;
break ;
case NEXT_MSG_ENUM_END :
break ;
}
next_msg_to_send = next_msg ( ( uint8_t ( next_msg_to_send ) + 1 ) % uint8_t ( NEXT_MSG_ENUM_END ) ) ;
}
}
// The send_location_message
// all open_drone_id send functions use data stored in the open drone id class.
//This location send function is an exception. It uses live location data from the ArduPilot system.
void AP_OpenDroneID : : send_location_message ( )
{
auto & ahrs = AP : : ahrs ( ) ;
const auto & barometer = AP : : baro ( ) ;
const auto & gps = AP : : gps ( ) ;
const AP_GPS : : GPS_Status gps_status = gps . status ( ) ;
const bool got_bad_gps_fix = ( gps_status < AP_GPS : : GPS_Status : : GPS_OK_FIX_3D ) ;
Location current_location ;
if ( ! ahrs . get_location ( current_location ) ) {
return ;
}
const uint8_t uav_status = hal . util - > get_soft_armed ( ) ? MAV_ODID_STATUS_AIRBORNE : MAV_ODID_STATUS_GROUND ;
float direction = ODID_INV_DIR ;
if ( ! got_bad_gps_fix ) {
direction = wrap_360 ( degrees ( ahrs . groundspeed_vector ( ) . angle ( ) ) ) ; // heading (degrees)
}
const float speed_horizontal = create_speed_horizontal ( ahrs . groundspeed ( ) ) ;
Vector3f velNED ;
UNUSED_RESULT ( ahrs . get_velocity_NED ( velNED ) ) ;
const float climb_rate = create_speed_vertical ( - velNED . z ) ; //make sure climb_rate is within Remote ID limit
int32_t latitude = 0 ;
int32_t longitude = 0 ;
if ( current_location . check_latlng ( ) ) { //set location if they are valid
latitude = current_location . lat ;
longitude = current_location . lng ;
}
// altitude referenced against 1013.2mb
const float base_press_mbar = 1013.2 ;
const float altitude_barometric = create_altitude ( barometer . get_altitude_difference ( base_press_mbar * 100 , barometer . get_pressure ( ) ) ) ;
float altitude_geodetic = - 1000 ;
int32_t alt_amsl_cm ;
float undulation ;
if ( current_location . get_alt_cm ( Location : : AltFrame : : ABSOLUTE , alt_amsl_cm ) ) {
altitude_geodetic = alt_amsl_cm * 0.01 ;
}
if ( gps . get_undulation ( undulation ) ) {
altitude_geodetic - = undulation ;
}
// Compute the current height above the takeoff location
float height_above_takeoff = 0 ; // height above takeoff (meters)
if ( hal . util - > get_soft_armed ( ) ) {
int32_t curr_alt_asml_cm ;
int32_t takeoff_alt_asml_cm ;
if ( current_location . get_alt_cm ( Location : : AltFrame : : ABSOLUTE , curr_alt_asml_cm ) & &
_takeoff_location . get_alt_cm ( Location : : AltFrame : : ABSOLUTE , takeoff_alt_asml_cm ) ) {
height_above_takeoff = ( curr_alt_asml_cm - takeoff_alt_asml_cm ) * 0.01 ;
}
}
// Accuracy
// If we have GPS 3D lock we presume that the accuracies of the system will track the GPS's reported accuracy
MAV_ODID_HOR_ACC horizontal_accuracy_mav = MAV_ODID_HOR_ACC_UNKNOWN ;
MAV_ODID_VER_ACC vertical_accuracy_mav = MAV_ODID_VER_ACC_UNKNOWN ;
MAV_ODID_SPEED_ACC speed_accuracy_mav = MAV_ODID_SPEED_ACC_UNKNOWN ;
MAV_ODID_TIME_ACC timestamp_accuracy_mav = MAV_ODID_TIME_ACC_UNKNOWN ;
float horizontal_accuracy ;
if ( gps . horizontal_accuracy ( horizontal_accuracy ) ) {
horizontal_accuracy_mav = create_enum_horizontal_accuracy ( horizontal_accuracy ) ;
}
float vertical_accuracy ;
if ( gps . vertical_accuracy ( vertical_accuracy ) ) {
vertical_accuracy_mav = create_enum_vertical_accuracy ( vertical_accuracy ) ;
}
float speed_accuracy ;
if ( gps . speed_accuracy ( speed_accuracy ) ) {
speed_accuracy_mav = create_enum_speed_accuracy ( speed_accuracy ) ;
}
// if we have ever had GPS lock then we will have better than 1s
// accuracy, as we use system timer to propogate time
timestamp_accuracy_mav = create_enum_timestamp_accuracy ( 1.0 ) ;
// Barometer altitude accuraacy will be highly dependent on the airframe and installation of the barometer in use
// thus ArduPilot cannot reasonably fill this in.
// Instead allow a manufacturer to use a parameter to fill this in
uint8_t barometer_accuracy = MAV_ODID_VER_ACC_UNKNOWN ; //ahrs class does not provide accuracy readings
if ( ! is_equal ( _baro_accuracy . get ( ) , - 1.0f ) ) {
barometer_accuracy = create_enum_vertical_accuracy ( _baro_accuracy ) ;
}
// Timestamp here is the number of seconds after into the current hour referenced to UTC time (up to one hour)
// FIX we need to only set this if w have a GPS lock is 2D good enough for that?
float timestamp = ODID_INV_TIMESTAMP ;
if ( ! got_bad_gps_fix ) {
uint32_t time_week_ms = gps . time_week_ms ( ) ;
timestamp = float ( time_week_ms % ( 3600 * 1000 ) ) * 0.001 ;
timestamp = create_location_timestamp ( timestamp ) ; //make sure timestamp is within Remote ID limit
}
{
WITH_SEMAPHORE ( _sem ) ;
// take semaphore so CAN gets a consistent packet
2022-08-10 20:20:29 -03:00
pkt_location = mavlink_open_drone_id_location_t {
2022-07-07 03:18:47 -03:00
latitude : latitude ,
longitude : longitude ,
altitude_barometric : altitude_barometric ,
altitude_geodetic : altitude_geodetic ,
height : height_above_takeoff ,
timestamp : timestamp ,
direction : uint16_t ( direction * 100.0 ) , // Heading (centi-degrees)
speed_horizontal : uint16_t ( speed_horizontal * 100.0 ) , // Ground speed (cm/s)
speed_vertical : int16_t ( climb_rate * 100.0 ) , // Climb rate (cm/s)
target_system : 0 ,
target_component : 0 ,
status : uint8_t ( uav_status ) ,
height_reference : MAV_ODID_HEIGHT_REF_OVER_TAKEOFF , // height reference enum: Above takeoff location or above ground
horizontal_accuracy : uint8_t ( horizontal_accuracy_mav ) ,
vertical_accuracy : uint8_t ( vertical_accuracy_mav ) ,
barometer_accuracy : barometer_accuracy ,
speed_accuracy : uint8_t ( speed_accuracy_mav ) ,
timestamp_accuracy : uint8_t ( timestamp_accuracy_mav )
} ;
need_send_location = dronecan_send_all ;
}
if ( _chan ! = MAV_CHAN_INVALID ) {
mavlink_msg_open_drone_id_location_send_struct ( _chan , & pkt_location ) ;
}
}
void AP_OpenDroneID : : send_basic_id_message ( )
{
// note that packet is filled in by the GCS
need_send_basic_id | = dronecan_send_all ;
if ( _chan ! = MAV_CHAN_INVALID ) {
mavlink_msg_open_drone_id_basic_id_send_struct ( _chan , & pkt_basic_id ) ;
}
}
void AP_OpenDroneID : : send_system_message ( )
{
// note that packet is filled in by the GCS
need_send_system | = dronecan_send_all ;
if ( _chan ! = MAV_CHAN_INVALID ) {
mavlink_msg_open_drone_id_system_send_struct ( _chan , & pkt_system ) ;
}
}
void AP_OpenDroneID : : send_self_id_message ( )
{
need_send_self_id | = dronecan_send_all ;
// note that packet is filled in by the GCS
if ( _chan ! = MAV_CHAN_INVALID ) {
mavlink_msg_open_drone_id_self_id_send_struct ( _chan , & pkt_self_id ) ;
}
}
void AP_OpenDroneID : : send_operator_id_message ( )
{
need_send_operator_id | = dronecan_send_all ;
// note that packet is filled in by the GCS
if ( _chan ! = MAV_CHAN_INVALID ) {
mavlink_msg_open_drone_id_operator_id_send_struct ( _chan , & pkt_operator_id ) ;
}
}
/*
* This converts a horizontal accuracy float value to the corresponding enum
*
* @ param Accuracy The horizontal accuracy in meters
* @ return Enum value representing the accuracy
*/
MAV_ODID_HOR_ACC AP_OpenDroneID : : create_enum_horizontal_accuracy ( float accuracy ) const
{
// Out of bounds return UKNOWN flag
if ( accuracy < 0.0 | | accuracy > = 18520.0 ) {
return MAV_ODID_HOR_ACC_UNKNOWN ;
}
static const struct {
float accuracy ; // Accuracy bound in meters
MAV_ODID_HOR_ACC mavoutput ; // mavlink enum output
} horiz_accuracy_table [ ] = {
{ 1.0 , MAV_ODID_HOR_ACC_1_METER } ,
{ 3.0 , MAV_ODID_HOR_ACC_3_METER } ,
{ 10.0 , MAV_ODID_HOR_ACC_10_METER } ,
{ 30.0 , MAV_ODID_HOR_ACC_30_METER } ,
{ 92.6 , MAV_ODID_HOR_ACC_0_05NM } ,
{ 185.2 , MAV_ODID_HOR_ACC_0_1NM } ,
{ 555.6 , MAV_ODID_HOR_ACC_0_3NM } ,
{ 926.0 , MAV_ODID_HOR_ACC_0_5NM } ,
{ 1852.0 , MAV_ODID_HOR_ACC_1NM } ,
{ 3704.0 , MAV_ODID_HOR_ACC_2NM } ,
{ 7408.0 , MAV_ODID_HOR_ACC_4NM } ,
{ 18520.0 , MAV_ODID_HOR_ACC_10NM } ,
} ;
for ( auto elem : horiz_accuracy_table ) {
if ( accuracy < elem . accuracy ) {
return elem . mavoutput ;
}
}
// Should not reach this
return MAV_ODID_HOR_ACC_UNKNOWN ;
}
/**
* This converts a vertical accuracy float value to the corresponding enum
*
* @ param Accuracy The vertical accuracy in meters
* @ return Enum value representing the accuracy
*/
MAV_ODID_VER_ACC AP_OpenDroneID : : create_enum_vertical_accuracy ( float accuracy ) const
{
// Out of bounds return UKNOWN flag
if ( accuracy < 0.0 | | accuracy > = 150.0 ) {
return MAV_ODID_VER_ACC_UNKNOWN ;
}
static const struct {
float accuracy ; // Accuracy bound in meters
MAV_ODID_VER_ACC mavoutput ; // mavlink enum output
} vertical_accuracy_table [ ] = {
{ 1.0 , MAV_ODID_VER_ACC_1_METER } ,
{ 3.0 , MAV_ODID_VER_ACC_3_METER } ,
{ 10.0 , MAV_ODID_VER_ACC_10_METER } ,
{ 25.0 , MAV_ODID_VER_ACC_25_METER } ,
{ 45.0 , MAV_ODID_VER_ACC_45_METER } ,
{ 150.0 , MAV_ODID_VER_ACC_150_METER } ,
} ;
for ( auto elem : vertical_accuracy_table ) {
if ( accuracy < elem . accuracy ) {
return elem . mavoutput ;
}
}
// Should not reach this
return MAV_ODID_VER_ACC_UNKNOWN ;
}
/**
* This converts a speed accuracy float value to the corresponding enum
*
* @ param Accuracy The speed accuracy in m / s
* @ return Enum value representing the accuracy
*/
MAV_ODID_SPEED_ACC AP_OpenDroneID : : create_enum_speed_accuracy ( float accuracy ) const
{
// Out of bounds return UKNOWN flag
if ( accuracy < 0.0 | | accuracy > = 10.0 ) {
return MAV_ODID_SPEED_ACC_UNKNOWN ;
}
if ( accuracy < 0.3 ) {
return MAV_ODID_SPEED_ACC_0_3_METERS_PER_SECOND ;
} else if ( accuracy < 1.0 ) {
return MAV_ODID_SPEED_ACC_1_METERS_PER_SECOND ;
} else if ( accuracy < 3.0 ) {
return MAV_ODID_SPEED_ACC_3_METERS_PER_SECOND ;
} else if ( accuracy < 10.0 ) {
return MAV_ODID_SPEED_ACC_10_METERS_PER_SECOND ;
}
// Should not reach this
return MAV_ODID_SPEED_ACC_UNKNOWN ;
}
/**
* This converts a timestamp accuracy float value to the corresponding enum
*
* @ param Accuracy The timestamp accuracy in seconds
* @ return Enum value representing the accuracy
*/
MAV_ODID_TIME_ACC AP_OpenDroneID : : create_enum_timestamp_accuracy ( float accuracy ) const
{
// Out of bounds return UKNOWN flag
if ( accuracy < 0.0 | | accuracy > = 1.5 ) {
return MAV_ODID_TIME_ACC_UNKNOWN ;
}
static const MAV_ODID_TIME_ACC mavoutput [ 15 ] = {
MAV_ODID_TIME_ACC_0_1_SECOND ,
MAV_ODID_TIME_ACC_0_2_SECOND ,
MAV_ODID_TIME_ACC_0_3_SECOND ,
MAV_ODID_TIME_ACC_0_4_SECOND ,
MAV_ODID_TIME_ACC_0_5_SECOND ,
MAV_ODID_TIME_ACC_0_6_SECOND ,
MAV_ODID_TIME_ACC_0_7_SECOND ,
MAV_ODID_TIME_ACC_0_8_SECOND ,
MAV_ODID_TIME_ACC_0_9_SECOND ,
MAV_ODID_TIME_ACC_1_0_SECOND ,
MAV_ODID_TIME_ACC_1_1_SECOND ,
MAV_ODID_TIME_ACC_1_2_SECOND ,
MAV_ODID_TIME_ACC_1_3_SECOND ,
MAV_ODID_TIME_ACC_1_4_SECOND ,
MAV_ODID_TIME_ACC_1_5_SECOND ,
} ;
for ( int8_t i = 1 ; i < = 15 ; i + + ) {
if ( accuracy < = 0.1 * i ) {
return mavoutput [ i - 1 ] ;
}
}
// Should not reach this
return MAV_ODID_TIME_ACC_UNKNOWN ;
}
// make sure value is within limits of remote ID standard
uint16_t AP_OpenDroneID : : create_speed_horizontal ( uint16_t speed ) const
{
if ( speed > ODID_MAX_SPEED_H ) { // constraint function can't be used, because out of range value is invalid
speed = ODID_INV_SPEED_H ;
}
return speed ;
}
// make sure value is within limits of remote ID standard
int16_t AP_OpenDroneID : : create_speed_vertical ( int16_t speed ) const
{
if ( speed > ODID_MAX_SPEED_V ) { // constraint function can't be used, because out of range value is invalid
speed = ODID_INV_SPEED_V ;
} else if ( speed < ODID_MIN_SPEED_V ) {
speed = ODID_INV_SPEED_V ;
}
return speed ;
}
// make sure value is within limits of remote ID standard
float AP_OpenDroneID : : create_altitude ( float altitude ) const
{
if ( altitude > ODID_MAX_ALT ) { // constraint function can't be used, because out of range value is invalid
altitude = ODID_INV_ALT ;
} else if ( altitude < ODID_MIN_ALT ) {
altitude = ODID_INV_ALT ;
}
return altitude ;
}
// make sure value is within limits of remote ID standard
float AP_OpenDroneID : : create_location_timestamp ( float timestamp ) const
{
if ( timestamp > ODID_MAX_TIMESTAMP ) { // constraint function can't be used, because out of range value is invalid
timestamp = ODID_INV_TIMESTAMP ;
} else if ( timestamp < 0 ) {
timestamp = ODID_INV_TIMESTAMP ;
}
return timestamp ;
}
// handle a message from the GCS
void AP_OpenDroneID : : handle_msg ( mavlink_channel_t chan , const mavlink_message_t & msg )
{
WITH_SEMAPHORE ( _sem ) ;
switch ( msg . msgid ) {
// only accept ARM_STATUS from the transmitter
case MAVLINK_MSG_ID_OPEN_DRONE_ID_ARM_STATUS : {
if ( chan = = _chan ) {
mavlink_msg_open_drone_id_arm_status_decode ( & msg , & arm_status ) ;
last_arm_status_ms = AP_HAL : : millis ( ) ;
}
break ;
}
// accept other messages from the GCS
case MAVLINK_MSG_ID_OPEN_DRONE_ID_OPERATOR_ID :
mavlink_msg_open_drone_id_operator_id_decode ( & msg , & pkt_operator_id ) ;
break ;
case MAVLINK_MSG_ID_OPEN_DRONE_ID_SELF_ID :
mavlink_msg_open_drone_id_self_id_decode ( & msg , & pkt_self_id ) ;
break ;
case MAVLINK_MSG_ID_OPEN_DRONE_ID_BASIC_ID :
mavlink_msg_open_drone_id_basic_id_decode ( & msg , & pkt_basic_id ) ;
break ;
case MAVLINK_MSG_ID_OPEN_DRONE_ID_SYSTEM :
mavlink_msg_open_drone_id_system_decode ( & msg , & pkt_system ) ;
last_system_ms = AP_HAL : : millis ( ) ;
break ;
}
}
// singleton instance
AP_OpenDroneID * AP_OpenDroneID : : _singleton ;
namespace AP
{
AP_OpenDroneID & opendroneid ( )
{
return * AP_OpenDroneID : : get_singleton ( ) ;
}
}
# endif //AP_OPENDRONEID_ENABLED