2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file table_helper.h
|
|
|
|
*
|
|
|
|
* @date Jul 6, 2014
|
2017-01-03 03:05:22 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2017
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
#ifndef TABLE_HELPER_H_
|
|
|
|
#define TABLE_HELPER_H_
|
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include "error_handling.h"
|
|
|
|
#include "interpolation.h"
|
2015-12-27 09:01:53 -08:00
|
|
|
#include "efilib.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2016-01-18 09:03:32 -08:00
|
|
|
/**
|
|
|
|
* this helper class brings together 3D table with two 2D axis curves
|
|
|
|
*/
|
2016-06-29 19:03:26 -07:00
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
2015-07-10 06:01:56 -07:00
|
|
|
class Map3D {
|
|
|
|
public:
|
2015-12-27 09:01:53 -08:00
|
|
|
Map3D(const char*name);
|
2016-06-30 19:02:49 -07:00
|
|
|
Map3D(const char*name, float multiplier);
|
|
|
|
void create(const char*name, float multiplier);
|
2016-06-29 19:03:26 -07:00
|
|
|
void init(vType table[RPM_BIN_SIZE][LOAD_BIN_SIZE], float loadBins[LOAD_BIN_SIZE], float rpmBins[RPM_BIN_SIZE]);
|
2016-03-11 12:01:58 -08:00
|
|
|
float getValue(float xRpm, float y);
|
2016-06-29 19:03:26 -07:00
|
|
|
void setAll(vType value);
|
|
|
|
vType *pointers[LOAD_BIN_SIZE];
|
2016-08-26 16:02:56 -07:00
|
|
|
private:
|
2015-07-10 06:01:56 -07:00
|
|
|
float *loadBins;
|
|
|
|
float *rpmBins;
|
2016-01-18 09:03:32 -08:00
|
|
|
bool initialized;
|
2015-12-27 09:01:53 -08:00
|
|
|
const char *name;
|
2016-06-30 19:02:49 -07:00
|
|
|
float multiplier;
|
2015-07-10 06:01:56 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
template<int SIZE>
|
|
|
|
class Table2D {
|
|
|
|
public:
|
|
|
|
Table2D();
|
|
|
|
void preCalc(float *bin, float *values);
|
|
|
|
float aTable[SIZE];
|
|
|
|
float bTable[SIZE];
|
|
|
|
float *bin;
|
|
|
|
};
|
|
|
|
|
|
|
|
template<int SIZE>
|
|
|
|
Table2D<SIZE>::Table2D() {
|
|
|
|
bin = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<int SIZE>
|
|
|
|
void Table2D<SIZE>::preCalc(float *bin, float *values) {
|
|
|
|
this->bin = bin;
|
|
|
|
for (int i = 0; i < SIZE - 1; i++) {
|
|
|
|
float x1 = bin[i];
|
|
|
|
float x2 = bin[i + 1];
|
|
|
|
if (x1 == x2) {
|
2018-01-23 09:05:14 -08:00
|
|
|
warning(CUSTOM_INTEPOLATE_ERROR_4, "preCalc: Same x1 and x2 in interpolate: %.2f/%.2f", x1, x2);
|
2015-07-10 06:01:56 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
float y1 = values[i];
|
|
|
|
float y2 = values[i + 1];
|
|
|
|
|
|
|
|
aTable[i] = INTERPOLATION_A(x1, y1, x2, y2);
|
|
|
|
bTable[i] = y1 - aTable[i] * x1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-29 19:03:26 -07:00
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
|
|
|
void Map3D<RPM_BIN_SIZE, LOAD_BIN_SIZE, vType>::init(vType table[RPM_BIN_SIZE][LOAD_BIN_SIZE],
|
2015-07-10 06:01:56 -07:00
|
|
|
float loadBins[LOAD_BIN_SIZE],
|
|
|
|
float rpmBins[RPM_BIN_SIZE]) {
|
2016-04-15 20:01:40 -07:00
|
|
|
// this method cannot use logger because it's invoked before everything
|
|
|
|
// that's because this method needs to be invoked before initial configuration processing
|
|
|
|
// and initial configuration load is done prior to logging initialization
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
for (int k = 0; k < LOAD_BIN_SIZE; k++) {
|
|
|
|
pointers[k] = table[k];
|
|
|
|
}
|
2016-01-18 09:03:32 -08:00
|
|
|
initialized = true;
|
2015-07-10 06:01:56 -07:00
|
|
|
this->loadBins = loadBins;
|
|
|
|
this->rpmBins = rpmBins;
|
|
|
|
}
|
|
|
|
|
2016-06-29 19:03:26 -07:00
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
|
|
|
float Map3D<RPM_BIN_SIZE, LOAD_BIN_SIZE, vType>::getValue(float xRpm, float y) {
|
2016-01-18 09:03:32 -08:00
|
|
|
efiAssert(initialized, "map not initialized", NAN);
|
2016-03-11 12:01:58 -08:00
|
|
|
if (cisnan(y)) {
|
2017-04-19 19:03:14 -07:00
|
|
|
warning(CUSTOM_PARAM_RANGE, "%s: y is NaN", name);
|
2015-12-27 09:01:53 -08:00
|
|
|
return NAN;
|
|
|
|
}
|
2016-03-11 12:01:58 -08:00
|
|
|
// todo: we have a bit of a mess: in TunerStudio, RPM is X-axis
|
2016-06-30 19:02:49 -07:00
|
|
|
return multiplier * interpolate3d<vType>(y, loadBins, LOAD_BIN_SIZE, xRpm, rpmBins, RPM_BIN_SIZE, pointers);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2016-06-29 19:03:26 -07:00
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
|
|
|
Map3D<RPM_BIN_SIZE, LOAD_BIN_SIZE, vType>::Map3D(const char *name) {
|
2016-06-30 19:02:49 -07:00
|
|
|
create(name, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
|
|
|
Map3D<RPM_BIN_SIZE, LOAD_BIN_SIZE, vType>::Map3D(const char *name, float multiplier) {
|
|
|
|
create(name, multiplier);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
|
|
|
void Map3D<RPM_BIN_SIZE, LOAD_BIN_SIZE, vType>::create(const char *name, float multiplier) {
|
2015-12-27 09:01:53 -08:00
|
|
|
this->name = name;
|
2016-06-30 19:02:49 -07:00
|
|
|
this->multiplier = multiplier;
|
2016-04-15 20:01:40 -07:00
|
|
|
initialized = false;
|
2015-07-10 06:01:56 -07:00
|
|
|
memset(&pointers, 0, sizeof(pointers));
|
|
|
|
loadBins = NULL;
|
|
|
|
rpmBins = NULL;
|
|
|
|
}
|
|
|
|
|
2016-06-29 19:03:26 -07:00
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
|
|
|
void Map3D<RPM_BIN_SIZE, LOAD_BIN_SIZE, vType>::setAll(vType value) {
|
2018-07-25 20:03:04 -07:00
|
|
|
efiAssertVoid(CUSTOM_ERR_6573, initialized, "map not initialized");
|
2015-07-10 06:01:56 -07:00
|
|
|
for (int l = 0; l < LOAD_BIN_SIZE; l++) {
|
|
|
|
for (int r = 0; r < RPM_BIN_SIZE; r++) {
|
2016-06-30 19:02:49 -07:00
|
|
|
pointers[l][r] = value / multiplier;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-02 15:02:12 -07:00
|
|
|
template<int RPM_BIN_SIZE, int LOAD_BIN_SIZE, typename vType>
|
|
|
|
void copy2DTable(const vType source[LOAD_BIN_SIZE][RPM_BIN_SIZE], vType destination[LOAD_BIN_SIZE][RPM_BIN_SIZE]) {
|
|
|
|
for (int k = 0; k < LOAD_BIN_SIZE; k++) {
|
|
|
|
for (int rpmIndex = 0; rpmIndex < RPM_BIN_SIZE; rpmIndex++) {
|
|
|
|
destination[k][rpmIndex] = source[k][rpmIndex];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-01 20:01:22 -07:00
|
|
|
/**
|
|
|
|
* AFR value is packed into uint8_t with a multiplier of 10
|
|
|
|
*/
|
|
|
|
#define AFR_STORAGE_MULT 10
|
|
|
|
|
|
|
|
typedef Map3D<FUEL_RPM_COUNT, FUEL_LOAD_COUNT, uint8_t> afr_Map3D_t;
|
2016-06-29 19:03:26 -07:00
|
|
|
typedef Map3D<IGN_RPM_COUNT, IGN_LOAD_COUNT, float> ign_Map3D_t;
|
|
|
|
typedef Map3D<FUEL_RPM_COUNT, FUEL_LOAD_COUNT, float> fuel_Map3D_t;
|
|
|
|
typedef Map3D<BARO_CORR_SIZE, BARO_CORR_SIZE, float> baroCorr_Map3D_t;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2016-01-21 18:03:08 -08:00
|
|
|
void setRpmBin(float array[], int size, float idleRpm, float topRpm);
|
|
|
|
|
2016-08-26 15:02:39 -07:00
|
|
|
void setTableBin(float array[], int size, float from, float to);
|
2016-08-26 16:02:56 -07:00
|
|
|
|
|
|
|
#define setArrayValues(array, size, value) setTableBin(array, size, value, value)
|
|
|
|
|
2018-01-07 08:17:49 -08:00
|
|
|
void setLinearCurve(float array[], int size, float from, float to, float precision);
|
2015-07-10 06:01:56 -07:00
|
|
|
void setRpmTableBin(float array[], int size);
|
|
|
|
|
|
|
|
#endif /* TABLE_HELPER_H_ */
|