2013-02-03 22:43:15 -08:00
//**************************************************************************************************
// Config section
//this section is where all the user set stuff is. This will eventually be replaced by a config file
/*
Need to calculate the req_fuel figure here , preferably in pre - processor macro
*/
2013-07-16 05:29:17 -07:00
2013-02-04 13:05:35 -08:00
//The following lines are configurable, but the defaults are probably pretty good for most applications
2013-07-03 03:27:46 -07:00
# define engineInjectorDeadTime 1500 //Time in uS that the injector takes to open
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
# define pinInjector 6 //Output pin the injector is on (Assumes 1 cyl only)
# define pinCoil 7 //Pin for the coil (AS above, 1 cyl only)
# define pinTPS 8 //TPS input pin
# define pinTrigger 2 //The CAS pin
2013-07-09 17:26:16 -07:00
# define pinMAP 0 //MAP sensor pin
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-02-04 03:43:38 -08:00
2013-06-26 20:08:49 -07:00
# include "fastAnalog.h"
2013-08-25 21:11:47 -07:00
//#include "digitalIOPerformance.h"
2013-02-03 22:43:15 -08:00
2013-07-16 05:29:17 -07:00
//NEED TO LOAD FROM EEPROM HERE
2013-07-16 20:09:18 -07:00
struct config1 configPage1 ;
struct config2 configPage2 ;
2013-07-16 05:29:17 -07:00
//float req_fuel = ((engineCapacity / engineInjectorSize) / engineCylinders / engineStoich) * 100; // This doesn't seem quite correct, but I can't find why. It will be close enough to start an engine
2013-07-16 20:09:18 -07:00
int req_fuel_uS = configPage1 . reqFuel * 1000 ; //Convert to uS and an int. This is the only variable to be used in calculations
2013-02-05 00:27:06 -08:00
2013-02-03 22:43:15 -08:00
// Setup section
2013-02-04 02:49:41 -08:00
// These aren't really configuration options, more so a description of how the hardware is setup. These are things that will be defined in the recommended hardware setup
2013-07-16 20:09:18 -07:00
int 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
int triggerToothAngle = 360 / configPage2 . triggerTeeth ; //The number of degrees that passes from tooth to tooth
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
2013-02-04 13:05:35 -08:00
struct table fuelTable ;
2013-02-13 22:34:29 -08:00
struct table ignitionTable ;
2013-02-03 22:43:15 -08:00
2013-07-16 05:31:01 -07:00
//unsigned long injectTime[(int)config1.nCylinders]; //The system time in uS that each injector needs to next fire at
//boolean intjectorNeedsFire[config1.nCylinders]; //Whether each injector needs to fire or not
2013-02-03 22:43:15 -08:00
2013-02-13 13:51:29 -08:00
unsigned long counter ;
2013-02-13 22:34:29 -08:00
unsigned long scheduleStart ;
unsigned long scheduleEnd ;
2013-02-05 05:17:54 -08:00
2013-07-09 17:26:16 -07:00
struct statuses currentStatus ;
2013-02-03 22:43:15 -08:00
void setup ( ) {
2013-02-04 02:49:41 -08: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-02-04 13:05:35 -08:00
int triggerInterrupt = 0 ; // By default, use the first interrupt. The user should always have set things up (Or even better, use the recommended pinouts)
2013-07-09 17:26:16 -07:00
currentStatus . RPM = 0 ;
currentStatus . hasSync = false ;
2013-06-27 17:21:22 -07:00
switch ( pinTrigger ) {
2013-06-28 04:52:23 -07:00
//Arduino Mega 2560 mapping (Uncomment to use)
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-06-28 04:52:23 -07:00
//Arduino Leo(nardo/stick) mapping (Comment this section if using a mega)
2013-08-25 21:11:47 -07:00
/*
2013-06-28 04:52:23 -07:00
case 3 :
triggerInterrupt = 0 ; break ;
case 2 :
triggerInterrupt = 1 ; break ;
case 0 :
triggerInterrupt = 2 ; break ;
case 1 :
triggerInterrupt = 3 ; break ;
case 7 :
triggerInterrupt = 4 ; 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 ) ;
digitalWrite ( pinTrigger , HIGH ) ;
attachInterrupt ( triggerInterrupt , trigger , RISING ) ; // 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)
2013-02-04 02:49:41 -08:00
2013-02-05 03:31:17 -08:00
Serial . begin ( 9600 ) ;
2013-07-09 05:12:35 -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
2013-06-28 04:52:23 -07:00
//Setup the dummy fuel and ignition tables
2013-02-05 05:17:54 -08:00
dummyFuelTable ( & fuelTable ) ;
2013-02-13 22:34:29 -08:00
dummyIgnitionTable ( & ignitionTable ) ;
2013-02-13 03:49:36 -08:00
initialiseScheduler ( ) ;
2013-02-13 13:51:29 -08:00
counter = 0 ;
2013-07-08 17:43:41 -07:00
2013-08-25 21:11:47 -07:00
configPage2 . triggerTeeth = 12 ; //TESTING ONLY!
configPage2 . triggerMissingTeeth = 1 ; //TESTING ONLY!
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
triggerToothAngle = 360 / configPage2 . triggerTeeth ;
2013-07-08 17:43:41 -07:00
//Setup some LEDs for testing
pinMode ( 10 , OUTPUT ) ;
pinMode ( 9 , OUTPUT ) ;
2013-02-03 22:43:15 -08:00
}
2013-02-04 02:49:41 -08:00
void loop ( )
{
2013-08-25 21:11:47 -07:00
//Check for any requets from serial
//Serial.println(toothCurrentCount);
//if (toothCurrentCount == 1) //Only check the serial buffer (And hence process serial commands) once per revolution
//{
if ( Serial . available ( ) > 0 )
{
command ( ) ;
}
//Serial.print("Trigger Teeth: ");
//Serial.println(configPage2.triggerTeeth);
/*
Serial . print ( " Tooth last time: " ) ;
Serial . println ( toothLastToothTime ) ;
Serial . print ( " Tooth last minus one time: " ) ;
Serial . println ( toothLastMinusOneToothTime ) ;
*/
Serial . print ( " RPM: " ) ;
Serial . println ( currentStatus . RPM ) ;
/*
//Serial.print("toothLastToothTime: ");
//Serial.println(toothLastToothTime);
*/
//}
2013-02-13 13:51:29 -08:00
//delay(2500);
2013-02-04 02:49:41 -08:00
//Always check for sync
//Main loop runs within this clause
2013-07-09 17:26:16 -07:00
if ( currentStatus . hasSync )
2013-02-04 02:49:41 -08:00
{
2013-02-04 13:05:35 -08:00
//Calculate the RPM based on the time between the last 2 teeth. I have no idea whether this will be accurate AT ALL, but it's fairly efficient and means there doesn't need to be another variable placed into the trigger interrupt
2013-08-25 21:11:47 -07:00
if ( toothCurrentCount ! = 1 & & toothLastMinusOneToothTime ! = 0 ) //We can't perform the RPM calculation if we're at the first tooth as the timing would be double (Well, we can, but it would need a different calculation and I don't think it's worth it, just use the last RPM value)
2013-02-04 13:05:35 -08:00
{
2013-08-25 21:11:47 -07:00
//unsigned long revolutionTime = ((long)configPage2.triggerTeeth * (toothLastToothTime - toothLastMinusOneToothTime)); //The time in uS that one revolution would take at current speed
unsigned long revolutionTime = ( toothOneTime - toothOneMinusOneTime ) ; //The time in uS that one revolution would take at current speed
//Serial.println((toothLastToothTime - toothLastMinusOneToothTime));
2013-07-09 17:26:16 -07:00
currentStatus . RPM = US_IN_MINUTE / revolutionTime ;
2013-02-04 13:05:35 -08:00
}
2013-08-25 21:11:47 -07:00
if ( ( micros ( ) - toothLastToothTime ) > 100000L ) { currentStatus . RPM = 0 ; toothLastMinusOneToothTime = 0 ; }
2013-02-04 13:05:35 -08:00
//Get the current MAP value
2013-07-09 17:26:16 -07:00
//currentStatus.MAP = 100; //Placeholder
currentStatus . MAP = map ( analogRead ( pinMAP ) , 0 , 1023 , 0 , 100 ) ;
currentStatus . TPS = 20 ; //Placeholder
2013-08-25 21:11:47 -07:00
//currentStatus.TPS = map(analogRead(pinTPS), 0, 1023, 0, 100);
2013-02-04 13:05:35 -08:00
2013-02-13 22:34:29 -08:00
//Begin the fuel calculation
2013-02-04 13:05:35 -08:00
//Perform lookup into fuel map for RPM vs MAP value
2013-07-09 17:26:16 -07:00
currentStatus . VE = getTableValue ( fuelTable , currentStatus . MAP , currentStatus . RPM ) ;
2013-02-13 22:34:29 -08:00
//Calculate an injector pulsewidth form the VE
2013-07-09 17:26:16 -07:00
currentStatus . PW = PW ( req_fuel_uS , currentStatus . VE , currentStatus . MAP , 100 , engineInjectorDeadTime ) ; //The 100 here is just a placeholder for any enrichment factors (Cold start, acceleration etc). To add 10% extra fuel, this would be 110
2013-02-13 22:34:29 -08:00
//Perform a lookup to get the desired ignition advance
2013-07-09 17:26:16 -07:00
int ignitionAdvance = getTableValue ( ignitionTable , currentStatus . MAP , currentStatus . RPM ) ;
2013-02-13 22:34:29 -08:00
//Determine the current crank angle
2013-07-16 20:09:18 -07:00
int crankAngle = ( toothCurrentCount - 1 ) * triggerToothAngle + configPage2 . triggerAngle ; //Number of teeth that have passed since tooth 1, multiplied by the angle each tooth represents, plus the angle that tooth 1 is from TDC
2013-02-13 22:34:29 -08:00
if ( crankAngle > 360 ) { crankAngle - = 360 ; } //Not sure if this is actually required
2013-06-26 20:08:49 -07:00
//Serial.print("Crank angle: "); Serial.println(crankAngle);
2013-02-14 04:19:34 -08:00
//How fast are we going? Can possibly work this out from RPM, but I don't think it's going to take a lot of CPU
unsigned long timePerDegree = ( toothLastToothTime - toothLastMinusOneToothTime ) / triggerToothAngle ; //The time (uS) it is currently taking to move 1 degree
2013-02-13 22:34:29 -08:00
2013-02-14 04:19:34 -08:00
//Determine next firing angles
2013-07-09 17:26:16 -07:00
int injectorStartAngle = 355 - ( currentStatus . PW / timePerDegree ) ; //This is a bit rough, 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
2013-02-14 04:19:34 -08:00
int ignitionStartAngle = 360 - ignitionAdvance ; //Simple
2013-02-13 22:34:29 -08:00
2013-06-26 20:08:49 -07:00
//Serial.print("Injector start angle: "); Serial.println(injectorStartAngle);
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
if ( injectorStartAngle > crankAngle )
{
setSchedule1 ( openInjector ,
( injectorStartAngle - crankAngle ) * timePerDegree ,
2013-07-09 17:26:16 -07:00
currentStatus . PW ,
2013-02-14 04:19:34 -08:00
closeInjector
) ;
}
//Likewise for the ignition
if ( ignitionStartAngle > crankAngle )
{
setSchedule2 ( beginCoilCharge ,
( ignitionStartAngle - crankAngle ) * timePerDegree ,
2013-07-16 20:09:18 -07:00
configPage2 . dwellRun ,
2013-02-14 04:19:34 -08:00
endCoilCharge
) ;
}
2013-02-13 22:34:29 -08:00
2013-02-04 02:49:41 -08:00
}
else
{ getSync ( ) ; }
}
//The get Sync function attempts to wait
void getSync ( )
{
2013-06-28 04:52:23 -07:00
//VERY basic waiting for sync routine. Artifically set the tooth count to be great than 1, then just wait for it to be reset to one (Which occurs in the trigger interrupt function)
/*
toothCurrentCount = 2 ;
while ( toothCurrentCount > 1 )
{
delay ( 1 ) ;
}
*/
2013-02-05 03:31:17 -08:00
//The are some placeholder values so we can get a fake RPM
toothLastMinusOneToothTime = micros ( ) ;
2013-08-25 21:11:47 -07:00
//delay(1); //A 1000us delay should make for about a 5000rpm test speed with a 12 tooth wheel(60000000us / (1000us * triggerTeeth)
2013-02-05 03:31:17 -08:00
toothLastToothTime = micros ( ) ;
2013-02-04 02:49:41 -08:00
2013-07-09 17:26:16 -07:00
currentStatus . hasSync = true ;
2013-02-04 02:49:41 -08:00
}
2013-02-03 22:43:15 -08:00
2013-02-04 02:49:41 -08:00
//Interrupts
2013-06-28 04:52:23 -07:00
//These 4 functions simply trigger the injector/coil driver off or on.
2013-02-13 22:47:49 -08:00
//void openInjector2() { scheduleEnd = micros();}
2013-06-28 04:52:23 -07:00
void openInjector ( ) { digitalWrite ( pinInjector , HIGH ) ; }
void closeInjector ( ) { digitalWrite ( pinInjector , LOW ) ; }
void beginCoilCharge ( ) { digitalWrite ( pinCoil , HIGH ) ; }
void endCoilCharge ( ) { digitalWrite ( pinCoil , LOW ) ; }
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
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-08-25 21:11:47 -07:00
2013-06-26 20:08:49 -07:00
unsigned long curTime = micros ( ) ;
2013-08-25 21:11:47 -07:00
if ( ( curTime - toothLastToothTime ) < 500 ) { return ; } //Debounce check. Pulses should never be less than 100uS, so if they are it means a false trigger. (A 36-1 wheel at 8000pm will have triggers approx. every 200uS)
toothCurrentCount + + ; //Increment the tooth counter
2013-02-04 13:05:35 -08:00
2013-08-25 21:11:47 -07:00
//Serial.println("Got trigger");
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; }
2013-08-25 21:11:47 -07:00
//if ( (curTime - toothLastToothTime) > ((3 * (toothLastToothTime - toothLastMinusOneToothTime))>>1)) { toothCurrentCount = 1; } //Same as above, but uses bitshift instead of multiplying by 1.5
if ( toothCurrentCount > triggerActualTeeth )
{
toothCurrentCount = 1 ;
toothOneMinusOneTime = toothOneTime ;
toothOneTime = curTime ;
} //For testing ONLY
2013-02-04 13:05:35 -08:00
toothLastMinusOneToothTime = toothLastToothTime ;
toothLastToothTime = curTime ;
2013-08-25 21:11:47 -07:00
// Update the last few tooth times
//toothLastMinusOneToothTime = toothLastToothTime;
//toothLastToothTime = curTime;
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