/* * This file 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 file 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 . * * Code by Andrew Tridgell and Siddharth Bharat Purohit */ #include "RCOutput.h" #include #include using namespace ChibiOS; extern const AP_HAL::HAL& hal; #if HAL_WITH_IO_MCU #include extern AP_IOMCU iomcu; #endif #define PWM_CLK_FREQ 8000000 #define PWM_US_WIDTH_FROM_CLK(x) ((PWM_CLK_FREQ/1000000)*x) const struct ChibiRCOutput::pwm_group ChibiRCOutput::pwm_group_list[] = { //Group 1 Config { //Channels in the Group and respective mapping {0, 1, 2, 3}, //Group Initial Config { 8000000, /* 8MHz PWM clock frequency. */ 160000, /* Initial PWM period 20ms. */ NULL, { //Channel Config {PWM_OUTPUT_ACTIVE_HIGH, NULL}, {PWM_OUTPUT_ACTIVE_HIGH, NULL}, {PWM_OUTPUT_ACTIVE_HIGH, NULL}, {PWM_OUTPUT_ACTIVE_HIGH, NULL} }, 0, 0 }, #if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_FMUV3 &PWMD1 #elif CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_SKYVIPER_F412 &PWMD3 #endif } }; #define NUM_GROUPS ARRAY_SIZE(pwm_group_list) void ChibiRCOutput::init() { for (uint8_t i = 0; i < NUM_GROUPS; i++ ) { //Start Pwm groups pwmStart(pwm_group_list[i].pwm_drv, &pwm_group_list[i].pwm_cfg); } #if HAL_WITH_IO_MCU if (AP_BoardConfig::io_enabled()) { iomcu.init(); // with IOMCU the local channels start at 8 chan_offset = 8; } #endif } void ChibiRCOutput::set_freq(uint32_t chmask, uint16_t freq_hz) { //check if the request spans accross any of the channel groups uint8_t update_mask = 0; uint32_t grp_ch_mask; // greater than 400 doesn't give enough room at higher periods for // the down pulse if (freq_hz > 400 && _output_mode != MODE_PWM_BRUSHED) { freq_hz = 400; } #if HAL_WITH_IO_MCU if (AP_BoardConfig::io_enabled()) { iomcu.set_freq(chmask, freq_hz); } #endif chmask >>= chan_offset; if (chmask == 0) { return; } for (uint8_t i = 0; i < NUM_GROUPS; i++ ) { grp_ch_mask = 0XF; if ((grp_ch_mask & chmask) == grp_ch_mask) { update_mask |= grp_ch_mask; pwmChangePeriod(pwm_group_list[i].pwm_drv, pwm_group_list[i].pwm_cfg.frequency/freq_hz); } } if (chmask != update_mask) { hal.console->printf("RCOutput: Failed to set PWM frequency req %x set %x\n", (unsigned)chmask, (unsigned)update_mask); } } uint16_t ChibiRCOutput::get_freq(uint8_t chan) { #if HAL_WITH_IO_MCU if (chan < chan_offset) { return iomcu.get_freq(chan); } #endif chan -= chan_offset; for (uint8_t i = 0; i < NUM_GROUPS; i++ ) { for (uint8_t j = 0; j < 4; j++) { if (pwm_group_list[i].chan[j] == chan) { return pwm_group_list[i].pwm_drv->config->frequency / pwm_group_list[i].pwm_drv->period; } } } // assume 50Hz default return 50; } void ChibiRCOutput::enable_ch(uint8_t chan) { if (chan < chan_offset) { return; } chan -= chan_offset; for (uint8_t i = 0; i < NUM_GROUPS; i++ ) { for (uint8_t j = 0; j < 4; j++) { if ((pwm_group_list[i].chan[j] == chan) && !(en_mask & 1<= _esc_pwm_max) { period_us = PWM_FRACTION_TO_WIDTH(pwm_group_list[i].pwm_drv, 1, 1); } else { period_us = PWM_FRACTION_TO_WIDTH(pwm_group_list[i].pwm_drv,\ (_esc_pwm_max - _esc_pwm_min), (period_us - _esc_pwm_min)); } pwmEnableChannel(pwm_group_list[i].pwm_drv, j, period_us); } else { pwmEnableChannel(pwm_group_list[i].pwm_drv, j, PWM_US_WIDTH_FROM_CLK(period_us)); } } } } } uint16_t ChibiRCOutput::read(uint8_t chan) { #if HAL_WITH_IO_MCU if (chan < chan_offset) { return iomcu.read_channel(chan); } #endif chan -= chan_offset; return period[chan]; } void ChibiRCOutput::read(uint16_t* period_us, uint8_t len) { #if HAL_WITH_IO_MCU for (uint8_t i=0; i