2011-10-28 06:22:39 +04:00
/*
* DMA controller driver for CSR SiRFprimaII
*
* Copyright ( c ) 2011 Cambridge Silicon Radio Limited , a CSR plc group company .
*
* Licensed under GPLv2 or later .
*/
# include <linux/module.h>
# include <linux/dmaengine.h>
# include <linux/dma-mapping.h>
2013-07-30 13:44:34 +04:00
# include <linux/pm_runtime.h>
2011-10-28 06:22:39 +04:00
# include <linux/interrupt.h>
# include <linux/io.h>
# include <linux/slab.h>
# include <linux/of_irq.h>
# include <linux/of_address.h>
# include <linux/of_device.h>
# include <linux/of_platform.h>
2013-03-18 12:33:43 +04:00
# include <linux/clk.h>
2014-03-27 11:49:31 +04:00
# include <linux/of_dma.h>
2011-10-28 06:22:39 +04:00
# include <linux/sirfsoc_dma.h>
2012-03-13 10:28:12 +04:00
# include "dmaengine.h"
2015-05-26 10:32:28 +03:00
# define SIRFSOC_DMA_VER_A7V1 1
# define SIRFSOC_DMA_VER_A7V2 2
# define SIRFSOC_DMA_VER_A6 4
2011-10-28 06:22:39 +04:00
# define SIRFSOC_DMA_DESCRIPTORS 16
# define SIRFSOC_DMA_CHANNELS 16
2015-05-26 10:32:28 +03:00
# define SIRFSOC_DMA_TABLE_NUM 256
2011-10-28 06:22:39 +04:00
# define SIRFSOC_DMA_CH_ADDR 0x00
# define SIRFSOC_DMA_CH_XLEN 0x04
# define SIRFSOC_DMA_CH_YLEN 0x08
# define SIRFSOC_DMA_CH_CTRL 0x0C
# define SIRFSOC_DMA_WIDTH_0 0x100
# define SIRFSOC_DMA_CH_VALID 0x140
# define SIRFSOC_DMA_CH_INT 0x144
# define SIRFSOC_DMA_INT_EN 0x148
2015-05-26 10:32:28 +03:00
# define SIRFSOC_DMA_INT_EN_CLR 0x14C
2011-10-28 06:22:39 +04:00
# define SIRFSOC_DMA_CH_LOOP_CTRL 0x150
2015-05-26 10:32:28 +03:00
# define SIRFSOC_DMA_CH_LOOP_CTRL_CLR 0x154
# define SIRFSOC_DMA_WIDTH_ATLAS7 0x10
# define SIRFSOC_DMA_VALID_ATLAS7 0x14
# define SIRFSOC_DMA_INT_ATLAS7 0x18
# define SIRFSOC_DMA_INT_EN_ATLAS7 0x1c
# define SIRFSOC_DMA_LOOP_CTRL_ATLAS7 0x20
# define SIRFSOC_DMA_CUR_DATA_ADDR 0x34
# define SIRFSOC_DMA_MUL_ATLAS7 0x38
# define SIRFSOC_DMA_CH_LOOP_CTRL_ATLAS7 0x158
# define SIRFSOC_DMA_CH_LOOP_CTRL_CLR_ATLAS7 0x15C
# define SIRFSOC_DMA_IOBG_SCMD_EN 0x800
# define SIRFSOC_DMA_EARLY_RESP_SET 0x818
# define SIRFSOC_DMA_EARLY_RESP_CLR 0x81C
2011-10-28 06:22:39 +04:00
# define SIRFSOC_DMA_MODE_CTRL_BIT 4
# define SIRFSOC_DMA_DIR_CTRL_BIT 5
2015-05-26 10:32:28 +03:00
# define SIRFSOC_DMA_MODE_CTRL_BIT_ATLAS7 2
# define SIRFSOC_DMA_CHAIN_CTRL_BIT_ATLAS7 3
# define SIRFSOC_DMA_DIR_CTRL_BIT_ATLAS7 4
# define SIRFSOC_DMA_TAB_NUM_ATLAS7 7
# define SIRFSOC_DMA_CHAIN_INT_BIT_ATLAS7 5
# define SIRFSOC_DMA_CHAIN_FLAG_SHIFT_ATLAS7 25
# define SIRFSOC_DMA_CHAIN_ADDR_SHIFT 32
# define SIRFSOC_DMA_INT_FINI_INT_ATLAS7 BIT(0)
# define SIRFSOC_DMA_INT_CNT_INT_ATLAS7 BIT(1)
# define SIRFSOC_DMA_INT_PAU_INT_ATLAS7 BIT(2)
# define SIRFSOC_DMA_INT_LOOP_INT_ATLAS7 BIT(3)
# define SIRFSOC_DMA_INT_INV_INT_ATLAS7 BIT(4)
# define SIRFSOC_DMA_INT_END_INT_ATLAS7 BIT(5)
# define SIRFSOC_DMA_INT_ALL_ATLAS7 0x3F
2011-10-28 06:22:39 +04:00
/* xlen and dma_width register is in 4 bytes boundary */
# define SIRFSOC_DMA_WORD_LEN 4
2015-05-26 10:32:28 +03:00
# define SIRFSOC_DMA_XLEN_MAX_V1 0x800
# define SIRFSOC_DMA_XLEN_MAX_V2 0x1000
2011-10-28 06:22:39 +04:00
struct sirfsoc_dma_desc {
struct dma_async_tx_descriptor desc ;
struct list_head node ;
/* SiRFprimaII 2D-DMA parameters */
int xlen ; /* DMA xlen */
int ylen ; /* DMA ylen */
int width ; /* DMA width */
int dir ;
bool cyclic ; /* is loop DMA? */
2015-05-26 10:32:28 +03:00
bool chain ; /* is chain DMA? */
2011-10-28 06:22:39 +04:00
u32 addr ; /* DMA buffer address */
2015-05-26 10:32:28 +03:00
u64 chain_table [ SIRFSOC_DMA_TABLE_NUM ] ; /* chain tbl */
2011-10-28 06:22:39 +04:00
} ;
struct sirfsoc_dma_chan {
struct dma_chan chan ;
struct list_head free ;
struct list_head prepared ;
struct list_head queued ;
struct list_head active ;
struct list_head completed ;
unsigned long happened_cyclic ;
unsigned long completed_cyclic ;
/* Lock for this structure */
spinlock_t lock ;
int mode ;
} ;
2013-07-30 13:44:34 +04:00
struct sirfsoc_dma_regs {
u32 ctrl [ SIRFSOC_DMA_CHANNELS ] ;
u32 interrupt_en ;
} ;
2011-10-28 06:22:39 +04:00
struct sirfsoc_dma {
struct dma_device dma ;
struct tasklet_struct tasklet ;
struct sirfsoc_dma_chan channels [ SIRFSOC_DMA_CHANNELS ] ;
void __iomem * base ;
int irq ;
2013-03-18 12:33:43 +04:00
struct clk * clk ;
2015-05-26 10:32:28 +03:00
int type ;
void ( * exec_desc ) ( struct sirfsoc_dma_desc * sdesc ,
int cid , int burst_mode , void __iomem * base ) ;
2013-07-30 13:44:34 +04:00
struct sirfsoc_dma_regs regs_save ;
2011-10-28 06:22:39 +04:00
} ;
2015-05-26 10:32:28 +03:00
struct sirfsoc_dmadata {
void ( * exec ) ( struct sirfsoc_dma_desc * sdesc ,
int cid , int burst_mode , void __iomem * base ) ;
int type ;
} ;
enum sirfsoc_dma_chain_flag {
SIRFSOC_DMA_CHAIN_NORMAL = 0x01 ,
SIRFSOC_DMA_CHAIN_PAUSE = 0x02 ,
SIRFSOC_DMA_CHAIN_LOOP = 0x03 ,
SIRFSOC_DMA_CHAIN_END = 0x04
} ;
2011-10-28 06:22:39 +04:00
# define DRV_NAME "sirfsoc_dma"
2013-07-30 13:44:34 +04:00
static int sirfsoc_dma_runtime_suspend ( struct device * dev ) ;
2011-10-28 06:22:39 +04:00
/* Convert struct dma_chan to struct sirfsoc_dma_chan */
static inline
struct sirfsoc_dma_chan * dma_chan_to_sirfsoc_dma_chan ( struct dma_chan * c )
{
return container_of ( c , struct sirfsoc_dma_chan , chan ) ;
}
/* Convert struct dma_chan to struct sirfsoc_dma */
static inline struct sirfsoc_dma * dma_chan_to_sirfsoc_dma ( struct dma_chan * c )
{
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( c ) ;
return container_of ( schan , struct sirfsoc_dma , channels [ c - > chan_id ] ) ;
}
2015-05-26 10:32:28 +03:00
static void sirfsoc_dma_execute_hw_a7v2 ( struct sirfsoc_dma_desc * sdesc ,
int cid , int burst_mode , void __iomem * base )
{
if ( sdesc - > chain ) {
/* DMA v2 HW chain mode */
writel_relaxed ( ( sdesc - > dir < < SIRFSOC_DMA_DIR_CTRL_BIT_ATLAS7 ) |
( sdesc - > chain < <
SIRFSOC_DMA_CHAIN_CTRL_BIT_ATLAS7 ) |
( 0x8 < < SIRFSOC_DMA_TAB_NUM_ATLAS7 ) | 0x3 ,
base + SIRFSOC_DMA_CH_CTRL ) ;
} else {
/* DMA v2 legacy mode */
writel_relaxed ( sdesc - > xlen , base + SIRFSOC_DMA_CH_XLEN ) ;
writel_relaxed ( sdesc - > ylen , base + SIRFSOC_DMA_CH_YLEN ) ;
writel_relaxed ( sdesc - > width , base + SIRFSOC_DMA_WIDTH_ATLAS7 ) ;
writel_relaxed ( ( sdesc - > width * ( ( sdesc - > ylen + 1 ) > > 1 ) ) ,
base + SIRFSOC_DMA_MUL_ATLAS7 ) ;
writel_relaxed ( ( sdesc - > dir < < SIRFSOC_DMA_DIR_CTRL_BIT_ATLAS7 ) |
( sdesc - > chain < <
SIRFSOC_DMA_CHAIN_CTRL_BIT_ATLAS7 ) |
0x3 , base + SIRFSOC_DMA_CH_CTRL ) ;
}
writel_relaxed ( sdesc - > chain ? SIRFSOC_DMA_INT_END_INT_ATLAS7 :
( SIRFSOC_DMA_INT_FINI_INT_ATLAS7 |
SIRFSOC_DMA_INT_LOOP_INT_ATLAS7 ) ,
base + SIRFSOC_DMA_INT_EN_ATLAS7 ) ;
writel ( sdesc - > addr , base + SIRFSOC_DMA_CH_ADDR ) ;
if ( sdesc - > cyclic )
writel ( 0x10001 , base + SIRFSOC_DMA_LOOP_CTRL_ATLAS7 ) ;
}
static void sirfsoc_dma_execute_hw_a7v1 ( struct sirfsoc_dma_desc * sdesc ,
int cid , int burst_mode , void __iomem * base )
{
writel_relaxed ( 1 , base + SIRFSOC_DMA_IOBG_SCMD_EN ) ;
writel_relaxed ( ( 1 < < cid ) , base + SIRFSOC_DMA_EARLY_RESP_SET ) ;
writel_relaxed ( sdesc - > width , base + SIRFSOC_DMA_WIDTH_0 + cid * 4 ) ;
writel_relaxed ( cid | ( burst_mode < < SIRFSOC_DMA_MODE_CTRL_BIT ) |
( sdesc - > dir < < SIRFSOC_DMA_DIR_CTRL_BIT ) ,
base + cid * 0x10 + SIRFSOC_DMA_CH_CTRL ) ;
writel_relaxed ( sdesc - > xlen , base + cid * 0x10 + SIRFSOC_DMA_CH_XLEN ) ;
writel_relaxed ( sdesc - > ylen , base + cid * 0x10 + SIRFSOC_DMA_CH_YLEN ) ;
writel_relaxed ( readl_relaxed ( base + SIRFSOC_DMA_INT_EN ) |
( 1 < < cid ) , base + SIRFSOC_DMA_INT_EN ) ;
writel ( sdesc - > addr > > 2 , base + cid * 0x10 + SIRFSOC_DMA_CH_ADDR ) ;
if ( sdesc - > cyclic ) {
writel ( ( 1 < < cid ) | 1 < < ( cid + 16 ) |
readl_relaxed ( base + SIRFSOC_DMA_CH_LOOP_CTRL_ATLAS7 ) ,
base + SIRFSOC_DMA_CH_LOOP_CTRL_ATLAS7 ) ;
}
}
static void sirfsoc_dma_execute_hw_a6 ( struct sirfsoc_dma_desc * sdesc ,
int cid , int burst_mode , void __iomem * base )
{
writel_relaxed ( sdesc - > width , base + SIRFSOC_DMA_WIDTH_0 + cid * 4 ) ;
writel_relaxed ( cid | ( burst_mode < < SIRFSOC_DMA_MODE_CTRL_BIT ) |
( sdesc - > dir < < SIRFSOC_DMA_DIR_CTRL_BIT ) ,
base + cid * 0x10 + SIRFSOC_DMA_CH_CTRL ) ;
writel_relaxed ( sdesc - > xlen , base + cid * 0x10 + SIRFSOC_DMA_CH_XLEN ) ;
writel_relaxed ( sdesc - > ylen , base + cid * 0x10 + SIRFSOC_DMA_CH_YLEN ) ;
writel_relaxed ( readl_relaxed ( base + SIRFSOC_DMA_INT_EN ) |
( 1 < < cid ) , base + SIRFSOC_DMA_INT_EN ) ;
writel ( sdesc - > addr > > 2 , base + cid * 0x10 + SIRFSOC_DMA_CH_ADDR ) ;
if ( sdesc - > cyclic ) {
writel ( ( 1 < < cid ) | 1 < < ( cid + 16 ) |
readl_relaxed ( base + SIRFSOC_DMA_CH_LOOP_CTRL ) ,
base + SIRFSOC_DMA_CH_LOOP_CTRL ) ;
}
}
2011-10-28 06:22:39 +04:00
/* Execute all queued DMA descriptors */
static void sirfsoc_dma_execute ( struct sirfsoc_dma_chan * schan )
{
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( & schan - > chan ) ;
int cid = schan - > chan . chan_id ;
struct sirfsoc_dma_desc * sdesc = NULL ;
2015-05-26 10:32:28 +03:00
void __iomem * base ;
2011-10-28 06:22:39 +04:00
/*
* lock has been held by functions calling this , so we don ' t hold
* lock again
*/
2015-05-26 10:32:28 +03:00
base = sdma - > base ;
2011-10-28 06:22:39 +04:00
sdesc = list_first_entry ( & schan - > queued , struct sirfsoc_dma_desc ,
2015-05-26 10:32:28 +03:00
node ) ;
2011-10-28 06:22:39 +04:00
/* Move the first queued descriptor to active list */
2012-09-27 12:36:10 +04:00
list_move_tail ( & sdesc - > node , & schan - > active ) ;
2011-10-28 06:22:39 +04:00
2015-05-26 10:32:28 +03:00
if ( sdma - > type = = SIRFSOC_DMA_VER_A7V2 )
cid = 0 ;
2011-10-28 06:22:39 +04:00
2015-05-26 10:32:28 +03:00
/* Start the DMA transfer */
sdma - > exec_desc ( sdesc , cid , schan - > mode , base ) ;
2011-10-28 06:22:39 +04:00
2015-05-26 10:32:28 +03:00
if ( sdesc - > cyclic )
2011-10-28 06:22:39 +04:00
schan - > happened_cyclic = schan - > completed_cyclic = 0 ;
}
/* Interrupt handler */
static irqreturn_t sirfsoc_dma_irq ( int irq , void * data )
{
struct sirfsoc_dma * sdma = data ;
struct sirfsoc_dma_chan * schan ;
struct sirfsoc_dma_desc * sdesc = NULL ;
u32 is ;
2015-05-26 10:32:28 +03:00
bool chain ;
2011-10-28 06:22:39 +04:00
int ch ;
2015-05-26 10:32:28 +03:00
void __iomem * reg ;
switch ( sdma - > type ) {
case SIRFSOC_DMA_VER_A6 :
case SIRFSOC_DMA_VER_A7V1 :
is = readl ( sdma - > base + SIRFSOC_DMA_CH_INT ) ;
reg = sdma - > base + SIRFSOC_DMA_CH_INT ;
while ( ( ch = fls ( is ) - 1 ) > = 0 ) {
is & = ~ ( 1 < < ch ) ;
writel_relaxed ( 1 < < ch , reg ) ;
schan = & sdma - > channels [ ch ] ;
spin_lock ( & schan - > lock ) ;
sdesc = list_first_entry ( & schan - > active ,
struct sirfsoc_dma_desc , node ) ;
if ( ! sdesc - > cyclic ) {
/* Execute queued descriptors */
list_splice_tail_init ( & schan - > active ,
& schan - > completed ) ;
dma_cookie_complete ( & sdesc - > desc ) ;
if ( ! list_empty ( & schan - > queued ) )
sirfsoc_dma_execute ( schan ) ;
} else
schan - > happened_cyclic + + ;
spin_unlock ( & schan - > lock ) ;
}
break ;
2011-10-28 06:22:39 +04:00
2015-05-26 10:32:28 +03:00
case SIRFSOC_DMA_VER_A7V2 :
is = readl ( sdma - > base + SIRFSOC_DMA_INT_ATLAS7 ) ;
2011-10-28 06:22:39 +04:00
2015-05-26 10:32:28 +03:00
reg = sdma - > base + SIRFSOC_DMA_INT_ATLAS7 ;
writel_relaxed ( SIRFSOC_DMA_INT_ALL_ATLAS7 , reg ) ;
schan = & sdma - > channels [ 0 ] ;
2011-10-28 06:22:39 +04:00
spin_lock ( & schan - > lock ) ;
2015-05-26 10:32:28 +03:00
sdesc = list_first_entry ( & schan - > active ,
struct sirfsoc_dma_desc , node ) ;
2011-10-28 06:22:39 +04:00
if ( ! sdesc - > cyclic ) {
2015-05-26 10:32:28 +03:00
chain = sdesc - > chain ;
if ( ( chain & & ( is & SIRFSOC_DMA_INT_END_INT_ATLAS7 ) ) | |
( ! chain & &
( is & SIRFSOC_DMA_INT_FINI_INT_ATLAS7 ) ) ) {
/* Execute queued descriptors */
list_splice_tail_init ( & schan - > active ,
& schan - > completed ) ;
dma_cookie_complete ( & sdesc - > desc ) ;
if ( ! list_empty ( & schan - > queued ) )
sirfsoc_dma_execute ( schan ) ;
}
} else if ( sdesc - > cyclic & & ( is &
SIRFSOC_DMA_INT_LOOP_INT_ATLAS7 ) )
2011-10-28 06:22:39 +04:00
schan - > happened_cyclic + + ;
spin_unlock ( & schan - > lock ) ;
2015-05-26 10:32:28 +03:00
break ;
default :
break ;
2011-10-28 06:22:39 +04:00
}
/* Schedule tasklet */
tasklet_schedule ( & sdma - > tasklet ) ;
return IRQ_HANDLED ;
}
/* process completed descriptors */
static void sirfsoc_dma_process_completed ( struct sirfsoc_dma * sdma )
{
dma_cookie_t last_cookie = 0 ;
struct sirfsoc_dma_chan * schan ;
struct sirfsoc_dma_desc * sdesc ;
struct dma_async_tx_descriptor * desc ;
unsigned long flags ;
unsigned long happened_cyclic ;
LIST_HEAD ( list ) ;
int i ;
for ( i = 0 ; i < sdma - > dma . chancnt ; i + + ) {
schan = & sdma - > channels [ i ] ;
/* Get all completed descriptors */
spin_lock_irqsave ( & schan - > lock , flags ) ;
if ( ! list_empty ( & schan - > completed ) ) {
list_splice_tail_init ( & schan - > completed , & list ) ;
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
/* Execute callbacks and run dependencies */
list_for_each_entry ( sdesc , & list , node ) {
desc = & sdesc - > desc ;
2016-07-20 23:13:05 +03:00
dmaengine_desc_get_callback_invoke ( desc , NULL ) ;
2011-10-28 06:22:39 +04:00
last_cookie = desc - > cookie ;
dma_run_dependencies ( desc ) ;
}
/* Free descriptors */
spin_lock_irqsave ( & schan - > lock , flags ) ;
list_splice_tail_init ( & list , & schan - > free ) ;
2012-03-07 02:34:06 +04:00
schan - > chan . completed_cookie = last_cookie ;
2011-10-28 06:22:39 +04:00
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
} else {
2015-05-26 10:32:28 +03:00
if ( list_empty ( & schan - > active ) ) {
2011-10-28 06:22:39 +04:00
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
continue ;
}
2015-05-26 10:32:28 +03:00
/* for cyclic channel, desc is always in active list */
sdesc = list_first_entry ( & schan - > active ,
struct sirfsoc_dma_desc , node ) ;
2011-10-28 06:22:39 +04:00
/* cyclic DMA */
happened_cyclic = schan - > happened_cyclic ;
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
desc = & sdesc - > desc ;
while ( happened_cyclic ! = schan - > completed_cyclic ) {
2016-07-20 23:13:05 +03:00
dmaengine_desc_get_callback_invoke ( desc , NULL ) ;
2011-10-28 06:22:39 +04:00
schan - > completed_cyclic + + ;
}
}
}
}
/* DMA Tasklet */
static void sirfsoc_dma_tasklet ( unsigned long data )
{
struct sirfsoc_dma * sdma = ( void * ) data ;
sirfsoc_dma_process_completed ( sdma ) ;
}
/* Submit descriptor to hardware */
static dma_cookie_t sirfsoc_dma_tx_submit ( struct dma_async_tx_descriptor * txd )
{
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( txd - > chan ) ;
struct sirfsoc_dma_desc * sdesc ;
unsigned long flags ;
dma_cookie_t cookie ;
sdesc = container_of ( txd , struct sirfsoc_dma_desc , desc ) ;
spin_lock_irqsave ( & schan - > lock , flags ) ;
/* Move descriptor to queue */
list_move_tail ( & sdesc - > node , & schan - > queued ) ;
2012-03-07 02:34:46 +04:00
cookie = dma_cookie_assign ( txd ) ;
2011-10-28 06:22:39 +04:00
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
return cookie ;
}
2014-11-17 16:42:34 +03:00
static int sirfsoc_dma_slave_config ( struct dma_chan * chan ,
struct dma_slave_config * config )
2011-10-28 06:22:39 +04:00
{
2014-11-17 16:42:34 +03:00
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
2011-10-28 06:22:39 +04:00
unsigned long flags ;
if ( ( config - > src_addr_width ! = DMA_SLAVE_BUSWIDTH_4_BYTES ) | |
( config - > dst_addr_width ! = DMA_SLAVE_BUSWIDTH_4_BYTES ) )
return - EINVAL ;
spin_lock_irqsave ( & schan - > lock , flags ) ;
schan - > mode = ( config - > src_maxburst = = 4 ? 1 : 0 ) ;
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
return 0 ;
}
2014-11-17 16:42:34 +03:00
static int sirfsoc_dma_terminate_all ( struct dma_chan * chan )
2011-10-28 06:22:39 +04:00
{
2014-11-17 16:42:34 +03:00
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
2011-10-28 06:22:39 +04:00
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( & schan - > chan ) ;
int cid = schan - > chan . chan_id ;
unsigned long flags ;
2012-12-14 15:06:58 +04:00
spin_lock_irqsave ( & schan - > lock , flags ) ;
2011-10-28 06:22:39 +04:00
2015-05-26 10:32:28 +03:00
switch ( sdma - > type ) {
case SIRFSOC_DMA_VER_A7V1 :
2012-11-01 18:54:43 +04:00
writel_relaxed ( 1 < < cid , sdma - > base + SIRFSOC_DMA_INT_EN_CLR ) ;
2015-07-27 08:50:21 +03:00
writel_relaxed ( 1 < < cid , sdma - > base + SIRFSOC_DMA_CH_INT ) ;
2012-11-01 18:54:43 +04:00
writel_relaxed ( ( 1 < < cid ) | 1 < < ( cid + 16 ) ,
2015-05-26 10:32:28 +03:00
sdma - > base +
SIRFSOC_DMA_CH_LOOP_CTRL_CLR_ATLAS7 ) ;
writel_relaxed ( 1 < < cid , sdma - > base + SIRFSOC_DMA_CH_VALID ) ;
break ;
case SIRFSOC_DMA_VER_A7V2 :
writel_relaxed ( 0 , sdma - > base + SIRFSOC_DMA_INT_EN_ATLAS7 ) ;
2015-07-27 08:50:21 +03:00
writel_relaxed ( SIRFSOC_DMA_INT_ALL_ATLAS7 ,
sdma - > base + SIRFSOC_DMA_INT_ATLAS7 ) ;
2015-05-26 10:32:28 +03:00
writel_relaxed ( 0 , sdma - > base + SIRFSOC_DMA_LOOP_CTRL_ATLAS7 ) ;
writel_relaxed ( 0 , sdma - > base + SIRFSOC_DMA_VALID_ATLAS7 ) ;
break ;
case SIRFSOC_DMA_VER_A6 :
writel_relaxed ( readl_relaxed ( sdma - > base + SIRFSOC_DMA_INT_EN ) &
~ ( 1 < < cid ) , sdma - > base + SIRFSOC_DMA_INT_EN ) ;
writel_relaxed ( readl_relaxed ( sdma - > base +
SIRFSOC_DMA_CH_LOOP_CTRL ) &
~ ( ( 1 < < cid ) | 1 < < ( cid + 16 ) ) ,
sdma - > base + SIRFSOC_DMA_CH_LOOP_CTRL ) ;
writel_relaxed ( 1 < < cid , sdma - > base + SIRFSOC_DMA_CH_VALID ) ;
break ;
default :
break ;
2012-11-01 18:54:43 +04:00
}
2011-10-28 06:22:39 +04:00
list_splice_tail_init ( & schan - > active , & schan - > free ) ;
list_splice_tail_init ( & schan - > queued , & schan - > free ) ;
2012-12-14 15:06:58 +04:00
2011-10-28 06:22:39 +04:00
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
return 0 ;
}
2014-11-17 16:42:34 +03:00
static int sirfsoc_dma_pause_chan ( struct dma_chan * chan )
2012-12-14 14:59:22 +04:00
{
2014-11-17 16:42:34 +03:00
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
2012-12-14 14:59:22 +04:00
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( & schan - > chan ) ;
int cid = schan - > chan . chan_id ;
unsigned long flags ;
spin_lock_irqsave ( & schan - > lock , flags ) ;
2015-05-26 10:32:28 +03:00
switch ( sdma - > type ) {
case SIRFSOC_DMA_VER_A7V1 :
2012-12-14 14:59:22 +04:00
writel_relaxed ( ( 1 < < cid ) | 1 < < ( cid + 16 ) ,
2015-05-26 10:32:28 +03:00
sdma - > base +
SIRFSOC_DMA_CH_LOOP_CTRL_CLR_ATLAS7 ) ;
break ;
case SIRFSOC_DMA_VER_A7V2 :
writel_relaxed ( 0 , sdma - > base + SIRFSOC_DMA_LOOP_CTRL_ATLAS7 ) ;
break ;
case SIRFSOC_DMA_VER_A6 :
writel_relaxed ( readl_relaxed ( sdma - > base +
SIRFSOC_DMA_CH_LOOP_CTRL ) &
~ ( ( 1 < < cid ) | 1 < < ( cid + 16 ) ) ,
sdma - > base + SIRFSOC_DMA_CH_LOOP_CTRL ) ;
break ;
default :
break ;
}
2012-12-14 14:59:22 +04:00
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
return 0 ;
}
2014-11-17 16:42:34 +03:00
static int sirfsoc_dma_resume_chan ( struct dma_chan * chan )
2012-12-14 14:59:22 +04:00
{
2014-11-17 16:42:34 +03:00
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
2012-12-14 14:59:22 +04:00
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( & schan - > chan ) ;
int cid = schan - > chan . chan_id ;
unsigned long flags ;
spin_lock_irqsave ( & schan - > lock , flags ) ;
2015-05-26 10:32:28 +03:00
switch ( sdma - > type ) {
case SIRFSOC_DMA_VER_A7V1 :
2012-12-14 14:59:22 +04:00
writel_relaxed ( ( 1 < < cid ) | 1 < < ( cid + 16 ) ,
2015-05-26 10:32:28 +03:00
sdma - > base + SIRFSOC_DMA_CH_LOOP_CTRL_ATLAS7 ) ;
break ;
case SIRFSOC_DMA_VER_A7V2 :
writel_relaxed ( 0x10001 ,
sdma - > base + SIRFSOC_DMA_LOOP_CTRL_ATLAS7 ) ;
break ;
case SIRFSOC_DMA_VER_A6 :
writel_relaxed ( readl_relaxed ( sdma - > base +
SIRFSOC_DMA_CH_LOOP_CTRL ) |
( ( 1 < < cid ) | 1 < < ( cid + 16 ) ) ,
sdma - > base + SIRFSOC_DMA_CH_LOOP_CTRL ) ;
break ;
default :
break ;
}
2012-12-14 14:59:22 +04:00
2011-10-28 06:22:39 +04:00
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
return 0 ;
}
/* Alloc channel resources */
static int sirfsoc_dma_alloc_chan_resources ( struct dma_chan * chan )
{
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( chan ) ;
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
struct sirfsoc_dma_desc * sdesc ;
unsigned long flags ;
LIST_HEAD ( descs ) ;
int i ;
2013-07-30 13:44:34 +04:00
pm_runtime_get_sync ( sdma - > dma . dev ) ;
2011-10-28 06:22:39 +04:00
/* Alloc descriptors for this channel */
for ( i = 0 ; i < SIRFSOC_DMA_DESCRIPTORS ; i + + ) {
sdesc = kzalloc ( sizeof ( * sdesc ) , GFP_KERNEL ) ;
if ( ! sdesc ) {
dev_notice ( sdma - > dma . dev , " Memory allocation error. "
" Allocated only %u descriptors \n " , i ) ;
break ;
}
dma_async_tx_descriptor_init ( & sdesc - > desc , chan ) ;
sdesc - > desc . flags = DMA_CTRL_ACK ;
sdesc - > desc . tx_submit = sirfsoc_dma_tx_submit ;
list_add_tail ( & sdesc - > node , & descs ) ;
}
/* Return error only if no descriptors were allocated */
if ( i = = 0 )
return - ENOMEM ;
spin_lock_irqsave ( & schan - > lock , flags ) ;
list_splice_tail_init ( & descs , & schan - > free ) ;
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
return i ;
}
/* Free channel resources */
static void sirfsoc_dma_free_chan_resources ( struct dma_chan * chan )
{
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
2013-07-30 13:44:34 +04:00
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( chan ) ;
2011-10-28 06:22:39 +04:00
struct sirfsoc_dma_desc * sdesc , * tmp ;
unsigned long flags ;
LIST_HEAD ( descs ) ;
spin_lock_irqsave ( & schan - > lock , flags ) ;
/* Channel must be idle */
BUG_ON ( ! list_empty ( & schan - > prepared ) ) ;
BUG_ON ( ! list_empty ( & schan - > queued ) ) ;
BUG_ON ( ! list_empty ( & schan - > active ) ) ;
BUG_ON ( ! list_empty ( & schan - > completed ) ) ;
/* Move data */
list_splice_tail_init ( & schan - > free , & descs ) ;
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
/* Free descriptors */
list_for_each_entry_safe ( sdesc , tmp , & descs , node )
kfree ( sdesc ) ;
2013-07-30 13:44:34 +04:00
pm_runtime_put ( sdma - > dma . dev ) ;
2011-10-28 06:22:39 +04:00
}
/* Send pending descriptor to hardware */
static void sirfsoc_dma_issue_pending ( struct dma_chan * chan )
{
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
unsigned long flags ;
spin_lock_irqsave ( & schan - > lock , flags ) ;
if ( list_empty ( & schan - > active ) & & ! list_empty ( & schan - > queued ) )
sirfsoc_dma_execute ( schan ) ;
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
}
/* Check request completion status */
static enum dma_status
sirfsoc_dma_tx_status ( struct dma_chan * chan , dma_cookie_t cookie ,
struct dma_tx_state * txstate )
{
2013-05-14 19:03:20 +04:00
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( chan ) ;
2011-10-28 06:22:39 +04:00
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
unsigned long flags ;
2012-03-07 02:35:27 +04:00
enum dma_status ret ;
2013-05-14 19:03:20 +04:00
struct sirfsoc_dma_desc * sdesc ;
int cid = schan - > chan . chan_id ;
unsigned long dma_pos ;
unsigned long dma_request_bytes ;
unsigned long residue ;
2011-10-28 06:22:39 +04:00
spin_lock_irqsave ( & schan - > lock , flags ) ;
2013-05-14 19:03:20 +04:00
2015-05-26 10:32:28 +03:00
if ( list_empty ( & schan - > active ) ) {
ret = dma_cookie_status ( chan , cookie , txstate ) ;
dma_set_residue ( txstate , 0 ) ;
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
return ret ;
}
sdesc = list_first_entry ( & schan - > active , struct sirfsoc_dma_desc , node ) ;
if ( sdesc - > cyclic )
dma_request_bytes = ( sdesc - > xlen + 1 ) * ( sdesc - > ylen + 1 ) *
( sdesc - > width * SIRFSOC_DMA_WORD_LEN ) ;
else
dma_request_bytes = sdesc - > xlen * SIRFSOC_DMA_WORD_LEN ;
2013-05-14 19:03:20 +04:00
2012-03-07 02:35:27 +04:00
ret = dma_cookie_status ( chan , cookie , txstate ) ;
2015-05-26 10:32:28 +03:00
if ( sdma - > type = = SIRFSOC_DMA_VER_A7V2 )
cid = 0 ;
if ( sdma - > type = = SIRFSOC_DMA_VER_A7V2 ) {
dma_pos = readl_relaxed ( sdma - > base + SIRFSOC_DMA_CUR_DATA_ADDR ) ;
} else {
dma_pos = readl_relaxed (
sdma - > base + cid * 0x10 + SIRFSOC_DMA_CH_ADDR ) < < 2 ;
}
2013-05-14 19:03:20 +04:00
residue = dma_request_bytes - ( dma_pos - sdesc - > addr ) ;
dma_set_residue ( txstate , residue ) ;
2011-10-28 06:22:39 +04:00
spin_unlock_irqrestore ( & schan - > lock , flags ) ;
2012-03-07 02:35:27 +04:00
return ret ;
2011-10-28 06:22:39 +04:00
}
static struct dma_async_tx_descriptor * sirfsoc_dma_prep_interleaved (
struct dma_chan * chan , struct dma_interleaved_template * xt ,
unsigned long flags )
{
struct sirfsoc_dma * sdma = dma_chan_to_sirfsoc_dma ( chan ) ;
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
struct sirfsoc_dma_desc * sdesc = NULL ;
unsigned long iflags ;
int ret ;
2012-09-27 12:35:38 +04:00
if ( ( xt - > dir ! = DMA_MEM_TO_DEV ) & & ( xt - > dir ! = DMA_DEV_TO_MEM ) ) {
2011-10-28 06:22:39 +04:00
ret = - EINVAL ;
goto err_dir ;
}
/* Get free descriptor */
spin_lock_irqsave ( & schan - > lock , iflags ) ;
if ( ! list_empty ( & schan - > free ) ) {
sdesc = list_first_entry ( & schan - > free , struct sirfsoc_dma_desc ,
node ) ;
list_del ( & sdesc - > node ) ;
}
spin_unlock_irqrestore ( & schan - > lock , iflags ) ;
if ( ! sdesc ) {
/* try to free completed descriptors */
sirfsoc_dma_process_completed ( sdma ) ;
ret = 0 ;
goto no_desc ;
}
/* Place descriptor in prepared list */
spin_lock_irqsave ( & schan - > lock , iflags ) ;
/*
* Number of chunks in a frame can only be 1 for prima2
* and ylen ( number of frame - 1 ) must be at least 0
*/
if ( ( xt - > frame_size = = 1 ) & & ( xt - > numf > 0 ) ) {
sdesc - > cyclic = 0 ;
sdesc - > xlen = xt - > sgl [ 0 ] . size / SIRFSOC_DMA_WORD_LEN ;
sdesc - > width = ( xt - > sgl [ 0 ] . size + xt - > sgl [ 0 ] . icg ) /
SIRFSOC_DMA_WORD_LEN ;
sdesc - > ylen = xt - > numf - 1 ;
if ( xt - > dir = = DMA_MEM_TO_DEV ) {
sdesc - > addr = xt - > src_start ;
sdesc - > dir = 1 ;
} else {
sdesc - > addr = xt - > dst_start ;
sdesc - > dir = 0 ;
}
list_add_tail ( & sdesc - > node , & schan - > prepared ) ;
} else {
pr_err ( " sirfsoc DMA Invalid xfer \n " ) ;
ret = - EINVAL ;
goto err_xfer ;
}
spin_unlock_irqrestore ( & schan - > lock , iflags ) ;
return & sdesc - > desc ;
err_xfer :
spin_unlock_irqrestore ( & schan - > lock , iflags ) ;
no_desc :
err_dir :
return ERR_PTR ( ret ) ;
}
static struct dma_async_tx_descriptor *
sirfsoc_dma_prep_cyclic ( struct dma_chan * chan , dma_addr_t addr ,
size_t buf_len , size_t period_len ,
2014-08-01 14:20:10 +04:00
enum dma_transfer_direction direction , unsigned long flags )
2011-10-28 06:22:39 +04:00
{
struct sirfsoc_dma_chan * schan = dma_chan_to_sirfsoc_dma_chan ( chan ) ;
struct sirfsoc_dma_desc * sdesc = NULL ;
unsigned long iflags ;
/*
* we only support cycle transfer with 2 period
* If the X - length is set to 0 , it would be the loop mode .
* The DMA address keeps increasing until reaching the end of a loop
* area whose size is defined by ( DMA_WIDTH x ( Y_LENGTH + 1 ) ) . Then
* the DMA address goes back to the beginning of this area .
* In loop mode , the DMA data region is divided into two parts , BUFA
* and BUFB . DMA controller generates interrupts twice in each loop :
* when the DMA address reaches the end of BUFA or the end of the
* BUFB
*/
if ( buf_len ! = 2 * period_len )
return ERR_PTR ( - EINVAL ) ;
/* Get free descriptor */
spin_lock_irqsave ( & schan - > lock , iflags ) ;
if ( ! list_empty ( & schan - > free ) ) {
sdesc = list_first_entry ( & schan - > free , struct sirfsoc_dma_desc ,
node ) ;
list_del ( & sdesc - > node ) ;
}
spin_unlock_irqrestore ( & schan - > lock , iflags ) ;
if ( ! sdesc )
2013-08-06 14:37:56 +04:00
return NULL ;
2011-10-28 06:22:39 +04:00
/* Place descriptor in prepared list */
spin_lock_irqsave ( & schan - > lock , iflags ) ;
sdesc - > addr = addr ;
sdesc - > cyclic = 1 ;
sdesc - > xlen = 0 ;
sdesc - > ylen = buf_len / SIRFSOC_DMA_WORD_LEN - 1 ;
sdesc - > width = 1 ;
list_add_tail ( & sdesc - > node , & schan - > prepared ) ;
spin_unlock_irqrestore ( & schan - > lock , iflags ) ;
return & sdesc - > desc ;
}
/*
* The DMA controller consists of 16 independent DMA channels .
* Each channel is allocated to a different function
*/
bool sirfsoc_dma_filter_id ( struct dma_chan * chan , void * chan_id )
{
unsigned int ch_nr = ( unsigned int ) chan_id ;
if ( ch_nr = = chan - > chan_id +
chan - > device - > dev_id * SIRFSOC_DMA_CHANNELS )
return true ;
return false ;
}
EXPORT_SYMBOL ( sirfsoc_dma_filter_id ) ;
2013-12-23 16:19:21 +04:00
# define SIRFSOC_DMA_BUSWIDTHS \
( BIT ( DMA_SLAVE_BUSWIDTH_UNDEFINED ) | \
BIT ( DMA_SLAVE_BUSWIDTH_1_BYTE ) | \
BIT ( DMA_SLAVE_BUSWIDTH_2_BYTES ) | \
BIT ( DMA_SLAVE_BUSWIDTH_4_BYTES ) | \
BIT ( DMA_SLAVE_BUSWIDTH_8_BYTES ) )
2014-03-27 11:49:31 +04:00
static struct dma_chan * of_dma_sirfsoc_xlate ( struct of_phandle_args * dma_spec ,
struct of_dma * ofdma )
{
struct sirfsoc_dma * sdma = ofdma - > of_dma_data ;
unsigned int request = dma_spec - > args [ 0 ] ;
2014-04-03 11:29:33 +04:00
if ( request > = SIRFSOC_DMA_CHANNELS )
2014-03-27 11:49:31 +04:00
return NULL ;
return dma_get_slave_channel ( & sdma - > channels [ request ] . chan ) ;
}
2012-11-19 22:22:55 +04:00
static int sirfsoc_dma_probe ( struct platform_device * op )
2011-10-28 06:22:39 +04:00
{
struct device_node * dn = op - > dev . of_node ;
struct device * dev = & op - > dev ;
struct dma_device * dma ;
struct sirfsoc_dma * sdma ;
struct sirfsoc_dma_chan * schan ;
2015-05-26 10:32:28 +03:00
struct sirfsoc_dmadata * data ;
2011-10-28 06:22:39 +04:00
struct resource res ;
ulong regs_start , regs_size ;
u32 id ;
int ret , i ;
sdma = devm_kzalloc ( dev , sizeof ( * sdma ) , GFP_KERNEL ) ;
2016-06-07 20:38:41 +03:00
if ( ! sdma )
2011-10-28 06:22:39 +04:00
return - ENOMEM ;
2016-06-07 20:38:41 +03:00
2015-05-26 10:32:28 +03:00
data = ( struct sirfsoc_dmadata * )
( of_match_device ( op - > dev . driver - > of_match_table ,
& op - > dev ) - > data ) ;
sdma - > exec_desc = data - > exec ;
sdma - > type = data - > type ;
2012-11-01 18:54:43 +04:00
2011-10-28 06:22:39 +04:00
if ( of_property_read_u32 ( dn , " cell-index " , & id ) ) {
dev_err ( dev , " Fail to get DMAC index \n " ) ;
2012-08-04 12:35:30 +04:00
return - ENODEV ;
2011-10-28 06:22:39 +04:00
}
sdma - > irq = irq_of_parse_and_map ( dn , 0 ) ;
2016-09-03 02:17:20 +03:00
if ( ! sdma - > irq ) {
2011-10-28 06:22:39 +04:00
dev_err ( dev , " Error mapping IRQ! \n " ) ;
2012-08-04 12:35:30 +04:00
return - EINVAL ;
2011-10-28 06:22:39 +04:00
}
2013-03-18 12:33:43 +04:00
sdma - > clk = devm_clk_get ( dev , NULL ) ;
if ( IS_ERR ( sdma - > clk ) ) {
dev_err ( dev , " failed to get a clock. \n " ) ;
return PTR_ERR ( sdma - > clk ) ;
}
2011-10-28 06:22:39 +04:00
ret = of_address_to_resource ( dn , 0 , & res ) ;
if ( ret ) {
dev_err ( dev , " Error parsing memory region! \n " ) ;
2012-08-04 12:35:30 +04:00
goto irq_dispose ;
2011-10-28 06:22:39 +04:00
}
regs_start = res . start ;
regs_size = resource_size ( & res ) ;
sdma - > base = devm_ioremap ( dev , regs_start , regs_size ) ;
if ( ! sdma - > base ) {
dev_err ( dev , " Error mapping memory region! \n " ) ;
ret = - ENOMEM ;
goto irq_dispose ;
}
2012-08-04 12:35:30 +04:00
ret = request_irq ( sdma - > irq , & sirfsoc_dma_irq , 0 , DRV_NAME , sdma ) ;
2011-10-28 06:22:39 +04:00
if ( ret ) {
dev_err ( dev , " Error requesting IRQ! \n " ) ;
ret = - EINVAL ;
2012-08-04 12:35:30 +04:00
goto irq_dispose ;
2011-10-28 06:22:39 +04:00
}
dma = & sdma - > dma ;
dma - > dev = dev ;
dma - > device_alloc_chan_resources = sirfsoc_dma_alloc_chan_resources ;
dma - > device_free_chan_resources = sirfsoc_dma_free_chan_resources ;
dma - > device_issue_pending = sirfsoc_dma_issue_pending ;
2014-11-17 16:42:34 +03:00
dma - > device_config = sirfsoc_dma_slave_config ;
dma - > device_pause = sirfsoc_dma_pause_chan ;
dma - > device_resume = sirfsoc_dma_resume_chan ;
dma - > device_terminate_all = sirfsoc_dma_terminate_all ;
2011-10-28 06:22:39 +04:00
dma - > device_tx_status = sirfsoc_dma_tx_status ;
dma - > device_prep_interleaved_dma = sirfsoc_dma_prep_interleaved ;
dma - > device_prep_dma_cyclic = sirfsoc_dma_prep_cyclic ;
2014-11-17 16:42:51 +03:00
dma - > src_addr_widths = SIRFSOC_DMA_BUSWIDTHS ;
dma - > dst_addr_widths = SIRFSOC_DMA_BUSWIDTHS ;
dma - > directions = BIT ( DMA_DEV_TO_MEM ) | BIT ( DMA_MEM_TO_DEV ) ;
2011-10-28 06:22:39 +04:00
INIT_LIST_HEAD ( & dma - > channels ) ;
dma_cap_set ( DMA_SLAVE , dma - > cap_mask ) ;
dma_cap_set ( DMA_CYCLIC , dma - > cap_mask ) ;
dma_cap_set ( DMA_INTERLEAVE , dma - > cap_mask ) ;
dma_cap_set ( DMA_PRIVATE , dma - > cap_mask ) ;
2014-10-16 13:01:02 +04:00
for ( i = 0 ; i < SIRFSOC_DMA_CHANNELS ; i + + ) {
2011-10-28 06:22:39 +04:00
schan = & sdma - > channels [ i ] ;
schan - > chan . device = dma ;
2012-03-07 02:35:47 +04:00
dma_cookie_init ( & schan - > chan ) ;
2011-10-28 06:22:39 +04:00
INIT_LIST_HEAD ( & schan - > free ) ;
INIT_LIST_HEAD ( & schan - > prepared ) ;
INIT_LIST_HEAD ( & schan - > queued ) ;
INIT_LIST_HEAD ( & schan - > active ) ;
INIT_LIST_HEAD ( & schan - > completed ) ;
spin_lock_init ( & schan - > lock ) ;
list_add_tail ( & schan - > chan . device_node , & dma - > channels ) ;
}
tasklet_init ( & sdma - > tasklet , sirfsoc_dma_tasklet , ( unsigned long ) sdma ) ;
/* Register DMA engine */
dev_set_drvdata ( dev , sdma ) ;
2013-07-30 13:44:34 +04:00
2011-10-28 06:22:39 +04:00
ret = dma_async_device_register ( dma ) ;
if ( ret )
goto free_irq ;
2014-03-27 11:49:31 +04:00
/* Device-tree DMA controller registration */
ret = of_dma_controller_register ( dn , of_dma_sirfsoc_xlate , sdma ) ;
if ( ret ) {
dev_err ( dev , " failed to register DMA controller \n " ) ;
goto unreg_dma_dev ;
}
2013-07-30 13:44:34 +04:00
pm_runtime_enable ( & op - > dev ) ;
2011-10-28 06:22:39 +04:00
dev_info ( dev , " initialized SIRFSOC DMAC driver \n " ) ;
return 0 ;
2014-03-27 11:49:31 +04:00
unreg_dma_dev :
dma_async_device_unregister ( dma ) ;
2011-10-28 06:22:39 +04:00
free_irq :
2012-08-04 12:35:30 +04:00
free_irq ( sdma - > irq , sdma ) ;
2011-10-28 06:22:39 +04:00
irq_dispose :
irq_dispose_mapping ( sdma - > irq ) ;
return ret ;
}
2012-12-22 03:09:59 +04:00
static int sirfsoc_dma_remove ( struct platform_device * op )
2011-10-28 06:22:39 +04:00
{
struct device * dev = & op - > dev ;
struct sirfsoc_dma * sdma = dev_get_drvdata ( dev ) ;
2014-03-27 11:49:31 +04:00
of_dma_controller_free ( op - > dev . of_node ) ;
2011-10-28 06:22:39 +04:00
dma_async_device_unregister ( & sdma - > dma ) ;
2012-08-04 12:35:30 +04:00
free_irq ( sdma - > irq , sdma ) ;
2016-07-05 09:26:10 +03:00
tasklet_kill ( & sdma - > tasklet ) ;
2011-10-28 06:22:39 +04:00
irq_dispose_mapping ( sdma - > irq ) ;
2013-07-30 13:44:34 +04:00
pm_runtime_disable ( & op - > dev ) ;
if ( ! pm_runtime_status_suspended ( & op - > dev ) )
sirfsoc_dma_runtime_suspend ( & op - > dev ) ;
return 0 ;
}
2016-03-02 18:58:58 +03:00
static int __maybe_unused sirfsoc_dma_runtime_suspend ( struct device * dev )
2013-07-30 13:44:34 +04:00
{
struct sirfsoc_dma * sdma = dev_get_drvdata ( dev ) ;
clk_disable_unprepare ( sdma - > clk ) ;
return 0 ;
}
2016-03-02 18:58:58 +03:00
static int __maybe_unused sirfsoc_dma_runtime_resume ( struct device * dev )
2013-07-30 13:44:34 +04:00
{
struct sirfsoc_dma * sdma = dev_get_drvdata ( dev ) ;
int ret ;
ret = clk_prepare_enable ( sdma - > clk ) ;
if ( ret < 0 ) {
dev_err ( dev , " clk_enable failed: %d \n " , ret ) ;
return ret ;
}
return 0 ;
}
2016-03-02 18:58:58 +03:00
static int __maybe_unused sirfsoc_dma_pm_suspend ( struct device * dev )
2013-07-30 13:44:34 +04:00
{
struct sirfsoc_dma * sdma = dev_get_drvdata ( dev ) ;
struct sirfsoc_dma_regs * save = & sdma - > regs_save ;
struct sirfsoc_dma_chan * schan ;
int ch ;
int ret ;
2015-05-26 10:32:28 +03:00
int count ;
u32 int_offset ;
2013-07-30 13:44:34 +04:00
/*
* if we were runtime - suspended before , resume to enable clock
* before accessing register
*/
if ( pm_runtime_status_suspended ( dev ) ) {
ret = sirfsoc_dma_runtime_resume ( dev ) ;
if ( ret < 0 )
return ret ;
}
2015-05-26 10:32:28 +03:00
if ( sdma - > type = = SIRFSOC_DMA_VER_A7V2 ) {
count = 1 ;
int_offset = SIRFSOC_DMA_INT_EN_ATLAS7 ;
} else {
count = SIRFSOC_DMA_CHANNELS ;
int_offset = SIRFSOC_DMA_INT_EN ;
}
2013-07-30 13:44:34 +04:00
/*
* DMA controller will lose all registers while suspending
* so we need to save registers for active channels
*/
2015-05-26 10:32:28 +03:00
for ( ch = 0 ; ch < count ; ch + + ) {
2013-07-30 13:44:34 +04:00
schan = & sdma - > channels [ ch ] ;
if ( list_empty ( & schan - > active ) )
continue ;
save - > ctrl [ ch ] = readl_relaxed ( sdma - > base +
ch * 0x10 + SIRFSOC_DMA_CH_CTRL ) ;
}
2015-05-26 10:32:28 +03:00
save - > interrupt_en = readl_relaxed ( sdma - > base + int_offset ) ;
2013-07-30 13:44:34 +04:00
/* Disable clock */
sirfsoc_dma_runtime_suspend ( dev ) ;
return 0 ;
}
2016-03-02 18:58:58 +03:00
static int __maybe_unused sirfsoc_dma_pm_resume ( struct device * dev )
2013-07-30 13:44:34 +04:00
{
struct sirfsoc_dma * sdma = dev_get_drvdata ( dev ) ;
struct sirfsoc_dma_regs * save = & sdma - > regs_save ;
struct sirfsoc_dma_desc * sdesc ;
struct sirfsoc_dma_chan * schan ;
int ch ;
int ret ;
2015-05-26 10:32:28 +03:00
int count ;
u32 int_offset ;
u32 width_offset ;
2013-07-30 13:44:34 +04:00
/* Enable clock before accessing register */
ret = sirfsoc_dma_runtime_resume ( dev ) ;
if ( ret < 0 )
return ret ;
2015-05-26 10:32:28 +03:00
if ( sdma - > type = = SIRFSOC_DMA_VER_A7V2 ) {
count = 1 ;
int_offset = SIRFSOC_DMA_INT_EN_ATLAS7 ;
width_offset = SIRFSOC_DMA_WIDTH_ATLAS7 ;
} else {
count = SIRFSOC_DMA_CHANNELS ;
int_offset = SIRFSOC_DMA_INT_EN ;
width_offset = SIRFSOC_DMA_WIDTH_0 ;
}
writel_relaxed ( save - > interrupt_en , sdma - > base + int_offset ) ;
for ( ch = 0 ; ch < count ; ch + + ) {
2013-07-30 13:44:34 +04:00
schan = & sdma - > channels [ ch ] ;
if ( list_empty ( & schan - > active ) )
continue ;
sdesc = list_first_entry ( & schan - > active ,
struct sirfsoc_dma_desc ,
node ) ;
writel_relaxed ( sdesc - > width ,
2015-05-26 10:32:28 +03:00
sdma - > base + width_offset + ch * 4 ) ;
2013-07-30 13:44:34 +04:00
writel_relaxed ( sdesc - > xlen ,
sdma - > base + ch * 0x10 + SIRFSOC_DMA_CH_XLEN ) ;
writel_relaxed ( sdesc - > ylen ,
sdma - > base + ch * 0x10 + SIRFSOC_DMA_CH_YLEN ) ;
writel_relaxed ( save - > ctrl [ ch ] ,
sdma - > base + ch * 0x10 + SIRFSOC_DMA_CH_CTRL ) ;
2015-05-26 10:32:28 +03:00
if ( sdma - > type = = SIRFSOC_DMA_VER_A7V2 ) {
writel_relaxed ( sdesc - > addr ,
sdma - > base + SIRFSOC_DMA_CH_ADDR ) ;
} else {
writel_relaxed ( sdesc - > addr > > 2 ,
sdma - > base + ch * 0x10 + SIRFSOC_DMA_CH_ADDR ) ;
}
2013-07-30 13:44:34 +04:00
}
/* if we were runtime-suspended before, suspend again */
if ( pm_runtime_status_suspended ( dev ) )
sirfsoc_dma_runtime_suspend ( dev ) ;
2011-10-28 06:22:39 +04:00
return 0 ;
}
2013-07-30 13:44:34 +04:00
static const struct dev_pm_ops sirfsoc_dma_pm_ops = {
SET_RUNTIME_PM_OPS ( sirfsoc_dma_runtime_suspend , sirfsoc_dma_runtime_resume , NULL )
SET_SYSTEM_SLEEP_PM_OPS ( sirfsoc_dma_pm_suspend , sirfsoc_dma_pm_resume )
} ;
2016-06-07 18:54:48 +03:00
static struct sirfsoc_dmadata sirfsoc_dmadata_a6 = {
2015-05-26 10:32:28 +03:00
. exec = sirfsoc_dma_execute_hw_a6 ,
. type = SIRFSOC_DMA_VER_A6 ,
} ;
2016-06-07 18:54:48 +03:00
static struct sirfsoc_dmadata sirfsoc_dmadata_a7v1 = {
2015-05-26 10:32:28 +03:00
. exec = sirfsoc_dma_execute_hw_a7v1 ,
. type = SIRFSOC_DMA_VER_A7V1 ,
} ;
2016-06-07 18:54:48 +03:00
static struct sirfsoc_dmadata sirfsoc_dmadata_a7v2 = {
2015-05-26 10:32:28 +03:00
. exec = sirfsoc_dma_execute_hw_a7v2 ,
. type = SIRFSOC_DMA_VER_A7V2 ,
} ;
2015-03-16 22:17:14 +03:00
static const struct of_device_id sirfsoc_dma_match [ ] = {
2015-05-26 10:32:28 +03:00
{ . compatible = " sirf,prima2-dmac " , . data = & sirfsoc_dmadata_a6 , } ,
{ . compatible = " sirf,atlas7-dmac " , . data = & sirfsoc_dmadata_a7v1 , } ,
{ . compatible = " sirf,atlas7-dmac-v2 " , . data = & sirfsoc_dmadata_a7v2 , } ,
2011-10-28 06:22:39 +04:00
{ } ,
} ;
2015-09-16 23:58:53 +03:00
MODULE_DEVICE_TABLE ( of , sirfsoc_dma_match ) ;
2011-10-28 06:22:39 +04:00
static struct platform_driver sirfsoc_dma_driver = {
. probe = sirfsoc_dma_probe ,
2012-11-19 22:20:04 +04:00
. remove = sirfsoc_dma_remove ,
2011-10-28 06:22:39 +04:00
. driver = {
. name = DRV_NAME ,
2013-07-30 13:44:34 +04:00
. pm = & sirfsoc_dma_pm_ops ,
2011-10-28 06:22:39 +04:00
. of_match_table = sirfsoc_dma_match ,
} ,
} ;
2013-04-11 10:09:28 +04:00
static __init int sirfsoc_dma_init ( void )
{
return platform_driver_register ( & sirfsoc_dma_driver ) ;
}
static void __exit sirfsoc_dma_exit ( void )
{
platform_driver_unregister ( & sirfsoc_dma_driver ) ;
}
subsys_initcall ( sirfsoc_dma_init ) ;
module_exit ( sirfsoc_dma_exit ) ;
2011-10-28 06:22:39 +04:00
2015-05-26 10:32:28 +03:00
MODULE_AUTHOR ( " Rongjun Ying <rongjun.ying@csr.com> " ) ;
MODULE_AUTHOR ( " Barry Song <baohua.song@csr.com> " ) ;
2011-10-28 06:22:39 +04:00
MODULE_DESCRIPTION ( " SIRFSOC DMA control driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;