Commit caf72df4 authored by Pratyush Yadav's avatar Pratyush Yadav Committed by Mark Brown

spi: spi-mem: allow specifying a command's extension

In xSPI mode, flashes expect 2-byte opcodes. The second byte is called
the "command extension". There can be 3 types of extensions in xSPI:
repeat, invert, and hex. When the extension type is "repeat", the same
opcode is sent twice. When it is "invert", the second byte is the
inverse of the opcode. When it is "hex" an additional opcode byte based
is sent with the command whose value can be anything.

So, make opcode a 16-bit value and add a 'nbytes', similar to how
multiple address widths are handled.

Some places use sizeof(op->cmd.opcode). Replace them with op->cmd.nbytes

The spi-mxic and spi-zynq-qspi drivers directly use op->cmd.opcode as a
buffer. Now that opcode is a 2-byte field, this can result in different
behaviour depending on if the machine is little endian or big endian.
Extract the opcode in a local 1-byte variable and use that as the buffer
instead. Both these drivers would reject multi-byte opcodes in their
supports_op() hook anyway, so we only need to worry about single-byte
opcodes for now.

The above two changes are put in this commit to keep the series
bisectable.
Signed-off-by: default avatarPratyush Yadav <p.yadav@ti.com>
Reviewed-by: default avatarTudor Ambarus <tudor.ambarus@microchip.com>
Link: https://lore.kernel.org/r/20200623183030.26591-3-p.yadav@ti.comSigned-off-by: default avatarMark Brown <broonie@kernel.org>
parent 4c5e2bba
...@@ -159,6 +159,9 @@ bool spi_mem_default_supports_op(struct spi_mem *mem, ...@@ -159,6 +159,9 @@ bool spi_mem_default_supports_op(struct spi_mem *mem,
if (op->cmd.dtr || op->addr.dtr || op->dummy.dtr || op->data.dtr) if (op->cmd.dtr || op->addr.dtr || op->dummy.dtr || op->data.dtr)
return false; return false;
if (op->cmd.nbytes != 1)
return false;
return true; return true;
} }
EXPORT_SYMBOL_GPL(spi_mem_default_supports_op); EXPORT_SYMBOL_GPL(spi_mem_default_supports_op);
...@@ -173,7 +176,7 @@ static bool spi_mem_buswidth_is_valid(u8 buswidth) ...@@ -173,7 +176,7 @@ static bool spi_mem_buswidth_is_valid(u8 buswidth)
static int spi_mem_check_op(const struct spi_mem_op *op) static int spi_mem_check_op(const struct spi_mem_op *op)
{ {
if (!op->cmd.buswidth) if (!op->cmd.buswidth || !op->cmd.nbytes)
return -EINVAL; return -EINVAL;
if ((op->addr.nbytes && !op->addr.buswidth) || if ((op->addr.nbytes && !op->addr.buswidth) ||
...@@ -309,8 +312,7 @@ int spi_mem_exec_op(struct spi_mem *mem, const struct spi_mem_op *op) ...@@ -309,8 +312,7 @@ int spi_mem_exec_op(struct spi_mem *mem, const struct spi_mem_op *op)
return ret; return ret;
} }
tmpbufsize = sizeof(op->cmd.opcode) + op->addr.nbytes + tmpbufsize = op->cmd.nbytes + op->addr.nbytes + op->dummy.nbytes;
op->dummy.nbytes;
/* /*
* Allocate a buffer to transmit the CMD, ADDR cycles with kmalloc() so * Allocate a buffer to transmit the CMD, ADDR cycles with kmalloc() so
...@@ -325,7 +327,7 @@ int spi_mem_exec_op(struct spi_mem *mem, const struct spi_mem_op *op) ...@@ -325,7 +327,7 @@ int spi_mem_exec_op(struct spi_mem *mem, const struct spi_mem_op *op)
tmpbuf[0] = op->cmd.opcode; tmpbuf[0] = op->cmd.opcode;
xfers[xferpos].tx_buf = tmpbuf; xfers[xferpos].tx_buf = tmpbuf;
xfers[xferpos].len = sizeof(op->cmd.opcode); xfers[xferpos].len = op->cmd.nbytes;
xfers[xferpos].tx_nbits = op->cmd.buswidth; xfers[xferpos].tx_nbits = op->cmd.buswidth;
spi_message_add_tail(&xfers[xferpos], &msg); spi_message_add_tail(&xfers[xferpos], &msg);
xferpos++; xferpos++;
...@@ -427,8 +429,7 @@ int spi_mem_adjust_op_size(struct spi_mem *mem, struct spi_mem_op *op) ...@@ -427,8 +429,7 @@ int spi_mem_adjust_op_size(struct spi_mem *mem, struct spi_mem_op *op)
return ctlr->mem_ops->adjust_op_size(mem, op); return ctlr->mem_ops->adjust_op_size(mem, op);
if (!ctlr->mem_ops || !ctlr->mem_ops->exec_op) { if (!ctlr->mem_ops || !ctlr->mem_ops->exec_op) {
len = sizeof(op->cmd.opcode) + op->addr.nbytes + len = op->cmd.nbytes + op->addr.nbytes + op->dummy.nbytes;
op->dummy.nbytes;
if (len > spi_max_transfer_size(mem->spi)) if (len > spi_max_transfer_size(mem->spi))
return -EINVAL; return -EINVAL;
......
...@@ -195,7 +195,7 @@ static int mtk_nor_adjust_op_size(struct spi_mem *mem, struct spi_mem_op *op) ...@@ -195,7 +195,7 @@ static int mtk_nor_adjust_op_size(struct spi_mem *mem, struct spi_mem_op *op)
} }
} }
len = MTK_NOR_PRG_MAX_SIZE - sizeof(op->cmd.opcode) - op->addr.nbytes - len = MTK_NOR_PRG_MAX_SIZE - op->cmd.nbytes - op->addr.nbytes -
op->dummy.nbytes; op->dummy.nbytes;
if (op->data.nbytes > len) if (op->data.nbytes > len)
op->data.nbytes = len; op->data.nbytes = len;
...@@ -219,7 +219,7 @@ static bool mtk_nor_supports_op(struct spi_mem *mem, ...@@ -219,7 +219,7 @@ static bool mtk_nor_supports_op(struct spi_mem *mem,
(op->dummy.buswidth == 0) && (op->dummy.buswidth == 0) &&
(op->data.buswidth == 1); (op->data.buswidth == 1);
} }
len = sizeof(op->cmd.opcode) + op->addr.nbytes + op->dummy.nbytes; len = op->cmd.nbytes + op->addr.nbytes + op->dummy.nbytes;
if ((len > MTK_NOR_PRG_MAX_SIZE) || if ((len > MTK_NOR_PRG_MAX_SIZE) ||
((op->data.nbytes) && (len == MTK_NOR_PRG_MAX_SIZE))) ((op->data.nbytes) && (len == MTK_NOR_PRG_MAX_SIZE)))
return false; return false;
......
...@@ -356,6 +356,7 @@ static int mxic_spi_mem_exec_op(struct spi_mem *mem, ...@@ -356,6 +356,7 @@ static int mxic_spi_mem_exec_op(struct spi_mem *mem,
int nio = 1, i, ret; int nio = 1, i, ret;
u32 ss_ctrl; u32 ss_ctrl;
u8 addr[8]; u8 addr[8];
u8 opcode = op->cmd.opcode;
ret = mxic_spi_set_freq(mxic, mem->spi->max_speed_hz); ret = mxic_spi_set_freq(mxic, mem->spi->max_speed_hz);
if (ret) if (ret)
...@@ -393,7 +394,7 @@ static int mxic_spi_mem_exec_op(struct spi_mem *mem, ...@@ -393,7 +394,7 @@ static int mxic_spi_mem_exec_op(struct spi_mem *mem,
writel(readl(mxic->regs + HC_CFG) | HC_CFG_MAN_CS_ASSERT, writel(readl(mxic->regs + HC_CFG) | HC_CFG_MAN_CS_ASSERT,
mxic->regs + HC_CFG); mxic->regs + HC_CFG);
ret = mxic_spi_data_xfer(mxic, &op->cmd.opcode, NULL, 1); ret = mxic_spi_data_xfer(mxic, &opcode, NULL, 1);
if (ret) if (ret)
goto out; goto out;
......
...@@ -527,20 +527,21 @@ static int zynq_qspi_exec_mem_op(struct spi_mem *mem, ...@@ -527,20 +527,21 @@ static int zynq_qspi_exec_mem_op(struct spi_mem *mem,
struct zynq_qspi *xqspi = spi_controller_get_devdata(mem->spi->master); struct zynq_qspi *xqspi = spi_controller_get_devdata(mem->spi->master);
int err = 0, i; int err = 0, i;
u8 *tmpbuf; u8 *tmpbuf;
u8 opcode = op->cmd.opcode;
dev_dbg(xqspi->dev, "cmd:%#x mode:%d.%d.%d.%d\n", dev_dbg(xqspi->dev, "cmd:%#x mode:%d.%d.%d.%d\n",
op->cmd.opcode, op->cmd.buswidth, op->addr.buswidth, opcode, op->cmd.buswidth, op->addr.buswidth,
op->dummy.buswidth, op->data.buswidth); op->dummy.buswidth, op->data.buswidth);
zynq_qspi_chipselect(mem->spi, true); zynq_qspi_chipselect(mem->spi, true);
zynq_qspi_config_op(xqspi, mem->spi); zynq_qspi_config_op(xqspi, mem->spi);
if (op->cmd.opcode) { if (op->cmd.nbytes) {
reinit_completion(&xqspi->data_completion); reinit_completion(&xqspi->data_completion);
xqspi->txbuf = (u8 *)&op->cmd.opcode; xqspi->txbuf = &opcode;
xqspi->rxbuf = NULL; xqspi->rxbuf = NULL;
xqspi->tx_bytes = sizeof(op->cmd.opcode); xqspi->tx_bytes = op->cmd.nbytes;
xqspi->rx_bytes = sizeof(op->cmd.opcode); xqspi->rx_bytes = op->cmd.nbytes;
zynq_qspi_write_op(xqspi, ZYNQ_QSPI_FIFO_DEPTH, true); zynq_qspi_write_op(xqspi, ZYNQ_QSPI_FIFO_DEPTH, true);
zynq_qspi_write(xqspi, ZYNQ_QSPI_IEN_OFFSET, zynq_qspi_write(xqspi, ZYNQ_QSPI_IEN_OFFSET,
ZYNQ_QSPI_IXR_RXTX_MASK); ZYNQ_QSPI_IXR_RXTX_MASK);
......
...@@ -17,6 +17,7 @@ ...@@ -17,6 +17,7 @@
{ \ { \
.buswidth = __buswidth, \ .buswidth = __buswidth, \
.opcode = __opcode, \ .opcode = __opcode, \
.nbytes = 1, \
} }
#define SPI_MEM_OP_ADDR(__nbytes, __val, __buswidth) \ #define SPI_MEM_OP_ADDR(__nbytes, __val, __buswidth) \
...@@ -69,6 +70,8 @@ enum spi_mem_data_dir { ...@@ -69,6 +70,8 @@ enum spi_mem_data_dir {
/** /**
* struct spi_mem_op - describes a SPI memory operation * struct spi_mem_op - describes a SPI memory operation
* @cmd.nbytes: number of opcode bytes (only 1 or 2 are valid). The opcode is
* sent MSB-first.
* @cmd.buswidth: number of IO lines used to transmit the command * @cmd.buswidth: number of IO lines used to transmit the command
* @cmd.opcode: operation opcode * @cmd.opcode: operation opcode
* @cmd.dtr: whether the command opcode should be sent in DTR mode or not * @cmd.dtr: whether the command opcode should be sent in DTR mode or not
...@@ -94,9 +97,10 @@ enum spi_mem_data_dir { ...@@ -94,9 +97,10 @@ enum spi_mem_data_dir {
*/ */
struct spi_mem_op { struct spi_mem_op {
struct { struct {
u8 nbytes;
u8 buswidth; u8 buswidth;
u8 dtr : 1; u8 dtr : 1;
u8 opcode; u16 opcode;
} cmd; } cmd;
struct { struct {
......
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