180 lines
3.8 KiB
C++
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());
|
|
}
|
|
|