提交 641eacd1 编写于 作者: Z Zain Wang 提交者: Herbert Xu

crypto: rockchip - move the crypto completion from interrupt context

It's illegal to call the completion function from hardirq context,
it will cause runtime tests to fail. Let's build a new task (done_task)
for moving update operation from hardirq context.
Signed-off-by: Nzain wang <wzz@rock-chips.com>
Signed-off-by: NHerbert Xu <herbert@gondor.apana.org.au>
上级 1d544944
...@@ -169,24 +169,22 @@ static irqreturn_t rk_crypto_irq_handle(int irq, void *dev_id) ...@@ -169,24 +169,22 @@ static irqreturn_t rk_crypto_irq_handle(int irq, void *dev_id)
{ {
struct rk_crypto_info *dev = platform_get_drvdata(dev_id); struct rk_crypto_info *dev = platform_get_drvdata(dev_id);
u32 interrupt_status; u32 interrupt_status;
int err = 0;
spin_lock(&dev->lock); spin_lock(&dev->lock);
interrupt_status = CRYPTO_READ(dev, RK_CRYPTO_INTSTS); interrupt_status = CRYPTO_READ(dev, RK_CRYPTO_INTSTS);
CRYPTO_WRITE(dev, RK_CRYPTO_INTSTS, interrupt_status); CRYPTO_WRITE(dev, RK_CRYPTO_INTSTS, interrupt_status);
if (interrupt_status & 0x0a) { if (interrupt_status & 0x0a) {
dev_warn(dev->dev, "DMA Error\n"); dev_warn(dev->dev, "DMA Error\n");
err = -EFAULT; dev->err = -EFAULT;
} else if (interrupt_status & 0x05) {
err = dev->update(dev);
} }
if (err) tasklet_schedule(&dev->done_task);
dev->complete(dev, err);
spin_unlock(&dev->lock); spin_unlock(&dev->lock);
return IRQ_HANDLED; return IRQ_HANDLED;
} }
static void rk_crypto_tasklet_cb(unsigned long data) static void rk_crypto_queue_task_cb(unsigned long data)
{ {
struct rk_crypto_info *dev = (struct rk_crypto_info *)data; struct rk_crypto_info *dev = (struct rk_crypto_info *)data;
struct crypto_async_request *async_req, *backlog; struct crypto_async_request *async_req, *backlog;
...@@ -210,11 +208,26 @@ static void rk_crypto_tasklet_cb(unsigned long data) ...@@ -210,11 +208,26 @@ static void rk_crypto_tasklet_cb(unsigned long data)
dev->ablk_req = ablkcipher_request_cast(async_req); dev->ablk_req = ablkcipher_request_cast(async_req);
else else
dev->ahash_req = ahash_request_cast(async_req); dev->ahash_req = ahash_request_cast(async_req);
dev->err = 0;
err = dev->start(dev); err = dev->start(dev);
if (err) if (err)
dev->complete(dev, err); dev->complete(dev, err);
} }
static void rk_crypto_done_task_cb(unsigned long data)
{
struct rk_crypto_info *dev = (struct rk_crypto_info *)data;
if (dev->err) {
dev->complete(dev, dev->err);
return;
}
dev->err = dev->update(dev);
if (dev->err)
dev->complete(dev, dev->err);
}
static struct rk_crypto_tmp *rk_cipher_algs[] = { static struct rk_crypto_tmp *rk_cipher_algs[] = {
&rk_ecb_aes_alg, &rk_ecb_aes_alg,
&rk_cbc_aes_alg, &rk_cbc_aes_alg,
...@@ -361,8 +374,10 @@ static int rk_crypto_probe(struct platform_device *pdev) ...@@ -361,8 +374,10 @@ static int rk_crypto_probe(struct platform_device *pdev)
crypto_info->dev = &pdev->dev; crypto_info->dev = &pdev->dev;
platform_set_drvdata(pdev, crypto_info); platform_set_drvdata(pdev, crypto_info);
tasklet_init(&crypto_info->crypto_tasklet, tasklet_init(&crypto_info->queue_task,
rk_crypto_tasklet_cb, (unsigned long)crypto_info); rk_crypto_queue_task_cb, (unsigned long)crypto_info);
tasklet_init(&crypto_info->done_task,
rk_crypto_done_task_cb, (unsigned long)crypto_info);
crypto_init_queue(&crypto_info->queue, 50); crypto_init_queue(&crypto_info->queue, 50);
crypto_info->enable_clk = rk_crypto_enable_clk; crypto_info->enable_clk = rk_crypto_enable_clk;
...@@ -380,7 +395,8 @@ static int rk_crypto_probe(struct platform_device *pdev) ...@@ -380,7 +395,8 @@ static int rk_crypto_probe(struct platform_device *pdev)
return 0; return 0;
err_register_alg: err_register_alg:
tasklet_kill(&crypto_info->crypto_tasklet); tasklet_kill(&crypto_info->queue_task);
tasklet_kill(&crypto_info->done_task);
err_crypto: err_crypto:
return err; return err;
} }
...@@ -390,7 +406,8 @@ static int rk_crypto_remove(struct platform_device *pdev) ...@@ -390,7 +406,8 @@ static int rk_crypto_remove(struct platform_device *pdev)
struct rk_crypto_info *crypto_tmp = platform_get_drvdata(pdev); struct rk_crypto_info *crypto_tmp = platform_get_drvdata(pdev);
rk_crypto_unregister(); rk_crypto_unregister();
tasklet_kill(&crypto_tmp->crypto_tasklet); tasklet_kill(&crypto_tmp->done_task);
tasklet_kill(&crypto_tmp->queue_task);
return 0; return 0;
} }
......
...@@ -190,9 +190,11 @@ struct rk_crypto_info { ...@@ -190,9 +190,11 @@ struct rk_crypto_info {
void __iomem *reg; void __iomem *reg;
int irq; int irq;
struct crypto_queue queue; struct crypto_queue queue;
struct tasklet_struct crypto_tasklet; struct tasklet_struct queue_task;
struct tasklet_struct done_task;
struct ablkcipher_request *ablk_req; struct ablkcipher_request *ablk_req;
struct ahash_request *ahash_req; struct ahash_request *ahash_req;
int err;
/* device lock */ /* device lock */
spinlock_t lock; spinlock_t lock;
......
...@@ -42,7 +42,7 @@ static int rk_handle_req(struct rk_crypto_info *dev, ...@@ -42,7 +42,7 @@ static int rk_handle_req(struct rk_crypto_info *dev,
spin_lock_irqsave(&dev->lock, flags); spin_lock_irqsave(&dev->lock, flags);
err = ablkcipher_enqueue_request(&dev->queue, req); err = ablkcipher_enqueue_request(&dev->queue, req);
spin_unlock_irqrestore(&dev->lock, flags); spin_unlock_irqrestore(&dev->lock, flags);
tasklet_schedule(&dev->crypto_tasklet); tasklet_schedule(&dev->queue_task);
return err; return err;
} }
......
...@@ -204,7 +204,7 @@ static int rk_ahash_digest(struct ahash_request *req) ...@@ -204,7 +204,7 @@ static int rk_ahash_digest(struct ahash_request *req)
ret = crypto_enqueue_request(&dev->queue, &req->base); ret = crypto_enqueue_request(&dev->queue, &req->base);
spin_unlock_irqrestore(&dev->lock, flags); spin_unlock_irqrestore(&dev->lock, flags);
tasklet_schedule(&dev->crypto_tasklet); tasklet_schedule(&dev->queue_task);
/* /*
* it will take some time to process date after last dma transmission. * it will take some time to process date after last dma transmission.
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册