/* 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 . */ // // Trimble GPS driver for ArduPilot. // Code by Michael Oborne // Maintained by Ryan Friedman // https://receiverhelp.trimble.com/oem-gnss/index.html#Welcome.html?TocPath=_____1 #pragma once #include "AP_GPS.h" #include "GPS_Backend.h" #if AP_GPS_GSOF_ENABLED class AP_GPS_GSOF : public AP_GPS_Backend { public: AP_GPS_GSOF(AP_GPS &_gps, AP_GPS::GPS_State &_state, AP_HAL::UARTDriver *_port); AP_GPS::GPS_Status highest_supported_status(void) override WARN_IF_UNUSED { return AP_GPS::GPS_OK_FIX_3D_RTK_FIXED; } // Methods bool read() override WARN_IF_UNUSED; const char *name() const override { return "GSOF"; } private: // A subset of the port identifiers in the GSOF protocol that are used for serial. // Ethernet, USB, etc are not supported by the GPS driver at this time so they are omitted. // These values are not documented in the API. enum class HW_Port { COM1 = 0, // RS232 serial COM2 = 1, // TTL serial }; // A subset of the data frequencies in the GSOF protocol that are used for serial. // These values are not documented in the API. enum class Output_Rate { FREQ_10_HZ = 1, FREQ_50_HZ = 15, FREQ_100_HZ = 16, }; // A subset of the supported baud rates in the GSOF protocol that are useful. // These values are not documented in the API. // The matches the GPS_GSOF_BAUD parameter. enum class HW_Baud { BAUD115K = 0x07, BAUD230K = 0x0B, }; bool parse(const uint8_t temp) WARN_IF_UNUSED; bool process_message() WARN_IF_UNUSED; // Send a request to the GPS to configure its baud rate on a certain (serial) port. // Note - these request functions expect an ACK from the device. // If the device is already sending serial traffic, there is no mechanism to prevent this. // According to the manufacturer, the best approach is to switch to ethernet. // Use one port for configuration data and another for stream data to prevent conflict. // Because there is only one TTL serial interface exposed, the approach here is using retry logic. bool requestBaud(const HW_Port portIndex, const HW_Baud baudRate) WARN_IF_UNUSED; // Send a request to the GPS to enable a message type on the port. bool requestGSOF(const uint8_t messageType, const HW_Port portIndex, const Output_Rate rateHz) WARN_IF_UNUSED; bool disableOutput(const HW_Port portIndex) WARN_IF_UNUSED; // Generic handler for sending command and checking return code. // Make sure to disableOutput() before doing configuration. bool requestResponse(const uint8_t* buf, const uint8_t len) WARN_IF_UNUSED; static void populateChecksum(uint8_t* buf, const uint8_t len); void populateOutgoingTransNumber(uint8_t* buf); double SwapDouble(const uint8_t* src, const uint32_t pos) const WARN_IF_UNUSED; float SwapFloat(const uint8_t* src, const uint32_t pos) const WARN_IF_UNUSED; uint32_t SwapUint32(const uint8_t* src, const uint32_t pos) const WARN_IF_UNUSED; uint16_t SwapUint16(const uint8_t* src, const uint32_t pos) const WARN_IF_UNUSED; bool validate_baud(const uint8_t baud) const WARN_IF_UNUSED; bool validate_com_port(const uint8_t com_port) const WARN_IF_UNUSED; struct Msg_Parser { enum class State { STARTTX = 0, STATUS, PACKETTYPE, LENGTH, DATA, CHECKSUM, ENDTX }; State state = State::STARTTX; uint8_t status; uint8_t packettype; uint8_t length; uint8_t data[256]; uint8_t checksum; uint8_t endtx; uint16_t read; uint8_t checksumcalc; } msg; static const uint8_t STX = 0x02; static const uint8_t ETX = 0x03; static const uint8_t ACK = 0x06; static const uint8_t NACK = 0x15; // How long to wait from sending configuration data for a response. // This assumes delay is the same regardless of baud rate. static const uint8_t configuration_wait_time_ms {5}; // How many attempts to attempt configuration. // Raising this makes the initialization more immune to data conflicts with streamed data. // Raising it too high will trigger the watchdog. // Lowering this makes the driver quicker to return from initialization calls. static const uint8_t configuration_attempts {3}; // The counter for number of outgoing packets uint8_t packetOutboundTransNumber; uint32_t gsofmsg_time; uint8_t gsofmsgreq_index; const uint8_t gsofmsgreq[5] = {1,2,8,9,12}; bool is_baud_configured {false}; bool gsof_configured {false}; }; #endif