1 // SPDX-License-Identifier: GPL-2.0-only
3 * Accelerated CRC32(C) using ARM CRC, NEON and Crypto Extensions instructions
8 #include <linux/cpufeature.h>
9 #include <linux/crc32.h>
10 #include <linux/init.h>
11 #include <linux/kernel.h>
12 #include <linux/module.h>
13 #include <linux/string.h>
15 #include <crypto/internal/simd.h>
17 #include <asm/hwcap.h>
21 static DEFINE_STATIC_KEY_FALSE(have_crc32);
22 static DEFINE_STATIC_KEY_FALSE(have_pmull);
24 #define PMULL_MIN_LEN 64 /* min size of buffer for pmull functions */
26 asmlinkage u32 crc32_pmull_le(const u8 buf[], u32 len, u32 init_crc);
27 asmlinkage u32 crc32_armv8_le(u32 init_crc, const u8 buf[], u32 len);
29 asmlinkage u32 crc32c_pmull_le(const u8 buf[], u32 len, u32 init_crc);
30 asmlinkage u32 crc32c_armv8_le(u32 init_crc, const u8 buf[], u32 len);
32 static u32 crc32_le_scalar(u32 crc, const u8 *p, size_t len)
34 if (static_branch_likely(&have_crc32))
35 return crc32_armv8_le(crc, p, len);
36 return crc32_le_base(crc, p, len);
39 u32 crc32_le_arch(u32 crc, const u8 *p, size_t len)
41 if (len >= PMULL_MIN_LEN + 15 &&
42 static_branch_likely(&have_pmull) && crypto_simd_usable()) {
43 size_t n = -(uintptr_t)p & 15;
45 /* align p to 16-byte boundary */
47 crc = crc32_le_scalar(crc, p, n);
51 n = round_down(len, 16);
53 crc = crc32_pmull_le(p, n, crc);
58 return crc32_le_scalar(crc, p, len);
60 EXPORT_SYMBOL(crc32_le_arch);
62 static u32 crc32c_le_scalar(u32 crc, const u8 *p, size_t len)
64 if (static_branch_likely(&have_crc32))
65 return crc32c_armv8_le(crc, p, len);
66 return crc32c_le_base(crc, p, len);
69 u32 crc32c_le_arch(u32 crc, const u8 *p, size_t len)
71 if (len >= PMULL_MIN_LEN + 15 &&
72 static_branch_likely(&have_pmull) && crypto_simd_usable()) {
73 size_t n = -(uintptr_t)p & 15;
75 /* align p to 16-byte boundary */
77 crc = crc32c_le_scalar(crc, p, n);
81 n = round_down(len, 16);
83 crc = crc32c_pmull_le(p, n, crc);
88 return crc32c_le_scalar(crc, p, len);
90 EXPORT_SYMBOL(crc32c_le_arch);
92 u32 crc32_be_arch(u32 crc, const u8 *p, size_t len)
94 return crc32_be_base(crc, p, len);
96 EXPORT_SYMBOL(crc32_be_arch);
98 static int __init crc32_arm_init(void)
100 if (elf_hwcap2 & HWCAP2_CRC32)
101 static_branch_enable(&have_crc32);
102 if (elf_hwcap2 & HWCAP2_PMULL)
103 static_branch_enable(&have_pmull);
106 arch_initcall(crc32_arm_init);
108 static void __exit crc32_arm_exit(void)
111 module_exit(crc32_arm_exit);
113 u32 crc32_optimizations(void)
115 if (elf_hwcap2 & (HWCAP2_CRC32 | HWCAP2_PMULL))
116 return CRC32_LE_OPTIMIZATION | CRC32C_OPTIMIZATION;
119 EXPORT_SYMBOL(crc32_optimizations);
122 MODULE_DESCRIPTION("Accelerated CRC32(C) using ARM CRC, NEON and Crypto Extensions");
123 MODULE_LICENSE("GPL v2");