/** * @file table_helper.h * * @date Jul 6, 2014 * @author Andrey Belomutskiy, (c) 2012-2017 */ #ifndef TABLE_HELPER_H_ #define TABLE_HELPER_H_ #include #include "error_handling.h" #include "interpolation.h" #include "efilib.h" // popular left edge of CLT-based correction curves #define CLT_CURVE_RANGE_FROM -40 /** * this helper class brings together 3D table with two 2D axis curves */ template class Map3D { public: explicit Map3D(const char*name); Map3D(const char*name, float multiplier); void init(vType table[RPM_BIN_SIZE][LOAD_BIN_SIZE], const kType loadBins[LOAD_BIN_SIZE], const kType rpmBins[RPM_BIN_SIZE]); float getValue(float xRpm, float y); void setAll(vType value); vType *pointers[LOAD_BIN_SIZE]; private: void create(const char*name, float multiplier); const kType *loadBins = NULL; const kType *rpmBins = NULL; bool initialized = false; const char *name; float multiplier; }; /* * this dead code is a questionable performance optimization idea: instead of division every time * we want interpolation for a curve we can pre-calculate A and B and save the division at the cost of more RAM usage * Realistically we probably value RAM over CPU at this time and the costs are not justified. template class Table2D { public: Table2D(); void preCalc(float *bin, float *values); float aTable[SIZE]; float bTable[SIZE]; float *bin; }; template Table2D::Table2D() { bin = NULL; } template void Table2D::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) { warning(CUSTOM_INTEPOLATE_ERROR_4, "preCalc: Same x1 and x2 in interpolate: %.2f/%.2f", x1, x2); return; } float y1 = values[i]; float y2 = values[i + 1]; aTable[i] = INTERPOLATION_A(x1, y1, x2, y2); bTable[i] = y1 - aTable[i] * x1; } } */ template void Map3D::init(vType table[RPM_BIN_SIZE][LOAD_BIN_SIZE], const kType loadBins[LOAD_BIN_SIZE], const kType rpmBins[RPM_BIN_SIZE]) { // 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 for (int k = 0; k < LOAD_BIN_SIZE; k++) { pointers[k] = table[k]; } initialized = true; this->loadBins = loadBins; this->rpmBins = rpmBins; } template float Map3D::getValue(float xRpm, float y) { efiAssert(CUSTOM_ERR_ASSERT, initialized, "map not initialized", NAN); if (cisnan(y)) { warning(CUSTOM_PARAM_RANGE, "%s: y is NaN", name); return NAN; } // todo: we have a bit of a mess: in TunerStudio, RPM is X-axis return multiplier * interpolate3d(y, loadBins, LOAD_BIN_SIZE, xRpm, rpmBins, RPM_BIN_SIZE, pointers); } template Map3D::Map3D(const char *name) { create(name, 1); } template Map3D::Map3D(const char *name, float multiplier) { create(name, multiplier); } template void Map3D::create(const char *name, float multiplier) { this->name = name; this->multiplier = multiplier; memset(&pointers, 0, sizeof(pointers)); } template void Map3D::setAll(vType value) { efiAssertVoid(CUSTOM_ERR_6573, initialized, "map not initialized"); for (int l = 0; l < LOAD_BIN_SIZE; l++) { for (int r = 0; r < RPM_BIN_SIZE; r++) { pointers[l][r] = value / multiplier; } } } template 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]; } } } /** * AFR value is packed into uint8_t with a multiplier of 10 */ #define AFR_STORAGE_MULT 10 /** * TPS-based Advance value is packed into int16_t with a multiplier of 100 */ #define ADVANCE_TPS_STORAGE_MULT 100 typedef Map3D afr_Map3D_t; typedef Map3D ign_Map3D_t; typedef Map3D ign_tps_Map3D_t; typedef Map3D fuel_Map3D_t; typedef Map3D baroCorr_Map3D_t; void setRpmBin(float array[], int size, float idleRpm, float topRpm); void setTableBin(float array[], int size, float from, float to); #define setArrayValues(array, size, value) setTableBin(array, size, value, value) /** * @param precision for example '0.1' for one digit fractional part */ template void setLinearCurveAny(vType array[], int size, float from, float to, float precision) { for (int i = 0; i < size; i++) { float value = interpolateMsg("setLinearCurve", 0, from, size - 1, to, i); /** * rounded values look nicer, also we want to avoid precision mismatch with Tuner Studio */ array[i] = efiRound(value, precision); } } #define setLinearCurve setLinearCurveAny void setRpmTableBin(float array[], int size); #endif /* TABLE_HELPER_H_ */