[PATCH 03/18] spi: qup: Add completion timeout for dma mode

2017-06-13 Thread Varadarajan Narayanan
Use different 'completion' structures to track the completion
of DMA Tx/Rx and PIO.

Signed-off-by: Andy Gross 
Signed-off-by: Varadarajan Narayanan 
---
 drivers/spi/spi-qup.c | 41 -
 1 file changed, 24 insertions(+), 17 deletions(-)

diff --git a/drivers/spi/spi-qup.c b/drivers/spi/spi-qup.c
index abe799b..272e48e 100644
--- a/drivers/spi/spi-qup.c
+++ b/drivers/spi/spi-qup.c
@@ -142,6 +142,8 @@ struct spi_qup {
 
struct spi_transfer *xfer;
struct completion   done;
+   struct completion   txc;
+   struct completion   rxc;
int error;
int w_size; /* bytes per SPI word */
int n_words;
@@ -283,16 +285,13 @@ static void spi_qup_fifo_write(struct spi_qup *controller,
 
 static void spi_qup_dma_done(void *data)
 {
-   struct spi_qup *qup = data;
-
-   complete(>done);
+   complete(data);
 }
 
 static int spi_qup_prep_sg(struct spi_master *master, struct spi_transfer 
*xfer,
   enum dma_transfer_direction dir,
-  dma_async_tx_callback callback)
+  dma_async_tx_callback callback, void *data)
 {
-   struct spi_qup *qup = spi_master_get_devdata(master);
unsigned long flags = DMA_PREP_INTERRUPT | DMA_PREP_FENCE;
struct dma_async_tx_descriptor *desc;
struct scatterlist *sgl;
@@ -315,7 +314,7 @@ static int spi_qup_prep_sg(struct spi_master *master, 
struct spi_transfer *xfer,
return -EINVAL;
 
desc->callback = callback;
-   desc->callback_param = qup;
+   desc->callback_param = data;
 
cookie = dmaengine_submit(desc);
 
@@ -333,16 +332,12 @@ static void spi_qup_dma_terminate(struct spi_master 
*master,
 
 static int spi_qup_do_dma(struct spi_master *master, struct spi_transfer *xfer)
 {
-   dma_async_tx_callback rx_done = NULL, tx_done = NULL;
+   struct spi_qup *qup = spi_master_get_devdata(master);
int ret;
 
-   if (xfer->rx_buf)
-   rx_done = spi_qup_dma_done;
-   else if (xfer->tx_buf)
-   tx_done = spi_qup_dma_done;
-
if (xfer->rx_buf) {
-   ret = spi_qup_prep_sg(master, xfer, DMA_DEV_TO_MEM, rx_done);
+   ret = spi_qup_prep_sg(master, xfer, DMA_DEV_TO_MEM,
+   spi_qup_dma_done, >rxc);
if (ret)
return ret;
 
@@ -350,13 +345,20 @@ static int spi_qup_do_dma(struct spi_master *master, 
struct spi_transfer *xfer)
}
 
if (xfer->tx_buf) {
-   ret = spi_qup_prep_sg(master, xfer, DMA_MEM_TO_DEV, tx_done);
+   ret = spi_qup_prep_sg(master, xfer, DMA_MEM_TO_DEV,
+   spi_qup_dma_done, >txc);
if (ret)
return ret;
 
dma_async_issue_pending(master->dma_tx);
}
 
+   if (xfer->rx_buf && !wait_for_completion_timeout(>rxc, timeout))
+   return -ETIMEDOUT;
+
+   if (xfer->tx_buf && !wait_for_completion_timeout(>txc, timeout))
+   return -ETIMEDOUT;
+
return 0;
 }
 
@@ -622,7 +624,6 @@ static int spi_qup_transfer_one(struct spi_master *master,
timeout = DIV_ROUND_UP(xfer->len * 8, timeout);
timeout = 100 * msecs_to_jiffies(timeout);
 
-   reinit_completion(>done);
 
spin_lock_irqsave(>lock, flags);
controller->xfer = xfer;
@@ -631,10 +632,14 @@ static int spi_qup_transfer_one(struct spi_master *master,
controller->tx_bytes = 0;
spin_unlock_irqrestore(>lock, flags);
 
-   if (spi_qup_is_dma_xfer(controller->mode))
+   if (spi_qup_is_dma_xfer(controller->mode)) {
+   reinit_completion(>rxc);
+   reinit_completion(>txc);
ret = spi_qup_do_dma(master, xfer);
-   else
+   } else {
+   reinit_completion(>done);
ret = spi_qup_do_pio(master, xfer);
+   }
 
if (ret)
goto exit;
@@ -860,6 +865,8 @@ static int spi_qup_probe(struct platform_device *pdev)
master->set_cs = spi_qup_set_cs;
 
spin_lock_init(>lock);
+   init_completion(>rxc);
+   init_completion(>txc);
init_completion(>done);
 
iomode = readl_relaxed(base + QUP_IO_M_MODES);
-- 
QUALCOMM INDIA, on behalf of Qualcomm Innovation Center, Inc. is a member of 
Code Aurora Forum, hosted by The Linux Foundation



[PATCH 03/18] spi: qup: Add completion timeout for dma mode

2017-06-13 Thread Varadarajan Narayanan
Use different 'completion' structures to track the completion
of DMA Tx/Rx and PIO.

Signed-off-by: Andy Gross 
Signed-off-by: Varadarajan Narayanan 
---
 drivers/spi/spi-qup.c | 41 -
 1 file changed, 24 insertions(+), 17 deletions(-)

diff --git a/drivers/spi/spi-qup.c b/drivers/spi/spi-qup.c
index abe799b..272e48e 100644
--- a/drivers/spi/spi-qup.c
+++ b/drivers/spi/spi-qup.c
@@ -142,6 +142,8 @@ struct spi_qup {
 
struct spi_transfer *xfer;
struct completion   done;
+   struct completion   txc;
+   struct completion   rxc;
int error;
int w_size; /* bytes per SPI word */
int n_words;
@@ -283,16 +285,13 @@ static void spi_qup_fifo_write(struct spi_qup *controller,
 
 static void spi_qup_dma_done(void *data)
 {
-   struct spi_qup *qup = data;
-
-   complete(>done);
+   complete(data);
 }
 
 static int spi_qup_prep_sg(struct spi_master *master, struct spi_transfer 
*xfer,
   enum dma_transfer_direction dir,
-  dma_async_tx_callback callback)
+  dma_async_tx_callback callback, void *data)
 {
-   struct spi_qup *qup = spi_master_get_devdata(master);
unsigned long flags = DMA_PREP_INTERRUPT | DMA_PREP_FENCE;
struct dma_async_tx_descriptor *desc;
struct scatterlist *sgl;
@@ -315,7 +314,7 @@ static int spi_qup_prep_sg(struct spi_master *master, 
struct spi_transfer *xfer,
return -EINVAL;
 
desc->callback = callback;
-   desc->callback_param = qup;
+   desc->callback_param = data;
 
cookie = dmaengine_submit(desc);
 
@@ -333,16 +332,12 @@ static void spi_qup_dma_terminate(struct spi_master 
*master,
 
 static int spi_qup_do_dma(struct spi_master *master, struct spi_transfer *xfer)
 {
-   dma_async_tx_callback rx_done = NULL, tx_done = NULL;
+   struct spi_qup *qup = spi_master_get_devdata(master);
int ret;
 
-   if (xfer->rx_buf)
-   rx_done = spi_qup_dma_done;
-   else if (xfer->tx_buf)
-   tx_done = spi_qup_dma_done;
-
if (xfer->rx_buf) {
-   ret = spi_qup_prep_sg(master, xfer, DMA_DEV_TO_MEM, rx_done);
+   ret = spi_qup_prep_sg(master, xfer, DMA_DEV_TO_MEM,
+   spi_qup_dma_done, >rxc);
if (ret)
return ret;
 
@@ -350,13 +345,20 @@ static int spi_qup_do_dma(struct spi_master *master, 
struct spi_transfer *xfer)
}
 
if (xfer->tx_buf) {
-   ret = spi_qup_prep_sg(master, xfer, DMA_MEM_TO_DEV, tx_done);
+   ret = spi_qup_prep_sg(master, xfer, DMA_MEM_TO_DEV,
+   spi_qup_dma_done, >txc);
if (ret)
return ret;
 
dma_async_issue_pending(master->dma_tx);
}
 
+   if (xfer->rx_buf && !wait_for_completion_timeout(>rxc, timeout))
+   return -ETIMEDOUT;
+
+   if (xfer->tx_buf && !wait_for_completion_timeout(>txc, timeout))
+   return -ETIMEDOUT;
+
return 0;
 }
 
@@ -622,7 +624,6 @@ static int spi_qup_transfer_one(struct spi_master *master,
timeout = DIV_ROUND_UP(xfer->len * 8, timeout);
timeout = 100 * msecs_to_jiffies(timeout);
 
-   reinit_completion(>done);
 
spin_lock_irqsave(>lock, flags);
controller->xfer = xfer;
@@ -631,10 +632,14 @@ static int spi_qup_transfer_one(struct spi_master *master,
controller->tx_bytes = 0;
spin_unlock_irqrestore(>lock, flags);
 
-   if (spi_qup_is_dma_xfer(controller->mode))
+   if (spi_qup_is_dma_xfer(controller->mode)) {
+   reinit_completion(>rxc);
+   reinit_completion(>txc);
ret = spi_qup_do_dma(master, xfer);
-   else
+   } else {
+   reinit_completion(>done);
ret = spi_qup_do_pio(master, xfer);
+   }
 
if (ret)
goto exit;
@@ -860,6 +865,8 @@ static int spi_qup_probe(struct platform_device *pdev)
master->set_cs = spi_qup_set_cs;
 
spin_lock_init(>lock);
+   init_completion(>rxc);
+   init_completion(>txc);
init_completion(>done);
 
iomode = readl_relaxed(base + QUP_IO_M_MODES);
-- 
QUALCOMM INDIA, on behalf of Qualcomm Innovation Center, Inc. is a member of 
Code Aurora Forum, hosted by The Linux Foundation