Release 4.12 drivers/crypto/mv_cesa.c
  
  
  
/*
 * 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/genalloc.h>
#include <linux/interrupt.h>
#include <linux/io.h>
#include <linux/kthread.h>
#include <linux/platform_device.h>
#include <linux/scatterlist.h>
#include <linux/slab.h>
#include <linux/module.h>
#include <linux/clk.h>
#include <crypto/internal/hash.h>
#include <crypto/sha.h>
#include <linux/of.h>
#include <linux/of_platform.h>
#include <linux/of_irq.h>
#include "mv_cesa.h"
#define MV_CESA	"MV-CESA:"
#define MAX_HW_HASH_SIZE	0xFFFF
#define MV_CESA_EXPIRE		500 
/* msec */
#define MV_CESA_DEFAULT_SRAM_SIZE	2048
/*
 * 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)
 * @crypt_len:          length of current hw crypt/hash process
 * @hw_nbytes:          total bytes to process in hw for this request
 * @copy_back:          whether to copy data back (crypt) or not (hash)
 * @sg_dst_left:        bytes left dst to process in this scatter list
 * @dst_start:          offset to add to dst start position (scatter list)
 * @hw_processed_bytes: number of bytes processed by hw (request).
 *
 * 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;
	
void (*complete) (void);
	
void (*process) (int is_first);
	/* src mostly */
	
int sg_src_left;
	
int src_start;
	
int crypt_len;
	
int hw_nbytes;
	/* dst mostly */
	
int copy_back;
	
int sg_dst_left;
	
int dst_start;
	
