127 lines
3.0 KiB
C++
127 lines
3.0 KiB
C++
#include "servo_state.hpp"
|
|
|
|
namespace servo {
|
|
ServoState::ServoState(CalibrationType default_type)
|
|
: table(default_type) {
|
|
}
|
|
|
|
float ServoState::enable() {
|
|
if(last_enabled_pulse < MIN_VALID_PULSE) {
|
|
servo_value = get_mid_value();
|
|
last_enabled_pulse = table.value_to_pulse(servo_value);
|
|
}
|
|
enabled = true;
|
|
return last_enabled_pulse;
|
|
}
|
|
|
|
float ServoState::disable() {
|
|
enabled = false;
|
|
return 0.0f; // A zero pulse
|
|
}
|
|
|
|
bool ServoState::is_enabled() const {
|
|
return enabled;
|
|
}
|
|
|
|
float ServoState::get_value() const {
|
|
return servo_value;
|
|
}
|
|
|
|
float ServoState::set_value(float value) {
|
|
servo_value = value;
|
|
float pulse = table.value_to_pulse(value);
|
|
if(pulse >= MIN_VALID_PULSE) {
|
|
last_enabled_pulse = pulse;
|
|
enabled = true;
|
|
}
|
|
else {
|
|
pulse = disable();
|
|
}
|
|
return pulse;
|
|
}
|
|
|
|
float ServoState::get_pulse() const {
|
|
return last_enabled_pulse;
|
|
}
|
|
|
|
float ServoState::set_pulse(float pulse) {
|
|
if(pulse >= MIN_VALID_PULSE) {
|
|
servo_value = table.value_from_pulse(pulse);
|
|
last_enabled_pulse = pulse;
|
|
enabled = true;
|
|
}
|
|
else {
|
|
pulse = disable();
|
|
}
|
|
return pulse;
|
|
}
|
|
|
|
float ServoState::get_min_value() const {
|
|
float value = 0.0f;
|
|
Calibration::Point *point = table.first_point();
|
|
if(point != nullptr) {
|
|
value = point->value;
|
|
}
|
|
return value;
|
|
}
|
|
|
|
float ServoState::get_mid_value() const {
|
|
float value = 0.0f;
|
|
Calibration::Point *first = table.first_point();
|
|
Calibration::Point *last = table.last_point();
|
|
if((first != nullptr) && (last != nullptr)) {
|
|
value = (first->value + last->value) / 2.0f;
|
|
}
|
|
return value;
|
|
}
|
|
|
|
float ServoState::get_max_value() const {
|
|
float value = 0.0f;
|
|
Calibration::Point *point = table.last_point();
|
|
if(point != nullptr) {
|
|
value = point->value;
|
|
}
|
|
return value;
|
|
}
|
|
|
|
float ServoState::to_min() {
|
|
return set_value(get_min_value());
|
|
}
|
|
|
|
float ServoState::to_mid() {
|
|
return set_value(get_mid_value());
|
|
}
|
|
|
|
float ServoState::to_max() {
|
|
return set_value(get_max_value());
|
|
}
|
|
|
|
float ServoState::to_percent(float in, float in_min, float in_max) {
|
|
float value = Calibration::map_float(in, in_min, in_max, get_min_value(), get_max_value());
|
|
return set_value(value);
|
|
}
|
|
|
|
float ServoState::to_percent(float in, float in_min, float in_max, float value_min, float value_max) {
|
|
float value = Calibration::map_float(in, in_min, in_max, value_min, value_max);
|
|
return set_value(value);
|
|
}
|
|
|
|
Calibration& ServoState::calibration() {
|
|
return table;
|
|
}
|
|
|
|
const Calibration& ServoState::calibration() const {
|
|
return table;
|
|
}
|
|
|
|
uint32_t ServoState::pulse_to_level(float pulse, uint32_t resolution) {
|
|
uint32_t level = 0;
|
|
if(pulse >= MIN_VALID_PULSE) {
|
|
// Constrain the level to hardcoded limits to protect the servo
|
|
pulse = MIN(MAX(pulse, LOWER_HARD_LIMIT), UPPER_HARD_LIMIT);
|
|
level = (uint32_t)((pulse * (float)resolution) / SERVO_PERIOD);
|
|
}
|
|
return level;
|
|
}
|
|
|
|
}; |