/* SPDX-License-Identifier: GPL-2.0-or-later */ /* * GHASH, arm optimized * * Copyright 2026 Google LLC */ #include #include #include static __ro_after_init DEFINE_STATIC_KEY_FALSE(have_neon); void pmull_ghash_update_p8(size_t blocks, struct polyval_elem *dg, const u8 *src, const struct polyval_elem *h); #define ghash_blocks_arch ghash_blocks_arch static void ghash_blocks_arch(struct polyval_elem *acc, const struct ghash_key *key, const u8 *data, size_t nblocks) { if (static_branch_likely(&have_neon) && may_use_simd()) { do { /* Allow rescheduling every 4 KiB. */ size_t n = min_t(size_t, nblocks, 4096 / GHASH_BLOCK_SIZE); scoped_ksimd() pmull_ghash_update_p8(n, acc, data, &key->h); data += n * GHASH_BLOCK_SIZE; nblocks -= n; } while (nblocks); } else { ghash_blocks_generic(acc, &key->h, data, nblocks); } } #define gf128hash_mod_init_arch gf128hash_mod_init_arch static void gf128hash_mod_init_arch(void) { if (elf_hwcap & HWCAP_NEON) static_branch_enable(&have_neon); }