2019-05-19 16:51:52 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2012-06-15 07:04:08 +04:00
/*
* Driver For Marvell Two - channel DMA Engine
*
* Copyright : Marvell International Ltd .
*/
2013-01-21 14:09:00 +04:00
# include <linux/err.h>
2012-06-15 07:04:08 +04:00
# include <linux/module.h>
# include <linux/init.h>
# include <linux/types.h>
# include <linux/interrupt.h>
# include <linux/dma-mapping.h>
# include <linux/slab.h>
# include <linux/dmaengine.h>
# include <linux/platform_device.h>
# include <linux/device.h>
2022-09-30 10:25:25 +03:00
# include <linux/genalloc.h>
2012-09-03 07:03:46 +04:00
# include <linux/of_device.h>
2014-01-20 16:39:01 +04:00
# include <linux/of_dma.h>
2012-06-15 07:04:08 +04:00
# include "dmaengine.h"
/*
* Two - Channel DMA registers
*/
# define TDBCR 0x00 /* Byte Count */
# define TDSAR 0x10 /* Src Addr */
# define TDDAR 0x20 /* Dst Addr */
# define TDNDPR 0x30 /* Next Desc */
# define TDCR 0x40 /* Control */
# define TDCP 0x60 /* Priority*/
# define TDCDPR 0x70 /* Current Desc */
# define TDIMR 0x80 /* Int Mask */
# define TDISR 0xa0 /* Int Status */
/* Two-Channel DMA Control Register */
# define TDCR_SSZ_8_BITS (0x0 << 22) /* Sample Size */
# define TDCR_SSZ_12_BITS (0x1 << 22)
# define TDCR_SSZ_16_BITS (0x2 << 22)
# define TDCR_SSZ_20_BITS (0x3 << 22)
# define TDCR_SSZ_24_BITS (0x4 << 22)
# define TDCR_SSZ_32_BITS (0x5 << 22)
# define TDCR_SSZ_SHIFT (0x1 << 22)
# define TDCR_SSZ_MASK (0x7 << 22)
# define TDCR_SSPMOD (0x1 << 21) /* SSP MOD */
# define TDCR_ABR (0x1 << 20) /* Channel Abort */
# define TDCR_CDE (0x1 << 17) /* Close Desc Enable */
# define TDCR_PACKMOD (0x1 << 16) /* Pack Mode (ADMA Only) */
# define TDCR_CHANACT (0x1 << 14) /* Channel Active */
# define TDCR_FETCHND (0x1 << 13) /* Fetch Next Desc */
# define TDCR_CHANEN (0x1 << 12) /* Channel Enable */
# define TDCR_INTMODE (0x1 << 10) /* Interrupt Mode */
# define TDCR_CHAINMOD (0x1 << 9) /* Chain Mode */
# define TDCR_BURSTSZ_MSK (0x7 << 6) /* Burst Size */
# define TDCR_BURSTSZ_4B (0x0 << 6)
# define TDCR_BURSTSZ_8B (0x1 << 6)
# define TDCR_BURSTSZ_16B (0x3 << 6)
# define TDCR_BURSTSZ_32B (0x6 << 6)
# define TDCR_BURSTSZ_64B (0x7 << 6)
2013-10-11 05:07:01 +04:00
# define TDCR_BURSTSZ_SQU_1B (0x5 << 6)
# define TDCR_BURSTSZ_SQU_2B (0x6 << 6)
# define TDCR_BURSTSZ_SQU_4B (0x0 << 6)
# define TDCR_BURSTSZ_SQU_8B (0x1 << 6)
# define TDCR_BURSTSZ_SQU_16B (0x3 << 6)
2012-06-15 07:04:08 +04:00
# define TDCR_BURSTSZ_SQU_32B (0x7 << 6)
# define TDCR_BURSTSZ_128B (0x5 << 6)
# define TDCR_DSTDIR_MSK (0x3 << 4) /* Dst Direction */
# define TDCR_DSTDIR_ADDR_HOLD (0x2 << 4) /* Dst Addr Hold */
# define TDCR_DSTDIR_ADDR_INC (0x0 << 4) /* Dst Addr Increment */
# define TDCR_SRCDIR_MSK (0x3 << 2) /* Src Direction */
# define TDCR_SRCDIR_ADDR_HOLD (0x2 << 2) /* Src Addr Hold */
# define TDCR_SRCDIR_ADDR_INC (0x0 << 2) /* Src Addr Increment */
# define TDCR_DSTDESCCONT (0x1 << 1)
# define TDCR_SRCDESTCONT (0x1 << 0)
/* Two-Channel DMA Int Mask Register */
# define TDIMR_COMP (0x1 << 0)
/* Two-Channel DMA Int Status Register */
# define TDISR_COMP (0x1 << 0)
/*
* Two - Channel DMA Descriptor Struct
* NOTE : desc ' s buf must be aligned to 16 bytes .
*/
struct mmp_tdma_desc {
u32 byte_cnt ;
u32 src_addr ;
u32 dst_addr ;
u32 nxt_desc ;
} ;
enum mmp_tdma_type {
MMP_AUD_TDMA = 0 ,
PXA910_SQU ,
} ;
# define TDMA_MAX_XFER_BYTES SZ_64K
struct mmp_tdma_chan {
struct device * dev ;
struct dma_chan chan ;
struct dma_async_tx_descriptor desc ;
struct tasklet_struct tasklet ;
struct mmp_tdma_desc * desc_arr ;
2015-03-03 04:16:08 +03:00
dma_addr_t desc_arr_phys ;
2012-06-15 07:04:08 +04:00
int desc_num ;
enum dma_transfer_direction dir ;
dma_addr_t dev_addr ;
u32 burst_sz ;
enum dma_slave_buswidth buswidth ;
enum dma_status status ;
2018-07-19 19:52:26 +03:00
struct dma_slave_config slave_config ;
2012-06-15 07:04:08 +04:00
int idx ;
enum mmp_tdma_type type ;
int irq ;
2013-11-28 13:29:39 +04:00
void __iomem * reg_base ;
2012-06-15 07:04:08 +04:00
size_t buf_len ;
size_t period_len ;
size_t pos ;
2013-12-13 12:14:31 +04:00
struct gen_pool * pool ;
2012-06-15 07:04:08 +04:00
} ;
# define TDMA_CHANNEL_NUM 2
struct mmp_tdma_device {
struct device * dev ;
void __iomem * base ;
struct dma_device device ;
struct mmp_tdma_chan * tdmac [ TDMA_CHANNEL_NUM ] ;
} ;
# define to_mmp_tdma_chan(dchan) container_of(dchan, struct mmp_tdma_chan, chan)
2018-07-19 19:52:26 +03:00
static int mmp_tdma_config_write ( struct dma_chan * chan ,
enum dma_transfer_direction dir ,
struct dma_slave_config * dmaengine_cfg ) ;
2012-06-15 07:04:08 +04:00
static void mmp_tdma_chan_set_desc ( struct mmp_tdma_chan * tdmac , dma_addr_t phys )
{
writel ( phys , tdmac - > reg_base + TDNDPR ) ;
writel ( readl ( tdmac - > reg_base + TDCR ) | TDCR_FETCHND ,
tdmac - > reg_base + TDCR ) ;
}
2014-09-10 12:40:48 +04:00
static void mmp_tdma_enable_irq ( struct mmp_tdma_chan * tdmac , bool enable )
{
if ( enable )
writel ( TDIMR_COMP , tdmac - > reg_base + TDIMR ) ;
else
writel ( 0 , tdmac - > reg_base + TDIMR ) ;
}
2012-06-15 07:04:08 +04:00
static void mmp_tdma_enable_chan ( struct mmp_tdma_chan * tdmac )
{
/* enable dma chan */
writel ( readl ( tdmac - > reg_base + TDCR ) | TDCR_CHANEN ,
tdmac - > reg_base + TDCR ) ;
tdmac - > status = DMA_IN_PROGRESS ;
}
2014-11-17 16:42:22 +03:00
static int mmp_tdma_disable_chan ( struct dma_chan * chan )
2012-06-15 07:04:08 +04:00
{
2014-11-17 16:42:22 +03:00
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
2015-03-03 04:16:08 +03:00
u32 tdcr ;
2014-11-17 16:42:22 +03:00
2015-03-03 04:16:08 +03:00
tdcr = readl ( tdmac - > reg_base + TDCR ) ;
tdcr | = TDCR_ABR ;
tdcr & = ~ TDCR_CHANEN ;
writel ( tdcr , tdmac - > reg_base + TDCR ) ;
2013-06-15 08:51:48 +04:00
2013-10-16 19:20:36 +04:00
tdmac - > status = DMA_COMPLETE ;
2014-11-17 16:42:22 +03:00
return 0 ;
2012-06-15 07:04:08 +04:00
}
2014-11-17 16:42:22 +03:00
static int mmp_tdma_resume_chan ( struct dma_chan * chan )
2012-06-15 07:04:08 +04:00
{
2014-11-17 16:42:22 +03:00
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
2012-06-15 07:04:08 +04:00
writel ( readl ( tdmac - > reg_base + TDCR ) | TDCR_CHANEN ,
tdmac - > reg_base + TDCR ) ;
tdmac - > status = DMA_IN_PROGRESS ;
2014-11-17 16:42:22 +03:00
return 0 ;
2012-06-15 07:04:08 +04:00
}
2014-11-17 16:42:22 +03:00
static int mmp_tdma_pause_chan ( struct dma_chan * chan )
2012-06-15 07:04:08 +04:00
{
2014-11-17 16:42:22 +03:00
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
2012-06-15 07:04:08 +04:00
writel ( readl ( tdmac - > reg_base + TDCR ) & ~ TDCR_CHANEN ,
tdmac - > reg_base + TDCR ) ;
tdmac - > status = DMA_PAUSED ;
2014-11-17 16:42:22 +03:00
return 0 ;
2012-06-15 07:04:08 +04:00
}
2014-11-17 16:42:22 +03:00
static int mmp_tdma_config_chan ( struct dma_chan * chan )
2012-06-15 07:04:08 +04:00
{
2014-11-17 16:42:22 +03:00
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
2013-11-29 09:22:52 +04:00
unsigned int tdcr = 0 ;
2012-06-15 07:04:08 +04:00
2014-11-17 16:42:22 +03:00
mmp_tdma_disable_chan ( chan ) ;
2012-06-15 07:04:08 +04:00
if ( tdmac - > dir = = DMA_MEM_TO_DEV )
tdcr = TDCR_DSTDIR_ADDR_HOLD | TDCR_SRCDIR_ADDR_INC ;
else if ( tdmac - > dir = = DMA_DEV_TO_MEM )
tdcr = TDCR_SRCDIR_ADDR_HOLD | TDCR_DSTDIR_ADDR_INC ;
if ( tdmac - > type = = MMP_AUD_TDMA ) {
tdcr | = TDCR_PACKMOD ;
switch ( tdmac - > burst_sz ) {
case 4 :
tdcr | = TDCR_BURSTSZ_4B ;
break ;
case 8 :
tdcr | = TDCR_BURSTSZ_8B ;
break ;
case 16 :
tdcr | = TDCR_BURSTSZ_16B ;
break ;
case 32 :
tdcr | = TDCR_BURSTSZ_32B ;
break ;
case 64 :
tdcr | = TDCR_BURSTSZ_64B ;
break ;
case 128 :
tdcr | = TDCR_BURSTSZ_128B ;
break ;
default :
2020-04-19 19:49:07 +03:00
dev_err ( tdmac - > dev , " unknown burst size. \n " ) ;
2012-06-15 07:04:08 +04:00
return - EINVAL ;
}
switch ( tdmac - > buswidth ) {
case DMA_SLAVE_BUSWIDTH_1_BYTE :
tdcr | = TDCR_SSZ_8_BITS ;
break ;
case DMA_SLAVE_BUSWIDTH_2_BYTES :
tdcr | = TDCR_SSZ_16_BITS ;
break ;
case DMA_SLAVE_BUSWIDTH_4_BYTES :
tdcr | = TDCR_SSZ_32_BITS ;
break ;
default :
2020-04-19 19:49:07 +03:00
dev_err ( tdmac - > dev , " unknown bus size. \n " ) ;
2012-06-15 07:04:08 +04:00
return - EINVAL ;
}
} else if ( tdmac - > type = = PXA910_SQU ) {
tdcr | = TDCR_SSPMOD ;
2013-10-11 05:07:01 +04:00
switch ( tdmac - > burst_sz ) {
case 1 :
tdcr | = TDCR_BURSTSZ_SQU_1B ;
break ;
case 2 :
tdcr | = TDCR_BURSTSZ_SQU_2B ;
break ;
case 4 :
tdcr | = TDCR_BURSTSZ_SQU_4B ;
break ;
case 8 :
tdcr | = TDCR_BURSTSZ_SQU_8B ;
break ;
case 16 :
tdcr | = TDCR_BURSTSZ_SQU_16B ;
break ;
case 32 :
tdcr | = TDCR_BURSTSZ_SQU_32B ;
break ;
default :
2020-04-19 19:49:07 +03:00
dev_err ( tdmac - > dev , " unknown burst size. \n " ) ;
2013-10-11 05:07:01 +04:00
return - EINVAL ;
}
2012-06-15 07:04:08 +04:00
}
writel ( tdcr , tdmac - > reg_base + TDCR ) ;
return 0 ;
}
static int mmp_tdma_clear_chan_irq ( struct mmp_tdma_chan * tdmac )
{
u32 reg = readl ( tdmac - > reg_base + TDISR ) ;
if ( reg & TDISR_COMP ) {
/* clear irq */
reg & = ~ TDISR_COMP ;
writel ( reg , tdmac - > reg_base + TDISR ) ;
return 0 ;
}
return - EAGAIN ;
}
2015-03-03 04:16:08 +03:00
static size_t mmp_tdma_get_pos ( struct mmp_tdma_chan * tdmac )
{
size_t reg ;
if ( tdmac - > idx = = 0 ) {
reg = __raw_readl ( tdmac - > reg_base + TDSAR ) ;
reg - = tdmac - > desc_arr [ 0 ] . src_addr ;
} else if ( tdmac - > idx = = 1 ) {
reg = __raw_readl ( tdmac - > reg_base + TDDAR ) ;
reg - = tdmac - > desc_arr [ 0 ] . dst_addr ;
} else
return - EINVAL ;
return reg ;
}
2012-06-15 07:04:08 +04:00
static irqreturn_t mmp_tdma_chan_handler ( int irq , void * dev_id )
{
struct mmp_tdma_chan * tdmac = dev_id ;
if ( mmp_tdma_clear_chan_irq ( tdmac ) = = 0 ) {
tasklet_schedule ( & tdmac - > tasklet ) ;
return IRQ_HANDLED ;
} else
return IRQ_NONE ;
}
static irqreturn_t mmp_tdma_int_handler ( int irq , void * dev_id )
{
struct mmp_tdma_device * tdev = dev_id ;
int i , ret ;
int irq_num = 0 ;
for ( i = 0 ; i < TDMA_CHANNEL_NUM ; i + + ) {
struct mmp_tdma_chan * tdmac = tdev - > tdmac [ i ] ;
ret = mmp_tdma_chan_handler ( irq , tdmac ) ;
if ( ret = = IRQ_HANDLED )
irq_num + + ;
}
if ( irq_num )
return IRQ_HANDLED ;
else
return IRQ_NONE ;
}
2020-08-31 13:35:21 +03:00
static void dma_do_tasklet ( struct tasklet_struct * t )
2012-06-15 07:04:08 +04:00
{
2020-08-31 13:35:21 +03:00
struct mmp_tdma_chan * tdmac = from_tasklet ( tdmac , t , tasklet ) ;
2012-06-15 07:04:08 +04:00
2016-07-20 23:12:01 +03:00
dmaengine_desc_get_callback_invoke ( & tdmac - > desc , NULL ) ;
2012-06-15 07:04:08 +04:00
}
static void mmp_tdma_free_descriptor ( struct mmp_tdma_chan * tdmac )
{
struct gen_pool * gpool ;
int size = tdmac - > desc_num * sizeof ( struct mmp_tdma_desc ) ;
2013-12-13 12:14:31 +04:00
gpool = tdmac - > pool ;
2015-03-03 04:16:08 +03:00
if ( gpool & & tdmac - > desc_arr )
2012-06-15 07:04:08 +04:00
gen_pool_free ( gpool , ( unsigned long ) tdmac - > desc_arr ,
size ) ;
tdmac - > desc_arr = NULL ;
2020-04-19 19:49:09 +03:00
if ( tdmac - > status = = DMA_ERROR )
tdmac - > status = DMA_COMPLETE ;
2012-06-15 07:04:08 +04:00
return ;
}
static dma_cookie_t mmp_tdma_tx_submit ( struct dma_async_tx_descriptor * tx )
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( tx - > chan ) ;
mmp_tdma_chan_set_desc ( tdmac , tdmac - > desc_arr_phys ) ;
return 0 ;
}
static int mmp_tdma_alloc_chan_resources ( struct dma_chan * chan )
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
int ret ;
dma_async_tx_descriptor_init ( & tdmac - > desc , chan ) ;
tdmac - > desc . tx_submit = mmp_tdma_tx_submit ;
if ( tdmac - > irq ) {
ret = devm_request_irq ( tdmac - > dev , tdmac - > irq ,
2013-10-13 09:10:51 +04:00
mmp_tdma_chan_handler , 0 , " tdma " , tdmac ) ;
2012-06-15 07:04:08 +04:00
if ( ret )
return ret ;
}
return 1 ;
}
static void mmp_tdma_free_chan_resources ( struct dma_chan * chan )
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
if ( tdmac - > irq )
devm_free_irq ( tdmac - > dev , tdmac - > irq , tdmac ) ;
mmp_tdma_free_descriptor ( tdmac ) ;
return ;
}
2016-07-04 13:09:48 +03:00
static struct mmp_tdma_desc * mmp_tdma_alloc_descriptor ( struct mmp_tdma_chan * tdmac )
2012-06-15 07:04:08 +04:00
{
struct gen_pool * gpool ;
int size = tdmac - > desc_num * sizeof ( struct mmp_tdma_desc ) ;
2013-12-13 12:14:31 +04:00
gpool = tdmac - > pool ;
2012-06-15 07:04:08 +04:00
if ( ! gpool )
return NULL ;
2013-11-13 03:09:55 +04:00
tdmac - > desc_arr = gen_pool_dma_alloc ( gpool , size , & tdmac - > desc_arr_phys ) ;
2012-06-15 07:04:08 +04:00
return tdmac - > desc_arr ;
}
static struct dma_async_tx_descriptor * mmp_tdma_prep_dma_cyclic (
struct dma_chan * chan , dma_addr_t dma_addr , size_t buf_len ,
size_t period_len , enum dma_transfer_direction direction ,
2014-08-01 14:20:10 +04:00
unsigned long flags )
2012-06-15 07:04:08 +04:00
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
struct mmp_tdma_desc * desc ;
int num_periods = buf_len / period_len ;
int i = 0 , buf = 0 ;
2020-04-25 00:50:20 +03:00
if ( ! is_slave_direction ( direction ) ) {
dev_err ( tdmac - > dev , " unsupported transfer direction \n " ) ;
2012-06-15 07:04:08 +04:00
return NULL ;
2020-04-25 00:50:20 +03:00
}
2020-04-19 19:49:10 +03:00
if ( tdmac - > status ! = DMA_COMPLETE ) {
dev_err ( tdmac - > dev , " controller busy " ) ;
2012-06-15 07:04:08 +04:00
return NULL ;
2020-04-19 19:49:10 +03:00
}
2012-06-15 07:04:08 +04:00
if ( period_len > TDMA_MAX_XFER_BYTES ) {
dev_err ( tdmac - > dev ,
2016-09-14 13:23:08 +03:00
" maximum period size exceeded: %zu > %d \n " ,
2012-06-15 07:04:08 +04:00
period_len , TDMA_MAX_XFER_BYTES ) ;
goto err_out ;
}
tdmac - > status = DMA_IN_PROGRESS ;
tdmac - > desc_num = num_periods ;
desc = mmp_tdma_alloc_descriptor ( tdmac ) ;
if ( ! desc )
goto err_out ;
2020-04-19 19:49:06 +03:00
if ( mmp_tdma_config_write ( chan , direction , & tdmac - > slave_config ) )
goto err_out ;
2018-07-19 19:52:26 +03:00
2012-06-15 07:04:08 +04:00
while ( buf < buf_len ) {
desc = & tdmac - > desc_arr [ i ] ;
if ( i + 1 = = num_periods )
desc - > nxt_desc = tdmac - > desc_arr_phys ;
else
desc - > nxt_desc = tdmac - > desc_arr_phys +
sizeof ( * desc ) * ( i + 1 ) ;
if ( direction = = DMA_MEM_TO_DEV ) {
desc - > src_addr = dma_addr ;
desc - > dst_addr = tdmac - > dev_addr ;
} else {
desc - > src_addr = tdmac - > dev_addr ;
desc - > dst_addr = dma_addr ;
}
desc - > byte_cnt = period_len ;
dma_addr + = period_len ;
buf + = period_len ;
i + + ;
}
2014-09-10 12:40:48 +04:00
/* enable interrupt */
if ( flags & DMA_PREP_INTERRUPT )
mmp_tdma_enable_irq ( tdmac , true ) ;
2012-06-15 07:04:08 +04:00
tdmac - > buf_len = buf_len ;
tdmac - > period_len = period_len ;
tdmac - > pos = 0 ;
return & tdmac - > desc ;
err_out :
tdmac - > status = DMA_ERROR ;
return NULL ;
}
2014-11-17 16:42:22 +03:00
static int mmp_tdma_terminate_all ( struct dma_chan * chan )
2012-06-15 07:04:08 +04:00
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
2014-11-17 16:42:22 +03:00
mmp_tdma_disable_chan ( chan ) ;
/* disable interrupt */
mmp_tdma_enable_irq ( tdmac , false ) ;
2015-01-13 16:31:46 +03:00
return 0 ;
2014-11-17 16:42:22 +03:00
}
static int mmp_tdma_config ( struct dma_chan * chan ,
struct dma_slave_config * dmaengine_cfg )
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
2018-07-19 19:52:26 +03:00
memcpy ( & tdmac - > slave_config , dmaengine_cfg , sizeof ( * dmaengine_cfg ) ) ;
return 0 ;
}
static int mmp_tdma_config_write ( struct dma_chan * chan ,
enum dma_transfer_direction dir ,
struct dma_slave_config * dmaengine_cfg )
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
if ( dir = = DMA_DEV_TO_MEM ) {
2014-11-17 16:42:22 +03:00
tdmac - > dev_addr = dmaengine_cfg - > src_addr ;
tdmac - > burst_sz = dmaengine_cfg - > src_maxburst ;
tdmac - > buswidth = dmaengine_cfg - > src_addr_width ;
} else {
tdmac - > dev_addr = dmaengine_cfg - > dst_addr ;
tdmac - > burst_sz = dmaengine_cfg - > dst_maxburst ;
tdmac - > buswidth = dmaengine_cfg - > dst_addr_width ;
2012-06-15 07:04:08 +04:00
}
2018-07-19 19:52:26 +03:00
tdmac - > dir = dir ;
2012-06-15 07:04:08 +04:00
2014-11-17 16:42:22 +03:00
return mmp_tdma_config_chan ( chan ) ;
2012-06-15 07:04:08 +04:00
}
static enum dma_status mmp_tdma_tx_status ( struct dma_chan * chan ,
dma_cookie_t cookie , struct dma_tx_state * txstate )
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
2015-03-03 04:16:08 +03:00
tdmac - > pos = mmp_tdma_get_pos ( tdmac ) ;
2013-05-27 16:14:41 +04:00
dma_set_tx_state ( txstate , chan - > completed_cookie , chan - > cookie ,
tdmac - > buf_len - tdmac - > pos ) ;
2012-06-15 07:04:08 +04:00
return tdmac - > status ;
}
static void mmp_tdma_issue_pending ( struct dma_chan * chan )
{
struct mmp_tdma_chan * tdmac = to_mmp_tdma_chan ( chan ) ;
mmp_tdma_enable_chan ( tdmac ) ;
}
2012-12-22 03:09:59 +04:00
static int mmp_tdma_remove ( struct platform_device * pdev )
2012-06-15 07:04:08 +04:00
{
2019-11-15 11:31:00 +03:00
if ( pdev - > dev . of_node )
of_dma_controller_free ( pdev - > dev . of_node ) ;
2012-06-15 07:04:08 +04:00
return 0 ;
}
2012-11-19 22:22:55 +04:00
static int mmp_tdma_chan_init ( struct mmp_tdma_device * tdev ,
2013-12-13 12:14:31 +04:00
int idx , int irq ,
int type , struct gen_pool * pool )
2012-06-15 07:04:08 +04:00
{
struct mmp_tdma_chan * tdmac ;
if ( idx > = TDMA_CHANNEL_NUM ) {
dev_err ( tdev - > dev , " too many channels for device! \n " ) ;
return - EINVAL ;
}
/* alloc channel */
tdmac = devm_kzalloc ( tdev - > dev , sizeof ( * tdmac ) , GFP_KERNEL ) ;
2016-06-07 20:38:41 +03:00
if ( ! tdmac )
2012-06-15 07:04:08 +04:00
return - ENOMEM ;
2016-06-07 20:38:41 +03:00
2012-06-15 07:04:08 +04:00
if ( irq )
2012-09-03 07:03:46 +04:00
tdmac - > irq = irq ;
2012-06-15 07:04:08 +04:00
tdmac - > dev = tdev - > dev ;
tdmac - > chan . device = & tdev - > device ;
tdmac - > idx = idx ;
tdmac - > type = type ;
2013-11-28 13:29:39 +04:00
tdmac - > reg_base = tdev - > base + idx * 4 ;
2013-12-13 12:14:31 +04:00
tdmac - > pool = pool ;
2013-10-16 19:20:36 +04:00
tdmac - > status = DMA_COMPLETE ;
2012-06-15 07:04:08 +04:00
tdev - > tdmac [ tdmac - > idx ] = tdmac ;
2020-08-31 13:35:21 +03:00
tasklet_setup ( & tdmac - > tasklet , dma_do_tasklet ) ;
2012-06-15 07:04:08 +04:00
/* add the channel to tdma_chan list */
list_add_tail ( & tdmac - > chan . device_node ,
& tdev - > device . channels ) ;
return 0 ;
}
2014-01-20 16:39:01 +04:00
struct mmp_tdma_filter_param {
unsigned int chan_id ;
} ;
static bool mmp_tdma_filter_fn ( struct dma_chan * chan , void * fn_param )
{
struct mmp_tdma_filter_param * param = fn_param ;
if ( chan - > chan_id ! = param - > chan_id )
return false ;
return true ;
}
2016-07-04 13:09:48 +03:00
static struct dma_chan * mmp_tdma_xlate ( struct of_phandle_args * dma_spec ,
2014-01-20 16:39:01 +04:00
struct of_dma * ofdma )
{
struct mmp_tdma_device * tdev = ofdma - > of_dma_data ;
dma_cap_mask_t mask = tdev - > device . cap_mask ;
struct mmp_tdma_filter_param param ;
if ( dma_spec - > args_count ! = 1 )
return NULL ;
param . chan_id = dma_spec - > args [ 0 ] ;
if ( param . chan_id > = TDMA_CHANNEL_NUM )
return NULL ;
2019-05-20 14:32:18 +03:00
return __dma_request_channel ( & mask , mmp_tdma_filter_fn , & param ,
ofdma - > of_node ) ;
2014-01-20 16:39:01 +04:00
}
2015-03-16 22:17:14 +03:00
static const struct of_device_id mmp_tdma_dt_ids [ ] = {
2012-09-03 07:03:46 +04:00
{ . compatible = " marvell,adma-1.0 " , . data = ( void * ) MMP_AUD_TDMA } ,
{ . compatible = " marvell,pxa910-squ " , . data = ( void * ) PXA910_SQU } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , mmp_tdma_dt_ids ) ;
2012-11-19 22:22:55 +04:00
static int mmp_tdma_probe ( struct platform_device * pdev )
2012-06-15 07:04:08 +04:00
{
2012-09-03 07:03:46 +04:00
enum mmp_tdma_type type ;
const struct of_device_id * of_id ;
2012-06-15 07:04:08 +04:00
struct mmp_tdma_device * tdev ;
struct resource * iores ;
int i , ret ;
2012-09-03 07:03:46 +04:00
int irq = 0 , irq_num = 0 ;
2012-06-15 07:04:08 +04:00
int chan_num = TDMA_CHANNEL_NUM ;
2015-03-03 04:16:08 +03:00
struct gen_pool * pool = NULL ;
2012-06-15 07:04:08 +04:00
2012-09-03 07:03:46 +04:00
of_id = of_match_device ( mmp_tdma_dt_ids , & pdev - > dev ) ;
if ( of_id )
type = ( enum mmp_tdma_type ) of_id - > data ;
else
type = platform_get_device_id ( pdev ) - > driver_data ;
2012-06-15 07:04:08 +04:00
/* always have couple channels */
tdev = devm_kzalloc ( & pdev - > dev , sizeof ( * tdev ) , GFP_KERNEL ) ;
if ( ! tdev )
return - ENOMEM ;
tdev - > dev = & pdev - > dev ;
2012-09-03 07:03:46 +04:00
for ( i = 0 ; i < chan_num ; i + + ) {
if ( platform_get_irq ( pdev , i ) > 0 )
irq_num + + ;
}
2012-06-15 07:04:08 +04:00
iores = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-01-21 14:09:00 +04:00
tdev - > base = devm_ioremap_resource ( & pdev - > dev , iores ) ;
if ( IS_ERR ( tdev - > base ) )
return PTR_ERR ( tdev - > base ) ;
2012-06-15 07:04:08 +04:00
2012-09-03 07:03:46 +04:00
INIT_LIST_HEAD ( & tdev - > device . channels ) ;
2022-09-30 10:25:25 +03:00
pool = of_gen_pool_get ( pdev - > dev . of_node , " asram " , 0 ) ;
2013-12-13 12:14:31 +04:00
if ( ! pool ) {
dev_err ( & pdev - > dev , " asram pool not available \n " ) ;
return - ENOMEM ;
}
2012-09-03 07:03:46 +04:00
if ( irq_num ! = chan_num ) {
irq = platform_get_irq ( pdev , 0 ) ;
ret = devm_request_irq ( & pdev - > dev , irq ,
2020-06-01 22:22:52 +03:00
mmp_tdma_int_handler , IRQF_SHARED , " tdma " , tdev ) ;
2012-06-15 07:04:08 +04:00
if ( ret )
return ret ;
}
/* initialize channel parameters */
for ( i = 0 ; i < chan_num ; i + + ) {
2012-09-03 07:03:46 +04:00
irq = ( irq_num ! = chan_num ) ? 0 : platform_get_irq ( pdev , i ) ;
2013-12-13 12:14:31 +04:00
ret = mmp_tdma_chan_init ( tdev , i , irq , type , pool ) ;
2012-06-15 07:04:08 +04:00
if ( ret )
return ret ;
}
2012-09-03 07:03:46 +04:00
dma_cap_set ( DMA_SLAVE , tdev - > device . cap_mask ) ;
dma_cap_set ( DMA_CYCLIC , tdev - > device . cap_mask ) ;
2012-06-15 07:04:08 +04:00
tdev - > device . dev = & pdev - > dev ;
tdev - > device . device_alloc_chan_resources =
mmp_tdma_alloc_chan_resources ;
tdev - > device . device_free_chan_resources =
mmp_tdma_free_chan_resources ;
tdev - > device . device_prep_dma_cyclic = mmp_tdma_prep_dma_cyclic ;
tdev - > device . device_tx_status = mmp_tdma_tx_status ;
tdev - > device . device_issue_pending = mmp_tdma_issue_pending ;
2014-11-17 16:42:22 +03:00
tdev - > device . device_config = mmp_tdma_config ;
tdev - > device . device_pause = mmp_tdma_pause_chan ;
tdev - > device . device_resume = mmp_tdma_resume_chan ;
tdev - > device . device_terminate_all = mmp_tdma_terminate_all ;
2015-07-20 11:41:32 +03:00
tdev - > device . copy_align = DMAENGINE_ALIGN_8_BYTES ;
2012-06-15 07:04:08 +04:00
2020-04-19 19:49:11 +03:00
tdev - > device . directions = BIT ( DMA_DEV_TO_MEM ) | BIT ( DMA_MEM_TO_DEV ) ;
if ( type = = MMP_AUD_TDMA ) {
tdev - > device . max_burst = SZ_128 ;
tdev - > device . src_addr_widths = BIT ( DMA_SLAVE_BUSWIDTH_4_BYTES ) ;
tdev - > device . dst_addr_widths = BIT ( DMA_SLAVE_BUSWIDTH_4_BYTES ) ;
} else if ( type = = PXA910_SQU ) {
tdev - > device . max_burst = SZ_32 ;
}
tdev - > device . residue_granularity = DMA_RESIDUE_GRANULARITY_BURST ;
tdev - > device . descriptor_reuse = true ;
2012-06-15 07:04:08 +04:00
dma_set_mask ( & pdev - > dev , DMA_BIT_MASK ( 64 ) ) ;
platform_set_drvdata ( pdev , tdev ) ;
2018-08-06 11:52:26 +03:00
ret = dmaenginem_async_device_register ( & tdev - > device ) ;
2012-06-15 07:04:08 +04:00
if ( ret ) {
dev_err ( tdev - > device . dev , " unable to register \n " ) ;
return ret ;
}
2014-01-20 16:39:01 +04:00
if ( pdev - > dev . of_node ) {
ret = of_dma_controller_register ( pdev - > dev . of_node ,
mmp_tdma_xlate , tdev ) ;
if ( ret ) {
dev_err ( tdev - > device . dev ,
" failed to register controller \n " ) ;
2018-08-06 11:52:26 +03:00
return ret ;
2014-01-20 16:39:01 +04:00
}
}
2012-06-15 07:04:08 +04:00
dev_info ( tdev - > device . dev , " initialized \n " ) ;
return 0 ;
}
static const struct platform_device_id mmp_tdma_id_table [ ] = {
{ " mmp-adma " , MMP_AUD_TDMA } ,
{ " pxa910-squ " , PXA910_SQU } ,
{ } ,
} ;
static struct platform_driver mmp_tdma_driver = {
. driver = {
. name = " mmp-tdma " ,
2012-09-03 07:03:46 +04:00
. of_match_table = mmp_tdma_dt_ids ,
2012-06-15 07:04:08 +04:00
} ,
. id_table = mmp_tdma_id_table ,
. probe = mmp_tdma_probe ,
2012-11-19 22:20:04 +04:00
. remove = mmp_tdma_remove ,
2012-06-15 07:04:08 +04:00
} ;
module_platform_driver ( mmp_tdma_driver ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " MMP Two-Channel DMA Driver " ) ;
MODULE_ALIAS ( " platform:mmp-tdma " ) ;
MODULE_AUTHOR ( " Leo Yan <leoy@marvell.com> " ) ;
MODULE_AUTHOR ( " Zhangfei Gao <zhangfei.gao@marvell.com> " ) ;