PostgreSQL Source Code git master
Loading...
Searching...
No Matches
pg_crc32c.h
Go to the documentation of this file.
1/*-------------------------------------------------------------------------
2 *
3 * pg_crc32c.h
4 * Routines for computing CRC-32C checksums.
5 *
6 * The speed of CRC-32C calculation has a big impact on performance, so we
7 * jump through some hoops to get the best implementation for each
8 * platform. Some CPU architectures have special instructions for speeding
9 * up CRC calculations (e.g. Intel SSE 4.2), on other platforms we use the
10 * Slicing-by-8 algorithm which uses lookup tables.
11 *
12 * The public interface consists of four macros:
13 *
14 * INIT_CRC32C(crc)
15 * Initialize a CRC accumulator
16 *
17 * COMP_CRC32C(crc, data, len)
18 * Accumulate some (more) bytes into a CRC
19 *
20 * FIN_CRC32C(crc)
21 * Finish a CRC calculation
22 *
23 * EQ_CRC32C(c1, c2)
24 * Check for equality of two CRCs.
25 *
26 * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
27 * Portions Copyright (c) 1994, Regents of the University of California
28 *
29 * src/include/port/pg_crc32c.h
30 *
31 *-------------------------------------------------------------------------
32 */
33#ifndef PG_CRC32C_H
34#define PG_CRC32C_H
35
36#include "port/pg_bswap.h"
37
39
40/* The INIT and EQ macros are the same for all implementations. */
41#define INIT_CRC32C(crc) ((crc) = 0xFFFFFFFF)
42#define EQ_CRC32C(c1, c2) ((c1) == (c2))
43
44#if defined(USE_SSE42_CRC32C)
45/*
46 * Use either Intel SSE 4.2 or AVX-512 instructions. We don't need a runtime check
47 * for SSE 4.2, so we can inline those in some cases.
48 */
49
50#include <nmmintrin.h>
51
52#define COMP_CRC32C(crc, data, len) \
53 ((crc) = pg_comp_crc32c_dispatch((crc), (data), (len)))
54#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
55
56extern pg_crc32c (*pg_comp_crc32c) (pg_crc32c crc, const void *data, size_t len);
57extern pg_crc32c pg_comp_crc32c_sse42(pg_crc32c crc, const void *data, size_t len);
58#ifdef USE_AVX512_CRC32C_WITH_RUNTIME_CHECK
59extern pg_crc32c pg_comp_crc32c_avx512(pg_crc32c crc, const void *data, size_t len);
60#endif
61
62/*
63 * We can only get here if the host compiler targets SSE 4.2, but on some
64 * systems gcc and clang don't have the same built-in targets, so we still
65 * must use a function attribute here to accommodate "--with-llvm" builds.
66 */
72{
73#ifdef HAVE_PG_INTEGER_CONSTANT_P
74 if (pg_integer_constant_p(len) && len < 32)
75 {
76 const unsigned char *p = (const unsigned char *) data;
77
78 /*
79 * For small constant inputs, inline the computation to avoid a
80 * function call and allow the compiler to unroll loops.
81 */
82#if SIZEOF_VOID_P >= 8
83 for (; len >= 8; p += 8, len -= 8)
84 crc = _mm_crc32_u64(crc, *(const uint64 *) p);
85#endif
86 for (; len >= 4; p += 4, len -= 4)
87 crc = _mm_crc32_u32(crc, *(const uint32 *) p);
88 for (; len > 0; --len)
89 crc = _mm_crc32_u8(crc, *p++);
90 return crc;
91 }
92 else
93 /* Otherwise, use a runtime check for AVX-512 instructions. */
94 return pg_comp_crc32c(crc, data, len);
95#else
96 return pg_comp_crc32c(crc, data, len);
97#endif /* HAVE_PG_INTEGER_CONSTANT_P */
98}
99
100#elif defined(USE_SSE42_CRC32C_WITH_RUNTIME_CHECK)
101
102/*
103 * Use Intel SSE 4.2 or AVX-512 instructions, but perform a runtime check first
104 * to check that they are available.
105 */
106#define COMP_CRC32C(crc, data, len) \
107 ((crc) = pg_comp_crc32c((crc), (data), (len)))
108#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
109
110extern pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len);
111extern pg_crc32c (*pg_comp_crc32c) (pg_crc32c crc, const void *data, size_t len);
112extern pg_crc32c pg_comp_crc32c_sse42(pg_crc32c crc, const void *data, size_t len);
113#ifdef USE_AVX512_CRC32C_WITH_RUNTIME_CHECK
114extern pg_crc32c pg_comp_crc32c_avx512(pg_crc32c crc, const void *data, size_t len);
115#endif
116
117#elif defined(USE_ARMV8_CRC32C)
118/*
119 * Use either ARMv8 CRC Extension or CRYPTO Extension (PMULL) instructions.
120 */
121#ifdef HAVE_PG_INTEGER_CONSTANT_P
122/*
123 * We don't need a runtime check for CRC, so for constant inputs, where
124 * we assume the input is small, we can avoid an indirect function call.
125 */
126#define COMP_CRC32C(crc, data, len) \
127 ((crc) = pg_integer_constant_p(len) ? \
128 pg_comp_crc32c_armv8((crc), (data), (len)) : \
129 pg_comp_crc32c((crc), (data), (len)))
130#else
131#define COMP_CRC32C(crc, data, len) \
132 ((crc) = pg_comp_crc32c((crc), (data), (len)))
133#endif /* HAVE_PG_INTEGER_CONSTANT_P */
134
135#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
136
137extern pg_crc32c (*pg_comp_crc32c) (pg_crc32c crc, const void *data, size_t len);
138extern pg_crc32c pg_comp_crc32c_armv8(pg_crc32c crc, const void *data, size_t len);
139#ifdef USE_PMULL_CRC32C_WITH_RUNTIME_CHECK
140extern pg_crc32c pg_comp_crc32c_pmull(pg_crc32c crc, const void *data, size_t len);
141#endif
142
143#elif defined(USE_LOONGARCH_CRC32C)
144/* Use LoongArch CRCC instructions. */
145
146#define COMP_CRC32C(crc, data, len) \
147 ((crc) = pg_comp_crc32c_loongarch((crc), (data), (len)))
148#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
149
150extern pg_crc32c pg_comp_crc32c_loongarch(pg_crc32c crc, const void *data, size_t len);
151
152#elif defined(USE_ARMV8_CRC32C_WITH_RUNTIME_CHECK)
153
154/*
155 * Use either ARMv8 CRC Extension or CRYPTO Extension (PMULL) instructions,
156 * but perform a runtime check first to check that they are available.
157 */
158#define COMP_CRC32C(crc, data, len) \
159 ((crc) = pg_comp_crc32c((crc), (data), (len)))
160#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
161
162extern pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len);
163extern pg_crc32c (*pg_comp_crc32c) (pg_crc32c crc, const void *data, size_t len);
164extern pg_crc32c pg_comp_crc32c_armv8(pg_crc32c crc, const void *data, size_t len);
165#ifdef USE_PMULL_CRC32C_WITH_RUNTIME_CHECK
166extern pg_crc32c pg_comp_crc32c_pmull(pg_crc32c crc, const void *data, size_t len);
167#endif
168
169#else
170/*
171 * Use slicing-by-8 algorithm.
172 *
173 * On big-endian systems, the intermediate value is kept in reverse byte
174 * order, to avoid byte-swapping during the calculation. FIN_CRC32C reverses
175 * the bytes to the final order.
176 */
177#define COMP_CRC32C(crc, data, len) \
178 ((crc) = pg_comp_crc32c_sb8((crc), (data), (len)))
179#ifdef WORDS_BIGENDIAN
180#define FIN_CRC32C(crc) ((crc) = pg_bswap32(crc) ^ 0xFFFFFFFF)
181#else
182#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
183#endif
184
185extern pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len);
186
187#endif
188
189#endif /* PG_CRC32C_H */
#define pg_attribute_no_sanitize_alignment()
Definition c.h:215
uint64_t uint64
Definition c.h:625
uint32_t uint32
Definition c.h:624
#define pg_attribute_target(...)
Definition c.h:238
uint32 pg_crc32c
Definition pg_crc32c.h:38
pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len)
pg_crc32c pg_comp_crc32c_armv8(pg_crc32c crc, const void *data, size_t len)
pg_crc32c(* pg_comp_crc32c)(pg_crc32c crc, const void *data, size_t len)
pg_crc32c pg_comp_crc32c_loongarch(pg_crc32c crc, const void *data, size_t len)
const void size_t len
const void * data
return crc
static int fb(int x)