2013-01-22 12:26:29 +02:00
/*
* PXA2xx SPI DMA engine support .
*
* Copyright ( C ) 2013 , Intel Corporation
* Author : Mika Westerberg < mika . westerberg @ linux . intel . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/device.h>
# include <linux/dma-mapping.h>
# include <linux/dmaengine.h>
# include <linux/pxa2xx_ssp.h>
# include <linux/scatterlist.h>
# include <linux/sizes.h>
# include <linux/spi/spi.h>
# include <linux/spi/pxa2xx_spi.h>
# include "spi-pxa2xx.h"
static void pxa2xx_spi_dma_transfer_complete ( struct driver_data * drv_data ,
bool error )
{
struct spi_message * msg = drv_data - > cur_msg ;
/*
* It is possible that one CPU is handling ROR interrupt and other
* just gets DMA completion . Calling pump_transfers ( ) twice for the
* same transfer leads to problems thus we prevent concurrent calls
* by using - > dma_running .
*/
if ( atomic_dec_and_test ( & drv_data - > dma_running ) ) {
/*
* If the other CPU is still handling the ROR interrupt we
* might not know about the error yet . So we re - check the
* ROR bit here before we clear the status register .
*/
if ( ! error ) {
2014-12-18 15:04:23 +02:00
u32 status = pxa2xx_spi_read ( drv_data , SSSR )
& drv_data - > mask_sr ;
2013-01-22 12:26:29 +02:00
error = status & SSSR_ROR ;
}
/* Clear status & disable interrupts */
2014-12-18 15:04:23 +02:00
pxa2xx_spi_write ( drv_data , SSCR1 ,
pxa2xx_spi_read ( drv_data , SSCR1 )
& ~ drv_data - > dma_cr1 ) ;
2013-01-22 12:26:29 +02:00
write_SSSR_CS ( drv_data , drv_data - > clear_sr ) ;
if ( ! pxa25x_ssp_comp ( drv_data ) )
2014-12-18 15:04:23 +02:00
pxa2xx_spi_write ( drv_data , SSTO , 0 ) ;
2013-01-22 12:26:29 +02:00
if ( ! error ) {
msg - > actual_length + = drv_data - > len ;
msg - > state = pxa2xx_spi_next_transfer ( drv_data ) ;
} else {
/* In case we got an error we disable the SSP now */
2014-12-18 15:04:23 +02:00
pxa2xx_spi_write ( drv_data , SSCR0 ,
pxa2xx_spi_read ( drv_data , SSCR0 )
& ~ SSCR0_SSE ) ;
2013-01-22 12:26:29 +02:00
msg - > state = ERROR_STATE ;
}
tasklet_schedule ( & drv_data - > pump_transfers ) ;
}
}
static void pxa2xx_spi_dma_callback ( void * data )
{
pxa2xx_spi_dma_transfer_complete ( data , false ) ;
}
static struct dma_async_tx_descriptor *
pxa2xx_spi_dma_prepare_one ( struct driver_data * drv_data ,
enum dma_transfer_direction dir )
{
struct chip_data * chip = drv_data - > cur_chip ;
2016-06-21 13:21:34 +03:00
struct spi_transfer * xfer = drv_data - > cur_transfer ;
2013-01-22 12:26:29 +02:00
enum dma_slave_buswidth width ;
struct dma_slave_config cfg ;
struct dma_chan * chan ;
struct sg_table * sgt ;
2016-06-21 13:21:34 +03:00
int ret ;
2013-01-22 12:26:29 +02:00
switch ( drv_data - > n_bytes ) {
case 1 :
width = DMA_SLAVE_BUSWIDTH_1_BYTE ;
break ;
case 2 :
width = DMA_SLAVE_BUSWIDTH_2_BYTES ;
break ;
default :
width = DMA_SLAVE_BUSWIDTH_4_BYTES ;
break ;
}
memset ( & cfg , 0 , sizeof ( cfg ) ) ;
cfg . direction = dir ;
if ( dir = = DMA_MEM_TO_DEV ) {
cfg . dst_addr = drv_data - > ssdr_physical ;
cfg . dst_addr_width = width ;
cfg . dst_maxburst = chip - > dma_burst_size ;
2016-06-21 13:21:34 +03:00
sgt = & xfer - > tx_sg ;
chan = drv_data - > master - > dma_tx ;
2013-01-22 12:26:29 +02:00
} else {
cfg . src_addr = drv_data - > ssdr_physical ;
cfg . src_addr_width = width ;
cfg . src_maxburst = chip - > dma_burst_size ;
2016-06-21 13:21:34 +03:00
sgt = & xfer - > rx_sg ;
chan = drv_data - > master - > dma_rx ;
2013-01-22 12:26:29 +02:00
}
ret = dmaengine_slave_config ( chan , & cfg ) ;
if ( ret ) {
dev_warn ( & drv_data - > pdev - > dev , " DMA slave config failed \n " ) ;
return NULL ;
}
2016-06-21 13:21:34 +03:00
return dmaengine_prep_slave_sg ( chan , sgt - > sgl , sgt - > nents , dir ,
2013-01-22 12:26:29 +02:00
DMA_PREP_INTERRUPT | DMA_CTRL_ACK ) ;
}
irqreturn_t pxa2xx_spi_dma_transfer ( struct driver_data * drv_data )
{
u32 status ;
2014-12-18 15:04:23 +02:00
status = pxa2xx_spi_read ( drv_data , SSSR ) & drv_data - > mask_sr ;
2013-01-22 12:26:29 +02:00
if ( status & SSSR_ROR ) {
dev_err ( & drv_data - > pdev - > dev , " FIFO overrun \n " ) ;
2016-06-21 13:21:34 +03:00
dmaengine_terminate_async ( drv_data - > master - > dma_rx ) ;
dmaengine_terminate_async ( drv_data - > master - > dma_tx ) ;
2013-01-22 12:26:29 +02:00
pxa2xx_spi_dma_transfer_complete ( drv_data , true ) ;
return IRQ_HANDLED ;
}
return IRQ_NONE ;
}
int pxa2xx_spi_dma_prepare ( struct driver_data * drv_data , u32 dma_burst )
{
struct dma_async_tx_descriptor * tx_desc , * rx_desc ;
2016-03-24 15:35:42 +02:00
int err = 0 ;
2013-01-22 12:26:29 +02:00
tx_desc = pxa2xx_spi_dma_prepare_one ( drv_data , DMA_MEM_TO_DEV ) ;
if ( ! tx_desc ) {
dev_err ( & drv_data - > pdev - > dev ,
" failed to get DMA TX descriptor \n " ) ;
2016-03-24 15:35:42 +02:00
err = - EBUSY ;
goto err_tx ;
2013-01-22 12:26:29 +02:00
}
rx_desc = pxa2xx_spi_dma_prepare_one ( drv_data , DMA_DEV_TO_MEM ) ;
if ( ! rx_desc ) {
dev_err ( & drv_data - > pdev - > dev ,
" failed to get DMA RX descriptor \n " ) ;
2016-03-24 15:35:42 +02:00
err = - EBUSY ;
goto err_rx ;
2013-01-22 12:26:29 +02:00
}
/* We are ready when RX completes */
rx_desc - > callback = pxa2xx_spi_dma_callback ;
rx_desc - > callback_param = drv_data ;
dmaengine_submit ( rx_desc ) ;
dmaengine_submit ( tx_desc ) ;
return 0 ;
2016-03-24 15:35:42 +02:00
err_rx :
2016-06-21 13:21:34 +03:00
dmaengine_terminate_async ( drv_data - > master - > dma_tx ) ;
2016-03-24 15:35:42 +02:00
err_tx :
return err ;
2013-01-22 12:26:29 +02:00
}
void pxa2xx_spi_dma_start ( struct driver_data * drv_data )
{
2016-06-21 13:21:34 +03:00
dma_async_issue_pending ( drv_data - > master - > dma_rx ) ;
dma_async_issue_pending ( drv_data - > master - > dma_tx ) ;
2013-01-22 12:26:29 +02:00
atomic_set ( & drv_data - > dma_running , 1 ) ;
}
int pxa2xx_spi_dma_setup ( struct driver_data * drv_data )
{
struct pxa2xx_spi_master * pdata = drv_data - > master_info ;
2013-05-13 13:45:10 +03:00
struct device * dev = & drv_data - > pdev - > dev ;
2016-06-21 13:21:34 +03:00
struct spi_master * master = drv_data - > master ;
2013-01-22 12:26:29 +02:00
dma_cap_mask_t mask ;
dma_cap_zero ( mask ) ;
dma_cap_set ( DMA_SLAVE , mask ) ;
2016-06-21 13:21:34 +03:00
master - > dma_tx = dma_request_slave_channel_compat ( mask ,
2014-08-19 20:29:19 +03:00
pdata - > dma_filter , pdata - > tx_param , dev , " tx " ) ;
2016-06-21 13:21:34 +03:00
if ( ! master - > dma_tx )
2013-01-22 12:26:29 +02:00
return - ENODEV ;
2016-06-21 13:21:34 +03:00
master - > dma_rx = dma_request_slave_channel_compat ( mask ,
2014-08-19 20:29:19 +03:00
pdata - > dma_filter , pdata - > rx_param , dev , " rx " ) ;
2016-06-21 13:21:34 +03:00
if ( ! master - > dma_rx ) {
dma_release_channel ( master - > dma_tx ) ;
master - > dma_tx = NULL ;
2013-01-22 12:26:29 +02:00
return - ENODEV ;
}
return 0 ;
}
void pxa2xx_spi_dma_release ( struct driver_data * drv_data )
{
2016-06-21 13:21:34 +03:00
struct spi_master * master = drv_data - > master ;
if ( master - > dma_rx ) {
dmaengine_terminate_sync ( master - > dma_rx ) ;
dma_release_channel ( master - > dma_rx ) ;
master - > dma_rx = NULL ;
2013-01-22 12:26:29 +02:00
}
2016-06-21 13:21:34 +03:00
if ( master - > dma_tx ) {
dmaengine_terminate_sync ( master - > dma_tx ) ;
dma_release_channel ( master - > dma_tx ) ;
master - > dma_tx = NULL ;
2013-01-22 12:26:29 +02:00
}
}
int pxa2xx_spi_set_dma_burst_and_threshold ( struct chip_data * chip ,
struct spi_device * spi ,
u8 bits_per_word , u32 * burst_code ,
u32 * threshold )
{
struct pxa2xx_spi_chip * chip_info = spi - > controller_data ;
/*
* If the DMA burst size is given in chip_info we use that ,
* otherwise we use the default . Also we use the default FIFO
* thresholds for now .
*/
2014-06-06 01:45:09 +08:00
* burst_code = chip_info ? chip_info - > dma_burst_size : 1 ;
2013-01-22 12:26:29 +02:00
* threshold = SSCR1_RxTresh ( RX_THRESH_DFLT )
| SSCR1_TxTresh ( TX_THRESH_DFLT ) ;
return 0 ;
}