2009-08-10 09:50:03 +07:00
|
|
|
/*
|
|
|
|
* Support for Marvell's crypto engine which can be found on some Orion5X
|
|
|
|
* boards.
|
|
|
|
*
|
|
|
|
* Author: Sebastian Andrzej Siewior < sebastian at breakpoint dot cc >
|
|
|
|
* License: GPLv2
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include <crypto/aes.h>
|
|
|
|
#include <crypto/algapi.h>
|
|
|
|
#include <linux/crypto.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/io.h>
|
|
|
|
#include <linux/kthread.h>
|
|
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <linux/scatterlist.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 15:04:11 +07:00
|
|
|
#include <linux/slab.h>
|
2011-05-28 01:41:48 +07:00
|
|
|
#include <linux/module.h>
|
2012-02-19 17:56:19 +07:00
|
|
|
#include <linux/clk.h>
|
2010-04-08 23:34:55 +07:00
|
|
|
#include <crypto/internal/hash.h>
|
|
|
|
#include <crypto/sha.h>
|
2012-09-04 01:29:34 +07:00
|
|
|
#include <linux/of.h>
|
|
|
|
#include <linux/of_platform.h>
|
|
|
|
#include <linux/of_irq.h>
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
#include "mv_cesa.h"
|
2010-04-08 23:34:55 +07:00
|
|
|
|
|
|
|
#define MV_CESA "MV-CESA:"
|
|
|
|
#define MAX_HW_HASH_SIZE 0xFFFF
|
2012-05-25 20:54:46 +07:00
|
|
|
#define MV_CESA_EXPIRE 500 /* msec */
|
2010-04-08 23:34:55 +07:00
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
/*
|
|
|
|
* STM:
|
|
|
|
* /---------------------------------------\
|
|
|
|
* | | request complete
|
|
|
|
* \./ |
|
|
|
|
* IDLE -> new request -> BUSY -> done -> DEQUEUE
|
|
|
|
* /°\ |
|
|
|
|
* | | more scatter entries
|
|
|
|
* \________________/
|
|
|
|
*/
|
|
|
|
enum engine_status {
|
|
|
|
ENGINE_IDLE,
|
|
|
|
ENGINE_BUSY,
|
|
|
|
ENGINE_W_DEQUEUE,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* struct req_progress - used for every crypt request
|
|
|
|
* @src_sg_it: sg iterator for src
|
|
|
|
* @dst_sg_it: sg iterator for dst
|
|
|
|
* @sg_src_left: bytes left in src to process (scatter list)
|
|
|
|
* @src_start: offset to add to src start position (scatter list)
|
2010-04-08 23:34:55 +07:00
|
|
|
* @crypt_len: length of current hw crypt/hash process
|
2010-04-08 23:27:33 +07:00
|
|
|
* @hw_nbytes: total bytes to process in hw for this request
|
2010-04-08 23:31:48 +07:00
|
|
|
* @copy_back: whether to copy data back (crypt) or not (hash)
|
2009-08-10 09:50:03 +07:00
|
|
|
* @sg_dst_left: bytes left dst to process in this scatter list
|
|
|
|
* @dst_start: offset to add to dst start position (scatter list)
|
2010-04-08 23:29:16 +07:00
|
|
|
* @hw_processed_bytes: number of bytes processed by hw (request).
|
2009-08-10 09:50:03 +07:00
|
|
|
*
|
|
|
|
* sg helper are used to iterate over the scatterlist. Since the size of the
|
|
|
|
* SRAM may be less than the scatter size, this struct struct is used to keep
|
|
|
|
* track of progress within current scatterlist.
|
|
|
|
*/
|
|
|
|
struct req_progress {
|
|
|
|
struct sg_mapping_iter src_sg_it;
|
|
|
|
struct sg_mapping_iter dst_sg_it;
|
2010-04-08 23:30:19 +07:00
|
|
|
void (*complete) (void);
|
|
|
|
void (*process) (int is_first);
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
/* src mostly */
|
|
|
|
int sg_src_left;
|
|
|
|
int src_start;
|
|
|
|
int crypt_len;
|
2010-04-08 23:27:33 +07:00
|
|
|
int hw_nbytes;
|
2009-08-10 09:50:03 +07:00
|
|
|
/* dst mostly */
|
2010-04-08 23:31:48 +07:00
|
|
|
int copy_back;
|
2009-08-10 09:50:03 +07:00
|
|
|
int sg_dst_left;
|
|
|
|
int dst_start;
|
2010-04-08 23:29:16 +07:00
|
|
|
int hw_processed_bytes;
|
2009-08-10 09:50:03 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct crypto_priv {
|
|
|
|
void __iomem *reg;
|
|
|
|
void __iomem *sram;
|
|
|
|
int irq;
|
2012-02-19 17:56:19 +07:00
|
|
|
struct clk *clk;
|
2009-08-10 09:50:03 +07:00
|
|
|
struct task_struct *queue_th;
|
|
|
|
|
|
|
|
/* the lock protects queue and eng_st */
|
|
|
|
spinlock_t lock;
|
|
|
|
struct crypto_queue queue;
|
|
|
|
enum engine_status eng_st;
|
2012-05-25 20:54:46 +07:00
|
|
|
struct timer_list completion_timer;
|
2010-04-08 23:27:33 +07:00
|
|
|
struct crypto_async_request *cur_req;
|
2009-08-10 09:50:03 +07:00
|
|
|
struct req_progress p;
|
|
|
|
int max_req_size;
|
|
|
|
int sram_size;
|
2010-04-08 23:34:55 +07:00
|
|
|
int has_sha1;
|
|
|
|
int has_hmac_sha1;
|
2009-08-10 09:50:03 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct crypto_priv *cpg;
|
|
|
|
|
|
|
|
struct mv_ctx {
|
|
|
|
u8 aes_enc_key[AES_KEY_LEN];
|
|
|
|
u32 aes_dec_key[8];
|
|
|
|
int key_len;
|
|
|
|
u32 need_calc_aes_dkey;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum crypto_op {
|
|
|
|
COP_AES_ECB,
|
|
|
|
COP_AES_CBC,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct mv_req_ctx {
|
|
|
|
enum crypto_op op;
|
|
|
|
int decrypt;
|
|
|
|
};
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
enum hash_op {
|
|
|
|
COP_SHA1,
|
|
|
|
COP_HMAC_SHA1
|
|
|
|
};
|
|
|
|
|
|
|
|
struct mv_tfm_hash_ctx {
|
|
|
|
struct crypto_shash *fallback;
|
|
|
|
struct crypto_shash *base_hash;
|
|
|
|
u32 ivs[2 * SHA1_DIGEST_SIZE / 4];
|
|
|
|
int count_add;
|
|
|
|
enum hash_op op;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct mv_req_hash_ctx {
|
|
|
|
u64 count;
|
|
|
|
u32 state[SHA1_DIGEST_SIZE / 4];
|
|
|
|
u8 buffer[SHA1_BLOCK_SIZE];
|
|
|
|
int first_hash; /* marks that we don't have previous state */
|
|
|
|
int last_chunk; /* marks that this is the 'final' request */
|
|
|
|
int extra_bytes; /* unprocessed bytes in buffer */
|
|
|
|
enum hash_op op;
|
|
|
|
int count_add;
|
|
|
|
};
|
|
|
|
|
2012-05-25 20:54:46 +07:00
|
|
|
static void mv_completion_timer_callback(unsigned long unused)
|
|
|
|
{
|
|
|
|
int active = readl(cpg->reg + SEC_ACCEL_CMD) & SEC_CMD_EN_SEC_ACCL0;
|
|
|
|
|
|
|
|
printk(KERN_ERR MV_CESA
|
|
|
|
"completion timer expired (CESA %sactive), cleaning up.\n",
|
|
|
|
active ? "" : "in");
|
|
|
|
|
|
|
|
del_timer(&cpg->completion_timer);
|
|
|
|
writel(SEC_CMD_DISABLE_SEC, cpg->reg + SEC_ACCEL_CMD);
|
|
|
|
while(readl(cpg->reg + SEC_ACCEL_CMD) & SEC_CMD_DISABLE_SEC)
|
|
|
|
printk(KERN_INFO MV_CESA "%s: waiting for engine finishing\n", __func__);
|
|
|
|
cpg->eng_st = ENGINE_W_DEQUEUE;
|
|
|
|
wake_up_process(cpg->queue_th);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mv_setup_timer(void)
|
|
|
|
{
|
|
|
|
setup_timer(&cpg->completion_timer, &mv_completion_timer_callback, 0);
|
|
|
|
mod_timer(&cpg->completion_timer,
|
|
|
|
jiffies + msecs_to_jiffies(MV_CESA_EXPIRE));
|
|
|
|
}
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
static void compute_aes_dec_key(struct mv_ctx *ctx)
|
|
|
|
{
|
|
|
|
struct crypto_aes_ctx gen_aes_key;
|
|
|
|
int key_pos;
|
|
|
|
|
|
|
|
if (!ctx->need_calc_aes_dkey)
|
|
|
|
return;
|
|
|
|
|
|
|
|
crypto_aes_expand_key(&gen_aes_key, ctx->aes_enc_key, ctx->key_len);
|
|
|
|
|
|
|
|
key_pos = ctx->key_len + 24;
|
|
|
|
memcpy(ctx->aes_dec_key, &gen_aes_key.key_enc[key_pos], 4 * 4);
|
|
|
|
switch (ctx->key_len) {
|
|
|
|
case AES_KEYSIZE_256:
|
|
|
|
key_pos -= 2;
|
|
|
|
/* fall */
|
|
|
|
case AES_KEYSIZE_192:
|
|
|
|
key_pos -= 2;
|
|
|
|
memcpy(&ctx->aes_dec_key[4], &gen_aes_key.key_enc[key_pos],
|
|
|
|
4 * 4);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ctx->need_calc_aes_dkey = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_setkey_aes(struct crypto_ablkcipher *cipher, const u8 *key,
|
|
|
|
unsigned int len)
|
|
|
|
{
|
|
|
|
struct crypto_tfm *tfm = crypto_ablkcipher_tfm(cipher);
|
|
|
|
struct mv_ctx *ctx = crypto_tfm_ctx(tfm);
|
|
|
|
|
|
|
|
switch (len) {
|
|
|
|
case AES_KEYSIZE_128:
|
|
|
|
case AES_KEYSIZE_192:
|
|
|
|
case AES_KEYSIZE_256:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
crypto_ablkcipher_set_flags(cipher, CRYPTO_TFM_RES_BAD_KEY_LEN);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
ctx->key_len = len;
|
|
|
|
ctx->need_calc_aes_dkey = 1;
|
|
|
|
|
|
|
|
memcpy(ctx->aes_enc_key, key, AES_KEY_LEN);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:27:02 +07:00
|
|
|
static void copy_src_to_buf(struct req_progress *p, char *dbuf, int len)
|
2009-08-10 09:50:03 +07:00
|
|
|
{
|
|
|
|
int ret;
|
2010-04-08 23:27:02 +07:00
|
|
|
void *sbuf;
|
2011-05-05 20:29:02 +07:00
|
|
|
int copy_len;
|
2009-08-10 09:50:03 +07:00
|
|
|
|
2011-05-05 20:29:02 +07:00
|
|
|
while (len) {
|
2010-04-08 23:27:02 +07:00
|
|
|
if (!p->sg_src_left) {
|
|
|
|
ret = sg_miter_next(&p->src_sg_it);
|
|
|
|
BUG_ON(!ret);
|
|
|
|
p->sg_src_left = p->src_sg_it.length;
|
|
|
|
p->src_start = 0;
|
|
|
|
}
|
2009-08-10 09:50:03 +07:00
|
|
|
|
2010-04-08 23:27:02 +07:00
|
|
|
sbuf = p->src_sg_it.addr + p->src_start;
|
|
|
|
|
2011-05-05 20:29:02 +07:00
|
|
|
copy_len = min(p->sg_src_left, len);
|
|
|
|
memcpy(dbuf, sbuf, copy_len);
|
|
|
|
|
|
|
|
p->src_start += copy_len;
|
|
|
|
p->sg_src_left -= copy_len;
|
|
|
|
|
|
|
|
len -= copy_len;
|
|
|
|
dbuf += copy_len;
|
2010-04-08 23:27:02 +07:00
|
|
|
}
|
|
|
|
}
|
2009-08-10 09:50:03 +07:00
|
|
|
|
2010-04-08 23:27:33 +07:00
|
|
|
static void setup_data_in(void)
|
2010-04-08 23:27:02 +07:00
|
|
|
{
|
|
|
|
struct req_progress *p = &cpg->p;
|
2010-04-08 23:33:26 +07:00
|
|
|
int data_in_sram =
|
2010-04-08 23:29:16 +07:00
|
|
|
min(p->hw_nbytes - p->hw_processed_bytes, cpg->max_req_size);
|
2010-04-08 23:33:26 +07:00
|
|
|
copy_src_to_buf(p, cpg->sram + SRAM_DATA_IN_START + p->crypt_len,
|
|
|
|
data_in_sram - p->crypt_len);
|
|
|
|
p->crypt_len = data_in_sram;
|
2009-08-10 09:50:03 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void mv_process_current_q(int first_block)
|
|
|
|
{
|
2010-04-08 23:27:33 +07:00
|
|
|
struct ablkcipher_request *req = ablkcipher_request_cast(cpg->cur_req);
|
2009-08-10 09:50:03 +07:00
|
|
|
struct mv_ctx *ctx = crypto_tfm_ctx(req->base.tfm);
|
|
|
|
struct mv_req_ctx *req_ctx = ablkcipher_request_ctx(req);
|
|
|
|
struct sec_accel_config op;
|
|
|
|
|
|
|
|
switch (req_ctx->op) {
|
|
|
|
case COP_AES_ECB:
|
|
|
|
op.config = CFG_OP_CRYPT_ONLY | CFG_ENCM_AES | CFG_ENC_MODE_ECB;
|
|
|
|
break;
|
|
|
|
case COP_AES_CBC:
|
2010-04-08 23:25:56 +07:00
|
|
|
default:
|
2009-08-10 09:50:03 +07:00
|
|
|
op.config = CFG_OP_CRYPT_ONLY | CFG_ENCM_AES | CFG_ENC_MODE_CBC;
|
|
|
|
op.enc_iv = ENC_IV_POINT(SRAM_DATA_IV) |
|
|
|
|
ENC_IV_BUF_POINT(SRAM_DATA_IV_BUF);
|
|
|
|
if (first_block)
|
|
|
|
memcpy(cpg->sram + SRAM_DATA_IV, req->info, 16);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (req_ctx->decrypt) {
|
|
|
|
op.config |= CFG_DIR_DEC;
|
|
|
|
memcpy(cpg->sram + SRAM_DATA_KEY_P, ctx->aes_dec_key,
|
|
|
|
AES_KEY_LEN);
|
|
|
|
} else {
|
|
|
|
op.config |= CFG_DIR_ENC;
|
|
|
|
memcpy(cpg->sram + SRAM_DATA_KEY_P, ctx->aes_enc_key,
|
|
|
|
AES_KEY_LEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (ctx->key_len) {
|
|
|
|
case AES_KEYSIZE_128:
|
|
|
|
op.config |= CFG_AES_LEN_128;
|
|
|
|
break;
|
|
|
|
case AES_KEYSIZE_192:
|
|
|
|
op.config |= CFG_AES_LEN_192;
|
|
|
|
break;
|
|
|
|
case AES_KEYSIZE_256:
|
|
|
|
op.config |= CFG_AES_LEN_256;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
op.enc_p = ENC_P_SRC(SRAM_DATA_IN_START) |
|
|
|
|
ENC_P_DST(SRAM_DATA_OUT_START);
|
|
|
|
op.enc_key_p = SRAM_DATA_KEY_P;
|
|
|
|
|
2010-04-08 23:27:33 +07:00
|
|
|
setup_data_in();
|
2009-08-10 09:50:03 +07:00
|
|
|
op.enc_len = cpg->p.crypt_len;
|
|
|
|
memcpy(cpg->sram + SRAM_CONFIG, &op,
|
|
|
|
sizeof(struct sec_accel_config));
|
|
|
|
|
|
|
|
/* GO */
|
2012-05-25 20:54:46 +07:00
|
|
|
mv_setup_timer();
|
2009-08-10 09:50:03 +07:00
|
|
|
writel(SEC_CMD_EN_SEC_ACCL0, cpg->reg + SEC_ACCEL_CMD);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mv_crypto_algo_completion(void)
|
|
|
|
{
|
2010-04-08 23:27:33 +07:00
|
|
|
struct ablkcipher_request *req = ablkcipher_request_cast(cpg->cur_req);
|
2009-08-10 09:50:03 +07:00
|
|
|
struct mv_req_ctx *req_ctx = ablkcipher_request_ctx(req);
|
|
|
|
|
2010-04-08 23:30:19 +07:00
|
|
|
sg_miter_stop(&cpg->p.src_sg_it);
|
|
|
|
sg_miter_stop(&cpg->p.dst_sg_it);
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
if (req_ctx->op != COP_AES_CBC)
|
|
|
|
return ;
|
|
|
|
|
|
|
|
memcpy(req->info, cpg->sram + SRAM_DATA_IV_BUF, 16);
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
static void mv_process_hash_current(int first_block)
|
|
|
|
{
|
|
|
|
struct ahash_request *req = ahash_request_cast(cpg->cur_req);
|
2011-05-05 20:29:03 +07:00
|
|
|
const struct mv_tfm_hash_ctx *tfm_ctx = crypto_tfm_ctx(req->base.tfm);
|
2010-04-08 23:34:55 +07:00
|
|
|
struct mv_req_hash_ctx *req_ctx = ahash_request_ctx(req);
|
|
|
|
struct req_progress *p = &cpg->p;
|
|
|
|
struct sec_accel_config op = { 0 };
|
|
|
|
int is_last;
|
|
|
|
|
|
|
|
switch (req_ctx->op) {
|
|
|
|
case COP_SHA1:
|
|
|
|
default:
|
|
|
|
op.config = CFG_OP_MAC_ONLY | CFG_MACM_SHA1;
|
|
|
|
break;
|
|
|
|
case COP_HMAC_SHA1:
|
|
|
|
op.config = CFG_OP_MAC_ONLY | CFG_MACM_HMAC_SHA1;
|
2011-05-05 20:29:03 +07:00
|
|
|
memcpy(cpg->sram + SRAM_HMAC_IV_IN,
|
|
|
|
tfm_ctx->ivs, sizeof(tfm_ctx->ivs));
|
2010-04-08 23:34:55 +07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
op.mac_src_p =
|
|
|
|
MAC_SRC_DATA_P(SRAM_DATA_IN_START) | MAC_SRC_TOTAL_LEN((u32)
|
|
|
|
req_ctx->
|
|
|
|
count);
|
|
|
|
|
|
|
|
setup_data_in();
|
|
|
|
|
|
|
|
op.mac_digest =
|
|
|
|
MAC_DIGEST_P(SRAM_DIGEST_BUF) | MAC_FRAG_LEN(p->crypt_len);
|
|
|
|
op.mac_iv =
|
|
|
|
MAC_INNER_IV_P(SRAM_HMAC_IV_IN) |
|
|
|
|
MAC_OUTER_IV_P(SRAM_HMAC_IV_OUT);
|
|
|
|
|
|
|
|
is_last = req_ctx->last_chunk
|
|
|
|
&& (p->hw_processed_bytes + p->crypt_len >= p->hw_nbytes)
|
|
|
|
&& (req_ctx->count <= MAX_HW_HASH_SIZE);
|
|
|
|
if (req_ctx->first_hash) {
|
|
|
|
if (is_last)
|
|
|
|
op.config |= CFG_NOT_FRAG;
|
|
|
|
else
|
|
|
|
op.config |= CFG_FIRST_FRAG;
|
|
|
|
|
|
|
|
req_ctx->first_hash = 0;
|
|
|
|
} else {
|
|
|
|
if (is_last)
|
|
|
|
op.config |= CFG_LAST_FRAG;
|
|
|
|
else
|
|
|
|
op.config |= CFG_MID_FRAG;
|
2011-05-05 20:29:04 +07:00
|
|
|
|
2011-11-17 00:28:01 +07:00
|
|
|
if (first_block) {
|
|
|
|
writel(req_ctx->state[0], cpg->reg + DIGEST_INITIAL_VAL_A);
|
|
|
|
writel(req_ctx->state[1], cpg->reg + DIGEST_INITIAL_VAL_B);
|
|
|
|
writel(req_ctx->state[2], cpg->reg + DIGEST_INITIAL_VAL_C);
|
|
|
|
writel(req_ctx->state[3], cpg->reg + DIGEST_INITIAL_VAL_D);
|
|
|
|
writel(req_ctx->state[4], cpg->reg + DIGEST_INITIAL_VAL_E);
|
|
|
|
}
|
2010-04-08 23:34:55 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(cpg->sram + SRAM_CONFIG, &op, sizeof(struct sec_accel_config));
|
|
|
|
|
|
|
|
/* GO */
|
2012-05-25 20:54:46 +07:00
|
|
|
mv_setup_timer();
|
2010-04-08 23:34:55 +07:00
|
|
|
writel(SEC_CMD_EN_SEC_ACCL0, cpg->reg + SEC_ACCEL_CMD);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int mv_hash_import_sha1_ctx(const struct mv_req_hash_ctx *ctx,
|
|
|
|
struct shash_desc *desc)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct sha1_state shash_state;
|
|
|
|
|
|
|
|
shash_state.count = ctx->count + ctx->count_add;
|
|
|
|
for (i = 0; i < 5; i++)
|
|
|
|
shash_state.state[i] = ctx->state[i];
|
|
|
|
memcpy(shash_state.buffer, ctx->buffer, sizeof(shash_state.buffer));
|
|
|
|
return crypto_shash_import(desc, &shash_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_hash_final_fallback(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
const struct mv_tfm_hash_ctx *tfm_ctx = crypto_tfm_ctx(req->base.tfm);
|
|
|
|
struct mv_req_hash_ctx *req_ctx = ahash_request_ctx(req);
|
|
|
|
struct {
|
|
|
|
struct shash_desc shash;
|
|
|
|
char ctx[crypto_shash_descsize(tfm_ctx->fallback)];
|
|
|
|
} desc;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
desc.shash.tfm = tfm_ctx->fallback;
|
|
|
|
desc.shash.flags = CRYPTO_TFM_REQ_MAY_SLEEP;
|
|
|
|
if (unlikely(req_ctx->first_hash)) {
|
|
|
|
crypto_shash_init(&desc.shash);
|
|
|
|
crypto_shash_update(&desc.shash, req_ctx->buffer,
|
|
|
|
req_ctx->extra_bytes);
|
|
|
|
} else {
|
|
|
|
/* only SHA1 for now....
|
|
|
|
*/
|
|
|
|
rc = mv_hash_import_sha1_ctx(req_ctx, &desc.shash);
|
|
|
|
if (rc)
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
rc = crypto_shash_final(&desc.shash, req->result);
|
|
|
|
out:
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2012-05-25 20:54:49 +07:00
|
|
|
static void mv_save_digest_state(struct mv_req_hash_ctx *ctx)
|
|
|
|
{
|
|
|
|
ctx->state[0] = readl(cpg->reg + DIGEST_INITIAL_VAL_A);
|
|
|
|
ctx->state[1] = readl(cpg->reg + DIGEST_INITIAL_VAL_B);
|
|
|
|
ctx->state[2] = readl(cpg->reg + DIGEST_INITIAL_VAL_C);
|
|
|
|
ctx->state[3] = readl(cpg->reg + DIGEST_INITIAL_VAL_D);
|
|
|
|
ctx->state[4] = readl(cpg->reg + DIGEST_INITIAL_VAL_E);
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
static void mv_hash_algo_completion(void)
|
|
|
|
{
|
|
|
|
struct ahash_request *req = ahash_request_cast(cpg->cur_req);
|
|
|
|
struct mv_req_hash_ctx *ctx = ahash_request_ctx(req);
|
|
|
|
|
|
|
|
if (ctx->extra_bytes)
|
|
|
|
copy_src_to_buf(&cpg->p, ctx->buffer, ctx->extra_bytes);
|
|
|
|
sg_miter_stop(&cpg->p.src_sg_it);
|
|
|
|
|
|
|
|
if (likely(ctx->last_chunk)) {
|
|
|
|
if (likely(ctx->count <= MAX_HW_HASH_SIZE)) {
|
|
|
|
memcpy(req->result, cpg->sram + SRAM_DIGEST_BUF,
|
|
|
|
crypto_ahash_digestsize(crypto_ahash_reqtfm
|
|
|
|
(req)));
|
2012-05-25 20:54:49 +07:00
|
|
|
} else {
|
|
|
|
mv_save_digest_state(ctx);
|
2010-04-08 23:34:55 +07:00
|
|
|
mv_hash_final_fallback(req);
|
2012-05-25 20:54:49 +07:00
|
|
|
}
|
2011-05-05 20:29:01 +07:00
|
|
|
} else {
|
2012-05-25 20:54:49 +07:00
|
|
|
mv_save_digest_state(ctx);
|
2010-04-08 23:34:55 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
static void dequeue_complete_req(void)
|
|
|
|
{
|
2010-04-08 23:27:33 +07:00
|
|
|
struct crypto_async_request *req = cpg->cur_req;
|
2009-08-10 09:50:03 +07:00
|
|
|
void *buf;
|
|
|
|
int ret;
|
2010-04-08 23:29:16 +07:00
|
|
|
cpg->p.hw_processed_bytes += cpg->p.crypt_len;
|
2010-04-08 23:31:48 +07:00
|
|
|
if (cpg->p.copy_back) {
|
|
|
|
int need_copy_len = cpg->p.crypt_len;
|
|
|
|
int sram_offset = 0;
|
|
|
|
do {
|
|
|
|
int dst_copy;
|
|
|
|
|
|
|
|
if (!cpg->p.sg_dst_left) {
|
|
|
|
ret = sg_miter_next(&cpg->p.dst_sg_it);
|
|
|
|
BUG_ON(!ret);
|
|
|
|
cpg->p.sg_dst_left = cpg->p.dst_sg_it.length;
|
|
|
|
cpg->p.dst_start = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf = cpg->p.dst_sg_it.addr;
|
|
|
|
buf += cpg->p.dst_start;
|
|
|
|
|
|
|
|
dst_copy = min(need_copy_len, cpg->p.sg_dst_left);
|
|
|
|
|
|
|
|
memcpy(buf,
|
|
|
|
cpg->sram + SRAM_DATA_OUT_START + sram_offset,
|
|
|
|
dst_copy);
|
|
|
|
sram_offset += dst_copy;
|
|
|
|
cpg->p.sg_dst_left -= dst_copy;
|
|
|
|
need_copy_len -= dst_copy;
|
|
|
|
cpg->p.dst_start += dst_copy;
|
|
|
|
} while (need_copy_len > 0);
|
|
|
|
}
|
2009-08-10 09:50:03 +07:00
|
|
|
|
2010-04-08 23:33:26 +07:00
|
|
|
cpg->p.crypt_len = 0;
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
BUG_ON(cpg->eng_st != ENGINE_W_DEQUEUE);
|
2010-04-08 23:29:16 +07:00
|
|
|
if (cpg->p.hw_processed_bytes < cpg->p.hw_nbytes) {
|
2009-08-10 09:50:03 +07:00
|
|
|
/* process next scatter list entry */
|
|
|
|
cpg->eng_st = ENGINE_BUSY;
|
2010-04-08 23:30:19 +07:00
|
|
|
cpg->p.process(0);
|
2009-08-10 09:50:03 +07:00
|
|
|
} else {
|
2010-04-08 23:30:19 +07:00
|
|
|
cpg->p.complete();
|
2009-08-10 09:50:03 +07:00
|
|
|
cpg->eng_st = ENGINE_IDLE;
|
2010-04-08 23:25:37 +07:00
|
|
|
local_bh_disable();
|
2010-04-08 23:27:33 +07:00
|
|
|
req->complete(req, 0);
|
2010-04-08 23:25:37 +07:00
|
|
|
local_bh_enable();
|
2009-08-10 09:50:03 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int count_sgs(struct scatterlist *sl, unsigned int total_bytes)
|
|
|
|
{
|
|
|
|
int i = 0;
|
2010-04-08 23:27:02 +07:00
|
|
|
size_t cur_len;
|
|
|
|
|
2011-05-05 20:29:06 +07:00
|
|
|
while (sl) {
|
2010-04-08 23:27:02 +07:00
|
|
|
cur_len = sl[i].length;
|
|
|
|
++i;
|
|
|
|
if (total_bytes > cur_len)
|
|
|
|
total_bytes -= cur_len;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
static void mv_start_new_crypt_req(struct ablkcipher_request *req)
|
2009-08-10 09:50:03 +07:00
|
|
|
{
|
2010-04-08 23:27:33 +07:00
|
|
|
struct req_progress *p = &cpg->p;
|
2009-08-10 09:50:03 +07:00
|
|
|
int num_sgs;
|
|
|
|
|
2010-04-08 23:27:33 +07:00
|
|
|
cpg->cur_req = &req->base;
|
|
|
|
memset(p, 0, sizeof(struct req_progress));
|
|
|
|
p->hw_nbytes = req->nbytes;
|
2010-04-08 23:30:19 +07:00
|
|
|
p->complete = mv_crypto_algo_completion;
|
|
|
|
p->process = mv_process_current_q;
|
2010-04-08 23:31:48 +07:00
|
|
|
p->copy_back = 1;
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
num_sgs = count_sgs(req->src, req->nbytes);
|
2010-04-08 23:27:33 +07:00
|
|
|
sg_miter_start(&p->src_sg_it, req->src, num_sgs, SG_MITER_FROM_SG);
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
num_sgs = count_sgs(req->dst, req->nbytes);
|
2010-04-08 23:27:33 +07:00
|
|
|
sg_miter_start(&p->dst_sg_it, req->dst, num_sgs, SG_MITER_TO_SG);
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
mv_process_current_q(1);
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
static void mv_start_new_hash_req(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
struct req_progress *p = &cpg->p;
|
|
|
|
struct mv_req_hash_ctx *ctx = ahash_request_ctx(req);
|
|
|
|
int num_sgs, hw_bytes, old_extra_bytes, rc;
|
|
|
|
cpg->cur_req = &req->base;
|
|
|
|
memset(p, 0, sizeof(struct req_progress));
|
|
|
|
hw_bytes = req->nbytes + ctx->extra_bytes;
|
|
|
|
old_extra_bytes = ctx->extra_bytes;
|
|
|
|
|
|
|
|
ctx->extra_bytes = hw_bytes % SHA1_BLOCK_SIZE;
|
|
|
|
if (ctx->extra_bytes != 0
|
|
|
|
&& (!ctx->last_chunk || ctx->count > MAX_HW_HASH_SIZE))
|
|
|
|
hw_bytes -= ctx->extra_bytes;
|
|
|
|
else
|
|
|
|
ctx->extra_bytes = 0;
|
|
|
|
|
|
|
|
num_sgs = count_sgs(req->src, req->nbytes);
|
|
|
|
sg_miter_start(&p->src_sg_it, req->src, num_sgs, SG_MITER_FROM_SG);
|
|
|
|
|
|
|
|
if (hw_bytes) {
|
|
|
|
p->hw_nbytes = hw_bytes;
|
|
|
|
p->complete = mv_hash_algo_completion;
|
|
|
|
p->process = mv_process_hash_current;
|
|
|
|
|
2011-05-05 20:29:05 +07:00
|
|
|
if (unlikely(old_extra_bytes)) {
|
|
|
|
memcpy(cpg->sram + SRAM_DATA_IN_START, ctx->buffer,
|
|
|
|
old_extra_bytes);
|
|
|
|
p->crypt_len = old_extra_bytes;
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
mv_process_hash_current(1);
|
|
|
|
} else {
|
|
|
|
copy_src_to_buf(p, ctx->buffer + old_extra_bytes,
|
|
|
|
ctx->extra_bytes - old_extra_bytes);
|
|
|
|
sg_miter_stop(&p->src_sg_it);
|
|
|
|
if (ctx->last_chunk)
|
|
|
|
rc = mv_hash_final_fallback(req);
|
|
|
|
else
|
|
|
|
rc = 0;
|
|
|
|
cpg->eng_st = ENGINE_IDLE;
|
|
|
|
local_bh_disable();
|
|
|
|
req->base.complete(&req->base, rc);
|
|
|
|
local_bh_enable();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
static int queue_manag(void *data)
|
|
|
|
{
|
|
|
|
cpg->eng_st = ENGINE_IDLE;
|
|
|
|
do {
|
|
|
|
struct crypto_async_request *async_req = NULL;
|
|
|
|
struct crypto_async_request *backlog;
|
|
|
|
|
|
|
|
__set_current_state(TASK_INTERRUPTIBLE);
|
|
|
|
|
|
|
|
if (cpg->eng_st == ENGINE_W_DEQUEUE)
|
|
|
|
dequeue_complete_req();
|
|
|
|
|
|
|
|
spin_lock_irq(&cpg->lock);
|
|
|
|
if (cpg->eng_st == ENGINE_IDLE) {
|
|
|
|
backlog = crypto_get_backlog(&cpg->queue);
|
|
|
|
async_req = crypto_dequeue_request(&cpg->queue);
|
|
|
|
if (async_req) {
|
|
|
|
BUG_ON(cpg->eng_st != ENGINE_IDLE);
|
|
|
|
cpg->eng_st = ENGINE_BUSY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&cpg->lock);
|
|
|
|
|
|
|
|
if (backlog) {
|
|
|
|
backlog->complete(backlog, -EINPROGRESS);
|
|
|
|
backlog = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (async_req) {
|
2010-04-08 23:34:55 +07:00
|
|
|
if (async_req->tfm->__crt_alg->cra_type !=
|
|
|
|
&crypto_ahash_type) {
|
|
|
|
struct ablkcipher_request *req =
|
2011-05-05 20:28:57 +07:00
|
|
|
ablkcipher_request_cast(async_req);
|
2010-04-08 23:34:55 +07:00
|
|
|
mv_start_new_crypt_req(req);
|
|
|
|
} else {
|
|
|
|
struct ahash_request *req =
|
|
|
|
ahash_request_cast(async_req);
|
|
|
|
mv_start_new_hash_req(req);
|
|
|
|
}
|
2009-08-10 09:50:03 +07:00
|
|
|
async_req = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
schedule();
|
|
|
|
|
|
|
|
} while (!kthread_should_stop());
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:27:33 +07:00
|
|
|
static int mv_handle_req(struct crypto_async_request *req)
|
2009-08-10 09:50:03 +07:00
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
spin_lock_irqsave(&cpg->lock, flags);
|
2010-04-08 23:27:33 +07:00
|
|
|
ret = crypto_enqueue_request(&cpg->queue, req);
|
2009-08-10 09:50:03 +07:00
|
|
|
spin_unlock_irqrestore(&cpg->lock, flags);
|
|
|
|
wake_up_process(cpg->queue_th);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_enc_aes_ecb(struct ablkcipher_request *req)
|
|
|
|
{
|
|
|
|
struct mv_req_ctx *req_ctx = ablkcipher_request_ctx(req);
|
|
|
|
|
|
|
|
req_ctx->op = COP_AES_ECB;
|
|
|
|
req_ctx->decrypt = 0;
|
|
|
|
|
2010-04-08 23:27:33 +07:00
|
|
|
return mv_handle_req(&req->base);
|
2009-08-10 09:50:03 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_dec_aes_ecb(struct ablkcipher_request *req)
|
|
|
|
{
|
|
|
|
struct mv_ctx *ctx = crypto_tfm_ctx(req->base.tfm);
|
|
|
|
struct mv_req_ctx *req_ctx = ablkcipher_request_ctx(req);
|
|
|
|
|
|
|
|
req_ctx->op = COP_AES_ECB;
|
|
|
|
req_ctx->decrypt = 1;
|
|
|
|
|
|
|
|
compute_aes_dec_key(ctx);
|
2010-04-08 23:27:33 +07:00
|
|
|
return mv_handle_req(&req->base);
|
2009-08-10 09:50:03 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_enc_aes_cbc(struct ablkcipher_request *req)
|
|
|
|
{
|
|
|
|
struct mv_req_ctx *req_ctx = ablkcipher_request_ctx(req);
|
|
|
|
|
|
|
|
req_ctx->op = COP_AES_CBC;
|
|
|
|
req_ctx->decrypt = 0;
|
|
|
|
|
2010-04-08 23:27:33 +07:00
|
|
|
return mv_handle_req(&req->base);
|
2009-08-10 09:50:03 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_dec_aes_cbc(struct ablkcipher_request *req)
|
|
|
|
{
|
|
|
|
struct mv_ctx *ctx = crypto_tfm_ctx(req->base.tfm);
|
|
|
|
struct mv_req_ctx *req_ctx = ablkcipher_request_ctx(req);
|
|
|
|
|
|
|
|
req_ctx->op = COP_AES_CBC;
|
|
|
|
req_ctx->decrypt = 1;
|
|
|
|
|
|
|
|
compute_aes_dec_key(ctx);
|
2010-04-08 23:27:33 +07:00
|
|
|
return mv_handle_req(&req->base);
|
2009-08-10 09:50:03 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_cra_init(struct crypto_tfm *tfm)
|
|
|
|
{
|
|
|
|
tfm->crt_ablkcipher.reqsize = sizeof(struct mv_req_ctx);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
static void mv_init_hash_req_ctx(struct mv_req_hash_ctx *ctx, int op,
|
|
|
|
int is_last, unsigned int req_len,
|
|
|
|
int count_add)
|
|
|
|
{
|
|
|
|
memset(ctx, 0, sizeof(*ctx));
|
|
|
|
ctx->op = op;
|
|
|
|
ctx->count = req_len;
|
|
|
|
ctx->first_hash = 1;
|
|
|
|
ctx->last_chunk = is_last;
|
|
|
|
ctx->count_add = count_add;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mv_update_hash_req_ctx(struct mv_req_hash_ctx *ctx, int is_last,
|
|
|
|
unsigned req_len)
|
|
|
|
{
|
|
|
|
ctx->last_chunk = is_last;
|
|
|
|
ctx->count += req_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_hash_init(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
const struct mv_tfm_hash_ctx *tfm_ctx = crypto_tfm_ctx(req->base.tfm);
|
|
|
|
mv_init_hash_req_ctx(ahash_request_ctx(req), tfm_ctx->op, 0, 0,
|
|
|
|
tfm_ctx->count_add);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_hash_update(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
if (!req->nbytes)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
mv_update_hash_req_ctx(ahash_request_ctx(req), 0, req->nbytes);
|
|
|
|
return mv_handle_req(&req->base);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_hash_final(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
struct mv_req_hash_ctx *ctx = ahash_request_ctx(req);
|
2011-05-05 20:29:06 +07:00
|
|
|
|
2012-02-27 18:17:04 +07:00
|
|
|
ahash_request_set_crypt(req, NULL, req->result, 0);
|
2010-04-08 23:34:55 +07:00
|
|
|
mv_update_hash_req_ctx(ctx, 1, 0);
|
|
|
|
return mv_handle_req(&req->base);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_hash_finup(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
mv_update_hash_req_ctx(ahash_request_ctx(req), 1, req->nbytes);
|
|
|
|
return mv_handle_req(&req->base);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_hash_digest(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
const struct mv_tfm_hash_ctx *tfm_ctx = crypto_tfm_ctx(req->base.tfm);
|
|
|
|
mv_init_hash_req_ctx(ahash_request_ctx(req), tfm_ctx->op, 1,
|
|
|
|
req->nbytes, tfm_ctx->count_add);
|
|
|
|
return mv_handle_req(&req->base);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mv_hash_init_ivs(struct mv_tfm_hash_ctx *ctx, const void *istate,
|
|
|
|
const void *ostate)
|
|
|
|
{
|
|
|
|
const struct sha1_state *isha1_state = istate, *osha1_state = ostate;
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 5; i++) {
|
|
|
|
ctx->ivs[i] = cpu_to_be32(isha1_state->state[i]);
|
|
|
|
ctx->ivs[i + 5] = cpu_to_be32(osha1_state->state[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_hash_setkey(struct crypto_ahash *tfm, const u8 * key,
|
|
|
|
unsigned int keylen)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
struct mv_tfm_hash_ctx *ctx = crypto_tfm_ctx(&tfm->base);
|
|
|
|
int bs, ds, ss;
|
|
|
|
|
|
|
|
if (!ctx->base_hash)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
rc = crypto_shash_setkey(ctx->fallback, key, keylen);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
/* Can't see a way to extract the ipad/opad from the fallback tfm
|
|
|
|
so I'm basically copying code from the hmac module */
|
|
|
|
bs = crypto_shash_blocksize(ctx->base_hash);
|
|
|
|
ds = crypto_shash_digestsize(ctx->base_hash);
|
|
|
|
ss = crypto_shash_statesize(ctx->base_hash);
|
|
|
|
|
|
|
|
{
|
|
|
|
struct {
|
|
|
|
struct shash_desc shash;
|
|
|
|
char ctx[crypto_shash_descsize(ctx->base_hash)];
|
|
|
|
} desc;
|
|
|
|
unsigned int i;
|
|
|
|
char ipad[ss];
|
|
|
|
char opad[ss];
|
|
|
|
|
|
|
|
desc.shash.tfm = ctx->base_hash;
|
|
|
|
desc.shash.flags = crypto_shash_get_flags(ctx->base_hash) &
|
|
|
|
CRYPTO_TFM_REQ_MAY_SLEEP;
|
|
|
|
|
|
|
|
if (keylen > bs) {
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err =
|
|
|
|
crypto_shash_digest(&desc.shash, key, keylen, ipad);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
keylen = ds;
|
|
|
|
} else
|
|
|
|
memcpy(ipad, key, keylen);
|
|
|
|
|
|
|
|
memset(ipad + keylen, 0, bs - keylen);
|
|
|
|
memcpy(opad, ipad, bs);
|
|
|
|
|
|
|
|
for (i = 0; i < bs; i++) {
|
|
|
|
ipad[i] ^= 0x36;
|
|
|
|
opad[i] ^= 0x5c;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = crypto_shash_init(&desc.shash) ? :
|
|
|
|
crypto_shash_update(&desc.shash, ipad, bs) ? :
|
|
|
|
crypto_shash_export(&desc.shash, ipad) ? :
|
|
|
|
crypto_shash_init(&desc.shash) ? :
|
|
|
|
crypto_shash_update(&desc.shash, opad, bs) ? :
|
|
|
|
crypto_shash_export(&desc.shash, opad);
|
|
|
|
|
|
|
|
if (rc == 0)
|
|
|
|
mv_hash_init_ivs(ctx, ipad, opad);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_cra_hash_init(struct crypto_tfm *tfm, const char *base_hash_name,
|
|
|
|
enum hash_op op, int count_add)
|
|
|
|
{
|
|
|
|
const char *fallback_driver_name = tfm->__crt_alg->cra_name;
|
|
|
|
struct mv_tfm_hash_ctx *ctx = crypto_tfm_ctx(tfm);
|
|
|
|
struct crypto_shash *fallback_tfm = NULL;
|
|
|
|
struct crypto_shash *base_hash = NULL;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
ctx->op = op;
|
|
|
|
ctx->count_add = count_add;
|
|
|
|
|
|
|
|
/* Allocate a fallback and abort if it failed. */
|
|
|
|
fallback_tfm = crypto_alloc_shash(fallback_driver_name, 0,
|
|
|
|
CRYPTO_ALG_NEED_FALLBACK);
|
|
|
|
if (IS_ERR(fallback_tfm)) {
|
|
|
|
printk(KERN_WARNING MV_CESA
|
|
|
|
"Fallback driver '%s' could not be loaded!\n",
|
|
|
|
fallback_driver_name);
|
|
|
|
err = PTR_ERR(fallback_tfm);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
ctx->fallback = fallback_tfm;
|
|
|
|
|
|
|
|
if (base_hash_name) {
|
|
|
|
/* Allocate a hash to compute the ipad/opad of hmac. */
|
|
|
|
base_hash = crypto_alloc_shash(base_hash_name, 0,
|
|
|
|
CRYPTO_ALG_NEED_FALLBACK);
|
|
|
|
if (IS_ERR(base_hash)) {
|
|
|
|
printk(KERN_WARNING MV_CESA
|
|
|
|
"Base driver '%s' could not be loaded!\n",
|
|
|
|
base_hash_name);
|
2011-01-04 11:37:16 +07:00
|
|
|
err = PTR_ERR(base_hash);
|
2010-04-08 23:34:55 +07:00
|
|
|
goto err_bad_base;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ctx->base_hash = base_hash;
|
|
|
|
|
|
|
|
crypto_ahash_set_reqsize(__crypto_ahash_cast(tfm),
|
|
|
|
sizeof(struct mv_req_hash_ctx) +
|
|
|
|
crypto_shash_descsize(ctx->fallback));
|
|
|
|
return 0;
|
|
|
|
err_bad_base:
|
|
|
|
crypto_free_shash(fallback_tfm);
|
|
|
|
out:
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mv_cra_hash_exit(struct crypto_tfm *tfm)
|
|
|
|
{
|
|
|
|
struct mv_tfm_hash_ctx *ctx = crypto_tfm_ctx(tfm);
|
|
|
|
|
|
|
|
crypto_free_shash(ctx->fallback);
|
|
|
|
if (ctx->base_hash)
|
|
|
|
crypto_free_shash(ctx->base_hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_cra_hash_sha1_init(struct crypto_tfm *tfm)
|
|
|
|
{
|
|
|
|
return mv_cra_hash_init(tfm, NULL, COP_SHA1, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_cra_hash_hmac_sha1_init(struct crypto_tfm *tfm)
|
|
|
|
{
|
|
|
|
return mv_cra_hash_init(tfm, "sha1", COP_HMAC_SHA1, SHA1_BLOCK_SIZE);
|
|
|
|
}
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
irqreturn_t crypto_int(int irq, void *priv)
|
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
val = readl(cpg->reg + SEC_ACCEL_INT_STATUS);
|
|
|
|
if (!(val & SEC_INT_ACCEL0_DONE))
|
|
|
|
return IRQ_NONE;
|
|
|
|
|
2012-05-25 20:54:46 +07:00
|
|
|
if (!del_timer(&cpg->completion_timer)) {
|
|
|
|
printk(KERN_WARNING MV_CESA
|
|
|
|
"got an interrupt but no pending timer?\n");
|
|
|
|
}
|
2009-08-10 09:50:03 +07:00
|
|
|
val &= ~SEC_INT_ACCEL0_DONE;
|
|
|
|
writel(val, cpg->reg + FPGA_INT_STATUS);
|
|
|
|
writel(val, cpg->reg + SEC_ACCEL_INT_STATUS);
|
|
|
|
BUG_ON(cpg->eng_st != ENGINE_BUSY);
|
|
|
|
cpg->eng_st = ENGINE_W_DEQUEUE;
|
|
|
|
wake_up_process(cpg->queue_th);
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct crypto_alg mv_aes_alg_ecb = {
|
|
|
|
.cra_name = "ecb(aes)",
|
|
|
|
.cra_driver_name = "mv-ecb-aes",
|
|
|
|
.cra_priority = 300,
|
2011-11-01 19:39:56 +07:00
|
|
|
.cra_flags = CRYPTO_ALG_TYPE_ABLKCIPHER |
|
|
|
|
CRYPTO_ALG_KERN_DRIVER_ONLY | CRYPTO_ALG_ASYNC,
|
2009-08-10 09:50:03 +07:00
|
|
|
.cra_blocksize = 16,
|
|
|
|
.cra_ctxsize = sizeof(struct mv_ctx),
|
|
|
|
.cra_alignmask = 0,
|
|
|
|
.cra_type = &crypto_ablkcipher_type,
|
|
|
|
.cra_module = THIS_MODULE,
|
|
|
|
.cra_init = mv_cra_init,
|
|
|
|
.cra_u = {
|
|
|
|
.ablkcipher = {
|
|
|
|
.min_keysize = AES_MIN_KEY_SIZE,
|
|
|
|
.max_keysize = AES_MAX_KEY_SIZE,
|
|
|
|
.setkey = mv_setkey_aes,
|
|
|
|
.encrypt = mv_enc_aes_ecb,
|
|
|
|
.decrypt = mv_dec_aes_ecb,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct crypto_alg mv_aes_alg_cbc = {
|
|
|
|
.cra_name = "cbc(aes)",
|
|
|
|
.cra_driver_name = "mv-cbc-aes",
|
|
|
|
.cra_priority = 300,
|
2011-11-01 19:39:56 +07:00
|
|
|
.cra_flags = CRYPTO_ALG_TYPE_ABLKCIPHER |
|
|
|
|
CRYPTO_ALG_KERN_DRIVER_ONLY | CRYPTO_ALG_ASYNC,
|
2009-08-10 09:50:03 +07:00
|
|
|
.cra_blocksize = AES_BLOCK_SIZE,
|
|
|
|
.cra_ctxsize = sizeof(struct mv_ctx),
|
|
|
|
.cra_alignmask = 0,
|
|
|
|
.cra_type = &crypto_ablkcipher_type,
|
|
|
|
.cra_module = THIS_MODULE,
|
|
|
|
.cra_init = mv_cra_init,
|
|
|
|
.cra_u = {
|
|
|
|
.ablkcipher = {
|
|
|
|
.ivsize = AES_BLOCK_SIZE,
|
|
|
|
.min_keysize = AES_MIN_KEY_SIZE,
|
|
|
|
.max_keysize = AES_MAX_KEY_SIZE,
|
|
|
|
.setkey = mv_setkey_aes,
|
|
|
|
.encrypt = mv_enc_aes_cbc,
|
|
|
|
.decrypt = mv_dec_aes_cbc,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2010-04-08 23:34:55 +07:00
|
|
|
struct ahash_alg mv_sha1_alg = {
|
|
|
|
.init = mv_hash_init,
|
|
|
|
.update = mv_hash_update,
|
|
|
|
.final = mv_hash_final,
|
|
|
|
.finup = mv_hash_finup,
|
|
|
|
.digest = mv_hash_digest,
|
|
|
|
.halg = {
|
|
|
|
.digestsize = SHA1_DIGEST_SIZE,
|
|
|
|
.base = {
|
|
|
|
.cra_name = "sha1",
|
|
|
|
.cra_driver_name = "mv-sha1",
|
|
|
|
.cra_priority = 300,
|
|
|
|
.cra_flags =
|
2011-11-01 19:39:56 +07:00
|
|
|
CRYPTO_ALG_ASYNC | CRYPTO_ALG_KERN_DRIVER_ONLY |
|
|
|
|
CRYPTO_ALG_NEED_FALLBACK,
|
2010-04-08 23:34:55 +07:00
|
|
|
.cra_blocksize = SHA1_BLOCK_SIZE,
|
|
|
|
.cra_ctxsize = sizeof(struct mv_tfm_hash_ctx),
|
|
|
|
.cra_init = mv_cra_hash_sha1_init,
|
|
|
|
.cra_exit = mv_cra_hash_exit,
|
|
|
|
.cra_module = THIS_MODULE,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ahash_alg mv_hmac_sha1_alg = {
|
|
|
|
.init = mv_hash_init,
|
|
|
|
.update = mv_hash_update,
|
|
|
|
.final = mv_hash_final,
|
|
|
|
.finup = mv_hash_finup,
|
|
|
|
.digest = mv_hash_digest,
|
|
|
|
.setkey = mv_hash_setkey,
|
|
|
|
.halg = {
|
|
|
|
.digestsize = SHA1_DIGEST_SIZE,
|
|
|
|
.base = {
|
|
|
|
.cra_name = "hmac(sha1)",
|
|
|
|
.cra_driver_name = "mv-hmac-sha1",
|
|
|
|
.cra_priority = 300,
|
|
|
|
.cra_flags =
|
2011-11-01 19:39:56 +07:00
|
|
|
CRYPTO_ALG_ASYNC | CRYPTO_ALG_KERN_DRIVER_ONLY |
|
|
|
|
CRYPTO_ALG_NEED_FALLBACK,
|
2010-04-08 23:34:55 +07:00
|
|
|
.cra_blocksize = SHA1_BLOCK_SIZE,
|
|
|
|
.cra_ctxsize = sizeof(struct mv_tfm_hash_ctx),
|
|
|
|
.cra_init = mv_cra_hash_hmac_sha1_init,
|
|
|
|
.cra_exit = mv_cra_hash_exit,
|
|
|
|
.cra_module = THIS_MODULE,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
static int mv_probe(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct crypto_priv *cp;
|
|
|
|
struct resource *res;
|
|
|
|
int irq;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (cpg) {
|
2010-04-08 23:34:55 +07:00
|
|
|
printk(KERN_ERR MV_CESA "Second crypto dev?\n");
|
2009-08-10 09:50:03 +07:00
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "regs");
|
|
|
|
if (!res)
|
|
|
|
return -ENXIO;
|
|
|
|
|
|
|
|
cp = kzalloc(sizeof(*cp), GFP_KERNEL);
|
|
|
|
if (!cp)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
spin_lock_init(&cp->lock);
|
|
|
|
crypto_init_queue(&cp->queue, 50);
|
2010-05-14 11:58:05 +07:00
|
|
|
cp->reg = ioremap(res->start, resource_size(res));
|
2009-08-10 09:50:03 +07:00
|
|
|
if (!cp->reg) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "sram");
|
|
|
|
if (!res) {
|
|
|
|
ret = -ENXIO;
|
|
|
|
goto err_unmap_reg;
|
|
|
|
}
|
2010-05-14 11:58:05 +07:00
|
|
|
cp->sram_size = resource_size(res);
|
2009-08-10 09:50:03 +07:00
|
|
|
cp->max_req_size = cp->sram_size - SRAM_CFG_SPACE;
|
|
|
|
cp->sram = ioremap(res->start, cp->sram_size);
|
|
|
|
if (!cp->sram) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto err_unmap_reg;
|
|
|
|
}
|
|
|
|
|
2012-09-04 01:29:34 +07:00
|
|
|
if (pdev->dev.of_node)
|
|
|
|
irq = irq_of_parse_and_map(pdev->dev.of_node, 0);
|
|
|
|
else
|
|
|
|
irq = platform_get_irq(pdev, 0);
|
2009-08-10 09:50:03 +07:00
|
|
|
if (irq < 0 || irq == NO_IRQ) {
|
|
|
|
ret = irq;
|
|
|
|
goto err_unmap_sram;
|
|
|
|
}
|
|
|
|
cp->irq = irq;
|
|
|
|
|
|
|
|
platform_set_drvdata(pdev, cp);
|
|
|
|
cpg = cp;
|
|
|
|
|
|
|
|
cp->queue_th = kthread_run(queue_manag, cp, "mv_crypto");
|
|
|
|
if (IS_ERR(cp->queue_th)) {
|
|
|
|
ret = PTR_ERR(cp->queue_th);
|
2010-05-26 07:45:22 +07:00
|
|
|
goto err_unmap_sram;
|
2009-08-10 09:50:03 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = request_irq(irq, crypto_int, IRQF_DISABLED, dev_name(&pdev->dev),
|
|
|
|
cp);
|
|
|
|
if (ret)
|
2010-05-26 07:45:22 +07:00
|
|
|
goto err_thread;
|
2009-08-10 09:50:03 +07:00
|
|
|
|
2012-02-19 17:56:19 +07:00
|
|
|
/* Not all platforms can gate the clock, so it is not
|
|
|
|
an error if the clock does not exists. */
|
|
|
|
cp->clk = clk_get(&pdev->dev, NULL);
|
|
|
|
if (!IS_ERR(cp->clk))
|
|
|
|
clk_prepare_enable(cp->clk);
|
|
|
|
|
2012-06-12 15:41:21 +07:00
|
|
|
writel(0, cpg->reg + SEC_ACCEL_INT_STATUS);
|
2009-08-10 09:50:03 +07:00
|
|
|
writel(SEC_INT_ACCEL0_DONE, cpg->reg + SEC_ACCEL_INT_MASK);
|
|
|
|
writel(SEC_CFG_STOP_DIG_ERR, cpg->reg + SEC_ACCEL_CFG);
|
2011-05-05 20:28:58 +07:00
|
|
|
writel(SRAM_CONFIG, cpg->reg + SEC_ACCEL_DESC_P0);
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
ret = crypto_register_alg(&mv_aes_alg_ecb);
|
2011-05-05 20:29:00 +07:00
|
|
|
if (ret) {
|
|
|
|
printk(KERN_WARNING MV_CESA
|
|
|
|
"Could not register aes-ecb driver\n");
|
2010-05-26 07:45:22 +07:00
|
|
|
goto err_irq;
|
2011-05-05 20:29:00 +07:00
|
|
|
}
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
ret = crypto_register_alg(&mv_aes_alg_cbc);
|
2011-05-05 20:29:00 +07:00
|
|
|
if (ret) {
|
|
|
|
printk(KERN_WARNING MV_CESA
|
|
|
|
"Could not register aes-cbc driver\n");
|
2009-08-10 09:50:03 +07:00
|
|
|
goto err_unreg_ecb;
|
2011-05-05 20:29:00 +07:00
|
|
|
}
|
2010-04-08 23:34:55 +07:00
|
|
|
|
|
|
|
ret = crypto_register_ahash(&mv_sha1_alg);
|
|
|
|
if (ret == 0)
|
|
|
|
cpg->has_sha1 = 1;
|
|
|
|
else
|
|
|
|
printk(KERN_WARNING MV_CESA "Could not register sha1 driver\n");
|
|
|
|
|
|
|
|
ret = crypto_register_ahash(&mv_hmac_sha1_alg);
|
|
|
|
if (ret == 0) {
|
|
|
|
cpg->has_hmac_sha1 = 1;
|
|
|
|
} else {
|
|
|
|
printk(KERN_WARNING MV_CESA
|
|
|
|
"Could not register hmac-sha1 driver\n");
|
|
|
|
}
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
return 0;
|
|
|
|
err_unreg_ecb:
|
|
|
|
crypto_unregister_alg(&mv_aes_alg_ecb);
|
2010-05-26 07:45:22 +07:00
|
|
|
err_irq:
|
2009-08-10 09:50:03 +07:00
|
|
|
free_irq(irq, cp);
|
2012-07-19 05:04:09 +07:00
|
|
|
if (!IS_ERR(cp->clk)) {
|
|
|
|
clk_disable_unprepare(cp->clk);
|
|
|
|
clk_put(cp->clk);
|
|
|
|
}
|
2010-05-26 07:45:22 +07:00
|
|
|
err_thread:
|
2009-08-10 09:50:03 +07:00
|
|
|
kthread_stop(cp->queue_th);
|
|
|
|
err_unmap_sram:
|
|
|
|
iounmap(cp->sram);
|
|
|
|
err_unmap_reg:
|
|
|
|
iounmap(cp->reg);
|
|
|
|
err:
|
|
|
|
kfree(cp);
|
|
|
|
cpg = NULL;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mv_remove(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct crypto_priv *cp = platform_get_drvdata(pdev);
|
|
|
|
|
|
|
|
crypto_unregister_alg(&mv_aes_alg_ecb);
|
|
|
|
crypto_unregister_alg(&mv_aes_alg_cbc);
|
2010-04-08 23:34:55 +07:00
|
|
|
if (cp->has_sha1)
|
|
|
|
crypto_unregister_ahash(&mv_sha1_alg);
|
|
|
|
if (cp->has_hmac_sha1)
|
|
|
|
crypto_unregister_ahash(&mv_hmac_sha1_alg);
|
2009-08-10 09:50:03 +07:00
|
|
|
kthread_stop(cp->queue_th);
|
|
|
|
free_irq(cp->irq, cp);
|
|
|
|
memset(cp->sram, 0, cp->sram_size);
|
|
|
|
iounmap(cp->sram);
|
|
|
|
iounmap(cp->reg);
|
2012-02-19 17:56:19 +07:00
|
|
|
|
|
|
|
if (!IS_ERR(cp->clk)) {
|
|
|
|
clk_disable_unprepare(cp->clk);
|
|
|
|
clk_put(cp->clk);
|
|
|
|
}
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
kfree(cp);
|
|
|
|
cpg = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-09-04 01:29:34 +07:00
|
|
|
static const struct of_device_id mv_cesa_of_match_table[] = {
|
|
|
|
{ .compatible = "marvell,orion-crypto", },
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(of, mv_cesa_of_match_table);
|
|
|
|
|
2009-08-10 09:50:03 +07:00
|
|
|
static struct platform_driver marvell_crypto = {
|
|
|
|
.probe = mv_probe,
|
2012-12-22 04:14:09 +07:00
|
|
|
.remove = mv_remove,
|
2009-08-10 09:50:03 +07:00
|
|
|
.driver = {
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.name = "mv_crypto",
|
2012-09-04 01:29:34 +07:00
|
|
|
.of_match_table = of_match_ptr(mv_cesa_of_match_table),
|
2009-08-10 09:50:03 +07:00
|
|
|
},
|
|
|
|
};
|
|
|
|
MODULE_ALIAS("platform:mv_crypto");
|
|
|
|
|
2011-11-26 20:26:19 +07:00
|
|
|
module_platform_driver(marvell_crypto);
|
2009-08-10 09:50:03 +07:00
|
|
|
|
|
|
|
MODULE_AUTHOR("Sebastian Andrzej Siewior <sebastian@breakpoint.cc>");
|
|
|
|
MODULE_DESCRIPTION("Support for Marvell's cryptographic engine");
|
|
|
|
MODULE_LICENSE("GPL");
|