mirror of
https://github.com/MariaDB/server.git
synced 2025-01-16 03:52:35 +01:00
d9d0e8fd60
In commit 9ec7819c58
the CRC-32 function
signatures had been unified somewhat, but not enough.
clang -fsanitize=undefined would flag a function pointer signature
mismatch between const char* and const void*, but not between
uint32_t and unsigned. We try to fix both inconsistencies anyway.
Reviewed by: Vladislav Vaintroub
719 lines
24 KiB
C++
719 lines
24 KiB
C++
/* Copyright (c) 2020, 2021, MariaDB
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1335 USA */
|
|
|
|
/*
|
|
* Copyright 2016 Ferry Toth, Exalon Delft BV, The Netherlands
|
|
* This software is provided 'as-is', without any express or implied
|
|
* warranty. In no event will the author be held liable for any damages
|
|
* arising from the use of this software.
|
|
* Permission is granted to anyone to use this software for any purpose,
|
|
* including commercial applications, and to alter it and redistribute it
|
|
* freely, subject to the following restrictions:
|
|
* 1. The origin of this software must not be misrepresented; you must not
|
|
* claim that you wrote the original software. If you use this software
|
|
* in a product, an acknowledgment in the product documentation would be
|
|
* appreciated but is not required.
|
|
* 2. Altered source versions must be plainly marked as such, and must not be
|
|
* misrepresented as being the original software.
|
|
* 3. This notice may not be removed or altered from any source distribution.
|
|
* Ferry Toth
|
|
* ftoth@exalondelft.nl
|
|
*
|
|
* https://github.com/htot/crc32c
|
|
*
|
|
* Modified by Facebook
|
|
*
|
|
* Original intel whitepaper:
|
|
* "Fast CRC Computation for iSCSI Polynomial Using CRC32 Instruction"
|
|
* https://www.intel.com/content/dam/www/public/us/en/documents/white-papers/crc-iscsi-polynomial-crc32-instruction-paper.pdf
|
|
*
|
|
* This version is from the folly library, created by Dave Watson <davejwatson@fb.com>
|
|
*
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
#include <nmmintrin.h>
|
|
#include <wmmintrin.h>
|
|
|
|
#ifdef _MSC_VER
|
|
# define USE_PCLMUL /* nothing */
|
|
#else
|
|
# define USE_PCLMUL __attribute__((target("sse4.2,pclmul")))
|
|
#endif
|
|
|
|
#define CRCtriplet(crc, buf, offset) \
|
|
crc##0 = _mm_crc32_u64(crc##0, *(buf##0 + offset)); \
|
|
crc##1 = _mm_crc32_u64(crc##1, *(buf##1 + offset)); \
|
|
crc##2 = _mm_crc32_u64(crc##2, *(buf##2 + offset));
|
|
|
|
#define CRCduplet(crc, buf, offset) \
|
|
crc##0 = _mm_crc32_u64(crc##0, *(buf##0 + offset)); \
|
|
crc##1 = _mm_crc32_u64(crc##1, *(buf##1 + offset));
|
|
|
|
#define CRCsinglet(crc, buf, offset) \
|
|
crc = _mm_crc32_u64(crc, *(uint64_t*)(buf + offset));
|
|
|
|
|
|
// Numbers taken directly from intel whitepaper.
|
|
// clang-format off
|
|
static const uint64_t clmul_constants alignas(16) [] = {
|
|
0x14cd00bd6, 0x105ec76f0, 0x0ba4fc28e, 0x14cd00bd6,
|
|
0x1d82c63da, 0x0f20c0dfe, 0x09e4addf8, 0x0ba4fc28e,
|
|
0x039d3b296, 0x1384aa63a, 0x102f9b8a2, 0x1d82c63da,
|
|
0x14237f5e6, 0x01c291d04, 0x00d3b6092, 0x09e4addf8,
|
|
0x0c96cfdc0, 0x0740eef02, 0x18266e456, 0x039d3b296,
|
|
0x0daece73e, 0x0083a6eec, 0x0ab7aff2a, 0x102f9b8a2,
|
|
0x1248ea574, 0x1c1733996, 0x083348832, 0x14237f5e6,
|
|
0x12c743124, 0x02ad91c30, 0x0b9e02b86, 0x00d3b6092,
|
|
0x018b33a4e, 0x06992cea2, 0x1b331e26a, 0x0c96cfdc0,
|
|
0x17d35ba46, 0x07e908048, 0x1bf2e8b8a, 0x18266e456,
|
|
0x1a3e0968a, 0x11ed1f9d8, 0x0ce7f39f4, 0x0daece73e,
|
|
0x061d82e56, 0x0f1d0f55e, 0x0d270f1a2, 0x0ab7aff2a,
|
|
0x1c3f5f66c, 0x0a87ab8a8, 0x12ed0daac, 0x1248ea574,
|
|
0x065863b64, 0x08462d800, 0x11eef4f8e, 0x083348832,
|
|
0x1ee54f54c, 0x071d111a8, 0x0b3e32c28, 0x12c743124,
|
|
0x0064f7f26, 0x0ffd852c6, 0x0dd7e3b0c, 0x0b9e02b86,
|
|
0x0f285651c, 0x0dcb17aa4, 0x010746f3c, 0x018b33a4e,
|
|
0x1c24afea4, 0x0f37c5aee, 0x0271d9844, 0x1b331e26a,
|
|
0x08e766a0c, 0x06051d5a2, 0x093a5f730, 0x17d35ba46,
|
|
0x06cb08e5c, 0x11d5ca20e, 0x06b749fb2, 0x1bf2e8b8a,
|
|
0x1167f94f2, 0x021f3d99c, 0x0cec3662e, 0x1a3e0968a,
|
|
0x19329634a, 0x08f158014, 0x0e6fc4e6a, 0x0ce7f39f4,
|
|
0x08227bb8a, 0x1a5e82106, 0x0b0cd4768, 0x061d82e56,
|
|
0x13c2b89c4, 0x188815ab2, 0x0d7a4825c, 0x0d270f1a2,
|
|
0x10f5ff2ba, 0x105405f3e, 0x00167d312, 0x1c3f5f66c,
|
|
0x0f6076544, 0x0e9adf796, 0x026f6a60a, 0x12ed0daac,
|
|
0x1a2adb74e, 0x096638b34, 0x19d34af3a, 0x065863b64,
|
|
0x049c3cc9c, 0x1e50585a0, 0x068bce87a, 0x11eef4f8e,
|
|
0x1524fa6c6, 0x19f1c69dc, 0x16cba8aca, 0x1ee54f54c,
|
|
0x042d98888, 0x12913343e, 0x1329d9f7e, 0x0b3e32c28,
|
|
0x1b1c69528, 0x088f25a3a, 0x02178513a, 0x0064f7f26,
|
|
0x0e0ac139e, 0x04e36f0b0, 0x0170076fa, 0x0dd7e3b0c,
|
|
0x141a1a2e2, 0x0bd6f81f8, 0x16ad828b4, 0x0f285651c,
|
|
0x041d17b64, 0x19425cbba, 0x1fae1cc66, 0x010746f3c,
|
|
0x1a75b4b00, 0x18db37e8a, 0x0f872e54c, 0x1c24afea4,
|
|
0x01e41e9fc, 0x04c144932, 0x086d8e4d2, 0x0271d9844,
|
|
0x160f7af7a, 0x052148f02, 0x05bb8f1bc, 0x08e766a0c,
|
|
0x0a90fd27a, 0x0a3c6f37a, 0x0b3af077a, 0x093a5f730,
|
|
0x04984d782, 0x1d22c238e, 0x0ca6ef3ac, 0x06cb08e5c,
|
|
0x0234e0b26, 0x063ded06a, 0x1d88abd4a, 0x06b749fb2,
|
|
0x04597456a, 0x04d56973c, 0x0e9e28eb4, 0x1167f94f2,
|
|
0x07b3ff57a, 0x19385bf2e, 0x0c9c8b782, 0x0cec3662e,
|
|
0x13a9cba9e, 0x0e417f38a, 0x093e106a4, 0x19329634a,
|
|
0x167001a9c, 0x14e727980, 0x1ddffc5d4, 0x0e6fc4e6a,
|
|
0x00df04680, 0x0d104b8fc, 0x02342001e, 0x08227bb8a,
|
|
0x00a2a8d7e, 0x05b397730, 0x168763fa6, 0x0b0cd4768,
|
|
0x1ed5a407a, 0x0e78eb416, 0x0d2c3ed1a, 0x13c2b89c4,
|
|
0x0995a5724, 0x1641378f0, 0x19b1afbc4, 0x0d7a4825c,
|
|
0x109ffedc0, 0x08d96551c, 0x0f2271e60, 0x10f5ff2ba,
|
|
0x00b0bf8ca, 0x00bf80dd2, 0x123888b7a, 0x00167d312,
|
|
0x1e888f7dc, 0x18dcddd1c, 0x002ee03b2, 0x0f6076544,
|
|
0x183e8d8fe, 0x06a45d2b2, 0x133d7a042, 0x026f6a60a,
|
|
0x116b0f50c, 0x1dd3e10e8, 0x05fabe670, 0x1a2adb74e,
|
|
0x130004488, 0x0de87806c, 0x000bcf5f6, 0x19d34af3a,
|
|
0x18f0c7078, 0x014338754, 0x017f27698, 0x049c3cc9c,
|
|
0x058ca5f00, 0x15e3e77ee, 0x1af900c24, 0x068bce87a,
|
|
0x0b5cfca28, 0x0dd07448e, 0x0ded288f8, 0x1524fa6c6,
|
|
0x059f229bc, 0x1d8048348, 0x06d390dec, 0x16cba8aca,
|
|
0x037170390, 0x0a3e3e02c, 0x06353c1cc, 0x042d98888,
|
|
0x0c4584f5c, 0x0d73c7bea, 0x1f16a3418, 0x1329d9f7e,
|
|
0x0531377e2, 0x185137662, 0x1d8d9ca7c, 0x1b1c69528,
|
|
0x0b25b29f2, 0x18a08b5bc, 0x19fb2a8b0, 0x02178513a,
|
|
0x1a08fe6ac, 0x1da758ae0, 0x045cddf4e, 0x0e0ac139e,
|
|
0x1a91647f2, 0x169cf9eb0, 0x1a0f717c4, 0x0170076fa,
|
|
};
|
|
|
|
// Compute the crc32c value for buffer smaller than 8
|
|
USE_PCLMUL
|
|
static inline void align_to_8(
|
|
size_t len,
|
|
uint64_t& crc0, // crc so far, updated on return
|
|
const unsigned char*& next) { // next data pointer, updated on return
|
|
uint32_t crc32bit = static_cast<uint32_t>(crc0);
|
|
if (len & 0x04) {
|
|
crc32bit = _mm_crc32_u32(crc32bit, *(uint32_t*)next);
|
|
next += sizeof(uint32_t);
|
|
}
|
|
if (len & 0x02) {
|
|
crc32bit = _mm_crc32_u16(crc32bit, *(uint16_t*)next);
|
|
next += sizeof(uint16_t);
|
|
}
|
|
if (len & 0x01) {
|
|
crc32bit = _mm_crc32_u8(crc32bit, *(next));
|
|
next++;
|
|
}
|
|
crc0 = crc32bit;
|
|
}
|
|
|
|
//
|
|
// CombineCRC performs pclmulqdq multiplication of 2 partial CRC's and a well
|
|
// chosen constant and xor's these with the remaining CRC.
|
|
//
|
|
USE_PCLMUL
|
|
static inline uint64_t CombineCRC(
|
|
size_t block_size,
|
|
uint64_t crc0,
|
|
uint64_t crc1,
|
|
uint64_t crc2,
|
|
const uint64_t* next2) {
|
|
const auto multiplier =
|
|
*(reinterpret_cast<const __m128i*>(clmul_constants) + block_size - 1);
|
|
const auto crc0_xmm = _mm_set_epi64x(0, crc0);
|
|
const auto res0 = _mm_clmulepi64_si128(crc0_xmm, multiplier, 0x00);
|
|
const auto crc1_xmm = _mm_set_epi64x(0, crc1);
|
|
const auto res1 = _mm_clmulepi64_si128(crc1_xmm, multiplier, 0x10);
|
|
const auto res = _mm_xor_si128(res0, res1);
|
|
crc0 = _mm_cvtsi128_si64(res);
|
|
crc0 = crc0 ^ *((uint64_t*)next2 - 1);
|
|
crc2 = _mm_crc32_u64(crc2, crc0);
|
|
return crc2;
|
|
}
|
|
|
|
// Compute CRC-32C using the Intel hardware instruction.
|
|
extern "C"
|
|
USE_PCLMUL
|
|
uint32_t crc32c_3way(uint32_t crc, const void *buf, size_t len)
|
|
{
|
|
const unsigned char* next = static_cast<const unsigned char*>(buf);
|
|
uint64_t count;
|
|
uint64_t crc0, crc1, crc2;
|
|
crc0 = crc ^ 0xffffffffu;
|
|
|
|
|
|
if (len >= 8) {
|
|
// if len > 216 then align and use triplets
|
|
if (len > 216) {
|
|
{
|
|
// Work on the bytes (< 8) before the first 8-byte alignment addr starts
|
|
auto align_bytes = (8 - (uintptr_t)next) & 7;
|
|
len -= align_bytes;
|
|
align_to_8(align_bytes, crc0, next);
|
|
}
|
|
|
|
// Now work on the remaining blocks
|
|
count = len / 24; // number of triplets
|
|
len %= 24; // bytes remaining
|
|
uint64_t n = count >> 7; // #blocks = first block + full blocks
|
|
uint64_t block_size = count & 127;
|
|
if (block_size == 0) {
|
|
block_size = 128;
|
|
} else {
|
|
n++;
|
|
}
|
|
// points to the first byte of the next block
|
|
const uint64_t* next0 = (uint64_t*)next + block_size;
|
|
const uint64_t* next1 = next0 + block_size;
|
|
const uint64_t* next2 = next1 + block_size;
|
|
|
|
crc1 = crc2 = 0;
|
|
// Use Duff's device, a for() loop inside a switch()
|
|
// statement. This needs to execute at least once, round len
|
|
// down to nearest triplet multiple
|
|
switch (block_size) {
|
|
case 128:
|
|
do {
|
|
// jumps here for a full block of len 128
|
|
CRCtriplet(crc, next, -128);
|
|
/* fallthrough */
|
|
case 127:
|
|
// jumps here or below for the first block smaller
|
|
CRCtriplet(crc, next, -127);
|
|
/* fallthrough */
|
|
case 126:
|
|
CRCtriplet(crc, next, -126); // than 128
|
|
/* fallthrough */
|
|
case 125:
|
|
CRCtriplet(crc, next, -125);
|
|
/* fallthrough */
|
|
case 124:
|
|
CRCtriplet(crc, next, -124);
|
|
/* fallthrough */
|
|
case 123:
|
|
CRCtriplet(crc, next, -123);
|
|
/* fallthrough */
|
|
case 122:
|
|
CRCtriplet(crc, next, -122);
|
|
/* fallthrough */
|
|
case 121:
|
|
CRCtriplet(crc, next, -121);
|
|
/* fallthrough */
|
|
case 120:
|
|
CRCtriplet(crc, next, -120);
|
|
/* fallthrough */
|
|
case 119:
|
|
CRCtriplet(crc, next, -119);
|
|
/* fallthrough */
|
|
case 118:
|
|
CRCtriplet(crc, next, -118);
|
|
/* fallthrough */
|
|
case 117:
|
|
CRCtriplet(crc, next, -117);
|
|
/* fallthrough */
|
|
case 116:
|
|
CRCtriplet(crc, next, -116);
|
|
/* fallthrough */
|
|
case 115:
|
|
CRCtriplet(crc, next, -115);
|
|
/* fallthrough */
|
|
case 114:
|
|
CRCtriplet(crc, next, -114);
|
|
/* fallthrough */
|
|
case 113:
|
|
CRCtriplet(crc, next, -113);
|
|
/* fallthrough */
|
|
case 112:
|
|
CRCtriplet(crc, next, -112);
|
|
/* fallthrough */
|
|
case 111:
|
|
CRCtriplet(crc, next, -111);
|
|
/* fallthrough */
|
|
case 110:
|
|
CRCtriplet(crc, next, -110);
|
|
/* fallthrough */
|
|
case 109:
|
|
CRCtriplet(crc, next, -109);
|
|
/* fallthrough */
|
|
case 108:
|
|
CRCtriplet(crc, next, -108);
|
|
/* fallthrough */
|
|
case 107:
|
|
CRCtriplet(crc, next, -107);
|
|
/* fallthrough */
|
|
case 106:
|
|
CRCtriplet(crc, next, -106);
|
|
/* fallthrough */
|
|
case 105:
|
|
CRCtriplet(crc, next, -105);
|
|
/* fallthrough */
|
|
case 104:
|
|
CRCtriplet(crc, next, -104);
|
|
/* fallthrough */
|
|
case 103:
|
|
CRCtriplet(crc, next, -103);
|
|
/* fallthrough */
|
|
case 102:
|
|
CRCtriplet(crc, next, -102);
|
|
/* fallthrough */
|
|
case 101:
|
|
CRCtriplet(crc, next, -101);
|
|
/* fallthrough */
|
|
case 100:
|
|
CRCtriplet(crc, next, -100);
|
|
/* fallthrough */
|
|
case 99:
|
|
CRCtriplet(crc, next, -99);
|
|
/* fallthrough */
|
|
case 98:
|
|
CRCtriplet(crc, next, -98);
|
|
/* fallthrough */
|
|
case 97:
|
|
CRCtriplet(crc, next, -97);
|
|
/* fallthrough */
|
|
case 96:
|
|
CRCtriplet(crc, next, -96);
|
|
/* fallthrough */
|
|
case 95:
|
|
CRCtriplet(crc, next, -95);
|
|
/* fallthrough */
|
|
case 94:
|
|
CRCtriplet(crc, next, -94);
|
|
/* fallthrough */
|
|
case 93:
|
|
CRCtriplet(crc, next, -93);
|
|
/* fallthrough */
|
|
case 92:
|
|
CRCtriplet(crc, next, -92);
|
|
/* fallthrough */
|
|
case 91:
|
|
CRCtriplet(crc, next, -91);
|
|
/* fallthrough */
|
|
case 90:
|
|
CRCtriplet(crc, next, -90);
|
|
/* fallthrough */
|
|
case 89:
|
|
CRCtriplet(crc, next, -89);
|
|
/* fallthrough */
|
|
case 88:
|
|
CRCtriplet(crc, next, -88);
|
|
/* fallthrough */
|
|
case 87:
|
|
CRCtriplet(crc, next, -87);
|
|
/* fallthrough */
|
|
case 86:
|
|
CRCtriplet(crc, next, -86);
|
|
/* fallthrough */
|
|
case 85:
|
|
CRCtriplet(crc, next, -85);
|
|
/* fallthrough */
|
|
case 84:
|
|
CRCtriplet(crc, next, -84);
|
|
/* fallthrough */
|
|
case 83:
|
|
CRCtriplet(crc, next, -83);
|
|
/* fallthrough */
|
|
case 82:
|
|
CRCtriplet(crc, next, -82);
|
|
/* fallthrough */
|
|
case 81:
|
|
CRCtriplet(crc, next, -81);
|
|
/* fallthrough */
|
|
case 80:
|
|
CRCtriplet(crc, next, -80);
|
|
/* fallthrough */
|
|
case 79:
|
|
CRCtriplet(crc, next, -79);
|
|
/* fallthrough */
|
|
case 78:
|
|
CRCtriplet(crc, next, -78);
|
|
/* fallthrough */
|
|
case 77:
|
|
CRCtriplet(crc, next, -77);
|
|
/* fallthrough */
|
|
case 76:
|
|
CRCtriplet(crc, next, -76);
|
|
/* fallthrough */
|
|
case 75:
|
|
CRCtriplet(crc, next, -75);
|
|
/* fallthrough */
|
|
case 74:
|
|
CRCtriplet(crc, next, -74);
|
|
/* fallthrough */
|
|
case 73:
|
|
CRCtriplet(crc, next, -73);
|
|
/* fallthrough */
|
|
case 72:
|
|
CRCtriplet(crc, next, -72);
|
|
/* fallthrough */
|
|
case 71:
|
|
CRCtriplet(crc, next, -71);
|
|
/* fallthrough */
|
|
case 70:
|
|
CRCtriplet(crc, next, -70);
|
|
/* fallthrough */
|
|
case 69:
|
|
CRCtriplet(crc, next, -69);
|
|
/* fallthrough */
|
|
case 68:
|
|
CRCtriplet(crc, next, -68);
|
|
/* fallthrough */
|
|
case 67:
|
|
CRCtriplet(crc, next, -67);
|
|
/* fallthrough */
|
|
case 66:
|
|
CRCtriplet(crc, next, -66);
|
|
/* fallthrough */
|
|
case 65:
|
|
CRCtriplet(crc, next, -65);
|
|
/* fallthrough */
|
|
case 64:
|
|
CRCtriplet(crc, next, -64);
|
|
/* fallthrough */
|
|
case 63:
|
|
CRCtriplet(crc, next, -63);
|
|
/* fallthrough */
|
|
case 62:
|
|
CRCtriplet(crc, next, -62);
|
|
/* fallthrough */
|
|
case 61:
|
|
CRCtriplet(crc, next, -61);
|
|
/* fallthrough */
|
|
case 60:
|
|
CRCtriplet(crc, next, -60);
|
|
/* fallthrough */
|
|
case 59:
|
|
CRCtriplet(crc, next, -59);
|
|
/* fallthrough */
|
|
case 58:
|
|
CRCtriplet(crc, next, -58);
|
|
/* fallthrough */
|
|
case 57:
|
|
CRCtriplet(crc, next, -57);
|
|
/* fallthrough */
|
|
case 56:
|
|
CRCtriplet(crc, next, -56);
|
|
/* fallthrough */
|
|
case 55:
|
|
CRCtriplet(crc, next, -55);
|
|
/* fallthrough */
|
|
case 54:
|
|
CRCtriplet(crc, next, -54);
|
|
/* fallthrough */
|
|
case 53:
|
|
CRCtriplet(crc, next, -53);
|
|
/* fallthrough */
|
|
case 52:
|
|
CRCtriplet(crc, next, -52);
|
|
/* fallthrough */
|
|
case 51:
|
|
CRCtriplet(crc, next, -51);
|
|
/* fallthrough */
|
|
case 50:
|
|
CRCtriplet(crc, next, -50);
|
|
/* fallthrough */
|
|
case 49:
|
|
CRCtriplet(crc, next, -49);
|
|
/* fallthrough */
|
|
case 48:
|
|
CRCtriplet(crc, next, -48);
|
|
/* fallthrough */
|
|
case 47:
|
|
CRCtriplet(crc, next, -47);
|
|
/* fallthrough */
|
|
case 46:
|
|
CRCtriplet(crc, next, -46);
|
|
/* fallthrough */
|
|
case 45:
|
|
CRCtriplet(crc, next, -45);
|
|
/* fallthrough */
|
|
case 44:
|
|
CRCtriplet(crc, next, -44);
|
|
/* fallthrough */
|
|
case 43:
|
|
CRCtriplet(crc, next, -43);
|
|
/* fallthrough */
|
|
case 42:
|
|
CRCtriplet(crc, next, -42);
|
|
/* fallthrough */
|
|
case 41:
|
|
CRCtriplet(crc, next, -41);
|
|
/* fallthrough */
|
|
case 40:
|
|
CRCtriplet(crc, next, -40);
|
|
/* fallthrough */
|
|
case 39:
|
|
CRCtriplet(crc, next, -39);
|
|
/* fallthrough */
|
|
case 38:
|
|
CRCtriplet(crc, next, -38);
|
|
/* fallthrough */
|
|
case 37:
|
|
CRCtriplet(crc, next, -37);
|
|
/* fallthrough */
|
|
case 36:
|
|
CRCtriplet(crc, next, -36);
|
|
/* fallthrough */
|
|
case 35:
|
|
CRCtriplet(crc, next, -35);
|
|
/* fallthrough */
|
|
case 34:
|
|
CRCtriplet(crc, next, -34);
|
|
/* fallthrough */
|
|
case 33:
|
|
CRCtriplet(crc, next, -33);
|
|
/* fallthrough */
|
|
case 32:
|
|
CRCtriplet(crc, next, -32);
|
|
/* fallthrough */
|
|
case 31:
|
|
CRCtriplet(crc, next, -31);
|
|
/* fallthrough */
|
|
case 30:
|
|
CRCtriplet(crc, next, -30);
|
|
/* fallthrough */
|
|
case 29:
|
|
CRCtriplet(crc, next, -29);
|
|
/* fallthrough */
|
|
case 28:
|
|
CRCtriplet(crc, next, -28);
|
|
/* fallthrough */
|
|
case 27:
|
|
CRCtriplet(crc, next, -27);
|
|
/* fallthrough */
|
|
case 26:
|
|
CRCtriplet(crc, next, -26);
|
|
/* fallthrough */
|
|
case 25:
|
|
CRCtriplet(crc, next, -25);
|
|
/* fallthrough */
|
|
case 24:
|
|
CRCtriplet(crc, next, -24);
|
|
/* fallthrough */
|
|
case 23:
|
|
CRCtriplet(crc, next, -23);
|
|
/* fallthrough */
|
|
case 22:
|
|
CRCtriplet(crc, next, -22);
|
|
/* fallthrough */
|
|
case 21:
|
|
CRCtriplet(crc, next, -21);
|
|
/* fallthrough */
|
|
case 20:
|
|
CRCtriplet(crc, next, -20);
|
|
/* fallthrough */
|
|
case 19:
|
|
CRCtriplet(crc, next, -19);
|
|
/* fallthrough */
|
|
case 18:
|
|
CRCtriplet(crc, next, -18);
|
|
/* fallthrough */
|
|
case 17:
|
|
CRCtriplet(crc, next, -17);
|
|
/* fallthrough */
|
|
case 16:
|
|
CRCtriplet(crc, next, -16);
|
|
/* fallthrough */
|
|
case 15:
|
|
CRCtriplet(crc, next, -15);
|
|
/* fallthrough */
|
|
case 14:
|
|
CRCtriplet(crc, next, -14);
|
|
/* fallthrough */
|
|
case 13:
|
|
CRCtriplet(crc, next, -13);
|
|
/* fallthrough */
|
|
case 12:
|
|
CRCtriplet(crc, next, -12);
|
|
/* fallthrough */
|
|
case 11:
|
|
CRCtriplet(crc, next, -11);
|
|
/* fallthrough */
|
|
case 10:
|
|
CRCtriplet(crc, next, -10);
|
|
/* fallthrough */
|
|
case 9:
|
|
CRCtriplet(crc, next, -9);
|
|
/* fallthrough */
|
|
case 8:
|
|
CRCtriplet(crc, next, -8);
|
|
/* fallthrough */
|
|
case 7:
|
|
CRCtriplet(crc, next, -7);
|
|
/* fallthrough */
|
|
case 6:
|
|
CRCtriplet(crc, next, -6);
|
|
/* fallthrough */
|
|
case 5:
|
|
CRCtriplet(crc, next, -5);
|
|
/* fallthrough */
|
|
case 4:
|
|
CRCtriplet(crc, next, -4);
|
|
/* fallthrough */
|
|
case 3:
|
|
CRCtriplet(crc, next, -3);
|
|
/* fallthrough */
|
|
case 2:
|
|
CRCtriplet(crc, next, -2);
|
|
/* fallthrough */
|
|
case 1:
|
|
CRCduplet(crc, next, -1); // the final triplet is actually only 2
|
|
//{ CombineCRC(); }
|
|
crc0 = CombineCRC(block_size, crc0, crc1, crc2, next2);
|
|
if (--n > 0) {
|
|
crc1 = crc2 = 0;
|
|
block_size = 128;
|
|
// points to the first byte of the next block
|
|
next0 = next2 + 128;
|
|
next1 = next0 + 128; // from here on all blocks are 128 long
|
|
next2 = next1 + 128;
|
|
}
|
|
/* fallthrough */
|
|
case 0:;
|
|
} while (n > 0);
|
|
}
|
|
next = (const unsigned char*)next2;
|
|
}
|
|
uint64_t count2 = len >> 3; // 216 of less bytes is 27 or less singlets
|
|
len = len & 7;
|
|
next += (count2 * 8);
|
|
switch (count2) {
|
|
case 27:
|
|
CRCsinglet(crc0, next, -27 * 8);
|
|
/* fallthrough */
|
|
case 26:
|
|
CRCsinglet(crc0, next, -26 * 8);
|
|
/* fallthrough */
|
|
case 25:
|
|
CRCsinglet(crc0, next, -25 * 8);
|
|
/* fallthrough */
|
|
case 24:
|
|
CRCsinglet(crc0, next, -24 * 8);
|
|
/* fallthrough */
|
|
case 23:
|
|
CRCsinglet(crc0, next, -23 * 8);
|
|
/* fallthrough */
|
|
case 22:
|
|
CRCsinglet(crc0, next, -22 * 8);
|
|
/* fallthrough */
|
|
case 21:
|
|
CRCsinglet(crc0, next, -21 * 8);
|
|
/* fallthrough */
|
|
case 20:
|
|
CRCsinglet(crc0, next, -20 * 8);
|
|
/* fallthrough */
|
|
case 19:
|
|
CRCsinglet(crc0, next, -19 * 8);
|
|
/* fallthrough */
|
|
case 18:
|
|
CRCsinglet(crc0, next, -18 * 8);
|
|
/* fallthrough */
|
|
case 17:
|
|
CRCsinglet(crc0, next, -17 * 8);
|
|
/* fallthrough */
|
|
case 16:
|
|
CRCsinglet(crc0, next, -16 * 8);
|
|
/* fallthrough */
|
|
case 15:
|
|
CRCsinglet(crc0, next, -15 * 8);
|
|
/* fallthrough */
|
|
case 14:
|
|
CRCsinglet(crc0, next, -14 * 8);
|
|
/* fallthrough */
|
|
case 13:
|
|
CRCsinglet(crc0, next, -13 * 8);
|
|
/* fallthrough */
|
|
case 12:
|
|
CRCsinglet(crc0, next, -12 * 8);
|
|
/* fallthrough */
|
|
case 11:
|
|
CRCsinglet(crc0, next, -11 * 8);
|
|
/* fallthrough */
|
|
case 10:
|
|
CRCsinglet(crc0, next, -10 * 8);
|
|
/* fallthrough */
|
|
case 9:
|
|
CRCsinglet(crc0, next, -9 * 8);
|
|
/* fallthrough */
|
|
case 8:
|
|
CRCsinglet(crc0, next, -8 * 8);
|
|
/* fallthrough */
|
|
case 7:
|
|
CRCsinglet(crc0, next, -7 * 8);
|
|
/* fallthrough */
|
|
case 6:
|
|
CRCsinglet(crc0, next, -6 * 8);
|
|
/* fallthrough */
|
|
case 5:
|
|
CRCsinglet(crc0, next, -5 * 8);
|
|
/* fallthrough */
|
|
case 4:
|
|
CRCsinglet(crc0, next, -4 * 8);
|
|
/* fallthrough */
|
|
case 3:
|
|
CRCsinglet(crc0, next, -3 * 8);
|
|
/* fallthrough */
|
|
case 2:
|
|
CRCsinglet(crc0, next, -2 * 8);
|
|
/* fallthrough */
|
|
case 1:
|
|
CRCsinglet(crc0, next, -1 * 8);
|
|
/* fallthrough */
|
|
case 0:;
|
|
}
|
|
}
|
|
{
|
|
align_to_8(len, crc0, next);
|
|
return (uint32_t)crc0 ^ 0xffffffffu;
|
|
}
|
|
}
|