263 lines
10 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/printHelpers/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-11-13 18:15:22 +01:00
[![Arduino-lint](https://github.com/RobTillaart/printHelpers/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/printHelpers/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/printHelpers/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/printHelpers/actions/workflows/jsoncheck.yml)
2021-01-29 12:31:58 +01:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/printHelpers/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/printHelpers.svg?maxAge=3600)](https://github.com/RobTillaart/printHelpers/releases)
# printHelpers
2021-12-24 13:26:40 +01:00
Arduino library to help formatting data for printing.
2021-01-29 12:31:58 +01:00
## Description
The printHelpers library contains a number of functions that help to print
data in a way not possible in the standard print library of the Arduino.
2023-01-28 13:52:36 +01:00
- **print64()** print **uint64_t** and **int64_t**
- **sci()** generates in scientific format - exponent has step 1.
- **eng()** generates in engineering format - exponent has step 3.
- **scieng()** generated exponential format - exponent has step 1 to 9.
- **toBytes()** generates KB MB GB etc.
- **hex()** generates hexadecimal output with leading zeros up to **uint64_t**.
- **bin()** generates binary output with leading zeros up to **uint64_t**.
2023-01-29 11:51:20 +01:00
- **toRoman()** generates a ROMAN representation of a (positive) number.
2023-01-28 13:52:36 +01:00
Details, see below.
2022-11-29 17:12:53 +01:00
2023-01-29 11:51:20 +01:00
#### Thread safety
2022-11-29 17:12:53 +01:00
2023-01-28 13:52:36 +01:00
Note the functions of this library all share an internal buffer, so the library is
definitely **not** thread safe.
Therefore one should copy / print the data (returned pointer) as fast as possible.
2022-11-29 17:12:53 +01:00
Thread-safe versions of these print functions might be made in the future.
## Interface
2023-01-28 13:52:36 +01:00
```cpp
#include "printHelpers.h"
```
2021-01-29 12:31:58 +01:00
The following functions are implemented:
2023-01-28 13:52:36 +01:00
2022-11-29 17:12:53 +01:00
#### print64()
2021-11-13 18:15:22 +01:00
- **char \* print64(int64_t value, uint8_t base)** converts a 64 bit integer
number to a char array.
2021-01-29 12:31:58 +01:00
The plus sign is not printed, neither are leading zero's.
Base 10 (DEC) and 16 (HEX) are supported and other bases up to 36 can be used.
Note that negative numbers will always get a minus sign for any base.
Cast the number to uint64_t to suppress the sign.
2021-11-13 18:15:22 +01:00
- **char \* print64(uint64_t value, uint8_t base)** converts a unsigned 64 bit
int number to a char array.
2021-01-29 12:31:58 +01:00
No sign is printed, neither are leading zero's.
Base 10 (DEC) and 16 (HEX) are supported and bases up to 36 can be used.
2023-01-28 13:52:36 +01:00
2022-11-29 17:12:53 +01:00
#### sci() eng()
2021-01-29 12:31:58 +01:00
2021-11-13 18:15:22 +01:00
- **char \* sci(double value, uint8_t decimals)** converts a float or double to a
char array.
2023-01-28 13:52:36 +01:00
E.g. **print(sci(f, 4))** ==> results in "6.7407E+21".
2021-01-29 12:31:58 +01:00
The existing Arduino print library only supports printing of floats and
2021-12-24 13:26:40 +01:00
doubles up to about 4E9 while the range of floats goes up to ~1E38.
2021-01-29 12:31:58 +01:00
The smallest float values will often be printed as 0.00 while floats
2023-01-28 13:52:36 +01:00
support down to about 1E-38 (subnormal even to 1E-45).
Existing (AVR) library functions **dtostrf()** has no scientific notation
and **dtostre()** is limited to 7 decimals. These latter two are faster.
Values printed with **sci()** do look pretty in column output.
2021-01-29 12:31:58 +01:00
2021-11-13 18:15:22 +01:00
- **char \* eng(double value, uint8_t decimals)** converts a float or double to a
char array.
2021-01-29 12:31:58 +01:00
E.g. print(eng(f, 4)) ==> results in "6.7407E+21".
Note the exponent created by **eng()** is always a multiple of 3.
2023-01-28 13:52:36 +01:00
Values printed with **eng()** do not always look pretty in column output.
This is due to the exponent power of 3. However its output translates easy to
thousands, millions etc which are powers of 3.
2021-01-29 12:31:58 +01:00
2021-11-13 18:15:22 +01:00
- **char \* scieng(double value, uint8_t decimals, uint8_t exponentMultiple)** converts a
float or double to a char array.
2021-01-29 12:31:58 +01:00
**sci()** and **eng()** use the same underlying function called **scieng()**
as the initial code for converting was almost identical.
2023-01-29 11:51:20 +01:00
Although not intended to be used directly, one may use it.
2021-11-13 18:15:22 +01:00
The last parameter **exponentMultiple** defines where the exponent is a multiple of.
2021-01-29 12:31:58 +01:00
For the **sci()** function this is 1, for the **eng()** function this is 3.
The **scieng()** function works for multiples from 1..9 for the exponent.
The usability of other values than 1 and 3 are not known.
Personally I like the multiple of 2 as I get 2 orders of magnitude in the
mantissa.
2023-01-28 13:52:36 +01:00
2022-11-29 17:12:53 +01:00
#### toBytes()
2021-01-29 12:31:58 +01:00
2021-11-13 18:15:22 +01:00
- **char \* toBytes(double value, uint8_t decimals = 2)** makes from a big number
2021-01-29 12:31:58 +01:00
representing an amount of bytes a shorter string usable for displaying.
The number of decimals is max 3
2023-01-28 13:52:36 +01:00
Example 3.292.528 ==> "3.140 MB"
2023-01-29 11:51:20 +01:00
2021-01-29 12:31:58 +01:00
Value ranges supported are in steps of powers of 1024.
2023-01-29 11:51:20 +01:00
These will all be shown in UPPERCASE so KB, MB etc.
| Unit | abbrev. | size | Unit | abbrev. | size |
|:-----------:|:---------:|:--------:|:------------:|:---------:|:--------:|
| kilobytes | KB | 1024 | zettabytes | KB | 1024^7 |
| megabytes | MB | 1024^2 | yottabytes | MB | 1024^8 |
| gigabytes | GB | 1024^3 | xonaytes | GB | 1024^9 |
| terabytes | TB | 1024^4 | wekabytes | TB | 1024^10 |
| petabytes | PB | 1024^5 | vundabytes | PB | 1024^11 |
| exabytes | EB | 1024^6 | udabytes | EB | 1024^12 |
Treda Byte is shortened as "TDB" and uses 2 chars to indicate the magnitude.
That would take extra memory or slightly more complex code.
As it is very seldom used, "official" support stops with UDA.
Should be big enough for some time.
Note: max uint64_t 2^64 is in the order of exa or zetta bytes.
To have some support for the really big sizes the code uses lowercase for the next 8 levels:
2021-01-29 12:31:58 +01:00
treda sorta rinta quexa pepta ocha nena minga luma (1024\^21 ~~ 10\^63)
2023-01-29 11:51:20 +01:00
To enable this patch the function in the **printHelpers.cpp** file.
2021-01-29 12:31:58 +01:00
2023-01-28 13:52:36 +01:00
2022-11-29 17:12:53 +01:00
#### hex() bin()
2023-01-28 13:52:36 +01:00
The default print() function of Arduino does not have leading zero's
for **HEX** and **BIN**.
This often causes a "broken" layout especially if one wants to print
in columns or so.
2022-11-29 17:12:53 +01:00
To solve this the following functions are added that will generate a
constant length char array.
- **char \* hex(uint64_t value, uint8_t digits = 16)**
- **char \* hex(uint32_t value, uint8_t digits = 8)**
- **char \* hex(uint16_t value, uint8_t digits = 4)**
- **char \* hex(uint8_t value, uint8_t digits = 2)**
- **char \* bin(uint64_t value, uint8_t digits = 64)**
- **char \* bin(uint32_t value, uint8_t digits = 32)**
- **char \* bin(uint16_t value, uint8_t digits = 16)**
- **char \* bin(uint8_t value, uint8_t digits = 8)**
Note: Data types not supported, must be cast to an supported type.
Note: There is overlap between **hex(value)** and **print64(value, HEX)**.
The latter does not produce the leading zero's or fixed length output.
2021-01-29 12:31:58 +01:00
2023-01-29 11:51:20 +01:00
#### toRoman()
https://en.wikipedia.org/wiki/Roman_numerals
A less used but well known print format are the Roman digits.
The library function **toRoman()** will convert any number from 0..100 million into a Roman number.
The numbers 1..5000 ("official" range) are the well known UPPER case characters.
- **char \* toRoman(uint32_t value)** returns Roman string.
| char | unit | notes |
|:------:|:-------|:------------|
| M | 1000 | M = Mille |
| D | 500 |
| C | 100 | C = Cent |
| L | 50 |
| X | 10 |
| V | 5 |
| I | 1 |
| N | 0 | extension |
2021-01-29 12:31:58 +01:00
2023-01-29 11:51:20 +01:00
Note: The maximum length returned is 16 characters in the "official" supported range.
4888 == MMMMDCCCLXXXVIII.
2021-01-29 12:31:58 +01:00
2023-01-29 11:51:20 +01:00
Notes:
- value == 0 => N is not part of the "official" numbers but we need it.
- values < 0 are not supported (note parameter is unsigned)
- values between 5K-10K are extended with extra M chars.
- values 10K-100M are represented with lower case characters.
This is not a standard, but it sort of works well.
- values > 100M return OVF == overflow.
- There is no special 'subtract code' for 9000 to have a clear distinction between
"official" and extended numbers.
- The number 4 is often written as IIII on clocks with Roman digits,
although IV would be (more?) correct and therefore IV is used.
## Shared print buffer
2021-01-29 12:31:58 +01:00
2021-11-13 18:15:22 +01:00
The implementation of the function all use a shared buffer to hold the
generated string.
This is done to reduce the memory overhead of embedding static buffers.
**Note this is not thread safe!**
In a coming release the functions will be able to pass a buffer to them
to become more thread safe.
The size of this shared buffer is default 66 to be able to print a 64 bit
integer in base 2.
To save memory one can change this buffer size in the code or compile time
2023-01-28 13:52:36 +01:00
by changing **PRINTBUFFERSIZE** in printHelpers.h.
2021-01-29 12:31:58 +01:00
Be aware that **sci()** and **eng()** use the same buffer.
2023-01-28 13:52:36 +01:00
These functions need about 10 bytes plus one bytes for every decimal used.
2021-01-29 12:31:58 +01:00
So for floats one need 15-20 bytes max, for doubles one need up to 30 bytes max.
In practice a size of 22 will work for most applications.
2021-11-13 18:15:22 +01:00
| PRINTBUFFERSIZE | BASE SUPPORTED | nr. decimals | Notes |
|:---------------:|:--------------:|:------------:|:----------|
| 66 | 02 - 36 | 0 - 50 | (default) |
| 34 | 04 - 36 | 0 - 20 |
| 24 | 08 - 36 | 0 - 14 |
| 22 | 10 - 36 | 0 - 12 |
| 18 | 16 - 36 | 0 - 07 |
2021-01-29 12:31:58 +01:00
When functions are added, the recommended minimum size might increase.
## Future
2023-01-28 13:52:36 +01:00
#### Must
2022-11-29 17:12:53 +01:00
- check TODO's in the code
2023-01-29 11:51:20 +01:00
2023-01-28 13:52:36 +01:00
#### Should
- documentation
2022-11-29 17:12:53 +01:00
- improve readability of the code
2023-01-28 13:52:36 +01:00
#### Could
2023-01-29 11:51:20 +01:00
- investigate separators in **hex()**
- space per 8, 4 or 2
2023-01-28 13:52:36 +01:00
- investigate thread safe version
- pass char buffer as parameter (breaking)
- could be the log10 pow version?
- investigate distance print helpers.
- feet(float cm) as 3'2" or 3-7/8 feet
- inch(float cm) as 32"
- yards(float meter),
- miles(float kilometre)
#### Wont
2022-11-29 17:12:53 +01:00
- add **float()** as Arduino limits floats to "MAXLONG" by code.
2023-01-28 13:52:36 +01:00
- use dtostrf() - is that portable?
2022-11-29 17:12:53 +01:00
- use sci() or eng()
2023-01-28 13:52:36 +01:00
- add **base(value, digits, base)** for any base > 1.
2022-11-29 17:12:53 +01:00
- only upon request.
2023-01-28 13:52:36 +01:00
- investigate separators in **bin()**
- point or space, per 8 or 4 or 2
- ==> printBuffer too small for bin(64) ==> need 75-100 bytes.
- Investigate performance and accuracy
- **sci()** and **eng()**.
- investigate sci() version based upon use of log()
- done => see examples.