Introduction
My job requires me to interface with a lot of embedded systems. Often times when communicating with those embedded systems a CRC check value will need to be calculated. A lot of the code I write is in C# and although there a few solutions out there for computing CRCs in .NET/Mono, the majority are not complete or generic. They may work for a CRC32 calculation, but then break when you need to use them for a CRC32 which requires reflection in the lookup table and check value calculation. The goal for this project was to produce a CRC algorithm implementation in C# which could handle any CRC from 8 to 64 bits using the most commonly recognized CRC parameterization model.
Kudos
This project would not have been possible if it were not for…
Ross Williams’ A PAINLESS GUIDE TO CRC ERROR DETECTION ALGORITHMS. Ross has graciously published this guide into the public domain. The guide goes through the theory and implementation of CRC algorithms and perhaps most famously defines a parameterization model (Rocksoft^tm Model) to categorize the many CRC variants. The parameterization model and C implementation of the CRC algorithm were a massive help in understanding how CRCs work and in implementing the CRC algorithm in C#.
The CRC RevEng project’s ‘Catalogue of parametrised CRC algorithms’ maintained by Greg Cook. The values in the catalog were used in the unit tests which verified the correctness of the implementation.
Code
Below is the C# class representing the aforementioned CRC Parameterization Model. It is composed of a constructor which validates and sets the properties which themselves are read-only.
The key characteristics of any CRC algorithm are:
- Width: The width of the CRC algorithm in bits.
- Polynomial: The polynomial of the CRC algorithm.
- Initial Value: The initial value used in the computation of the CRC check value.
- XOR Out Value: The value which is XORed to the final computed value before returning the check value.
- Reflect In: Indicates whether bytes are reflected before being processed.
- Reflect Out: Indicates whether the final computed value is reflected before the XOR stage.
/* Simplified BSD License Copyright 2017 Derek Will Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ using System; namespace CrcSharp { /// <summary> /// CRC algorithm parameters. /// </summary> public class CrcParameters { private readonly int _width; private readonly ulong _polynomial; private readonly ulong _initialValue; private readonly ulong _xorOutValue; private readonly bool _reflectIn; private readonly bool _reflectOut; /// <summary> /// Gets the width of the CRC algorithm in bits. /// </summary> /// <value>The width of the CRC algorithm in bits.</value> public int Width { get { return _width; } } /// <summary> /// Gets the polynomial of the CRC algorithm. /// </summary> /// <value>The polynomial of the CRC algorithm.</value> public ulong Polynomial { get { return _polynomial; } } /// <summary> /// Gets the initial value used in the computation of the CRC check value. /// </summary> /// <value>The initial value used in the computation of the CRC check value.</value> public ulong InitialValue { get { return _initialValue; } } /// <summary> /// Gets the value which is XORed to the final computed value before returning the check value. /// </summary> /// <value>The value which is XORed to the final computed value before returning the check value.</value> public ulong XorOutValue { get { return _xorOutValue; } } /// <summary> /// Gets a value indicating whether bytes are reflected before being processed. /// </summary> /// <value><c>true</c> if each byte is to be reflected before being processed; otherwise, <c>false</c>.</value> public bool ReflectIn { get { return _reflectIn; } } /// <summary> /// Gets a value indicating whether the final computed value is reflected before the XOR stage. /// </summary> /// <value><c>true</c> if the final computed value is reflected before the XOR stage; otherwise, <c>false</c>.</value> public bool ReflectOut { get { return _reflectOut; } } /// <summary> /// Initializes a new instance of the <see cref="CrcSharp.CrcParameters"/> class. /// </summary> /// <param name="width">Width of the CRC algorithm in bits.</param> /// <param name="polynomial">Polynomial of the CRC algorithm.</param> /// <param name="initialValue">Initial value used in the computation of the CRC check value.</param> /// <param name="xorOutValue">The value which is XORed to the final computed value before returning the check value.</param> /// <param name="reflectIn">If set to <c>true</c> each byte is to be reflected before being processed.</param> /// <param name="reflectOut">If set to <c>true</c> the final computed value is reflected before the XOR stage.</param> public CrcParameters (int width, ulong polynomial, ulong initialValue, ulong xorOutValue, bool reflectIn, bool reflectOut) { ThrowIfParametersInvalid (width, polynomial, initialValue, xorOutValue); _width = width; _polynomial = polynomial; _initialValue = initialValue; _xorOutValue = xorOutValue; _reflectIn = reflectIn; _reflectOut = reflectOut; } /// <summary> /// Verifies if the parameter values are valid. /// </summary> /// <param name="width">Width of the CRC algorithm in bits.</param> /// <param name="polynomial">Polynomial of the CRC algorithm.</param> /// <param name="initialValue">Initial value used in the computation of the CRC check value.</param> /// <param name="xorOutValue">The value which is XORed to the final computed value before returning the check value.</param> private void ThrowIfParametersInvalid(int width, ulong polynomial, ulong initialValue, ulong xorOutValue) { if (width < 8 || width > 64) throw new ArgumentOutOfRangeException ("width", "Width must be between 8-64 bits."); ulong maxValue = (UInt64.MaxValue >> (64 - width)); if (polynomial > maxValue) throw new ArgumentOutOfRangeException ("polynomial", string.Format("Polynomial exceeds {0} bits.", width)); if (initialValue > maxValue) throw new ArgumentOutOfRangeException ("initialValue", string.Format("Initial Value exceeds {0} bits.", width)); if (xorOutValue > maxValue) throw new ArgumentOutOfRangeException ("xorOutValue", string.Format ("XOR Out Value exceeds {0} bits.", width)); } } }
The following CRC class generates a lookup table at runtime using the CRC Parameters passed into the constructor. The CRC check values for data can then be computed and returned to the caller as numeric values (i.e. 0xcbf43926) or as a byte array (i.e. { 0x26, 0x39, 0xf4, 0xcb }) depending on the method used.
/* Simplified BSD License Copyright 2017 Derek Will Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ using System; using System.Linq; namespace CrcSharp { /// <summary> /// CRC algorithm. /// </summary> public class Crc { private readonly CrcParameters _parameters; private readonly ulong[] _lookupTable; /// <summary> /// Gets the CRC algorithm parameters. /// </summary> /// <value>The CRC algorithm parameters.</value> public CrcParameters Parameters { get { return _parameters; } } /// <summary> /// Gets the lookup table used in calculating check values. /// </summary> /// <value>The lookup table.</value> public ulong[] LookupTable { get { return _lookupTable; } } /// <summary> /// Initializes a new instance of the <see cref="CrcSharp.Crc"/> class. /// </summary> /// <param name="parameters">CRC algorithm parameters.</param> public Crc(CrcParameters parameters) { if (parameters == null) throw new ArgumentNullException ("parameters", "Parameters argument cannot be null."); _parameters = parameters; _lookupTable = GenerateLookupTable(); } /// <summary> /// Calculates the CRC check value as a numeric value. /// </summary> /// <returns>The CRC check value as a numeric value.</returns> /// <param name="data">Data to compute the check value of.</param> public ulong CalculateAsNumeric(byte[] data) { byte[] crcCheckVal = CalculateCheckValue (data); Array.Resize (ref crcCheckVal, 8); return BitConverter.ToUInt64(crcCheckVal, 0); } /// <summary> /// Calculates the CRC check value as a byte array. /// </summary> /// <returns>The CRC check value as a byte array.</returns> /// <param name="data">Data to compute the check value of.</param> public byte[] CalculateCheckValue(byte[] data) { if (data == null) throw new ArgumentNullException ("data", "Data argument cannot be null."); ulong crc = _parameters.InitialValue; if (_parameters.ReflectIn) { crc = ReflectBits (crc, _parameters.Width); } foreach (byte b in data) { if (_parameters.ReflectIn) { crc = _lookupTable [(crc ^ b) & 0xFF] ^ (crc >> 8); } else { crc = _lookupTable[((crc >> (_parameters.Width - 8)) ^ b) & 0xFF] ^ (crc << 8); } crc &= (UInt64.MaxValue >> (64 - _parameters.Width)); } // Source: https://stackoverflow.com/questions/28656471/how-to-configure-calculation-of-crc-table/28661073#28661073 // Per Mark Adler - ...the reflect out different from the reflect in (CRC-12/3GPP). // In that one case, you need to bit reverse the output since the input is not reflected, but the output is. if (_parameters.ReflectIn ^ _parameters.ReflectOut) { crc = ReflectBits (crc, _parameters.Width); } ulong crcFinalValue = crc ^ _parameters.XorOutValue; return BitConverter.GetBytes(crcFinalValue).Take((_parameters.Width + 7)/ 8).ToArray(); } /// <summary> /// Generates the lookup table using the CRC algorithm parameters. /// </summary> /// <returns>The lookup table.</returns> private ulong[] GenerateLookupTable() { if (_parameters == null) throw new InvalidOperationException ("CRC parameters must be set prior to calling this method."); var lookupTable = new ulong[256]; ulong topBit = (ulong)((ulong)1 << (_parameters.Width - 1)); for (int i = 0; i < lookupTable.Length; i++) { byte inByte = (byte)i; if (_parameters.ReflectIn) { inByte = (byte)ReflectBits(inByte, 8); } ulong r = (ulong)((ulong)inByte << (_parameters.Width - 8)); for (int j = 0; j < 8; j++) { if ((r & topBit) != 0) { r = ((r << 1) ^ _parameters.Polynomial); } else { r = (r << 1); } } if (_parameters.ReflectIn) { r = ReflectBits(r, _parameters.Width); } lookupTable[i] = r & (UInt64.MaxValue >> (64 - _parameters.Width)); } return lookupTable; } /// <summary> /// Reflects the bits of a provided numeric value. /// </summary> /// <returns>Bit-reflected version of the provided numeric value.</returns> /// <param name="b">Value to reflect the bits of.</param> /// <param name="bitCount">Number of bits in the provided value.</param> private static ulong ReflectBits(ulong b, int bitCount) { ulong reflection = 0x00; for (int bitNumber = 0; bitNumber < bitCount; ++bitNumber) { if (((b >> bitNumber) & 0x01) == 0x01) { reflection |= (ulong)(((ulong)1 << ((bitCount - 1) - bitNumber))); } } return reflection; } } }
Testing
At the time of this post, there are 83 CRC algorithm variants between 8 and 64 bits in width documented in CRC RevEng’s Catalog. All 83 variants have been unit tested using NUnit and successfully pass.
Below is a sample snippet from the CRC-32 Tests:
[TestFixture] class Crc32Tests { private byte[] _data; [SetUp] protected void SetUp() { _data = System.Text.ASCIIEncoding.ASCII.GetBytes ("123456789"); } [Test] public void Crc32_Standard_Calculate() { var crc32 = new Crc(new CrcParameters(32, 0x04c11db7, 0xffffffff, 0xffffffff, true, true)); Assert.AreEqual(0xcbf43926, crc32.CalculateAsNumeric (_data)); Assert.IsTrue (crc32.CalculateCheckValue (_data).SequenceEqual (new byte[] { 0x26, 0x39, 0xf4, 0xcb })); } }
Concluding Remarks
The focus of this project was to create a lightweight CRC implementation for .NET/Mono software projects which could be easily imported into the source code of a software project or compiled and used as a DLL. I plan on hosting the project for download here on my website and soon I will put the project up on Github and as a NuGet package.
The project covers CRC 8-64 bits. CRC support for width under 8 bits is provided by a wonderful open source .NET/Mono project called Data.HashFunction which covers those CRCs as well as many more non-cryptographic hash functions.
Link to download project: here
Hi Derek,
I’m so glad to have come across this work.
I downloaded the project. But I don’t know how to run the test part.
I’m new in c#. That would be great if you could help me on the test part.
Best regards,
Gokberk.
Hi Gokberk, I posted the project on Github as well. If you pull down that project you can run the tests as simple as “dotnet test” from the commandline when in the solution or project directory.
Hi Derek,
Can you share the C# dll which can be integrated to other programs for CRC calculation?
The code is available here: https://github.com/derek-will/CrcSharp
Nuget package has been published: https://www.nuget.org/packages/CrcSharp
You can also get the DLL in the ZIP archive I published here: https://github.com/derek-will/CrcSharp/releases/tag/1.2.0
With regards to the polynomial are we leaving off the +x^0 or x^n?
Hi Austin, in the parameterization model that this library utilizes – X^n is left off. The upper bit is implicit and not part of the polynomial within the model.