1 /* SPDX-License-Identifier: GPL-2.0 */
3 * sha3-ce-glue.c - core SHA-3 transform using v8.2 Crypto Extensions
5 * Copyright (C) 2018 Linaro Ltd <ard.biesheuvel@linaro.org>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License version 2 as
9 * published by the Free Software Foundation.
12 #include <asm/hwcap.h>
15 #include <asm/unaligned.h>
16 #include <crypto/internal/hash.h>
17 #include <crypto/sha3.h>
18 #include <linux/cpufeature.h>
19 #include <linux/crypto.h>
20 #include <linux/module.h>
22 MODULE_DESCRIPTION("SHA3 secure hash using ARMv8 Crypto Extensions");
23 MODULE_AUTHOR("Ard Biesheuvel <ard.biesheuvel@linaro.org>");
24 MODULE_LICENSE("GPL v2");
26 asmlinkage
void sha3_ce_transform(u64
*st
, const u8
*data
, int blocks
,
29 static int sha3_update(struct shash_desc
*desc
, const u8
*data
,
32 struct sha3_state
*sctx
= shash_desc_ctx(desc
);
33 unsigned int digest_size
= crypto_shash_digestsize(desc
->tfm
);
36 return crypto_sha3_update(desc
, data
, len
);
38 if ((sctx
->partial
+ len
) >= sctx
->rsiz
) {
42 int p
= sctx
->rsiz
- sctx
->partial
;
44 memcpy(sctx
->buf
+ sctx
->partial
, data
, p
);
46 sha3_ce_transform(sctx
->st
, sctx
->buf
, 1, digest_size
);
54 blocks
= len
/ sctx
->rsiz
;
59 sha3_ce_transform(sctx
->st
, data
, blocks
, digest_size
);
61 data
+= blocks
* sctx
->rsiz
;
66 memcpy(sctx
->buf
+ sctx
->partial
, data
, len
);
72 static int sha3_final(struct shash_desc
*desc
, u8
*out
)
74 struct sha3_state
*sctx
= shash_desc_ctx(desc
);
75 unsigned int digest_size
= crypto_shash_digestsize(desc
->tfm
);
76 __le64
*digest
= (__le64
*)out
;
80 return crypto_sha3_final(desc
, out
);
82 sctx
->buf
[sctx
->partial
++] = 0x06;
83 memset(sctx
->buf
+ sctx
->partial
, 0, sctx
->rsiz
- sctx
->partial
);
84 sctx
->buf
[sctx
->rsiz
- 1] |= 0x80;
87 sha3_ce_transform(sctx
->st
, sctx
->buf
, 1, digest_size
);
90 for (i
= 0; i
< digest_size
/ 8; i
++)
91 put_unaligned_le64(sctx
->st
[i
], digest
++);
94 put_unaligned_le32(sctx
->st
[i
], (__le32
*)digest
);
96 *sctx
= (struct sha3_state
){};
100 static struct shash_alg algs
[] = { {
101 .digestsize
= SHA3_224_DIGEST_SIZE
,
102 .init
= crypto_sha3_init
,
103 .update
= sha3_update
,
105 .descsize
= sizeof(struct sha3_state
),
106 .base
.cra_name
= "sha3-224",
107 .base
.cra_driver_name
= "sha3-224-ce",
108 .base
.cra_flags
= CRYPTO_ALG_TYPE_SHASH
,
109 .base
.cra_blocksize
= SHA3_224_BLOCK_SIZE
,
110 .base
.cra_module
= THIS_MODULE
,
111 .base
.cra_priority
= 200,
113 .digestsize
= SHA3_256_DIGEST_SIZE
,
114 .init
= crypto_sha3_init
,
115 .update
= sha3_update
,
117 .descsize
= sizeof(struct sha3_state
),
118 .base
.cra_name
= "sha3-256",
119 .base
.cra_driver_name
= "sha3-256-ce",
120 .base
.cra_flags
= CRYPTO_ALG_TYPE_SHASH
,
121 .base
.cra_blocksize
= SHA3_256_BLOCK_SIZE
,
122 .base
.cra_module
= THIS_MODULE
,
123 .base
.cra_priority
= 200,
125 .digestsize
= SHA3_384_DIGEST_SIZE
,
126 .init
= crypto_sha3_init
,
127 .update
= sha3_update
,
129 .descsize
= sizeof(struct sha3_state
),
130 .base
.cra_name
= "sha3-384",
131 .base
.cra_driver_name
= "sha3-384-ce",
132 .base
.cra_flags
= CRYPTO_ALG_TYPE_SHASH
,
133 .base
.cra_blocksize
= SHA3_384_BLOCK_SIZE
,
134 .base
.cra_module
= THIS_MODULE
,
135 .base
.cra_priority
= 200,
137 .digestsize
= SHA3_512_DIGEST_SIZE
,
138 .init
= crypto_sha3_init
,
139 .update
= sha3_update
,
141 .descsize
= sizeof(struct sha3_state
),
142 .base
.cra_name
= "sha3-512",
143 .base
.cra_driver_name
= "sha3-512-ce",
144 .base
.cra_flags
= CRYPTO_ALG_TYPE_SHASH
,
145 .base
.cra_blocksize
= SHA3_512_BLOCK_SIZE
,
146 .base
.cra_module
= THIS_MODULE
,
147 .base
.cra_priority
= 200,
150 static int __init
sha3_neon_mod_init(void)
152 return crypto_register_shashes(algs
, ARRAY_SIZE(algs
));
155 static void __exit
sha3_neon_mod_fini(void)
157 crypto_unregister_shashes(algs
, ARRAY_SIZE(algs
));
160 module_cpu_feature_match(SHA3
, sha3_neon_mod_init
);
161 module_exit(sha3_neon_mod_fini
);