From 2d5588ae02560aae029844410212aab38785d671 Mon Sep 17 00:00:00 2001 From: Mark Whitehorn <kd0aij@gmail.com> Date: Sat, 18 Feb 2017 20:35:13 -0700 Subject: [PATCH] simplify oneshot mode selection; use zero PWM rate as indicator cleanup and remove unused (new) params --- src/drivers/px4fmu/fmu.cpp | 60 +++++++-------------------- src/drivers/stm32/drv_io_timer.c | 32 +++++++------- src/drivers/stm32/drv_io_timer.h | 3 +- src/drivers/stm32/drv_pwm_servo.c | 50 ++++++---------------- src/modules/px4iofirmware/mixer.cpp | 4 +- src/modules/px4iofirmware/px4io.c | 1 - src/modules/px4iofirmware/registers.c | 19 +++++---- src/systemcmds/pwm/pwm.c | 11 ++--- 8 files changed, 64 insertions(+), 116 deletions(-) diff --git a/src/drivers/px4fmu/fmu.cpp b/src/drivers/px4fmu/fmu.cpp index 9aae69f5ac..526fb71b14 100644 --- a/src/drivers/px4fmu/fmu.cpp +++ b/src/drivers/px4fmu/fmu.cpp @@ -184,6 +184,7 @@ private: static const unsigned _max_actuators = DIRECT_PWM_OUTPUT_CHANNELS; Mode _mode; + bool _oneshot_mode; unsigned _pwm_default_rate; unsigned _pwm_alt_rate; uint32_t _pwm_alt_rate_channels; @@ -322,6 +323,7 @@ PX4FMU *g_fmu; PX4FMU::PX4FMU() : CDev("fmu", PX4FMU_DEVICE_PATH), _mode(MODE_NONE), + _oneshot_mode(false), _pwm_default_rate(50), _pwm_alt_rate(50), _pwm_alt_rate_channels(0), @@ -466,7 +468,7 @@ PX4FMU::init() _task = px4_task_spawn_cmd("fmuservo", SCHED_DEFAULT, SCHED_PRIORITY_FAST_DRIVER - 1, - 1200, + 1280, (main_t)&PX4FMU::task_main_trampoline, nullptr); @@ -693,6 +695,11 @@ PX4FMU::set_mode(Mode mode) return OK; } +/* This routine is called from two IOCTLs: PWM_SERVO_SET_UPDATE_RATE and PWM_SERVO_SET_SELECT_UPDATE_RATE + * In the first case, it is intended to set the "alternate" PWM rate, which may be [25,400]Hz. + * In the second case, it specifies which channels are set at the alternate rate, with all others + * implicitly set to the default rate. + */ int PX4FMU::set_pwm_rate(uint32_t rate_map, unsigned default_rate, unsigned alt_rate) { @@ -714,7 +721,7 @@ PX4FMU::set_pwm_rate(uint32_t rate_map, unsigned default_rate, unsigned alt_rate if (pass == 0) { // preflight if ((alt != 0) && (alt != mask)) { - warn("rate group %u mask %x bad overlap %x", group, mask, alt); + PX4_WARN("rate group %u mask %x bad overlap %x", group, mask, alt); // not a legal map, bail return -EINVAL; } @@ -722,14 +729,14 @@ PX4FMU::set_pwm_rate(uint32_t rate_map, unsigned default_rate, unsigned alt_rate } else { // set it - errors here are unexpected if (alt != 0) { - if (up_pwm_servo_set_rate_group_update(group, _pwm_alt_rate) != OK) { - warn("rate group set alt failed"); + if (up_pwm_servo_set_rate_group_update(group, alt_rate) != OK) { + PX4_WARN("rate group set alt failed"); return -EINVAL; } } else { - if (up_pwm_servo_set_rate_group_update(group, _pwm_default_rate) != OK) { - warn("rate group set default failed"); + if (up_pwm_servo_set_rate_group_update(group, default_rate) != OK) { + PX4_WARN("rate group set default failed"); return -EINVAL; } } @@ -984,7 +991,6 @@ void PX4FMU::update_pwm_out_state(bool on) { if (on && !_pwm_initialized && _pwm_mask != 0) { - up_pwm_set_oneshot_mode(true); up_pwm_servo_init(_pwm_mask); set_pwm_rate(_pwm_alt_rate_channels, _pwm_default_rate, _pwm_alt_rate); _pwm_initialized = true; @@ -1047,40 +1053,6 @@ PX4FMU::task_main() _current_update_rate = 0; } - /* - * Adjust actuator topic update rate to keep up with - * the highest servo update rate configured. - * - * We always mix at max rate; some channels may update slower. - */ - unsigned max_rate = (_pwm_default_rate > _pwm_alt_rate) ? _pwm_default_rate : _pwm_alt_rate; - - if (_current_update_rate != max_rate) { - _current_update_rate = max_rate; - int update_rate_in_ms = int(1000 / _current_update_rate); - - /* reject faster than 500 Hz updates */ - if (update_rate_in_ms < 2) { - update_rate_in_ms = 2; - } - - /* reject slower than 10 Hz updates */ - if (update_rate_in_ms > 100) { - update_rate_in_ms = 100; - } - - PX4_DEBUG("adjusted actuator update interval to %ums", update_rate_in_ms); - - for (unsigned i = 0; i < actuator_controls_s::NUM_ACTUATOR_CONTROL_GROUPS; i++) { - if (_control_subs[i] > 0) { - orb_set_interval(_control_subs[i], update_rate_in_ms); - } - } - - // set to current max rate, even if we are actually checking slower/faster - _current_update_rate = max_rate; - } - /* wait for an update */ int ret = ::poll(_poll_fds, _poll_fds_num, 20); @@ -1261,10 +1233,8 @@ PX4FMU::task_main() pwm_output_set(i, pwm_limited[i]); } - // force an update if in oneshot mode - if (up_pwm_get_oneshot_mode()) { - up_pwm_force_update(); - } + // force an update of oneshot channels + up_pwm_force_update(); publish_pwm_outputs(pwm_limited, num_outputs); perf_end(_ctl_latency); diff --git a/src/drivers/stm32/drv_io_timer.c b/src/drivers/stm32/drv_io_timer.c index 82eae6776f..0c1d26e9bb 100644 --- a/src/drivers/stm32/drv_io_timer.c +++ b/src/drivers/stm32/drv_io_timer.c @@ -378,8 +378,9 @@ static int allocate_channel(unsigned channel, io_timer_channel_mode_t mode) static int timer_set_rate(unsigned timer, unsigned rate) { - /* configure the timer to update at the desired rate */ + timer_freq[timer] = 1; + rPSC(timer) = (io_timers[timer].clock_freq / 1000000) - 1; rARR(timer) = timer_freq[timer] * 1000000 / rate; /* generate an update event; reloads the counter and all registers */ @@ -388,30 +389,29 @@ static int timer_set_rate(unsigned timer, unsigned rate) return 0; } -//#define FAKE_ONESHOT void io_timer_set_oneshot_mode(unsigned timer) { timer_freq[timer] = 8; + rPSC(timer) = (io_timers[timer].clock_freq / 8000000) - 1; -#ifdef FAKE_ONESHOT - // to eliminate jitter with max additional latency of 500usec, set PWM rate to 2KHz. - // Note that the FMU mixer jitter is on the order of 2msec. - timer_set_rate(timer, 2000); -#else // set rate to 125Hz (lowest possible at 8MHz is ~122Hz) // io_timer_set_ccr() must be called at > 125Hz to minimize latency - timer_set_rate(timer, 125); -#endif + rARR(timer) = timer_freq[timer] * 1000000 / 125; } -extern void io_timer_force_update(unsigned timer) +uint8_t io_timer_get_freq(unsigned timer) { -#ifdef FAKE_ONESHOT - // let's not and say we did -#else - // force update of channel compare register - rEGR(timer) |= GTIM_EGR_UG; -#endif + return timer_freq[timer]; +} + +void io_timer_force_update() +{ + for (int i = 0; i < MAX_IO_TIMERS; i++) { + if (timer_freq[i] == 8) { + // force update of channel compare register + rEGR(i) |= GTIM_EGR_UG; + } + } } int io_timer_init_timer(unsigned timer) diff --git a/src/drivers/stm32/drv_io_timer.h b/src/drivers/stm32/drv_io_timer.h index 9bc61ea5e1..30774784f9 100644 --- a/src/drivers/stm32/drv_io_timer.h +++ b/src/drivers/stm32/drv_io_timer.h @@ -134,6 +134,7 @@ __EXPORT int io_timer_get_mode_channels(io_timer_channel_mode_t mode); * * @param timer The timer to force. */ -__EXPORT extern void io_timer_force_update(unsigned timer); +__EXPORT extern void io_timer_force_update(void); +__EXPORT extern uint8_t io_timer_get_freq(unsigned timer); __END_DECLS diff --git a/src/drivers/stm32/drv_pwm_servo.c b/src/drivers/stm32/drv_pwm_servo.c index 0810ed50d3..2bd86bc9f0 100644 --- a/src/drivers/stm32/drv_pwm_servo.c +++ b/src/drivers/stm32/drv_pwm_servo.c @@ -63,22 +63,6 @@ #include <stm32_tim.h> -/* pwm_oneshot_mode true implies all servo outputs are oneshot */ -static bool pwm_oneshot_mode = false; - -/* this is a bitfield: if bit N is set, that timer is used for oneshot only */ -static uint8_t oneshot_timers; - -void up_pwm_set_oneshot_mode(bool value) -{ - pwm_oneshot_mode = value; -} - -bool up_pwm_get_oneshot_mode() -{ - return pwm_oneshot_mode; -} - int up_pwm_servo_set(unsigned channel, servo_position_t value) { return io_timer_set_ccr(channel, value); @@ -94,10 +78,6 @@ int up_pwm_servo_init(uint32_t channel_mask) /* Init channels */ io_timer_channel_mode_t chmode = IOTimerChanMode_PWMOut; - if (pwm_oneshot_mode) { - chmode = IOTimerChanMode_OneShot; - } - uint32_t current = io_timer_get_mode_channels(chmode); // First free the current set of PWMs @@ -109,8 +89,6 @@ int up_pwm_servo_init(uint32_t channel_mask) } } - oneshot_timers = 0; - // Now allocate the new set for (unsigned channel = 0; channel_mask != 0 && channel < MAX_TIMER_IO_CHANNELS; channel++) { @@ -122,13 +100,13 @@ int up_pwm_servo_init(uint32_t channel_mask) io_timer_free_channel(channel); } + if (io_timer_get_freq(timer_io_channels[channel].timer_index) == 8) { + chmode = IOTimerChanMode_OneShot; + } + io_timer_channel_init(channel, chmode, NULL, NULL); channel_mask &= ~(1 << channel); - if (pwm_oneshot_mode) { - // update the oneshot_timers bitfield - oneshot_timers |= (1 << timer_io_channels[channel].timer_index); - } } } @@ -143,6 +121,12 @@ void up_pwm_servo_deinit(void) int up_pwm_servo_set_rate_group_update(unsigned group, unsigned rate) { + if (rate == 0) { + /* configure this group for OneShot125 PWM */ + io_timer_set_oneshot_mode(group); + return OK; + } + /* limit update rate to 1..10000Hz; somewhat arbitrary but safe */ if (rate < 1) { return -ERANGE; @@ -163,11 +147,7 @@ int up_pwm_servo_set_rate_group_update(unsigned group, unsigned rate) void up_pwm_force_update(void) { - for (unsigned i = 0; i < 8; i++) { - if (oneshot_timers & (1 << i)) { - io_timer_force_update(i); - } - } + io_timer_force_update(); } int up_pwm_servo_set_rate(unsigned rate) @@ -187,10 +167,6 @@ uint32_t up_pwm_servo_get_rate_group(unsigned group) void up_pwm_servo_arm(bool armed) { - if (pwm_oneshot_mode) { - io_timer_set_enable(armed, IOTimerChanMode_OneShot, IO_TIMER_ALL_MODES_CHANNELS); - - } else { - io_timer_set_enable(armed, IOTimerChanMode_PWMOut, IO_TIMER_ALL_MODES_CHANNELS); - } + io_timer_set_enable(armed, IOTimerChanMode_OneShot, IO_TIMER_ALL_MODES_CHANNELS); + io_timer_set_enable(armed, IOTimerChanMode_PWMOut, IO_TIMER_ALL_MODES_CHANNELS); } diff --git a/src/modules/px4iofirmware/mixer.cpp b/src/modules/px4iofirmware/mixer.cpp index 284d194bff..0cdf639d85 100644 --- a/src/modules/px4iofirmware/mixer.cpp +++ b/src/modules/px4iofirmware/mixer.cpp @@ -313,9 +313,7 @@ mixer_tick(void) if (new_fmu_data) { new_fmu_data = false; - if (up_pwm_get_oneshot_mode()) { - up_pwm_force_update(); - } + up_pwm_force_update(); } } diff --git a/src/modules/px4iofirmware/px4io.c b/src/modules/px4iofirmware/px4io.c index ee424a0bb3..b20a6e1f06 100644 --- a/src/modules/px4iofirmware/px4io.c +++ b/src/modules/px4iofirmware/px4io.c @@ -235,7 +235,6 @@ int user_start(int argc, char *argv[]) { /* configure the first 8 PWM outputs (i.e. all of them) */ - up_pwm_set_oneshot_mode(true); up_pwm_servo_init(0xff); #if defined(CONFIG_HAVE_CXX) && defined(CONFIG_HAVE_CXXINITIALIZE) diff --git a/src/modules/px4iofirmware/registers.c b/src/modules/px4iofirmware/registers.c index 5de27777a2..63ecbe1df3 100644 --- a/src/modules/px4iofirmware/registers.c +++ b/src/modules/px4iofirmware/registers.c @@ -325,9 +325,7 @@ registers_set(uint8_t page, uint8_t offset, const uint16_t *values, unsigned num system_state.fmu_data_received_time = hrt_absolute_time(); r_status_flags |= PX4IO_P_STATUS_FLAGS_RAW_PWM; - if (up_pwm_get_oneshot_mode()) { - up_pwm_force_update(); - } + up_pwm_force_update(); break; @@ -632,12 +630,17 @@ registers_set_one(uint8_t page, uint8_t offset, uint16_t value) break; case PX4IO_P_SETUP_PWM_ALTRATE: - if (value < 25) { - value = 25; - } - if (value > 400) { - value = 400; + /* a rate of zero implies oneshot mode */ + if (value != 0) { + /* constrain normal PWM to [25,400]Hz */ + if (value < 25) { + value = 25; + } + + if (value > 400) { + value = 400; + } } pwm_configure_rates(r_setup_pwm_rates, r_setup_pwm_defaultrate, value); diff --git a/src/systemcmds/pwm/pwm.c b/src/systemcmds/pwm/pwm.c index e5fd2f6368..4e37c14303 100644 --- a/src/systemcmds/pwm/pwm.c +++ b/src/systemcmds/pwm/pwm.c @@ -327,15 +327,16 @@ pwm_main(int argc, char *argv[]) } else if (!strcmp(argv[1], "rate")) { /* change alternate PWM rate */ - if (alt_rate > 0) { +// if (alt_rate > 0) { ret = px4_ioctl(fd, PWM_SERVO_SET_UPDATE_RATE, alt_rate); - if (ret != OK) { - PX4_ERR("PWM_SERVO_SET_UPDATE_RATE (check rate for sanity)"); - return error_on_warn; - } + if (ret != OK) { + PX4_ERR("PWM_SERVO_SET_UPDATE_RATE (check rate for sanity)"); + return error_on_warn; } +// } + /* directly supplied channel mask */ if (set_mask > 0) { ret = px4_ioctl(fd, PWM_SERVO_SET_SELECT_UPDATE_RATE, set_mask); -- GitLab