2015-05-29 00:33:00 -07:00
/*
Speeduino - Simple engine management for the Arduino Mega 2560 platform
Copyright ( C ) Josh Stewart
A full copy of the license may be found in the projects root directory
*/
2017-01-17 22:37:55 -08:00
# include "idle.h"
2018-12-29 09:26:30 -08:00
# include "maths.h"
# include "timers.h"
# include "src/PID_v1/PID_v1.h"
2015-05-29 00:33:00 -07:00
/*
2015-09-25 17:12:59 -07:00
These functions cover the PWM and stepper idle control
2015-05-29 00:33:00 -07:00
*/
2015-08-20 06:21:27 -07:00
/*
Idle Control
Currently limited to on / off control and open loop PWM and stepper drive
*/
2018-01-23 17:05:50 -08:00
integerPID idlePID ( & currentStatus . longRPM , & idle_pid_target_value , & idle_cl_target_rpm , configPage6 . idleKP , configPage6 . idleKI , configPage6 . idleKD , DIRECT ) ; //This is the PID object if that algorithm is used. Needs to be global as it maintains state outside of each function call
2016-04-06 20:28:13 -07:00
2015-05-29 00:33:00 -07:00
void initialiseIdle ( )
{
2017-02-28 19:42:24 -08:00
//By default, turn off the PWM interrupt (It gets turned on below if needed)
IDLE_TIMER_DISABLE ( ) ;
2017-02-19 22:57:46 -08:00
2015-08-20 06:21:27 -07:00
//Initialising comprises of setting the 2D tables with the relevant values from the config pages
2018-01-23 17:05:50 -08:00
switch ( configPage6 . iacAlgorithm )
2015-08-20 06:21:27 -07:00
{
2019-01-06 10:16:04 -08:00
case IAC_ALGORITHM_NONE :
//Case 0 is no idle control ('None')
2015-08-20 06:21:27 -07:00
break ;
2017-02-19 22:57:46 -08:00
2017-03-20 04:29:42 -07:00
case IAC_ALGORITHM_ONOFF :
2015-08-20 06:21:27 -07:00
//Case 1 is on/off idle control
2018-01-23 17:05:50 -08:00
if ( ( currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) < configPage6 . iacFastTemp )
2015-08-20 06:21:27 -07:00
{
digitalWrite ( pinIdle1 , HIGH ) ;
2017-03-29 06:01:52 -07:00
idleOn = true ;
2015-08-20 06:21:27 -07:00
}
break ;
2017-02-19 22:57:46 -08:00
2017-03-20 04:29:42 -07:00
case IAC_ALGORITHM_PWM_OL :
2015-08-20 06:21:27 -07:00
//Case 2 is PWM open loop
iacPWMTable . xSize = 10 ;
2015-08-25 20:27:50 -07:00
iacPWMTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacPWMTable . values = configPage6 . iacOLPWMVal ;
iacPWMTable . axisX = configPage6 . iacBins ;
2017-02-19 22:57:46 -08:00
2017-03-22 18:25:51 -07:00
2015-08-20 06:21:27 -07:00
iacCrankDutyTable . xSize = 4 ;
2016-05-08 21:00:52 -07:00
iacCrankDutyTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacCrankDutyTable . values = configPage6 . iacCrankDuty ;
iacCrankDutyTable . axisX = configPage6 . iacCrankBins ;
2017-02-19 22:57:46 -08:00
2015-09-25 17:12:59 -07:00
idle_pin_port = portOutputRegister ( digitalPinToPort ( pinIdle1 ) ) ;
idle_pin_mask = digitalPinToBitMask ( pinIdle1 ) ;
2016-01-12 22:06:55 -08:00
idle2_pin_port = portOutputRegister ( digitalPinToPort ( pinIdle2 ) ) ;
idle2_pin_mask = digitalPinToBitMask ( pinIdle2 ) ;
2019-01-21 13:56:25 -08:00
# if defined(CORE_AVR)
2018-01-23 17:05:50 -08:00
idle_pwm_max_count = 1000000L / ( 16 * configPage6 . idleFreq * 2 ) ; //Converts the frequency in Hz to the number of ticks (at 16uS) it takes to complete 1 cycle. Note that the frequency is divided by 2 coming from TS to allow for up to 512hz
2018-12-29 09:26:30 -08:00
# elif defined(CORE_TEENSY)
idle_pwm_max_count = 1000000L / ( 32 * configPage6 . idleFreq * 2 ) ; //Converts the frequency in Hz to the number of ticks (at 16uS) it takes to complete 1 cycle. Note that the frequency is divided by 2 coming from TS to allow for up to 512hz
2017-08-03 20:13:32 -07:00
# endif
2016-10-06 23:34:27 -07:00
enableIdle ( ) ;
2015-08-20 06:21:27 -07:00
break ;
2017-02-19 22:57:46 -08:00
2017-03-20 04:29:42 -07:00
case IAC_ALGORITHM_PWM_CL :
2015-08-20 06:21:27 -07:00
//Case 3 is PWM closed loop
iacClosedLoopTable . xSize = 10 ;
2016-05-08 21:00:52 -07:00
iacClosedLoopTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacClosedLoopTable . values = configPage6 . iacCLValues ;
iacClosedLoopTable . axisX = configPage6 . iacBins ;
2017-02-19 22:57:46 -08:00
2015-08-20 06:21:27 -07:00
iacCrankDutyTable . xSize = 4 ;
2016-05-08 21:00:52 -07:00
iacCrankDutyTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacCrankDutyTable . values = configPage6 . iacCrankDuty ;
iacCrankDutyTable . axisX = configPage6 . iacCrankBins ;
2016-04-06 20:28:13 -07:00
idle_pin_port = portOutputRegister ( digitalPinToPort ( pinIdle1 ) ) ;
idle_pin_mask = digitalPinToBitMask ( pinIdle1 ) ;
idle2_pin_port = portOutputRegister ( digitalPinToPort ( pinIdle2 ) ) ;
idle2_pin_mask = digitalPinToBitMask ( pinIdle2 ) ;
2019-01-21 13:56:25 -08:00
# if defined(CORE_AVR)
2018-01-23 17:05:50 -08:00
idle_pwm_max_count = 1000000L / ( 16 * configPage6 . idleFreq * 2 ) ; //Converts the frequency in Hz to the number of ticks (at 16uS) it takes to complete 1 cycle. Note that the frequency is divided by 2 coming from TS to allow for up to 512hz
2018-12-29 09:26:30 -08:00
# elif defined(CORE_TEENSY)
idle_pwm_max_count = 1000000L / ( 32 * configPage6 . idleFreq * 2 ) ; //Converts the frequency in Hz to the number of ticks (at 16uS) it takes to complete 1 cycle. Note that the frequency is divided by 2 coming from TS to allow for up to 512hz
2017-08-03 20:13:32 -07:00
# endif
2018-01-23 17:05:50 -08:00
idlePID . SetOutputLimits ( percentage ( configPage2 . iacCLminDuty , idle_pwm_max_count ) , percentage ( configPage2 . iacCLmaxDuty , idle_pwm_max_count ) ) ;
idlePID . SetTunings ( configPage6 . idleKP , configPage6 . idleKI , configPage6 . idleKD ) ;
2016-04-08 06:52:32 -07:00
idlePID . SetMode ( AUTOMATIC ) ; //Turn PID on
2017-03-13 23:25:59 -07:00
idleCounter = 0 ;
2015-08-20 06:21:27 -07:00
break ;
2017-02-19 22:57:46 -08:00
2017-03-20 04:29:42 -07:00
case IAC_ALGORITHM_STEP_OL :
2015-08-20 06:21:27 -07:00
//Case 2 is Stepper open loop
iacStepTable . xSize = 10 ;
2015-08-25 20:27:50 -07:00
iacStepTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacStepTable . values = configPage6 . iacOLStepVal ;
iacStepTable . axisX = configPage6 . iacBins ;
2017-02-19 22:57:46 -08:00
2015-08-20 06:21:27 -07:00
iacCrankStepsTable . xSize = 4 ;
2018-12-29 09:26:30 -08:00
iacCrankStepsTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacCrankStepsTable . values = configPage6 . iacCrankSteps ;
iacCrankStepsTable . axisX = configPage6 . iacCrankBins ;
iacStepTime = configPage6 . iacStepTime * 1000 ;
2019-02-26 14:53:43 -08:00
iacCoolTime = configPage9 . iacCoolTime * 1000 ;
2017-02-19 22:57:46 -08:00
2016-12-11 04:23:54 -08:00
completedHomeSteps = 0 ;
2017-03-20 04:29:42 -07:00
idleStepper . curIdleStep = 0 ;
2015-08-25 20:27:50 -07:00
idleStepper . stepperStatus = SOFF ;
2018-12-02 14:36:41 -08:00
if ( ! configPage9 . iacStepperInv )
{
idleStepper . lessAirDirection = STEPPER_BACKWARD ;
idleStepper . moreAirDirection = STEPPER_FORWARD ;
}
else
{
idleStepper . lessAirDirection = STEPPER_FORWARD ;
idleStepper . moreAirDirection = STEPPER_BACKWARD ;
}
2015-08-20 06:21:27 -07:00
break ;
2017-02-19 22:57:46 -08:00
2017-03-20 04:29:42 -07:00
case IAC_ALGORITHM_STEP_CL :
2015-08-20 06:21:27 -07:00
//Case 5 is Stepper closed loop
iacClosedLoopTable . xSize = 10 ;
2017-03-10 01:34:39 -08:00
iacClosedLoopTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacClosedLoopTable . values = configPage6 . iacCLValues ;
iacClosedLoopTable . axisX = configPage6 . iacBins ;
2017-02-19 22:57:46 -08:00
2015-08-20 06:21:27 -07:00
iacCrankStepsTable . xSize = 4 ;
2018-12-29 09:26:30 -08:00
iacCrankStepsTable . valueSize = SIZE_BYTE ;
2018-01-23 17:05:50 -08:00
iacCrankStepsTable . values = configPage6 . iacCrankSteps ;
iacCrankStepsTable . axisX = configPage6 . iacCrankBins ;
iacStepTime = configPage6 . iacStepTime * 1000 ;
2019-02-26 14:53:43 -08:00
iacCoolTime = configPage9 . iacCoolTime * 1000 ;
2017-02-19 22:57:46 -08:00
2017-03-10 01:34:39 -08:00
completedHomeSteps = 0 ;
2017-03-13 23:25:59 -07:00
idleCounter = 0 ;
2017-03-20 04:29:42 -07:00
idleStepper . curIdleStep = 0 ;
2015-08-25 20:27:50 -07:00
idleStepper . stepperStatus = SOFF ;
2017-03-10 01:34:39 -08:00
2018-12-02 14:36:41 -08:00
if ( ! configPage9 . iacStepperInv )
{
idleStepper . lessAirDirection = STEPPER_BACKWARD ;
idleStepper . moreAirDirection = STEPPER_FORWARD ;
}
else
{
idleStepper . lessAirDirection = STEPPER_FORWARD ;
idleStepper . moreAirDirection = STEPPER_BACKWARD ;
}
2018-01-23 17:05:50 -08:00
idlePID . SetOutputLimits ( 0 , ( configPage6 . iacStepHome * 3 ) ) ; //Maximum number of steps probably needs its own setting
idlePID . SetTunings ( configPage6 . idleKP , configPage6 . idleKI , configPage6 . idleKD ) ;
2017-03-10 01:34:39 -08:00
idlePID . SetMode ( AUTOMATIC ) ; //Turn PID on
2015-08-20 06:21:27 -07:00
break ;
2017-08-02 18:46:48 -07:00
default :
//Well this just shouldn't happen
break ;
2015-08-20 06:21:27 -07:00
}
2018-01-23 17:05:50 -08:00
idleInitComplete = configPage6 . iacAlgorithm ; //Sets which idle method was initialised
2017-03-20 04:29:42 -07:00
currentStatus . idleLoad = 0 ;
2015-05-29 00:33:00 -07:00
}
2015-08-20 06:21:27 -07:00
void idleControl ( )
2015-05-29 00:33:00 -07:00
{
2018-01-23 17:05:50 -08:00
if ( idleInitComplete ! = configPage6 . iacAlgorithm ) { initialiseIdle ( ) ; }
2018-04-07 21:40:12 -07:00
if ( currentStatus . RPM > 0 ) { enableIdle ( ) ; }
2017-02-19 22:57:46 -08:00
2018-05-04 00:55:58 -07:00
//Check whether the idleUp is active
if ( configPage2 . idleUpEnabled = = true )
{
if ( configPage2 . idleUpPolarity = = 0 ) { currentStatus . idleUpActive = ! digitalRead ( pinIdleUp ) ; } //Normal mode (ground switched)
else { currentStatus . idleUpActive = digitalRead ( pinIdleUp ) ; } //Inverted mode (5v activates idleUp)
}
else { currentStatus . idleUpActive = false ; }
2018-01-23 17:05:50 -08:00
switch ( configPage6 . iacAlgorithm )
2015-08-20 06:21:27 -07:00
{
2017-03-13 23:25:59 -07:00
case IAC_ALGORITHM_NONE : //Case 0 is no idle control ('None')
2015-08-20 06:21:27 -07:00
break ;
2017-02-19 22:57:46 -08:00
2017-03-13 23:25:59 -07:00
case IAC_ALGORITHM_ONOFF : //Case 1 is on/off idle control
2018-01-23 17:05:50 -08:00
if ( ( currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) < configPage6 . iacFastTemp ) //All temps are offset by 40 degrees
2015-08-20 06:21:27 -07:00
{
digitalWrite ( pinIdle1 , HIGH ) ;
idleOn = true ;
}
else if ( idleOn ) { digitalWrite ( pinIdle1 , LOW ) ; idleOn = false ; }
break ;
2017-02-19 22:57:46 -08:00
2017-03-13 23:25:59 -07:00
case IAC_ALGORITHM_PWM_OL : //Case 2 is PWM open loop
2015-08-20 06:21:27 -07:00
//Check for cranking pulsewidth
if ( BIT_CHECK ( currentStatus . engine , BIT_ENGINE_CRANK ) )
{
//Currently cranking. Use the cranking table
2016-01-12 22:06:55 -08:00
currentStatus . idleDuty = table2D_getValue ( & iacCrankDutyTable , currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) ; //All temps are offset by 40 degrees
2015-08-20 06:21:27 -07:00
}
2016-05-08 20:18:13 -07:00
else
2015-08-20 06:21:27 -07:00
{
//Standard running
2016-01-12 22:06:55 -08:00
currentStatus . idleDuty = table2D_getValue ( & iacPWMTable , currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) ; //All temps are offset by 40 degrees
2015-08-20 06:21:27 -07:00
}
2019-02-03 19:32:05 -08:00
if ( currentStatus . idleUpActive = = true ) { currentStatus . idleDuty + = configPage2 . idleUpAdder ; } //Add Idle Up amount if active
if ( currentStatus . idleDuty = = 0 )
{
disableIdle ( ) ;
break ;
}
BIT_SET ( currentStatus . spark , BIT_SPARK_IDLE ) ; //Turn the idle control flag on
idle_pwm_target_value = percentage ( currentStatus . idleDuty , idle_pwm_max_count ) ;
currentStatus . idleLoad = currentStatus . idleDuty > > 1 ; //Idle Load is divided by 2 in order to send to TS
idleOn = true ;
2018-12-29 09:26:30 -08:00
2015-08-20 06:21:27 -07:00
break ;
2017-02-19 22:57:46 -08:00
2017-03-13 23:25:59 -07:00
case IAC_ALGORITHM_PWM_CL : //Case 3 is PWM closed loop
2016-04-06 20:28:13 -07:00
//No cranking specific value for closed loop (yet?)
2016-05-08 21:00:52 -07:00
idle_cl_target_rpm = table2D_getValue ( & iacClosedLoopTable , currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) * 10 ; //All temps are offset by 40 degrees
2018-01-23 17:05:50 -08:00
if ( ( idleCounter & 31 ) = = 1 ) { idlePID . SetTunings ( configPage6 . idleKP , configPage6 . idleKI , configPage6 . idleKD ) ; } //This only needs to be run very infrequently, once every 32 calls to idleControl(). This is approx. once per second
2016-04-06 20:28:13 -07:00
idlePID . Compute ( ) ;
2017-03-10 01:34:39 -08:00
idle_pwm_target_value = idle_pid_target_value ;
2019-02-03 19:32:05 -08:00
if ( idle_pwm_target_value = = 0 )
{
disableIdle ( ) ;
break ;
}
BIT_SET ( currentStatus . spark , BIT_SPARK_IDLE ) ; //Turn the idle control flag on
2017-03-20 04:29:42 -07:00
currentStatus . idleLoad = ( ( unsigned long ) ( idle_pwm_target_value * 100UL ) / idle_pwm_max_count ) > > 1 ;
2018-05-04 00:55:58 -07:00
if ( currentStatus . idleUpActive = = true ) { currentStatus . idleDuty + = configPage2 . idleUpAdder ; } //Add Idle Up amount if active
2017-03-13 23:25:59 -07:00
idleCounter + + ;
2015-08-20 06:21:27 -07:00
break ;
2017-02-19 22:57:46 -08:00
2017-03-13 23:25:59 -07:00
case IAC_ALGORITHM_STEP_OL : //Case 4 is open loop stepper control
2015-08-20 21:14:47 -07:00
//First thing to check is whether there is currently a step going on and if so, whether it needs to be turned off
2017-11-26 14:24:00 -08:00
if ( ( checkForStepping ( ) = = false ) & & ( isStepperHomed ( ) = = true ) ) //Check that homing is complete and that there's not currently a step already taking place. MUST BE IN THIS ORDER!
2015-08-20 06:21:27 -07:00
{
2017-08-02 18:46:48 -07:00
//Check for cranking pulsewidth
if ( BIT_CHECK ( currentStatus . engine , BIT_ENGINE_CRANK ) )
2016-12-11 04:23:54 -08:00
{
2017-08-02 18:46:48 -07:00
//Currently cranking. Use the cranking table
idleStepper . targetIdleStep = table2D_getValue ( & iacCrankStepsTable , ( currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) ) * 3 ; //All temps are offset by 40 degrees. Step counts are divided by 3 in TS. Multiply back out here
2018-05-04 00:55:58 -07:00
if ( currentStatus . idleUpActive = = true ) { idleStepper . targetIdleStep + = configPage2 . idleUpAdder ; } //Add Idle Up amount if active
2017-08-02 18:46:48 -07:00
doStep ( ) ;
2016-12-11 04:23:54 -08:00
}
2017-08-02 18:46:48 -07:00
else if ( ( currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) < iacStepTable . axisX [ IDLE_TABLE_SIZE - 1 ] )
{
//Standard running
2019-02-26 14:52:37 -08:00
//We must also have more than zero RPM for the running state
if ( ( ( mainLoopCount & 255 ) = = 1 ) & & ( currentStatus . RPM > 0 ) )
2017-08-02 18:46:48 -07:00
{
//Only do a lookup of the required value around 4 times per second. Any more than this can create too much jitter and require a hyster value that is too high
idleStepper . targetIdleStep = table2D_getValue ( & iacStepTable , ( currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) ) * 3 ; //All temps are offset by 40 degrees. Step counts are divided by 3 in TS. Multiply back out here
2018-05-04 00:55:58 -07:00
if ( currentStatus . idleUpActive = = true ) { idleStepper . targetIdleStep + = configPage2 . idleUpAdder ; } //Add Idle Up amount if active
2018-01-23 17:05:50 -08:00
iacStepTime = configPage6 . iacStepTime * 1000 ;
2019-02-26 14:53:43 -08:00
iacCoolTime = configPage9 . iacCoolTime * 1000 ;
2017-08-02 18:46:48 -07:00
}
doStep ( ) ;
}
currentStatus . idleLoad = idleStepper . curIdleStep > > 1 ; //Current step count (Divided by 2 for byte)
2015-08-20 06:21:27 -07:00
}
2017-03-10 01:34:39 -08:00
break ;
2019-01-06 10:16:04 -08:00
case IAC_ALGORITHM_STEP_CL : //Case 5 is closed loop stepper control
2017-03-10 01:34:39 -08:00
//First thing to check is whether there is currently a step going on and if so, whether it needs to be turned off
2017-11-26 14:24:00 -08:00
if ( ( checkForStepping ( ) = = false ) & & ( isStepperHomed ( ) = = true ) ) //Check that homing is complete and that there's not currently a step already taking place. MUST BE IN THIS ORDER!
2017-04-02 04:17:29 -07:00
{
2017-08-02 18:46:48 -07:00
if ( ( idleCounter & 31 ) = = 1 )
{
//This only needs to be run very infrequently, once every 32 calls to idleControl(). This is approx. once per second
2018-01-23 17:05:50 -08:00
idlePID . SetTunings ( configPage6 . idleKP , configPage6 . idleKI , configPage6 . idleKD ) ;
iacStepTime = configPage6 . iacStepTime * 1000 ;
2019-02-26 14:53:43 -08:00
iacCoolTime = configPage9 . iacCoolTime * 1000 ;
2017-08-02 18:46:48 -07:00
}
2017-03-10 01:34:39 -08:00
2017-08-02 18:46:48 -07:00
idle_cl_target_rpm = table2D_getValue ( & iacClosedLoopTable , currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) * 10 ; //All temps are offset by 40 degrees
2018-05-04 00:55:58 -07:00
if ( currentStatus . idleUpActive = = true ) { idle_pid_target_value + = configPage2 . idleUpAdder ; } //Add Idle Up amount if active
2017-08-02 18:46:48 -07:00
idlePID . Compute ( ) ;
idleStepper . targetIdleStep = idle_pid_target_value ;
2017-02-19 22:57:46 -08:00
2017-08-02 18:46:48 -07:00
doStep ( ) ;
currentStatus . idleLoad = idleStepper . curIdleStep > > 1 ; //Current step count (Divided by 2 for byte)
idleCounter + + ;
}
break ;
default :
//There really should be a valid idle type
2015-08-20 06:21:27 -07:00
break ;
}
2015-05-29 00:33:00 -07:00
}
2015-08-20 21:14:47 -07:00
/*
2017-03-10 01:34:39 -08:00
Checks whether the stepper has been homed yet . If it hasn ' t , will handle the next step
Returns :
True : If the system has been homed . No other action is taken
False : If the motor has not yet been homed . Will also perform another homing step .
*/
static inline byte isStepperHomed ( )
{
2017-08-02 18:46:48 -07:00
bool isHomed = true ; //As it's the most common scenario, default value is true
2018-01-23 17:05:50 -08:00
if ( completedHomeSteps < ( configPage6 . iacStepHome * 3 ) ) //Home steps are divided by 3 from TS
2017-03-10 01:34:39 -08:00
{
2018-12-02 14:36:41 -08:00
digitalWrite ( pinStepperDir , idleStepper . lessAirDirection ) ; //homing the stepper closes off the air bleed
2017-11-26 14:24:00 -08:00
digitalWrite ( pinStepperEnable , LOW ) ; //Enable the DRV8825
2017-03-10 01:34:39 -08:00
digitalWrite ( pinStepperStep , HIGH ) ;
2018-03-09 19:07:06 -08:00
idleStepper . stepStartTime = micros_safe ( ) ;
2017-03-10 01:34:39 -08:00
idleStepper . stepperStatus = STEPPING ;
completedHomeSteps + + ;
idleOn = true ;
2017-08-02 18:46:48 -07:00
isHomed = false ;
2017-03-10 01:34:39 -08:00
}
2017-08-02 18:46:48 -07:00
return isHomed ;
2017-03-10 01:34:39 -08:00
}
/*
Checks whether a step is currently underway or whether the motor is in ' cooling ' state ( ie whether it ' s ready to begin another step or not )
Returns :
True : If a step is underway or motor is ' cooling '
False : If the motor is ready for another step
*/
static inline byte checkForStepping ( )
{
2017-08-02 18:46:48 -07:00
bool isStepping = false ;
2019-02-26 14:53:43 -08:00
unsigned int timeCheck ;
2017-08-02 18:46:48 -07:00
if ( ( idleStepper . stepperStatus = = STEPPING ) | | ( idleStepper . stepperStatus = = COOLING ) )
2017-03-10 01:34:39 -08:00
{
2019-02-26 14:53:43 -08:00
if ( idleStepper . stepperStatus = = STEPPING )
2017-03-10 01:34:39 -08:00
{
2019-02-26 14:53:43 -08:00
timeCheck = iacStepTime ;
}
else
{
timeCheck = iacCoolTime ;
}
if ( micros_safe ( ) > ( idleStepper . stepStartTime + timeCheck ) )
{
2017-03-10 01:34:39 -08:00
if ( idleStepper . stepperStatus = = STEPPING )
{
//Means we're currently in a step, but it needs to be turned off
digitalWrite ( pinStepperStep , LOW ) ; //Turn off the step
2018-03-09 19:07:06 -08:00
idleStepper . stepStartTime = micros_safe ( ) ;
2019-02-26 14:53:43 -08:00
// if there is no cool time we can miss that step out completely.
if ( iacCoolTime > 0 )
{
idleStepper . stepperStatus = COOLING ; //'Cooling' is the time the stepper needs to sit in LOW state before the next step can be made
}
else
{
idleStepper . stepperStatus = SOFF ;
}
2017-08-02 18:46:48 -07:00
isStepping = true ;
2017-03-10 01:34:39 -08:00
}
else
{
2017-04-02 04:17:29 -07:00
//Means we're in COOLING status but have been in this state long enough. Go into off state
2017-03-10 01:34:39 -08:00
idleStepper . stepperStatus = SOFF ;
2017-04-02 04:17:29 -07:00
digitalWrite ( pinStepperEnable , HIGH ) ; //Disable the DRV8825
2017-03-10 01:34:39 -08:00
}
}
else
{
//Means we're in a step, but it doesn't need to turn off yet. No further action at this time
2017-08-02 18:46:48 -07:00
isStepping = true ;
2017-03-10 01:34:39 -08:00
}
}
2017-08-02 18:46:48 -07:00
return isStepping ;
2017-03-10 01:34:39 -08:00
}
/*
Performs a step
2015-08-20 21:14:47 -07:00
*/
2017-03-10 01:34:39 -08:00
static inline void doStep ( )
2015-08-20 21:14:47 -07:00
{
2018-01-23 17:05:50 -08:00
if ( ( idleStepper . targetIdleStep < = ( idleStepper . curIdleStep - configPage6 . iacStepHyster ) ) | | ( idleStepper . targetIdleStep > = ( idleStepper . curIdleStep + configPage6 . iacStepHyster ) ) ) //Hysteris check
2017-08-02 18:46:48 -07:00
{
2018-12-02 14:36:41 -08:00
// the home position for a stepper is pintle fully seated, i.e. no airflow.
if ( idleStepper . targetIdleStep < idleStepper . curIdleStep )
{
// we are moving toward the home position (reducing air)
digitalWrite ( pinStepperDir , idleStepper . lessAirDirection ) ;
idleStepper . curIdleStep - - ;
}
else
if ( idleStepper . targetIdleStep > idleStepper . curIdleStep )
{
// we are moving away from the home position (adding air).
digitalWrite ( pinStepperDir , idleStepper . moreAirDirection ) ;
idleStepper . curIdleStep + + ;
}
2017-08-02 18:46:48 -07:00
digitalWrite ( pinStepperEnable , LOW ) ; //Enable the DRV8825
digitalWrite ( pinStepperStep , HIGH ) ;
2018-03-09 19:07:06 -08:00
idleStepper . stepStartTime = micros_safe ( ) ;
2017-08-02 18:46:48 -07:00
idleStepper . stepperStatus = STEPPING ;
idleOn = true ;
}
2015-08-20 21:14:47 -07:00
}
2015-09-25 17:12:59 -07:00
2016-10-06 23:34:27 -07:00
//This function simply turns off the idle PWM and sets the pin low
static inline void disableIdle ( )
{
2018-01-23 17:05:50 -08:00
if ( ( configPage6 . iacAlgorithm = = IAC_ALGORITHM_PWM_CL ) | | ( configPage6 . iacAlgorithm = = IAC_ALGORITHM_PWM_OL ) )
2017-03-20 04:29:42 -07:00
{
IDLE_TIMER_DISABLE ( ) ;
digitalWrite ( pinIdle1 , LOW ) ;
}
2019-02-26 14:52:37 -08:00
else if ( ( configPage6 . iacAlgorithm = = IAC_ALGORITHM_STEP_OL ) )
2017-03-20 04:29:42 -07:00
{
2017-11-27 15:09:54 -08:00
//Only disable the stepper motor if homing is completed
2018-07-22 15:28:56 -07:00
if ( ( checkForStepping ( ) = = false ) & & ( isStepperHomed ( ) = = true ) )
2017-11-26 22:39:53 -08:00
{
2019-02-26 14:52:37 -08:00
/* for open loop stepper we should just move to the cranking position when
disabling idle , since the only time this function is called in this scenario
is if the engine stops .
*/
idleStepper . targetIdleStep = table2D_getValue ( & iacCrankStepsTable , ( currentStatus . coolant + CALIBRATION_TEMPERATURE_OFFSET ) ) * 3 ; //All temps are offset by 40 degrees. Step counts are divided by 3 in TS. Multiply back out here
if ( currentStatus . idleUpActive = = true ) { idleStepper . targetIdleStep + = configPage2 . idleUpAdder ; } //Add Idle Up amount if active?
2017-11-26 22:39:53 -08:00
}
2017-03-20 04:29:42 -07:00
}
2019-02-03 19:32:05 -08:00
BIT_CLEAR ( currentStatus . spark , BIT_SPARK_IDLE ) ; //Turn the idle control flag off
currentStatus . idleLoad = 0 ;
2016-10-06 23:34:27 -07:00
}
//Any common functions associated with starting the Idle
//Typically this is enabling the PWM interrupt
static inline void enableIdle ( )
{
2018-01-23 17:05:50 -08:00
if ( ( configPage6 . iacAlgorithm = = IAC_ALGORITHM_PWM_CL ) | | ( configPage6 . iacAlgorithm = = IAC_ALGORITHM_PWM_OL ) )
2017-03-20 04:29:42 -07:00
{
IDLE_TIMER_ENABLE ( ) ;
}
2018-01-23 17:05:50 -08:00
else if ( ( configPage6 . iacAlgorithm = = IAC_ALGORITHM_STEP_CL ) | | ( configPage6 . iacAlgorithm = = IAC_ALGORITHM_STEP_OL ) )
2017-03-20 04:29:42 -07:00
{
}
2016-10-06 23:34:27 -07:00
}
2017-08-02 18:46:48 -07:00
# if defined(CORE_AVR) //AVR chips use the ISR for this
2015-09-29 00:21:00 -07:00
ISR ( TIMER4_COMPC_vect )
2019-01-18 02:46:03 -08:00
# else
2017-02-28 19:42:24 -08:00
static inline void idleInterrupt ( ) //Most ARM chips can simply call a function
# endif
2015-09-25 17:12:59 -07:00
{
if ( idle_pwm_state )
{
2018-01-23 17:05:50 -08:00
if ( configPage6 . iacPWMdir = = 0 )
2017-02-19 22:57:46 -08:00
{
2016-05-27 05:44:40 -07:00
//Normal direction
* idle_pin_port & = ~ ( idle_pin_mask ) ; // Switch pin to low (1 pin mode)
2018-01-23 17:05:50 -08:00
if ( configPage6 . iacChannels = = 1 ) { * idle2_pin_port | = ( idle2_pin_mask ) ; } //If 2 idle channels are in use, flip idle2 to be the opposite of idle1
2016-05-27 05:44:40 -07:00
}
else
{
//Reversed direction
* idle_pin_port | = ( idle_pin_mask ) ; // Switch pin high
2018-01-23 17:05:50 -08:00
if ( configPage6 . iacChannels = = 1 ) { * idle2_pin_port & = ~ ( idle2_pin_mask ) ; } //If 2 idle channels are in use, flip idle2 to be the opposite of idle1
2016-05-27 05:44:40 -07:00
}
2017-02-28 19:42:24 -08:00
IDLE_COMPARE = IDLE_COUNTER + ( idle_pwm_max_count - idle_pwm_cur_value ) ;
2017-02-19 22:57:46 -08:00
idle_pwm_state = false ;
2015-09-25 17:12:59 -07:00
}
else
{
2018-01-23 17:05:50 -08:00
if ( configPage6 . iacPWMdir = = 0 )
2017-02-19 22:57:46 -08:00
{
2016-05-27 05:44:40 -07:00
//Normal direction
* idle_pin_port | = ( idle_pin_mask ) ; // Switch pin high
2018-01-23 17:05:50 -08:00
if ( configPage6 . iacChannels = = 1 ) { * idle2_pin_port & = ~ ( idle2_pin_mask ) ; } //If 2 idle channels are in use, flip idle2 to be the opposite of idle1
2016-05-27 05:44:40 -07:00
}
else
{
//Reversed direction
* idle_pin_port & = ~ ( idle_pin_mask ) ; // Switch pin to low (1 pin mode)
2018-01-23 17:05:50 -08:00
if ( configPage6 . iacChannels = = 1 ) { * idle2_pin_port | = ( idle2_pin_mask ) ; } //If 2 idle channels are in use, flip idle2 to be the opposite of idle1
2016-05-27 05:44:40 -07:00
}
2017-02-28 19:42:24 -08:00
IDLE_COMPARE = IDLE_COUNTER + idle_pwm_target_value ;
2015-09-29 16:27:37 -07:00
idle_pwm_cur_value = idle_pwm_target_value ;
2017-02-19 22:57:46 -08:00
idle_pwm_state = true ;
2015-09-25 17:12:59 -07:00
}
}
2018-12-29 09:26:30 -08:00
# if defined(CORE_TEENSY)
void ftm2_isr ( void )
{
//FTM2 only has 2 compare channels
//Use separate variables for each test to ensure conversion to bool
bool interrupt1 = ( FTM2_C0SC & FTM_CSC_CHF ) ;
bool interrupt2 = ( FTM2_C1SC & FTM_CSC_CHF ) ; //Not currently used
if ( interrupt1 ) { FTM2_C0SC & = ~ FTM_CSC_CHF ; idleInterrupt ( ) ; }
else if ( interrupt2 ) { FTM2_C1SC & = ~ FTM_CSC_CHF ; } //Add a callback function here if this is ever used
}
# endif