| /* | 
 |  * Cryptographic API. | 
 |  * | 
 |  * Digest operations. | 
 |  * | 
 |  * Copyright (c) 2002 James Morris <jmorris@intercode.com.au> | 
 |  * | 
 |  * 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/crypto.h> | 
 | #include <linux/mm.h> | 
 | #include <linux/errno.h> | 
 | #include <linux/highmem.h> | 
 | #include <asm/scatterlist.h> | 
 | #include "internal.h" | 
 |  | 
 | static void init(struct crypto_tfm *tfm) | 
 | { | 
 | 	tfm->__crt_alg->cra_digest.dia_init(crypto_tfm_ctx(tfm)); | 
 | } | 
 |  | 
 | static void update(struct crypto_tfm *tfm, | 
 |                    struct scatterlist *sg, unsigned int nsg) | 
 | { | 
 | 	unsigned int i; | 
 |  | 
 | 	for (i = 0; i < nsg; i++) { | 
 |  | 
 | 		struct page *pg = sg[i].page; | 
 | 		unsigned int offset = sg[i].offset; | 
 | 		unsigned int l = sg[i].length; | 
 |  | 
 | 		do { | 
 | 			unsigned int bytes_from_page = min(l, ((unsigned int) | 
 | 							   (PAGE_SIZE)) -  | 
 | 							   offset); | 
 | 			char *p = crypto_kmap(pg, 0) + offset; | 
 |  | 
 | 			tfm->__crt_alg->cra_digest.dia_update | 
 | 					(crypto_tfm_ctx(tfm), p, | 
 | 					 bytes_from_page); | 
 | 			crypto_kunmap(p, 0); | 
 | 			crypto_yield(tfm); | 
 | 			offset = 0; | 
 | 			pg++; | 
 | 			l -= bytes_from_page; | 
 | 		} while (l > 0); | 
 | 	} | 
 | } | 
 |  | 
 | static void final(struct crypto_tfm *tfm, u8 *out) | 
 | { | 
 | 	tfm->__crt_alg->cra_digest.dia_final(crypto_tfm_ctx(tfm), out); | 
 | } | 
 |  | 
 | static int setkey(struct crypto_tfm *tfm, const u8 *key, unsigned int keylen) | 
 | { | 
 | 	u32 flags; | 
 | 	if (tfm->__crt_alg->cra_digest.dia_setkey == NULL) | 
 | 		return -ENOSYS; | 
 | 	return tfm->__crt_alg->cra_digest.dia_setkey(crypto_tfm_ctx(tfm), | 
 | 						     key, keylen, &flags); | 
 | } | 
 |  | 
 | static void digest(struct crypto_tfm *tfm, | 
 |                    struct scatterlist *sg, unsigned int nsg, u8 *out) | 
 | { | 
 | 	unsigned int i; | 
 |  | 
 | 	tfm->crt_digest.dit_init(tfm); | 
 | 		 | 
 | 	for (i = 0; i < nsg; i++) { | 
 | 		char *p = crypto_kmap(sg[i].page, 0) + sg[i].offset; | 
 | 		tfm->__crt_alg->cra_digest.dia_update(crypto_tfm_ctx(tfm), | 
 | 		                                      p, sg[i].length); | 
 | 		crypto_kunmap(p, 0); | 
 | 		crypto_yield(tfm); | 
 | 	} | 
 | 	crypto_digest_final(tfm, out); | 
 | } | 
 |  | 
 | int crypto_init_digest_flags(struct crypto_tfm *tfm, u32 flags) | 
 | { | 
 | 	return flags ? -EINVAL : 0; | 
 | } | 
 |  | 
 | int crypto_init_digest_ops(struct crypto_tfm *tfm) | 
 | { | 
 | 	struct digest_tfm *ops = &tfm->crt_digest; | 
 | 	 | 
 | 	ops->dit_init	= init; | 
 | 	ops->dit_update	= update; | 
 | 	ops->dit_final	= final; | 
 | 	ops->dit_digest	= digest; | 
 | 	ops->dit_setkey	= setkey; | 
 | 	 | 
 | 	return crypto_alloc_hmac_block(tfm); | 
 | } | 
 |  | 
 | void crypto_exit_digest_ops(struct crypto_tfm *tfm) | 
 | { | 
 | 	crypto_free_hmac_block(tfm); | 
 | } |