2013-02-03 22:43:15 -08:00
//**************************************************************************************************
// Config section
2013-02-04 13:05:35 -08:00
//The following lines are configurable, but the defaults are probably pretty good for most applications
2014-12-29 16:12:01 -08:00
//#define engineInjectorDeadTime 2500 //Time in uS that the injector takes to open minus the time it takes to close
2013-02-05 00:27:06 -08:00
# define engineSquirtsPerCycle 2 //Would be 1 for a 2 stroke
2013-06-26 20:08:49 -07:00
2013-02-03 22:43:15 -08:00
//**************************************************************************************************
2013-07-18 04:53:40 -07:00
# include "globals.h"
2013-02-04 03:43:38 -08:00
# include "utils.h"
2013-02-04 13:05:35 -08:00
# include "table.h"
2013-02-05 00:27:06 -08:00
# include "testing.h"
2013-02-13 03:49:36 -08:00
# include "scheduler.h"
2013-07-18 04:53:40 -07:00
# include "storage.h"
2013-07-08 17:43:41 -07:00
# include "comms.h"
2013-09-23 05:23:34 -07:00
# include "math.h"
2014-01-07 00:02:00 -08:00
# include "corrections.h"
2014-02-06 01:48:19 -08:00
# include "timers.h"
2013-02-04 03:43:38 -08:00
2013-06-26 20:08:49 -07:00
# include "fastAnalog.h"
2013-09-17 23:45:53 -07:00
# define DIGITALIO_NO_MIX_ANALOGWRITE
2013-09-16 00:38:52 -07:00
# include "digitalIOPerformance.h"
2013-02-03 22:43:15 -08:00
2013-07-16 20:09:18 -07:00
struct config1 configPage1 ;
struct config2 configPage2 ;
2013-07-16 05:29:17 -07:00
2014-12-29 16:12:01 -08:00
int req_fuel_uS , triggerToothAngle , inj_opentime_uS ;
2013-09-15 17:18:33 -07:00
volatile int triggerActualTeeth ;
2014-09-08 20:59:27 -07:00
unsigned int triggerFilterTime ; // The shortest time (in uS) that pulses will be accepted (Used for debounce filtering)
2014-07-07 16:05:32 -07:00
# define MAX_RPM 9000 //This is the maximum rpm that the ECU will attempt to run at. It is NOT related to the rev limiter, but is instead dictates how fast certain operations will be allowed to run. Lower number gives better performance
2013-02-04 02:49:41 -08:00
2013-02-04 13:05:35 -08:00
volatile int toothCurrentCount = 0 ; //The current number of teeth (Onec sync has been achieved, this can never actually be 0
volatile unsigned long toothLastToothTime = 0 ; //The time (micros()) that the last tooth was registered
volatile unsigned long toothLastMinusOneToothTime = 0 ; //The time (micros()) that the tooth before the last tooth was registered
2013-08-25 21:11:47 -07:00
volatile unsigned long toothOneTime = 0 ; //The time (micros()) that tooth 1 last triggered
volatile unsigned long toothOneMinusOneTime = 0 ; //The 2nd to last time (micros()) that tooth 1 last triggered
2014-10-02 21:19:37 -07:00
volatile int toothHistory [ 512 ] ;
volatile int toothHistoryIndex = 0 ;
2014-08-25 05:03:57 -07:00
volatile byte startRevolutions = 0 ; //A counter for how many revolutions have been completed since sync was achieved.
volatile bool ignitionOn = true ; //The current state of the ignition system
2013-02-04 13:05:35 -08:00
2014-01-07 00:02:00 -08:00
struct table3D fuelTable ; //8x8 fuel map
struct table3D ignitionTable ; //8x8 ignition map
2014-02-26 18:26:04 -08:00
struct table2D taeTable ; //4 bin TPS Acceleration Enrichment map (2D)
2014-02-26 20:17:14 -08:00
struct table2D WUETable ; //10 bin Warm Up Enrichment map (2D)
2014-12-10 15:21:59 -08:00
byte cltCalibrationTable [ CALIBRATION_TABLE_SIZE ] ;
byte iatCalibrationTable [ CALIBRATION_TABLE_SIZE ] ;
byte o2CalibrationTable [ CALIBRATION_TABLE_SIZE ] ;
2013-02-03 22:43:15 -08:00
2013-02-13 13:51:29 -08:00
unsigned long counter ;
2014-02-13 14:27:33 -08:00
unsigned long currentLoopTime ; //The time the current loop started (uS)
unsigned long previousLoopTime ; //The time the previous loop started (uS)
2013-02-13 22:34:29 -08:00
unsigned long scheduleStart ;
unsigned long scheduleEnd ;
2013-02-05 05:17:54 -08:00
2013-10-14 05:53:20 -07:00
byte coilHIGH = HIGH ;
byte coilLOW = LOW ;
2013-07-09 17:26:16 -07:00
struct statuses currentStatus ;
2014-02-18 02:05:13 -08:00
volatile int mainLoopCount ;
2014-01-29 21:28:21 -08:00
unsigned long secCounter ; //The next time to increment 'runSecs' counter.
2015-01-02 14:09:19 -08:00
int channel1Degrees ; //The number of crank degrees until cylinder 1 is at TDC (This is obviously 0 for virtually ALL engines, but there's some weird ones)
int channel2Degrees ; //The number of crank degrees until cylinder 2 (and 5/6/7/8) is at TDC
int channel3Degrees ; //The number of crank degrees until cylinder 3 (and 5/6/7/8) is at TDC
int channel4Degrees ; //The number of crank degrees until cylinder 4 (and 5/6/7/8) is at TDC
2014-12-08 20:31:48 -08:00
int timePerDegree ;
2013-07-09 17:26:16 -07:00
2013-09-15 17:18:33 -07:00
void setup ( )
{
2013-11-13 22:17:58 -08:00
pinMode ( pinCoil1 , OUTPUT ) ;
pinMode ( pinCoil2 , OUTPUT ) ;
pinMode ( pinCoil3 , OUTPUT ) ;
pinMode ( pinCoil4 , OUTPUT ) ;
2013-11-14 01:08:14 -08:00
pinMode ( pinInjector1 , OUTPUT ) ;
pinMode ( pinInjector2 , OUTPUT ) ;
pinMode ( pinInjector3 , OUTPUT ) ;
pinMode ( pinInjector4 , OUTPUT ) ;
2013-09-24 06:30:45 -07:00
2014-02-26 20:17:14 -08:00
2013-09-15 17:18:33 -07:00
//Setup the dummy fuel and ignition tables
//dummyFuelTable(&fuelTable);
//dummyIgnitionTable(&ignitionTable);
loadConfig ( ) ;
2014-02-26 20:17:14 -08:00
2014-05-06 04:07:49 -07:00
//Repoint the 2D table structs to the config pages that were just loaded
2014-05-12 04:28:24 -07:00
taeTable . valueSize = SIZE_BYTE ; //Set this table to use byte values
2014-02-26 20:17:14 -08:00
taeTable . xSize = 4 ;
taeTable . values = configPage2 . taeValues ;
taeTable . axisX = configPage2 . taeBins ;
2014-05-12 04:28:24 -07:00
WUETable . valueSize = SIZE_BYTE ; //Set this table to use byte values
2014-02-26 20:17:14 -08:00
WUETable . xSize = 10 ;
WUETable . values = configPage1 . wueValues ;
WUETable . axisX = configPage2 . wueBins ;
2015-01-18 14:10:01 -08:00
WUETable . axisX [ 0 ] = 0 ;
WUETable . axisX [ 1 ] = 11 ;
WUETable . axisX [ 2 ] = 22 ;
WUETable . axisX [ 3 ] = 33 ;
WUETable . axisX [ 4 ] = 44 ;
WUETable . axisX [ 5 ] = 56 ;
WUETable . axisX [ 6 ] = 67 ;
WUETable . axisX [ 7 ] = 78 ;
WUETable . axisX [ 8 ] = 94 ;
WUETable . axisX [ 9 ] = 111 ;
2014-05-12 04:28:24 -07:00
//Setup the calibration tables
loadCalibration ( ) ;
2013-10-14 05:53:20 -07:00
//Need to check early on whether the coil charging is inverted. If this is not set straight away it can cause an unwanted spark at bootup
if ( configPage2 . IgInv = = 1 ) { coilHIGH = LOW , coilLOW = HIGH ; }
else { coilHIGH = HIGH , coilLOW = LOW ; }
2013-11-13 22:17:58 -08:00
digitalWrite ( pinCoil1 , coilLOW ) ;
digitalWrite ( pinCoil2 , coilLOW ) ;
digitalWrite ( pinCoil3 , coilLOW ) ;
digitalWrite ( pinCoil4 , coilLOW ) ;
2013-10-14 05:53:20 -07:00
2014-12-23 01:27:46 -08:00
//Similar for injectors, make sure they're turned off
digitalWrite ( pinInjector1 , LOW ) ;
digitalWrite ( pinInjector2 , LOW ) ;
digitalWrite ( pinInjector3 , LOW ) ;
digitalWrite ( pinInjector4 , LOW ) ;
2013-09-15 17:18:33 -07:00
initialiseSchedulers ( ) ;
2014-01-30 20:02:32 -08:00
initialiseTimers ( ) ;
2013-09-15 17:18:33 -07:00
//Once the configs have been loaded, a number of one time calculations can be completed
2014-10-09 22:39:38 -07:00
req_fuel_uS = configPage1 . reqFuel * 100 ; //Convert to uS and an int. This is the only variable to be used in calculations
2013-09-15 17:18:33 -07:00
triggerToothAngle = 360 / configPage2 . triggerTeeth ; //The number of degrees that passes from tooth to tooth
triggerActualTeeth = configPage2 . triggerTeeth - configPage2 . triggerMissingTeeth ; //The number of physical teeth on the wheel. Doing this here saves us a calculation each time in the interrupt
2014-12-29 16:12:01 -08:00
inj_opentime_uS = configPage1 . injOpen * 100 ; //Injector open time. Comes through as ms*10 (Eg 15.5ms = 155).
2013-09-15 17:18:33 -07:00
2013-02-04 13:05:35 -08:00
//Begin the main crank trigger interrupt pin setup
2013-02-04 02:49:41 -08:00
//The interrupt numbering is a bit odd - See here for reference: http://arduino.cc/en/Reference/AttachInterrupt
2013-06-28 04:52:23 -07:00
//These assignments are based on the Arduino Mega AND VARY BETWEEN BOARDS. Please confirm the board you are using and update acordingly.
2013-11-14 01:08:14 -08:00
int triggerInterrupt = 0 ; // By default, use the first interrupt
2013-07-09 17:26:16 -07:00
currentStatus . RPM = 0 ;
2013-09-24 06:30:45 -07:00
currentStatus . hasSync = false ;
2014-01-29 21:28:21 -08:00
currentStatus . runSecs = 0 ;
2014-05-08 03:46:38 -07:00
currentStatus . secl = 0 ;
2014-07-07 16:05:32 -07:00
triggerFilterTime = ( int ) ( 1000000 / ( MAX_RPM / 60 * configPage2 . triggerTeeth ) ) ; //Trigger filter time is the shortest possible time (in uS) that there can be between crank teeth (ie at max RPM). Any pulses that occur faster than this time will be disgarded as noise
2014-01-29 21:28:21 -08:00
2013-06-27 17:21:22 -07:00
switch ( pinTrigger ) {
2013-06-28 04:52:23 -07:00
2013-11-14 01:08:14 -08:00
//Arduino Mega 2560 mapping
2013-02-04 02:49:41 -08:00
case 2 :
triggerInterrupt = 0 ; break ;
case 3 :
triggerInterrupt = 1 ; break ;
case 18 :
triggerInterrupt = 5 ; break ;
case 19 :
triggerInterrupt = 4 ; break ;
case 20 :
triggerInterrupt = 3 ; break ;
case 21 :
triggerInterrupt = 2 ; break ;
2013-08-25 21:11:47 -07:00
2013-02-03 22:43:15 -08:00
}
2013-08-25 21:11:47 -07:00
pinMode ( pinTrigger , INPUT ) ;
2014-12-29 21:23:18 -08:00
//digitalWrite(pinTrigger, HIGH);
2013-09-13 07:00:17 -07:00
attachInterrupt ( triggerInterrupt , trigger , FALLING ) ; // Attach the crank trigger wheel interrupt (Hall sensor drags to ground when triggering)
2013-02-04 02:49:41 -08:00
//End crank triger interrupt attachment
2013-07-03 03:27:46 -07:00
req_fuel_uS = req_fuel_uS / engineSquirtsPerCycle ; //The req_fuel calculation above gives the total required fuel (At VE 100%) in the full cycle. If we're doing more than 1 squirt per cycle then we need to split the amount accordingly. (Note that in a non-sequential 4-stroke setup you cannot have less than 2 squirts as you cannot determine the stroke to make the single squirt on)
2014-02-13 14:27:33 -08:00
//Initial values for loop times
previousLoopTime = 0 ;
currentLoopTime = micros ( ) ;
2013-02-04 02:49:41 -08:00
2013-09-03 05:13:28 -07:00
Serial . begin ( 115200 ) ;
2013-02-04 02:49:41 -08:00
2013-06-28 04:52:23 -07:00
//This sets the ADC (Analog to Digitial Converter) to run at 1Mhz, greatly reducing analog read times (MAP/TPS)
//1Mhz is the fastest speed permitted by the CPU without affecting accuracy
//Please see chapter 11 of 'Practical Arduino' (http://books.google.com.au/books?id=HsTxON1L6D4C&printsec=frontcover#v=onepage&q&f=false) for more details
//Can be disabled by removing the #include "fastAnalog.h" above
2013-06-26 20:08:49 -07:00
# ifdef sbi
2013-06-27 17:21:22 -07:00
sbi ( ADCSRA , ADPS2 ) ;
cbi ( ADCSRA , ADPS1 ) ;
cbi ( ADCSRA , ADPS0 ) ;
2013-06-26 20:08:49 -07:00
# endif
2014-02-17 22:08:55 -08:00
mainLoopCount = 0 ;
2013-07-08 17:43:41 -07:00
2013-09-26 04:27:20 -07:00
//Setup other relevant pins
pinMode ( pinMAP , INPUT ) ;
pinMode ( pinO2 , INPUT ) ;
pinMode ( pinTPS , INPUT ) ;
2013-11-14 01:08:14 -08:00
pinMode ( pinIAT , INPUT ) ;
2014-12-23 01:27:46 -08:00
pinMode ( pinCLT , INPUT ) ;
2013-09-26 04:27:20 -07:00
//Turn on pullups for above pins
2013-10-14 05:53:20 -07:00
digitalWrite ( pinMAP , HIGH ) ;
2014-12-23 01:27:46 -08:00
//digitalWrite(pinO2, LOW);
2013-09-26 04:27:20 -07:00
digitalWrite ( pinTPS , LOW ) ;
2014-05-26 17:06:02 -07:00
//Calculate the number of degrees between cylinders
switch ( configPage1 . nCylinders ) {
case 1 :
2015-01-02 14:09:19 -08:00
channel1Degrees = 0 ;
2014-05-26 17:06:02 -07:00
break ;
case 2 :
2015-01-02 14:09:19 -08:00
channel1Degrees = 0 ;
channel2Degrees = 180 ;
break ;
case 3 :
channel1Degrees = 0 ;
channel2Degrees = 120 ;
channel3Degrees = 240 ;
2014-05-26 17:06:02 -07:00
break ;
case 4 :
2015-01-02 14:09:19 -08:00
channel1Degrees = 0 ;
channel2Degrees = 180 ;
2014-05-26 17:06:02 -07:00
break ;
case 6 :
2015-01-02 14:09:19 -08:00
channel1Degrees = 0 ;
channel2Degrees = 120 ;
channel3Degrees = 240 ;
2014-05-26 17:06:02 -07:00
break ;
default : //Handle this better!!!
2015-01-02 14:09:19 -08:00
channel1Degrees = 0 ;
channel2Degrees = 180 ;
2014-05-26 17:06:02 -07:00
break ;
}
2013-02-03 22:43:15 -08:00
}
2013-02-04 02:49:41 -08:00
void loop ( )
{
2014-02-17 22:08:55 -08:00
mainLoopCount + + ;
2015-01-26 00:40:48 -08:00
//Check for any requets from serial. Serial operations are checked under 2 scenarios:
// 1) Every 64 loops (64 Is more than fast enough for TunerStudio). This function is equivalent to ((loopCount % 64) == 1) but is considerably faster due to not using the mod or division operations
// 2) If the amount of data in the serial buffer is greater than a set threhold (See globals.h). This is to avoid serial buffer overflow when large amounts of data is being sent
2015-01-27 18:04:23 -08:00
if ( ( ( mainLoopCount & 63 ) = = 1 ) or ( Serial . available ( ) > SERIAL_BUFFER_THRESHOLD ) )
2013-09-14 02:38:15 -07:00
{
2013-08-25 21:11:47 -07:00
if ( Serial . available ( ) > 0 )
{
command ( ) ;
}
2013-09-14 02:38:15 -07:00
}
2013-08-25 21:11:47 -07:00
2013-09-14 02:38:15 -07:00
//Calculate the RPM based on the uS between the last 2 times tooth One was seen.
2014-02-13 14:27:33 -08:00
previousLoopTime = currentLoopTime ;
currentLoopTime = micros ( ) ;
2014-12-23 15:25:40 -08:00
long timeToLastTooth = ( currentLoopTime - toothLastToothTime ) ;
if ( ( timeToLastTooth < 500000L ) | | ( toothLastToothTime > currentLoopTime ) ) //Check how long ago the last tooth was seen compared to now. If it was more than half a second ago then the engine is probably stopped. toothLastToothTime can be greater than currentLoopTime if a pulse occurs between getting the lastest time and doing the comparison
2013-09-14 02:38:15 -07:00
{
noInterrupts ( ) ;
2014-02-06 01:48:19 -08:00
unsigned long revolutionTime = ( toothOneTime - toothOneMinusOneTime ) ; //The time in uS that one revolution would take at current speed (The time tooth 1 was last seen, minus the time it was seen prior to that)
2013-09-14 02:38:15 -07:00
interrupts ( ) ;
2014-06-05 18:29:43 -07:00
currentStatus . RPM = ldiv ( US_IN_MINUTE , revolutionTime ) . quot ; //Calc RPM based on last full revolution time (uses ldiv rather than div as US_IN_MINUTE is a long)
2013-09-14 02:38:15 -07:00
}
else
{
//We reach here if the time between teeth is too great. This VERY likely means the engine has stopped
currentStatus . RPM = 0 ;
2014-12-29 16:12:01 -08:00
currentStatus . PW = 0 ;
currentStatus . VE = 0 ;
2013-09-19 03:49:28 -07:00
currentStatus . hasSync = false ;
2014-01-29 21:28:21 -08:00
currentStatus . runSecs = 0 ; //Reset the counter for number of seconds running.
secCounter = 0 ; //Reset our seconds counter.
2013-09-14 02:38:15 -07:00
}
2013-12-29 04:43:19 -08:00
//Uncomment the following for testing
2014-05-12 04:50:02 -07:00
/*
2013-12-29 04:43:19 -08:00
currentStatus . hasSync = true ;
2014-06-29 20:04:55 -07:00
currentStatus . RPM = 500 ;
2014-05-12 04:50:02 -07:00
*/
2013-09-14 02:38:15 -07:00
2013-12-28 18:03:55 -08:00
//***SET STATUSES***
//-----------------------------------------------------------------------------------------------------
2014-02-17 02:54:28 -08:00
currentStatus . TPSlast = currentStatus . TPS ;
2014-10-09 22:39:38 -07:00
currentStatus . MAP = map ( analogRead ( pinMAP ) , 0 , 1023 , 10 , 255 ) ; //Get the current MAP value
2015-01-30 04:51:01 -08:00
currentStatus . tpsADC = map ( analogRead ( pinTPS ) , 0 , 1023 , 0 , 255 ) ; //Get the current raw TPS ADC value and map it into a byte
currentStatus . TPS = map ( currentStatus . tpsADC , configPage1 . tpsMin , configPage1 . tpsMax , 0 , 100 ) ; //Take the raw TPS ADC value and convert it into a TPS% based on the calibrated values
2015-01-12 02:23:25 -08:00
2015-01-28 00:31:09 -08:00
//The IAT and CLT readings can be done less frequently. This still runs about 4 times per second
if ( ( mainLoopCount & 255 ) = = 1 )
2014-05-11 20:43:33 -07:00
{
2015-01-12 04:23:43 -08:00
currentStatus . cltADC = map ( analogRead ( pinCLT ) , 0 , 1023 , 0 , 511 ) ; //Get the current raw CLT value
currentStatus . iatADC = map ( analogRead ( pinIAT ) , 0 , 1023 , 0 , 511 ) ; //Get the current raw IAT value
currentStatus . O2ADC = map ( analogRead ( pinO2 ) , 0 , 1023 , 0 , 511 ) ; //Get the current O2 value. Calibration is from AFR values 7.35 to 22.4. This is the correct calibration for an Innovate Wideband 0v - 5V unit. Proper calibration is still a WIP
2014-10-14 17:49:11 -07:00
currentStatus . battery10 = map ( analogRead ( pinBat ) , 0 , 1023 , 0 , 245 ) ; //Get the current raw Battery value. Permissible values are from 0v to 24.5v (245)
//currentStatus.batADC = map(analogRead(pinBat), 0, 1023, 0, 255); //Get the current raw Battery value
2014-05-11 20:43:33 -07:00
2014-12-15 05:34:02 -08:00
currentStatus . coolant = cltCalibrationTable [ currentStatus . cltADC ] - CALIBRATION_TEMPERATURE_OFFSET ; //Temperature calibration values are stored as positive bytes. We subtract 40 from them to allow for negative temperatures
currentStatus . IAT = iatCalibrationTable [ currentStatus . iatADC ] - CALIBRATION_TEMPERATURE_OFFSET ;
2015-01-12 02:23:25 -08:00
currentStatus . O2 = o2CalibrationTable [ currentStatus . O2ADC ] ;
2014-05-11 20:43:33 -07:00
}
2014-02-18 02:05:13 -08:00
2013-02-04 02:49:41 -08:00
//Always check for sync
//Main loop runs within this clause
2014-11-17 03:20:40 -08:00
if ( currentStatus . hasSync & & ( currentStatus . RPM > 0 ) )
2013-02-04 02:49:41 -08:00
{
2014-09-20 15:17:14 -07:00
//If it is, check is we're running or cranking
2014-12-23 15:25:40 -08:00
if ( currentStatus . RPM > ( ( unsigned int ) configPage2 . crankRPM * 100 ) ) //Crank RPM stored in byte as RPM / 100
2014-09-20 15:17:14 -07:00
{ //Sets the engine running bit, clears the engine cranking bit
BIT_SET ( currentStatus . engine , BIT_ENGINE_RUN ) ;
2015-01-30 04:51:01 -08:00
BIT_CLEAR ( currentStatus . engine , BIT_ENGINE_CRANK ) ;
2014-09-20 15:17:14 -07:00
}
else
{ //Sets the engine cranking bit, clears the engine running bit
BIT_SET ( currentStatus . engine , BIT_ENGINE_CRANK ) ;
BIT_CLEAR ( currentStatus . engine , BIT_ENGINE_RUN ) ;
currentStatus . runSecs = 0 ; //We're cranking (hopefully), so reset the engine run time to prompt ASE.
//Check whether enough cranking revolutions have been performed to turn the ignition on
if ( startRevolutions > configPage2 . StgCycles )
{ ignitionOn = true ; }
}
2014-01-29 21:28:21 -08:00
2013-09-19 03:49:28 -07:00
//END SETTING STATUSES
//-----------------------------------------------------------------------------------------------------
2013-02-04 13:05:35 -08:00
2013-02-13 22:34:29 -08:00
//Begin the fuel calculation
2013-12-26 02:34:32 -08:00
//Calculate an injector pulsewidth from the VE
2014-12-21 03:43:30 -08:00
currentStatus . corrections = correctionsTotal ( ) ;
//currentStatus.corrections = 100;
2014-01-09 23:54:33 -08:00
if ( configPage1 . algorithm = = 0 ) //Check with fuelling algorithm is being used
2013-12-26 02:34:32 -08:00
{
//Speed Density
2014-01-07 00:02:00 -08:00
currentStatus . VE = get3DTableValue ( fuelTable , currentStatus . MAP , currentStatus . RPM ) ; //Perform lookup into fuel map for RPM vs MAP value
2014-12-29 16:12:01 -08:00
currentStatus . PW = PW_SD ( req_fuel_uS , currentStatus . VE , currentStatus . MAP , currentStatus . corrections , inj_opentime_uS ) ;
2014-09-14 16:08:20 -07:00
if ( configPage2 . FixAng = = 0 ) //Check whether the user has set a fixed timing angle
{ currentStatus . advance = get3DTableValue ( ignitionTable , currentStatus . MAP , currentStatus . RPM ) ; } //As above, but for ignition advance
else
{ currentStatus . advance = configPage2 . FixAng ; }
2013-12-26 02:34:32 -08:00
}
else
{
//Alpha-N
2014-01-07 00:02:00 -08:00
currentStatus . VE = get3DTableValue ( fuelTable , currentStatus . TPS , currentStatus . RPM ) ; //Perform lookup into fuel map for RPM vs TPS value
2014-12-29 16:12:01 -08:00
currentStatus . PW = PW_AN ( req_fuel_uS , currentStatus . VE , currentStatus . TPS , currentStatus . corrections , inj_opentime_uS ) ; //Calculate pulsewidth using the Alpha-N algorithm (in uS)
2014-09-14 16:08:20 -07:00
if ( configPage2 . FixAng = = 0 ) //Check whether the user has set a fixed timing angle
{ currentStatus . advance = get3DTableValue ( ignitionTable , currentStatus . TPS , currentStatus . RPM ) ; } //As above, but for ignition advance
else
{ currentStatus . advance = configPage2 . FixAng ; }
2013-12-26 02:34:32 -08:00
}
2013-12-29 04:43:19 -08:00
2014-05-26 17:06:02 -07:00
int injector1StartAngle = 0 ;
int injector2StartAngle = 0 ;
2015-01-02 14:09:19 -08:00
int injector3StartAngle = 0 ; //Currently used for 3 cylinder only
2014-05-26 17:06:02 -07:00
int injector4StartAngle = 0 ; //Not used until sequential gets written
int ignition1StartAngle = 0 ;
int ignition2StartAngle = 0 ;
int ignition3StartAngle = 0 ; //Not used until sequential or 4+ cylinders support gets written
int ignition4StartAngle = 0 ; //Not used until sequential or 4+ cylinders support gets written
2013-02-13 22:34:29 -08:00
//Determine the current crank angle
2014-01-09 22:51:28 -08:00
//This is the current angle ATDC the engine is at. This is the last known position based on what tooth was last 'seen'. It is only accurate to the resolution of the trigger wheel (Eg 36-1 is 10 degrees)
2014-10-02 21:19:37 -07:00
int crankAngle = ( toothCurrentCount - 1 ) * triggerToothAngle + ( ( int ) ( configPage2 . triggerAngle ) * 4 ) ; //Number of teeth that have passed since tooth 1, multiplied by the angle each tooth represents, plus the angle that tooth 1 is ATDC. This gives accuracy only to the nearest tooth. Needs to be multipled by 4 as the trigger angle is divided by 4 for the serial protocol
2013-06-26 20:08:49 -07:00
2014-01-09 22:51:28 -08:00
//How fast are we going? Need to know how long (uS) it will take to get from one tooth to the next. We then use that to estimate how far we are between the last tooth and the next one
2014-12-08 20:31:48 -08:00
timePerDegree = ldiv ( ( toothOneTime - toothOneMinusOneTime ) , 360 ) . quot ; //The time (uS) it is currently taking to move 1 degree
2014-07-07 05:48:50 -07:00
//crankAngle += div( (int)(micros() - toothLastToothTime), timePerDegree).quot; //Estimate the number of degrees travelled since the last tooth
crankAngle + = ldiv ( ( micros ( ) - toothLastToothTime ) , timePerDegree ) . quot ; //Estimate the number of degrees travelled since the last tooth
2014-12-08 20:31:48 -08:00
if ( crankAngle > 360 ) { crankAngle - = 360 ; }
2013-02-13 22:34:29 -08:00
2015-01-02 14:09:19 -08:00
//***********************************************************************************************
//BEGIN INJECTION TIMING
2013-02-14 04:19:34 -08:00
//Determine next firing angles
2014-06-29 02:36:29 -07:00
//1
2015-01-28 00:31:09 -08:00
int PWdivTimerPerDegree = div ( currentStatus . PW , timePerDegree ) . quot ; //This variable is used multiple times, so only do the division once.
injector1StartAngle = 355 - ( PWdivTimerPerDegree ) ; //This is a little primitive, but is based on the idea that all fuel needs to be delivered before the inlet valve opens. I am using 355 as the point at which the injector MUST be closed by. See http://www.extraefi.co.uk/sequential_fuel.html for more detail
2014-05-26 17:06:02 -07:00
//Repeat the above for each cylinder
2014-06-29 02:36:29 -07:00
//2
2014-12-18 16:25:53 -08:00
if ( configPage1 . nCylinders = = 2 )
{
2015-01-28 00:31:09 -08:00
injector2StartAngle = ( 355 + channel2Degrees - ( PWdivTimerPerDegree ) ) ;
2014-12-18 16:25:53 -08:00
if ( injector2StartAngle > 360 ) { injector2StartAngle - = 360 ; }
}
2015-01-02 14:09:19 -08:00
//3
2015-01-28 00:31:09 -08:00
else if ( configPage1 . nCylinders = = 3 )
2015-01-02 14:09:19 -08:00
{
2015-01-28 00:31:09 -08:00
injector2StartAngle = ( 355 + channel2Degrees - ( PWdivTimerPerDegree ) ) ;
2015-01-02 14:09:19 -08:00
if ( injector2StartAngle > 360 ) { injector2StartAngle - = 360 ; }
2015-01-28 00:31:09 -08:00
injector3StartAngle = ( 355 + channel3Degrees - ( PWdivTimerPerDegree ) ) ;
2015-01-02 14:09:19 -08:00
if ( injector3StartAngle > 360 ) { injector3StartAngle - = 360 ; }
}
2014-06-29 02:36:29 -07:00
//4
2015-01-02 14:09:19 -08:00
else if ( configPage1 . nCylinders = = 4 )
2014-12-18 16:25:53 -08:00
{
2015-01-28 00:31:09 -08:00
injector2StartAngle = ( 355 + channel2Degrees - ( PWdivTimerPerDegree ) ) ;
2014-12-18 16:25:53 -08:00
if ( injector2StartAngle > 360 ) { injector2StartAngle - = 360 ; }
2015-01-02 14:09:19 -08:00
}
//***********************************************************************************************
//BEGIN IGNITION CALCULATIONS
2014-06-05 18:29:43 -07:00
if ( currentStatus . RPM > ( ( unsigned int ) ( configPage2 . SoftRevLim * 100 ) ) ) { currentStatus . advance - = configPage2 . SoftLimRetard ; } //Softcut RPM limit (If we're above softcut limit, delay timing by configured number of degrees)
//Calculate start angle for each channel
//1
2014-12-08 20:31:48 -08:00
int dwell = ( configPage2 . dwellRun * 100 ) ; //Dwell is stored as ms * 10. ie Dwell of 4.3ms would be 43 in configPage2. This number therefore needs to be multiplied by 100 to get dwell in uS
int dwellAngle = ( div ( dwell , timePerDegree ) . quot ) ;
ignition1StartAngle = 360 - currentStatus . advance - dwellAngle ; // 360 - desired advance angle - number of degrees the dwell will take
2014-06-05 18:29:43 -07:00
//2
2014-12-18 16:25:53 -08:00
if ( configPage1 . nCylinders = = 2 )
{
2015-01-02 14:09:19 -08:00
( ignition2StartAngle = channel2Degrees + 360 - currentStatus . advance - ( div ( ( configPage2 . dwellRun * 100 ) , timePerDegree ) . quot ) ) ;
2014-12-18 16:25:53 -08:00
if ( ignition2StartAngle > 360 ) { ignition2StartAngle - = 360 ; }
}
2015-01-02 14:09:19 -08:00
//3
else if ( configPage1 . nCylinders = = 3 )
{
( ignition2StartAngle = channel2Degrees + 360 - currentStatus . advance - ( div ( ( configPage2 . dwellRun * 100 ) , timePerDegree ) . quot ) ) ;
if ( ignition2StartAngle > 360 ) { ignition2StartAngle - = 360 ; }
( ignition3StartAngle = channel3Degrees + 360 - currentStatus . advance - ( div ( ( configPage2 . dwellRun * 100 ) , timePerDegree ) . quot ) ) ;
if ( ignition3StartAngle > 360 ) { ignition3StartAngle - = 360 ; }
}
2014-06-29 02:36:29 -07:00
//4
2015-01-02 14:09:19 -08:00
else if ( configPage1 . nCylinders = = 4 )
{
( ignition2StartAngle = channel2Degrees + 360 - currentStatus . advance - ( div ( ( configPage2 . dwellRun * 100 ) , timePerDegree ) . quot ) ) ;
2014-12-18 16:25:53 -08:00
if ( ignition2StartAngle > 360 ) { ignition2StartAngle - = 360 ; }
2015-01-02 14:09:19 -08:00
}
2013-02-13 22:47:49 -08:00
2013-02-14 04:19:34 -08:00
//Finally calculate the time (uS) until we reach the firing angles and set the schedules
//We only need to set the shcedule if we're BEFORE the open angle
//This may potentially be called a number of times as we get closer and closer to the opening time
2013-11-14 01:08:14 -08:00
if ( injector1StartAngle > crankAngle )
2013-02-14 04:19:34 -08:00
{
2013-09-17 23:45:53 -07:00
setFuelSchedule1 ( openInjector1 ,
2014-12-08 20:31:48 -08:00
( ( unsigned long ) ( injector1StartAngle - crankAngle ) * ( unsigned long ) timePerDegree ) ,
2014-09-21 12:52:32 -07:00
( unsigned long ) currentStatus . PW ,
2013-09-17 23:45:53 -07:00
closeInjector1
2013-02-14 04:19:34 -08:00
) ;
}
2014-05-26 17:06:02 -07:00
if ( injector2StartAngle > crankAngle )
{
setFuelSchedule2 ( openInjector2 ,
2014-12-08 20:31:48 -08:00
( ( unsigned long ) ( injector2StartAngle - crankAngle ) * ( unsigned long ) timePerDegree ) ,
( unsigned long ) currentStatus . PW ,
2014-05-26 17:06:02 -07:00
closeInjector2
) ;
}
2015-01-02 14:09:19 -08:00
if ( injector3StartAngle > crankAngle )
{
setFuelSchedule3 ( openInjector3 ,
( ( unsigned long ) ( injector3StartAngle - crankAngle ) * ( unsigned long ) timePerDegree ) ,
( unsigned long ) currentStatus . PW ,
closeInjector3
) ;
}
2013-02-14 04:19:34 -08:00
//Likewise for the ignition
2014-08-25 05:03:57 -07:00
//Perform an initial check to see if the ignition is turned on
if ( ignitionOn )
{
2014-12-08 20:31:48 -08:00
if ( ( ignition1StartAngle > crankAngle ) )
2014-08-25 05:03:57 -07:00
{
if ( currentStatus . RPM < ( ( unsigned int ) ( configPage2 . HardRevLim ) * 100 ) ) //Check for hard cut rev limit (If we're above the hardcut limit, we simply don't set a spark schedule)
{
setIgnitionSchedule1 ( beginCoil1Charge ,
2014-12-08 20:31:48 -08:00
( ( unsigned long ) ( ignition1StartAngle - crankAngle ) * ( unsigned long ) timePerDegree ) ,
2014-08-25 05:03:57 -07:00
dwell ,
endCoil1Charge
) ;
}
2014-01-09 22:17:10 -08:00
}
2014-08-25 05:03:57 -07:00
if ( ignition2StartAngle > crankAngle )
{
if ( currentStatus . RPM < ( ( unsigned int ) ( configPage2 . HardRevLim ) * 100 ) ) //Check for hard cut rev limit (If we're above the hardcut limit, we simply don't set a spark schedule)
{
setIgnitionSchedule2 ( beginCoil2Charge ,
2014-12-08 20:31:48 -08:00
( ( unsigned long ) ( ignition2StartAngle - crankAngle ) * ( unsigned long ) timePerDegree ) ,
2014-08-25 05:03:57 -07:00
dwell ,
endCoil2Charge
) ;
}
2014-05-26 17:06:02 -07:00
}
2015-01-02 14:09:19 -08:00
if ( ignition3StartAngle > crankAngle )
{
if ( currentStatus . RPM < ( ( unsigned int ) ( configPage2 . HardRevLim ) * 100 ) ) //Check for hard cut rev limit (If we're above the hardcut limit, we simply don't set a spark schedule)
{
setIgnitionSchedule3 ( beginCoil3Charge ,
( ( unsigned long ) ( ignition2StartAngle - crankAngle ) * ( unsigned long ) timePerDegree ) ,
dwell ,
endCoil3Charge
) ;
}
}
2014-05-26 17:06:02 -07:00
}
2013-02-13 22:34:29 -08:00
2013-02-04 02:49:41 -08:00
}
2013-09-14 02:38:15 -07:00
2013-02-04 02:49:41 -08:00
}
2013-11-13 22:17:58 -08:00
//************************************************************************************************
2013-02-04 02:49:41 -08:00
//Interrupts
2013-09-17 23:45:53 -07:00
//These functions simply trigger the injector/coil driver off or on.
//NOTE: squirt status is changed as per http://www.msextra.com/doc/ms1extra/COM_RS232.htm#Acmd
2013-11-13 22:17:58 -08:00
void openInjector1 ( ) { digitalWrite ( pinInjector1 , HIGH ) ; BIT_SET ( currentStatus . squirt , 0 ) ; }
void closeInjector1 ( ) { digitalWrite ( pinInjector1 , LOW ) ; BIT_CLEAR ( currentStatus . squirt , 0 ) ; }
void beginCoil1Charge ( ) { digitalWrite ( pinCoil1 , coilHIGH ) ; }
void endCoil1Charge ( ) { digitalWrite ( pinCoil1 , coilLOW ) ; }
void openInjector2 ( ) { digitalWrite ( pinInjector2 , HIGH ) ; BIT_SET ( currentStatus . squirt , 1 ) ; } //Sets the relevant pin HIGH and changes the current status bit for injector 2 (2nd bit of currentStatus.squirt)
2013-11-14 01:08:14 -08:00
void closeInjector2 ( ) { digitalWrite ( pinInjector2 , LOW ) ; BIT_CLEAR ( currentStatus . squirt , 1 ) ; }
2013-11-13 22:17:58 -08:00
void beginCoil2Charge ( ) { digitalWrite ( pinCoil2 , coilHIGH ) ; }
void endCoil2Charge ( ) { digitalWrite ( pinCoil2 , coilLOW ) ; }
2014-07-06 04:50:02 -07:00
void openInjector3 ( ) { digitalWrite ( pinInjector3 , HIGH ) ; BIT_SET ( currentStatus . squirt , 2 ) ; } //Sets the relevant pin HIGH and changes the current status bit for injector 3 (3rd bit of currentStatus.squirt)
void closeInjector3 ( ) { digitalWrite ( pinInjector3 , LOW ) ; BIT_CLEAR ( currentStatus . squirt , 2 ) ; }
2013-11-13 22:17:58 -08:00
void beginCoil3Charge ( ) { digitalWrite ( pinCoil3 , coilHIGH ) ; }
void endCoil3Charge ( ) { digitalWrite ( pinCoil3 , coilLOW ) ; }
2013-09-17 23:45:53 -07:00
2014-07-06 04:50:02 -07:00
void openInjector4 ( ) { digitalWrite ( pinInjector4 , HIGH ) ; BIT_SET ( currentStatus . squirt , 3 ) ; } //Sets the relevant pin HIGH and changes the current status bit for injector 4 (4th bit of currentStatus.squirt)
void closeInjector4 ( ) { digitalWrite ( pinInjector4 , LOW ) ; BIT_CLEAR ( currentStatus . squirt , 3 ) ; }
2013-11-13 22:17:58 -08:00
void beginCoil4Charge ( ) { digitalWrite ( pinCoil4 , coilHIGH ) ; }
void endCoil4Charge ( ) { digitalWrite ( pinCoil4 , coilLOW ) ; }
2013-02-04 02:49:41 -08:00
2013-02-04 13:05:35 -08:00
//The trigger function is called everytime a crank tooth passes the sensor
2014-12-21 02:00:44 -08:00
volatile unsigned long curTime ;
volatile int curGap ;
volatile unsigned int targetGap ;
2013-02-04 02:49:41 -08:00
void trigger ( )
{
2013-02-04 13:05:35 -08:00
// http://www.msextra.com/forums/viewtopic.php?f=94&t=22976
2013-06-26 20:08:49 -07:00
// http://www.megamanual.com/ms2/wheel.htm
2013-09-13 07:00:17 -07:00
noInterrupts ( ) ; //Turn off interrupts whilst in this routine
2013-08-25 21:11:47 -07:00
2014-12-21 02:00:44 -08:00
curTime = micros ( ) ;
curGap = curTime - toothLastToothTime ;
2014-10-02 21:19:37 -07:00
if ( curGap < triggerFilterTime ) { interrupts ( ) ; return ; } //Debounce check. Pulses should never be less than triggerFilterTime, so if they are it means a false trigger. (A 36-1 wheel at 8000pm will have triggers approx. every 200uS)
2014-09-21 16:07:34 -07:00
toothCurrentCount + + ; //Increment the tooth counter
2014-10-02 21:19:37 -07:00
//High speed tooth logging history
toothHistory [ toothHistoryIndex ] = curGap ;
if ( toothHistoryIndex = = 511 )
{ toothHistoryIndex = 0 ; }
else
{ toothHistoryIndex + + ; }
2013-02-04 13:05:35 -08:00
//Begin the missing tooth detection
//If the time between the current tooth and the last is greater than 1.5x the time between the last tooth and the tooth before that, we make the assertion that we must be at the first tooth after the gap
2013-02-07 14:34:51 -08:00
//if ( (curTime - toothLastToothTime) > (1.5 * (toothLastToothTime - toothLastMinusOneToothTime))) { toothCurrentCount = 1; }
2014-12-21 02:00:44 -08:00
if ( configPage2 . triggerMissingTeeth = = 1 ) { targetGap = ( 3 * ( toothLastToothTime - toothLastMinusOneToothTime ) ) > > 1 ; } //Multiply by 1.5 (Checks for a gap 1.5x greater than the last one) (Uses bitshift to multiply by 3 then divide by 2. Much faster than multiplying by 1.5)
//else { targetGap = (10 * (toothLastToothTime - toothLastMinusOneToothTime)) >> 2; } //Multiply by 2.5 (Checks for a gap 2.5x greater than the last one)
else { targetGap = ( ( toothLastToothTime - toothLastMinusOneToothTime ) ) * 2 ; } //Multiply by 2 (Checks for a gap 2x greater than the last one)
if ( curGap > targetGap )
2013-09-04 17:27:16 -07:00
{
toothCurrentCount = 1 ;
toothOneMinusOneTime = toothOneTime ;
toothOneTime = curTime ;
2013-09-14 02:38:15 -07:00
currentStatus . hasSync = true ;
2014-08-25 05:03:57 -07:00
startRevolutions + + ; //Counter
2013-09-04 17:27:16 -07:00
}
2013-09-13 07:00:17 -07:00
2013-02-04 13:05:35 -08:00
toothLastMinusOneToothTime = toothLastToothTime ;
toothLastToothTime = curTime ;
2013-09-03 05:13:28 -07:00
interrupts ( ) ; //Turn interrupts back on
2013-02-04 02:49:41 -08:00
2013-02-03 22:43:15 -08:00
}
2013-02-04 02:49:41 -08:00
2013-02-03 22:43:15 -08:00