crypto: omap-aes - Remove usage of private DMA API
[platform/adaptation/renesas_rcar/renesas_kernel.git] / drivers / crypto / omap-aes.c
index e66e8ee..faf522f 100644 (file)
 #include <linux/init.h>
 #include <linux/errno.h>
 #include <linux/kernel.h>
-#include <linux/clk.h>
 #include <linux/platform_device.h>
 #include <linux/scatterlist.h>
 #include <linux/dma-mapping.h>
+#include <linux/dmaengine.h>
+#include <linux/omap-dma.h>
+#include <linux/pm_runtime.h>
 #include <linux/io.h>
 #include <linux/crypto.h>
 #include <linux/interrupt.h>
 #include <crypto/scatterwalk.h>
 #include <crypto/aes.h>
 
-#include <linux/omap-dma.h>
+#define DST_MAXBURST                   4
+#define DMA_MIN                                (DST_MAXBURST * sizeof(u32))
 
 /* OMAP TRM gives bitfields as start:end, where start is the higher bit
    number. For example 7:0 */
@@ -96,7 +99,6 @@ struct omap_aes_dev {
        struct list_head        list;
        unsigned long           phys_base;
        void __iomem            *io_base;
-       struct clk              *iclk;
        struct omap_aes_ctx     *ctx;
        struct device           *dev;
        unsigned long           flags;
@@ -111,19 +113,21 @@ struct omap_aes_dev {
        struct ablkcipher_request       *req;
        size_t                          total;
        struct scatterlist              *in_sg;
+       struct scatterlist              in_sgl;
        size_t                          in_offset;
        struct scatterlist              *out_sg;
+       struct scatterlist              out_sgl;
        size_t                          out_offset;
 
        size_t                  buflen;
        void                    *buf_in;
        size_t                  dma_size;
        int                     dma_in;
-       int                     dma_lch_in;
+       struct dma_chan         *dma_lch_in;
        dma_addr_t              dma_addr_in;
        void                    *buf_out;
        int                     dma_out;
-       int                     dma_lch_out;
+       struct dma_chan         *dma_lch_out;
        dma_addr_t              dma_addr_out;
 };
 
@@ -160,19 +164,6 @@ static void omap_aes_write_n(struct omap_aes_dev *dd, u32 offset,
                omap_aes_write(dd, offset, *value);
 }
 
-static int omap_aes_wait(struct omap_aes_dev *dd, u32 offset, u32 bit)
-{
-       unsigned long timeout = jiffies + DEFAULT_TIMEOUT;
-
-       while (!(omap_aes_read(dd, offset) & bit)) {
-               if (time_is_before_jiffies(timeout)) {
-                       dev_err(dd->dev, "omap-aes timeout\n");
-                       return -ETIMEDOUT;
-               }
-       }
-       return 0;
-}
-
 static int omap_aes_hw_init(struct omap_aes_dev *dd)
 {
        /*
@@ -180,23 +171,9 @@ static int omap_aes_hw_init(struct omap_aes_dev *dd)
         * It may be long delays between requests.
         * Device might go to off mode to save power.
         */
-       clk_enable(dd->iclk);
+       pm_runtime_get_sync(dd->dev);
 
        if (!(dd->flags & FLAGS_INIT)) {
-               /* is it necessary to reset before every operation? */
-               omap_aes_write_mask(dd, AES_REG_MASK, AES_REG_MASK_SOFTRESET,
-                                       AES_REG_MASK_SOFTRESET);
-               /*
-                * prevent OCP bus error (SRESP) in case an access to the module
-                * is performed while the module is coming out of soft reset
-                */
-               __asm__ __volatile__("nop");
-               __asm__ __volatile__("nop");
-
-               if (omap_aes_wait(dd, AES_REG_SYSSTATUS,
-                               AES_REG_SYSSTATUS_RESETDONE))
-                       return -ETIMEDOUT;
-
                dd->flags |= FLAGS_INIT;
                dd->err = 0;
        }
@@ -215,9 +192,9 @@ static int omap_aes_write_ctrl(struct omap_aes_dev *dd)
                return err;
 
        val = 0;
-       if (dd->dma_lch_out >= 0)
+       if (dd->dma_lch_out != NULL)
                val |= AES_REG_MASK_DMA_OUT_EN;
-       if (dd->dma_lch_in >= 0)
+       if (dd->dma_lch_in != NULL)
                val |= AES_REG_MASK_DMA_IN_EN;
 
        mask = AES_REG_MASK_DMA_IN_EN | AES_REG_MASK_DMA_OUT_EN;
@@ -246,20 +223,6 @@ static int omap_aes_write_ctrl(struct omap_aes_dev *dd)
 
        omap_aes_write_mask(dd, AES_REG_CTRL, val, mask);
 
-       /* IN */
-       omap_set_dma_dest_params(dd->dma_lch_in, 0, OMAP_DMA_AMODE_CONSTANT,
-                                dd->phys_base + AES_REG_DATA, 0, 4);
-
-       omap_set_dma_dest_burst_mode(dd->dma_lch_in, OMAP_DMA_DATA_BURST_4);
-       omap_set_dma_src_burst_mode(dd->dma_lch_in, OMAP_DMA_DATA_BURST_4);
-
-       /* OUT */
-       omap_set_dma_src_params(dd->dma_lch_out, 0, OMAP_DMA_AMODE_CONSTANT,
-                               dd->phys_base + AES_REG_DATA, 0, 4);
-
-       omap_set_dma_src_burst_mode(dd->dma_lch_out, OMAP_DMA_DATA_BURST_4);
-       omap_set_dma_dest_burst_mode(dd->dma_lch_out, OMAP_DMA_DATA_BURST_4);
-
        return 0;
 }
 
@@ -284,18 +247,10 @@ static struct omap_aes_dev *omap_aes_find_dev(struct omap_aes_ctx *ctx)
        return dd;
 }
 
