Commit 34904623 authored by Mark Brown's avatar Mark Brown

Merge remote-tracking branch 'spi/topic/dma' into spi-next

parents 42af2f5c 1ce24864
...@@ -73,7 +73,7 @@ ...@@ -73,7 +73,7 @@
#define MTK_SPI_IDLE 0 #define MTK_SPI_IDLE 0
#define MTK_SPI_PAUSED 1 #define MTK_SPI_PAUSED 1
#define MTK_SPI_MAX_FIFO_SIZE 32 #define MTK_SPI_MAX_FIFO_SIZE 32U
#define MTK_SPI_PACKET_SIZE 1024 #define MTK_SPI_PACKET_SIZE 1024
struct mtk_spi_compatible { struct mtk_spi_compatible {
...@@ -333,7 +333,7 @@ static int mtk_spi_fifo_transfer(struct spi_master *master, ...@@ -333,7 +333,7 @@ static int mtk_spi_fifo_transfer(struct spi_master *master,
struct mtk_spi *mdata = spi_master_get_devdata(master); struct mtk_spi *mdata = spi_master_get_devdata(master);
mdata->cur_transfer = xfer; mdata->cur_transfer = xfer;
mdata->xfer_len = xfer->len; mdata->xfer_len = min(MTK_SPI_MAX_FIFO_SIZE, xfer->len);
mtk_spi_prepare_transfer(master, xfer); mtk_spi_prepare_transfer(master, xfer);
mtk_spi_setup_packet(master); mtk_spi_setup_packet(master);
...@@ -410,7 +410,10 @@ static bool mtk_spi_can_dma(struct spi_master *master, ...@@ -410,7 +410,10 @@ static bool mtk_spi_can_dma(struct spi_master *master,
struct spi_device *spi, struct spi_device *spi,
struct spi_transfer *xfer) struct spi_transfer *xfer)
{ {
return xfer->len > MTK_SPI_MAX_FIFO_SIZE; /* Buffers for DMA transactions must be 4-byte aligned */
return (xfer->len > MTK_SPI_MAX_FIFO_SIZE &&
(unsigned long)xfer->tx_buf % 4 == 0 &&
(unsigned long)xfer->rx_buf % 4 == 0);
} }
static int mtk_spi_setup(struct spi_device *spi) static int mtk_spi_setup(struct spi_device *spi)
...@@ -451,10 +454,36 @@ static irqreturn_t mtk_spi_interrupt(int irq, void *dev_id) ...@@ -451,10 +454,36 @@ static irqreturn_t mtk_spi_interrupt(int irq, void *dev_id)
&reg_val, remainder); &reg_val, remainder);
} }
} }
trans->len -= mdata->xfer_len;
if (!trans->len) {
spi_finalize_current_transfer(master); spi_finalize_current_transfer(master);
return IRQ_HANDLED; return IRQ_HANDLED;
} }
if (trans->tx_buf)
trans->tx_buf += mdata->xfer_len;
if (trans->rx_buf)
trans->rx_buf += mdata->xfer_len;
mdata->xfer_len = min(MTK_SPI_MAX_FIFO_SIZE, trans->len);
mtk_spi_setup_packet(master);
cnt = trans->len / 4;
iowrite32_rep(mdata->base + SPI_TX_DATA_REG, trans->tx_buf, cnt);
remainder = trans->len % 4;
if (remainder > 0) {
reg_val = 0;
memcpy(&reg_val, trans->tx_buf + (cnt * 4), remainder);
writel(reg_val, mdata->base + SPI_TX_DATA_REG);
}
mtk_spi_enable_transfer(master);
return IRQ_HANDLED;
}
if (mdata->tx_sgl) if (mdata->tx_sgl)
trans->tx_dma += mdata->xfer_len; trans->tx_dma += mdata->xfer_len;
if (mdata->rx_sgl) if (mdata->rx_sgl)
......
...@@ -807,12 +807,12 @@ static int __spi_map_msg(struct spi_master *master, struct spi_message *msg) ...@@ -807,12 +807,12 @@ static int __spi_map_msg(struct spi_master *master, struct spi_message *msg)
if (master->dma_tx) if (master->dma_tx)
tx_dev = master->dma_tx->device->dev; tx_dev = master->dma_tx->device->dev;
else else
tx_dev = &master->dev; tx_dev = master->dev.parent;
if (master->dma_rx) if (master->dma_rx)
rx_dev = master->dma_rx->device->dev; rx_dev = master->dma_rx->device->dev;
else else
rx_dev = &master->dev; rx_dev = master->dev.parent;
list_for_each_entry(xfer, &msg->transfers, transfer_list) { list_for_each_entry(xfer, &msg->transfers, transfer_list) {
if (!master->can_dma(master, msg->spi, xfer)) if (!master->can_dma(master, msg->spi, xfer))
...@@ -854,12 +854,12 @@ static int __spi_unmap_msg(struct spi_master *master, struct spi_message *msg) ...@@ -854,12 +854,12 @@ static int __spi_unmap_msg(struct spi_master *master, struct spi_message *msg)
if (master->dma_tx) if (master->dma_tx)
tx_dev = master->dma_tx->device->dev; tx_dev = master->dma_tx->device->dev;
else else
tx_dev = &master->dev; tx_dev = master->dev.parent;
if (master->dma_rx) if (master->dma_rx)
rx_dev = master->dma_rx->device->dev; rx_dev = master->dma_rx->device->dev;
else else
rx_dev = &master->dev; rx_dev = master->dev.parent;
list_for_each_entry(xfer, &msg->transfers, transfer_list) { list_for_each_entry(xfer, &msg->transfers, transfer_list) {
if (!master->can_dma(master, msg->spi, xfer)) if (!master->can_dma(master, msg->spi, xfer))
......
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