int hw_processed_bytes;
};
struct crypto_priv {
	
void __iomem *reg;
	
void __iomem *sram;
	
struct gen_pool *sram_pool;
	
dma_addr_t sram_dma;
	
int irq;
	
struct clk *clk;
	
struct task_struct *queue_th;
	/* the lock protects queue and eng_st */
	
spinlock_t lock;
	
struct crypto_queue queue;
	
enum engine_status eng_st;
	
struct timer_list completion_timer;
	
struct crypto_async_request *cur_req;
	
struct req_progress p;
	
int max_req_size;
	
int sram_size;
	
int has_sha1;
	
int has_hmac_sha1;
};
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;
};
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;
};
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Phil Sutter | 88 | 100.00% | 1 | 100.00% | 
| Total | 88 | 100.00% | 1 | 100.00% | 
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));
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Phil Sutter | 36 | 100.00% | 1 | 100.00% | 
| Total | 36 | 100.00% | 1 | 100.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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 120 | 100.00% | 1 | 100.00% | 
| Total | 120 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 94 | 100.00% | 1 | 100.00% | 
| Total | 94 | 100.00% | 1 | 100.00% | 
static void copy_src_to_buf(struct req_progress *p, char *dbuf, int len)
{
	int ret;
	void *sbuf;
	int copy_len;
	while (len) {
		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;
		}
		sbuf = p->src_sg_it.addr + p->src_start;
		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;
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 65 | 51.18% | 1 | 33.33% | 
| Uri Simchoni | 47 | 37.01% | 1 | 33.33% | 
| Phil Sutter | 15 | 11.81% | 1 | 33.33% | 
| Total | 127 | 100.00% | 3 | 100.00% | 
static void setup_data_in(void)
{
	struct req_progress *p = &cpg->p;
	int data_in_sram =
	    min(p->hw_nbytes - p->hw_processed_bytes, cpg->max_req_size);
	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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 51 | 80.95% | 4 | 80.00% | 
| Sebastian Andrzej Siewior | 12 | 19.05% | 1 | 20.00% | 
| Total | 63 | 100.00% | 5 | 100.00% | 
static void mv_process_current_q(int first_block)
{
	struct ablkcipher_request *req = ablkcipher_request_cast(cpg->cur_req);
	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:
	default:
		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;
	setup_data_in();
	op.enc_len = cpg->p.crypt_len;
	memcpy(cpg->sram + SRAM_CONFIG, &op,
			sizeof(struct sec_accel_config));
	/* GO */
	mv_setup_timer();
	writel(SEC_CMD_EN_SEC_ACCL0, cpg->reg + SEC_ACCEL_CMD);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 268 | 97.10% | 1 | 25.00% | 
| Uri Simchoni | 5 | 1.81% | 2 | 50.00% | 
| Phil Sutter | 3 | 1.09% | 1 | 25.00% | 
| Total | 276 | 100.00% | 4 | 100.00% | 
static void mv_crypto_algo_completion(void)
{
	struct ablkcipher_request *req = ablkcipher_request_cast(cpg->cur_req);
	struct mv_req_ctx *req_ctx = ablkcipher_request_ctx(req);
	sg_miter_stop(&cpg->p.src_sg_it);
	sg_miter_stop(&cpg->p.dst_sg_it);
	if (req_ctx->op != COP_AES_CBC)
		return ;
	memcpy(req->info, cpg->sram + SRAM_DATA_IV_BUF, 16);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 51 | 68.92% | 1 | 33.33% | 
| Uri Simchoni | 23 | 31.08% | 2 | 66.67% | 
| Total | 74 | 100.00% | 3 | 100.00% | 
static void mv_process_hash_current(int first_block)
{
	struct ahash_request *req = ahash_request_cast(cpg->cur_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 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;
		memcpy(cpg->sram + SRAM_HMAC_IV_IN,
				tfm_ctx->ivs, sizeof(tfm_ctx->ivs));
		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;
		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);
		}
	}
	memcpy(cpg->sram + SRAM_CONFIG, &op, sizeof(struct sec_accel_config));
	/* GO */
	mv_setup_timer();
	writel(SEC_CMD_EN_SEC_ACCL0, cpg->reg + SEC_ACCEL_CMD);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 183 | 49.46% | 2 | 28.57% | 
| Phil Sutter | 124 | 33.51% | 4 | 57.14% | 
| Sebastian Andrzej Siewior | 63 | 17.03% | 1 | 14.29% | 
| Total | 370 | 100.00% | 7 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 80 | 87.91% | 5 | 83.33% | 
| Sebastian Andrzej Siewior | 11 | 12.09% | 1 | 16.67% | 
| Total | 91 | 100.00% | 6 | 100.00% | 
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);
	SHASH_DESC_ON_STACK(shash, tfm_ctx->fallback);
	int rc;
	shash->tfm = tfm_ctx->fallback;
	shash->flags = CRYPTO_TFM_REQ_MAY_SLEEP;
	if (unlikely(req_ctx->first_hash)) {
		crypto_shash_init(shash);
		crypto_shash_update(shash, req_ctx->buffer,
				    req_ctx->extra_bytes);
	} else {
		/* only SHA1 for now....
                 */
		rc = mv_hash_import_sha1_ctx(req_ctx, shash);
		if (rc)
			goto out;
	}
	rc = crypto_shash_final(shash, req->result);
out:
	return rc;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 109 | 85.83% | 2 | 50.00% | 
| Sebastian Andrzej Siewior | 12 | 9.45% | 1 | 25.00% | 
| Behan Webster | 6 | 4.72% | 1 | 25.00% | 
| Total | 127 | 100.00% | 4 | 100.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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Phil Sutter | 91 | 100.00% | 1 | 100.00% | 
| Total | 91 | 100.00% | 1 | 100.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)));
		} else {
			mv_save_digest_state(ctx);
			mv_hash_final_fallback(req);
		}
	} else {
		mv_save_digest_state(ctx);
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 74 | 57.81% | 1 | 33.33% | 
| Phil Sutter | 54 | 42.19% | 2 | 66.67% | 
| Total | 128 | 100.00% | 3 | 100.00% | 
static void dequeue_complete_req(void)
{
	struct crypto_async_request *req = cpg->cur_req;
	void *buf;
	int ret;
	cpg->p.hw_processed_bytes += cpg->p.crypt_len;
	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);
	}
	cpg->p.crypt_len = 0;
	BUG_ON(cpg->eng_st != ENGINE_W_DEQUEUE);
	if (cpg->p.hw_processed_bytes < cpg->p.hw_nbytes) {
		/* process next scatter list entry */
		cpg->eng_st = ENGINE_BUSY;
		cpg->p.process(0);
	} else {
		cpg->p.complete();
		cpg->eng_st = ENGINE_IDLE;
		local_bh_disable();
		req->complete(req, 0);
		local_bh_enable();
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 274 | 100.00% | 1 | 100.00% | 
| Total | 274 | 100.00% | 1 | 100.00% | 
static int count_sgs(struct scatterlist *sl, unsigned int total_bytes)
{
	int i = 0;
	size_t cur_len;
	while (sl) {
		cur_len = sl[i].length;
		++i;
		if (total_bytes > cur_len)
			total_bytes -= cur_len;
		else
			break;
	}
	return i;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 55 | 98.21% | 1 | 50.00% | 
| Phil Sutter | 1 | 1.79% | 1 | 50.00% | 
| Total | 56 | 100.00% | 2 | 100.00% | 
static void mv_start_new_crypt_req(struct ablkcipher_request *req)
{
	struct req_progress *p = &cpg->p;
	int num_sgs;
	cpg->cur_req = &req->base;
	memset(p, 0, sizeof(struct req_progress));
	p->hw_nbytes = req->nbytes;
	p->complete = mv_crypto_algo_completion;
	p->process = mv_process_current_q;
	p->copy_back = 1;
	num_sgs = count_sgs(req->src, req->nbytes);
	sg_miter_start(&p->src_sg_it, req->src, num_sgs, SG_MITER_FROM_SG);
	num_sgs = count_sgs(req->dst, req->nbytes);
	sg_miter_start(&p->dst_sg_it, req->dst, num_sgs, SG_MITER_TO_SG);
	mv_process_current_q(1);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 135 | 100.00% | 1 | 100.00% | 
| Total | 135 | 100.00% | 1 | 100.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;
		if (unlikely(old_extra_bytes)) {
			memcpy(cpg->sram + SRAM_DATA_IN_START, ctx->buffer,
			       old_extra_bytes);
			p->crypt_len = old_extra_bytes;
		}
		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();
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 250 | 89.29% | 1 | 50.00% | 
| Phil Sutter | 30 | 10.71% | 1 | 50.00% | 
| Total | 280 | 100.00% | 2 | 100.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 = NULL;
		__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) {
			if (crypto_tfm_alg_type(async_req->tfm) !=
			    CRYPTO_ALG_TYPE_AHASH) {
				struct ablkcipher_request *req =
				    ablkcipher_request_cast(async_req);
				mv_start_new_crypt_req(req);
			} else {
				struct ahash_request *req =
				    ahash_request_cast(async_req);
				mv_start_new_hash_req(req);
			}
			async_req = NULL;
		}
		schedule();
	} while (!kthread_should_stop());
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 198 | 96.59% | 1 | 25.00% | 
| Marek Vašut | 4 | 1.95% | 1 | 25.00% | 
| Colin Ian King | 2 | 0.98% | 1 | 25.00% | 
| Phil Sutter | 1 | 0.49% | 1 | 25.00% | 
| Total | 205 | 100.00% | 4 | 100.00% | 
static int mv_handle_req(struct crypto_async_request *req)
{
	unsigned long flags;
	int ret;
	spin_lock_irqsave(&cpg->lock, flags);
	ret = crypto_enqueue_request(&cpg->queue, req);
	spin_unlock_irqrestore(&cpg->lock, flags);
	wake_up_process(cpg->queue_th);
	return ret;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 60 | 100.00% | 1 | 100.00% | 
| Total | 60 | 100.00% | 1 | 100.00% | 
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;
	return mv_handle_req(&req->base);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 42 | 100.00% | 1 | 100.00% | 
| Total | 42 | 100.00% | 1 | 100.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);
	return mv_handle_req(&req->base);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 61 | 100.00% | 1 | 100.00% | 
| Total | 61 | 100.00% | 1 | 100.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;
	return mv_handle_req(&req->base);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 42 | 100.00% | 1 | 100.00% | 
| Total | 42 | 100.00% | 1 | 100.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);
	return mv_handle_req(&req->base);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 61 | 100.00% | 1 | 100.00% | 
