AP_HAL: add RCOutput::get_output_mode_banner
This commit is contained in:
parent
dfbccf3811
commit
95ec629923
52
libraries/AP_HAL/RCOutput.cpp
Normal file
52
libraries/AP_HAL/RCOutput.cpp
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
#include "AP_HAL.h"
|
||||||
|
|
||||||
|
extern const AP_HAL::HAL &hal;
|
||||||
|
|
||||||
|
// helper function for implementation of get_output_mode_banner
|
||||||
|
const char* AP_HAL::RCOutput::get_output_mode_string(enum output_mode out_mode) const
|
||||||
|
{
|
||||||
|
// convert mode to string
|
||||||
|
switch (out_mode) {
|
||||||
|
case MODE_PWM_NONE:
|
||||||
|
return "None";
|
||||||
|
case MODE_PWM_NORMAL:
|
||||||
|
return "PWM";
|
||||||
|
case MODE_PWM_ONESHOT:
|
||||||
|
return "OneS";
|
||||||
|
case MODE_PWM_ONESHOT125:
|
||||||
|
return "OS125";
|
||||||
|
case MODE_PWM_BRUSHED:
|
||||||
|
return "Brush";
|
||||||
|
case MODE_PWM_DSHOT150:
|
||||||
|
return "DS150";
|
||||||
|
case MODE_PWM_DSHOT300:
|
||||||
|
return "DS300";
|
||||||
|
case MODE_PWM_DSHOT600:
|
||||||
|
return "DS600";
|
||||||
|
case MODE_PWM_DSHOT1200:
|
||||||
|
return "DS1200";
|
||||||
|
case MODE_NEOPIXEL:
|
||||||
|
return "NeoP";
|
||||||
|
}
|
||||||
|
|
||||||
|
// we should never reach here but just in case
|
||||||
|
return "Unknown";
|
||||||
|
}
|
||||||
|
|
||||||
|
// convert output mode to string. helper function for implementation of get_output_mode_banner
|
||||||
|
void AP_HAL::RCOutput::append_to_banner(char banner_msg[], uint8_t banner_msg_len, output_mode out_mode, uint8_t low_ch, uint8_t high_ch) const
|
||||||
|
{
|
||||||
|
const char* mode_str = get_output_mode_string(out_mode);
|
||||||
|
|
||||||
|
// make copy of banner_msg
|
||||||
|
char banner_msg_temp[banner_msg_len];
|
||||||
|
memcpy(banner_msg_temp, banner_msg, banner_msg_len);
|
||||||
|
|
||||||
|
if (low_ch == high_ch) {
|
||||||
|
// handle single channel case
|
||||||
|
hal.util->snprintf(banner_msg, banner_msg_len, "%s %s:%u", banner_msg_temp, mode_str, (unsigned)low_ch);
|
||||||
|
} else {
|
||||||
|
// the general case
|
||||||
|
hal.util->snprintf(banner_msg, banner_msg_len, "%s %s:%u-%u", banner_msg_temp, mode_str, (unsigned)low_ch, (unsigned)high_ch);
|
||||||
|
}
|
||||||
|
}
|
@ -182,6 +182,11 @@ public:
|
|||||||
};
|
};
|
||||||
virtual void set_output_mode(uint16_t mask, enum output_mode mode) {}
|
virtual void set_output_mode(uint16_t mask, enum output_mode mode) {}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* get output mode banner to inform user of how outputs are configured
|
||||||
|
*/
|
||||||
|
virtual bool get_output_mode_banner(char banner_msg[], uint8_t banner_msg_len) const { return false; }
|
||||||
|
|
||||||
/*
|
/*
|
||||||
set default update rate
|
set default update rate
|
||||||
*/
|
*/
|
||||||
@ -209,4 +214,10 @@ public:
|
|||||||
trigger send of neopixel data
|
trigger send of neopixel data
|
||||||
*/
|
*/
|
||||||
virtual void neopixel_send(void) {}
|
virtual void neopixel_send(void) {}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
// helper functions for implementation of get_output_mode_banner
|
||||||
|
void append_to_banner(char banner_msg[], uint8_t banner_msg_len, output_mode out_mode, uint8_t low_ch, uint8_t high_ch) const;
|
||||||
|
const char* get_output_mode_string(enum output_mode out_mode) const;
|
||||||
};
|
};
|
||||||
|
@ -312,7 +312,7 @@ void RCOutput::write(uint8_t chan, uint16_t period_us)
|
|||||||
// handle IO MCU channels
|
// handle IO MCU channels
|
||||||
if (AP_BoardConfig::io_enabled()) {
|
if (AP_BoardConfig::io_enabled()) {
|
||||||
uint16_t io_period_us = period_us;
|
uint16_t io_period_us = period_us;
|
||||||
if (iomcu_oneshot125 && ((1U<<chan) & io_fast_channel_mask)) {
|
if ((iomcu_mode == MODE_PWM_ONESHOT125) && ((1U<<chan) & io_fast_channel_mask)) {
|
||||||
// the iomcu only has one oneshot setting, so we need to scale by a factor
|
// the iomcu only has one oneshot setting, so we need to scale by a factor
|
||||||
// of 8 here for oneshot125
|
// of 8 here for oneshot125
|
||||||
io_period_us /= 8;
|
io_period_us /= 8;
|
||||||
@ -714,20 +714,85 @@ void RCOutput::set_output_mode(uint16_t mask, enum output_mode mode)
|
|||||||
mode == MODE_PWM_ONESHOT125) &&
|
mode == MODE_PWM_ONESHOT125) &&
|
||||||
(mask & ((1U<<chan_offset)-1)) &&
|
(mask & ((1U<<chan_offset)-1)) &&
|
||||||
AP_BoardConfig::io_enabled()) {
|
AP_BoardConfig::io_enabled()) {
|
||||||
iomcu_oneshot125 = (mode == MODE_PWM_ONESHOT125);
|
iomcu_mode = mode;
|
||||||
// also setup IO to use a 1Hz frequency, so we only get output
|
// also setup IO to use a 1Hz frequency, so we only get output
|
||||||
// when we trigger
|
// when we trigger
|
||||||
iomcu.set_freq(io_fast_channel_mask, 1);
|
iomcu.set_freq(io_fast_channel_mask, 1);
|
||||||
return iomcu.set_oneshot_mode();
|
iomcu.set_oneshot_mode();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
if (mode == MODE_PWM_BRUSHED &&
|
if (mode == MODE_PWM_BRUSHED &&
|
||||||
(mask & ((1U<<chan_offset)-1)) &&
|
(mask & ((1U<<chan_offset)-1)) &&
|
||||||
AP_BoardConfig::io_enabled()) {
|
AP_BoardConfig::io_enabled()) {
|
||||||
return iomcu.set_brushed_mode();
|
iomcu_mode = mode;
|
||||||
|
iomcu.set_brushed_mode();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* get output mode banner to inform user of how outputs are configured
|
||||||
|
*/
|
||||||
|
bool RCOutput::get_output_mode_banner(char banner_msg[], uint8_t banner_msg_len) const
|
||||||
|
{
|
||||||
|
// create array of each channel's mode
|
||||||
|
output_mode ch_mode[chan_offset + NUM_GROUPS * ARRAY_SIZE(pwm_group::chan)] = {};
|
||||||
|
bool have_nonzero_modes = false;
|
||||||
|
|
||||||
|
#if HAL_WITH_IO_MCU
|
||||||
|
// fill in ch_mode array for IOMCU channels
|
||||||
|
if (AP_BoardConfig::io_enabled()) {
|
||||||
|
for (uint8_t i = 0; i < chan_offset; i++ ) {
|
||||||
|
ch_mode[i] = iomcu_mode;
|
||||||
|
}
|
||||||
|
have_nonzero_modes = (chan_offset > 0) && (iomcu_mode != MODE_PWM_NONE);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// fill in ch_mode array for FMU channels
|
||||||
|
for (uint8_t i = 0; i < NUM_GROUPS; i++ ) {
|
||||||
|
const pwm_group &group = pwm_group_list[i];
|
||||||
|
if (group.current_mode != MODE_PWM_NONE) {
|
||||||
|
for (uint8_t j = 0; j < ARRAY_SIZE(group.chan); j++) {
|
||||||
|
if (group.chan[j] != CHAN_DISABLED) {
|
||||||
|
const uint8_t chan_num = group.chan[j] + chan_offset;
|
||||||
|
if (chan_num < ARRAY_SIZE(ch_mode)) {
|
||||||
|
ch_mode[chan_num] = group.current_mode;
|
||||||
|
have_nonzero_modes = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle simple case
|
||||||
|
if (!have_nonzero_modes) {
|
||||||
|
hal.util->snprintf(banner_msg, banner_msg_len, "RCOut: None");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// write banner to banner_msg
|
||||||
|
hal.util->snprintf(banner_msg, banner_msg_len, "RCOut:");
|
||||||
|
uint8_t curr_mode_lowest_ch = 0;
|
||||||
|
for (uint8_t k = 1; k < ARRAY_SIZE(ch_mode); k++) {
|
||||||
|
if (ch_mode[k-1] != ch_mode[k]) {
|
||||||
|
if (ch_mode[k-1] != MODE_PWM_NONE) {
|
||||||
|
append_to_banner(banner_msg, banner_msg_len, ch_mode[k-1], curr_mode_lowest_ch + 1, k);
|
||||||
|
}
|
||||||
|
curr_mode_lowest_ch = k;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// add final few channel's mode to banner (won't have been done by above loop)
|
||||||
|
const uint8_t final_index = ARRAY_SIZE(ch_mode)-1;
|
||||||
|
if (ch_mode[final_index] != MODE_PWM_NONE) {
|
||||||
|
append_to_banner(banner_msg, banner_msg_len, ch_mode[final_index], curr_mode_lowest_ch + 1, final_index + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
start corking output
|
start corking output
|
||||||
*/
|
*/
|
||||||
|
@ -49,6 +49,7 @@ public:
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
void set_output_mode(uint16_t mask, enum output_mode mode) override;
|
void set_output_mode(uint16_t mask, enum output_mode mode) override;
|
||||||
|
bool get_output_mode_banner(char banner_msg[], uint8_t banner_msg_len) const override;
|
||||||
|
|
||||||
float scale_esc_to_unity(uint16_t pwm) override {
|
float scale_esc_to_unity(uint16_t pwm) override {
|
||||||
return 2.0 * ((float) pwm - _esc_pwm_min) / (_esc_pwm_max - _esc_pwm_min) - 1.0;
|
return 2.0 * ((float) pwm - _esc_pwm_min) / (_esc_pwm_max - _esc_pwm_min) - 1.0;
|
||||||
@ -294,8 +295,8 @@ private:
|
|||||||
// widest pulse for oneshot triggering
|
// widest pulse for oneshot triggering
|
||||||
uint16_t trigger_widest_pulse;
|
uint16_t trigger_widest_pulse;
|
||||||
|
|
||||||
// are we using oneshot125 for the iomcu?
|
// iomcu output mode (pwm, oneshot or oneshot125)
|
||||||
bool iomcu_oneshot125;
|
enum output_mode iomcu_mode = MODE_PWM_NORMAL;
|
||||||
|
|
||||||
// find a channel group given a channel number
|
// find a channel group given a channel number
|
||||||
struct pwm_group *find_chan(uint8_t chan, uint8_t &group_idx);
|
struct pwm_group *find_chan(uint8_t chan, uint8_t &group_idx);
|
||||||
|
Loading…
Reference in New Issue
Block a user