236 lines
7.7 KiB
C++
236 lines
7.7 KiB
C++
/*
|
|
* 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/>.
|
|
*/
|
|
/*
|
|
DroneCAN support for OpenDroneID
|
|
*/
|
|
|
|
#include "AP_OpenDroneID.h"
|
|
|
|
#if AP_OPENDRONEID_ENABLED
|
|
|
|
#include <AP_DroneCAN/AP_DroneCAN.h>
|
|
|
|
#if HAL_ENABLE_DRONECAN_DRIVERS
|
|
#include <GCS_MAVLink/GCS.h>
|
|
|
|
static Canard::Publisher<dronecan_remoteid_Location>* dc_location[HAL_MAX_CAN_PROTOCOL_DRIVERS];
|
|
static Canard::Publisher<dronecan_remoteid_BasicID>* dc_basic_id[HAL_MAX_CAN_PROTOCOL_DRIVERS];
|
|
static Canard::Publisher<dronecan_remoteid_SelfID>* dc_self_id[HAL_MAX_CAN_PROTOCOL_DRIVERS];
|
|
static Canard::Publisher<dronecan_remoteid_System>* dc_system[HAL_MAX_CAN_PROTOCOL_DRIVERS];
|
|
static Canard::Publisher<dronecan_remoteid_OperatorID>* dc_operator_id[HAL_MAX_CAN_PROTOCOL_DRIVERS];
|
|
|
|
static void handle_arm_status(AP_DroneCAN* ap_dronecan, const CanardRxTransfer &transfer, const dronecan_remoteid_ArmStatus &msg);
|
|
|
|
void AP_OpenDroneID::dronecan_init(AP_DroneCAN *uavcan)
|
|
{
|
|
const uint8_t driver_index = uavcan->get_driver_index();
|
|
driver_mask = 1U<<driver_index;
|
|
if (dronecan_done_init & driver_mask) {
|
|
// already initialised
|
|
return;
|
|
}
|
|
|
|
dc_location[driver_index] = new Canard::Publisher<dronecan_remoteid_Location>(uavcan->get_canard_iface());
|
|
if (dc_location[driver_index] == nullptr) {
|
|
goto alloc_failed;
|
|
}
|
|
dc_location[driver_index]->set_timeout_ms(20);
|
|
dc_location[driver_index]->set_priority(CANARD_TRANSFER_PRIORITY_LOW);
|
|
|
|
dc_basic_id[driver_index] = new Canard::Publisher<dronecan_remoteid_BasicID>(uavcan->get_canard_iface());
|
|
if (dc_basic_id[driver_index] == nullptr) {
|
|
goto alloc_failed;
|
|
}
|
|
dc_basic_id[driver_index]->set_timeout_ms(20);
|
|
dc_basic_id[driver_index]->set_priority(CANARD_TRANSFER_PRIORITY_LOW);
|
|
|
|
dc_self_id[driver_index] = new Canard::Publisher<dronecan_remoteid_SelfID>(uavcan->get_canard_iface());
|
|
if (dc_self_id[driver_index] == nullptr) {
|
|
goto alloc_failed;
|
|
}
|
|
dc_self_id[driver_index]->set_timeout_ms(20);
|
|
dc_self_id[driver_index]->set_priority(CANARD_TRANSFER_PRIORITY_LOW);
|
|
|
|
dc_system[driver_index] = new Canard::Publisher<dronecan_remoteid_System>(uavcan->get_canard_iface());
|
|
if (dc_system[driver_index] == nullptr) {
|
|
goto alloc_failed;
|
|
}
|
|
dc_system[driver_index]->set_timeout_ms(20);
|
|
dc_system[driver_index]->set_priority(CANARD_TRANSFER_PRIORITY_LOW);
|
|
|
|
dc_operator_id[driver_index] = new Canard::Publisher<dronecan_remoteid_OperatorID>(uavcan->get_canard_iface());
|
|
if (dc_operator_id[driver_index] == nullptr) {
|
|
goto alloc_failed;
|
|
}
|
|
dc_operator_id[driver_index]->set_timeout_ms(20);
|
|
dc_operator_id[driver_index]->set_priority(CANARD_TRANSFER_PRIORITY_LOW);
|
|
|
|
if (Canard::allocate_sub_arg_callback(uavcan, &handle_arm_status, driver_index) == nullptr)
|
|
{
|
|
goto alloc_failed;
|
|
}
|
|
|
|
dronecan_done_init |= driver_mask;
|
|
return;
|
|
|
|
alloc_failed:
|
|
dronecan_init_failed |= driver_mask;
|
|
GCS_SEND_TEXT(MAV_SEVERITY_NOTICE, "OpenDroneID DroneCAN alloc failed");
|
|
}
|
|
|
|
/*
|
|
send pending DroneCAN OpenDroneID packets
|
|
*/
|
|
void AP_OpenDroneID::dronecan_send(AP_DroneCAN *uavcan)
|
|
{
|
|
dronecan_init(uavcan);
|
|
|
|
if (dronecan_init_failed & driver_mask) {
|
|
return;
|
|
}
|
|
|
|
if (need_send_basic_id & driver_mask) {
|
|
WITH_SEMAPHORE(_sem);
|
|
dronecan_send_basic_id(uavcan);
|
|
need_send_basic_id &= ~driver_mask;
|
|
}
|
|
if (need_send_system & driver_mask) {
|
|
WITH_SEMAPHORE(_sem);
|
|
dronecan_send_system(uavcan);
|
|
need_send_system &= ~driver_mask;
|
|
}
|
|
if (need_send_self_id & driver_mask) {
|
|
WITH_SEMAPHORE(_sem);
|
|
dronecan_send_self_id(uavcan);
|
|
need_send_self_id &= ~driver_mask;
|
|
}
|
|
if (need_send_operator_id & driver_mask) {
|
|
WITH_SEMAPHORE(_sem);
|
|
dronecan_send_operator_id(uavcan);
|
|
need_send_operator_id &= ~driver_mask;
|
|
}
|
|
if (need_send_location & driver_mask) {
|
|
WITH_SEMAPHORE(_sem);
|
|
dronecan_send_location(uavcan);
|
|
need_send_location &= ~driver_mask;
|
|
}
|
|
}
|
|
|
|
#define ODID_COPY(name) msg.name = pkt.name
|
|
#define ODID_COPY_STR(name) do { msg.name.len = strncpy_noterm((char*)msg.name.data, (const char*)pkt.name, sizeof(msg.name.data)); } while(0)
|
|
|
|
|
|
void AP_OpenDroneID::dronecan_send_location(AP_DroneCAN *uavcan)
|
|
{
|
|
dronecan_remoteid_Location msg {};
|
|
const auto &pkt = pkt_location;
|
|
ODID_COPY_STR(id_or_mac);
|
|
ODID_COPY(status);
|
|
ODID_COPY(direction);
|
|
ODID_COPY(speed_horizontal);
|
|
ODID_COPY(speed_vertical);
|
|
ODID_COPY(latitude);
|
|
ODID_COPY(longitude);
|
|
ODID_COPY(altitude_barometric);
|
|
ODID_COPY(altitude_geodetic);
|
|
ODID_COPY(height_reference);
|
|
ODID_COPY(height);
|
|
ODID_COPY(horizontal_accuracy);
|
|
ODID_COPY(vertical_accuracy);
|
|
ODID_COPY(barometer_accuracy);
|
|
ODID_COPY(speed_accuracy);
|
|
ODID_COPY(timestamp);
|
|
ODID_COPY(timestamp_accuracy);
|
|
dc_location[uavcan->get_driver_index()]->broadcast(msg);
|
|
}
|
|
|
|
void AP_OpenDroneID::dronecan_send_basic_id(AP_DroneCAN *uavcan)
|
|
{
|
|
dronecan_remoteid_BasicID msg {};
|
|
const auto &pkt = pkt_basic_id;
|
|
ODID_COPY_STR(id_or_mac);
|
|
ODID_COPY(id_type);
|
|
ODID_COPY(ua_type);
|
|
ODID_COPY_STR(uas_id);
|
|
dc_basic_id[uavcan->get_driver_index()]->broadcast(msg);
|
|
}
|
|
|
|
void AP_OpenDroneID::dronecan_send_system(AP_DroneCAN *uavcan)
|
|
{
|
|
dronecan_remoteid_System msg {};
|
|
const auto &pkt = pkt_system;
|
|
ODID_COPY_STR(id_or_mac);
|
|
ODID_COPY(operator_location_type);
|
|
ODID_COPY(classification_type);
|
|
ODID_COPY(operator_latitude);
|
|
ODID_COPY(operator_longitude);
|
|
ODID_COPY(area_count);
|
|
ODID_COPY(area_radius);
|
|
ODID_COPY(area_ceiling);
|
|
ODID_COPY(area_floor);
|
|
ODID_COPY(category_eu);
|
|
ODID_COPY(class_eu);
|
|
ODID_COPY(operator_altitude_geo);
|
|
ODID_COPY(timestamp);
|
|
dc_system[uavcan->get_driver_index()]->broadcast(msg);
|
|
}
|
|
|
|
void AP_OpenDroneID::dronecan_send_self_id(AP_DroneCAN *uavcan)
|
|
{
|
|
dronecan_remoteid_SelfID msg {};
|
|
const auto &pkt = pkt_self_id;
|
|
ODID_COPY_STR(id_or_mac);
|
|
ODID_COPY(description_type);
|
|
ODID_COPY_STR(description);
|
|
dc_self_id[uavcan->get_driver_index()]->broadcast(msg);
|
|
}
|
|
|
|
void AP_OpenDroneID::dronecan_send_operator_id(AP_DroneCAN *uavcan)
|
|
{
|
|
dronecan_remoteid_OperatorID msg {};
|
|
const auto &pkt = pkt_operator_id;
|
|
ODID_COPY_STR(id_or_mac);
|
|
ODID_COPY(operator_id_type);
|
|
ODID_COPY_STR(operator_id);
|
|
dc_operator_id[uavcan->get_driver_index()]->broadcast(msg);
|
|
}
|
|
|
|
/*
|
|
handle ArmStatus message from DroneCAN
|
|
*/
|
|
static void handle_arm_status(AP_DroneCAN *ap_dronecan, const CanardRxTransfer& transfer, const dronecan_remoteid_ArmStatus &msg)
|
|
{
|
|
mavlink_open_drone_id_arm_status_t status {};
|
|
|
|
status.status = msg.status;
|
|
strncpy_noterm((char*)status.error, (const char*)msg.error.data, sizeof(status.error));
|
|
|
|
AP::opendroneid().set_arm_status(status);
|
|
|
|
// Push DroneCAN Arm Message to GCS
|
|
gcs().send_to_active_channels(MAVLINK_MSG_ID_OPEN_DRONE_ID_ARM_STATUS,(const char*)&status);
|
|
}
|
|
|
|
// copy arm status for DroneCAN
|
|
void AP_OpenDroneID::set_arm_status(mavlink_open_drone_id_arm_status_t &status)
|
|
{
|
|
WITH_SEMAPHORE(_sem);
|
|
arm_status = status;
|
|
last_arm_status_ms = AP_HAL::millis();
|
|
}
|
|
|
|
#endif // HAL_ENABLE_DRONECAN_DRIVERS
|
|
#endif // AP_OPENDRONEID_ENABLED
|