custom-board-bundle-sample-.../firmware/util/math/interpolation.cpp

276 lines
7.0 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file interpolation.cpp
* @brief Linear interpolation algorithms
*
2018-08-19 07:53:25 -07:00
* See test_interpolation_3d.cpp
*
*
2015-07-10 06:01:56 -07:00
* @date Oct 17, 2013
2018-01-20 17:55:31 -08:00
* @author Andrey Belomutskiy, (c) 2012-2018
2015-07-10 06:01:56 -07:00
* @author Dmitry Sidin, (c) 2015
*/
2018-09-16 19:26:57 -07:00
#include "global.h"
2015-07-10 06:01:56 -07:00
2015-07-10 06:01:56 -07:00
#include "interpolation.h"
2016-06-29 22:01:38 -07:00
bool needInterpolationLoggingValue = true;
2016-06-29 20:01:53 -07:00
int needInterpolationLogging(void) {
2016-06-29 22:01:38 -07:00
return needInterpolationLoggingValue;
2016-06-29 20:01:53 -07:00
}
2015-07-10 06:01:56 -07:00
#define BINARY_PERF true
Logging * logger;
2016-06-29 17:02:00 -07:00
#if BINARY_PERF && ! EFI_UNIT_TEST
2015-07-10 06:01:56 -07:00
#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;
{
2019-05-07 16:32:08 -07:00
uint32_t start = getTimeNowLowerNt();
2015-07-10 06:01:56 -07:00
int temp = 0;
for (int i = 0; i < COUNT; i++) {
temp += findIndex(array16, size16, v);
}
2019-05-07 16:32:08 -07:00
timeOld = getTimeNowLowerNt() - start;
2015-07-10 06:01:56 -07:00
}
uint32_t timeNew;
{
2019-05-07 16:32:08 -07:00
uint32_t start = getTimeNowLowerNt();
2015-07-10 06:01:56 -07:00
int temp = 0;
for (int i = 0; i < COUNT; i++) {
temp += findIndex2(array16, size16, v);
}
2019-05-07 16:32:08 -07:00
timeNew = getTimeNowLowerNt() - start;
2015-07-10 06:01:56 -07:00
}
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) {
2018-01-23 09:05:14 -08:00
firmwareError(CUSTOM_ERR_INTERPOLATE, "init: Same x1 and x2 in interpolate: %.2f/%.2f", x1, x2);
2015-07-10 06:01:56 -07:00
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
*
2018-06-12 02:45:11 -07:00
* @note For example, "interpolateMsg("", engineConfiguration.tpsMin, 0, engineConfiguration.tpsMax, 100, adc);"
2015-07-10 06:01:56 -07:00
*/
2016-01-09 12:01:41 -08:00
float interpolateMsg(const char *msg, float x1, float y1, float x2, float y2, float x) {
2015-07-10 06:01:56 -07:00
// todo: double comparison using EPS
if (x1 == x2) {
/**
* we could end up here for example while resetting bins while changing engine type
*/
2018-01-23 09:05:14 -08:00
warning(CUSTOM_INTEPOLATE_ERROR, "interpolate%s: Same x1 and x2 in interpolate: %.2f/%.2f", msg, x1, x2);
2015-07-10 06:01:56 -07:00
return NAN;
}
// a*x1 + b = y1
// a*x2 + b = y2
2018-07-25 20:03:04 -07:00
// efiAssertVoid(CUSTOM_ERR_ASSERT_VOID, x1 != x2, "no way we can interpolate");
2015-07-10 06:01:56 -07:00
float a = INTERPOLATION_A(x1, y1, x2, y2);
float b = y1 - a * x1;
float result = a * x + b;
#if DEBUG_FUEL
2018-01-23 09:05:14 -08:00
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);
2015-07-10 06:01:56 -07:00
#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;
}
2017-05-28 21:02:22 -07:00
/**
* Another implementation, which one is faster?
*/
2015-07-10 06:01:56 -07:00
int findIndex2(const float array[], unsigned size, float value) {
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(value), "NaN in findIndex2", 0);
efiAssert(CUSTOM_ERR_ASSERT, size > 1, "size in findIndex", 0);
2015-07-10 06:01:56 -07:00
// 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
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, b + b == size, "Size not power of 2", -1);
2015-07-10 06:01:56 -07:00
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
*/
2018-07-25 20:30:00 -07:00
// efiAssert(CUSTOM_ERR_ASSERT, j < size, "size", 0);
2015-07-10 06:01:56 -07:00
if (array[j] <= value)
i = j;
}
return i || *array <= value ? i : -1;
}
2017-06-06 20:11:29 -07:00
/**
* 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]) {
2017-12-03 15:18:50 -08:00
// todo: this should become a warning under https://github.com/rusefi/rusefi/issues/440
2018-01-23 09:05:14 -08:00
firmwareError(CUSTOM_ERR_AXIS_ORDER, "invalid axis %s at %.2f", msg, array[i]);
2017-06-06 20:11:29 -07:00
}
}
}
2015-07-10 06:01:56 -07:00
/** @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.
2017-06-12 15:48:55 -07:00
*
* See also ensureArrayIsAscending
2015-07-10 06:01:56 -07:00
*/
2017-05-28 21:02:22 -07:00
int findIndexMsg(const char *msg, const float array[], int size, float value) {
if (cisnan(value)) {
2017-05-29 16:23:15 -07:00
firmwareError(ERROR_NAN_FIND_INDEX, "NaN in findIndex%s", msg);
2017-05-28 21:02:22 -07:00
return 0;
}
2015-07-10 06:01:56 -07:00
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--;
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, size > 0, "Unexpected state in binary search", 0);
2015-07-10 06:01:56 -07:00
#endif
// todo: compare current implementation with
// http://eigenjoy.com/2011/01/21/worlds-fastest-binary-search/
// ?
middle = (left + right) / 2;
2018-01-23 09:05:14 -08:00
// print("left=%d middle=%d right=%d: %.2f\r\n", left, middle, right, array[middle]);
2015-07-10 06:01:56 -07:00
if (middle == left)
break;
2018-01-21 09:28:28 -08:00
if (middle != 0 && array[middle - 1] > array[middle]) {
2019-04-12 19:10:57 -07:00
#if EFI_UNIT_TEST
2018-09-10 19:10:55 -07:00
firmwareError(CUSTOM_ERR_6610, "%s: out of order %.2f %.2f", msg, array[middle - 1], array[middle]);
2018-01-21 09:28:28 -08:00
#else
2018-09-10 19:00:13 -07:00
warning(CUSTOM_ERR_OUT_OF_ORDER, "%s: out of order %.2f %.2f", msg, array[middle - 1], array[middle]);
2018-01-21 09:28:28 -08:00
#endif /* EFI_UNIT_TEST */
}
2015-07-10 06:01:56 -07:00
if (value < array[middle]) {
right = middle;
} else if (value > array[middle]) {
left = middle;
} else {
break;
}
}
return middle;
}
2017-05-28 21:02:22 -07:00
int findIndex(const float array[], int size, float value) {
return findIndexMsg("", array, size, value);
}
2015-07-10 06:01:56 -07:00
/**
* @brief One-dimensional table lookup with linear interpolation
2019-05-25 10:36:29 -07:00
*
* @see setLinearCurve()
2015-07-10 06:01:56 -07:00
*/
2019-01-16 05:24:37 -08:00
float interpolate2d(const char *msg, float value, const float bin[], const float values[], int size) {
2017-05-29 08:31:07 -07:00
if (isnan(value)) {
2017-12-08 17:04:58 -08:00
firmwareError(CUSTOM_INTERPOLATE_NAN, "NaN in interpolate2d %s", msg);
2017-05-29 08:31:07 -07:00
return NAN;
}
2017-05-29 08:09:14 -07:00
int index = findIndexMsg("value", bin, size, value);
2015-07-10 06:01:56 -07:00
if (index == -1)
return values[0];
if (index == size - 1)
return values[size - 1];
2016-01-09 12:01:41 -08:00
return interpolateMsg("2d", bin[index], values[index], bin[index + 1], values[index + 1], value);
2015-07-10 06:01:56 -07:00
}
2017-06-11 12:06:05 -07:00
/**
* Sets specified value for specified key in a correction curve
2018-01-07 08:23:28 -08:00
* see also setLinearCurve()
2017-06-11 12:06:05 -07:00
*/
void setCurveValue(float bins[], float values[], int size, float key, float value) {
2017-05-29 08:09:14 -07:00
int index = findIndexMsg("tbVl", bins, size, key);
2015-07-10 06:01:56 -07:00
if (index == -1)
index = 0;
values[index] = value;
}
void initInterpolation(Logging *sharedLogger) {
logger = sharedLogger;
#if BINARY_PERF && ! EFI_UNIT_TEST
addConsoleAction("binarytest", testBinary);
#endif
}