-static void omap_aes_dma_callback(int lch, u16 ch_status, void *data)
+static void omap_aes_dma_out_callback(void *data)
 {
        struct omap_aes_dev *dd = data;
 
-       if (ch_status != OMAP_DMA_BLOCK_IRQ) {
-               pr_err("omap-aes DMA error status: 0x%hx\n", ch_status);
-               dd->err = -EIO;
-               dd->flags &= ~FLAGS_INIT; /* request to re-initialize */
-       } else if (lch == dd->dma_lch_in) {
-               return;
-       }
-
        /* dma_lch_out - completed */
        tasklet_schedule(&dd->done_task);
 }
@@ -303,9 +258,10 @@ static void omap_aes_dma_callback(int lch, u16 ch_status, void *data)
 static int omap_aes_dma_init(struct omap_aes_dev *dd)
 {
        int err = -ENOMEM;
+       dma_cap_mask_t mask;
 
-       dd->dma_lch_out = -1;
-       dd->dma_lch_in = -1;
+       dd->dma_lch_out = NULL;
+       dd->dma_lch_in = NULL;
 
        dd->buf_in = (void *)__get_free_pages(GFP_KERNEL, OMAP_AES_CACHE_SIZE);
        dd->buf_out = (void *)__get_free_pages(GFP_KERNEL, OMAP_AES_CACHE_SIZE);
@@ -334,23 +290,27 @@ static int omap_aes_dma_init(struct omap_aes_dev *dd)
                goto err_map_out;
        }
 
