2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
2012-08-21 03:58:13 -07:00
/* Glue code for AES encryption optimized for sparc64 crypto opcodes.
*
* This is based largely upon arch / x86 / crypto / aesni - intel_glue . c
*
* Copyright ( C ) 2008 , Intel Corp .
* Author : Huang Ying < ying . huang @ intel . com >
*
* Added RFC4106 AES - GCM support for 128 - bit keys under the AEAD
* interface for 64 - bit kernels .
* Authors : Adrian Hoban < adrian . hoban @ intel . com >
* Gabriele Paoloni < gabriele . paoloni @ intel . com >
* Tadeusz Struk ( tadeusz . struk @ intel . com )
* Aidan O ' Mahony ( aidan . o . mahony @ intel . com )
* Copyright ( c ) 2010 , Intel Corporation .
*/
2012-09-15 09:17:10 -07:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2012-08-21 03:58:13 -07:00
# include <linux/crypto.h>
# include <linux/init.h>
# include <linux/module.h>
# include <linux/mm.h>
# include <linux/types.h>
# include <crypto/algapi.h>
# include <crypto/aes.h>
2019-10-11 21:38:48 -07:00
# include <crypto/internal/skcipher.h>
2012-08-21 03:58:13 -07:00
# include <asm/fpumacro.h>
# include <asm/pstate.h>
# include <asm/elf.h>
2012-09-15 09:06:30 -07:00
# include "opcodes.h"
2012-08-29 12:50:16 -07:00
struct aes_ops {
void ( * encrypt ) ( const u64 * key , const u32 * input , u32 * output ) ;
void ( * decrypt ) ( const u64 * key , const u32 * input , u32 * output ) ;
void ( * load_encrypt_keys ) ( const u64 * key ) ;
void ( * load_decrypt_keys ) ( const u64 * key ) ;
void ( * ecb_encrypt ) ( const u64 * key , const u64 * input , u64 * output ,
unsigned int len ) ;
void ( * ecb_decrypt ) ( const u64 * key , const u64 * input , u64 * output ,
unsigned int len ) ;
void ( * cbc_encrypt ) ( const u64 * key , const u64 * input , u64 * output ,
unsigned int len , u64 * iv ) ;
void ( * cbc_decrypt ) ( const u64 * key , const u64 * input , u64 * output ,
unsigned int len , u64 * iv ) ;
2012-08-29 14:49:23 -07:00
void ( * ctr_crypt ) ( const u64 * key , const u64 * input , u64 * output ,
unsigned int len , u64 * iv ) ;
2012-08-29 12:50:16 -07:00
} ;
2012-08-21 03:58:13 -07:00
struct crypto_sparc64_aes_ctx {
2012-08-29 12:50:16 -07:00
struct aes_ops * ops ;
2012-08-21 03:58:13 -07:00
u64 key [ AES_MAX_KEYLENGTH / sizeof ( u64 ) ] ;
u32 key_length ;
u32 expanded_key_length ;
} ;
2012-08-29 12:50:16 -07:00
extern void aes_sparc64_encrypt_128 ( const u64 * key , const u32 * input ,
u32 * output ) ;
extern void aes_sparc64_encrypt_192 ( const u64 * key , const u32 * input ,
u32 * output ) ;
extern void aes_sparc64_encrypt_256 ( const u64 * key , const u32 * input ,
u32 * output ) ;
extern void aes_sparc64_decrypt_128 ( const u64 * key , const u32 * input ,
u32 * output ) ;
extern void aes_sparc64_decrypt_192 ( const u64 * key , const u32 * input ,
u32 * output ) ;
extern void aes_sparc64_decrypt_256 ( const u64 * key , const u32 * input ,
u32 * output ) ;
extern void aes_sparc64_load_encrypt_keys_128 ( const u64 * key ) ;
extern void aes_sparc64_load_encrypt_keys_192 ( const u64 * key ) ;
extern void aes_sparc64_load_encrypt_keys_256 ( const u64 * key ) ;
extern void aes_sparc64_load_decrypt_keys_128 ( const u64 * key ) ;
extern void aes_sparc64_load_decrypt_keys_192 ( const u64 * key ) ;
extern void aes_sparc64_load_decrypt_keys_256 ( const u64 * key ) ;
extern void aes_sparc64_ecb_encrypt_128 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ) ;
extern void aes_sparc64_ecb_encrypt_192 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ) ;
extern void aes_sparc64_ecb_encrypt_256 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ) ;
extern void aes_sparc64_ecb_decrypt_128 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ) ;
extern void aes_sparc64_ecb_decrypt_192 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ) ;
extern void aes_sparc64_ecb_decrypt_256 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ) ;
extern void aes_sparc64_cbc_encrypt_128 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
extern void aes_sparc64_cbc_encrypt_192 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
extern void aes_sparc64_cbc_encrypt_256 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
extern void aes_sparc64_cbc_decrypt_128 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
extern void aes_sparc64_cbc_decrypt_192 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
extern void aes_sparc64_cbc_decrypt_256 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
2012-08-29 14:49:23 -07:00
extern void aes_sparc64_ctr_crypt_128 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
extern void aes_sparc64_ctr_crypt_192 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
extern void aes_sparc64_ctr_crypt_256 ( const u64 * key , const u64 * input ,
u64 * output , unsigned int len ,
u64 * iv ) ;
2014-05-16 23:26:06 +02:00
static struct aes_ops aes128_ops = {
2012-08-29 12:50:16 -07:00
. encrypt = aes_sparc64_encrypt_128 ,
. decrypt = aes_sparc64_decrypt_128 ,
. load_encrypt_keys = aes_sparc64_load_encrypt_keys_128 ,
. load_decrypt_keys = aes_sparc64_load_decrypt_keys_128 ,
. ecb_encrypt = aes_sparc64_ecb_encrypt_128 ,
. ecb_decrypt = aes_sparc64_ecb_decrypt_128 ,
. cbc_encrypt = aes_sparc64_cbc_encrypt_128 ,
. cbc_decrypt = aes_sparc64_cbc_decrypt_128 ,
2012-08-29 14:49:23 -07:00
. ctr_crypt = aes_sparc64_ctr_crypt_128 ,
2012-08-29 12:50:16 -07:00
} ;
2014-05-16 23:26:06 +02:00
static struct aes_ops aes192_ops = {
2012-08-29 12:50:16 -07:00
. encrypt = aes_sparc64_encrypt_192 ,
. decrypt = aes_sparc64_decrypt_192 ,
. load_encrypt_keys = aes_sparc64_load_encrypt_keys_192 ,
. load_decrypt_keys = aes_sparc64_load_decrypt_keys_192 ,
. ecb_encrypt = aes_sparc64_ecb_encrypt_192 ,
. ecb_decrypt = aes_sparc64_ecb_decrypt_192 ,
. cbc_encrypt = aes_sparc64_cbc_encrypt_192 ,
. cbc_decrypt = aes_sparc64_cbc_decrypt_192 ,
2012-08-29 14:49:23 -07:00
. ctr_crypt = aes_sparc64_ctr_crypt_192 ,
2012-08-29 12:50:16 -07:00
} ;
2014-05-16 23:26:06 +02:00
static struct aes_ops aes256_ops = {
2012-08-29 12:50:16 -07:00
. encrypt = aes_sparc64_encrypt_256 ,
. decrypt = aes_sparc64_decrypt_256 ,
. load_encrypt_keys = aes_sparc64_load_encrypt_keys_256 ,
. load_decrypt_keys = aes_sparc64_load_decrypt_keys_256 ,
. ecb_encrypt = aes_sparc64_ecb_encrypt_256 ,
. ecb_decrypt = aes_sparc64_ecb_decrypt_256 ,
. cbc_encrypt = aes_sparc64_cbc_encrypt_256 ,
. cbc_decrypt = aes_sparc64_cbc_decrypt_256 ,
2012-08-29 14:49:23 -07:00
. ctr_crypt = aes_sparc64_ctr_crypt_256 ,
2012-08-29 12:50:16 -07:00
} ;
2012-08-21 03:58:13 -07:00
extern void aes_sparc64_key_expand ( const u32 * in_key , u64 * output_key ,
unsigned int key_len ) ;
static int aes_set_key ( struct crypto_tfm * tfm , const u8 * in_key ,
unsigned int key_len )
{
struct crypto_sparc64_aes_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
switch ( key_len ) {
case AES_KEYSIZE_128 :
ctx - > expanded_key_length = 0xb0 ;
2012-08-29 12:50:16 -07:00
ctx - > ops = & aes128_ops ;
2012-08-21 03:58:13 -07:00
break ;
case AES_KEYSIZE_192 :
ctx - > expanded_key_length = 0xd0 ;
2012-08-29 12:50:16 -07:00
ctx - > ops = & aes192_ops ;
2012-08-21 03:58:13 -07:00
break ;
case AES_KEYSIZE_256 :
ctx - > expanded_key_length = 0xf0 ;
2012-08-29 12:50:16 -07:00
ctx - > ops = & aes256_ops ;
2012-08-21 03:58:13 -07:00
break ;
default :
return - EINVAL ;
}
aes_sparc64_key_expand ( ( const u32 * ) in_key , & ctx - > key [ 0 ] , key_len ) ;
ctx - > key_length = key_len ;
return 0 ;
}
2019-10-11 21:38:48 -07:00
static int aes_set_key_skcipher ( struct crypto_skcipher * tfm , const u8 * in_key ,
unsigned int key_len )
{
return aes_set_key ( crypto_skcipher_tfm ( tfm ) , in_key , key_len ) ;
}
2019-07-02 21:41:20 +02:00
static void crypto_aes_encrypt ( struct crypto_tfm * tfm , u8 * dst , const u8 * src )
2012-08-21 03:58:13 -07:00
{
struct crypto_sparc64_aes_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2012-08-29 12:50:16 -07:00
ctx - > ops - > encrypt ( & ctx - > key [ 0 ] , ( const u32 * ) src , ( u32 * ) dst ) ;
2012-08-21 03:58:13 -07:00
}
2019-07-02 21:41:20 +02:00
static void crypto_aes_decrypt ( struct crypto_tfm * tfm , u8 * dst , const u8 * src )
2012-08-21 03:58:13 -07:00
{
struct crypto_sparc64_aes_ctx * ctx = crypto_tfm_ctx ( tfm ) ;
2012-08-29 12:50:16 -07:00
ctx - > ops - > decrypt ( & ctx - > key [ 0 ] , ( const u32 * ) src , ( u32 * ) dst ) ;
2012-08-21 03:58:13 -07:00
}
2019-10-11 21:38:48 -07:00
static int ecb_encrypt ( struct skcipher_request * req )
2012-08-21 03:58:13 -07:00
{
2019-10-11 21:38:48 -07:00
struct crypto_skcipher * tfm = crypto_skcipher_reqtfm ( req ) ;
const struct crypto_sparc64_aes_ctx * ctx = crypto_skcipher_ctx ( tfm ) ;
struct skcipher_walk walk ;
unsigned int nbytes ;
2012-08-21 03:58:13 -07:00
int err ;
2019-10-11 21:38:48 -07:00
err = skcipher_walk_virt ( & walk , req , true ) ;
if ( err )
return err ;
2012-08-21 03:58:13 -07:00
2012-08-29 12:50:16 -07:00
ctx - > ops - > load_encrypt_keys ( & ctx - > key [ 0 ] ) ;
2019-10-11 21:38:48 -07:00
while ( ( nbytes = walk . nbytes ) ! = 0 ) {
ctx - > ops - > ecb_encrypt ( & ctx - > key [ 0 ] , walk . src . virt . addr ,
walk . dst . virt . addr ,
round_down ( nbytes , AES_BLOCK_SIZE ) ) ;
err = skcipher_walk_done ( & walk , nbytes % AES_BLOCK_SIZE ) ;
2012-08-21 03:58:13 -07:00
}
fprs_write ( 0 ) ;
return err ;
}
2019-10-11 21:38:48 -07:00
static int ecb_decrypt ( struct skcipher_request * req )
2012-08-21 03:58:13 -07:00
{
2019-10-11 21:38:48 -07:00
struct crypto_skcipher * tfm = crypto_skcipher_reqtfm ( req ) ;
const struct crypto_sparc64_aes_ctx * ctx = crypto_skcipher_ctx ( tfm ) ;
const u64 * key_end ;
struct skcipher_walk walk ;
unsigned int nbytes ;
2012-08-21 03:58:13 -07:00
int err ;
2019-10-11 21:38:48 -07:00
err = skcipher_walk_virt ( & walk , req , true ) ;
if ( err )
return err ;
2012-08-21 03:58:13 -07:00
2012-08-29 12:50:16 -07:00
ctx - > ops - > load_decrypt_keys ( & ctx - > key [ 0 ] ) ;
2012-08-21 03:58:13 -07:00
key_end = & ctx - > key [ ctx - > expanded_key_length / sizeof ( u64 ) ] ;
2019-10-11 21:38:48 -07:00
while ( ( nbytes = walk . nbytes ) ! = 0 ) {
ctx - > ops - > ecb_decrypt ( key_end , walk . src . virt . addr ,
walk . dst . virt . addr ,
round_down ( nbytes , AES_BLOCK_SIZE ) ) ;
err = skcipher_walk_done ( & walk , nbytes % AES_BLOCK_SIZE ) ;
2012-08-21 03:58:13 -07:00
}
fprs_write ( 0 ) ;
return err ;
}
2019-10-11 21:38:48 -07:00
static int cbc_encrypt ( struct skcipher_request * req )
2012-08-21 03:58:13 -07:00
{
2019-10-11 21:38:48 -07:00
struct crypto_skcipher * tfm = crypto_skcipher_reqtfm ( req ) ;
const struct crypto_sparc64_aes_ctx * ctx = crypto_skcipher_ctx ( tfm ) ;
struct skcipher_walk walk ;
unsigned int nbytes ;
2012-08-21 03:58:13 -07:00
int err ;
2019-10-11 21:38:48 -07:00
err = skcipher_walk_virt ( & walk , req , true ) ;
if ( err )
return err ;
2012-08-21 03:58:13 -07:00
2012-08-29 12:50:16 -07:00
ctx - > ops - > load_encrypt_keys ( & ctx - > key [ 0 ] ) ;
2019-10-11 21:38:48 -07:00
while ( ( nbytes = walk . nbytes ) ! = 0 ) {
ctx - > ops - > cbc_encrypt ( & ctx - > key [ 0 ] , walk . src . virt . addr ,
walk . dst . virt . addr ,
round_down ( nbytes , AES_BLOCK_SIZE ) ,
walk . iv ) ;
err = skcipher_walk_done ( & walk , nbytes % AES_BLOCK_SIZE ) ;
2012-08-21 03:58:13 -07:00
}
fprs_write ( 0 ) ;
return err ;
}
2019-10-11 21:38:48 -07:00
static int cbc_decrypt ( struct skcipher_request * req )
2012-08-21 03:58:13 -07:00
{
2019-10-11 21:38:48 -07:00
struct crypto_skcipher * tfm = crypto_skcipher_reqtfm ( req ) ;
const struct crypto_sparc64_aes_ctx * ctx = crypto_skcipher_ctx ( tfm ) ;
const u64 * key_end ;
struct skcipher_walk walk ;
unsigned int nbytes ;
2012-08-21 03:58:13 -07:00
int err ;
2019-10-11 21:38:48 -07:00
err = skcipher_walk_virt ( & walk , req , true ) ;
if ( err )
return err ;
2012-08-21 03:58:13 -07:00
2012-08-29 12:50:16 -07:00
ctx - > ops - > load_decrypt_keys ( & ctx - > key [ 0 ] ) ;
2012-08-21 03:58:13 -07:00
key_end = & ctx - > key [ ctx - > expanded_key_length / sizeof ( u64 ) ] ;
2019-10-11 21:38:48 -07:00
while ( ( nbytes = walk . nbytes ) ! = 0 ) {
ctx - > ops - > cbc_decrypt ( key_end , walk . src . virt . addr ,
walk . dst . virt . addr ,
round_down ( nbytes , AES_BLOCK_SIZE ) ,
walk . iv ) ;
err = skcipher_walk_done ( & walk , nbytes % AES_BLOCK_SIZE ) ;
2012-08-21 03:58:13 -07:00
}
fprs_write ( 0 ) ;
return err ;
}
2019-10-11 21:38:48 -07:00
static void ctr_crypt_final ( const struct crypto_sparc64_aes_ctx * ctx ,
struct skcipher_walk * walk )
2012-12-19 15:20:23 -08:00
{
u8 * ctrblk = walk - > iv ;
u64 keystream [ AES_BLOCK_SIZE / sizeof ( u64 ) ] ;
u8 * src = walk - > src . virt . addr ;
u8 * dst = walk - > dst . virt . addr ;
unsigned int nbytes = walk - > nbytes ;
ctx - > ops - > ecb_encrypt ( & ctx - > key [ 0 ] , ( const u64 * ) ctrblk ,
keystream , AES_BLOCK_SIZE ) ;
crypto: algapi - make crypto_xor() take separate dst and src arguments
There are quite a number of occurrences in the kernel of the pattern
if (dst != src)
memcpy(dst, src, walk.total % AES_BLOCK_SIZE);
crypto_xor(dst, final, walk.total % AES_BLOCK_SIZE);
or
crypto_xor(keystream, src, nbytes);
memcpy(dst, keystream, nbytes);
where crypto_xor() is preceded or followed by a memcpy() invocation
that is only there because crypto_xor() uses its output parameter as
one of the inputs. To avoid having to add new instances of this pattern
in the arm64 code, which will be refactored to implement non-SIMD
fallbacks, add an alternative implementation called crypto_xor_cpy(),
taking separate input and output arguments. This removes the need for
the separate memcpy().
Signed-off-by: Ard Biesheuvel <ard.biesheuvel@linaro.org>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
2017-07-24 11:28:04 +01:00
crypto_xor_cpy ( dst , ( u8 * ) keystream , src , nbytes ) ;
2012-12-19 15:20:23 -08:00
crypto_inc ( ctrblk , AES_BLOCK_SIZE ) ;
}
2019-10-11 21:38:48 -07:00
static int ctr_crypt ( struct skcipher_request * req )
2012-08-29 14:49:23 -07:00
{
2019-10-11 21:38:48 -07:00
struct crypto_skcipher * tfm = crypto_skcipher_reqtfm ( req ) ;
const struct crypto_sparc64_aes_ctx * ctx = crypto_skcipher_ctx ( tfm ) ;
struct skcipher_walk walk ;
unsigned int nbytes ;
2012-08-29 14:49:23 -07:00
int err ;
2019-10-11 21:38:48 -07:00
err = skcipher_walk_virt ( & walk , req , true ) ;
if ( err )
return err ;
2012-08-29 14:49:23 -07:00
ctx - > ops - > load_encrypt_keys ( & ctx - > key [ 0 ] ) ;
2012-12-19 15:20:23 -08:00
while ( ( nbytes = walk . nbytes ) > = AES_BLOCK_SIZE ) {
2019-10-11 21:38:48 -07:00
ctx - > ops - > ctr_crypt ( & ctx - > key [ 0 ] , walk . src . virt . addr ,
walk . dst . virt . addr ,
round_down ( nbytes , AES_BLOCK_SIZE ) ,
walk . iv ) ;
err = skcipher_walk_done ( & walk , nbytes % AES_BLOCK_SIZE ) ;
2012-08-29 14:49:23 -07:00
}
2012-12-19 15:20:23 -08:00
if ( walk . nbytes ) {
ctr_crypt_final ( ctx , & walk ) ;
2019-10-11 21:38:48 -07:00
err = skcipher_walk_done ( & walk , 0 ) ;
2012-12-19 15:20:23 -08:00
}
2012-08-29 14:49:23 -07:00
fprs_write ( 0 ) ;
return err ;
}
2019-10-11 21:38:48 -07:00
static struct crypto_alg cipher_alg = {
2012-08-21 03:58:13 -07:00
. cra_name = " aes " ,
. cra_driver_name = " aes-sparc64 " ,
2012-09-15 09:06:30 -07:00
. cra_priority = SPARC_CR_OPCODE_PRIORITY ,
2012-08-21 03:58:13 -07:00
. cra_flags = CRYPTO_ALG_TYPE_CIPHER ,
. cra_blocksize = AES_BLOCK_SIZE ,
. cra_ctxsize = sizeof ( struct crypto_sparc64_aes_ctx ) ,
. cra_alignmask = 3 ,
. cra_module = THIS_MODULE ,
. cra_u = {
. cipher = {
. cia_min_keysize = AES_MIN_KEY_SIZE ,
. cia_max_keysize = AES_MAX_KEY_SIZE ,
. cia_setkey = aes_set_key ,
2019-07-02 21:41:20 +02:00
. cia_encrypt = crypto_aes_encrypt ,
. cia_decrypt = crypto_aes_decrypt
2012-08-21 03:58:13 -07:00
}
}
2019-10-11 21:38:48 -07:00
} ;
static struct skcipher_alg skcipher_algs [ ] = {
{
. base . cra_name = " ecb(aes) " ,
. base . cra_driver_name = " ecb-aes-sparc64 " ,
. base . cra_priority = SPARC_CR_OPCODE_PRIORITY ,
. base . cra_blocksize = AES_BLOCK_SIZE ,
. base . cra_ctxsize = sizeof ( struct crypto_sparc64_aes_ctx ) ,
. base . cra_alignmask = 7 ,
. base . cra_module = THIS_MODULE ,
. min_keysize = AES_MIN_KEY_SIZE ,
. max_keysize = AES_MAX_KEY_SIZE ,
. setkey = aes_set_key_skcipher ,
. encrypt = ecb_encrypt ,
. decrypt = ecb_decrypt ,
} , {
. base . cra_name = " cbc(aes) " ,
. base . cra_driver_name = " cbc-aes-sparc64 " ,
. base . cra_priority = SPARC_CR_OPCODE_PRIORITY ,
. base . cra_blocksize = AES_BLOCK_SIZE ,
. base . cra_ctxsize = sizeof ( struct crypto_sparc64_aes_ctx ) ,
. base . cra_alignmask = 7 ,
. base . cra_module = THIS_MODULE ,
. min_keysize = AES_MIN_KEY_SIZE ,
. max_keysize = AES_MAX_KEY_SIZE ,
. ivsize = AES_BLOCK_SIZE ,
. setkey = aes_set_key_skcipher ,
. encrypt = cbc_encrypt ,
. decrypt = cbc_decrypt ,
} , {
. base . cra_name = " ctr(aes) " ,
. base . cra_driver_name = " ctr-aes-sparc64 " ,
. base . cra_priority = SPARC_CR_OPCODE_PRIORITY ,
. base . cra_blocksize = 1 ,
. base . cra_ctxsize = sizeof ( struct crypto_sparc64_aes_ctx ) ,
. base . cra_alignmask = 7 ,
. base . cra_module = THIS_MODULE ,
. min_keysize = AES_MIN_KEY_SIZE ,
. max_keysize = AES_MAX_KEY_SIZE ,
. ivsize = AES_BLOCK_SIZE ,
. setkey = aes_set_key_skcipher ,
. encrypt = ctr_crypt ,
. decrypt = ctr_crypt ,
. chunksize = AES_BLOCK_SIZE ,
}
} ;
2012-08-21 03:58:13 -07:00
static bool __init sparc64_has_aes_opcode ( void )
{
unsigned long cfr ;
if ( ! ( sparc64_elf_hwcap & HWCAP_SPARC_CRYPTO ) )
return false ;
__asm__ __volatile__ ( " rd %%asr26, %0 " : " =r " ( cfr ) ) ;
if ( ! ( cfr & CFR_AES ) )
return false ;
return true ;
}
static int __init aes_sparc64_mod_init ( void )
{
2019-10-11 21:38:48 -07:00
int err ;
if ( ! sparc64_has_aes_opcode ( ) ) {
pr_info ( " sparc64 aes opcodes not available. \n " ) ;
return - ENODEV ;
2012-08-21 03:58:13 -07:00
}
2019-10-11 21:38:48 -07:00
pr_info ( " Using sparc64 aes opcodes optimized AES implementation \n " ) ;
err = crypto_register_alg ( & cipher_alg ) ;
if ( err )
return err ;
err = crypto_register_skciphers ( skcipher_algs ,
ARRAY_SIZE ( skcipher_algs ) ) ;
if ( err )
crypto_unregister_alg ( & cipher_alg ) ;
return err ;
2012-08-21 03:58:13 -07:00
}
static void __exit aes_sparc64_mod_fini ( void )
{
2019-10-11 21:38:48 -07:00
crypto_unregister_alg ( & cipher_alg ) ;
crypto_unregister_skciphers ( skcipher_algs , ARRAY_SIZE ( skcipher_algs ) ) ;
2012-08-21 03:58:13 -07:00
}
module_init ( aes_sparc64_mod_init ) ;
module_exit ( aes_sparc64_mod_fini ) ;
MODULE_LICENSE ( " GPL " ) ;
2015-01-11 18:17:43 +01:00
MODULE_DESCRIPTION ( " Rijndael (AES) Cipher Algorithm, sparc64 aes opcode accelerated " ) ;
2012-08-21 03:58:13 -07:00
2014-11-20 17:05:53 -08:00
MODULE_ALIAS_CRYPTO ( " aes " ) ;
2012-11-09 20:53:32 -08:00
# include "crop_devid.c"