/** * @file interpolation.cpp * @brief Linear interpolation algorithms * * See test_interpolation_3d.cpp * * * @date Oct 17, 2013 * @author Andrey Belomutskiy, (c) 2012-2018 * @author Dmitry Sidin, (c) 2015 */ #include "main.h" #if DEBUG_FUEL #include #endif #include #include "efilib2.h" #include "interpolation.h" bool needInterpolationLoggingValue = true; int needInterpolationLogging(void) { return needInterpolationLoggingValue; } #define BINARY_PERF true Logging * logger; #if BINARY_PERF && ! EFI_UNIT_TEST #define COUNT 10000 float array16[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 }; static void testBinary(void) { const int size16 = 16; uint32_t totalOld = 0; uint32_t totalNew = 0; for (int v = 0; v <= 16; v++) { uint32_t timeOld; { uint32_t start = GET_TIMESTAMP(); int temp = 0; for (int i = 0; i < COUNT; i++) { temp += findIndex(array16, size16, v); } timeOld = GET_TIMESTAMP() - start; } uint32_t timeNew; { uint32_t start = GET_TIMESTAMP(); int temp = 0; for (int i = 0; i < COUNT; i++) { temp += findIndex2(array16, size16, v); } timeNew = GET_TIMESTAMP() - start; } scheduleMsg(logger, "for v=%d old=%d ticks", v, timeOld); scheduleMsg(logger, "for v=%d new=%d ticks", v, timeNew); totalOld += timeOld; totalNew += timeNew; } scheduleMsg(logger, "totalOld=%d ticks", totalOld); scheduleMsg(logger, "totalNew=%d ticks", totalNew); } #endif FastInterpolation::FastInterpolation() { init(0, 0, 1, 1); } FastInterpolation::FastInterpolation(float x1, float y1, float x2, float y2) { init(x1, y1, x2, y2); } void FastInterpolation::init(float x1, float y1, float x2, float y2) { if (x1 == x2) { firmwareError(CUSTOM_ERR_INTERPOLATE, "init: Same x1 and x2 in interpolate: %.2f/%.2f", x1, x2); return; } a = INTERPOLATION_A(x1, y1, x2, y2); b = y1 - a * x1; } float FastInterpolation::getValue(float x) { return a * x + b; } /** @brief Linear interpolation by two points * * @param x1 key of the first point * @param y1 value of the first point * @param x2 key of the second point * @param y2 value of the second point * @param X key to be interpolated * * @note For example, "interpolateMsg("", engineConfiguration.tpsMin, 0, engineConfiguration.tpsMax, 100, adc);" */ float interpolateMsg(const char *msg, float x1, float y1, float x2, float y2, float x) { // todo: double comparison using EPS if (x1 == x2) { /** * we could end up here for example while resetting bins while changing engine type */ warning(CUSTOM_INTEPOLATE_ERROR, "interpolate%s: Same x1 and x2 in interpolate: %.2f/%.2f", msg, x1, x2); return NAN; } // a*x1 + b = y1 // a*x2 + b = y2 // efiAssertVoid(CUSTOM_ERR_ASSERT_VOID, x1 != x2, "no way we can interpolate"); float a = INTERPOLATION_A(x1, y1, x2, y2); float b = y1 - a * x1; float result = a * x + b; #if DEBUG_FUEL printf("x1=%.2f y1=%.2f x2=%.2f y2=%.2f\r\n", x1, y1, x2, y2); printf("a=%.2f b=%.2f result=%.2f\r\n", a, b, result); #endif return result; } float interpolateClamped(float x1, float y1, float x2, float y2, float x) { if (x <= x1) return y1; if (x >= x2) return y2; float a = INTERPOLATION_A(x1, y1, x2, y2); float b = y1 - a * x1; return a * x + b; } /** * Another implementation, which one is faster? */ int findIndex2(const float array[], unsigned size, float value) { efiAssert(CUSTOM_ERR_ASSERT, !cisnan(value), "NaN in findIndex2", 0); efiAssert(CUSTOM_ERR_ASSERT, size > 1, "size in findIndex", 0); // if (size <= 1) // return size && *array <= value ? 0 : -1; signed i = 0; //unsigned b = 1 << int(log(float(size) - 1) / 0.69314718055994530942); unsigned b = size >> 1; // in our case size is always a power of 2 efiAssert(CUSTOM_ERR_ASSERT, b + b == size, "Size not power of 2", -1); for (; b; b >>= 1) { unsigned j = i | b; /** * it should be * "if (j < size && array[j] <= value)" * but in our case size is always power of 2 thus size is always more then j */ // efiAssert(CUSTOM_ERR_ASSERT, j < size, "size", 0); if (array[j] <= value) i = j; } return i || *array <= value ? i : -1; } /** * in order to use binary search we need to know that axis elements are sorted */ void ensureArrayIsAscending(const char *msg, const float array[], int size) { for (int i = 0; i < size - 1; i ++) { if (array[i] >= array[i+ 1]) { // todo: this should become a warning under https://github.com/rusefi/rusefi/issues/440 firmwareError(CUSTOM_ERR_AXIS_ORDER, "invalid axis %s at %.2f", msg, array[i]); } } } /** @brief Binary search * @returns the highest index within sorted array such that array[i] is greater than or equal to the parameter * @note If the parameter is smaller than the first element of the array, -1 is returned. * * See also ensureArrayIsAscending */ int findIndexMsg(const char *msg, const float array[], int size, float value) { if (cisnan(value)) { firmwareError(ERROR_NAN_FIND_INDEX, "NaN in findIndex%s", msg); return 0; } if (value < array[0]) return -1; int middle; int left = 0; int right = size; // todo: extract binary search as template method? while (true) { #if 0 // that's an assertion to make sure we do not loop here size--; efiAssert(CUSTOM_ERR_ASSERT, size > 0, "Unexpected state in binary search", 0); #endif // todo: compare current implementation with // http://eigenjoy.com/2011/01/21/worlds-fastest-binary-search/ // ? middle = (left + right) / 2; // print("left=%d middle=%d right=%d: %.2f\r\n", left, middle, right, array[middle]); if (middle == left) break; if (middle != 0 && array[middle - 1] > array[middle]) { #if EFI_UNIT_TEST || defined(__DOXYGEN__) firmwareError(CUSTOM_ERR_6147, "%s: out of order %.2f %.2f", msg, array[middle - 1], array[middle]); #else warning(CUSTOM_ERR_6147, "%s: out of order %.2f %.2f", msg, array[middle - 1], array[middle]); #endif /* EFI_UNIT_TEST */ } if (value < array[middle]) { right = middle; } else if (value > array[middle]) { left = middle; } else { break; } } return middle; } int findIndex(const float array[], int size, float value) { return findIndexMsg("", array, size, value); } /** * @brief One-dimensional table lookup with linear interpolation */ float interpolate2d(const char *msg, float value, float bin[], float values[], int size) { if (isnan(value)) { firmwareError(CUSTOM_INTERPOLATE_NAN, "NaN in interpolate2d %s", msg); return NAN; } int index = findIndexMsg("value", bin, size, value); if (index == -1) return values[0]; if (index == size - 1) return values[size - 1]; return interpolateMsg("2d", bin[index], values[index], bin[index + 1], values[index + 1], value); } /** * Sets specified value for specified key in a correction curve * see also setLinearCurve() */ void setCurveValue(float bins[], float values[], int size, float key, float value) { int index = findIndexMsg("tbVl", bins, size, key); if (index == -1) index = 0; values[index] = value; } void initInterpolation(Logging *sharedLogger) { logger = sharedLogger; #if BINARY_PERF && ! EFI_UNIT_TEST addConsoleAction("binarytest", testBinary); #endif }