Kernel  |  4.4

下载     查看原文件
C++程序  |  215行  |  5.08 KB
/*
 * Cryptographic API.
 *
 * Skein256 Hash Algorithm.
 *
 * Derived from cryptoapi implementation, adapted for in-place
 * scatterlist interface.
 *
 * Copyright (c) Eric Rost <eric.rost@mybabylon.net>
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the Free
 * Software Foundation; either version 2 of the License, or (at your option)
 * any later version.
 *
 */
#include <linux/types.h>
#include <linux/init.h>
#include <linux/module.h>
#include <crypto/internal/hash.h>
#include "skein_base.h"

static int skein256_init(struct shash_desc *desc)
{
	return skein_256_init((struct skein_256_ctx *)shash_desc_ctx(desc),
			SKEIN256_DIGEST_BIT_SIZE);
}

static int skein256_update(struct shash_desc *desc, const u8 *data,
			unsigned int len)
{
	return skein_256_update((struct skein_256_ctx *)shash_desc_ctx(desc),
				data, len);
}

static int skein256_final(struct shash_desc *desc, u8 *out)
{
	return skein_256_final((struct skein_256_ctx *)shash_desc_ctx(desc),
				out);
}

static int skein256_export(struct shash_desc *desc, void *out)
{
	struct skein_256_ctx *sctx = shash_desc_ctx(desc);

	memcpy(out, sctx, sizeof(*sctx));
	return 0;
}

static int skein256_import(struct shash_desc *desc, const void *in)
{
	struct skein_256_ctx *sctx = shash_desc_ctx(desc);

	memcpy(sctx, in, sizeof(*sctx));
	return 0;
}

static int skein512_init(struct shash_desc *desc)
{
	return skein_512_init((struct skein_512_ctx *)shash_desc_ctx(desc),
				SKEIN512_DIGEST_BIT_SIZE);
}

static int skein512_update(struct shash_desc *desc, const u8 *data,
			unsigned int len)
{
	return skein_512_update((struct skein_512_ctx *)shash_desc_ctx(desc),
				data, len);
}

static int skein512_final(struct shash_desc *desc, u8 *out)
{
	return skein_512_final((struct skein_512_ctx *)shash_desc_ctx(desc),
				out);
}

static int skein512_export(struct shash_desc *desc, void *out)
{
	struct skein_512_ctx *sctx = shash_desc_ctx(desc);

	memcpy(out, sctx, sizeof(*sctx));
	return 0;
}

static int skein512_import(struct shash_desc *desc, const void *in)
{
	struct skein_512_ctx *sctx = shash_desc_ctx(desc);

	memcpy(sctx, in, sizeof(*sctx));
	return 0;
}

static int skein1024_init(struct shash_desc *desc)
{
	return skein_1024_init((struct skein_1024_ctx *)shash_desc_ctx(desc),
				SKEIN1024_DIGEST_BIT_SIZE);
}

static int skein1024_update(struct shash_desc *desc, const u8 *data,
			unsigned int len)
{
	return skein_1024_update((struct skein_1024_ctx *)shash_desc_ctx(desc),
				data, len);
}

static int skein1024_final(struct shash_desc *desc, u8 *out)
{
	return skein_1024_final((struct skein_1024_ctx *)shash_desc_ctx(desc),
			out);
}

static int skein1024_export(struct shash_desc *desc, void *out)
{
	struct skein_1024_ctx *sctx = shash_desc_ctx(desc);

	memcpy(out, sctx, sizeof(*sctx));
	return 0;
}

static int skein1024_import(struct shash_desc *desc, const void *in)
{
	struct skein_1024_ctx *sctx = shash_desc_ctx(desc);

	memcpy(sctx, in, sizeof(*sctx));
	return 0;
}

static struct shash_alg alg256 = {
	.digestsize	=	(SKEIN256_DIGEST_BIT_SIZE / 8),
	.init		=	skein256_init,
	.update		=	skein256_update,
	.final		=	skein256_final,
	.export		=	skein256_export,
	.import		=	skein256_import,
	.descsize	=	sizeof(struct skein_256_ctx),
	.statesize	=	sizeof(struct skein_256_ctx),
	.base		=	{
		.cra_name		=	"skein256",
		.cra_driver_name	=	"skein",
		.cra_flags		=	CRYPTO_ALG_TYPE_SHASH,
		.cra_blocksize		=	SKEIN_256_BLOCK_BYTES,
		.cra_module		=	THIS_MODULE,
	}
};

static struct shash_alg alg512 = {
	.digestsize	=	(SKEIN512_DIGEST_BIT_SIZE / 8),
	.init		=	skein512_init,
	.update		=	skein512_update,
	.final		=	skein512_final,
	.export		=	skein512_export,
	.import		=	skein512_import,
	.descsize	=	sizeof(struct skein_512_ctx),
	.statesize	=	sizeof(struct skein_512_ctx),
	.base		=	{
		.cra_name		=	"skein512",
		.cra_driver_name	=	"skein",
		.cra_flags		=	CRYPTO_ALG_TYPE_SHASH,
		.cra_blocksize		=	SKEIN_512_BLOCK_BYTES,
		.cra_module		=	THIS_MODULE,
	}
};

static struct shash_alg alg1024 = {
	.digestsize	=	(SKEIN1024_DIGEST_BIT_SIZE / 8),
	.init		=	skein1024_init,
	.update		=	skein1024_update,
	.final		=	skein1024_final,
	.export		=	skein1024_export,
	.import		=	skein1024_import,
	.descsize	=	sizeof(struct skein_1024_ctx),
	.statesize	=	sizeof(struct skein_1024_ctx),
	.base		=	{
		.cra_name		=	"skein1024",
		.cra_driver_name	=	"skein",
		.cra_flags		=	CRYPTO_ALG_TYPE_SHASH,
		.cra_blocksize		=	SKEIN_1024_BLOCK_BYTES,
		.cra_module		=	THIS_MODULE,
	}
};

static int __init skein_generic_init(void)
{
	if (crypto_register_shash(&alg256))
		goto out;
	if (crypto_register_shash(&alg512))
		goto unreg256;
	if (crypto_register_shash(&alg1024))
		goto unreg512;

	return 0;

unreg512:
	crypto_unregister_shash(&alg512);
unreg256:
	crypto_unregister_shash(&alg256);
out:
	return -1;
}

static void __exit skein_generic_fini(void)
{
	crypto_unregister_shash(&alg256);
	crypto_unregister_shash(&alg512);
	crypto_unregister_shash(&alg1024);
}

module_init(skein_generic_init);
module_exit(skein_generic_fini);

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("Skein Hash Algorithm");

MODULE_ALIAS("skein");