222 lines
8.2 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/CRC/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-10-27 13:42:30 +02:00
[![Arduino-lint](https://github.com/RobTillaart/CRC/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/CRC/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/CRC/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/CRC/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/CRC/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/CRC.svg?maxAge=3600)](https://github.com/RobTillaart/CRC/releases)
# CRC
2022-01-26 10:27:58 +01:00
Arduino library with CRC8, CRC12, CRC16, CRC32 and CRC64 functions.
2021-01-29 12:31:58 +01:00
## Description
2023-01-17 16:18:12 +01:00
Goal of this library is to have a flexible and portable set of generic
2021-12-14 20:22:40 +01:00
CRC functions and classes.
2021-01-29 12:31:58 +01:00
2023-01-17 16:18:12 +01:00
The CRCx classes have a number of added values.
Most important is that they allow one to verify intermediate CRC values.
This is useful if one sends a "train of packets" which include a CRC so far.
This detects both errors in one single packet but also optional missing packets,
2022-05-20 09:04:29 +02:00
or even injected packets.
2021-01-29 12:31:58 +01:00
2023-01-17 16:18:12 +01:00
Another trick one can do with the class CRCx is to change the polynome or
2022-05-20 09:04:29 +02:00
the reverse flag runtime during the process. This makes it harder to imitate.
2021-04-07 13:31:22 +02:00
Furthermore the class allows to add values in single steps and continue too.
2023-01-17 16:18:12 +01:00
Finally the class version gives more readable code (IMHO) as the parameters
2021-12-14 20:22:40 +01:00
are explicitly set.
2021-01-29 12:31:58 +01:00
**Note** the classes have same names as the static functions, except the class
2023-01-17 16:18:12 +01:00
is UPPER case. So **CRC8** is a class and **crc8()** is the function.
2021-01-29 12:31:58 +01:00
2022-05-20 09:04:29 +02:00
Deeper tech info - https://en.wikipedia.org/wiki/Cyclic_redundancy_check
2021-01-29 12:31:58 +01:00
and many other websites.
2023-01-17 16:18:12 +01:00
#### Related
- https://github.com/RobTillaart/Adler
- https://github.com/RobTillaart/CRC
- https://github.com/RobTillaart/Fletcher
- https://github.com/RobTillaart/LUHN
2021-01-29 12:31:58 +01:00
## Interface CRC classes
2022-05-20 09:04:29 +02:00
The interfaces are very similar for CRC8, CRC12, CRC16, CRC32 and CRC64 class.
2023-01-17 16:18:12 +01:00
The only difference is the data type for polynome, start- and end-mask,
2021-12-14 20:22:40 +01:00
and the returned CRC.
2021-01-29 12:31:58 +01:00
2022-02-07 14:10:53 +01:00
2023-01-17 16:18:12 +01:00
#### Base
2022-01-28 14:56:11 +01:00
2021-01-29 12:31:58 +01:00
Use **\#include "CRC8.h"**
2022-05-20 09:04:29 +02:00
- **CRC8()** default - parameterless - constructor.
- **CRC8(polynome, XORstart, XORend, reverseIn, reverseOut)** Constructor to set all parameters at once.
- **void reset()** set all internals to defaults of the **CRC8()** parameterless constructor.
2022-02-07 14:10:53 +01:00
- **void restart()** reset internal CRC and count only;
reuse values for other e.g polynome, XOR masks and reverse flags.
2021-01-29 12:31:58 +01:00
- **void add(value)** add a single value to CRC calculation.
2023-01-17 16:18:12 +01:00
- **void add(array, uint16_t length)** add an array of values to the CRC.
2022-02-07 14:10:53 +01:00
In case of a warning/error for the array type, use casting to (uint8_t \*).
2023-01-17 16:18:12 +01:00
- **uint8_t getCRC()** returns CRC calculated so far. This allows to check the CRC of
2021-12-14 20:22:40 +01:00
a really large stream at intermediate moments, e.g. to link multiple packets.
- **uint32_t count()** returns number of values added so far. Default 0.
2021-01-29 12:31:58 +01:00
2022-02-07 14:10:53 +01:00
2023-01-17 16:18:12 +01:00
#### Parameters
2022-01-28 14:56:11 +01:00
2022-05-20 09:04:29 +02:00
The parameters do not have defaults so the user must set them explicitly.
2022-01-28 14:56:11 +01:00
- **void setPolynome(polynome)** set polynome, note reset sets a default polynome.
- **void setStartXOR(start)** set start-mask, default 0.
- **void setEndXOR(end)** set end-mask, default 0.
- **void setReverseIn(bool reverseIn)** reverse the bit pattern of input data (MSB vs LSB).
- **void setReverseOut(bool reverseOut)** reverse the bit pattern of CRC (MSB vs LSB).
- **uint8_t getPolyNome()** return parameter set above or default.
- **uint8_t getStartXOR()** return parameter set above or default.
- **uint8_t getEndXOR()** return parameter set above or default.
- **bool getReverseIn()** return parameter set above or default.
- **bool getReverseOut()** return parameter set above or default.
2021-01-29 12:31:58 +01:00
2023-01-17 16:18:12 +01:00
#### Power users only
2022-02-08 10:10:54 +01:00
As CRC calculations of large blocks can take serious time (in milliseconds),
the classes call **yield()** after every 256 **add()** calls to keep RTOS
2023-01-17 16:18:12 +01:00
environments happy.
2022-02-08 10:10:54 +01:00
2023-01-17 16:18:12 +01:00
The following two calls allows one to enable and disable these calls to
2022-02-08 10:10:54 +01:00
**yield()** to get optimal performance. The risk is missing context switching
to handle interrupts etc. So use at own risk.
- **void enableYield()** enables the calls to **yield()**.
- **void disableYield()** disables the calls to **yield()**.
2023-01-17 16:18:12 +01:00
_Note: the static functions in this library also call **yield()** but this
2022-02-08 10:10:54 +01:00
cannot be disabled (for now)._
2023-01-17 16:18:12 +01:00
_Note: a parameter could be a future option to set the number of adds before
2022-05-20 09:04:29 +02:00
**yield()** is called. **setYield(0)** would be disable it._
2022-02-08 10:10:54 +01:00
2021-01-29 12:31:58 +01:00
### Example snippet
2023-01-17 16:18:12 +01:00
A minimal usage only needs:
2021-01-29 12:31:58 +01:00
- the constructor, the add() function and the getCRC() function.
```cpp
2021-04-07 13:31:22 +02:00
#include "CRC32.h"
...
2021-01-29 12:31:58 +01:00
CRC32 crc;
...
while (Serial.available())
{
int c = Serial.read();
crc.add(c);
}
Serial.println(crc.getCRC());
```
## Interface static functions
Use **\#include "CRC.h"**
Most functions have a default polynome, same start and end masks, and default there is no reversing.
2023-01-17 16:18:12 +01:00
However these parameters allow one to tweak the CRC in all aspects known.
In all the examples encountered the reverse flags were set both to false or both to true.
For flexibility both parameters are kept available.
2021-01-29 12:31:58 +01:00
2021-12-14 20:22:40 +01:00
- **uint8_t crc8(array, length, polynome = 0xD5, start = 0, end = 0, reverseIn = false, reverseOut = false)** idem with default polynome.
2022-01-26 10:27:58 +01:00
- **uint16_t crc12(array, length, polynome = 0x080D, start = 0, end = 0, reverseIn = false, reverseOut = false)** idem with default polynome.
2022-02-08 10:10:54 +01:00
- **uint16_t crc16(array, length, polynome = 0x8001, start = 0, end = 0, reverseIn = false, reverseOut = false)** idem with default polynome.
2021-01-29 12:31:58 +01:00
- **uint16_t crc16-CCITT(array, length)** fixed polynome **0x1021**, non zero start / end masks.
2021-12-14 20:22:40 +01:00
- **uint32_t crc32(array, length, polynome = 0x04C11DB7, start = 0, end = 0, reverseIn = false, reverseOut = false)** idem with default polynome.
2022-02-07 14:10:53 +01:00
- **uint64_t crc64(array, length, polynome = 0x42F0E1EBA9EA3693, start = 0, end = 0, reverseIn = false, reverseOut = false)** - experimental version, no reference found except on Wikipedia.
2021-01-29 12:31:58 +01:00
2023-01-17 16:18:12 +01:00
Note these functions are limited to one call per block of data.
2022-02-08 10:10:54 +01:00
These functions will call **yield()** every 256 bytes to keep RTOS happy.
For more flexibility use the specific classes.
2021-01-29 12:31:58 +01:00
2023-01-17 16:18:12 +01:00
The static CRC functions use fast reverse functions that can be also be
2022-02-08 10:10:54 +01:00
used outside CRC context. Their usage is straightforward.
2022-01-28 14:56:11 +01:00
- **uint8_t reverse8(uint8_t in)** idem.
- **uint16_t reverse16(uint16_t in)** idem.
- **uint16_t reverse12(uint16_t in)** idem.
- **uint32_t reverse32(uint32_t in)** idem.
- **uint64_t reverse64(uint64_t in)** idem.
Reverse12 is based upon reverse16, with a final shift.
Other reverses can be created in similar way.
2021-01-29 12:31:58 +01:00
2022-02-08 10:10:54 +01:00
## CRC_polynomes.h
Since version 0.2.1 the file CRC_polynomes.h is added to hold symbolic names for certain polynomes.
These can be used in your code too to minimize the number of "magic HEX codes".
If standard polynomes are missing, please open an issue and report, with reference.
2021-12-14 20:22:40 +01:00
## Operational
See examples.
2023-01-17 16:18:12 +01:00
## Links
2021-12-14 20:22:40 +01:00
- https://en.wikipedia.org/wiki/Cyclic_redundancy_check - generic background.
2022-01-28 14:56:11 +01:00
- http://zorc.breitbandkatze.de/crc.html - online CRC calculator (any base up to 64 is supported.)
2021-12-14 20:22:40 +01:00
- https://crccalc.com/ - online CRC calculator to verify.
2022-05-20 09:04:29 +02:00
- https://www.lddgo.net/en/encrypt/crc - online CRC calculator
2021-12-14 20:22:40 +01:00
2021-01-29 12:31:58 +01:00
## Future
2023-01-17 16:18:12 +01:00
#### Must
#### Should
- add examples.
2022-01-28 14:56:11 +01:00
- example showing multiple packages of data linked by their CRC.
2023-01-17 16:18:12 +01:00
sort of "blockchain"
#### Could
2022-05-20 09:04:29 +02:00
- table versions for performance? (performance - memory discussion).
2021-01-29 12:31:58 +01:00
- stream version - 4 classes class?
2022-02-07 14:10:53 +01:00
- **setCRC(value)** to be able to pick up where one left ?
- can be done with **setStartXOR()**
2022-05-20 09:04:29 +02:00
- needs **getRawCRC()** without reverse and end mask
2023-01-17 16:18:12 +01:00
- Think about default parameters for constructor **CRC8(polynome, XORstart, XORend, reverseIn, reverseOut)**
2022-05-20 09:04:29 +02:00
- same as reset so constructors merge? Note the CRC-functions do have defaults too.
2021-01-29 12:31:58 +01:00
#### Exotic CRC's ?
- **CRC1()** // parity :)
2021-12-14 20:22:40 +01:00
- **CRC4(array, length, polynome, start, end, reverseIn, reverseOut)** nibbles?
2023-01-17 16:18:12 +01:00
- default polynome 0x03 ITU
2022-02-07 14:10:53 +01:00
- One CRC() with #bits as parameter?
2023-01-17 16:18:12 +01:00
- up to 64 bit for all missing ones?
2022-02-07 14:10:53 +01:00
- performance penalty
2022-05-20 09:04:29 +02:00
- One CRC() template class?
2021-01-29 12:31:58 +01:00
2022-02-07 14:10:53 +01:00
#### Won't
2021-01-29 12:31:58 +01:00
2022-02-07 14:10:53 +01:00
- add a dump(Stream = Serial) to see all the settings at once.
user can access parameters, so no need.
2021-01-29 12:31:58 +01:00
2023-01-17 16:18:12 +01:00