// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*- /// @file GCS.h /// @brief Interface definition for the various Ground Control System protocols. #ifndef __GCS_H #define __GCS_H #include #include #include #include #include #include #include /// /// @class GCS /// @brief Class describing the interface between the APM code /// proper and the GCS implementation. /// /// GCS' are currently implemented inside the sketch and as such have /// access to all global state. The sketch should not, however, call GCS /// internal functions - all calls to the GCS should be routed through /// this interface (or functions explicitly exposed by a subclass). /// class GCS_Class { public: /// Startup initialisation. /// /// This routine performs any one-off initialisation required before /// GCS messages are exchanged. /// /// @note The stream is expected to be set up and configured for the /// correct bitrate before ::init is called. /// /// @note The stream is currently BetterStream so that we can use the _P /// methods; this may change if Arduino adds them to Print. /// /// @param port The stream over which messages are exchanged. /// void init(BetterStream *port) { _port = port; } /// Update GCS state. /// /// This may involve checking for received bytes on the stream, /// or sending additional periodic messages. void update(void) {} /// Send a message with a single numeric parameter. /// /// This may be a standalone message, or the GCS driver may /// have its own way of locating additional parameters to send. /// /// @param id ID of the message to send. /// @param param Explicit message parameter. /// void send_message(uint8_t id, int32_t param = 0) {} /// Send a text message. /// /// @param severity A value describing the importance of the message. /// @param str The text to be sent. /// void send_text(uint8_t severity, const char *str) {} /// Send acknowledgement for a message. /// /// @param id The message ID being acknowledged. /// @param sum1 Checksum byte 1 from the message being acked. /// @param sum2 Checksum byte 2 from the message being acked. /// void acknowledge(uint8_t id, uint8_t sum1, uint8_t sum2) {} /// Emit an update of the "current" waypoints, often previous, current and /// next. /// void print_current_waypoints() {} // // The following interfaces are not currently implemented as their counterparts // are not called in the mainline code. XXX ripe for re-specification. // /// Send a text message with printf-style formatting. /// /// @param severity A value describing the importance of the message. /// @param fmt The format string to send. /// @param ... Additional arguments to the format string. /// // void send_message(uint8_t severity, const char *fmt, ...) {} /// Log a waypoint /// /// @param wp The waypoint to log. /// @param index The index of the waypoint. // void print_waypoint(struct Location *wp, uint8_t index) {} // test if frequency within range requested for loop // used by data_stream_send static bool freqLoopMatch(uint16_t freq, uint16_t freqMin, uint16_t freqMax) { if (freq != 0 && freq >= freqMin && freq < freqMax) return true; else return false; } // send streams which match frequency range void data_stream_send(uint16_t freqMin, uint16_t freqMax); protected: /// The stream we are communicating over BetterStream *_port; }; // // GCS class definitions. // // These are here so that we can declare the GCS object early in the sketch // and then reference it statically rather than via a pointer. // /// /// @class GCS_STANDARD /// @brief The default APM GCS protocol /// #if GCS_PROTOCOL == GCS_PROTOCOL_STANDARD class GCS_STANDARD : public GCS_Class { public: GCS_STANDARD(BinComm::MessageHandler GCS_MessageHandlers[]); void update(void); void init(BetterStream *port); void send_message(uint8_t id, uint32_t param = 0); void send_text(uint8_t severity, const char *str); void acknowledge(uint8_t id, uint8_t sum1, uint8_t sum2); BinComm & getBinComm(void) { return _binComm; } private: BinComm _binComm; }; #endif // GCS_PROTOCOL_STANDARD /// /// @class GCS_MAVLINK /// @brief The mavlink protocol for qgroundcontrol /// #if GCS_PROTOCOL == GCS_PROTOCOL_MAVLINK class GCS_MAVLINK : public GCS_Class { public: GCS_MAVLINK(); void update(void); void init(BetterStream *port); void send_message(uint8_t id, uint32_t param = 0); void send_text(uint8_t severity, const char *str); void acknowledge(uint8_t id, uint8_t sum1, uint8_t sum2); void data_stream_send(uint16_t freqMin, uint16_t freqMax); private: void handleMessage(mavlink_message_t * msg); mavlink_channel_t chan; uint16_t packet_drops; uint16_t rawSensorStreamRate; uint16_t attitudeStreamRate; uint16_t positionStreamRate; uint16_t rawControllerStreamRate; uint16_t rcStreamRate; uint16_t extraStreamRate[3]; }; #endif // GCS_PROTOCOL_MAVLINK /// /// @class GCS_LEGACY /// @brief Support for the legacy Ardupilot GCS /// #if GCS_PROTOCOL == GCS_PROTOCOL_LEGACY class GCS_LEGACY : public GCS_Class { public: void send_message(uint8_t id, uint32_t param = 0); void send_text(uint8_t severity, const char *str); private: void print_attitude(void); void print_control_mode(void); void print_position(void); void print_waypoint(struct Location *cmd,byte index); }; #endif // GCS_PROTOTOL_LEGACY /// /// @class GCS_DEBUGTERMINAL /// @brief A GCS driver with an interactive management interface /// #if GCS_PROTOCOL == GCS_PROTOCOL_DEBUGTERMINAL class GCS_DEBUGTERMINAL : public GCS_Class { public: GCS_DEBUGTERMINAL() : report_command(1) {} void update(void); void send_message(uint8_t id, uint32_t param = 0); void send_text(uint8_t severity, const char *str); void print_current_waypoints(void); void print_commands(); void print_PID(long PID_error, long dt, float scaler, float derivative, float integrator, float last_error); private: char gcsinputbuffer[120]; // read buffer byte bufferidx; // Reporting flags byte report_heartbeat; byte report_attitude; byte report_location; byte report_command; byte report_severity; byte first_location; byte report_cpu_load; void processgcsinput(char c); void run_debugt_command(char *buf); static int strmatch(char* s1, const char* s2); static long readfloat(char* s, unsigned char* i); void process_set_cmd(char *buffer, int bufferlen); static int get_pid_offset_name(char *buffer, int *offset, unsigned char *length); void print_rlocation(Location *wp); void print_error(const char *msg); void print_position(void); void print_attitude(void); void print_new_wp_info(); void print_control_mode(void); void print_tuning(void); void print_command(struct Location *cmd, byte index); void print_command_id(byte id); void print_gain(unsigned char g); void print_gains(); void print_gain_keyword_error(); void print_commands(unsigned char i1, unsigned char i2); }; #endif // GCS_PROTOCOL_DEBUGTERMINAL /// /// @class GCS_XPLANE /// @brief GCS driver for HIL operation with the X-plane simulator /// #if GCS_PROTOCOL == GCS_PROTOCOL_XPLANE class GCS_XPLANE : public GCS_Class { public: void send_message(uint8_t id, uint32_t param = 0); void send_text(uint8_t severity, const char *str); void print_current_waypoints(void); private: void print_control_mode(void); void print_waypoint(struct Location *wp, uint8_t index); void print_waypoints(void); }; #endif // GCS_PROTOCOL_XPLANE #endif // __GCS_H