-       err = omap_request_dma(dd->dma_in, "omap-aes-rx",
-                              omap_aes_dma_callback, dd, &dd->dma_lch_in);
-       if (err) {
-               dev_err(dd->dev, "Unable to request DMA channel\n");
+       dma_cap_zero(mask);
+       dma_cap_set(DMA_SLAVE, mask);
+
+       dd->dma_lch_in = dma_request_channel(mask, omap_dma_filter_fn,
+                                            &dd->dma_in);
+       if (!dd->dma_lch_in) {
+               dev_err(dd->dev, "Unable to request in DMA channel\n");
                goto err_dma_in;
        }
-       err = omap_request_dma(dd->dma_out, "omap-aes-tx",
-                              omap_aes_dma_callback, dd, &dd->dma_lch_out);
-       if (err) {
-               dev_err(dd->dev, "Unable to request DMA channel\n");
+
+       dd->dma_lch_out = dma_request_channel(mask, omap_dma_filter_fn,
+                                            &dd->dma_out);
+       if (!dd->dma_lch_out) {
+               dev_err(dd->dev, "Unable to request out DMA channel\n");
                goto err_dma_out;
        }
 
        return 0;
 
 err_dma_out:
-       omap_free_dma(dd->dma_lch_in);
+       dma_release_channel(dd->dma_lch_in);
 err_dma_in:
        dma_unmap_single(dd->dev, dd->dma_addr_out, dd->buflen,
                         DMA_FROM_DEVICE);
@@ -367,8 +327,8 @@ err_alloc:
 
 static void omap_aes_dma_cleanup(struct omap_aes_dev *dd)
 {
-       omap_free_dma(dd->dma_lch_out);
-       omap_free_dma(dd->dma_lch_in);
+       dma_release_channel(dd->dma_lch_out);
+       dma_release_channel(dd->dma_lch_in);
        dma_unmap_single(dd->dev, dd->dma_addr_out, dd->buflen,
                         DMA_FROM_DEVICE);
        dma_unmap_single(dd->dev, dd->dma_addr_in, dd->buflen, DMA_TO_DEVICE);
@@ -426,12 +386,15 @@ static int sg_copy(struct scatterlist **sg, size_t *offset, void *buf,
        return off;
 }
 
-static int omap_aes_crypt_dma(struct crypto_tfm *tfm, dma_addr_t dma_addr_in,
-                              dma_addr_t dma_addr_out, int length)
+static int omap_aes_crypt_dma(struct crypto_tfm *tfm,
+               struct scatterlist *in_sg, struct scatterlist *out_sg)
 {
        struct omap_aes_ctx *ctx = crypto_tfm_ctx(tfm);
        struct omap_aes_dev *dd = ctx->dd;
-       int len32;
+       struct dma_async_tx_descriptor *tx_in, *tx_out;
+       struct dma_slave_config cfg;
+       dma_addr_t dma_addr_in = sg_dma_address(in_sg);
+       int ret, length = sg_dma_len(in_sg);
 
        pr_debug("len: %d\n", length);
 
@@ -441,26 +404,58 @@ static int omap_aes_crypt_dma(struct crypto_tfm *tfm, dma_addr_t dma_addr_in,
                dma_sync_single_for_device(dd->dev, dma_addr_in, length,
                                           DMA_TO_DEVICE);
 
-       len32 = DIV_ROUND_UP(length, sizeof(u32));
+       memset(&cfg, 0, sizeof(cfg));
+
+       cfg.src_addr = dd->phys_base + AES_REG_DATA;
+       cfg.dst_addr = dd->phys_base + AES_REG_DATA;
+       cfg.src_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
+       cfg.dst_addr_width = DMA_SLAVE_BUSWIDTH_4_BYTES;
+       cfg.src_maxburst = DST_MAXBURST;
+       cfg.dst_maxburst = DST_MAXBURST;
 
        /* IN */
-       omap_set_dma_transfer_params(dd->dma_lch_in, OMAP_DMA_DATA_TYPE_S32,
-                                    len32, 1, OMAP_DMA_SYNC_PACKET, dd->dma_in,
-                                       OMAP_DMA_DST_SYNC);
+       ret = dmaengine_slave_config(dd->dma_lch_in, &cfg);
+       if (ret) {
+               dev_err(dd->dev, "can't configure IN dmaengine slave: %d\n",
+                       ret);
+               return ret;
+       }
+
+       tx_in = dmaengine_prep_slave_sg(dd->dma_lch_in, in_sg, 1,
+                                       DMA_MEM_TO_DEV,
+                                       DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
+       if (!tx_in) {
+               dev_err(dd->dev, "IN prep_slave_sg() failed\n");
+               return -EINVAL;
+       }
 
-       omap_set_dma_src_params(dd->dma_lch_in, 0, OMAP_DMA_AMODE_POST_INC,
-                               dma_addr_in, 0, 0);
+       /* No callback necessary */
+       tx_in->callback_param = dd;
 
        /* OUT */
-       omap_set_dma_transfer_params(dd->dma_lch_out, OMAP_DMA_DATA_TYPE_S32,
-                                    len32, 1, OMAP_DMA_SYNC_PACKET,
-                                       dd->dma_out, OMAP_DMA_SRC_SYNC);
+       ret = dmaengine_slave_config(dd->dma_lch_out, &cfg);
+       if (ret) {
+               dev_err(dd->dev, "can't configure OUT dmaengine slave: %d\n",
+                       ret);
+               return ret;
+       }
+
+       tx_out = dmaengine_prep_slave_sg(dd->dma_lch_out, out_sg, 1,
+                                       DMA_DEV_TO_MEM,
+                                       DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
+       if (!tx_out) {
+               dev_err(dd->dev, "OUT prep_slave_sg() failed\n");
+               return -EINVAL;
+       }
+
+       tx_out->callback = omap_aes_dma_out_callback;
+       tx_out->callback_param = dd;
 
-       omap_set_dma_dest_params(dd->dma_lch_out, 0, OMAP_DMA_AMODE_POST_INC,
-                                dma_addr_out, 0, 0);
+       dmaengine_submit(tx_in);
+       dmaengine_submit(tx_out);
 
-       omap_start_dma(dd->dma_lch_in);
-       omap_start_dma(dd->dma_lch_out);
+       dma_async_issue_pending(dd->dma_lch_in);
+       dma_async_issue_pending(dd->dma_lch_out);
 
        /* start DMA or disable idle mode */
        omap_aes_write_mask(dd, AES_REG_MASK, AES_REG_MASK_START,
@@ -476,6 +471,8 @@ static int omap_aes_crypt_dma_start(struct omap_aes_dev *dd)
        int err, fast = 0, in, out;
        size_t count;
        dma_addr_t addr_in, addr_out;
+       struct scatterlist *in_sg, *out_sg;
+       int len32;
 
        pr_debug("total: %d\n", dd->total);
 
@@ -514,6 +511,9 @@ static int omap_aes_crypt_dma_start(struct omap_aes_dev *dd)
                addr_in = sg_dma_address(dd->in_sg);
                addr_out = sg_dma_address(dd->out_sg);
 
+               in_sg = dd->in_sg;
+               out_sg = dd->out_sg;
+
                dd->flags |= FLAGS_FAST;
 
        } else {
@@ -521,6 +521,27 @@ static int omap_aes_crypt_dma_start(struct omap_aes_dev *dd)
                count = sg_copy(&dd->in_sg, &dd->in_offset, dd->buf_in,
                                 dd->buflen, dd->total, 0);
 
+               len32 = DIV_ROUND_UP(count, DMA_MIN) * DMA_MIN;
+
+               /*
+                * The data going into the AES module has been copied
+                * to a local buffer and the data coming out will go
+                * into a local buffer so set up local SG entries for
+                * both.
+                */
+               sg_init_table(&dd->in_sgl, 1);
+               dd->in_sgl.offset = dd->in_offset;
+               sg_dma_len(&dd->in_sgl) = len32;
+               sg_dma_address(&dd->in_sgl) = dd->dma_addr_in;
+
+               sg_init_table(&dd->out_sgl, 1);
+               dd->out_sgl.offset = dd->out_offset;
+               sg_dma_len(&dd->out_sgl) = len32;
+               sg_dma_address(&dd->out_sgl) = dd->dma_addr_out;
+
+               in_sg = &dd->in_sgl;
+               out_sg = &dd->out_sgl;
+
                addr_in = dd->dma_addr_in;
                addr_out = dd->dma_addr_out;
 
@@ -530,7 +551,7 @@ static int omap_aes_crypt_dma_start(struct omap_aes_dev *dd)
 
        dd->total -= count;
 
-       err = omap_aes_crypt_dma(tfm, addr_in, addr_out, count);
+       err = omap_aes_crypt_dma(tfm, in_sg, out_sg);
        if (err) {
                dma_unmap_sg(dd->dev, dd->in_sg, 1, DMA_TO_DEVICE);
                dma_unmap_sg(dd->dev, dd->out_sg, 1, DMA_TO_DEVICE);
@@ -545,7 +566,7 @@ static void omap_aes_finish_req(struct omap_aes_dev *dd, int err)
 
        pr_debug("err: %d\n", err);
 
-       clk_disable(dd->iclk);
+       pm_runtime_put_sync(dd->dev);
        dd->flags &= ~FLAGS_BUSY;
 
        req->base.complete(&req->base, err);
@@ -560,8 +581,8 @@ static int omap_aes_crypt_dma_stop(struct omap_aes_dev *dd)
 
        omap_aes_write_mask(dd, AES_REG_MASK, 0, AES_REG_MASK_START);
 
-       omap_stop_dma(dd->dma_lch_in);
-       omap_stop_dma(dd->dma_lch_out);
+       dmaengine_terminate_all(dd->dma_lch_in);
+       dmaengine_terminate_all(dd->dma_lch_out);
 
        if (dd->flags & FLAGS_FAST) {
                dma_unmap_sg(dd->dev, dd->out_sg, 1, DMA_FROM_DEVICE);
@@ -840,26 +861,21 @@ static int omap_aes_probe(struct platform_device *pdev)
        else
                dd->dma_in = res->start;
 
-       /* Initializing the clock */
-       dd->iclk = clk_get(dev, "ick");
-       if (IS_ERR(dd->iclk)) {
-               dev_err(dev, "clock intialization failed.\n");
-               err = PTR_ERR(dd->iclk);
-               goto err_res;
-       }
-
        dd->io_base = ioremap(dd->phys_base, SZ_4K);
        if (!dd->io_base) {
                dev_err(dev, "can't ioremap\n");
                err = -ENOMEM;
-               goto err_io;
+               goto err_res;
        }
 
-       clk_enable(dd->iclk);
+       pm_runtime_enable(dev);
+       pm_runtime_get_sync(dev);
+
        reg = omap_aes_read(dd, AES_REG_REV);
        dev_info(dev, "OMAP AES hw accel rev: %u.%u\n",
                 (reg & AES_REG_REV_MAJOR) >> 4, reg & AES_REG_REV_MINOR);
-       clk_disable(dd->iclk);
+
+       pm_runtime_put_sync(dev);
 
        tasklet_init(&dd->done_task, omap_aes_done_task, (unsigned long)dd);
        tasklet_init(&dd->queue_task, omap_aes_queue_task, (unsigned long)dd);
@@ -880,8 +896,6 @@ static int omap_aes_probe(struct platform_device *pdev)
                        goto err_algs;
        }
 
-       pr_info("probe() done\n");
-
        return 0;
 err_algs:
        for (j = 0; j < i; j++)
@@ -891,8 +905,7 @@ err_dma:
        tasklet_kill(&dd->done_task);
        tasklet_kill(&dd->queue_task);
        iounmap(dd->io_base);
-err_io:
-       clk_put(dd->iclk);
+       pm_runtime_disable(dev);
 err_res:
        kfree(dd);
        dd = NULL;
@@ -920,26 +933,43 @@ static int omap_aes_remove(struct platform_device *pdev)
        tasklet_kill(&dd->queue_task);
        omap_aes_dma_cleanup(dd);
        iounmap(dd->io_base);
-       clk_put(dd->iclk);
+       pm_runtime_disable(dd->dev);
        kfree(dd);
        dd = NULL;
 
        return 0;
 }
 
+#ifdef CONFIG_PM_SLEEP
+static int omap_aes_suspend(struct device *dev)
+{
+       pm_runtime_put_sync(dev);
+       return 0;
+}
+
+static int omap_aes_resume(struct device *dev)
+{
+       pm_runtime_get_sync(dev);
+       return 0;
+}
+#endif
+
+static const struct dev_pm_ops omap_aes_pm_ops = {
+       SET_SYSTEM_SLEEP_PM_OPS(omap_aes_suspend, omap_aes_resume)
+};
+
 static struct platform_driver omap_aes_driver = {
        .probe  = omap_aes_probe,
        .remove = omap_aes_remove,
        .driver = {
                .name   = "omap-aes",
                .owner  = THIS_MODULE,
+               .pm     = &omap_aes_pm_ops,
        },
 };
 
 static int __init omap_aes_mod_init(void)
 {
-       pr_info("loading %s driver\n", "omap-aes");
-
        return  platform_driver_register(&omap_aes_driver);
 }