/* 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 . */ #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 32 // note that we allow for more than is in the enum #define SERVO_OUT_MOTOR_MAX 12 // SRV_Channel::k_motor1 ... SRV_Channel::k_motor8, SRV_Channel::k_motor9 ... SRV_Channel::k_motor12 extern const AP_HAL::HAL &hal; void AP_Periph_FW::rcout_init() { // start up with safety enabled. This disables the pwm output until we receive an packet from the rempte system hal.rcout->force_safety_on(); #if HAL_WITH_ESC_TELEM && !HAL_GCS_ENABLED if (g.esc_telem_port >= 0) { serial_manager.set_protocol_and_baud(g.esc_telem_port, AP_SerialManager::SerialProtocol_ESCTelemetry, 115200); } #endif SRV_Channels::init(); #if HAL_PWM_COUNT > 0 for (uint8_t i=0; iset_output_mode(esc_mask, esc_type); if (esc_type >= AP_HAL::RCOutput::MODE_PWM_DSHOT150 && esc_type <= AP_HAL::RCOutput::MODE_PWM_DSHOT1200) { SRV_Channels::set_digital_outputs(esc_mask, 0); } // run this once and at 1Hz to configure aux and esc ranges rcout_init_1Hz(); // run DShot at 1kHz hal.rcout->set_dshot_rate(0, 400); } void AP_Periph_FW::rcout_init_1Hz() { // this runs at 1Hz to allow for run-time param changes SRV_Channels::enable_aux_servos(); for (uint8_t i=0; i 0 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); rcout_has_new_data_to_update = true; #endif } void AP_Periph_FW::rcout_srv_PWM(uint8_t actuator_id, const float command_value) { #if HAL_PWM_COUNT > 0 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_pwm(function, uint16_t(command_value+0.5)); rcout_has_new_data_to_update = true; #endif } 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(); } rcout_has_new_data_to_update = true; } void AP_Periph_FW::rcout_update() { if (!rcout_has_new_data_to_update) { return; } rcout_has_new_data_to_update = false; SRV_Channels::calc_pwm(); SRV_Channels::cork(); SRV_Channels::output_ch_all(); SRV_Channels::push(); #if HAL_WITH_ESC_TELEM uint32_t now_ms = AP_HAL::millis(); if (now_ms - last_esc_telem_update_ms >= 20) { last_esc_telem_update_ms = now_ms; esc_telem_update(); } #endif } #endif // HAL_PERIPH_ENABLE_RC_OUT