Commit b95cb394 authored by Boris Brezillon's avatar Boris Brezillon Committed by Mark Brown

spi: ti-qspi: Implement the spi_mem interface

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: default avatarBoris Brezillon <boris.brezillon@bootlin.com>
Reviewed-by: default avatarVignesh R <vigneshr@ti.com>
Tested-by: default avatarVignesh R <vigneshr@ti.com>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent 5f195ee7
...@@ -36,6 +36,7 @@ ...@@ -36,6 +36,7 @@
#include <linux/sizes.h> #include <linux/sizes.h>
#include <linux/spi/spi.h> #include <linux/spi/spi.h>
#include <linux/spi/spi-mem.h>
struct ti_qspi_regs { struct ti_qspi_regs {
u32 clkctrl; u32 clkctrl;
...@@ -50,6 +51,7 @@ struct ti_qspi { ...@@ -50,6 +51,7 @@ struct ti_qspi {
struct spi_master *master; struct spi_master *master;
void __iomem *base; void __iomem *base;
void __iomem *mmap_base; void __iomem *mmap_base;
size_t mmap_size;
struct regmap *ctrl_base; struct regmap *ctrl_base;
unsigned int ctrl_reg; unsigned int ctrl_reg;
struct clk *fclk; struct clk *fclk;
...@@ -434,12 +436,10 @@ static int ti_qspi_dma_xfer(struct ti_qspi *qspi, dma_addr_t dma_dst, ...@@ -434,12 +436,10 @@ static int ti_qspi_dma_xfer(struct ti_qspi *qspi, dma_addr_t dma_dst,
return 0; return 0;
} }
static int ti_qspi_dma_bounce_buffer(struct ti_qspi *qspi, static int ti_qspi_dma_bounce_buffer(struct ti_qspi *qspi, loff_t offs,
struct spi_flash_read_message *msg) void *to, size_t readsize)
{ {
size_t readsize = msg->len; dma_addr_t dma_src = qspi->mmap_phys_base + offs;
void *to = msg->buf;
dma_addr_t dma_src = qspi->mmap_phys_base + msg->from;
int ret = 0; int ret = 0;
/* /*
...@@ -507,13 +507,14 @@ static void ti_qspi_disable_memory_map(struct spi_device *spi) ...@@ -507,13 +507,14 @@ static void ti_qspi_disable_memory_map(struct spi_device *spi)
qspi->mmap_enabled = false; qspi->mmap_enabled = false;
} }
static void ti_qspi_setup_mmap_read(struct spi_device *spi, static void ti_qspi_setup_mmap_read(struct spi_device *spi, u8 opcode,
struct spi_flash_read_message *msg) u8 data_nbits, u8 addr_width,
u8 dummy_bytes)
{ {
struct ti_qspi *qspi = spi_master_get_devdata(spi->master); 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: case SPI_NBITS_QUAD:
memval |= QSPI_SETUP_RD_QUAD; memval |= QSPI_SETUP_RD_QUAD;
break; break;
...@@ -524,8 +525,8 @@ static void ti_qspi_setup_mmap_read(struct spi_device *spi, ...@@ -524,8 +525,8 @@ static void ti_qspi_setup_mmap_read(struct spi_device *spi,
memval |= QSPI_SETUP_RD_NORMAL; memval |= QSPI_SETUP_RD_NORMAL;
break; break;
} }
memval |= ((msg->addr_width - 1) << QSPI_SETUP_ADDR_SHIFT | memval |= ((addr_width - 1) << QSPI_SETUP_ADDR_SHIFT |
msg->dummy_bytes << QSPI_SETUP_DUMMY_SHIFT); dummy_bytes << QSPI_SETUP_DUMMY_SHIFT);
ti_qspi_write(qspi, memval, ti_qspi_write(qspi, memval,
QSPI_SPI_SETUP_REG(spi->chip_select)); QSPI_SPI_SETUP_REG(spi->chip_select));
} }
...@@ -546,13 +547,15 @@ static int ti_qspi_spi_flash_read(struct spi_device *spi, ...@@ -546,13 +547,15 @@ static int ti_qspi_spi_flash_read(struct spi_device *spi,
if (!qspi->mmap_enabled) if (!qspi->mmap_enabled)
ti_qspi_enable_memory_map(spi); 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 (qspi->rx_chan) {
if (msg->cur_msg_mapped) if (msg->cur_msg_mapped)
ret = ti_qspi_dma_xfer_sg(qspi, msg->rx_sg, msg->from); ret = ti_qspi_dma_xfer_sg(qspi, msg->rx_sg, msg->from);
else else
ret = ti_qspi_dma_bounce_buffer(qspi, msg); ret = ti_qspi_dma_bounce_buffer(qspi, msg->from,
msg->buf, msg->len);
if (ret) if (ret)
goto err_unlock; goto err_unlock;
} else { } else {
...@@ -566,6 +569,58 @@ static int ti_qspi_spi_flash_read(struct spi_device *spi, ...@@ -566,6 +569,58 @@ static int ti_qspi_spi_flash_read(struct spi_device *spi,
return ret; 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, static int ti_qspi_start_transfer_one(struct spi_master *master,
struct spi_message *m) struct spi_message *m)
{ {
...@@ -673,6 +728,7 @@ static int ti_qspi_probe(struct platform_device *pdev) ...@@ -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) | master->bits_per_word_mask = SPI_BPW_MASK(32) | SPI_BPW_MASK(16) |
SPI_BPW_MASK(8); SPI_BPW_MASK(8);
master->spi_flash_read = ti_qspi_spi_flash_read; 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)) if (!of_property_read_u32(np, "num-cs", &num_cs))
master->num_chipselect = num_cs; master->num_chipselect = num_cs;
...@@ -701,6 +757,7 @@ static int ti_qspi_probe(struct platform_device *pdev) ...@@ -701,6 +757,7 @@ static int ti_qspi_probe(struct platform_device *pdev)
"memory mapped resource not required\n"); "memory mapped resource not required\n");
} }
} }
qspi->mmap_size = resource_size(res_mmap);
irq = platform_get_irq(pdev, 0); irq = platform_get_irq(pdev, 0);
if (irq < 0) { if (irq < 0) {
...@@ -785,6 +842,7 @@ static int ti_qspi_probe(struct platform_device *pdev) ...@@ -785,6 +842,7 @@ static int ti_qspi_probe(struct platform_device *pdev)
PTR_ERR(qspi->mmap_base)); PTR_ERR(qspi->mmap_base));
qspi->mmap_base = NULL; qspi->mmap_base = NULL;
master->spi_flash_read = NULL; master->spi_flash_read = NULL;
master->mem_ops = NULL;
} }
} }
qspi->mmap_enabled = false; qspi->mmap_enabled = false;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment