2013-02-04 13:05:35 -08:00
/*
This file is used for everything related to maps / tables including their definition , functions etc
*/
2013-07-09 05:12:35 -07:00
# include <Arduino.h>
2013-02-04 13:05:35 -08:00
struct table {
//All tables must be the same size for simplicity
2013-07-16 05:29:17 -07:00
const static byte xSize = 8 ;
const static byte ySize = 8 ;
2013-02-04 13:05:35 -08:00
2013-07-09 05:12:35 -07:00
byte values [ ySize ] [ xSize ] ;
2013-02-10 16:34:30 -08:00
int axisX [ xSize ] ;
int axisY [ ySize ] ;
2013-02-04 13:05:35 -08:00
} ;
2013-02-05 00:27:06 -08:00
/*
Tables have an origin ( 0 , 0 ) in the top left hand corner . Vertical axis is expressed first .
Eg : 2 x2 table
- - - - -
| 2 7 |
| 1 4 |
- - - - -
( 0 , 1 ) = 7
( 0 , 0 ) = 2
( 1 , 0 ) = 1
*/
2013-02-05 01:41:41 -08:00
//This function pulls a value from a table given a target for X and Y coordinates.
//It performs a 2D linear interpolation as descibred in: http://www.megamanual.com/v22manual/ve_tuner.pdf
2013-02-05 00:27:06 -08:00
int getTableValue ( struct table fromTable , int Y , int X )
2013-02-04 13:05:35 -08:00
{
2013-02-05 01:41:41 -08:00
//Loop through the X axis bins for the min/max pair
//Note: For the X axis specifically, rather than looping from tableAxisX[0] up to tableAxisX[max], we start at tableAxisX[Max] and go down.
// This is because the important tables (fuel and injection) will have the highest RPM at the top of the X axis, so starting there will mean the best case occurs when the RPM is highest (And hence the CPU is needed most)
2013-07-03 15:50:07 -07:00
int xMinValue = fromTable . axisX [ 0 ] ;
int xMaxValue = fromTable . axisX [ fromTable . xSize - 1 ] ;
int xMin = 0 ;
int xMax = 0 ;
2013-02-05 05:17:54 -08:00
2013-02-05 02:11:59 -08:00
for ( int x = fromTable . xSize - 1 ; x > 0 ; x - - )
2013-02-05 00:27:06 -08:00
{
2013-07-03 15:50:07 -07:00
//Checks the case where the X value is exactly what was requested
if ( X = = fromTable . axisX [ x - 1 ] )
{
xMaxValue = fromTable . axisX [ x - 1 ] ;
xMinValue = fromTable . axisX [ x - 1 ] ;
xMax = x - 1 ;
xMin = x - 1 ;
break ;
}
//Normal case
2013-02-05 02:11:59 -08:00
if ( ( X < = fromTable . axisX [ x ] ) & & ( X > = fromTable . axisX [ x - 1 ] ) )
2013-02-05 01:41:41 -08:00
{
2013-02-05 05:17:54 -08:00
xMaxValue = fromTable . axisX [ x ] ;
xMinValue = fromTable . axisX [ x - 1 ] ;
xMax = x ;
xMin = x - 1 ;
2013-02-05 01:41:41 -08:00
break ;
}
2013-02-05 00:27:06 -08:00
}
2013-02-05 01:41:41 -08:00
//Loop through the Y axis bins for the min/max pair
2013-07-03 15:50:07 -07:00
int yMinValue = fromTable . axisY [ 0 ] ;
int yMaxValue = fromTable . axisY [ fromTable . ySize - 1 ] ;
int yMin = 0 ;
int yMax = 0 ;
2013-02-05 05:17:54 -08:00
2013-02-05 02:11:59 -08:00
for ( int y = fromTable . ySize - 1 ; y > 0 ; y - - )
2013-02-05 01:41:41 -08:00
{
2013-07-03 15:50:07 -07:00
//Checks the case where the Y value is exactly what was requested
if ( Y = = fromTable . axisY [ y - 1 ] )
{
yMaxValue = fromTable . axisY [ y - 1 ] ;
yMinValue = fromTable . axisY [ y - 1 ] ;
yMax = y - 1 ;
yMin = y - 1 ;
break ;
}
//Normal case
2013-02-05 02:11:59 -08:00
if ( ( Y > = fromTable . axisY [ y ] ) & & ( Y < = fromTable . axisY [ y - 1 ] ) )
2013-02-05 01:41:41 -08:00
{
2013-02-05 05:17:54 -08:00
yMaxValue = fromTable . axisY [ y ] ;
yMinValue = fromTable . axisY [ y - 1 ] ;
yMax = y ;
yMin = y - 1 ;
2013-02-05 01:41:41 -08:00
break ;
}
}
2013-02-05 05:17:54 -08:00
2013-02-05 01:41:41 -08:00
/*
At this point we have the 4 corners of the map where the interpolated value will fall in
Eg : ( yMin , xMin ) ( yMin , xMax )
( yMax , xMin ) ( yMax , xMax )
In the following calculation the table values are referred to by the following variables :
A B
C D
*/
int A = fromTable . values [ yMin ] [ xMin ] ;
int B = fromTable . values [ yMin ] [ xMax ] ;
int C = fromTable . values [ yMax ] [ xMin ] ;
int D = fromTable . values [ yMax ] [ xMax ] ;
//Create some normalised position values
2013-02-05 05:17:54 -08:00
//These are essentially percentages (between 0 and 1) of where the desired value falls between the nearest bins on each axis
2013-02-10 15:53:00 -08:00
/*
// Float version
2013-02-05 05:17:54 -08:00
float p = ( ( float ) ( X - xMinValue ) ) / ( float ) ( xMaxValue - xMinValue ) ;
float q = ( ( float ) ( Y - yMaxValue ) ) / ( float ) ( yMinValue - yMaxValue ) ;
2013-02-05 01:41:41 -08:00
2013-02-05 02:11:59 -08:00
float m = ( 1.0 - p ) * ( 1.0 - q ) ;
float n = p * ( 1 - q ) ;
float o = ( 1 - p ) * q ;
float r = p * q ;
2013-02-05 01:41:41 -08:00
2013-02-05 02:11:59 -08:00
return ( ( A * m ) + ( B * n ) + ( C * o ) + ( D * r ) ) ;
2013-02-10 15:53:00 -08:00
*/
// Non-Float version:
2013-07-03 15:50:07 -07:00
//Initial check incase the values were hit straight on
int p ;
if ( xMaxValue = = xMinValue )
{ p = ( ( X - xMinValue ) < < 7 ) ; } //This only occurs if the requested X value was equal to one of the X axis bins
else
{ p = ( ( X - xMinValue ) < < 7 ) / ( xMaxValue - xMinValue ) ; } //This is the standard case
int q ;
if ( yMaxValue = = yMinValue )
{ q = ( ( Y - yMinValue ) < < 7 ) ; }
else
{ q = ( ( Y - yMaxValue ) < < 7 ) / ( yMinValue - yMaxValue ) ; }
2013-02-10 15:53:00 -08:00
int m = ( ( 128 - p ) * ( 128 - q ) ) > > 7 ;
int n = ( p * ( 128 - q ) ) > > 7 ;
int o = ( ( 128 - p ) * q ) > > 7 ;
int r = ( p * q ) > > 7 ;
return ( ( A * m ) + ( B * n ) + ( C * o ) + ( D * r ) ) > > 7 ;
2013-02-04 13:05:35 -08:00
}