110 lines
2.2 KiB
C++
110 lines
2.2 KiB
C++
/**
|
|
* @file DcMotor.cpp
|
|
* @brief DC motor controller
|
|
*
|
|
* @date Dec 22, 2018
|
|
* @author Matthew Kennedy
|
|
*/
|
|
|
|
#include "pch.h"
|
|
|
|
#include "dc_motor.h"
|
|
|
|
TwoPinDcMotor::TwoPinDcMotor(OutputPin& disablePin)
|
|
: m_disable(&disablePin)
|
|
{
|
|
disable("init");
|
|
}
|
|
|
|
void TwoPinDcMotor::configure(IPwm& enable, IPwm& dir1, IPwm& dir2, bool isInverted) {
|
|
m_enable = &enable;
|
|
m_dir1 = &dir1;
|
|
m_dir2 = &dir2;
|
|
m_isInverted = isInverted;
|
|
}
|
|
|
|
void TwoPinDcMotor::enable() {
|
|
if (m_disable) {
|
|
m_disable->setValue(false);
|
|
}
|
|
|
|
m_msg = nullptr;
|
|
}
|
|
|
|
void TwoPinDcMotor::disable(const char *msg) {
|
|
m_msg = msg;
|
|
if (m_disable) {
|
|
m_disable->setValue(true);
|
|
}
|
|
|
|
// Also set the duty to zero
|
|
set(0);
|
|
}
|
|
|
|
bool TwoPinDcMotor::isOpenDirection() const {
|
|
return m_value >= 0;
|
|
}
|
|
|
|
float TwoPinDcMotor::get() const {
|
|
return m_value;
|
|
}
|
|
|
|
/**
|
|
* @param duty value between -1.0 and 1.0
|
|
*/
|
|
bool TwoPinDcMotor::set(float duty)
|
|
{
|
|
m_value = duty;
|
|
|
|
// For low voltage, voltageRatio will be >1 to boost duty so that motor current stays the same
|
|
// At high voltage, the inverse will be true to keep behavior always the same.
|
|
float voltageRatio = 14 / Sensor::get(SensorType::BatteryVoltage).value_or(14);
|
|
duty *= voltageRatio;
|
|
|
|
// If not init, don't try to set
|
|
if (!m_dir1 || !m_dir2 || !m_enable) {
|
|
if (m_disable) {
|
|
m_disable->setValue(true);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool isPositive = duty > 0;
|
|
|
|
if (!isPositive) {
|
|
duty = -duty;
|
|
}
|
|
|
|
// below here 'duty' is a not negative
|
|
|
|
// Clamp to 100%
|
|
if (duty > 1.0f) {
|
|
duty = 1.0f;
|
|
}
|
|
// Disable for very small duty
|
|
else if (duty < 0.01f)
|
|
{
|
|
duty = 0.0f;
|
|
}
|
|
|
|
// If we're in two pin mode, force 100%, else use this pin to PWM
|
|
float enableDuty = m_type == ControlType::PwmEnablePin ? duty : 1;
|
|
|
|
// Direction pins get 100% duty unless we're in PwmDirectionPins mode
|
|
float dirDuty = m_type == ControlType::PwmDirectionPins ? duty : 1;
|
|
|
|
m_enable->setSimplePwmDutyCycle(enableDuty);
|
|
float recipDuty = 0;
|
|
if (m_isInverted) {
|
|
dirDuty = 1.0f - dirDuty;
|
|
recipDuty = 1.0f;
|
|
}
|
|
|
|
m_dir1->setSimplePwmDutyCycle(isPositive ? dirDuty : recipDuty);
|
|
m_dir2->setSimplePwmDutyCycle(isPositive ? recipDuty : dirDuty);
|
|
|
|
// This motor has no fault detection, so always return false (indicate success).
|
|
return false;
|
|
}
|