183 lines
6.7 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/FastMap/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
[![Arduino-lint](https://github.com/RobTillaart/FastMap/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/FastMap/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/FastMap/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/FastMap/actions/workflows/jsoncheck.yml)
2023-10-31 11:37:49 +01:00
[![GitHub issues](https://img.shields.io/github/issues/RobTillaart/FastMap.svg)](https://github.com/RobTillaart/FastMap/issues)
2021-01-29 12:31:58 +01:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/FastMap/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/FastMap.svg?maxAge=3600)](https://github.com/RobTillaart/FastMap/releases)
2023-10-31 11:37:49 +01:00
[![PlatformIO Registry](https://badges.registry.platformio.org/packages/robtillaart/library/FastMap.svg)](https://registry.platformio.org/libraries/robtillaart/FastMap)
2021-01-29 12:31:58 +01:00
2020-11-27 11:16:22 +01:00
# FastMap
2020-04-16 12:59:37 +02:00
Fast mapping and constraining.
2020-11-27 11:16:22 +01:00
## Description
FastMap is an object that pre-calculates (internal) floats to make a mapping function especially for floats.
2022-11-05 10:57:25 +01:00
The FastMap also provides a **back()** function to reverse the mapping.
This only works well with floats, and less with integers, so use with care.
2020-11-27 11:16:22 +01:00
An important difference with the traditional **map()** function is that both **init()** and **map()**
2022-11-05 10:57:25 +01:00
accepts floats as parameters.
This allows mapping that would be hard to achieve with the normal **map()** function.
2020-11-27 11:16:22 +01:00
2022-11-05 10:57:25 +01:00
Since 0.4.0 the **init()** function will not accept zero range defining input or output parameters.
2023-10-31 11:37:49 +01:00
2020-11-27 11:16:22 +01:00
## Performance notes
(based upon tests https://github.com/RobTillaart/FastMap/issues/4 )
2022-11-05 10:57:25 +01:00
- On AVR (UNO and MEGA) no gain is made mapping integers with fastMap, 130% slower = substantial.
- On AVR the gain for float is limited, 10% faster.
- On ESP32 the gain for integers and float is both in the order of 25%.
2020-11-27 11:16:22 +01:00
To see the actual gain in your project on your hardware you should test and compare.
FastMap is faster when mapping floats as it uses less float operations than the standard map formula does.
2022-11-05 10:57:25 +01:00
The performance results from pre-calculating values in the **init()** function.
An actual mapping therefore needs only one multiply and one add operation where the standard **map()** function
uses four adds, a multiplication and a division.
The pre-calculation in **init()** should be taken in account and if every **map()** call needs an **init()**
2020-11-27 11:16:22 +01:00
there will be no gain, on contrary.
2022-11-05 10:57:25 +01:00
## Precision notes
The implementation of **fastMap()** uses floats (typical 32 bits) which might result in more memory usage
2022-11-05 10:57:25 +01:00
and loss of precision for mapping of larger values, especially 32 and 64 bit integers.
This is caused by the limits of the mantissa (~23 bits) of the standard 4 byte float.
2022-11-05 10:57:25 +01:00
To solve this a **FastMapDouble** class is added which uses the **double** type for the platforms
that support 8 byte floats.
If your platform does not support double it will often be mapped to float, so no gain.
Furthermore using double might imply a performance penalty on some platforms.
2020-11-27 11:16:22 +01:00
## Interface
2023-10-31 11:37:49 +01:00
```cpp
#include "FastMap.h"
```
### Base
- **FastMap()** Constructor
2022-11-05 10:57:25 +01:00
- **bool init(float in_min, float in_max, float out_min, float out_max)** defines the linear mapping parameters.
2020-11-27 11:16:22 +01:00
The **init()** function calculates all needed values for the **map()**, the **back()** call and the **constrainXX()** functions.
2022-11-05 10:57:25 +01:00
The **init()** function can be called again with new values when needed to do other mapping,
although it will give less overhead if you create an fastMap object per conversion needed.
Returns false if (out_max == out_min) or (in_max == in_min). (breaking change in 0.4.0).
2020-11-27 11:16:22 +01:00
- **float map(float value)** maps the parameter.
2022-11-05 10:57:25 +01:00
- **float back(float value)** does the inverse mapping.
2020-11-27 11:16:22 +01:00
2021-01-29 12:31:58 +01:00
### Constrains
2022-11-05 10:57:25 +01:00
FastMap supports three versions of constraining the map function, based upon the parameters of **init()**.
2020-11-27 11:16:22 +01:00
- **float constrainedMap(float value);** returns a value between outMin .. outMax
2022-11-05 10:57:25 +01:00
- **float lowerConstrainedMap(float value);** returns a value between outMin .. infinity, ==> no upper limit.
- **float upperConstrainedMap(float value);** returns a value between -infinity .. outMax ==> no lower limit.
2020-11-27 11:16:22 +01:00
2022-11-05 10:57:25 +01:00
To change the constrain values call **init()** with new limits, or use the standard **constrain()**.
2020-11-27 11:16:22 +01:00
Note there are **NO** constrain-versions for **back(value)** function.
## FastMapDouble
Version 3.0 adds **fastMapDouble** which has the same interface as **fastMap()**.
2020-11-27 11:16:22 +01:00
This class is meant to support 8 bytes doubles in their native accuracy and precision.
To display doubles one might need the **sci()** function of my **printHelpers** class.
2021-01-29 12:31:58 +01:00
https://github.com/RobTillaart/printHelpers
2020-11-27 11:16:22 +01:00
2022-11-05 10:57:25 +01:00
Note that on most embedded platforms the performance of doubles is less than floats.
2023-10-31 11:37:49 +01:00
See below.
2022-11-05 10:57:25 +01:00
#### boards supporting double
| board | float | double |
|:---------------|:-------:|:--------:|
| UNO | yes | no |
| ATMEGA | yes | no |
| MKR1000 | yes | yes |
| Zero | yes | yes |
| Teensy | yes | ? |
| ESP32 | yes | yes |
| RP2040 | yes | ? |
to elaborate table. (if someone has a good link, please let me know).
test code.
```cpp
void setup() {
Serial.begin(115200);
Serial.print("size of double:\t");
Serial.println(sizeof(double));
Serial.print("size of float: \t");
Serial.println(sizeof(float));
}
void loop() {}
```
2023-10-31 11:37:49 +01:00
## Performance
Tested version 0.4.1 with **fastMap_performance_test.ino**
| | MAP | FASTMAP | FASTMAP dbl | Notes |
|:--------|:--------:|:---------:|:-------------:|:-------:|
| UNO | 496072 | 211888 | 211888 | float == double
| ESP32 | 1814 | 627 | 6924 |
UNO scores factor 2.34
ESP32 scores factor 2.89
Note: the 8 byte double (ESP32) is ~11 x slower than the float version,
and ~4 x slower than the default map function.
So unless the precision of 8 bytes double is required one better
uses the float version.
2020-11-27 11:16:22 +01:00
2023-10-31 11:37:49 +01:00
Note: always do your own performance measurements!
If you have additional performance figures for other boards,
please let me know (report via an issue).
## Future
2023-10-31 11:37:49 +01:00
#### Must
2022-11-05 10:57:25 +01:00
- update documentation
2023-10-31 11:37:49 +01:00
#### Should
2022-11-05 10:57:25 +01:00
- test performance fastMapDouble on ESP32.
2023-10-31 11:37:49 +01:00
- need good test example
#### Could
2022-11-05 10:57:25 +01:00
- investigate map function for complex numbers? / coordinates?
- Template class?
2022-11-05 10:57:25 +01:00
2023-10-31 11:37:49 +01:00
#### Wont
- can fastMap and fastMapDouble be in a class hierarchy? limited gain?
## Support
If you appreciate my libraries, you can support the development and maintenance.
Improve the quality of the libraries by providing issues and Pull Requests, or
donate through PayPal or GitHub sponsors.
Thank you,