PostgreSQL Source Code git master
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
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-2025, 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 */
68pg_attribute_target("sse4.2")
69static inline
71pg_comp_crc32c_dispatch(pg_crc32c crc, const void *data, size_t len)
72{
73 if (__builtin_constant_p(len) && len < 32)
74 {
75 const unsigned char *p = data;
76
77 /*
78 * For small constant inputs, inline the computation to avoid a
79 * function call and allow the compiler to unroll loops.
80 */
81#if SIZEOF_VOID_P >= 8
82 for (; len >= 8; p += 8, len -= 8)
83 crc = _mm_crc32_u64(crc, *(const uint64 *) p);
84#endif
85 for (; len >= 4; p += 4, len -= 4)
86 crc = _mm_crc32_u32(crc, *(const uint32 *) p);
87 for (; len > 0; --len)
88 crc = _mm_crc32_u8(crc, *p++);
89 return crc;
90 }
91 else
92 /* Otherwise, use a runtime check for AVX-512 instructions. */
93 return pg_comp_crc32c(crc, data, len);
94}
95
96#elif defined(USE_SSE42_CRC32C_WITH_RUNTIME_CHECK)
97
98/*
99 * Use Intel SSE 4.2 or AVX-512 instructions, but perform a runtime check first
100 * to check that they are available.
101 */
102#define COMP_CRC32C(crc, data, len) \
103 ((crc) = pg_comp_crc32c((crc), (data), (len)))
104#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
105
106extern pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len);
107extern PGDLLIMPORT pg_crc32c (*pg_comp_crc32c) (pg_crc32c crc, const void *data, size_t len);
108extern pg_crc32c pg_comp_crc32c_sse42(pg_crc32c crc, const void *data, size_t len);
109#ifdef USE_AVX512_CRC32C_WITH_RUNTIME_CHECK
110extern pg_crc32c pg_comp_crc32c_avx512(pg_crc32c crc, const void *data, size_t length);
111#endif
112
113#elif defined(USE_ARMV8_CRC32C)
114/* Use ARMv8 CRC Extension instructions. */
115
116#define COMP_CRC32C(crc, data, len) \
117 ((crc) = pg_comp_crc32c_armv8((crc), (data), (len)))
118#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
119
120extern pg_crc32c pg_comp_crc32c_armv8(pg_crc32c crc, const void *data, size_t len);
121
122#elif defined(USE_LOONGARCH_CRC32C)
123/* Use LoongArch CRCC instructions. */
124
125#define COMP_CRC32C(crc, data, len) \
126 ((crc) = pg_comp_crc32c_loongarch((crc), (data), (len)))
127#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
128
129extern pg_crc32c pg_comp_crc32c_loongarch(pg_crc32c crc, const void *data, size_t len);
130
131#elif defined(USE_ARMV8_CRC32C_WITH_RUNTIME_CHECK)
132
133/*
134 * Use ARMv8 instructions, but perform a runtime check first
135 * to check that they are available.
136 */
137#define COMP_CRC32C(crc, data, len) \
138 ((crc) = pg_comp_crc32c((crc), (data), (len)))
139#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
140
141extern pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len);
142extern pg_crc32c (*pg_comp_crc32c) (pg_crc32c crc, const void *data, size_t len);
143extern pg_crc32c pg_comp_crc32c_armv8(pg_crc32c crc, const void *data, size_t len);
144
145#else
146/*
147 * Use slicing-by-8 algorithm.
148 *
149 * On big-endian systems, the intermediate value is kept in reverse byte
150 * order, to avoid byte-swapping during the calculation. FIN_CRC32C reverses
151 * the bytes to the final order.
152 */
153#define COMP_CRC32C(crc, data, len) \
154 ((crc) = pg_comp_crc32c_sb8((crc), (data), (len)))
155#ifdef WORDS_BIGENDIAN
156#define FIN_CRC32C(crc) ((crc) = pg_bswap32(crc) ^ 0xFFFFFFFF)
157#else
158#define FIN_CRC32C(crc) ((crc) ^= 0xFFFFFFFF)
159#endif
160
161extern pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len);
162
163#endif
164
165#endif /* PG_CRC32C_H */
#define PGDLLIMPORT
Definition: c.h:1291
uint64_t uint64
Definition: c.h:503
uint32_t uint32
Definition: c.h:502
#define pg_attribute_target(...)
Definition: c.h:213
uint32 pg_crc32c
Definition: pg_crc32c.h:38
pg_crc32c pg_comp_crc32c_sb8(pg_crc32c crc, const void *data, size_t len)
Definition: pg_crc32c_sb8.c:35
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
pg_attribute_no_sanitize_alignment() pg_attribute_target("sse4.2") pg_crc32c pg_comp_crc32c_sse42(pg_crc32c crc