2022-02-16 00:40:42 +00:00
|
|
|
#include "servo.hpp"
|
2022-02-17 22:38:59 +00:00
|
|
|
#include "hardware/clocks.h"
|
2022-02-16 00:40:42 +00:00
|
|
|
|
|
|
|
namespace servo {
|
2022-02-17 22:38:59 +00:00
|
|
|
Servo::Servo(uint pin, CalibrationType type)
|
2022-02-16 22:06:07 +00:00
|
|
|
: pin(pin), state(type) {
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
2022-02-16 00:40:42 +00:00
|
|
|
|
|
|
|
Servo::~Servo() {
|
|
|
|
gpio_set_function(pin, GPIO_FUNC_NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Servo::init() {
|
2022-02-19 01:07:19 +00:00
|
|
|
bool success = false;
|
2022-02-16 00:40:42 +00:00
|
|
|
|
2022-02-19 01:07:19 +00:00
|
|
|
uint16_t period; uint16_t div16;
|
|
|
|
if(Servo::calculate_pwm_factors(pwm_frequency, period, div16)) {
|
|
|
|
pwm_period = period;
|
2022-02-16 00:40:42 +00:00
|
|
|
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_cfg = pwm_get_default_config();
|
2022-02-16 00:40:42 +00:00
|
|
|
|
2022-02-19 01:07:19 +00:00
|
|
|
// Set the new wrap (should be 1 less than the period to get full 0 to 100%)
|
|
|
|
pwm_config_set_wrap(&pwm_cfg, pwm_period - 1);
|
2022-02-16 10:28:47 +00:00
|
|
|
|
2022-02-19 01:07:19 +00:00
|
|
|
// Apply the divider
|
|
|
|
pwm_config_set_clkdiv(&pwm_cfg, (float)div16 / 16.0f); // There's no 'pwm_config_set_clkdiv_int_frac' for some reason...
|
|
|
|
|
|
|
|
pwm_init(pwm_gpio_to_slice_num(pin), &pwm_cfg, true);
|
|
|
|
gpio_set_function(pin, GPIO_FUNC_PWM);
|
|
|
|
|
|
|
|
pwm_set_gpio_level(pin, 0);
|
|
|
|
|
|
|
|
success = true;
|
|
|
|
}
|
|
|
|
return success;
|
2022-02-16 00:40:42 +00:00
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
uint Servo::get_pin() const {
|
2022-02-17 17:59:09 +00:00
|
|
|
return pin;
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Servo::enable() {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.enable();
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Servo::disable() {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.disable();
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
bool Servo::is_enabled() const {
|
2022-02-17 17:59:09 +00:00
|
|
|
return state.is_enabled();
|
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
float Servo::get_value() const {
|
2022-02-16 22:06:07 +00:00
|
|
|
return state.get_value();
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
2022-02-16 00:40:42 +00:00
|
|
|
void Servo::set_value(float value) {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.set_value(value);
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
float Servo::get_pulse() const {
|
2022-02-16 22:06:07 +00:00
|
|
|
return state.get_pulse();
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Servo::set_pulse(float pulse) {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.set_pulse(pulse);
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float Servo::get_frequency() const {
|
|
|
|
return pwm_frequency;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Servo::set_frequency(float freq) {
|
|
|
|
bool success = false;
|
|
|
|
|
|
|
|
// Calculate a suitable pwm wrap period for this frequency
|
|
|
|
uint16_t period; uint16_t div16;
|
|
|
|
if(Servo::calculate_pwm_factors(freq, period, div16)) {
|
|
|
|
|
|
|
|
// Record if the new period will be larger or smaller.
|
|
|
|
// This is used to apply new pwm values either before or after the wrap is applied,
|
|
|
|
// to avoid momentary blips in PWM output on SLOW_DECAY
|
|
|
|
bool pre_update_pwm = (period > pwm_period);
|
|
|
|
|
|
|
|
pwm_period = period;
|
|
|
|
pwm_frequency = freq;
|
|
|
|
|
|
|
|
uint pin_num = pwm_gpio_to_slice_num(pin);
|
|
|
|
|
|
|
|
// Apply the new divider
|
|
|
|
uint8_t div = div16 >> 4;
|
|
|
|
uint8_t mod = div16 % 16;
|
|
|
|
pwm_set_clkdiv_int_frac(pin_num, div, mod);
|
|
|
|
|
|
|
|
// If the the period is larger, update the pwm before setting the new wraps
|
|
|
|
if(pre_update_pwm) {
|
|
|
|
float current_pulse = get_pulse();
|
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(current_pulse, pwm_period, pwm_frequency));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the new wrap (should be 1 less than the period to get full 0 to 100%)
|
|
|
|
pwm_set_wrap(pin_num, pwm_period - 1);
|
|
|
|
|
|
|
|
// If the the period is smaller, update the pwm after setting the new wraps
|
|
|
|
if(!pre_update_pwm) {
|
|
|
|
float current_pulse = get_pulse();
|
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(current_pulse, pwm_period, pwm_frequency));
|
|
|
|
}
|
|
|
|
|
|
|
|
success = true;
|
|
|
|
}
|
|
|
|
return success;
|
2022-02-17 22:38:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
float Servo::get_min_value() const {
|
|
|
|
return state.get_min_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
float Servo::get_mid_value() const {
|
|
|
|
return state.get_mid_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
float Servo::get_max_value() const {
|
|
|
|
return state.get_max_value();
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Servo::to_min() {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.to_min();
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Servo::to_mid() {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.to_mid();
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Servo::to_max() {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.to_max();
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-17 17:59:09 +00:00
|
|
|
}
|
|
|
|
|
2022-02-16 10:28:47 +00:00
|
|
|
void Servo::to_percent(float in, float in_min, float in_max) {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.to_percent(in, in_min, in_max);
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Servo::to_percent(float in, float in_min, float in_max, float value_min, float value_max) {
|
2022-02-16 22:06:07 +00:00
|
|
|
float new_pulse = state.to_percent(in, in_min, in_max, value_min, value_max);
|
2022-02-19 01:07:19 +00:00
|
|
|
pwm_set_gpio_level(pin, (uint16_t)ServoState::pulse_to_level(new_pulse, pwm_period, pwm_frequency));
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
Calibration& Servo::calibration() {
|
|
|
|
return state.calibration();
|
2022-02-17 17:59:09 +00:00
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
const Calibration& Servo::calibration() const {
|
2022-02-16 22:06:07 +00:00
|
|
|
return state.calibration();
|
2022-02-16 10:28:47 +00:00
|
|
|
}
|
2022-02-19 01:07:19 +00:00
|
|
|
|
|
|
|
// Derived from the rp2 Micropython implementation: https://github.com/micropython/micropython/blob/master/ports/rp2/machine_pwm.c
|
|
|
|
bool Servo::calculate_pwm_factors(float freq, uint16_t& top_out, uint16_t& div16_out) {
|
|
|
|
bool success = false;
|
|
|
|
uint32_t source_hz = clock_get_hz(clk_sys);
|
|
|
|
|
|
|
|
// Check the provided frequency is valid
|
|
|
|
if((freq >= 1.0f) && (freq <= (float)(source_hz >> 1))) {
|
|
|
|
uint32_t div16_top = (uint32_t)((float)(source_hz << 4) / freq);
|
|
|
|
uint32_t top = 1;
|
|
|
|
|
|
|
|
while(true) {
|
|
|
|
// Try a few small prime factors to get close to the desired frequency.
|
|
|
|
if((div16_top >= (5 << 4)) && (div16_top % 5 == 0) && (top * 5 <= MAX_PWM_WRAP)) {
|
|
|
|
div16_top /= 5;
|
|
|
|
top *= 5;
|
|
|
|
}
|
|
|
|
else if((div16_top >= (3 << 4)) && (div16_top % 3 == 0) && (top * 3 <= MAX_PWM_WRAP)) {
|
|
|
|
div16_top /= 3;
|
|
|
|
top *= 3;
|
|
|
|
}
|
|
|
|
else if((div16_top >= (2 << 4)) && (top * 2 <= MAX_PWM_WRAP)) {
|
|
|
|
div16_top /= 2;
|
|
|
|
top *= 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(div16_top >= 16 && div16_top <= (UINT8_MAX << 4)) {
|
|
|
|
top_out = top;
|
|
|
|
div16_out = div16_top;
|
|
|
|
|
|
|
|
success = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2022-02-16 00:40:42 +00:00
|
|
|
};
|