spi: ti-qspi: Implement the spi_mem interface
authorBoris Brezillon <boris.brezillon@bootlin.com>
Thu, 26 Apr 2018 16:18:18 +0000 (18:18 +0200)
committerMark Brown <broonie@kernel.org>
Fri, 11 May 2018 02:33:42 +0000 (11:33 +0900)
The spi_mem interface is meant to replace the spi_flash_read() one.
Implement the ->exec_op() method so that we can smoothly get rid of the
old interface.

Signed-off-by: Boris Brezillon <boris.brezillon@bootlin.com>
Reviewed-by: Vignesh R <vigneshr@ti.com>
Tested-by: Vignesh R <vigneshr@ti.com>
Signed-off-by: Mark Brown <broonie@kernel.org>
drivers/spi/spi-ti-qspi.c

index c24d9b4..1b12c2f 100644 (file)
@@ -36,6 +36,7 @@
 #include <linux/sizes.h>
 
 #include <linux/spi/spi.h>
+#include <linux/spi/spi-mem.h>
 
 struct ti_qspi_regs {
        u32 clkctrl;
@@ -50,6 +51,7 @@ struct ti_qspi {
        struct spi_master       *master;
        void __iomem            *base;
        void __iomem            *mmap_base;
+       size_t                  mmap_size;
        struct regmap           *ctrl_base;
        unsigned int            ctrl_reg;
        struct clk              *fclk;
@@ -434,12 +436,10 @@ static int ti_qspi_dma_xfer(struct ti_qspi *qspi, dma_addr_t dma_dst,
        return 0;
 }
 
-static int ti_qspi_dma_bounce_buffer(struct ti_qspi *qspi,
-                                    struct spi_flash_read_message *msg)
+static int ti_qspi_dma_bounce_buffer(struct ti_qspi *qspi, loff_t offs,
+                                    void *to, size_t readsize)
 {
-       size_t readsize = msg->len;
-       void *to = msg->buf;
-       dma_addr_t dma_src = qspi->mmap_phys_base + msg->from;
+       dma_addr_t dma_src = qspi->mmap_phys_base + offs;
        int ret = 0;
 
        /*
@@ -507,13 +507,14 @@ static void ti_qspi_disable_memory_map(struct spi_device *spi)
        qspi->mmap_enabled = false;
 }
 
-static void ti_qspi_setup_mmap_read(struct spi_device *spi,
-                                   struct spi_flash_read_message *msg)
+static void ti_qspi_setup_mmap_read(struct spi_device *spi, u8 opcode,
+                                   u8 data_nbits, u8 addr_width,
+                                   u8 dummy_bytes)
 {
        struct ti_qspi  *qspi = spi_master_get_devdata(spi->master);
-       u32 memval = msg->read_opcode;
+       u32 memval = opcode;
 
-       switch (msg->data_nbits) {
+       switch (data_nbits) {
        case SPI_NBITS_QUAD:
                memval |= QSPI_SETUP_RD_QUAD;
                break;
@@ -524,8 +525,8 @@ static void ti_qspi_setup_mmap_read(struct spi_device *spi,
                memval |= QSPI_SETUP_RD_NORMAL;
                break;
        }
-       memval |= ((msg->addr_width - 1) << QSPI_SETUP_ADDR_SHIFT |
-                  msg->dummy_bytes << QSPI_SETUP_DUMMY_SHIFT);
+       memval |= ((addr_width - 1) << QSPI_SETUP_ADDR_SHIFT |
+                  dummy_bytes << QSPI_SETUP_DUMMY_SHIFT);
        ti_qspi_write(qspi, memval,
                      QSPI_SPI_SETUP_REG(spi->chip_select));
 }
@@ -546,13 +547,15 @@ static int ti_qspi_spi_flash_read(struct spi_device *spi,
 
        if (!qspi->mmap_enabled)
                ti_qspi_enable_memory_map(spi);
-       ti_qspi_setup_mmap_read(spi, msg);
+       ti_qspi_setup_mmap_read(spi, msg->read_opcode, msg->data_nbits,
+                               msg->addr_width, msg->dummy_bytes);
 
        if (qspi->rx_chan) {
                if (msg->cur_msg_mapped)
                        ret = ti_qspi_dma_xfer_sg(qspi, msg->rx_sg, msg->from);
                else
-                       ret = ti_qspi_dma_bounce_buffer(qspi, msg);
+                       ret = ti_qspi_dma_bounce_buffer(qspi, msg->from,
+                                                       msg->buf, msg->len);
                if (ret)
                        goto err_unlock;
        } else {
@@ -566,6 +569,58 @@ err_unlock:
        return ret;
 }
 
+static int ti_qspi_exec_mem_op(struct spi_mem *mem,
+                              const struct spi_mem_op *op)
+{
+       struct ti_qspi *qspi = spi_master_get_devdata(mem->spi->master);
+       u32 from = 0;
+       int ret = 0;
+
+       /* Only optimize read path. */
+       if (!op->data.nbytes || op->data.dir != SPI_MEM_DATA_IN ||
+           !op->addr.nbytes || op->addr.nbytes > 4)
+               return -ENOTSUPP;
+
+       /* Address exceeds MMIO window size, fall back to regular mode. */
+       from = op->addr.val;
+       if (from + op->data.nbytes > qspi->mmap_size)
+               return -ENOTSUPP;
+
+       mutex_lock(&qspi->list_lock);
+
+       if (!qspi->mmap_enabled)
+               ti_qspi_enable_memory_map(mem->spi);
+       ti_qspi_setup_mmap_read(mem->spi, op->cmd.opcode, op->data.buswidth,
+                               op->addr.nbytes, op->dummy.nbytes);
+
+       if (qspi->rx_chan) {
+               struct sg_table sgt;
+
+               if (virt_addr_valid(op->data.buf.in) &&
+                   !spi_controller_dma_map_mem_op_data(mem->spi->master, op,
+                                                       &sgt)) {
+                       ret = ti_qspi_dma_xfer_sg(qspi, sgt, from);
+                       spi_controller_dma_unmap_mem_op_data(mem->spi->master,
+                                                            op, &sgt);
+               } else {
+                       ret = ti_qspi_dma_bounce_buffer(qspi, from,
+                                                       op->data.buf.in,
+                                                       op->data.nbytes);
+               }
+       } else {
+               memcpy_fromio(op->data.buf.in, qspi->mmap_base + from,
+                             op->data.nbytes);
+       }
+
+       mutex_unlock(&qspi->list_lock);
+
+       return ret;
+}
+
+static const struct spi_controller_mem_ops ti_qspi_mem_ops = {
+       .exec_op = ti_qspi_exec_mem_op,
+};
+
 static int ti_qspi_start_transfer_one(struct spi_master *master,
                struct spi_message *m)
 {
@@ -673,6 +728,7 @@ static int ti_qspi_probe(struct platform_device *pdev)
        master->bits_per_word_mask = SPI_BPW_MASK(32) | SPI_BPW_MASK(16) |
                                     SPI_BPW_MASK(8);
        master->spi_flash_read = ti_qspi_spi_flash_read;
+       master->mem_ops = &ti_qspi_mem_ops;
 
        if (!of_property_read_u32(np, "num-cs", &num_cs))
                master->num_chipselect = num_cs;
@@ -701,6 +757,7 @@ static int ti_qspi_probe(struct platform_device *pdev)
                                "memory mapped resource not required\n");
                }
        }
+       qspi->mmap_size = resource_size(res_mmap);
 
        irq = platform_get_irq(pdev, 0);
        if (irq < 0) {
@@ -785,6 +842,7 @@ no_dma:
                                 PTR_ERR(qspi->mmap_base));
                        qspi->mmap_base = NULL;
                        master->spi_flash_read = NULL;
+                       master->mem_ops = NULL;
                }
        }
        qspi->mmap_enabled = false;