|  | /* SPDX-License-Identifier: Apache-2.0 OR BSD-2-Clause */ | 
|  | // | 
|  | // This file is dual-licensed, meaning that you can use it under your | 
|  | // choice of either of the following two licenses: | 
|  | // | 
|  | // Copyright 2023 The OpenSSL Project Authors. All Rights Reserved. | 
|  | // | 
|  | // Licensed under the Apache License 2.0 (the "License"). You can obtain | 
|  | // a copy in the file LICENSE in the source distribution or at | 
|  | // https://www.openssl.org/source/license.html | 
|  | // | 
|  | // or | 
|  | // | 
|  | // Copyright (c) 2023, Christoph Müllner <christoph.muellner@vrull.eu> | 
|  | // Copyright (c) 2023, Phoebe Chen <phoebe.chen@sifive.com> | 
|  | // Copyright (c) 2023, Jerry Shih <jerry.shih@sifive.com> | 
|  | // Copyright 2024 Google LLC | 
|  | // All rights reserved. | 
|  | // | 
|  | // Redistribution and use in source and binary forms, with or without | 
|  | // modification, are permitted provided that the following conditions | 
|  | // are met: | 
|  | // 1. Redistributions of source code must retain the above copyright | 
|  | //    notice, this list of conditions and the following disclaimer. | 
|  | // 2. Redistributions in binary form must reproduce the above copyright | 
|  | //    notice, this list of conditions and the following disclaimer in the | 
|  | //    documentation and/or other materials provided with the distribution. | 
|  | // | 
|  | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | 
|  | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | 
|  | // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | 
|  | // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | 
|  | // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
|  | // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
|  | // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|  | // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  |  | 
|  | // This file contains macros that are shared by the other aes-*.S files.  The | 
|  | // generated code of these macros depends on the following RISC-V extensions: | 
|  | // - RV64I | 
|  | // - RISC-V Vector ('V') with VLEN >= 128 | 
|  | // - RISC-V Vector AES block cipher extension ('Zvkned') | 
|  |  | 
|  | // Loads the AES round keys from \keyp into vector registers and jumps to code | 
|  | // specific to the length of the key.  Specifically: | 
|  | //   - If AES-128, loads round keys into v1-v11 and jumps to \label128. | 
|  | //   - If AES-192, loads round keys into v1-v13 and jumps to \label192. | 
|  | //   - If AES-256, loads round keys into v1-v15 and continues onwards. | 
|  | // | 
|  | // Also sets vl=4 and vtype=e32,m1,ta,ma.  Clobbers t0 and t1. | 
|  | .macro	aes_begin	keyp, label128, label192 | 
|  | lwu		t0, 480(\keyp)	// t0 = key length in bytes | 
|  | li		t1, 24		// t1 = key length for AES-192 | 
|  | vsetivli	zero, 4, e32, m1, ta, ma | 
|  | vle32.v		v1, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v2, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v3, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v4, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v5, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v6, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v7, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v8, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v9, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v10, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v11, (\keyp) | 
|  | blt		t0, t1, \label128	// If AES-128, goto label128. | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v12, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v13, (\keyp) | 
|  | beq		t0, t1, \label192	// If AES-192, goto label192. | 
|  | // Else, it's AES-256. | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v14, (\keyp) | 
|  | addi		\keyp, \keyp, 16 | 
|  | vle32.v		v15, (\keyp) | 
|  | .endm | 
|  |  | 
|  | // Encrypts \data using zvkned instructions, using the round keys loaded into | 
|  | // v1-v11 (for AES-128), v1-v13 (for AES-192), or v1-v15 (for AES-256).  \keylen | 
|  | // is the AES key length in bits.  vl and vtype must already be set | 
|  | // appropriately.  Note that if vl > 4, multiple blocks are encrypted. | 
|  | .macro	aes_encrypt	data, keylen | 
|  | vaesz.vs	\data, v1 | 
|  | vaesem.vs	\data, v2 | 
|  | vaesem.vs	\data, v3 | 
|  | vaesem.vs	\data, v4 | 
|  | vaesem.vs	\data, v5 | 
|  | vaesem.vs	\data, v6 | 
|  | vaesem.vs	\data, v7 | 
|  | vaesem.vs	\data, v8 | 
|  | vaesem.vs	\data, v9 | 
|  | vaesem.vs	\data, v10 | 
|  | .if \keylen == 128 | 
|  | vaesef.vs	\data, v11 | 
|  | .elseif \keylen == 192 | 
|  | vaesem.vs	\data, v11 | 
|  | vaesem.vs	\data, v12 | 
|  | vaesef.vs	\data, v13 | 
|  | .else | 
|  | vaesem.vs	\data, v11 | 
|  | vaesem.vs	\data, v12 | 
|  | vaesem.vs	\data, v13 | 
|  | vaesem.vs	\data, v14 | 
|  | vaesef.vs	\data, v15 | 
|  | .endif | 
|  | .endm | 
|  |  | 
|  | // Same as aes_encrypt, but decrypts instead of encrypts. | 
|  | .macro	aes_decrypt	data, keylen | 
|  | .if \keylen == 128 | 
|  | vaesz.vs	\data, v11 | 
|  | .elseif \keylen == 192 | 
|  | vaesz.vs	\data, v13 | 
|  | vaesdm.vs	\data, v12 | 
|  | vaesdm.vs	\data, v11 | 
|  | .else | 
|  | vaesz.vs	\data, v15 | 
|  | vaesdm.vs	\data, v14 | 
|  | vaesdm.vs	\data, v13 | 
|  | vaesdm.vs	\data, v12 | 
|  | vaesdm.vs	\data, v11 | 
|  | .endif | 
|  | vaesdm.vs	\data, v10 | 
|  | vaesdm.vs	\data, v9 | 
|  | vaesdm.vs	\data, v8 | 
|  | vaesdm.vs	\data, v7 | 
|  | vaesdm.vs	\data, v6 | 
|  | vaesdm.vs	\data, v5 | 
|  | vaesdm.vs	\data, v4 | 
|  | vaesdm.vs	\data, v3 | 
|  | vaesdm.vs	\data, v2 | 
|  | vaesdf.vs	\data, v1 | 
|  | .endm | 
|  |  | 
|  | // Expands to aes_encrypt or aes_decrypt according to \enc, which is 1 or 0. | 
|  | .macro	aes_crypt	data, enc, keylen | 
|  | .if \enc | 
|  | aes_encrypt	\data, \keylen | 
|  | .else | 
|  | aes_decrypt	\data, \keylen | 
|  | .endif | 
|  | .endm |