2013-07-16 20:09:18 -07:00
/*
This is called when a command is received over serial from TunerStudio / Megatune
It parses the command and calls the relevant function
A detailed description of each call can be found at : http : //www.msextra.com/doc/ms1extra/COM_RS232.htm
*/
2013-07-08 17:43:41 -07:00
void command ( )
{
switch ( Serial . read ( ) )
{
case ' A ' : // send 22 bytes of realtime values
sendValues ( 22 ) ;
break ;
2013-07-16 20:09:18 -07:00
case ' B ' : // Burn current values to eeprom
2013-07-18 04:53:40 -07:00
writeConfig ( ) ;
2013-07-08 17:43:41 -07:00
break ;
2013-07-16 20:09:18 -07:00
case ' C ' : // test communications. This is used by Tunerstudio to see whether there is an ECU on a given serial port
2013-07-08 17:43:41 -07:00
testComm ( ) ;
break ;
case ' P ' : // set the current page
2013-07-09 05:12:35 -07:00
//Blue
2013-09-14 04:29:21 -07:00
//digitalWrite(10, HIGH);
//digitalWrite(9, LOW);
//digitalWrite(13, LOW);
2013-07-16 20:09:18 -07:00
//A 2nd byte of data is required after the 'P' specifying the new page number.
//This loop should never need to run as the byte should already be in the buffer, but is here just in case
while ( Serial . available ( ) = = 0 ) { }
2013-07-09 05:12:35 -07:00
currentPage = Serial . read ( ) ;
2013-07-08 17:43:41 -07:00
break ;
case ' R ' : // send 39 bytes of realtime values
sendValues ( 39 ) ;
break ;
case ' S ' : // send code version
Serial . write ( ms_version ) ;
break ;
case ' Q ' : // send code version
2013-07-09 05:12:35 -07:00
//Off
2013-09-14 04:29:21 -07:00
//digitalWrite(9, LOW);
//digitalWrite(10, LOW);
//digitalWrite(13, LOW);
2013-07-08 17:43:41 -07:00
Serial . write ( ms_version ) ;
break ;
case ' V ' : // send VE table and constants
2013-07-09 05:12:35 -07:00
//Red
2013-09-14 04:29:21 -07:00
//digitalWrite(9, LOW);
//digitalWrite(10, LOW);
//digitalWrite(13, HIGH);
2013-07-08 17:43:41 -07:00
sendPage ( ) ;
break ;
case ' W ' : // receive new VE or constant at 'W'+<offset>+<newbyte>
2013-07-09 05:12:35 -07:00
//Green
2013-09-14 04:29:21 -07:00
//digitalWrite(9, HIGH);
//digitalWrite(10, LOW);
//digitalWrite(13, LOW);
2013-07-16 20:09:18 -07:00
2013-09-15 08:10:16 -07:00
byte offset ;
2013-09-24 06:29:35 -07:00
while ( Serial . available ( ) = = 0 ) { }
2013-09-15 08:10:16 -07:00
offset = Serial . read ( ) ;
while ( Serial . available ( ) = = 0 ) { }
receiveValue ( offset , Serial . read ( ) ) ;
2013-07-08 17:43:41 -07:00
break ;
default :
break ;
}
}
2013-07-16 20:09:18 -07:00
/*
This function returns the current values of a fixed group of variables
*/
2013-07-08 17:43:41 -07:00
void sendValues ( int length )
{
2013-09-24 21:56:38 -07:00
byte response [ 23 ] ;
2013-07-08 17:43:41 -07:00
response [ 0 ] = ( uint8_t ) 1 ; //rtc.sec;
2013-09-17 23:45:53 -07:00
response [ 1 ] = currentStatus . squirt ;
2013-08-25 21:11:47 -07:00
response [ 2 ] = currentStatus . engine ; // Engine Status NOT YET WORKING
2013-07-08 17:43:41 -07:00
response [ 3 ] = 0x00 ; //baro
2013-07-09 17:26:16 -07:00
response [ 4 ] = currentStatus . MAP ; //map
2013-07-08 17:43:41 -07:00
response [ 5 ] = 0x00 ; //mat
response [ 6 ] = 0x00 ; //Coolant
2013-09-16 00:39:24 -07:00
response [ 7 ] = currentStatus . TPS ; //TPS
2013-07-08 17:43:41 -07:00
response [ 8 ] = 0x00 ; //battery voltage
response [ 9 ] = 0x00 ; //O2
response [ 10 ] = 0x00 ; //Exhaust gas correction (%)
response [ 11 ] = 0x00 ; //Air Correction (%)
response [ 12 ] = 0x00 ; //Warmup enrichment (%)
2013-07-09 17:26:16 -07:00
response [ 13 ] = ( currentStatus . RPM / 100 ) ; //rpm / 100
response [ 14 ] = currentStatus . PW / 100 ; //Pulsewidth 1 divided by 10 (in ms)
2013-07-08 17:43:41 -07:00
response [ 15 ] = 0x00 ; //acceleration enrichment (ms)
response [ 16 ] = 0x00 ; //Barometer correction (%)
response [ 17 ] = 0x00 ; //Total GammaE (%)
2013-07-09 17:26:16 -07:00
response [ 18 ] = currentStatus . VE ; //Current VE 1 (%)
2013-07-08 17:43:41 -07:00
response [ 19 ] = 0x00 ; //Pulsewidth 2 divided by 10 (in ms)
2013-09-24 21:56:38 -07:00
response [ 20 ] = 0x00 ; //Current VE 2 (%)
2013-07-08 17:43:41 -07:00
response [ 21 ] = 0x00 ; //Idle
2013-09-24 21:56:38 -07:00
response [ 22 ] = currentStatus . advance ;
2013-07-08 17:43:41 -07:00
Serial . write ( response , ( size_t ) 22 ) ;
2013-09-15 08:10:16 -07:00
Serial . flush ( ) ;
2013-07-08 17:43:41 -07:00
return ;
}
2013-07-18 00:26:24 -07:00
void receiveValue ( byte offset , byte newValue )
{
2013-07-18 03:36:36 -07:00
byte * pnt_configPage ;
switch ( currentPage )
{
case vePage :
2013-07-19 04:02:19 -07:00
pnt_configPage = ( byte * ) & configPage1 ; //Setup a pointer to the relevant config page
2013-07-18 03:36:36 -07:00
if ( offset < 64 ) //New value is part of the fuel map
{
2013-07-19 04:02:19 -07:00
fuelTable . values [ 7 - offset / 8 ] [ offset % 8 ] = newValue ;
2013-09-15 05:39:55 -07:00
return ;
2013-07-18 03:36:36 -07:00
}
else if ( offset < 80 ) //New value is one of the X or Y axis bins
{
2013-07-19 04:02:19 -07:00
//Check whether this is on the X (RPM) or Y (MAP/TPS) axis
if ( offset < 72 )
{
//X Axis
2013-09-15 05:39:55 -07:00
fuelTable . axisX [ ( offset - 64 ) ] = ( int ( newValue ) * 100 ) ; //The RPM values sent by megasquirt are divided by 100, need to multiple it back by 100 to make it correct
2013-07-19 04:02:19 -07:00
}
else
{
//Y Axis
offset = 7 - ( offset - 72 ) ; //Need to do a translation to flip the order (Due to us using (0,0) in the top left rather than bottom right
2013-09-15 08:10:16 -07:00
fuelTable . axisY [ offset ] = int ( newValue ) ;
2013-07-19 04:02:19 -07:00
}
2013-09-15 05:39:55 -07:00
return ;
2013-07-18 03:36:36 -07:00
}
else //New value is one of the remaining config items
{
2013-09-15 17:18:33 -07:00
//For some reason, TunerStudio is sending offsets greater than the maximum page size. I'm not sure if it's their bug or mine, but the fix is to only update the config page if the offset is less than the maximum size
if ( offset < page_size )
{
* ( pnt_configPage + byte ( offset - 80 ) ) = newValue ; //Need to subtract 80 because the map and bins (Which make up 80 bytes) aren't part of the config pages
}
2013-09-15 05:39:55 -07:00
return ;
2013-07-18 03:36:36 -07:00
}
break ;
case ignPage : //Ignition settings page (Page 2)
pnt_configPage = ( byte * ) & configPage2 ;
if ( offset < 64 ) //New value is part of the ignition map
{
2013-07-19 04:02:19 -07:00
ignitionTable . values [ 7 - offset / 8 ] [ offset % 8 ] = newValue ;
2013-09-15 08:10:16 -07:00
return ;
2013-07-18 03:36:36 -07:00
}
else if ( offset < 80 ) //New value is one of the X or Y axis bins
{
2013-07-19 04:02:19 -07:00
//Check whether this is on the X (RPM) or Y (MAP/TPS) axis
if ( offset < 72 )
{
//X Axis
2013-09-15 08:10:16 -07:00
ignitionTable . axisX [ ( offset - 64 ) ] = int ( newValue ) * int ( 100 ) ; //The RPM values sent by megasquirt are divided by 100, need to multiple it back by 100 to make it correct
2013-07-19 04:02:19 -07:00
}
else
{
2013-09-15 08:10:16 -07:00
2013-07-19 04:02:19 -07:00
//Y Axis
offset = 7 - ( offset - 72 ) ; //Need to do a translation to flip the order
2013-09-15 08:10:16 -07:00
ignitionTable . axisY [ offset ] = int ( newValue ) ;
2013-07-19 04:02:19 -07:00
}
2013-09-15 08:10:16 -07:00
return ;
2013-07-18 03:36:36 -07:00
}
else //New value is one of the remaining config items
{
2013-09-15 17:18:33 -07:00
//For some reason, TunerStudio is sending offsets greater than the maximum page size. I'm not sure if it's their bug or mine, but the fix is to only update the config page if the offset is less than the maximum size
if ( offset < page_size )
{
* ( pnt_configPage + byte ( offset - 80 ) ) = newValue ; //Need to subtract 80 because the map and bins (Which make up 80 bytes) aren't part of the config pages
}
2013-09-15 08:10:16 -07:00
return ;
2013-07-18 03:36:36 -07:00
}
break ;
default :
break ;
}
2013-07-18 00:26:24 -07:00
}
2013-08-25 21:11:47 -07:00
/*
sendPage ( ) packs the data within the current page ( As set with the ' P ' command )
into a buffer and sends it .
Note that some translation of the data is required to lay it out in the way Megasqurit / TunerStudio expect it
*/
2013-07-08 17:43:41 -07:00
void sendPage ( )
{
2013-07-18 03:36:36 -07:00
byte response [ page_size ] ;
2013-07-18 00:26:24 -07:00
byte offset ;
byte * pnt_configPage ;
2013-07-08 21:06:45 -07:00
2013-09-14 04:29:21 -07:00
switch ( currentPage )
2013-07-08 21:06:45 -07:00
{
case vePage :
2013-07-09 05:12:35 -07:00
//Need to perform a translation of the values[MAP/TPS][RPM] into the MS expected format
2013-07-16 20:09:18 -07:00
//MS format has origin (0,0) in the bottom left corner, we use the top left for efficiency reasons
2013-08-25 21:11:47 -07:00
for ( byte x = 0 ; x < 64 ; x + + ) { response [ x ] = fuelTable . values [ 7 - x / 8 ] [ x % 8 ] ; } //This is slightly non-intuitive, but essentially just flips the table vertically (IE top line becomes the bottom line etc). Columns are unchanged
2013-09-14 04:29:21 -07:00
for ( byte x = 64 ; x < 72 ; x + + ) { response [ x ] = byte ( fuelTable . axisX [ ( x - 64 ) ] / 100 ) ; } //RPM Bins for VE table (Need to be dvidied by 100)
2013-09-15 08:10:16 -07:00
for ( byte y = 72 ; y < 80 ; y + + ) { response [ y ] = byte ( fuelTable . axisY [ 7 - ( y - 72 ) ] ) ; } //MAP or TPS bins for VE table
2013-07-18 00:26:24 -07:00
//All other bytes can simply be copied from the config table
pnt_configPage = ( byte * ) & configPage1 ; //Create a pointer to Page 1 in memory
offset = 80 ; //Offset is based on the amount already copied above (table + bins)
2013-09-14 04:29:21 -07:00
2013-07-18 03:36:36 -07:00
for ( byte x = offset ; x < page_size ; x + + )
2013-07-18 00:26:24 -07:00
{
2013-09-15 17:18:33 -07:00
response [ x ] = * ( pnt_configPage + byte ( x - offset ) ) ; //Each byte is simply the location in memory of configPage1 + the offset + the variable number (x)
2013-07-18 00:26:24 -07:00
}
2013-07-18 03:36:36 -07:00
Serial . write ( ( byte * ) & response , sizeof ( response ) ) ;
2013-09-15 08:10:16 -07:00
Serial . flush ( ) ;
2013-07-08 21:06:45 -07:00
break ;
2013-07-10 04:18:18 -07:00
2013-07-08 21:06:45 -07:00
case ignPage :
2013-07-09 05:12:35 -07:00
//Need to perform a translation of the values[MAP/TPS][RPM] into the MS expected format
for ( byte x = 0 ; x < 64 ; x + + ) { response [ x ] = ignitionTable . values [ 7 - x / 8 ] [ x % 8 ] ; }
2013-09-15 08:10:16 -07:00
for ( byte x = 64 ; x < 72 ; x + + ) { response [ x ] = byte ( ignitionTable . axisX [ ( x - 64 ) ] / 100 ) ; }
for ( byte y = 72 ; y < 80 ; y + + ) { response [ y ] = byte ( ignitionTable . axisY [ 7 - ( y - 72 ) ] ) ; }
2013-07-08 21:06:45 -07:00
2013-07-18 00:26:24 -07:00
//All other bytes can simply be copied from the config table
2013-07-18 03:36:36 -07:00
pnt_configPage = ( byte * ) & configPage2 ; //Create a pointer to Page 2 in memory
2013-07-18 00:26:24 -07:00
offset = 80 ; //Offset is based on the amount already copied above (table + bins)
2013-07-18 03:36:36 -07:00
for ( byte x = offset ; x < page_size ; x + + )
2013-07-18 00:26:24 -07:00
{
2013-09-17 23:45:53 -07:00
response [ x ] = * ( pnt_configPage + byte ( x - offset ) ) ; //Each byte is simply the location in memory of configPage2 + the offset + the variable number (x)
2013-07-18 00:26:24 -07:00
}
Serial . write ( ( byte * ) & response , sizeof ( response ) ) ;
2013-09-15 08:10:16 -07:00
Serial . flush ( ) ;
2013-07-10 04:18:18 -07:00
break ;
2013-07-08 21:06:45 -07:00
default :
break ;
}
2013-07-08 17:43:41 -07:00
return ;
}
void testComm ( )
{
Serial . write ( 1 ) ;
return ;
}