// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- /* 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 . */ // // Emlid Reach Binary (ERB) GPS driver for ArduPilot. // ERB protocol: http://files.emlid.com/ERB.pdf #pragma once #include #include "AP_GPS.h" class AP_GPS_ERB : public AP_GPS_Backend { public: AP_GPS_ERB(AP_GPS &_gps, AP_GPS::GPS_State &_state, AP_HAL::UARTDriver *_port); // Methods bool read(); AP_GPS::GPS_Status highest_supported_status(void) { return AP_GPS::GPS_OK_FIX_3D_RTK; } static bool _detect(struct ERB_detect_state &state, uint8_t data); private: struct PACKED erb_header { uint8_t preamble1; uint8_t preamble2; uint8_t msg_id; uint16_t length; }; struct PACKED erb_ver { uint32_t time; uint8_t ver_high; uint8_t ver_medium; uint8_t ver_low; }; struct PACKED erb_pos { uint32_t time; double longitude; double latitude; double altitude_ellipsoid; double altitude_msl; uint32_t horizontal_accuracy; uint32_t vertical_accuracy; }; struct PACKED erb_stat { uint32_t time; uint16_t week; uint8_t fix_type; uint8_t fix_status; uint8_t satellites; }; struct PACKED erb_dops { uint32_t time; uint16_t gDOP; uint16_t pDOP; uint16_t vDOP; uint16_t hDOP; }; struct PACKED erb_vel { uint32_t time; int32_t vel_north; int32_t vel_east; int32_t vel_down; uint32_t speed_2d; int32_t heading_2d; uint32_t speed_accuracy; }; // Receive buffer union PACKED { erb_ver ver; erb_pos pos; erb_stat stat; erb_dops dops; erb_vel vel; uint8_t bytes[]; } _buffer; enum erb_protocol_bytes { PREAMBLE1 = 0x45, PREAMBLE2 = 0x52, MSG_VER = 0x01, MSG_POS = 0x02, MSG_STAT = 0x03, MSG_DOPS = 0x04, MSG_VEL = 0x05, }; enum erb_fix_type { FIX_NONE = 0x00, FIX_SINGLE = 0x01, FIX_FLOAT = 0x02, FIX_FIX = 0x03, }; // Packet checksum accumulators uint8_t _ck_a; uint8_t _ck_b; // State machine state uint8_t _step; uint8_t _msg_id; uint16_t _payload_length; uint16_t _payload_counter; // 8 bit count of fix messages processed, used for periodic processing uint8_t _fix_count; uint32_t _last_pos_time; uint32_t _last_vel_time; // do we have new position information? bool _new_position:1; // do we have new speed information? bool _new_speed:1; // Buffer parse & GPS state update bool _parse_gps(); void inject_data(uint8_t *data, uint8_t len); // used to update fix between status and position packets AP_GPS::GPS_Status next_fix; };