mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-18 06:38:29 -04:00
AP_HAL: add RCOutput::get_output_mode_banner
This commit is contained in:
parent
a54f24c8c0
commit
a71ae54d3a
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) {}
|
||||
|
||||
/*
|
||||
* 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
|
||||
*/
|
||||
@ -209,4 +214,10 @@ public:
|
||||
trigger send of neopixel data
|
||||
*/
|
||||
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
|
||||
if (AP_BoardConfig::io_enabled()) {
|
||||
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
|
||||
// of 8 here for oneshot125
|
||||
io_period_us /= 8;
|
||||
@ -714,20 +714,85 @@ void RCOutput::set_output_mode(uint16_t mask, enum output_mode mode)
|
||||
mode == MODE_PWM_ONESHOT125) &&
|
||||
(mask & ((1U<<chan_offset)-1)) &&
|
||||
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
|
||||
// when we trigger
|
||||
iomcu.set_freq(io_fast_channel_mask, 1);
|
||||
return iomcu.set_oneshot_mode();
|
||||
iomcu.set_oneshot_mode();
|
||||
return;
|
||||
}
|
||||
if (mode == MODE_PWM_BRUSHED &&
|
||||
(mask & ((1U<<chan_offset)-1)) &&
|
||||
AP_BoardConfig::io_enabled()) {
|
||||
return iomcu.set_brushed_mode();
|
||||
iomcu_mode = mode;
|
||||
iomcu.set_brushed_mode();
|
||||
return;
|
||||
}
|
||||
#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
|
||||
*/
|
||||
|
@ -49,6 +49,7 @@ public:
|
||||
return true;
|
||||
}
|
||||
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 {
|
||||
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
|
||||
uint16_t trigger_widest_pulse;
|
||||
|
||||
// are we using oneshot125 for the iomcu?
|
||||
bool iomcu_oneshot125;
|
||||
// iomcu output mode (pwm, oneshot or oneshot125)
|
||||
enum output_mode iomcu_mode = MODE_PWM_NORMAL;
|
||||
|
||||
// find a channel group given a channel number
|
||||
struct pwm_group *find_chan(uint8_t chan, uint8_t &group_idx);
|
||||
|
Loading…
Reference in New Issue
Block a user