/*
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 .
*/
#include "AP_ADSB_Sagetech.h"
#if HAL_ADSB_ENABLED
#include
#include
#include
#include
#include
#include
#include
extern const AP_HAL::HAL& hal;
#define SAGETECH_SCALER_LATLNG (1.0f/2.145767E-5f) // 180/(2^23)
#define SAGETECH_SCALER_KNOTS_TO_CM ((KNOTS_TO_M_PER_SEC/0.125f) * 100.0f)
#define SAGETECH_SCALER_ALTITUDE (1.0f/0.015625f)
#define SAGETECH_SCALER_HEADING_CM ((360.0f/256.0f) * 100.0f)
#define SAGETECH_VALIDFLAG_LATLNG (1<<0)
#define SAGETECH_VALIDFLAG_ALTITUDE (1<<1)
#define SAGETECH_VALIDFLAG_VELOCITY (1<<2)
#define SAGETECH_VALIDFLAG_GND_SPEED (1<<3)
#define SAGETECH_VALIDFLAG_HEADING (1<<4)
#define SAGETECH_VALIDFLAG_V_RATE_GEO (1<<5)
#define SAGETECH_VALIDFLAG_V_RATE_BARO (1<<6)
#define SAGETECH_VALIDFLAG_EST_LATLNG (1<<7)
#define SAGETECH_VALIDFLAG_EST_VELOCITY (1<<8)
#define SAGETECH_ENFORCE_ACKS 0
#define SAGETECH_DEBUG_ACK_TIMEOUTS 0
#define SAGETECH_DEBUG_TX_ID_ONLY 0
#define SAGETECH_DEBUG_TX_ID_PAYLOAD 0
#define SAGETECH_DEBUG_TX_ALL_RAW 0
#define SAGETECH_DEBUG_RX 0
#define SAGETECH_DEBUG_RX_ACK 0
// constructor
AP_ADSB_Sagetech::AP_ADSB_Sagetech(AP_ADSB &adsb) :
AP_ADSB_Backend(adsb)
{
uart = AP::serialmanager().find_serial(AP_SerialManager::SerialProtocol_Sagetech, 0);
if (uart != nullptr) {
baudrate = AP::serialmanager().find_baudrate(AP_SerialManager::SerialProtocol_Sagetech, 0);
uart->begin(baudrate);
// no sagtech hardware have flow control pins exposed
uart->set_flow_control(AP_HAL::UARTDriver::FLOW_CONTROL_DISABLE);
}
}
// detect if a sensor is connected by looking for a configured serial port
bool AP_ADSB_Sagetech::detect()
{
return AP::serialmanager().find_serial(AP_SerialManager::SerialProtocol_Sagetech, 0) != nullptr;
}
void AP_ADSB_Sagetech::init()
{
if (uart == nullptr) {
gcs().send_text(MAV_SEVERITY_DEBUG, "%sInit failed, check SERIALx_PROTOCOL cfg", frontend.GcsHeader);
}
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
if (uart != nullptr) {
gcs().send_text(MAV_SEVERITY_DEBUG, "%sUART Init successful", frontend.GcsHeader);
}
#endif
}
void AP_ADSB_Sagetech::update()
{
const uint32_t now_ms = AP_HAL::millis();
// -----------------------------
// read any available data on serial port
// -----------------------------
int32_t nbytes = 0;
if (uart != nullptr) {
nbytes = uart->available();
}
while (nbytes-- > 0) {
const uint8_t data = (uint8_t)uart->read();
switch (protocol) {
default:
case AP_ADSB_Sagetech::Protocol::NONE:
// parse all protocols until we find one that works
if (parse_byte_XP(data)) {
protocol = AP_ADSB_Sagetech::Protocol::XP;
parse_packet_XP(message_in_xp.packet);
}
if (parse_byte_MX(data)) {
protocol = AP_ADSB_Sagetech::Protocol::MX;
parse_packet_MX();
}
break;
case AP_ADSB_Sagetech::Protocol::XP:
if (parse_byte_XP(data)) {
parse_packet_XP(message_in_xp.packet);
}
break;
case AP_ADSB_Sagetech::Protocol::MX:
if (parse_byte_MX(data)) {
parse_packet_MX();
}
break;
}
}
// -----------------------------
// handle timers for generating data
// -----------------------------
#if SAGETECH_ENFORCE_ACKS
if (last_packet_type_sent != MsgTypes_XP::INVALID) {
// we're expecting an ACK
if (now_ms - last_packet_send_ms >= 1000) {
response_timeout_count++;
#if SAGETECH_DEBUG_ACK_TIMEOUTS
gcs().send_text(MAV_SEVERITY_DEBUG, "%sACK Timeout type=%u, cnt=%u", frontend.GcsHeader, (unsigned)last_packet_type_sent, (unsigned)response_timeout_count);
#endif
// retry forever until we get an ACK
send_packet(last_packet_type_sent);
}
} else
#endif
{
//if (!last_packet_initialize_ms) {
if (!last_packet_initialize_ms || (now_ms - last_packet_initialize_ms >= 5000)) {
last_packet_initialize_ms = now_ms;
send_packet(MsgTypes_XP::Installation_Set);
//} else if (last_packet_PreFlight_ms == 0) {
} else if (!last_packet_PreFlight_ms || (now_ms - last_packet_PreFlight_ms >= 8200)) {
// send once, for now..
last_packet_PreFlight_ms = now_ms;
// TODO: allow callsign to not require a reboot
send_packet(MsgTypes_XP::Preflight_Set);
} else if (now_ms - last_packet_Operating_ms >= 1000 && (
last_packet_Operating_ms == 0 || // send once at boot
// send as data changes
last_operating_squawk != frontend.out_state.cfg.squawk_octal ||
abs(last_operating_alt - frontend._my_loc.alt) > 1555 || // 1493cm == 49ft. The output resolution is 100ft per bit
last_operating_rf_select != frontend.out_state.cfg.rfSelect))
{
last_packet_Operating_ms = now_ms;
last_operating_squawk = frontend.out_state.cfg.squawk_octal;
last_operating_alt = frontend._my_loc.alt;
last_operating_rf_select = frontend.out_state.cfg.rfSelect;
send_packet(MsgTypes_XP::Operating_Set);
} else if (now_ms - last_packet_GPS_ms >= (frontend.out_state.is_flying ? 200 : 1000)) {
// 1Hz when not flying, 5Hz when flying
last_packet_GPS_ms = now_ms;
send_packet(MsgTypes_XP::GPS_Set);
}
}
}
void AP_ADSB_Sagetech::send_packet(const MsgTypes_XP type)
{
switch (type) {
case MsgTypes_XP::Installation_Set:
send_Installation();
break;
case MsgTypes_XP::Preflight_Set:
send_PreFlight();
break;
case MsgTypes_XP::Operating_Set:
send_Operating();
break;
case MsgTypes_XP::GPS_Set:
send_GPS();
break;
default:
break;
}
}
void AP_ADSB_Sagetech::request_packet(const MsgTypes_XP type)
{
Packet_XP pkt {};
pkt.type = MsgTypes_XP::Request;
pkt.id = 0;
pkt.payload_length = 4;
pkt.payload[0] = static_cast(type);
send_msg(pkt);
}
void AP_ADSB_Sagetech::parse_packet_XP(const Packet_XP &msg)
{
#if SAGETECH_DEBUG_RX
gcs().send_text(MAV_SEVERITY_DEBUG, "%sRX type=%u, id=%d", frontend.GcsHeader, (unsigned)msg.type, msg.id);
#endif
switch (msg.type) {
case MsgTypes_XP::ACK: {
// ACK received!
const uint8_t system_state = msg.payload[2];
transponder_type = (Transponder_Type)msg.payload[6];
const char* rfmode = "RF mode: ";
const uint8_t prev_transponder_mode = last_ack_transponder_mode;
last_ack_transponder_mode = (system_state >> 6) & 0x03;
if (prev_transponder_mode != last_ack_transponder_mode) {
switch (last_ack_transponder_mode) {
case 0: gcs().send_text(MAV_SEVERITY_INFO, "%s%sOFF", frontend.GcsHeader, rfmode); break;
case 1: gcs().send_text(MAV_SEVERITY_INFO, "%s%sSTBY", frontend.GcsHeader, rfmode); break;
case 2: gcs().send_text(MAV_SEVERITY_INFO, "%s%sON", frontend.GcsHeader, rfmode); break;
case 3: gcs().send_text(MAV_SEVERITY_INFO, "%s%sON-ALT",frontend.GcsHeader, rfmode); break;
default: break;
}
}
#if SAGETECH_DEBUG_RX_ACK
const uint8_t acked_type = msg.payload[0];
const uint8_t acked_id = msg.payload[1];
const int32_t pres_altitude = (int32_t)fetchU32(&msg.payload[3]);
const uint16_t squawk = fetchU16(&msg.payload[7]);
gcs().send_text(MAV_SEVERITY_DEBUG, "%sACK %u, %u, 0x%02X, %d, %u, %u %u", frontend.GcsHeader,
acked_type,
acked_id,
system_state,
pres_altitude,
transponder_type,
squawk,
last_ack_transponder_mode); // mode
for (uint8_t i=0; i<6; i++) {
const uint8_t stateBits = (system_state & (1<< i));
if (stateBits != 0 && stateBits != SystemStateBits::Altitidue_Source) {
gcs().send_text(MAV_SEVERITY_DEBUG, "%sACK status: %s", frontend.GcsHeader, systemStatsBits_to_str((SystemStateBits)stateBits));
}
}
#endif
}
break;
case MsgTypes_XP::Installatioin_Response:
case MsgTypes_XP::Preflight_Response:
case MsgTypes_XP::Status_Response:
// TODO add support for these
break;
case MsgTypes_XP::ADSB_StateVector_Report:
case MsgTypes_XP::ADSB_ModeStatus_Report:
case MsgTypes_XP::TISB_StateVector_Report:
case MsgTypes_XP::TISB_ModeStatus_Report:
case MsgTypes_XP::TISB_CorasePos_Report:
case MsgTypes_XP::TISB_ADSB_Mgr_Report:
handle_adsb_in_msg(msg);
break;
case MsgTypes_XP::Installation_Set:
case MsgTypes_XP::Preflight_Set:
case MsgTypes_XP::Operating_Set:
case MsgTypes_XP::GPS_Set:
case MsgTypes_XP::Request:
// these are out-bound only and are not expected to be received
FALLTHROUGH;
default:
last_packet_type_sent = INVALID;
break;
}
}
void AP_ADSB_Sagetech::handle_adsb_in_msg(const Packet_XP &msg)
{
AP_ADSB::adsb_vehicle_t vehicle {};
const uint32_t now = AP_HAL::millis();
vehicle.last_update_ms = now;
uint16_t validFlags;
switch (msg.type) {
case MsgTypes_XP::ADSB_StateVector_Report: // 0x91
validFlags = fetchU16(&msg.payload[8]);
vehicle.info.ICAO_address = fetchU24(&msg.payload[10]);
if (validFlags & SAGETECH_VALIDFLAG_LATLNG) {
vehicle.info.lat = ((int32_t)fetchU24(&msg.payload[20])) * SAGETECH_SCALER_LATLNG;
vehicle.info.lon = ((int32_t)fetchU24(&msg.payload[23])) * SAGETECH_SCALER_LATLNG;
vehicle.info.flags |= ADSB_FLAGS_VALID_COORDS;
}
if (validFlags & SAGETECH_VALIDFLAG_ALTITUDE) {
vehicle.info.altitude = (int32_t)fetchU24(&msg.payload[26]);
vehicle.info.flags |= ADSB_FLAGS_VALID_ALTITUDE;
}
if (validFlags & SAGETECH_VALIDFLAG_VELOCITY) {
const float velNS = ((int32_t)fetchU16(&msg.payload[29])) * SAGETECH_SCALER_KNOTS_TO_CM;
const float velEW = ((int32_t)fetchU16(&msg.payload[31])) * SAGETECH_SCALER_KNOTS_TO_CM;
vehicle.info.hor_velocity = Vector2f(velEW, velNS).angle();
vehicle.info.flags |= ADSB_FLAGS_VALID_VELOCITY;
}
if (validFlags & SAGETECH_VALIDFLAG_HEADING) {
vehicle.info.heading = ((float)msg.payload[29]) * SAGETECH_SCALER_HEADING_CM;
vehicle.info.flags |= ADSB_FLAGS_VALID_HEADING;
}
if ((validFlags & SAGETECH_VALIDFLAG_V_RATE_GEO) || (validFlags & SAGETECH_VALIDFLAG_V_RATE_BARO)) {
vehicle.info.ver_velocity = (int16_t)fetchU16(&msg.payload[38]);
vehicle.info.flags |= ADSB_FLAGS_VERTICAL_VELOCITY_VALID;
}
if (vehicle.info.flags != 0) {
frontend.handle_adsb_vehicle(vehicle);
}
break;
case MsgTypes_XP::ADSB_ModeStatus_Report: // 0x92
validFlags = msg.payload[8];
vehicle.info.ICAO_address = fetchU24(&msg.payload[9]);
if (msg.payload[16] != 0) {
// if string is non-null, consider it valid
memcpy(&vehicle.info, &msg.payload[16], 8);
vehicle.info.flags |= ADSB_FLAGS_VALID_CALLSIGN;
}
if (vehicle.info.flags != 0) {
frontend.handle_adsb_vehicle(vehicle);
}
break;
case MsgTypes_XP::TISB_StateVector_Report:
case MsgTypes_XP::TISB_ModeStatus_Report:
case MsgTypes_XP::TISB_CorasePos_Report:
case MsgTypes_XP::TISB_ADSB_Mgr_Report:
// TODO
return;
default:
return;
}
}
bool AP_ADSB_Sagetech::parse_byte_XP(const uint8_t data)
{
switch (message_in_xp.state) {
default:
case ParseState::WaitingFor_Start:
if (data == 0xA5) {
message_in_xp.state = ParseState::WaitingFor_AssmAddr;
}
break;
case ParseState::WaitingFor_AssmAddr:
message_in_xp.state = (data == 0x01) ? ParseState::WaitingFor_MsgType : ParseState::WaitingFor_Start;
break;
case ParseState::WaitingFor_MsgType:
message_in_xp.packet.type = static_cast(data);
message_in_xp.state = ParseState::WaitingFor_MsgId;
break;
case ParseState::WaitingFor_MsgId:
message_in_xp.packet.id = data;
message_in_xp.state = ParseState::WaitingFor_PayloadLen;
break;
case ParseState::WaitingFor_PayloadLen:
message_in_xp.packet.payload_length = data;
message_in_xp.index = 0;
message_in_xp.state = (data == 0) ? ParseState::WaitingFor_ChecksumFletcher : ParseState::WaitingFor_PayloadContents;
break;
case ParseState::WaitingFor_PayloadContents:
message_in_xp.packet.payload[message_in_xp.index++] = data;
if (message_in_xp.index >= message_in_xp.packet.payload_length) {
message_in_xp.state = ParseState::WaitingFor_ChecksumFletcher;
message_in_xp.index = 0;
}
break;
case ParseState::WaitingFor_ChecksumFletcher:
message_in_xp.packet.checksumFletcher = data;
message_in_xp.state = ParseState::WaitingFor_Checksum;
break;
case ParseState::WaitingFor_Checksum:
message_in_xp.packet.checksum = data;
message_in_xp.state = ParseState::WaitingFor_End;
if (checksum_XP(message_in_xp.packet)) {
parse_packet_XP(message_in_xp.packet);
}
break;
case ParseState::WaitingFor_End:
// we don't care if the end value matches
message_in_xp.state = ParseState::WaitingFor_Start;
break;
}
return false;
}
// compute Sum and FletcherSum and write them into msg.
// returns true if the values in msg were already the correct ones
//
// this allows a single function to tell you if inbound msg is correct wirh true result, and ignore result for outbound
bool AP_ADSB_Sagetech::checksum_XP(Packet_XP &msg)
{
uint8_t sum = 0;
uint8_t sumFletcher = 0;
const uint8_t header_message_format[5] = {
0xA5, // start
0x01, // assembly address
static_cast(msg.type),
msg.id,
msg.payload_length
};
for (uint8_t i=0; i<5; i++) {
sum += header_message_format[i];
sumFletcher += sum;
}
for (uint8_t i=0; i(msg.type),
msg.id,
msg.payload_length
};
const uint8_t message_format_tail[3] = {
msg.checksumFletcher,
msg.checksum,
0x5A // end
};
if (uart != nullptr) {
uart->write(message_format_header, sizeof(message_format_header));
uart->write(msg.payload, msg.payload_length);
uart->write(message_format_tail, sizeof(message_format_tail));
}
last_packet_type_sent = msg.type;
last_packet_send_ms = AP_HAL::millis();
#if SAGETECH_DEBUG_TX_ALL_RAW
gcs().send_text(MAV_SEVERITY_DEBUG, "%s-------", frontend.GcsHeader);
for (uint8_t i=0; i(msg.type), type_to_str(msg.type));
#endif
#if SAGETECH_DEBUG_TX_ID_PAYLOAD || SAGETECH_DEBUG_TX_ALL_RAW
#if SAGETECH_DEBUG_TX_ID_PAYLOAD
gcs().send_text(MAV_SEVERITY_DEBUG, "%sTx type=%u, payload.len=%u", frontend.GcsHeader, (unsigned)msg.type, (unsigned)msg.payload_length);
#endif
for (uint8_t i=0; i= 0) ? 0x01 : 0; // isNorth
hemisphere |= (longitude >= 0) ? 0x02 : 0; // isEast
hemisphere |= (AP::gps().status() < AP_GPS::GPS_OK_FIX_2D) ? 0x80 : 0; // isInvalid
pkt.payload[35] = hemisphere;
// time
uint64_t time_usec;
if (!AP::rtc().get_utc_usec(time_usec)) {
memset(&pkt.payload[36],' ', 10);
} else {
// not completely accurate, our time includes leap seconds and time_t should be without
const time_t time_sec = time_usec / 1000000;
struct tm* tm = gmtime(&time_sec);
// format time string
snprintf((char*)&pkt.payload[36], 11, "%02u%02u%06.3f", tm->tm_hour, tm->tm_min, tm->tm_sec + (time_usec % 1000000) * 1.0e-6);
}
send_msg(pkt);
}
const char* AP_ADSB_Sagetech::type_to_str(const uint8_t type)
{
switch (type) {
case 0x01: return "Install";
case 0x02: return "PreFlight";
case 0x03: return "Operate";
case 0x04: return "GPS Data";
case 0x05: return "DataReq";
default: return "Unknown";
}
}
const char* AP_ADSB_Sagetech::systemStatsBits_to_str(const SystemStateBits systemStateBits)
{
switch (systemStateBits) {
case SystemStateBits::Error_Transponder: return "Error_Transponder";
case SystemStateBits::Altitidue_Source: return "Altitidue_Source";
case SystemStateBits::Error_GPS: return "Error_GPS";
case SystemStateBits::Error_ICAO: return "Error_ICAO";
case SystemStateBits::Error_Over_Temperature: return "Error_Over_Temperature";
case SystemStateBits::Error_Extended_Squitter: return "Error_Extended_Squitter";
default: return "Unknown";
}
}
#endif // HAL_ADSB_ENABLED