mirror of
https://github.com/torvalds/linux.git
synced 2026-06-04 04:23:35 +02:00
lib/crypto: sha256: Add HMAC-SHA224 and HMAC-SHA256 support
Since HMAC support is commonly needed and is fairly simple, include it as a first-class citizen of the SHA-256 library. The API supports both incremental and one-shot computation, and either preparing the key ahead of time or just using a raw key. The implementation is much more streamlined than crypto/hmac.c. I've kept it consistent with the HMAC-SHA384 and HMAC-SHA512 code as much as possible. Testing of these functions will be via sha224_kunit and sha256_kunit, added by a later commit. Acked-by: Ard Biesheuvel <ardb@kernel.org> Link: https://lore.kernel.org/r/20250630160645.3198-9-ebiggers@kernel.org Signed-off-by: Eric Biggers <ebiggers@kernel.org>
This commit is contained in:
parent
4c855d5069
commit
077833cd60
|
|
@ -131,6 +131,22 @@ struct __sha256_ctx {
|
|||
};
|
||||
void __sha256_update(struct __sha256_ctx *ctx, const u8 *data, size_t len);
|
||||
|
||||
/*
|
||||
* HMAC key and message context structs, shared by HMAC-SHA224 and HMAC-SHA256.
|
||||
* The hmac_sha224_* and hmac_sha256_* structs wrap this one so that the API has
|
||||
* proper typing and doesn't allow mixing the functions arbitrarily.
|
||||
*/
|
||||
struct __hmac_sha256_key {
|
||||
struct sha256_block_state istate;
|
||||
struct sha256_block_state ostate;
|
||||
};
|
||||
struct __hmac_sha256_ctx {
|
||||
struct __sha256_ctx sha_ctx;
|
||||
struct sha256_block_state ostate;
|
||||
};
|
||||
void __hmac_sha256_init(struct __hmac_sha256_ctx *ctx,
|
||||
const struct __hmac_sha256_key *key);
|
||||
|
||||
/**
|
||||
* struct sha224_ctx - Context for hashing a message with SHA-224
|
||||
* @ctx: private
|
||||
|
|
@ -148,6 +164,109 @@ static inline void sha224_update(struct sha224_ctx *ctx,
|
|||
void sha224_final(struct sha224_ctx *ctx, u8 out[SHA224_DIGEST_SIZE]);
|
||||
void sha224(const u8 *data, size_t len, u8 out[SHA224_DIGEST_SIZE]);
|
||||
|
||||
/**
|
||||
* struct hmac_sha224_key - Prepared key for HMAC-SHA224
|
||||
* @key: private
|
||||
*/
|
||||
struct hmac_sha224_key {
|
||||
struct __hmac_sha256_key key;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct hmac_sha224_ctx - Context for computing HMAC-SHA224 of a message
|
||||
* @ctx: private
|
||||
*/
|
||||
struct hmac_sha224_ctx {
|
||||
struct __hmac_sha256_ctx ctx;
|
||||
};
|
||||
|
||||
/**
|
||||
* hmac_sha224_preparekey() - Prepare a key for HMAC-SHA224
|
||||
* @key: (output) the key structure to initialize
|
||||
* @raw_key: the raw HMAC-SHA224 key
|
||||
* @raw_key_len: the key length in bytes. All key lengths are supported.
|
||||
*
|
||||
* Note: the caller is responsible for zeroizing both the struct hmac_sha224_key
|
||||
* and the raw key once they are no longer needed.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha224_preparekey(struct hmac_sha224_key *key,
|
||||
const u8 *raw_key, size_t raw_key_len);
|
||||
|
||||
/**
|
||||
* hmac_sha224_init() - Initialize an HMAC-SHA224 context for a new message
|
||||
* @ctx: (output) the HMAC context to initialize
|
||||
* @key: the prepared HMAC key
|
||||
*
|
||||
* If you don't need incremental computation, consider hmac_sha224() instead.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
static inline void hmac_sha224_init(struct hmac_sha224_ctx *ctx,
|
||||
const struct hmac_sha224_key *key)
|
||||
{
|
||||
__hmac_sha256_init(&ctx->ctx, &key->key);
|
||||
}
|
||||
|
||||
/**
|
||||
* hmac_sha224_update() - Update an HMAC-SHA224 context with message data
|
||||
* @ctx: the HMAC context to update; must have been initialized
|
||||
* @data: the message data
|
||||
* @data_len: the data length in bytes
|
||||
*
|
||||
* This can be called any number of times.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
static inline void hmac_sha224_update(struct hmac_sha224_ctx *ctx,
|
||||
const u8 *data, size_t data_len)
|
||||
{
|
||||
__sha256_update(&ctx->ctx.sha_ctx, data, data_len);
|
||||
}
|
||||
|
||||
/**
|
||||
* hmac_sha224_final() - Finish computing an HMAC-SHA224 value
|
||||
* @ctx: the HMAC context to finalize; must have been initialized
|
||||
* @out: (output) the resulting HMAC-SHA224 value
|
||||
*
|
||||
* After finishing, this zeroizes @ctx. So the caller does not need to do it.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha224_final(struct hmac_sha224_ctx *ctx, u8 out[SHA224_DIGEST_SIZE]);
|
||||
|
||||
/**
|
||||
* hmac_sha224() - Compute HMAC-SHA224 in one shot, using a prepared key
|
||||
* @key: the prepared HMAC key
|
||||
* @data: the message data
|
||||
* @data_len: the data length in bytes
|
||||
* @out: (output) the resulting HMAC-SHA224 value
|
||||
*
|
||||
* If you're using the key only once, consider using hmac_sha224_usingrawkey().
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha224(const struct hmac_sha224_key *key,
|
||||
const u8 *data, size_t data_len, u8 out[SHA224_DIGEST_SIZE]);
|
||||
|
||||
/**
|
||||
* hmac_sha224_usingrawkey() - Compute HMAC-SHA224 in one shot, using a raw key
|
||||
* @raw_key: the raw HMAC-SHA224 key
|
||||
* @raw_key_len: the key length in bytes. All key lengths are supported.
|
||||
* @data: the message data
|
||||
* @data_len: the data length in bytes
|
||||
* @out: (output) the resulting HMAC-SHA224 value
|
||||
*
|
||||
* If you're using the key multiple times, prefer to use
|
||||
* hmac_sha224_preparekey() followed by multiple calls to hmac_sha224() instead.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha224_usingrawkey(const u8 *raw_key, size_t raw_key_len,
|
||||
const u8 *data, size_t data_len,
|
||||
u8 out[SHA224_DIGEST_SIZE]);
|
||||
|
||||
/**
|
||||
* struct sha256_ctx - Context for hashing a message with SHA-256
|
||||
* @ctx: private
|
||||
|
|
@ -165,6 +284,109 @@ static inline void sha256_update(struct sha256_ctx *ctx,
|
|||
void sha256_final(struct sha256_ctx *ctx, u8 out[SHA256_DIGEST_SIZE]);
|
||||
void sha256(const u8 *data, size_t len, u8 out[SHA256_DIGEST_SIZE]);
|
||||
|
||||
/**
|
||||
* struct hmac_sha256_key - Prepared key for HMAC-SHA256
|
||||
* @key: private
|
||||
*/
|
||||
struct hmac_sha256_key {
|
||||
struct __hmac_sha256_key key;
|
||||
};
|
||||
|
||||
/**
|
||||
* struct hmac_sha256_ctx - Context for computing HMAC-SHA256 of a message
|
||||
* @ctx: private
|
||||
*/
|
||||
struct hmac_sha256_ctx {
|
||||
struct __hmac_sha256_ctx ctx;
|
||||
};
|
||||
|
||||
/**
|
||||
* hmac_sha256_preparekey() - Prepare a key for HMAC-SHA256
|
||||
* @key: (output) the key structure to initialize
|
||||
* @raw_key: the raw HMAC-SHA256 key
|
||||
* @raw_key_len: the key length in bytes. All key lengths are supported.
|
||||
*
|
||||
* Note: the caller is responsible for zeroizing both the struct hmac_sha256_key
|
||||
* and the raw key once they are no longer needed.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha256_preparekey(struct hmac_sha256_key *key,
|
||||
const u8 *raw_key, size_t raw_key_len);
|
||||
|
||||
/**
|
||||
* hmac_sha256_init() - Initialize an HMAC-SHA256 context for a new message
|
||||
* @ctx: (output) the HMAC context to initialize
|
||||
* @key: the prepared HMAC key
|
||||
*
|
||||
* If you don't need incremental computation, consider hmac_sha256() instead.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
static inline void hmac_sha256_init(struct hmac_sha256_ctx *ctx,
|
||||
const struct hmac_sha256_key *key)
|
||||
{
|
||||
__hmac_sha256_init(&ctx->ctx, &key->key);
|
||||
}
|
||||
|
||||
/**
|
||||
* hmac_sha256_update() - Update an HMAC-SHA256 context with message data
|
||||
* @ctx: the HMAC context to update; must have been initialized
|
||||
* @data: the message data
|
||||
* @data_len: the data length in bytes
|
||||
*
|
||||
* This can be called any number of times.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
static inline void hmac_sha256_update(struct hmac_sha256_ctx *ctx,
|
||||
const u8 *data, size_t data_len)
|
||||
{
|
||||
__sha256_update(&ctx->ctx.sha_ctx, data, data_len);
|
||||
}
|
||||
|
||||
/**
|
||||
* hmac_sha256_final() - Finish computing an HMAC-SHA256 value
|
||||
* @ctx: the HMAC context to finalize; must have been initialized
|
||||
* @out: (output) the resulting HMAC-SHA256 value
|
||||
*
|
||||
* After finishing, this zeroizes @ctx. So the caller does not need to do it.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha256_final(struct hmac_sha256_ctx *ctx, u8 out[SHA256_DIGEST_SIZE]);
|
||||
|
||||
/**
|
||||
* hmac_sha256() - Compute HMAC-SHA256 in one shot, using a prepared key
|
||||
* @key: the prepared HMAC key
|
||||
* @data: the message data
|
||||
* @data_len: the data length in bytes
|
||||
* @out: (output) the resulting HMAC-SHA256 value
|
||||
*
|
||||
* If you're using the key only once, consider using hmac_sha256_usingrawkey().
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha256(const struct hmac_sha256_key *key,
|
||||
const u8 *data, size_t data_len, u8 out[SHA256_DIGEST_SIZE]);
|
||||
|
||||
/**
|
||||
* hmac_sha256_usingrawkey() - Compute HMAC-SHA256 in one shot, using a raw key
|
||||
* @raw_key: the raw HMAC-SHA256 key
|
||||
* @raw_key_len: the key length in bytes. All key lengths are supported.
|
||||
* @data: the message data
|
||||
* @data_len: the data length in bytes
|
||||
* @out: (output) the resulting HMAC-SHA256 value
|
||||
*
|
||||
* If you're using the key multiple times, prefer to use
|
||||
* hmac_sha256_preparekey() followed by multiple calls to hmac_sha256() instead.
|
||||
*
|
||||
* Context: Any context.
|
||||
*/
|
||||
void hmac_sha256_usingrawkey(const u8 *raw_key, size_t raw_key_len,
|
||||
const u8 *data, size_t data_len,
|
||||
u8 out[SHA256_DIGEST_SIZE]);
|
||||
|
||||
/* State for the SHA-512 (and SHA-384) compression function */
|
||||
struct sha512_block_state {
|
||||
u64 h[8];
|
||||
|
|
|
|||
|
|
@ -1,9 +1,6 @@
|
|||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
/*
|
||||
* SHA-256, as specified in
|
||||
* http://csrc.nist.gov/groups/STM/cavp/documents/shs/sha256-384-512.pdf
|
||||
*
|
||||
* SHA-256 code by Jean-Luc Cooke <jlcooke@certainkey.com>.
|
||||
* SHA-224, SHA-256, HMAC-SHA224, and HMAC-SHA256 library functions
|
||||
*
|
||||
* Copyright (c) Jean-Luc Cooke <jlcooke@certainkey.com>
|
||||
* Copyright (c) Andrew McDonald <andrew@mcdonald.org.uk>
|
||||
|
|
@ -11,12 +8,14 @@
|
|||
* Copyright (c) 2014 Red Hat Inc.
|
||||
*/
|
||||
|
||||
#include <crypto/hmac.h>
|
||||
#include <crypto/internal/blockhash.h>
|
||||
#include <crypto/internal/sha2.h>
|
||||
#include <linux/export.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/module.h>
|
||||
#include <linux/string.h>
|
||||
#include <linux/wordpart.h>
|
||||
|
||||
static const struct sha256_block_state sha224_iv = {
|
||||
.h = {
|
||||
|
|
@ -136,5 +135,143 @@ void sha256(const u8 *data, size_t len, u8 out[SHA256_DIGEST_SIZE])
|
|||
}
|
||||
EXPORT_SYMBOL(sha256);
|
||||
|
||||
MODULE_DESCRIPTION("SHA-256 Algorithm");
|
||||
/* pre-boot environment (as indicated by __DISABLE_EXPORTS) doesn't need HMAC */
|
||||
#ifndef __DISABLE_EXPORTS
|
||||
static void __hmac_sha256_preparekey(struct __hmac_sha256_key *key,
|
||||
const u8 *raw_key, size_t raw_key_len,
|
||||
const struct sha256_block_state *iv)
|
||||
{
|
||||
union {
|
||||
u8 b[SHA256_BLOCK_SIZE];
|
||||
unsigned long w[SHA256_BLOCK_SIZE / sizeof(unsigned long)];
|
||||
} derived_key = { 0 };
|
||||
|
||||
if (unlikely(raw_key_len > SHA256_BLOCK_SIZE)) {
|
||||
if (iv == &sha224_iv)
|
||||
sha224(raw_key, raw_key_len, derived_key.b);
|
||||
else
|
||||
sha256(raw_key, raw_key_len, derived_key.b);
|
||||
} else {
|
||||
memcpy(derived_key.b, raw_key, raw_key_len);
|
||||
}
|
||||
|
||||
for (size_t i = 0; i < ARRAY_SIZE(derived_key.w); i++)
|
||||
derived_key.w[i] ^= REPEAT_BYTE(HMAC_IPAD_VALUE);
|
||||
key->istate = *iv;
|
||||
sha256_blocks(&key->istate, derived_key.b, 1);
|
||||
|
||||
for (size_t i = 0; i < ARRAY_SIZE(derived_key.w); i++)
|
||||
derived_key.w[i] ^= REPEAT_BYTE(HMAC_OPAD_VALUE ^
|
||||
HMAC_IPAD_VALUE);
|
||||
key->ostate = *iv;
|
||||
sha256_blocks(&key->ostate, derived_key.b, 1);
|
||||
|
||||
memzero_explicit(&derived_key, sizeof(derived_key));
|
||||
}
|
||||
|
||||
void hmac_sha224_preparekey(struct hmac_sha224_key *key,
|
||||
const u8 *raw_key, size_t raw_key_len)
|
||||
{
|
||||
__hmac_sha256_preparekey(&key->key, raw_key, raw_key_len, &sha224_iv);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha224_preparekey);
|
||||
|
||||
void hmac_sha256_preparekey(struct hmac_sha256_key *key,
|
||||
const u8 *raw_key, size_t raw_key_len)
|
||||
{
|
||||
__hmac_sha256_preparekey(&key->key, raw_key, raw_key_len, &sha256_iv);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha256_preparekey);
|
||||
|
||||
void __hmac_sha256_init(struct __hmac_sha256_ctx *ctx,
|
||||
const struct __hmac_sha256_key *key)
|
||||
{
|
||||
__sha256_init(&ctx->sha_ctx, &key->istate, SHA256_BLOCK_SIZE);
|
||||
ctx->ostate = key->ostate;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(__hmac_sha256_init);
|
||||
|
||||
static void __hmac_sha256_final(struct __hmac_sha256_ctx *ctx,
|
||||
u8 *out, size_t digest_size)
|
||||
{
|
||||
/* Generate the padded input for the outer hash in ctx->sha_ctx.buf. */
|
||||
__sha256_final(&ctx->sha_ctx, ctx->sha_ctx.buf, digest_size);
|
||||
memset(&ctx->sha_ctx.buf[digest_size], 0,
|
||||
SHA256_BLOCK_SIZE - digest_size);
|
||||
ctx->sha_ctx.buf[digest_size] = 0x80;
|
||||
*(__be32 *)&ctx->sha_ctx.buf[SHA256_BLOCK_SIZE - 4] =
|
||||
cpu_to_be32(8 * (SHA256_BLOCK_SIZE + digest_size));
|
||||
|
||||
/* Compute the outer hash, which gives the HMAC value. */
|
||||
sha256_blocks(&ctx->ostate, ctx->sha_ctx.buf, 1);
|
||||
for (size_t i = 0; i < digest_size; i += 4)
|
||||
put_unaligned_be32(ctx->ostate.h[i / 4], out + i);
|
||||
|
||||
memzero_explicit(ctx, sizeof(*ctx));
|
||||
}
|
||||
|
||||
void hmac_sha224_final(struct hmac_sha224_ctx *ctx,
|
||||
u8 out[SHA224_DIGEST_SIZE])
|
||||
{
|
||||
__hmac_sha256_final(&ctx->ctx, out, SHA224_DIGEST_SIZE);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha224_final);
|
||||
|
||||
void hmac_sha256_final(struct hmac_sha256_ctx *ctx,
|
||||
u8 out[SHA256_DIGEST_SIZE])
|
||||
{
|
||||
__hmac_sha256_final(&ctx->ctx, out, SHA256_DIGEST_SIZE);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha256_final);
|
||||
|
||||
void hmac_sha224(const struct hmac_sha224_key *key,
|
||||
const u8 *data, size_t data_len, u8 out[SHA224_DIGEST_SIZE])
|
||||
{
|
||||
struct hmac_sha224_ctx ctx;
|
||||
|
||||
hmac_sha224_init(&ctx, key);
|
||||
hmac_sha224_update(&ctx, data, data_len);
|
||||
hmac_sha224_final(&ctx, out);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha224);
|
||||
|
||||
void hmac_sha256(const struct hmac_sha256_key *key,
|
||||
const u8 *data, size_t data_len, u8 out[SHA256_DIGEST_SIZE])
|
||||
{
|
||||
struct hmac_sha256_ctx ctx;
|
||||
|
||||
hmac_sha256_init(&ctx, key);
|
||||
hmac_sha256_update(&ctx, data, data_len);
|
||||
hmac_sha256_final(&ctx, out);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha256);
|
||||
|
||||
void hmac_sha224_usingrawkey(const u8 *raw_key, size_t raw_key_len,
|
||||
const u8 *data, size_t data_len,
|
||||
u8 out[SHA224_DIGEST_SIZE])
|
||||
{
|
||||
struct hmac_sha224_key key;
|
||||
|
||||
hmac_sha224_preparekey(&key, raw_key, raw_key_len);
|
||||
hmac_sha224(&key, data, data_len, out);
|
||||
|
||||
memzero_explicit(&key, sizeof(key));
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha224_usingrawkey);
|
||||
|
||||
void hmac_sha256_usingrawkey(const u8 *raw_key, size_t raw_key_len,
|
||||
const u8 *data, size_t data_len,
|
||||
u8 out[SHA256_DIGEST_SIZE])
|
||||
{
|
||||
struct hmac_sha256_key key;
|
||||
|
||||
hmac_sha256_preparekey(&key, raw_key, raw_key_len);
|
||||
hmac_sha256(&key, data, data_len, out);
|
||||
|
||||
memzero_explicit(&key, sizeof(key));
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(hmac_sha256_usingrawkey);
|
||||
#endif /* !__DISABLE_EXPORTS */
|
||||
|
||||
MODULE_DESCRIPTION("SHA-224, SHA-256, HMAC-SHA224, and HMAC-SHA256 library functions");
|
||||
MODULE_LICENSE("GPL");
|
||||
|
|
|
|||
Loading…
Reference in New Issue
Block a user