/* SPDX-License-Identifier: GPL-2.0-only */ /* * GHASH, RISC-V optimized * * Copyright (C) 2023 VRULL GmbH * Copyright (C) 2023 SiFive, Inc. * Copyright 2026 Google LLC */ #include #include static __ro_after_init DEFINE_STATIC_KEY_FALSE(have_zvkg); asmlinkage void ghash_zvkg(u8 accumulator[GHASH_BLOCK_SIZE], const u8 key[GHASH_BLOCK_SIZE], const u8 *data, size_t nblocks); #define ghash_preparekey_arch ghash_preparekey_arch static void ghash_preparekey_arch(struct ghash_key *key, const u8 raw_key[GHASH_BLOCK_SIZE]) { /* Save key in POLYVAL format for fallback */ ghash_key_to_polyval(raw_key, &key->h); /* Save key in GHASH format for zvkg */ memcpy(key->h_raw, raw_key, GHASH_BLOCK_SIZE); } #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_zvkg) && likely(may_use_simd())) { u8 ghash_acc[GHASH_BLOCK_SIZE]; polyval_acc_to_ghash(acc, ghash_acc); kernel_vector_begin(); ghash_zvkg(ghash_acc, key->h_raw, data, nblocks); kernel_vector_end(); ghash_acc_to_polyval(ghash_acc, acc); memzero_explicit(ghash_acc, sizeof(ghash_acc)); } 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 (riscv_isa_extension_available(NULL, ZVKG) && riscv_vector_vlen() >= 128) static_branch_enable(&have_zvkg); }