111 lines
2.8 KiB
C++
111 lines
2.8 KiB
C++
|
|
#include "pch.h"
|
|
|
|
#if !EFI_UNIT_TEST
|
|
|
|
#include "stepper.h"
|
|
#include "dc_motor.h"
|
|
|
|
static const int8_t phaseA[] =
|
|
{
|
|
1,
|
|
1,
|
|
-1,
|
|
-1
|
|
};
|
|
|
|
static const int8_t phaseB[] =
|
|
{
|
|
-1,
|
|
1,
|
|
1,
|
|
-1
|
|
};
|
|
|
|
static const int8_t microPhases[] = {
|
|
0, 20, 38, 56, 71, 83, 92, 98,
|
|
100, 98, 92, 83, 71, 56, 38, 20,
|
|
0, -20, -38, -56, -71, -83, -92, -98,
|
|
-100, -98, -92, -83, -71, -56, -38, -20
|
|
};
|
|
|
|
static const int maxNumSteps = 8;
|
|
static constexpr int tableSizeMask = efi::size(microPhases) - 1;
|
|
static constexpr float phaseDutyCycleDivisor = 1.0f / (100.0f * 100.0f); // both the phase degrees and duty cycle settings are in %
|
|
|
|
|
|
void DualHBridgeStepper::initialize(DcMotor* motorPhaseA, DcMotor* motorPhaseB, float reactionTime)
|
|
{
|
|
setReactionTime(reactionTime);
|
|
|
|
m_motorPhaseA = motorPhaseA;
|
|
m_motorPhaseB = motorPhaseB;
|
|
|
|
efiAssertVoid(CUSTOM_ERR_ASSERT, CONFIG(stepperNumMicroSteps) <= maxNumSteps, "stepperNumMicroSteps");
|
|
}
|
|
|
|
bool DualHBridgeStepper::step(bool positive) {
|
|
// Check that we've been initialized
|
|
if (!m_motorPhaseA || !m_motorPhaseB) {
|
|
return false;
|
|
}
|
|
|
|
if (CONFIG(stepperNumMicroSteps) > 1) {
|
|
float dutyMult = CONFIG(stepperMaxDutyCycle) * phaseDutyCycleDivisor;
|
|
int numStepIncr = maxNumSteps / CONFIG(stepperNumMicroSteps);
|
|
if (!positive)
|
|
numStepIncr = -numStepIncr;
|
|
for (int i = CONFIG(stepperNumMicroSteps); i > 0; i--) {
|
|
m_phase = (m_phase + numStepIncr) & tableSizeMask;
|
|
update(dutyMult);
|
|
// sleep 1/Nth of the pause time
|
|
pause(CONFIG(stepperNumMicroSteps));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// For the full-stepping mode, we use a traditional "two phase on" drive model
|
|
// because "wave drive" (one phase on) method has less torque.
|
|
// For explanation, pls see: https://github.com/rusefi/rusefi/pull/3213#discussion_r700746453
|
|
|
|
// step phase, wrapping
|
|
if (positive) {
|
|
m_phase = (m_phase + 1) & 0x03;
|
|
} else {
|
|
m_phase = (m_phase - 1) & 0x03;
|
|
}
|
|
|
|
update(1.0f);
|
|
pause();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DualHBridgeStepper::update(float dutyMult) {
|
|
if (!m_motorPhaseA || !m_motorPhaseB) {
|
|
return false;
|
|
}
|
|
|
|
if (CONFIG(stepperNumMicroSteps) > 1) {
|
|
// phase B is 90 degrees shifted
|
|
uint8_t m_phaseB = (m_phase + CONFIG(stepperNumMicroSteps)) & tableSizeMask;
|
|
|
|
// Set phases according to the table
|
|
m_motorPhaseA->set(dutyMult * microPhases[m_phase]);
|
|
m_motorPhaseB->set(dutyMult * microPhases[m_phaseB]);
|
|
return true;
|
|
}
|
|
// Set phases according to the table
|
|
m_motorPhaseA->set(phaseA[m_phase]);
|
|
m_motorPhaseB->set(phaseB[m_phase]);
|
|
return true;
|
|
}
|
|
|
|
void DualHBridgeStepper::sleep() {
|
|
float sleepingCoef = minI(CONFIG(stepperMinDutyCycle), CONFIG(stepperMaxDutyCycle)) * phaseDutyCycleDivisor;
|
|
update(sleepingCoef);
|
|
pause();
|
|
}
|
|
|
|
#endif
|