OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [linux/] [linux-2.4/] [crypto/] [api.c] - Diff between revs 1275 and 1765

Only display areas with differences | Details | Blame | View Log

Rev 1275 Rev 1765
/*
/*
 * Scatterlist Cryptographic API.
 * Scatterlist Cryptographic API.
 *
 *
 * Copyright (c) 2002 James Morris <jmorris@intercode.com.au>
 * Copyright (c) 2002 James Morris <jmorris@intercode.com.au>
 * Copyright (c) 2002 David S. Miller (davem@redhat.com)
 * Copyright (c) 2002 David S. Miller (davem@redhat.com)
 *
 *
 * Portions derived from Cryptoapi, by Alexander Kjeldaas <astor@fast.no>
 * Portions derived from Cryptoapi, by Alexander Kjeldaas <astor@fast.no>
 * and Nettle, by Niels Möller.
 * and Nettle, by Niels Möller.
 *
 *
 * This program is free software; you can redistribute it and/or modify it
 * 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
 * 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)
 * Software Foundation; either version 2 of the License, or (at your option)
 * any later version.
 * any later version.
 *
 *
 */
 */
#include <linux/init.h>
#include <linux/init.h>
#include <linux/crypto.h>
#include <linux/crypto.h>
#include <linux/errno.h>
#include <linux/errno.h>
#include <linux/rwsem.h>
#include <linux/rwsem.h>
#include <linux/slab.h>
#include <linux/slab.h>
#include "internal.h"
#include "internal.h"
 
 
LIST_HEAD(crypto_alg_list);
LIST_HEAD(crypto_alg_list);
DECLARE_RWSEM(crypto_alg_sem);
DECLARE_RWSEM(crypto_alg_sem);
 
 
static inline int crypto_alg_get(struct crypto_alg *alg)
static inline int crypto_alg_get(struct crypto_alg *alg)
{
{
        return try_inc_mod_count(alg->cra_module);
        return try_inc_mod_count(alg->cra_module);
}
}
 
 
static inline void crypto_alg_put(struct crypto_alg *alg)
static inline void crypto_alg_put(struct crypto_alg *alg)
{
{
        if (alg->cra_module)
        if (alg->cra_module)
                __MOD_DEC_USE_COUNT(alg->cra_module);
                __MOD_DEC_USE_COUNT(alg->cra_module);
}
}
 
 
struct crypto_alg *crypto_alg_lookup(const char *name)
struct crypto_alg *crypto_alg_lookup(const char *name)
{
{
        struct crypto_alg *q, *alg = NULL;
        struct crypto_alg *q, *alg = NULL;
 
 
        if (!name)
        if (!name)
                return NULL;
                return NULL;
 
 
        down_read(&crypto_alg_sem);
        down_read(&crypto_alg_sem);
 
 
        list_for_each_entry(q, &crypto_alg_list, cra_list) {
        list_for_each_entry(q, &crypto_alg_list, cra_list) {
                if (!(strcmp(q->cra_name, name))) {
                if (!(strcmp(q->cra_name, name))) {
                        if (crypto_alg_get(q))
                        if (crypto_alg_get(q))
                                alg = q;
                                alg = q;
                        break;
                        break;
                }
                }
        }
        }
 
 
        up_read(&crypto_alg_sem);
        up_read(&crypto_alg_sem);
        return alg;
        return alg;
}
}
 
 
static int crypto_init_flags(struct crypto_tfm *tfm, u32 flags)
static int crypto_init_flags(struct crypto_tfm *tfm, u32 flags)
{
{
        tfm->crt_flags = 0;
        tfm->crt_flags = 0;
 
 
        switch (crypto_tfm_alg_type(tfm)) {
        switch (crypto_tfm_alg_type(tfm)) {
        case CRYPTO_ALG_TYPE_CIPHER:
        case CRYPTO_ALG_TYPE_CIPHER:
                return crypto_init_cipher_flags(tfm, flags);
                return crypto_init_cipher_flags(tfm, flags);
 
 
        case CRYPTO_ALG_TYPE_DIGEST:
        case CRYPTO_ALG_TYPE_DIGEST:
                return crypto_init_digest_flags(tfm, flags);
                return crypto_init_digest_flags(tfm, flags);
 
 
        case CRYPTO_ALG_TYPE_COMPRESS:
        case CRYPTO_ALG_TYPE_COMPRESS:
                return crypto_init_compress_flags(tfm, flags);
                return crypto_init_compress_flags(tfm, flags);
 
 
        default:
        default:
                break;
                break;
        }
        }
 
 
        BUG();
        BUG();
        return -EINVAL;
        return -EINVAL;
}
}
 
 
static int crypto_init_ops(struct crypto_tfm *tfm)
static int crypto_init_ops(struct crypto_tfm *tfm)
{
{
        switch (crypto_tfm_alg_type(tfm)) {
        switch (crypto_tfm_alg_type(tfm)) {
        case CRYPTO_ALG_TYPE_CIPHER:
        case CRYPTO_ALG_TYPE_CIPHER:
                return crypto_init_cipher_ops(tfm);
                return crypto_init_cipher_ops(tfm);
 
 
        case CRYPTO_ALG_TYPE_DIGEST:
        case CRYPTO_ALG_TYPE_DIGEST:
                return crypto_init_digest_ops(tfm);
                return crypto_init_digest_ops(tfm);
 
 
        case CRYPTO_ALG_TYPE_COMPRESS:
        case CRYPTO_ALG_TYPE_COMPRESS:
                return crypto_init_compress_ops(tfm);
                return crypto_init_compress_ops(tfm);
 
 
        default:
        default:
                break;
                break;
        }
        }
 
 
        BUG();
        BUG();
        return -EINVAL;
        return -EINVAL;
}
}
 
 
static void crypto_exit_ops(struct crypto_tfm *tfm)
static void crypto_exit_ops(struct crypto_tfm *tfm)
{
{
        switch (crypto_tfm_alg_type(tfm)) {
        switch (crypto_tfm_alg_type(tfm)) {
        case CRYPTO_ALG_TYPE_CIPHER:
        case CRYPTO_ALG_TYPE_CIPHER:
                crypto_exit_cipher_ops(tfm);
                crypto_exit_cipher_ops(tfm);
                break;
                break;
 
 
        case CRYPTO_ALG_TYPE_DIGEST:
        case CRYPTO_ALG_TYPE_DIGEST:
                crypto_exit_digest_ops(tfm);
                crypto_exit_digest_ops(tfm);
                break;
                break;
 
 
        case CRYPTO_ALG_TYPE_COMPRESS:
        case CRYPTO_ALG_TYPE_COMPRESS:
                crypto_exit_compress_ops(tfm);
                crypto_exit_compress_ops(tfm);
                break;
                break;
 
 
        default:
        default:
                BUG();
                BUG();
 
 
        }
        }
}
}
 
 
struct crypto_tfm *crypto_alloc_tfm(const char *name, u32 flags)
struct crypto_tfm *crypto_alloc_tfm(const char *name, u32 flags)
{
{
        struct crypto_tfm *tfm = NULL;
        struct crypto_tfm *tfm = NULL;
        struct crypto_alg *alg;
        struct crypto_alg *alg;
 
 
        alg = crypto_alg_mod_lookup(name);
        alg = crypto_alg_mod_lookup(name);
        if (alg == NULL)
        if (alg == NULL)
                goto out;
                goto out;
 
 
        tfm = kmalloc(sizeof(*tfm) + alg->cra_ctxsize, GFP_KERNEL);
        tfm = kmalloc(sizeof(*tfm) + alg->cra_ctxsize, GFP_KERNEL);
        if (tfm == NULL)
        if (tfm == NULL)
                goto out_put;
                goto out_put;
 
 
        memset(tfm, 0, sizeof(*tfm) + alg->cra_ctxsize);
        memset(tfm, 0, sizeof(*tfm) + alg->cra_ctxsize);
 
 
        tfm->__crt_alg = alg;
        tfm->__crt_alg = alg;
 
 
        if (crypto_init_flags(tfm, flags))
        if (crypto_init_flags(tfm, flags))
                goto out_free_tfm;
                goto out_free_tfm;
 
 
        if (crypto_init_ops(tfm)) {
        if (crypto_init_ops(tfm)) {
                crypto_exit_ops(tfm);
                crypto_exit_ops(tfm);
                goto out_free_tfm;
                goto out_free_tfm;
        }
        }
 
 
        goto out;
        goto out;
 
 
out_free_tfm:
out_free_tfm:
        kfree(tfm);
        kfree(tfm);
        tfm = NULL;
        tfm = NULL;
out_put:
out_put:
        crypto_alg_put(alg);
        crypto_alg_put(alg);
out:
out:
        return tfm;
        return tfm;
}
}
 
 
void crypto_free_tfm(struct crypto_tfm *tfm)
void crypto_free_tfm(struct crypto_tfm *tfm)
{
{
        crypto_exit_ops(tfm);
        crypto_exit_ops(tfm);
        crypto_alg_put(tfm->__crt_alg);
        crypto_alg_put(tfm->__crt_alg);
        kfree(tfm);
        kfree(tfm);
}
}
 
 
int crypto_register_alg(struct crypto_alg *alg)
int crypto_register_alg(struct crypto_alg *alg)
{
{
        int ret = 0;
        int ret = 0;
        struct crypto_alg *q;
        struct crypto_alg *q;
 
 
        down_write(&crypto_alg_sem);
        down_write(&crypto_alg_sem);
 
 
        list_for_each_entry(q, &crypto_alg_list, cra_list) {
        list_for_each_entry(q, &crypto_alg_list, cra_list) {
                if (!(strcmp(q->cra_name, alg->cra_name))) {
                if (!(strcmp(q->cra_name, alg->cra_name))) {
                        ret = -EEXIST;
                        ret = -EEXIST;
                        goto out;
                        goto out;
                }
                }
        }
        }
 
 
        list_add_tail(&alg->cra_list, &crypto_alg_list);
        list_add_tail(&alg->cra_list, &crypto_alg_list);
out:
out:
        up_write(&crypto_alg_sem);
        up_write(&crypto_alg_sem);
        return ret;
        return ret;
}
}
 
 
int crypto_unregister_alg(struct crypto_alg *alg)
int crypto_unregister_alg(struct crypto_alg *alg)
{
{
        int ret = -ENOENT;
        int ret = -ENOENT;
        struct crypto_alg *q;
        struct crypto_alg *q;
 
 
        BUG_ON(!alg->cra_module);
        BUG_ON(!alg->cra_module);
 
 
        down_write(&crypto_alg_sem);
        down_write(&crypto_alg_sem);
        list_for_each_entry(q, &crypto_alg_list, cra_list) {
        list_for_each_entry(q, &crypto_alg_list, cra_list) {
                if (alg == q) {
                if (alg == q) {
                        list_del(&alg->cra_list);
                        list_del(&alg->cra_list);
                        ret = 0;
                        ret = 0;
                        goto out;
                        goto out;
                }
                }
        }
        }
out:
out:
        up_write(&crypto_alg_sem);
        up_write(&crypto_alg_sem);
        return ret;
        return ret;
}
}
 
 
int crypto_alg_available(const char *name, u32 flags)
int crypto_alg_available(const char *name, u32 flags)
{
{
        int ret = 0;
        int ret = 0;
        struct crypto_alg *alg = crypto_alg_mod_lookup(name);
        struct crypto_alg *alg = crypto_alg_mod_lookup(name);
 
 
        if (alg) {
        if (alg) {
                crypto_alg_put(alg);
                crypto_alg_put(alg);
                ret = 1;
                ret = 1;
        }
        }
 
 
        return ret;
        return ret;
}
}
 
 
static int __init init_crypto(void)
static int __init init_crypto(void)
{
{
        printk(KERN_INFO "Initializing Cryptographic API\n");
        printk(KERN_INFO "Initializing Cryptographic API\n");
        crypto_init_proc();
        crypto_init_proc();
        return 0;
        return 0;
}
}
 
 
__initcall(init_crypto);
__initcall(init_crypto);
 
 
EXPORT_SYMBOL_GPL(crypto_register_alg);
EXPORT_SYMBOL_GPL(crypto_register_alg);
EXPORT_SYMBOL_GPL(crypto_unregister_alg);
EXPORT_SYMBOL_GPL(crypto_unregister_alg);
EXPORT_SYMBOL_GPL(crypto_alloc_tfm);
EXPORT_SYMBOL_GPL(crypto_alloc_tfm);
EXPORT_SYMBOL_GPL(crypto_free_tfm);
EXPORT_SYMBOL_GPL(crypto_free_tfm);
EXPORT_SYMBOL_GPL(crypto_alg_available);
EXPORT_SYMBOL_GPL(crypto_alg_available);
 
 

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.