/* 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 . */ /* implementation of RunCam camera protocols With thanks to betaflight for a great reference implementation. Several of the functions below are based on betaflight equivalent functions */ #pragma once #include #include #ifndef HAL_RUNCAM_ENABLED #define HAL_RUNCAM_ENABLED !HAL_MINIMIZE_FEATURES && !APM_BUILD_TYPE(APM_BUILD_Replay) #endif #if HAL_RUNCAM_ENABLED #include #include #include #include #include #define RUNCAM_MAX_PACKET_SIZE 64 /// @class AP_RunCam /// @brief Object managing a RunCam device class AP_RunCam { public: AP_RunCam(); // do not allow copies AP_RunCam(const AP_RunCam &other) = delete; AP_RunCam &operator=(const AP_RunCam &) = delete; // get singleton instance static AP_RunCam *get_singleton() { return _singleton; } enum class DeviceType { Disabled = 0, Split = 1, }; // operation of camera button simulation enum class ControlOperation { RCDEVICE_PROTOCOL_SIMULATE_WIFI_BTN = 0x00, // WiFi/Mode button RCDEVICE_PROTOCOL_SIMULATE_POWER_BTN = 0x01, RCDEVICE_PROTOCOL_CHANGE_MODE = 0x02, RCDEVICE_PROTOCOL_CHANGE_START_RECORDING = 0x03, RCDEVICE_PROTOCOL_CHANGE_STOP_RECORDING = 0x04, UNKNOWN_CAMERA_OPERATION = 0xFF }; // control for OSD menu entry enum class ControlOption { STICK_YAW_RIGHT = (1 << 0), STICK_ROLL_RIGHT = (1 << 1), THREE_POS_SWITCH = (1 << 2), TWO_POS_SWITCH = (1 << 3) }; // initialize the RunCam driver void init(); // camera button simulation bool simulate_camera_button(const ControlOperation operation); // start the video void start_recording(); // stop the video void stop_recording(); // enter the OSD menu void enter_osd(); // exit the OSD menu void exit_osd(); // OSD control determined by camera options void osd_option(); // update loop void update(); // Check whether arming is allowed bool pre_arm_check(char *failure_msg, const uint8_t failure_msg_len) const; static const struct AP_Param::GroupInfo var_info[]; private: // definitions prefixed with RCDEVICE taken from https://support.runcam.com/hc/en-us/articles/360014537794-RunCam-Device-Protocol // possible supported features enum class Feature { RCDEVICE_PROTOCOL_FEATURE_SIMULATE_POWER_BUTTON = (1 << 0), RCDEVICE_PROTOCOL_FEATURE_SIMULATE_WIFI_BUTTON = (1 << 1), // WiFi/Mode button RCDEVICE_PROTOCOL_FEATURE_CHANGE_MODE = (1 << 2), RCDEVICE_PROTOCOL_FEATURE_SIMULATE_5_KEY_OSD_CABLE = (1 << 3), RCDEVICE_PROTOCOL_FEATURE_DEVICE_SETTINGS_ACCESS = (1 << 4), RCDEVICE_PROTOCOL_FEATURE_DISPLAY_PORT = (1 << 5), RCDEVICE_PROTOCOL_FEATURE_START_RECORDING = (1 << 6), RCDEVICE_PROTOCOL_FEATURE_STOP_RECORDING = (1 << 7), FEATURES_OVERRIDE = (1 << 14) }; // camera control commands enum class Command { RCDEVICE_PROTOCOL_COMMAND_GET_DEVICE_INFO = 0x00, RCDEVICE_PROTOCOL_COMMAND_CAMERA_CONTROL = 0x01, RCDEVICE_PROTOCOL_COMMAND_5KEY_SIMULATION_PRESS = 0x02, RCDEVICE_PROTOCOL_COMMAND_5KEY_SIMULATION_RELEASE = 0x03, RCDEVICE_PROTOCOL_COMMAND_5KEY_CONNECTION = 0x04, COMMAND_NONE }; // operation of RC5KEY_CONNECTION enum class ConnectionOperation { RCDEVICE_PROTOCOL_5KEY_FUNCTION_OPEN = 0x01, RCDEVICE_PROTOCOL_5KEY_FUNCTION_CLOSE = 0x02 }; // operation of 5 Key OSD cable simulation enum class SimulationOperation { SIMULATION_NONE = 0x00, RCDEVICE_PROTOCOL_5KEY_SIMULATION_SET = 0x01, RCDEVICE_PROTOCOL_5KEY_SIMULATION_LEFT = 0x02, RCDEVICE_PROTOCOL_5KEY_SIMULATION_RIGHT = 0x03, RCDEVICE_PROTOCOL_5KEY_SIMULATION_UP = 0x04, RCDEVICE_PROTOCOL_5KEY_SIMULATION_DOWN = 0x05 }; // protocol versions, only version 1.0 is supported enum class ProtocolVersion { RCSPLIT_VERSION = 0x00, // unsupported firmware version <= 1.1.0 VERSION_1_0 = 0x01, UNKNOWN }; // status of command enum class RequestStatus { NONE, PENDING, SUCCESS, INCORRECT_CRC, TIMEOUT }; enum class State { INITIALIZING, // uart open INITIALIZED, // features received READY, VIDEO_RECORDING, ENTERING_MENU, IN_MENU, EXITING_MENU }; enum class Event { NONE, ENTER_MENU, EXIT_MENU, IN_MENU_ENTER, IN_MENU_RIGHT, // only used by the 5-key process IN_MENU_UP, IN_MENU_DOWN, IN_MENU_EXIT, BUTTON_RELEASE, STOP_RECORDING, START_RECORDING }; enum class OSDOption { NONE, ENTER, EXIT, OPTION, NO_OPTION }; static const uint8_t RUNCAM_NUM_SUB_MENUS = 5; static const uint8_t RUNCAM_NUM_EXPECTED_RESPONSES = 4; static const uint8_t RUNCAM_MAX_MENUS = 1; static const uint8_t RUNCAM_MAX_MENU_LENGTH = 6; static const uint8_t RUNCAM_MAX_DEVICE_TYPES = 1; // supported features, usually probed from the device AP_Int16 _features; // delay time to make sure the camera is fully booted AP_Int32 _boot_delay_ms; // delay time to make sure a button press has been activated AP_Int32 _button_delay_ms; // delay time to make sure a mode change has been activated AP_Int32 _mode_delay_ms; // runcam type/firmware revision AP_Int8 _cam_type; // runcam control options AP_Int8 _cam_control_option; // video on/off bool _video_recording = true; // detected protocol version ProtocolVersion _protocol_version = ProtocolVersion::UNKNOWN; // uart for the device AP_HAL::UARTDriver *uart; // camera state State _state = State::INITIALIZING; // time since last OSD cycle uint32_t _last_osd_update_ms; // start time of the current button press or boot sequence uint32_t _transition_start_ms; // timeout of the current button press or boot sequence uint32_t _transition_timeout_ms; // OSD state machine: button has been pressed bool _button_pressed; // OSD state machine: waiting for a response bool _waiting_device_response; // OSD option from RC switches OSDOption _osd_option; // OSD state mechine: in the menu, value indicates depth uint8_t _in_menu; // OSD state machine: current selection in the top menu int8_t _top_menu_pos; // OSD state machine: current selection in the sub menu uint8_t _sub_menu_pos; // lengths of the sub-menus static uint8_t _sub_menu_lengths[RUNCAM_NUM_SUB_MENUS]; // shared inbound scratch space uint8_t _recv_buf[RUNCAM_MAX_PACKET_SIZE]; // all the response contexts use same recv buffer class Request; FUNCTOR_TYPEDEF(parse_func_t, void, const Request&); // class to represent a request class Request { friend class AP_RunCam; public: Request(AP_RunCam *device, Command commandID, uint8_t param, uint32_t timeout, uint16_t maxRetryTimes, parse_func_t parserFunc); Request() { _command = Command::COMMAND_NONE; } uint8_t *_recv_buf; // response data buffer AP_RunCam *_device; // parent device Command _command; // command for which a response is expected uint8_t _param; // parameter data, the protocol can take more but we never use it private: uint8_t _recv_response_length; // length of the data received uint8_t _expected_response_length; // total length of response data wanted uint32_t _timeout_ms; // how long to wait before giving up uint32_t _request_timestamp_ms; // when the request was sent, if it's zero keep waiting for the response uint16_t _max_retry_times; // number of times to resend the request parse_func_t _parser_func; // function to parse the response RequestStatus _result; // whether we were successful or not // get the length of the expected response uint8_t get_expected_response_length(const Command command) const; // calculate a crc uint8_t get_crc() const; // parse the response void parse_response() { if (_parser_func != nullptr) { _parser_func(*this); } } struct Length { Command command; uint8_t reponse_length; }; static Length _expected_responses_length[RUNCAM_NUM_EXPECTED_RESPONSES]; } _pending_request; // menu structure of the runcam device struct Menu { uint8_t _top_menu_length; uint8_t _sub_menu_lengths[RUNCAM_MAX_MENU_LENGTH]; }; static Menu _menus[RUNCAM_MAX_DEVICE_TYPES]; // return the length of the top menu uint8_t get_top_menu_length() const { return _menus[_cam_type]._top_menu_length; } // return the length of a particular sub-menu uint8_t get_sub_menu_length(uint8_t submenu) const { return _menus[_cam_type]._sub_menu_lengths[submenu]; } // start the counter for a button press void set_button_press_timeout() { _transition_timeout_ms = _button_delay_ms; _button_pressed = true; } // start the counter for a mode change void set_mode_change_timeout() { _transition_timeout_ms = _mode_delay_ms; _button_pressed = true; } // disable the OSD display void disable_osd() { #if OSD_ENABLED AP_OSD* osd = AP::osd(); if (osd != nullptr) { osd->disable(); } #endif } // enable the OSD display void enable_osd() { #if OSD_ENABLED AP_OSD* osd = AP::osd(); if (osd != nullptr) { osd->enable(); } #endif } // OSD update loop void update_osd(); // return radio values as LOW, MIDDLE, HIGH RC_Channel::aux_switch_pos_t get_channel_pos(uint8_t rcmapchan) const; // update the state machine when armed or flying void update_state_machine_armed(); // update the state machine when disarmed void update_state_machine_disarmed(); // handle the initialized state void handle_initialized(Event ev); // handle the ready state void handle_ready(Event ev); // handle the recording state void handle_recording(Event ev); // run the 2-key OSD simulation process void handle_in_menu(Event ev); // map rc input to an event AP_RunCam::Event map_rc_input_to_event() const; // run the 2-key OSD simulation process void handle_2_key_simulation_process(Event ev); // enter the 2 key OSD menu void enter_2_key_osd_menu(); // eexit the 2 key OSD menu void exit_2_key_osd_menu(); // run the 5-key OSD simulation process void handle_5_key_simulation_process(Event ev); // handle a response void handle_5_key_simulation_response(const Request& request); // process a response from the serial port void receive(); // empty the receive side of the serial port void drain(); // get the RunCam device information void get_device_info(); // 5 key osd cable simulation SimulationOperation map_key_to_protocol_operation(const Event ev) const; // send an event void send_5_key_OSD_cable_simulation_event(const Event key); // enter the menu void open_5_key_OSD_cable_connection(parse_func_t parseFunc); // exit the menu void close_5_key_OSD_cable_connection(parse_func_t parseFunc); // press a button void simulate_5_key_OSD_cable_button_press(const SimulationOperation operation, parse_func_t parseFunc); // release a button void simulate_5_key_OSD_cable_button_release(parse_func_t parseFunc); // send a RunCam request and register a response to be processed void send_request_and_waiting_response(Command commandID, uint8_t param, uint32_t timeout, uint16_t maxRetryTimes, parse_func_t parseFunc); // send a packet to the serial port void send_packet(Command command, uint8_t param); // crc functions static uint8_t crc8_high_first(uint8_t *ptr, uint8_t len); static uint8_t crc8_dvb_s2(uint8_t crc, uint8_t a); // handle a device info response void parse_device_info(const Request& request); // wait for the RunCam device to be fully ready bool camera_ready() const; // whether or not the requested feature is supported bool has_feature(const Feature feature) { return _features.get() & uint16_t(feature); } // error handler for OSD simulation void simulation_OSD_cable_failed(const Request& request); // process pending request, retrying as necessary bool request_pending(uint32_t now); static AP_RunCam *_singleton; }; namespace AP { AP_RunCam *runcam(); }; #endif // HAL_RUNCAM_ENABLED