| Total | 61 | 100.00% | 1 | 100.00% | 
static int mv_cra_init(struct crypto_tfm *tfm)
{
	tfm->crt_ablkcipher.reqsize = sizeof(struct mv_req_ctx);
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 26 | 100.00% | 1 | 100.00% | 
| Total | 26 | 100.00% | 1 | 100.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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 67 | 100.00% | 1 | 100.00% | 
| Total | 67 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 29 | 100.00% | 1 | 100.00% | 
| Total | 29 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 49 | 100.00% | 1 | 100.00% | 
| Total | 49 | 100.00% | 1 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 44 | 100.00% | 1 | 100.00% | 
| Total | 44 | 100.00% | 1 | 100.00% | 
static int mv_hash_final(struct ahash_request *req)
{
	struct mv_req_hash_ctx *ctx = ahash_request_ctx(req);
	ahash_request_set_crypt(req, NULL, req->result, 0);
	mv_update_hash_req_ctx(ctx, 1, 0);
	return mv_handle_req(&req->base);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 39 | 75.00% | 1 | 50.00% | 
| Phil Sutter | 13 | 25.00% | 1 | 50.00% | 
| Total | 52 | 100.00% | 2 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 31 | 91.18% | 2 | 66.67% | 
| Sebastian Andrzej Siewior | 3 | 8.82% | 1 | 33.33% | 
| Total | 34 | 100.00% | 3 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 36 | 63.16% | 3 | 75.00% | 
| Sebastian Andrzej Siewior | 21 | 36.84% | 1 | 25.00% | 
| Total | 57 | 100.00% | 4 | 100.00% | 
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]);
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 78 | 88.64% | 1 | 50.00% | 
| Sebastian Andrzej Siewior | 10 | 11.36% | 1 | 50.00% | 
| Total | 88 | 100.00% | 2 | 100.00% | 
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);
	{
		SHASH_DESC_ON_STACK(shash, ctx->base_hash);
		unsigned int i;
		char ipad[ss];
		char opad[ss];
		shash->tfm = ctx->base_hash;
		shash->flags = crypto_shash_get_flags(ctx->base_hash) &
		    CRYPTO_TFM_REQ_MAY_SLEEP;
		if (keylen > bs) {
			int err;
			err =
			    crypto_shash_digest(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(shash) ? :
		    crypto_shash_update(shash, ipad, bs) ? :
		    crypto_shash_export(shash, ipad) ? :
		    crypto_shash_init(shash) ? :
		    crypto_shash_update(shash, opad, bs) ? :
		    crypto_shash_export(shash, opad);
		if (rc == 0)
			mv_hash_init_ivs(ctx, ipad, opad);
		return rc;
	}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 221 | 70.83% | 1 | 33.33% | 
| Sebastian Andrzej Siewior | 85 | 27.24% | 1 | 33.33% | 
| Behan Webster | 6 | 1.92% | 1 | 33.33% | 
| Total | 312 | 100.00% | 3 | 100.00% | 
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 = crypto_tfm_alg_name(tfm);
	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);
			err = PTR_ERR(base_hash);
			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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 158 | 76.33% | 2 | 40.00% | 
| Sebastian Andrzej Siewior | 45 | 21.74% | 1 | 20.00% | 
| Marek Vašut | 3 | 1.45% | 1 | 20.00% | 
| Roel Kluin | 1 | 0.48% | 1 | 20.00% | 
| Total | 207 | 100.00% | 5 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 23 | 56.10% | 1 | 50.00% | 
| Uri Simchoni | 18 | 43.90% | 1 | 50.00% | 
| Total | 41 | 100.00% | 2 | 100.00% | 
static int mv_cra_hash_sha1_init(struct crypto_tfm *tfm)
{
	return mv_cra_hash_init(tfm, NULL, COP_SHA1, 0);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 18 | 78.26% | 1 | 50.00% | 
| Sebastian Andrzej Siewior | 5 | 21.74% | 1 | 50.00% | 
| Total | 23 | 100.00% | 2 | 100.00% | 
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);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 13 | 56.52% | 1 | 50.00% | 
| Uri Simchoni | 10 | 43.48% | 1 | 50.00% | 
| Total | 23 | 100.00% | 2 | 100.00% | 
static 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;
	if (!del_timer(&cpg->completion_timer)) {
		printk(KERN_WARNING MV_CESA
		       "got an interrupt but no pending timer?\n");
	}
	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;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 90 | 81.82% | 1 | 33.33% | 
| Phil Sutter | 19 | 17.27% | 1 | 33.33% | 
| Sachin Kamat | 1 | 0.91% | 1 | 33.33% | 
| Total | 110 | 100.00% | 3 | 100.00% | 
static struct crypto_alg mv_aes_alg_ecb = {
	.cra_name		= "ecb(aes)",
	.cra_driver_name	= "mv-ecb-aes",
	.cra_priority	= 300,
	.cra_flags	= CRYPTO_ALG_TYPE_ABLKCIPHER |
			  CRYPTO_ALG_KERN_DRIVER_ONLY | CRYPTO_ALG_ASYNC,
	.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,
                },
        },
};
static struct crypto_alg mv_aes_alg_cbc = {
	.cra_name		= "cbc(aes)",
	.cra_driver_name	= "mv-cbc-aes",
	.cra_priority	= 300,
	.cra_flags	= CRYPTO_ALG_TYPE_ABLKCIPHER |
			  CRYPTO_ALG_KERN_DRIVER_ONLY | CRYPTO_ALG_ASYNC,
	.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,
                },
        },
};
static 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 =
			  CRYPTO_ALG_ASYNC | CRYPTO_ALG_KERN_DRIVER_ONLY |
			  CRYPTO_ALG_NEED_FALLBACK,
			  .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,
                          }
		 }
};
static 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 =
			  CRYPTO_ALG_ASYNC | CRYPTO_ALG_KERN_DRIVER_ONLY |
			  CRYPTO_ALG_NEED_FALLBACK,
			  .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,
                          }
		 }
};
static int mv_cesa_get_sram(struct platform_device *pdev,
			    struct crypto_priv *cp)
{
	struct resource *res;
	u32 sram_size = MV_CESA_DEFAULT_SRAM_SIZE;
	of_property_read_u32(pdev->dev.of_node, "marvell,crypto-sram-size",
			     &sram_size);
	cp->sram_size = sram_size;
	cp->sram_pool = of_gen_pool_get(pdev->dev.of_node,
					"marvell,crypto-srams", 0);
	if (cp->sram_pool) {
		cp->sram = gen_pool_dma_alloc(cp->sram_pool, sram_size,
					      &cp->sram_dma);
		if (cp->sram)
			return 0;
		return -ENOMEM;
	}
	res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
					   "sram");
	if (!res || resource_size(res) < cp->sram_size)
		return -EINVAL;
	cp->sram = devm_ioremap_resource(&pdev->dev, res);
	if (IS_ERR(cp->sram))
		return PTR_ERR(cp->sram);
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Boris Brezillon | 164 | 99.39% | 1 | 50.00% | 
| Stephen Rothwell | 1 | 0.61% | 1 | 50.00% | 
| Total | 165 | 100.00% | 2 | 100.00% | 
static int mv_probe(struct platform_device *pdev)
{
	struct crypto_priv *cp;
	struct resource *res;
	int irq;
	int ret;
	if (cpg) {
		printk(KERN_ERR MV_CESA "Second crypto dev?\n");
		return -EEXIST;
	}
	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "regs");
	if (!res)
		return -ENXIO;
	cp = devm_kzalloc(&pdev->dev, sizeof(*cp), GFP_KERNEL);
	if (!cp)
		return -ENOMEM;
	spin_lock_init(&cp->lock);
	crypto_init_queue(&cp->queue, 50);
	cp->reg = devm_ioremap_resource(&pdev->dev, res);
	if (IS_ERR(cp->reg)) {
		ret = PTR_ERR(cp->reg);
		goto err;
	}
	ret = mv_cesa_get_sram(pdev, cp);
	if (ret)
		goto err;
	cp->max_req_size = cp->sram_size - SRAM_CFG_SPACE;
	irq = platform_get_irq(pdev, 0);
	if (irq < 0) {
		ret = irq;
		goto err;
	}
	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);
		goto err;
	}
	ret = request_irq(irq, crypto_int, 0, dev_name(&pdev->dev),
			cp);
	if (ret)
		goto err_thread;
	/* 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);
	writel(0, cpg->reg + SEC_ACCEL_INT_STATUS);
	writel(SEC_INT_ACCEL0_DONE, cpg->reg + SEC_ACCEL_INT_MASK);
	writel(SEC_CFG_STOP_DIG_ERR, cpg->reg + SEC_ACCEL_CFG);
	writel(SRAM_CONFIG, cpg->reg + SEC_ACCEL_DESC_P0);
	ret = crypto_register_alg(&mv_aes_alg_ecb);
	if (ret) {
		printk(KERN_WARNING MV_CESA
		       "Could not register aes-ecb driver\n");
		goto err_irq;
	}
	ret = crypto_register_alg(&mv_aes_alg_cbc);
	if (ret) {
		printk(KERN_WARNING MV_CESA
		       "Could not register aes-cbc driver\n");
		goto err_unreg_ecb;
	}
	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");
	}
	return 0;
err_unreg_ecb:
	crypto_unregister_alg(&mv_aes_alg_ecb);
err_irq:
	free_irq(irq, cp);
	if (!IS_ERR(cp->clk)) {
		clk_disable_unprepare(cp->clk);
		clk_put(cp->clk);
	}
err_thread:
	kthread_stop(cp->queue_th);
err:
	cpg = NULL;
	return ret;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 332 | 63.85% | 1 | 7.69% | 
| Uri Simchoni | 60 | 11.54% | 1 | 7.69% | 
| Phil Sutter | 38 | 7.31% | 3 | 23.08% | 
| Andrew Lunn | 32 | 6.15% | 1 | 7.69% | 
| Simon Baatz | 26 | 5.00% | 1 | 7.69% | 
| Boris Brezillon | 20 | 3.85% | 2 | 15.38% | 
| Nadim Almas | 6 | 1.15% | 1 | 7.69% | 
| Dan Carpenter | 4 | 0.77% | 1 | 7.69% | 
| Michael Opdenacker | 1 | 0.19% | 1 | 7.69% | 
| Tobias Klauser | 1 | 0.19% | 1 | 7.69% | 
| Total | 520 | 100.00% | 13 | 100.00% | 
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);
	if (cp->has_sha1)
		crypto_unregister_ahash(&mv_sha1_alg);
	if (cp->has_hmac_sha1)
		crypto_unregister_ahash(&mv_hmac_sha1_alg);
	kthread_stop(cp->queue_th);
	free_irq(cp->irq, cp);
	memset(cp->sram, 0, cp->sram_size);
	if (!IS_ERR(cp->clk)) {
		clk_disable_unprepare(cp->clk);
		clk_put(cp->clk);
	}
	cpg = NULL;
	return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Sebastian Andrzej Siewior | 69 | 57.98% | 1 | 33.33% | 
| Andrew Lunn | 26 | 21.85% | 1 | 33.33% | 
| Uri Simchoni | 24 | 20.17% | 1 | 33.33% | 
| Total | 119 | 100.00% | 3 | 100.00% | 
static const struct of_device_id mv_cesa_of_match_table[] = {
	{ .compatible = "marvell,orion-crypto", },
	{ .compatible = "marvell,kirkwood-crypto", },
	{ .compatible = "marvell,dove-crypto", },
	{}
};
MODULE_DEVICE_TABLE(of, mv_cesa_of_match_table);
static struct platform_driver marvell_crypto = {
	.probe		= mv_probe,
	.remove		= mv_remove,
	.driver		= {
		.name	= "mv_crypto",
		.of_match_table = mv_cesa_of_match_table,
        },
};
MODULE_ALIAS("platform:mv_crypto");
module_platform_driver(marvell_crypto);
MODULE_AUTHOR("Sebastian Andrzej Siewior <sebastian@breakpoint.cc>");
MODULE_DESCRIPTION("Support for Marvell's cryptographic engine");
MODULE_LICENSE("GPL");
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp | 
| Uri Simchoni | 2986 | 52.20% | 9 | 20.00% | 
| Sebastian Andrzej Siewior | 1813 | 31.70% | 1 | 2.22% | 
| Phil Sutter | 522 | 9.13% | 14 | 31.11% | 
| Boris Brezillon | 213 | 3.72% | 3 | 6.67% | 
| Andrew Lunn | 105 | 1.84% | 2 | 4.44% | 
| Simon Baatz | 26 | 0.45% | 1 | 2.22% | 
| Behan Webster | 12 | 0.21% | 1 | 2.22% | 
| Nikos Mavrogiannopoulos | 8 | 0.14% | 1 | 2.22% | 
| Marek Vašut | 7 | 0.12% | 2 | 4.44% | 
| Nadim Almas | 6 | 0.10% | 1 | 2.22% | 
| Sachin Kamat | 5 | 0.09% | 1 | 2.22% | 
| Dan Carpenter | 4 | 0.07% | 1 | 2.22% | 
| Paul Gortmaker | 3 | 0.05% | 1 | 2.22% | 
| Tejun Heo | 3 | 0.05% | 1 | 2.22% | 
| Colin Ian King | 2 | 0.03% | 1 | 2.22% | 
| Stephen Rothwell | 1 | 0.02% | 1 | 2.22% | 
| Roel Kluin | 1 | 0.02% | 1 | 2.22% | 
| Tobias Klauser | 1 | 0.02% | 1 | 2.22% | 
| Axel Lin | 1 | 0.02% | 1 | 2.22% | 
| Michael Opdenacker | 1 | 0.02% | 1 | 2.22% | 
| Total | 5720 | 100.00% | 45 | 100.00% | 
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.