2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file efilib.cpp
|
|
|
|
*
|
|
|
|
* We cannot use stdlib because we do not have malloc - so, we have to implement these functions
|
|
|
|
*
|
|
|
|
* @date Feb 21, 2014
|
2020-01-13 18:57:43 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#include <string.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include "datalogging.h"
|
|
|
|
#include "histogram.h"
|
|
|
|
|
2022-03-19 16:33:37 -07:00
|
|
|
// also known as bool2string and boolean2string
|
2015-07-10 06:01:56 -07:00
|
|
|
const char * boolToString(bool value) {
|
|
|
|
return value ? "Yes" : "No";
|
|
|
|
}
|
|
|
|
|
2017-12-10 07:19:05 -08:00
|
|
|
/*
|
2017-12-10 12:05:58 -08:00
|
|
|
float efiFloor(float value, float precision) {
|
2015-07-10 06:01:56 -07:00
|
|
|
int a = (int) (value / precision);
|
|
|
|
return a * precision;
|
|
|
|
}
|
2017-12-10 07:19:05 -08:00
|
|
|
*/
|
|
|
|
|
2018-01-07 09:11:49 -08:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param precision for example '0.1' for one digit fractional part
|
|
|
|
*/
|
2017-12-10 07:19:05 -08:00
|
|
|
float efiRound(float value, float precision) {
|
2024-07-18 20:57:40 -07:00
|
|
|
efiAssert(ObdCode::CUSTOM_ERR_ASSERT, precision != 0, "Zero precision is not valid for efiRound maybe you mean '1'?", NAN);
|
2020-12-05 21:11:57 -08:00
|
|
|
float a = round(value / precision);
|
2020-06-18 20:29:08 -07:00
|
|
|
return fixNegativeZero(a * precision);
|
2017-12-10 07:19:05 -08:00
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
char * efiTrim(char *param) {
|
|
|
|
while (param[0] == ' ') {
|
|
|
|
param++; // that would skip leading spaces
|
|
|
|
}
|
|
|
|
int len = efiStrlen(param);
|
|
|
|
while (len > 0 && param[len - 1] == ' ') {
|
|
|
|
param[len - 1] = 0;
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
return param;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool startsWith(const char *line, const char *prefix) {
|
|
|
|
uint32_t len = efiStrlen(prefix);
|
|
|
|
if (efiStrlen(line) < len) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (uint32_t i = 0; i < len; i++) {
|
|
|
|
if (line[i] != prefix[i]) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-09-23 05:45:45 -07:00
|
|
|
static char *ltoa_internal(char *p, uint32_t num, unsigned radix) {
|
|
|
|
constexpr int bufferLength = 10;
|
2023-12-30 20:51:59 -08:00
|
|
|
|
2019-09-23 05:45:45 -07:00
|
|
|
char buffer[bufferLength];
|
|
|
|
|
|
|
|
size_t idx = bufferLength - 1;
|
|
|
|
|
|
|
|
// First, we write from right-to-left so that we don't have to compute
|
|
|
|
// log(num)/log(radix)
|
|
|
|
do
|
|
|
|
{
|
|
|
|
auto digit = num % radix;
|
|
|
|
|
|
|
|
// Digits 0-9 -> '0'-'9'
|
|
|
|
// Digits 10-15 -> 'a'-'f'
|
|
|
|
char c = digit < 10
|
|
|
|
? digit + '0'
|
|
|
|
: digit + 'a' - 10;
|
|
|
|
|
|
|
|
// Write this digit in to the buffer
|
|
|
|
buffer[idx] = c;
|
|
|
|
idx--;
|
2015-07-10 06:01:56 -07:00
|
|
|
} while ((num /= radix) != 0);
|
2023-12-30 20:51:59 -08:00
|
|
|
|
2019-09-23 05:45:45 -07:00
|
|
|
idx++;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-09-23 05:45:45 -07:00
|
|
|
// Now, we copy characters in to place in the final buffer
|
|
|
|
while (idx < bufferLength)
|
|
|
|
{
|
|
|
|
*p++ = buffer[idx++];
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2017-03-05 11:17:48 -08:00
|
|
|
/**
|
|
|
|
* @return pointer at the end zero symbol after the digits
|
|
|
|
*/
|
2015-07-10 06:01:56 -07:00
|
|
|
static char* itoa_signed(char *p, int num, unsigned radix) {
|
|
|
|
if (num < 0) {
|
|
|
|
*p++ = '-';
|
|
|
|
char *end = ltoa_internal(p, -num, radix);
|
|
|
|
*end = 0;
|
|
|
|
return end;
|
|
|
|
}
|
|
|
|
char *end = ltoa_internal(p, num, radix);
|
|
|
|
*end = 0;
|
|
|
|
return end;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Integer to string
|
2017-03-05 11:17:48 -08:00
|
|
|
*
|
|
|
|
* @return pointer at the end zero symbol after the digits
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
char* itoa10(char *p, int num) {
|
|
|
|
return itoa_signed(p, num, 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
int efiPow10(int param) {
|
|
|
|
switch (param) {
|
|
|
|
case 0:
|
|
|
|
return 1;
|
|
|
|
case 1:
|
|
|
|
return 10;
|
|
|
|
case 2:
|
|
|
|
return 100;
|
|
|
|
case 3:
|
|
|
|
return 1000;
|
|
|
|
case 4:
|
|
|
|
return 10000;
|
|
|
|
case 5:
|
|
|
|
return 100000;
|
|
|
|
case 6:
|
|
|
|
return 1000000;
|
|
|
|
case 7:
|
|
|
|
return 10000000;
|
|
|
|
case 8:
|
|
|
|
return 100000000;
|
|
|
|
}
|
|
|
|
return 10 * efiPow10(10 - 1);
|
|
|
|
}
|
|
|
|
|
2017-04-04 20:00:21 -07:00
|
|
|
/*
|
|
|
|
** return lower-case of c if upper-case, else c
|
|
|
|
*/
|
|
|
|
int mytolower(const char c) {
|
|
|
|
return TO_LOWER(c);
|
|
|
|
}
|
|
|
|
|
2022-12-02 21:51:41 -08:00
|
|
|
int djb2lowerCase(const char *str) {
|
|
|
|
unsigned long hash = 5381;
|
|
|
|
int c;
|
|
|
|
|
2022-12-02 22:12:31 -08:00
|
|
|
while ( (c = *str++) ) {
|
2022-12-02 21:51:41 -08:00
|
|
|
c = TO_LOWER(c);
|
|
|
|
hash = ((hash << 5) + hash) + c; /* hash * 33 + c */
|
|
|
|
}
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @brief This function knows how to print a histogram_s summary
|
|
|
|
*/
|
|
|
|
void printHistogram(Logging *logging, histogram_s *histogram) {
|
|
|
|
#if EFI_HISTOGRAMS && ! EFI_UNIT_TEST
|
|
|
|
int report[5];
|
|
|
|
int len = hsReport(histogram, report);
|
|
|
|
|
2020-09-13 01:49:25 -07:00
|
|
|
logging->reset();
|
2021-11-19 01:01:45 -08:00
|
|
|
logging.append(PROTOCOL_MSG LOG_DELIMITER);
|
2020-09-13 01:49:25 -07:00
|
|
|
logging.appendPrintf("histogram %s *", histogram->name);
|
2015-07-10 06:01:56 -07:00
|
|
|
for (int i = 0; i < len; i++)
|
2020-09-13 01:49:25 -07:00
|
|
|
logging.appendPrintf("%d ", report[i]);
|
|
|
|
logging.appendPrintf("*");
|
2021-11-19 01:01:45 -08:00
|
|
|
logging.append(LOG_DELIMITER);
|
2015-07-10 06:01:56 -07:00
|
|
|
scheduleLogging(logging);
|
|
|
|
#else
|
|
|
|
UNUSED(logging);
|
|
|
|
UNUSED(histogram);
|
2023-12-30 20:51:59 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif /* EFI_HISTOGRAMS */
|
|
|
|
}
|
|
|
|
|
2018-09-29 09:16:36 -07:00
|
|
|
float limitRateOfChange(float newValue, float oldValue, float incrLimitPerSec, float decrLimitPerSec, float secsPassed) {
|
|
|
|
if (newValue >= oldValue)
|
|
|
|
return (incrLimitPerSec <= 0.0f) ? newValue : oldValue + minF(newValue - oldValue, incrLimitPerSec * secsPassed);
|
|
|
|
return (decrLimitPerSec <= 0.0f) ? newValue : oldValue - minF(oldValue - newValue, decrLimitPerSec * secsPassed);
|
|
|
|
}
|
2022-09-20 03:19:51 -07:00
|
|
|
|
|
|
|
bool isPhaseInRange(float test, float current, float next) {
|
|
|
|
bool afterCurrent = test >= current;
|
|
|
|
bool beforeNext = test < next;
|
|
|
|
|
|
|
|
if (next > current) {
|
|
|
|
// we're not near the end of the cycle, comparison is simple
|
|
|
|
// 0 |------------------------| 720
|
|
|
|
// next current
|
|
|
|
return afterCurrent && beforeNext;
|
|
|
|
} else {
|
|
|
|
// we're near the end of the cycle so we have to check the wraparound
|
|
|
|
// 0 -----------| |------ 720
|
|
|
|
// next current
|
|
|
|
// Check whether test is after current (ie, between current tooth and end of cycle)
|
|
|
|
// or if test if before next (ie, between start of cycle and next tooth)
|
|
|
|
return afterCurrent || beforeNext;
|
|
|
|
}
|
|
|
|
}
|
2023-12-30 20:51:59 -08:00
|
|
|
|
2024-04-07 12:49:02 -07:00
|
|
|
static int getBitRangeCommon(const uint8_t data[], int bitIndex, int bitWidth, int secondByteOffset) {
|
2023-12-30 20:51:59 -08:00
|
|
|
int byteIndex = bitIndex >> 3;
|
|
|
|
int shift = bitIndex - byteIndex * 8;
|
|
|
|
int value = data[byteIndex];
|
|
|
|
if (shift + bitWidth > 8) {
|
2024-04-07 12:49:02 -07:00
|
|
|
value = value + data[secondByteOffset + byteIndex] * 256;
|
2023-12-30 20:51:59 -08:00
|
|
|
}
|
|
|
|
int mask = (1 << bitWidth) - 1;
|
|
|
|
return (value >> shift) & mask;
|
|
|
|
}
|
2024-04-07 12:49:02 -07:00
|
|
|
|
|
|
|
// see also getBitRange in lua_lib.h
|
|
|
|
int getBitRangeLsb(const uint8_t data[], int bitIndex, int bitWidth) {
|
|
|
|
return getBitRangeCommon(data, bitIndex, bitWidth, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// see also getBitRangeMsb in lua_lib.h
|
|
|
|
int getBitRangeMsb(const uint8_t data[], int bitIndex, int bitWidth) {
|
|
|
|
return getBitRangeCommon(data, bitIndex, bitWidth, -1);
|
|
|
|
}
|