| /* |
| * 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"); |