Fix BLTouch PWM reliability in HAL/STM32 (#18702)
This commit is contained in:
parent
1c94033f7c
commit
12bc63913c
|
@ -79,7 +79,7 @@ void HAL_init() {
|
||||||
while (!LL_PWR_IsActiveFlag_BRR()); // Wait until backup regulator is initialized
|
while (!LL_PWR_IsActiveFlag_BRR()); // Wait until backup regulator is initialized
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
SetSoftwareSerialTimerInterruptPriority();
|
SetTimerInterruptPriorities();
|
||||||
|
|
||||||
TERN_(EMERGENCY_PARSER, USB_Hook_init());
|
TERN_(EMERGENCY_PARSER, USB_Hook_init());
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,6 +33,18 @@ static libServo *servos[NUM_SERVOS] = {0};
|
||||||
constexpr millis_t servoDelay[] = SERVO_DELAY;
|
constexpr millis_t servoDelay[] = SERVO_DELAY;
|
||||||
static_assert(COUNT(servoDelay) == NUM_SERVOS, "SERVO_DELAY must be an array NUM_SERVOS long.");
|
static_assert(COUNT(servoDelay) == NUM_SERVOS, "SERVO_DELAY must be an array NUM_SERVOS long.");
|
||||||
|
|
||||||
|
// Initialize to the default timer priority. This will be overridden by a call from timers.cpp.
|
||||||
|
// This allows all timer interrupt priorities to be managed from a single location in the HAL.
|
||||||
|
static uint32_t servo_interrupt_priority = NVIC_EncodePriority(NVIC_GetPriorityGrouping(), TIM_IRQ_PRIO, TIM_IRQ_SUBPRIO);
|
||||||
|
|
||||||
|
// This must be called after the STM32 Servo class has intialized the timer.
|
||||||
|
// It may only be needed after the first call to attach(), but it is possible
|
||||||
|
// that is is necessary after every detach() call. To be safe this is currently
|
||||||
|
// called after every call to attach().
|
||||||
|
static void fixServoTimerInterruptPriority() {
|
||||||
|
NVIC_SetPriority(getTimerUpIrq(TIMER_SERVO), servo_interrupt_priority);
|
||||||
|
}
|
||||||
|
|
||||||
libServo::libServo()
|
libServo::libServo()
|
||||||
: delay(servoDelay[servoCount]),
|
: delay(servoDelay[servoCount]),
|
||||||
was_attached_before_pause(false),
|
was_attached_before_pause(false),
|
||||||
|
@ -44,13 +56,17 @@ libServo::libServo()
|
||||||
int8_t libServo::attach(const int pin) {
|
int8_t libServo::attach(const int pin) {
|
||||||
if (servoCount >= MAX_SERVOS) return -1;
|
if (servoCount >= MAX_SERVOS) return -1;
|
||||||
if (pin > 0) servo_pin = pin;
|
if (pin > 0) servo_pin = pin;
|
||||||
return stm32_servo.attach(servo_pin);
|
auto result = stm32_servo.attach(servo_pin);
|
||||||
|
fixServoTimerInterruptPriority();
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
int8_t libServo::attach(const int pin, const int min, const int max) {
|
int8_t libServo::attach(const int pin, const int min, const int max) {
|
||||||
if (servoCount >= MAX_SERVOS) return -1;
|
if (servoCount >= MAX_SERVOS) return -1;
|
||||||
if (pin > 0) servo_pin = pin;
|
if (pin > 0) servo_pin = pin;
|
||||||
return stm32_servo.attach(servo_pin, min, max);
|
auto result = stm32_servo.attach(servo_pin, min, max);
|
||||||
|
fixServoTimerInterruptPriority();
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
void libServo::move(const int value) {
|
void libServo::move(const int value) {
|
||||||
|
@ -86,5 +102,9 @@ void libServo::resume_all_servos() {
|
||||||
if (servo) servo->resume();
|
if (servo) servo->resume();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void libServo::setInterruptPriority(uint32_t preemptPriority, uint32_t subPriority) {
|
||||||
|
servo_interrupt_priority = NVIC_EncodePriority(NVIC_GetPriorityGrouping(), preemptPriority, subPriority);
|
||||||
|
}
|
||||||
|
|
||||||
#endif // HAS_SERVOS
|
#endif // HAS_SERVOS
|
||||||
#endif // ARDUINO_ARCH_STM32 && !STM32GENERIC
|
#endif // ARDUINO_ARCH_STM32 && !STM32GENERIC
|
||||||
|
|
|
@ -41,6 +41,7 @@ class libServo {
|
||||||
|
|
||||||
static void pause_all_servos();
|
static void pause_all_servos();
|
||||||
static void resume_all_servos();
|
static void resume_all_servos();
|
||||||
|
static void setInterruptPriority(uint32_t preemptPriority, uint32_t subPriority);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Servo stm32_servo;
|
Servo stm32_servo;
|
||||||
|
|
|
@ -29,26 +29,41 @@
|
||||||
|
|
||||||
#define NUM_HARDWARE_TIMERS 2
|
#define NUM_HARDWARE_TIMERS 2
|
||||||
|
|
||||||
|
// Default timer priorities. Override by specifying alternate priorities in the board pins file.
|
||||||
|
// The TONE timer is not present here, as it currently cannot be set programmatically. It is set
|
||||||
|
// by defining TIM_IRQ_PRIO in the variant.h or platformio.ini file, which adjusts the default
|
||||||
|
// priority for STM32 HardwareTimer objects.
|
||||||
|
#define SWSERIAL_TIMER_IRQ_PRIO_DEFAULT 1 // Requires tight bit timing to communicate reliably with TMC drivers
|
||||||
|
#define SERVO_TIMER_IRQ_PRIO_DEFAULT 1 // Requires tight PWM timing to control a BLTouch reliably
|
||||||
|
#define STEP_TIMER_IRQ_PRIO_DEFAULT 2
|
||||||
|
#define TEMP_TIMER_IRQ_PRIO_DEFAULT 14 // Low priority avoids interference with other hardware and timers
|
||||||
|
|
||||||
#ifndef STEP_TIMER_IRQ_PRIO
|
#ifndef STEP_TIMER_IRQ_PRIO
|
||||||
#define STEP_TIMER_IRQ_PRIO 2
|
#define STEP_TIMER_IRQ_PRIO STEP_TIMER_IRQ_PRIO_DEFAULT
|
||||||
#endif
|
#endif
|
||||||
#ifndef TEMP_TIMER_IRQ_PRIO
|
#ifndef TEMP_TIMER_IRQ_PRIO
|
||||||
#define TEMP_TIMER_IRQ_PRIO 14 // 14 = after hardware ISRs
|
#define TEMP_TIMER_IRQ_PRIO TEMP_TIMER_IRQ_PRIO_DEFAULT
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Ensure the default timer priority is somewhere between the STEP and TEMP priorities.
|
|
||||||
// The STM32 framework defaults to interrupt 14 for all timers. This should be increased so that
|
|
||||||
// timing-sensitive operations such as speaker output are note impacted by the long-running
|
|
||||||
// temperature ISR. This must be defined in the platformio.ini file or the board's variant.h,
|
|
||||||
// so that it will be consumed by framework code.
|
|
||||||
#if !(TIM_IRQ_PRIO > STEP_TIMER_IRQ_PRIO && TIM_IRQ_PRIO < TEMP_TIMER_IRQ_PRIO)
|
|
||||||
#error "Default timer interrupt priority is unspecified or set to a value which may degrade performance."
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if HAS_TMC_SW_SERIAL
|
#if HAS_TMC_SW_SERIAL
|
||||||
#include <SoftwareSerial.h>
|
#include <SoftwareSerial.h>
|
||||||
#ifndef SWSERIAL_TIMER_IRQ_PRIO
|
#ifndef SWSERIAL_TIMER_IRQ_PRIO
|
||||||
#define SWSERIAL_TIMER_IRQ_PRIO 1
|
#define SWSERIAL_TIMER_IRQ_PRIO SWSERIAL_TIMER_IRQ_PRIO_DEFAULT
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#if HAS_SERVOS
|
||||||
|
#include "Servo.h"
|
||||||
|
#ifndef SERVO_TIMER_IRQ_PRIO
|
||||||
|
#define SERVO_TIMER_IRQ_PRIO SERVO_TIMER_IRQ_PRIO_DEFAULT
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
#if ENABLED(SPEAKER)
|
||||||
|
// Ensure the default timer priority is somewhere between the STEP and TEMP priorities.
|
||||||
|
// The STM32 framework defaults to interrupt 14 for all timers. This should be increased so that
|
||||||
|
// timing-sensitive operations such as speaker output are not impacted by the long-running
|
||||||
|
// temperature ISR. This must be defined in the platformio.ini file or the board's variant.h,
|
||||||
|
// so that it will be consumed by framework code.
|
||||||
|
#if !(TIM_IRQ_PRIO > STEP_TIMER_IRQ_PRIO && TIM_IRQ_PRIO < TEMP_TIMER_IRQ_PRIO)
|
||||||
|
#error "Default timer interrupt priority is unspecified or set to a value which may degrade performance."
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -189,8 +204,9 @@ TIM_TypeDef * HAL_timer_device(const uint8_t timer_num) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetSoftwareSerialTimerInterruptPriority() {
|
void SetTimerInterruptPriorities() {
|
||||||
TERN_(HAS_TMC_SW_SERIAL, SoftwareSerial::setInterruptPriority(SWSERIAL_TIMER_IRQ_PRIO, 0));
|
TERN_(HAS_TMC_SW_SERIAL, SoftwareSerial::setInterruptPriority(SWSERIAL_TIMER_IRQ_PRIO, 0));
|
||||||
|
TERN_(HAS_SERVOS, libServo::setInterruptPriority(SERVO_TIMER_IRQ_PRIO, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // ARDUINO_ARCH_STM32 && !STM32GENERIC
|
#endif // ARDUINO_ARCH_STM32 && !STM32GENERIC
|
||||||
|
|
|
@ -86,8 +86,9 @@ void HAL_timer_enable_interrupt(const uint8_t timer_num);
|
||||||
void HAL_timer_disable_interrupt(const uint8_t timer_num);
|
void HAL_timer_disable_interrupt(const uint8_t timer_num);
|
||||||
bool HAL_timer_interrupt_enabled(const uint8_t timer_num);
|
bool HAL_timer_interrupt_enabled(const uint8_t timer_num);
|
||||||
|
|
||||||
|
// Configure timer priorities for peripherals such as Software Serial or Servos.
|
||||||
// Exposed here to allow all timer priority information to reside in timers.cpp
|
// Exposed here to allow all timer priority information to reside in timers.cpp
|
||||||
void SetSoftwareSerialTimerInterruptPriority();
|
void SetTimerInterruptPriorities();
|
||||||
|
|
||||||
//TIM_TypeDef* HAL_timer_device(const uint8_t timer_num); no need to be public for now. not public = not used externally
|
//TIM_TypeDef* HAL_timer_device(const uint8_t timer_num); no need to be public for now. not public = not used externally
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue