aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/spi/spi-ti-qspi.c
diff options
context:
space:
mode:
authorVignesh R <vigneshr@ti.com>2015-08-20 06:30:59 -0400
committerMark Brown <broonie@kernel.org>2015-08-20 13:56:42 -0400
commitf682c4ffd25a19594d21987c19a69fa123242eb7 (patch)
tree40bb253223b77994bca219c996af5238c6632178 /drivers/spi/spi-ti-qspi.c
parentbc0195aad0daa2ad5b0d76cce22b167bc3435590 (diff)
spi: ti-qspi: use 128 bit transfer mode where possible
TI QSPI has four 32 bit data regsiters which can be used to transfer 16 bytes of data at once. The register group QSPI_SPI_DATA_REG_3, QSPI_SPI_DATA_REG_2, QSPI_SPI_DATA_REG_1 and QSPI_SPI_DATA_REG is treated as a single 128-bit word for shifting data in and out. The bit at QSPI_SPI_DATA_REG_3[31] position is the first bit to be shifted out in case of 128 bit transfer mode. Therefore the first byte to be written to flash should be at QSPI_SPI_DATA_REG_3[31-25] position. Instead of writing 1 byte at a time when interacting with spi-nor flash, make use of all the four registers so that 16 bytes can be transferred in one go. This reduces number of register writes and Word Complete interrupts for a given transfer message size, thereby increasing the write performance. Without this patch the raw flash write speed is ~100KB/s, with this patch the write speed increases to ~400 kB/s on DRA74 EVM. Signed-off-by: Vignesh R <vigneshr@ti.com> Signed-off-by: Mark Brown <broonie@kernel.org>
Diffstat (limited to 'drivers/spi/spi-ti-qspi.c')
-rw-r--r--drivers/spi/spi-ti-qspi.c34
1 files changed, 30 insertions, 4 deletions
diff --git a/drivers/spi/spi-ti-qspi.c b/drivers/spi/spi-ti-qspi.c
index 5c0616870358..aa6d284131e0 100644
--- a/drivers/spi/spi-ti-qspi.c
+++ b/drivers/spi/spi-ti-qspi.c
@@ -99,6 +99,8 @@ struct ti_qspi {
99#define QSPI_INVAL (4 << 16) 99#define QSPI_INVAL (4 << 16)
100#define QSPI_WC_CMD_INT_EN (1 << 14) 100#define QSPI_WC_CMD_INT_EN (1 << 14)
101#define QSPI_FLEN(n) ((n - 1) << 0) 101#define QSPI_FLEN(n) ((n - 1) << 0)
102#define QSPI_WLEN_MAX_BITS 128
103#define QSPI_WLEN_MAX_BYTES 16
102 104
103/* STATUS REGISTER */ 105/* STATUS REGISTER */
104#define BUSY 0x01 106#define BUSY 0x01
@@ -217,14 +219,16 @@ static inline u32 qspi_is_busy(struct ti_qspi *qspi)
217 219
218static int qspi_write_msg(struct ti_qspi *qspi, struct spi_transfer *t) 220static int qspi_write_msg(struct ti_qspi *qspi, struct spi_transfer *t)
219{ 221{
220 int wlen, count; 222 int wlen, count, xfer_len;
221 unsigned int cmd; 223 unsigned int cmd;
222 const u8 *txbuf; 224 const u8 *txbuf;
225 u32 data;
223 226
224 txbuf = t->tx_buf; 227 txbuf = t->tx_buf;
225 cmd = qspi->cmd | QSPI_WR_SNGL; 228 cmd = qspi->cmd | QSPI_WR_SNGL;
226 count = t->len; 229 count = t->len;
227 wlen = t->bits_per_word >> 3; /* in bytes */ 230 wlen = t->bits_per_word >> 3; /* in bytes */
231 xfer_len = wlen;
228 232
229 while (count) { 233 while (count) {
230 if (qspi_is_busy(qspi)) 234 if (qspi_is_busy(qspi))
@@ -234,7 +238,29 @@ static int qspi_write_msg(struct ti_qspi *qspi, struct spi_transfer *t)
234 case 1: 238 case 1:
235 dev_dbg(qspi->dev, "tx cmd %08x dc %08x data %02x\n", 239 dev_dbg(qspi->dev, "tx cmd %08x dc %08x data %02x\n",
236 cmd, qspi->dc, *txbuf); 240 cmd, qspi->dc, *txbuf);
237 writeb(*txbuf, qspi->base + QSPI_SPI_DATA_REG); 241 if (count >= QSPI_WLEN_MAX_BYTES) {
242 u32 *txp = (u32 *)txbuf;
243
244 data = cpu_to_be32(*txp++);
245 writel(data, qspi->base +
246 QSPI_SPI_DATA_REG_3);
247 data = cpu_to_be32(*txp++);
248 writel(data, qspi->base +
249 QSPI_SPI_DATA_REG_2);
250 data = cpu_to_be32(*txp++);
251 writel(data, qspi->base +
252 QSPI_SPI_DATA_REG_1);
253 data = cpu_to_be32(*txp++);
254 writel(data, qspi->base +
255 QSPI_SPI_DATA_REG);
256 xfer_len = QSPI_WLEN_MAX_BYTES;
257 cmd |= QSPI_WLEN(QSPI_WLEN_MAX_BITS);
258 } else {
259 writeb(*txbuf, qspi->base + QSPI_SPI_DATA_REG);
260 cmd = qspi->cmd | QSPI_WR_SNGL;
261 xfer_len = wlen;
262 cmd |= QSPI_WLEN(wlen);
263 }
238 break; 264 break;
239 case 2: 265 case 2:
240 dev_dbg(qspi->dev, "tx cmd %08x dc %08x data %04x\n", 266 dev_dbg(qspi->dev, "tx cmd %08x dc %08x data %04x\n",
@@ -254,8 +280,8 @@ static int qspi_write_msg(struct ti_qspi *qspi, struct spi_transfer *t)
254 dev_err(qspi->dev, "write timed out\n"); 280 dev_err(qspi->dev, "write timed out\n");
255 return -ETIMEDOUT; 281 return -ETIMEDOUT;
256 } 282 }
257 txbuf += wlen; 283 txbuf += xfer_len;
258 count -= wlen; 284 count -= xfer_len;
259 } 285 }
260 286
261 return 0; 287 return 0;