rusefi-full/firmware/controllers/core/fsio_core.cpp

464 lines
11 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file fsio_core.cpp
* @brief core FSUI handling logic
*
* Here we parse and evaluate logical expressions in
* http://en.wikipedia.org/wiki/Reverse_Polish_notation
*
* Once the expressions are parsed on startup (that's a heavy operation),
* evaluating those is relatively efficient.
*
*
* @date Oct 3, 2014
2020-01-13 18:57:43 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
2015-07-10 06:01:56 -07:00
*/
2018-09-16 19:26:57 -07:00
#include "global.h"
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_FSIO
#include "os_access.h"
2015-07-10 06:01:56 -07:00
#include "fsio_core.h"
#include "fsio_impl.h"
2019-09-19 22:06:15 -07:00
#include "adc_inputs.h"
2015-07-10 06:01:56 -07:00
2016-06-29 22:01:38 -07:00
extern fsio8_Map3D_f32t fsioTable1;
2016-07-01 20:01:22 -07:00
extern fsio8_Map3D_u8t fsioTable2;
2016-07-05 11:02:26 -07:00
extern fsio8_Map3D_u8t fsioTable3;
extern fsio8_Map3D_u8t fsioTable4;
static fsio8_Map3D_u8t * fsio8t_tables[] = {NULL, NULL, &fsioTable2, &fsioTable3, &fsioTable4};
2017-06-04 12:25:37 -07:00
EXTERN_ENGINE;
2016-04-04 07:01:43 -07:00
2019-10-07 22:26:35 -07:00
LENameOrdinalPair * LE_FIRST = nullptr;
2015-07-10 06:01:56 -07:00
/**
* the main point of these static fields is that their constructor would register
* them in the magic list of operator name/ordinal pairs
*/
static LENameOrdinalPair leAnd(LE_OPERATOR_AND, "and");
static LENameOrdinalPair leAnd2(LE_OPERATOR_AND, "&");
static LENameOrdinalPair leOr(LE_OPERATOR_OR, "or");
static LENameOrdinalPair leOr2(LE_OPERATOR_OR, "|");
static LENameOrdinalPair leNot(LE_OPERATOR_NOT, "not");
2021-02-17 07:53:35 -08:00
static LENameOrdinalPair leNot2(LE_OPERATOR_NOT, "!");
2015-07-10 06:01:56 -07:00
static LENameOrdinalPair leAdd(LE_OPERATOR_ADDITION, "+");
static LENameOrdinalPair leSub(LE_OPERATOR_SUBTRACTION, "-");
static LENameOrdinalPair leMul(LE_OPERATOR_MULTIPLICATION, "*");
static LENameOrdinalPair leDiv(LE_OPERATOR_DIVISION, "/");
static LENameOrdinalPair leMore(LE_OPERATOR_MORE, ">");
static LENameOrdinalPair leMoreOrEqual(LE_OPERATOR_MORE_OR_EQUAL, ">=");
static LENameOrdinalPair leLess(LE_OPERATOR_LESS, "<");
static LENameOrdinalPair leLessOrEquals(LE_OPERATOR_LESS_OR_EQUAL, "<=");
static LENameOrdinalPair leMax(LE_METHOD_MAX, "max");
static LENameOrdinalPair leMin(LE_METHOD_MIN, "min");
static LENameOrdinalPair leIf(LE_METHOD_IF, "if");
2016-03-02 19:02:37 -08:00
static LENameOrdinalPair leSelf(LE_METHOD_SELF, "self");
2015-07-10 06:01:56 -07:00
LENameOrdinalPair::LENameOrdinalPair(le_action_e action, const char *name) {
this->action = action;
this->name = name;
this->next = LE_FIRST;
LE_FIRST = this;
}
LEElement::LEElement() {
2016-04-19 17:03:53 -07:00
clear();
}
void LEElement::clear() {
2015-07-10 06:01:56 -07:00
action = LE_UNDEFINED;
fValue = NAN;
}
void LEElement::init(le_action_e action) {
this->action = action;
}
void LEElement::init(le_action_e action, float fValue) {
this->action = action;
this->fValue = fValue;
}
void LEElement::init(le_action_e action, bool bValue) {
this->action = action;
this->fValue = bValue ? 1 : 0;
}
2015-07-10 06:01:56 -07:00
LECalculator::LECalculator() {
reset();
}
void LECalculator::reset() {
m_program = nullptr;
2015-07-10 06:01:56 -07:00
stack.reset();
currentCalculationLogPosition = 0;
2016-08-30 18:02:38 -07:00
memset(calcLogAction, 0, sizeof(calcLogAction));
2015-07-10 06:01:56 -07:00
}
void LECalculator::reset(LEElement *element) {
reset();
setProgram(element);
2015-07-10 06:01:56 -07:00
}
void LECalculator::setProgram(LEElement* program) {
m_program = program;
2015-07-10 06:01:56 -07:00
}
bool float2bool(float v) {
2015-07-10 06:01:56 -07:00
return v != 0;
}
float LECalculator::pop(le_action_e action) {
if (stack.size() == 0) {
2017-04-19 19:03:14 -07:00
warning(CUSTOM_EMPTY_FSIO_STACK, "empty stack for action=%d", action);
2015-07-10 06:01:56 -07:00
return NAN;
}
return stack.pop();
}
void LECalculator::push(le_action_e action, float value) {
stack.push(value);
if (currentCalculationLogPosition < MAX_CALC_LOG) {
calcLogAction[currentCalculationLogPosition] = action;
calcLogValue[currentCalculationLogPosition] = value;
currentCalculationLogPosition++;
}
}
static FsioResult doBinaryBoolean(le_action_e action, float lhs, float rhs) {
bool v1 = float2bool(lhs);
bool v2 = float2bool(rhs);
switch (action) {
case LE_OPERATOR_AND:
return v1 && v2;
case LE_OPERATOR_OR:
return v1 || v2;
default:
return unexpected;
}
}
static FsioResult doBinaryNumeric(le_action_e action, float v1, float v2) {
// Process based on the action type
switch (action) {
case LE_OPERATOR_ADDITION:
return v1 + v2;
case LE_OPERATOR_SUBTRACTION:
return v1 - v2;
case LE_OPERATOR_MULTIPLICATION:
return v1 * v2;
case LE_OPERATOR_DIVISION:
return v1 / v2;
case LE_OPERATOR_LESS:
return v1 < v2;
case LE_OPERATOR_MORE:
return v1 > v2;
case LE_OPERATOR_LESS_OR_EQUAL:
return v1 <= v2;
case LE_OPERATOR_MORE_OR_EQUAL:
return v1 >= v2;
case LE_METHOD_MIN:
return minF(v1, v2);
case LE_METHOD_MAX:
return maxF(v1, v2);
default:
return unexpected;
}
}
2015-07-10 06:01:56 -07:00
/**
* @return true in case of error, false otherwise
*/
FsioResult LECalculator::processElement(const LEElement *element DECLARE_ENGINE_PARAMETER_SUFFIX) {
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
efiAssert(CUSTOM_ERR_ASSERT, getCurrentRemainingStack() > 64, "FSIO logic", unexpected);
2015-07-10 06:01:56 -07:00
#endif
switch (element->action) {
// Literal values
2015-07-10 06:01:56 -07:00
case LE_NUMERIC_VALUE:
return element->fValue;
case LE_BOOLEAN_VALUE:
return element->fValue != 0;
// Boolean input binary operators
case LE_OPERATOR_AND:
2015-07-10 06:01:56 -07:00
case LE_OPERATOR_OR: {
float v1 = pop(LE_OPERATOR_OR);
float v2 = pop(LE_OPERATOR_OR);
return doBinaryBoolean(element->action, v1, v2);
2015-07-10 06:01:56 -07:00
}
// Numeric input binary operators
case LE_OPERATOR_ADDITION:
case LE_OPERATOR_SUBTRACTION:
case LE_OPERATOR_MULTIPLICATION:
case LE_OPERATOR_DIVISION:
case LE_OPERATOR_LESS:
case LE_OPERATOR_MORE:
case LE_OPERATOR_LESS_OR_EQUAL:
case LE_OPERATOR_MORE_OR_EQUAL:
case LE_METHOD_MIN:
case LE_METHOD_MAX: {
2015-07-10 06:01:56 -07:00
// elements on stack are in reverse order
float v2 = pop(element->action);
float v1 = pop(element->action);
return doBinaryNumeric(element->action, v1, v2);
2015-07-10 06:01:56 -07:00
}
// Boolean input unary operator
2015-07-10 06:01:56 -07:00
case LE_OPERATOR_NOT: {
float v = pop(LE_OPERATOR_NOT);
return !float2bool(v) ? 1 : 0;
2015-07-10 06:01:56 -07:00
}
case LE_METHOD_IF: {
// elements on stack are in reverse order
float vFalse = pop(LE_METHOD_IF);
float vTrue = pop(LE_METHOD_IF);
float vCond = pop(LE_METHOD_IF);
return vCond != 0 ? vTrue : vFalse;
2015-07-10 06:01:56 -07:00
}
case LE_METHOD_FSIO_SETTING: {
2017-03-12 11:47:21 -07:00
float humanIndex = pop(LE_METHOD_FSIO_SETTING);
int index = (int) humanIndex - 1;
2017-06-25 23:14:31 -07:00
if (index >= 0 && index < FSIO_COMMAND_COUNT) {
return CONFIG(fsio_setting)[index];
2015-07-10 06:01:56 -07:00
} else {
return unexpected;
2015-07-10 06:01:56 -07:00
}
2016-03-22 11:03:44 -07:00
}
case LE_METHOD_FSIO_TABLE: {
float i = pop(LE_METHOD_FSIO_TABLE);
float yValue = pop(LE_METHOD_FSIO_TABLE);
2016-04-04 07:01:43 -07:00
float xValue = pop(LE_METHOD_FSIO_TABLE);
2016-03-22 11:03:44 -07:00
int index = (int) i;
2016-07-05 11:02:26 -07:00
if (index < 1 || index > MAX_TABLE_INDEX) {
return unexpected;
2016-03-22 11:03:44 -07:00
} else {
2016-06-29 22:01:38 -07:00
if (index == 1) {
fsio8_Map3D_f32t *t = &fsioTable1;
2016-03-22 11:03:44 -07:00
return t->getValue(xValue, yValue);
2016-06-29 22:01:38 -07:00
} else {
2016-07-05 11:02:26 -07:00
fsio8_Map3D_u8t *t = fsio8t_tables[index];
2016-06-29 22:01:38 -07:00
return t->getValue(xValue, yValue);
2016-06-29 22:01:38 -07:00
}
2016-03-22 11:03:44 -07:00
}
2015-07-10 06:01:56 -07:00
}
2018-08-01 20:27:22 -07:00
case LE_METHOD_FSIO_DIGITAL_INPUT:
2020-06-01 19:39:56 -07:00
// todo: implement code for digital input!!!
return unexpected;
2016-10-22 21:03:08 -07:00
case LE_METHOD_FSIO_ANALOG_INPUT:
2020-06-01 19:39:56 -07:00
{
int index = clampF(0, pop(LE_METHOD_FSIO_ANALOG_INPUT), FSIO_ANALOG_INPUT_COUNT - 1);
return getVoltage("fsio", engineConfiguration->fsioAdc[index] PASS_ENGINE_PARAMETER_SUFFIX);
2020-06-01 19:39:56 -07:00
}
2015-07-10 06:01:56 -07:00
case LE_METHOD_KNOCK:
return ENGINE(knockCount);
2015-07-10 06:01:56 -07:00
case LE_UNDEFINED:
2017-04-19 19:03:14 -07:00
warning(CUSTOM_UNKNOWN_FSIO, "FSIO undefined action");
return unexpected;
2015-07-10 06:01:56 -07:00
default:
return getEngineValue(element->action PASS_ENGINE_PARAMETER_SUFFIX);
2015-07-10 06:01:56 -07:00
}
}
2017-06-04 12:25:37 -07:00
float LECalculator::getValue2(float selfValue, LEElement *fistElementInList DECLARE_ENGINE_PARAMETER_SUFFIX) {
2016-03-02 19:02:37 -08:00
reset(fistElementInList);
2017-06-04 12:25:37 -07:00
return getValue(selfValue PASS_ENGINE_PARAMETER_SUFFIX);
2015-07-10 06:01:56 -07:00
}
bool LECalculator::isEmpty() const {
return !m_program;
2016-10-01 06:02:04 -07:00
}
2017-06-04 12:25:37 -07:00
float LECalculator::getValue(float selfValue DECLARE_ENGINE_PARAMETER_SUFFIX) {
2016-10-01 06:02:04 -07:00
if (isEmpty()) {
warning(CUSTOM_NO_FSIO, "no FSIO code");
2015-07-10 06:01:56 -07:00
return NAN;
}
const LEElement* element = m_program;
2015-07-10 06:01:56 -07:00
stack.reset();
int counter = 0;
// while not a return statement, execute instructions
while (element->action != LE_METHOD_RETURN) {
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, counter < 200, "FSIOcount", NAN); // just in case
2015-07-10 06:01:56 -07:00
2016-03-02 19:02:37 -08:00
if (element->action == LE_METHOD_SELF) {
push(element->action, selfValue);
} else {
FsioResult result = processElement(element PASS_ENGINE_PARAMETER_SUFFIX);
if (!result) {
2016-03-02 19:02:37 -08:00
// error already reported
return NAN;
}
push(element->action, result.Value);
2015-07-10 06:01:56 -07:00
}
// Step forward to the next instruction in sequence
element++;
2015-07-10 06:01:56 -07:00
counter++;
}
// The stack should have exactly one element on it
2015-07-10 06:01:56 -07:00
if (stack.size() != 1) {
warning(CUSTOM_FSIO_STACK_SIZE, "unexpected FSIO stack size at return: %d", stack.size());
2015-07-10 06:01:56 -07:00
return NAN;
}
return stack.pop();
}
LEElementPool::LEElementPool(LEElement *pool, int size) {
this->m_pool = pool;
2015-07-10 06:01:56 -07:00
this->size = size;
reset();
}
void LEElementPool::reset() {
// Next free element is the first one
m_nextFree = m_pool;
2015-07-10 06:01:56 -07:00
}
int LEElementPool::getSize() const {
return m_nextFree - m_pool;
2015-07-10 06:01:56 -07:00
}
bool isNumeric(const char* line) {
return line[0] >= '0' && line[0] <= '9';
}
bool isBoolean(const char* line) {
bool isTrue = 0 == strcmp(line, "true");
bool isFalse = 0 == strcmp(line, "false");
return isTrue || isFalse;
}
2015-07-10 06:01:56 -07:00
/**
* @return pointer at the position after the consumed token, null if no token consumed
*/
const char *getNextToken(const char *line, char *buffer, const int bufferSize) {
while (line[0] != 0 && line[0] == ' ') {
line++;
}
if (line[0] == 0) {
return NULL;
}
int tokenLen = indexOf(line, ' ');
if (tokenLen == -1) {
// no space - the whole remaining line is the token
tokenLen = strlen(line);
}
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, tokenLen < bufferSize, "token overflow", NULL);
2015-07-10 06:01:56 -07:00
strncpy(buffer, line, tokenLen);
buffer[tokenLen] = 0;
line += tokenLen;
return line;
}
le_action_e parseAction(const char * line) {
LENameOrdinalPair *pair = LE_FIRST;
while (pair != NULL) {
if (strEqualCaseInsensitive(pair->name, line)) {
return pair->action;
}
pair = pair->next;
}
return LE_UNDEFINED;
}
static char parsingBuffer[64];
LEElement* LEElementPool::parseExpression(const char * line) {
LEElement* expressionHead = m_nextFree;
LEElement* n = expressionHead;
2015-07-10 06:01:56 -07:00
while (true) {
line = getNextToken(line, parsingBuffer, sizeof(parsingBuffer));
if (!line) {
2015-07-10 06:01:56 -07:00
/**
* No more tokens in this line, parsing complete!
2015-07-10 06:01:56 -07:00
*/
// Push a return statement on the end
n->init(LE_METHOD_RETURN);
// The next available element is the one after the return
m_nextFree = n + 1;
return expressionHead;
2015-07-10 06:01:56 -07:00
}
if (isNumeric(parsingBuffer)) {
n->init(LE_NUMERIC_VALUE, atoff(parsingBuffer));
} else if (isBoolean(parsingBuffer)) {
n->init(LE_BOOLEAN_VALUE, parsingBuffer[0] == 't');
2015-07-10 06:01:56 -07:00
} else {
le_action_e action = parseAction(parsingBuffer);
if (action == LE_UNDEFINED) {
/**
* Cannot recognize token
*/
firmwareError(CUSTOM_ERR_PARSING_ERROR, "unrecognized FSIO keyword [%s]", parsingBuffer);
2020-01-20 22:47:58 -08:00
return nullptr;
2015-07-10 06:01:56 -07:00
}
n->init(action);
}
n++;
2015-07-10 06:01:56 -07:00
}
}
FsioValue::FsioValue(float f)
{
u.f32 = f;
// The low bit represents whether this is a bool or not, clear it for float
u.u32 &= 0xFFFFFFFE;
}
FsioValue::FsioValue(bool b)
{
u.u32 = (b ? 2 : 0); // second bit is the actual value of the bool
// Low bit indicates this is a bool
u.u32 |= 0x1;
}
bool FsioValue::isFloat() const {
uint32_t typeBit = u.u32 & 0x1;
return typeBit == 0;
}
float FsioValue::asFloat() const {
return u.f32;
}
bool FsioValue::isBool() const {
uint32_t typeBit = u.u32 & 0x1;
return typeBit == 1;
}
bool FsioValue::asBool() const {
uint32_t boolBit = u.u32 & 0x2;
return boolBit != 0;
}
2015-07-10 06:01:56 -07:00
#endif /* EFI_FSIO */