/* * 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 struct RCOutput::pwm_group RCOutput::pwm_group_list[] = { HAL_PWM_GROUPS }; #define NUM_GROUPS ARRAY_SIZE_SIMPLE(pwm_group_list) #define CHAN_DISABLED 255 void RCOutput::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); for (uint8_t j = 0; j < 4; j++ ) { if (pwm_group_list[i].chan[j] != CHAN_DISABLED) { total_channels = MAX(total_channels, pwm_group_list[i].chan[j]+1); } } } #if HAL_WITH_IO_MCU if (AP_BoardConfig::io_enabled()) { iomcu.init(); // with IOMCU the local channels start at 8 chan_offset = 8; total_channels += chan_offset; } #endif } void RCOutput::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; // 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++ ) { uint16_t grp_ch_mask = 0; for (uint8_t j=0; j<4; j++) { if (pwm_group_list[i].chan[j] != CHAN_DISABLED) { grp_ch_mask |= (1U< 400) { freq_set = 400; } bool changed_clock = false; if (freq_set > 400 && pwm_group_list[i].pwm_cfg.frequency == 1000000) { // need to change to an 8MHz clock pwm_group_list[i].pwm_cfg.frequency = 8000000; changed_clock = true; } else if (freq_set <= 400 && pwm_group_list[i].pwm_cfg.frequency == 8000000) { // need to change to an 1MHz clock pwm_group_list[i].pwm_cfg.frequency = 1000000; } if (changed_clock) { pwmStop(pwm_group_list[i].pwm_drv); pwmStart(pwm_group_list[i].pwm_drv, &pwm_group_list[i].pwm_cfg); } pwmChangePeriod(pwm_group_list[i].pwm_drv, pwm_group_list[i].pwm_cfg.frequency/freq_set); } } if (freq_hz > 50) { fast_channel_mask |= update_mask; } if (chmask != update_mask) { hal.console->printf("RCOutput: Failed to set PWM frequency req %x set %x\n", (unsigned)chmask, (unsigned)update_mask); } } /* set default output rate */ void RCOutput::set_default_rate(uint16_t freq_hz) { #if HAL_WITH_IO_MCU if (AP_BoardConfig::io_enabled()) { iomcu.set_default_rate(freq_hz); } #endif for (uint8_t i = 0; i < NUM_GROUPS; i++ ) { uint16_t grp_ch_mask = 0; for (uint8_t j=0; j<4; j++) { if (pwm_group_list[i].chan[j] != CHAN_DISABLED) { grp_ch_mask |= (1U<= total_channels) { return 0; } #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 RCOutput::enable_ch(uint8_t chan) { if (chan >= total_channels) { return; } 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<= total_channels) { return; } 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) { pwmDisableChannel(pwm_group_list[i].pwm_drv, j); en_mask &= ~(1<= total_channels) { return; } last_sent[chan] = period_us; #if HAL_WITH_IO_MCU // handle IO MCU channels if (AP_BoardConfig::io_enabled()) { iomcu.write_channel(chan, period_us); } #endif if (chan < chan_offset) { return; } chan -= chan_offset; period[chan] = period_us; num_channels = MAX(chan+1, num_channels); if (!corked) { push_local(); } } /* push values to local channels from period[] array */ void RCOutput::push_local(void) { if (num_channels == 0) { return; } uint16_t outmask = (1U<= _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 { uint32_t width = (pwm_group_list[i].pwm_cfg.frequency/1000000) * period_us; pwmEnableChannel(pwm_group_list[i].pwm_drv, j, width); } } } } } uint16_t RCOutput::read(uint8_t chan) { if (chan >= total_channels) { return 0; } #if HAL_WITH_IO_MCU if (chan < chan_offset) { return iomcu.read_channel(chan); } #endif chan -= chan_offset; return period[chan]; } void RCOutput::read(uint16_t* period_us, uint8_t len) { if (len > total_channels) { len = total_channels; } #if HAL_WITH_IO_MCU for (uint8_t i=0; i= total_channels) { return 0; } return last_sent[chan]; } void RCOutput::read_last_sent(uint16_t* period_us, uint8_t len) { if (len > total_channels) { len = total_channels; } for (uint8_t i=0; i