2022-02-16 22:06:07 +00:00
|
|
|
#include "servo_cluster.hpp"
|
2022-02-20 15:12:02 +00:00
|
|
|
#include "pwm.hpp"
|
|
|
|
#include <cstdio>
|
2022-02-16 22:06:07 +00:00
|
|
|
|
|
|
|
namespace servo {
|
2022-02-21 00:04:36 +00:00
|
|
|
ServoCluster::ServoCluster(PIO pio, uint sm, uint pin_mask)
|
|
|
|
: pwms(pio, sm, pin_mask) {
|
|
|
|
}
|
2022-02-20 15:12:02 +00:00
|
|
|
|
2022-02-21 00:04:36 +00:00
|
|
|
ServoCluster::ServoCluster(PIO pio, uint sm, uint pin_base, uint pin_count)
|
|
|
|
: pwms(pio, sm, pin_base, pin_count) {
|
|
|
|
}
|
2022-02-20 15:12:02 +00:00
|
|
|
|
2022-02-21 00:04:36 +00:00
|
|
|
ServoCluster::ServoCluster(PIO pio, uint sm, std::initializer_list<uint8_t> pins)
|
|
|
|
: pwms(pio, sm, pins) {
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ServoCluster::~ServoCluster() {
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ServoCluster::init() {
|
2022-02-21 00:04:36 +00:00
|
|
|
bool success = false;
|
|
|
|
|
|
|
|
if(pwms.init()) {
|
|
|
|
// Calculate a suitable pwm wrap period for this frequency
|
|
|
|
uint32_t period; uint16_t div16;
|
|
|
|
if(pimoroni::PWMCluster::calculate_pwm_factors(pwm_frequency, period, div16)) {
|
|
|
|
pwm_period = period;
|
|
|
|
|
|
|
|
// Update the pwm before setting the new wrap
|
|
|
|
for(uint servo = 0; servo < NUM_BANK0_GPIOS; servo++) {
|
|
|
|
pwms.set_chan_level(servo, 0, false);
|
|
|
|
}
|
2022-02-16 22:06:07 +00:00
|
|
|
|
2022-02-21 00:04:36 +00:00
|
|
|
// Set the new wrap (should be 1 less than the period to get full 0 to 100%)
|
|
|
|
pwms.set_wrap(pwm_period); // NOTE Minus 1 not needed here. Maybe should change Wrap behaviour so it is needed, for consistency with hardware pwm?
|
2022-02-16 22:06:07 +00:00
|
|
|
|
2022-02-21 00:04:36 +00:00
|
|
|
// Apply the new divider
|
|
|
|
// This is done after loading new PWM values to avoid a lockup condition
|
|
|
|
uint8_t div = div16 >> 4;
|
|
|
|
uint8_t mod = div16 % 16;
|
|
|
|
pwms.set_clkdiv_int_frac(div, mod);
|
2022-02-16 22:06:07 +00:00
|
|
|
|
2022-02-21 00:04:36 +00:00
|
|
|
success = true;
|
|
|
|
}
|
|
|
|
}
|
2022-02-16 22:06:07 +00:00
|
|
|
|
2022-02-21 00:04:36 +00:00
|
|
|
return success;
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
uint ServoCluster::get_pin_mask() const {
|
2022-02-21 00:04:36 +00:00
|
|
|
return pwms.get_pin_mask();
|
2022-02-17 22:38:59 +00:00
|
|
|
}
|
|
|
|
|
2022-02-16 22:06:07 +00:00
|
|
|
void ServoCluster::enable(uint servo, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].enable();
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ServoCluster::disable(uint servo, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].disable();
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
bool ServoCluster::is_enabled(uint servo) const {
|
2022-02-17 17:59:09 +00:00
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return servos[servo].is_enabled();
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
float ServoCluster::get_value(uint servo) const {
|
2022-02-16 22:06:07 +00:00
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return servos[servo].get_value();
|
|
|
|
else
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ServoCluster::set_value(uint servo, float value, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].set_value(value);
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
float ServoCluster::get_pulse(uint servo) const {
|
2022-02-16 22:06:07 +00:00
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return servos[servo].get_pulse();
|
|
|
|
else
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ServoCluster::set_pulse(uint servo, float pulse, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].set_pulse(pulse);
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
float ServoCluster::get_frequency() const {
|
|
|
|
return pwm_frequency;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ServoCluster::set_frequency(float freq) {
|
|
|
|
bool success = false;
|
|
|
|
|
|
|
|
if((freq >= ServoState::MIN_FREQUENCY) && (freq <= ServoState::MAX_FREQUENCY)) {
|
|
|
|
// Calculate a suitable pwm wrap period for this frequency
|
|
|
|
uint32_t period; uint16_t div16;
|
|
|
|
if(pimoroni::PWMCluster::calculate_pwm_factors(freq, period, div16)) {
|
|
|
|
|
|
|
|
pwm_period = period;
|
|
|
|
pwm_frequency = freq;
|
|
|
|
|
|
|
|
// Update the pwm before setting the new wrap
|
|
|
|
for(uint servo = 0; servo < NUM_BANK0_GPIOS; servo++) {
|
|
|
|
float current_pulse = servos[servo].get_pulse();
|
|
|
|
apply_pulse(servo, current_pulse, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the new wrap (should be 1 less than the period to get full 0 to 100%)
|
|
|
|
pwms.set_wrap(pwm_period, true);
|
|
|
|
|
|
|
|
// Apply the new divider
|
|
|
|
// This is done after loading new PWM values to avoid a lockup condition
|
|
|
|
uint8_t div = div16 >> 4;
|
|
|
|
uint8_t mod = div16 % 16;
|
|
|
|
pwms.set_clkdiv_int_frac(div, mod);
|
|
|
|
|
|
|
|
success = true;
|
|
|
|
}
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
2022-02-20 15:12:02 +00:00
|
|
|
return success;
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
|
2022-02-17 22:38:59 +00:00
|
|
|
float ServoCluster::get_min_value(uint servo) const {
|
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return servos[servo].get_min_value();
|
|
|
|
else
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
float ServoCluster::get_mid_value(uint servo) const {
|
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return servos[servo].get_mid_value();
|
|
|
|
else
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
float ServoCluster::get_max_value(uint servo) const {
|
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return servos[servo].get_max_value();
|
|
|
|
else
|
|
|
|
return 0.0f;
|
|
|
|
}
|
|
|
|
|
2022-02-16 22:06:07 +00:00
|
|
|
void ServoCluster::to_min(uint servo, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].to_min();
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ServoCluster::to_mid(uint servo, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].to_mid();
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ServoCluster::to_max(uint servo, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].to_max();
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ServoCluster::to_percent(uint servo, float in, float in_min, float in_max, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].to_percent(in, in_min, in_max);
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ServoCluster::to_percent(uint servo, float in, float in_min, float in_max, float value_min, float value_max, bool load) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS) {
|
|
|
|
float new_pulse = servos[servo].to_percent(in, in_min, in_max, value_min, value_max);
|
2022-02-20 15:12:02 +00:00
|
|
|
apply_pulse(servo, new_pulse, load);
|
2022-02-16 22:06:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Calibration* ServoCluster::calibration(uint servo) {
|
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return &servos[servo].calibration();
|
|
|
|
else
|
|
|
|
return nullptr;
|
|
|
|
}
|
2022-02-17 22:38:59 +00:00
|
|
|
|
|
|
|
const Calibration* ServoCluster::calibration(uint servo) const {
|
|
|
|
if(servo < NUM_BANK0_GPIOS)
|
|
|
|
return &servos[servo].calibration();
|
|
|
|
else
|
|
|
|
return nullptr;
|
|
|
|
}
|
2022-02-20 15:12:02 +00:00
|
|
|
|
|
|
|
void ServoCluster::apply_pulse(uint servo, float pulse, bool load) {
|
|
|
|
pwms.set_chan_level(servo, ServoState::pulse_to_level(pulse, pwm_period, pwm_frequency), load);
|
|
|
|
}
|
2022-02-16 22:06:07 +00:00
|
|
|
};
|