## The content below is mostly from 2018 and is in the process of being updated. This wiki article needs to be viewed as a work in progress at this time and may contain contradictory information.
Flexible Logic is the fanciest-schmanciest feature of rusEfi. Using the FSIO a user can quickly add a logical function directly into the program of the ECU.
Flexible Logic allows advanced users to get unprecedented level of configurability for custom outputs and engine control. rusEfi supports up to 16 flexible outputs, each of these could be either an on/off or PWM signal. In case of a PWM signal the frequency is defined in the configuration and the duty cycle is dynamically controlled by the evaluated expression.
Because the FSIO is like coding a new function directly into the ECU from TunerStudio (or console) it comes with an additional difficulty step and that is understanding "reverse Polish Notation"
This is a form of "postfix" mathematical notation that is particularly suited to computer use and allows a processor to work directly with an equation.
If differs from the common mathematical notation as instead of having the operators (+ - < * ) between the numbers it puts all of them at the end of the expression.
At first look this is very confusing to read and understand but it is actually quite simple and very effective.
**The two videos below explain this very well and we strongly recommend watching them.**
https://www.youtube.com/watch?v=7ha78yWRDlE
https://www.youtube.com/watch?v=TrfcJCulsF4
The TLDR is that the processor is critical to how this notation works, if a value is a number it loads it onto its "stack" if a value is an operator it performs an operation on the numbers in the stack.
A simple example of how this works is as follows:
A B +
| Stack | Value |
| ------------- |:-------------:|
| + | Operand |
| B | Number |
| A | Number |
How the processor interprets this is: *"Add, Number B, to Number A."*
A B C * +
| Stack | Value |
| ------------- |:-------------:|
| * | Operand |
| C | Number |
| B | Number |
| A | Number |
The Processor hits the * operator first so performs a multiplication on the B and C. *"Multiply, by Number C, Number B"*
The result of B*C would then be put back on the stack, for our example we will call this D. At which point the processor would continue the calculation. "A, B, C and *" have already been taken care of so the result would be as shown below.
As previously shown this would be *"add, Number D, to Number A"*
The clever part of this notation is that it does not need to have parenthesis in order to conform to correct calculation order [See here for deeper understanding of the importance of calculation order in maths](https://en.wikipedia.org/wiki/Order_of_operations)
Taking a simple calculation to prevent a starter button working above cranking speed (as used on the Prometheus ECU):
Standard notation -> RPM <Cranking_RPM
RPN -> RPM Cranking_RPM <
| Stack | Value |
| ------------- |:-------------:|
| < | Operand |
| Cranking_RPM | Number |
| RPM | Number |
Resulting in the processor reading the <operandandperforminga"lessthan"assessmentofCranking_RPMandRPM.
*"Less than check for, Cranking_RPM, being less than RPM"*
RPN is not easy at first but hopefully this short guide and the examples shown below in a similar format will make this as easy as possible while giving a little bit of understanding of how an ECU processes the FSIO.
----
## rusEFI console now has build-in decoder of RPN form.
For example entering the formula below into the command window:
FSIO setting 6 has been configured for an adjustable RPM limiter, this can be configured as shown above, the first number (2000) is the limit that will be active when the signal is on, the second (7000) is the standard limiter when the switch is deactivated.
---
# See [legacy wiki](https://rusefi.com/wiki/index.php?title=Manual:Flexible_Logic) content below.
Fuel pump is already using this mechanism - see https://github.com/rusefi/rusefi/blob/master/firmware/controllers/system_fsio.txt
Here are some commands to try:
eval "2 + 3"
Prints the result of "2 + 3"
eval "max (rpm, 100)"
Prints the result of "max(100, rpm)"
eval "if(1, 200, 100)"
Prints the result of "if(1, 200, 100)"
set_fsio_expression 1 "rpm > fsio_setting(4)"
Turn output #1 hi if rpm is greater than fsio setting #4
set_fsio_output_pin 1 pd11
Tells output #1 to use pin PD11
See https://github.com/rusefi/rusefi/blob/master/firmware/controllers/core/fsio_impl.cpp for list of available methods
---
## Contents
1 [More Examples](##More-Examples)
2 [FSIO parameters](##FSIO-parameters)
3 [FSIO digital inputs](##FSIO-digital-inputs)
4 [FSIO analog inputs](##FSIO-analog-inputs)
5 [Case study #1: shift light](##Case-study-#1:-shift-light)
6 [Case study #2: A/C compressor control](##Case-study-#2:-A/C-compressor-control)
7 [Case study #3: digital inputs for extra devices](##Case-study-#3:-digital-inputs-for-extra-devices)
8 [Case study #4: boost controller](##Case-study-#4:-boost-controller)
9 [Case study #5: solenoid exhaust cam actuator](##Case-study-#5:-solenoid-exhaust-cam-actuator)
Now, let's assume we want duty cycle to be 10% if rpm is below 1000, 90% if rpm is above 5000, and have it grow linearly from 10% to 90% between 1000 rpm and 5000 rpm. The human-readable expression for that would be
FSIO expressions could be customized using 16 user-defined 'settings' which are stored within the configuration. To change a setting from dev console your command is set_fsio_setting INDEX VAlUE
To access for example setting #3 your FSIO expression is
fsio_setting(3)
---
## FSIO digital inputs
These feature allows you to pull toggle switches state from your fsio expressions.
work in progress
---
## FSIO analog inputs
As of Aug 1, 2018 only one analog input is implemented:
where input would be assigned via FSIO analog input #1
---
## Case study #1: shift light
We want to turn on a warning light if RPM is above 4500
In order to set an output, three commands have to be invoked, also user has to decide which FSIO unit between 1 and 16 to use. Separate commands are used to set duty cycle, output pin, and condition. All three commands referencing the same FSIO index. This is poor usability and one day someone would improve this somehow.
First we need to configure user output, let's use pin PE5 and index #3 for this example:
set_fsio_output_frequency 3 0
this would disable PWM on channel #3 and set this port to simple digital output mode
set_fsio_output_pin 3 PE5
Now we will set FSIO expression to control this output pin:
Say we want to use PE0 as A/C toggle button, and we want A/C compressor logic to be 'if (rpm > 1200 AND A/C button is depressed) then output 200Hz @ 80% duty cycle on PB1 else output nothing on PB1'.
First we need to configure A/C button input pin which is currently implemented as analog input (that's a temporary hack) - see engineConfiguration->acSwitchAdc
As for the magic expression, that's RPN for "fsio_table (3, rpm, map) / 100".
---
## Case study #5: solenoid exhaust cam actuator
Let's assume we want to turn on 200Hz solenoid at 80% duty cycle if RPM is above 6000.
Here is how this would look in Tuner Studio:
That's RPN notation for "(rpm > 6000) * 0.8"
See https://github.com/rusefi/rusefi/blob/master/firmware/controllers/system_fsio.txt see https://github.com/rusefi/rusefi/blob/master/firmware/controllers/system_fsio.h
We have a usability feature with microRusEfi where stm32 pin names are hidden from users. Unfortunately for FSIO these smt32 pin names are needed. Those are visible in https://github.com/rusefi/rusefi/blob/master/firmware/config/boards/microrusefi/mapping.yaml