custom-board-bundle-sample-.../firmware/controllers/actuators/dc_motors.cpp

180 lines
3.8 KiB
C++

/**
* @file dc_motors.cpp
*
* @date March 3, 2020
* @author Matthew Kennedy (c) 2020
*/
#include "pch.h"
#include "periodic_task.h"
#include "dc_motors.h"
#include "dc_motor.h"
// Simple wrapper to use an OutputPin as "PWM" that can only do 0 or 1
struct PwmWrapper : public IPwm {
OutputPin& m_pin;
PwmWrapper(OutputPin& pin) : m_pin(pin) { }
void setSimplePwmDutyCycle(float dutyCycle) override {
m_pin.setValue(dutyCycle > 0.5f);
}
};
class DcHardware {
private:
OutputPin m_pinEnable;
OutputPin m_pinDir1;
OutputPin m_pinDir2;
OutputPin m_disablePin;
PwmWrapper wrappedEnable{m_pinEnable};
PwmWrapper wrappedDir1{m_pinDir1};
PwmWrapper wrappedDir2{m_pinDir2};
SimplePwm m_pwm1;
SimplePwm m_pwm2;
bool isStarted = false;
public:
DcHardware() : dcMotor(m_disablePin) {}
TwoPinDcMotor dcMotor;
void setFrequency(int frequency) {
m_pwm1.setFrequency(frequency);
m_pwm2.setFrequency(frequency);
}
void stop() {
// todo: replace 'isStarted' with 'stop'
}
void start(bool useTwoWires,
brain_pin_e pinEnable,
brain_pin_e pinDir1,
brain_pin_e pinDir2,
brain_pin_e pinDisable,
bool isInverted,
ExecutorInterface* executor,
int frequency) {
if (isStarted) {
// actually implement stop()
return;
}
isStarted = true;
dcMotor.setType(useTwoWires ? TwoPinDcMotor::ControlType::PwmDirectionPins : TwoPinDcMotor::ControlType::PwmEnablePin);
// Configure the disable pin first - ensure things are in a safe state
m_disablePin.initPin("ETB Disable", pinDisable);
m_disablePin.setValue(0);
// Clamp to >100hz
int clampedFrequency = maxI(100, frequency);
if (clampedFrequency > ETB_HW_MAX_FREQUENCY) {
firmwareError(OBD_PCM_Processor_Fault, "Electronic throttle frequency too high, maximum %d hz", ETB_HW_MAX_FREQUENCY);
return;
}
if (useTwoWires) {
m_pinEnable.initPin("ETB Enable", pinEnable);
// no need to complicate event queue with ETB PWM in unit tests
#if ! EFI_UNIT_TEST
startSimplePwmHard(&m_pwm1, "ETB Dir 1",
executor,
pinDir1,
&m_pinDir1,
clampedFrequency,
0
);
startSimplePwmHard(&m_pwm2, "ETB Dir 2",
executor,
pinDir2,
&m_pinDir2,
clampedFrequency,
0
);
#endif // EFI_UNIT_TEST
dcMotor.configure(wrappedEnable, m_pwm1, m_pwm2, isInverted);
} else {
m_pinDir1.initPin("ETB Dir 1", pinDir1);
m_pinDir2.initPin("ETB Dir 2", pinDir2);
// no need to complicate event queue with ETB PWM in unit tests
#if ! EFI_UNIT_TEST
startSimplePwmHard(&m_pwm1, "ETB Enable",
executor,
pinEnable,
&m_pinEnable,
clampedFrequency,
0
);
#endif // EFI_UNIT_TEST
dcMotor.configure(m_pwm1, wrappedDir1, wrappedDir2, isInverted);
}
}
};
static DcHardware dcHardware[ETB_COUNT + DC_PER_STEPPER];
DcMotor* initDcMotor(const dc_io& io, size_t index, bool useTwoWires) {
auto& hw = dcHardware[index];
hw.start(
useTwoWires,
io.controlPin,
io.directionPin1,
io.directionPin2,
io.disablePin,
// todo You would not believe how you invert TLE9201 #4579
engineConfiguration->stepperDcInvertedPins,
&engine->executor,
engineConfiguration->etbFreq
);
return &hw.dcMotor;
}
DcMotor* initDcMotor(brain_pin_e coil_p, brain_pin_e coil_m, size_t index) {
auto& hw = dcHardware[index];
hw.start(
true, /* useTwoWires */
Gpio::Unassigned, /* pinEnable */
coil_p,
coil_m,
Gpio::Unassigned, /* pinDisable */
engineConfiguration->stepperDcInvertedPins,
&engine->executor,
engineConfiguration->etbFreq /* same in case of stepper? */
);
return &hw.dcMotor;
}
void setDcMotorFrequency(size_t index, int hz) {
dcHardware[index].setFrequency(hz);
}
void setDcMotorDuty(size_t index, float duty) {
dcHardware[index].dcMotor.set(duty);
}
void showDcMotorInfo(int i) {
DcHardware *dc = &dcHardware[i];
efiPrintf(" motor: dir=%d DC=%f", dc->dcMotor.isOpenDirection(), dc->dcMotor.get());
}