From 07aeab5c44c9c1c8cc1764770a1e240c062b85c8 Mon Sep 17 00:00:00 2001 From: Tom Pittenger Date: Sat, 12 Dec 2020 01:08:45 -0800 Subject: [PATCH] AP_Periph: add Servo OUT support --- Tools/AP_Periph/AP_Periph.cpp | 13 ++++- Tools/AP_Periph/AP_Periph.h | 13 ++++- Tools/AP_Periph/Parameters.cpp | 7 +++ Tools/AP_Periph/Parameters.h | 1 + Tools/AP_Periph/README.md | 1 + Tools/AP_Periph/can.cpp | 88 +++++++++++++++++++++++++++-- Tools/AP_Periph/rc_out.cpp | 100 +++++++++++++++++++++++++++++++++ Tools/AP_Periph/wscript | 1 + 8 files changed, 218 insertions(+), 6 deletions(-) create mode 100644 Tools/AP_Periph/rc_out.cpp diff --git a/Tools/AP_Periph/AP_Periph.cpp b/Tools/AP_Periph/AP_Periph.cpp index d4365d2a91..f2b505de25 100644 --- a/Tools/AP_Periph/AP_Periph.cpp +++ b/Tools/AP_Periph/AP_Periph.cpp @@ -125,11 +125,18 @@ void AP_Periph_FW::init() battery.lib.init(); #endif -#ifdef HAL_PERIPH_NEOPIXEL_COUNT +#if defined(HAL_PERIPH_NEOPIXEL_COUNT) || defined(HAL_PERIPH_ENABLE_RC_OUT) hal.rcout->init(); +#endif + +#ifdef HAL_PERIPH_NEOPIXEL_COUNT hal.rcout->set_serial_led_num_LEDs(HAL_PERIPH_NEOPIXEL_CHAN, AP_HAL::RCOutput::MODE_NEOPIXEL); #endif +#ifdef HAL_PERIPH_ENABLE_RC_OUT + rcout_init(); +#endif + #ifdef HAL_PERIPH_ENABLE_ADSB adsb_init(); #endif @@ -272,6 +279,10 @@ void AP_Periph_FW::update() #ifdef HAL_PERIPH_LISTEN_FOR_SERIAL_UART_REBOOT_CMD_PORT check_for_serial_reboot_cmd(HAL_PERIPH_LISTEN_FOR_SERIAL_UART_REBOOT_CMD_PORT); #endif + +#ifdef HAL_PERIPH_ENABLE_RC_OUT + SRV_Channels::enable_aux_servos(); +#endif } static uint32_t last_error_ms; diff --git a/Tools/AP_Periph/AP_Periph.h b/Tools/AP_Periph/AP_Periph.h index 1489dc0261..fc8064e234 100644 --- a/Tools/AP_Periph/AP_Periph.h +++ b/Tools/AP_Periph/AP_Periph.h @@ -5,6 +5,7 @@ #include #include #include +#include "SRV_Channel/SRV_Channel.h" #include #include #include @@ -127,7 +128,17 @@ public: HWESC_Telem hwesc_telem; void hwesc_telem_update(); #endif - + +#ifdef HAL_PERIPH_ENABLE_RC_OUT + SRV_Channels servo_channels; + + void rcout_init(); + void rcout_esc(int16_t *rc, uint8_t num_channels); + void rcout_srv(const uint8_t actuator_id, const float command_value); + void rcout_update(); + void rcout_handle_safety_state(uint8_t safety_state); +#endif + // setup the var_info table AP_Param param_loader{var_info}; diff --git a/Tools/AP_Periph/Parameters.cpp b/Tools/AP_Periph/Parameters.cpp index f11ebf5010..3c6cb70b0d 100644 --- a/Tools/AP_Periph/Parameters.cpp +++ b/Tools/AP_Periph/Parameters.cpp @@ -113,6 +113,13 @@ const AP_Param::Info AP_Periph_FW::var_info[] = { GSCALAR(esc_number, "ESC_NUMBER", 0), #endif +#ifdef HAL_PERIPH_ENABLE_RC_OUT + // Servo driver + // @Group: OUT + // @Path: ../libraries/SRV_Channel/SRV_Channels.cpp + GOBJECT(servo_channels, "OUT", SRV_Channels), +#endif + AP_VAREND }; diff --git a/Tools/AP_Periph/Parameters.h b/Tools/AP_Periph/Parameters.h index 1e3704d963..0fb5501cd1 100644 --- a/Tools/AP_Periph/Parameters.h +++ b/Tools/AP_Periph/Parameters.h @@ -32,6 +32,7 @@ public: k_param_debug, k_param_serial_number, k_param_adsb_port, + k_param_servo_channels, }; AP_Int16 format_version; diff --git a/Tools/AP_Periph/README.md b/Tools/AP_Periph/README.md index fbf7cf6ef9..cd723266bb 100644 --- a/Tools/AP_Periph/README.md +++ b/Tools/AP_Periph/README.md @@ -34,6 +34,7 @@ UAVCAN sensor types. Support is included for: - LEDs (GPIO, I2C or WS2812 serial) - Safety LED and Safety Switch - Buzzer (tonealarm or simple GPIO) + - RC Output (All standard RCOutput protocols) An AP_Periph UAVCAN firmware supports these UAVCAN features: diff --git a/Tools/AP_Periph/can.cpp b/Tools/AP_Periph/can.cpp index 2edb861939..7cf40083f0 100644 --- a/Tools/AP_Periph/can.cpp +++ b/Tools/AP_Periph/can.cpp @@ -44,6 +44,9 @@ #include #include #include +#include +#include +#include #include #include #include @@ -495,7 +498,7 @@ static void can_buzzer_update(void) } #endif // HAL_PERIPH_ENABLE_BUZZER -#ifdef HAL_GPIO_PIN_SAFE_LED +#if defined(HAL_GPIO_PIN_SAFE_LED) || defined(HAL_PERIPH_ENABLE_RC_OUT) static uint8_t safety_state; /* @@ -508,6 +511,9 @@ static void handle_safety_state(CanardInstance* ins, CanardRxTransfer* transfer) return; } safety_state = req.status; +#ifdef HAL_PERIPH_ENABLE_RC_OUT + periph.rcout_handle_safety_state(safety_state); +#endif } #endif // HAL_GPIO_PIN_SAFE_LED @@ -604,6 +610,59 @@ static void handle_lightscommand(CanardInstance* ins, CanardRxTransfer* transfer } #endif // AP_PERIPH_HAVE_LED +#ifdef HAL_PERIPH_ENABLE_RC_OUT +static void handle_esc_rawcommand(CanardInstance* ins, CanardRxTransfer* transfer) +{ + uavcan_equipment_esc_RawCommand cmd; + uint8_t arraybuf[UAVCAN_EQUIPMENT_ESC_RAWCOMMAND_MAX_SIZE]; + uint8_t *arraybuf_ptr = arraybuf; + if (uavcan_equipment_esc_RawCommand_decode(transfer, transfer->payload_len, &cmd, &arraybuf_ptr) < 0) { + return; + } + periph.rcout_esc(cmd.cmd.data, cmd.cmd.len); +} + +static void handle_act_command(CanardInstance* ins, CanardRxTransfer* transfer) +{ + // manual decoding due to TAO bug in libcanard generated code + if (transfer->payload_len < 1 || transfer->payload_len > UAVCAN_EQUIPMENT_ACTUATOR_ARRAYCOMMAND_MAX_SIZE+1) { + return; + } + + const uint8_t data_count = (transfer->payload_len / UAVCAN_EQUIPMENT_ACTUATOR_COMMAND_MAX_SIZE); + uavcan_equipment_actuator_Command data[data_count] {}; + + uint32_t offset = 0; + for (uint8_t i=0; i. + */ +#include +#ifdef HAL_PERIPH_ENABLE_RC_OUT +#include "AP_Periph.h" + +// magic value from UAVCAN driver packet +// dsdl/uavcan/equipment/esc/1030.RawCommand.uavcan +// Raw ESC command normalized into [-8192, 8191] +#define UAVCAN_ESC_MAX_VALUE 8191 + +#define SERVO_OUT_RCIN_MAX 16 // SRV_Channel::k_rcin1 ... SRV_Channel::k_rcin16 +#define SERVO_OUT_MOTOR_MAX 12 // SRV_Channel::k_motor1 ... SRV_Channel::k_motor8, SRV_Channel::k_motor9 ... SRV_Channel::k_motor12 + +#if HAL_PWM_COUNT == 0 + #error "You must define a PWM output in your hwdef.dat" +#endif + +static bool has_new_data_to_update; + +extern const AP_HAL::HAL &hal; + +void AP_Periph_FW::rcout_init() +{ + for (uint8_t i=0; i HAL_PWM_COUNT)) { + // not supported or out of range + return; + } + + const SRV_Channel::Aux_servo_function_t function = SRV_Channel::Aux_servo_function_t(SRV_Channel::k_rcin1 + actuator_id - 1); + SRV_Channels::set_output_norm(function, command_value); + + has_new_data_to_update = true; +} + +void AP_Periph_FW::rcout_handle_safety_state(uint8_t safety_state) +{ + if (safety_state == 255) { + hal.rcout->force_safety_off(); + } else { + hal.rcout->force_safety_on(); + } + has_new_data_to_update = true; +} + +void AP_Periph_FW::rcout_update() +{ + if (!has_new_data_to_update) { + return; + } + has_new_data_to_update = false; + + SRV_Channels::calc_pwm(); + SRV_Channels::cork(); + SRV_Channels::output_ch_all(); + SRV_Channels::push(); +} + +#endif // HAL_PERIPH_ENABLE_RC_OUT + diff --git a/Tools/AP_Periph/wscript b/Tools/AP_Periph/wscript index 9d2729ddce..905fe40acb 100644 --- a/Tools/AP_Periph/wscript +++ b/Tools/AP_Periph/wscript @@ -37,6 +37,7 @@ def build(bld): 'AP_RangeFinder', 'AP_ROMFS', 'AP_MSP', + 'SRV_Channel', ], exclude_src=[ 'libraries/AP_HAL_ChibiOS/Storage.cpp'