2010-05-31 13:49:14 +02:00
/*
* kirkwood - dma . c
*
* ( c ) 2010 Arnaud Patard < apatard @ mandriva . com >
2010-09-09 14:10:33 +02:00
* ( c ) 2010 Arnaud Patard < arnaud . patard @ rtp - net . org >
2010-05-31 13:49:14 +02:00
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/device.h>
# include <linux/io.h>
# include <linux/slab.h>
# include <linux/interrupt.h>
# include <linux/dma-mapping.h>
# include <linux/mbus.h>
# include <sound/soc.h>
# include "kirkwood.h"
# define KIRKWOOD_RATES \
2012-11-20 12:20:14 +00:00
( SNDRV_PCM_RATE_8000_192000 | \
SNDRV_PCM_RATE_CONTINUOUS | \
SNDRV_PCM_RATE_KNOT )
2010-05-31 13:49:14 +02:00
# define KIRKWOOD_FORMATS \
( SNDRV_PCM_FMTBIT_S16_LE | \
SNDRV_PCM_FMTBIT_S24_LE | \
2012-11-20 12:20:55 +00:00
SNDRV_PCM_FMTBIT_S32_LE | \
SNDRV_PCM_FMTBIT_IEC958_SUBFRAME_LE | \
SNDRV_PCM_FMTBIT_IEC958_SUBFRAME_BE )
2010-05-31 13:49:14 +02:00
struct kirkwood_dma_priv {
struct snd_pcm_substream * play_stream ;
struct snd_pcm_substream * rec_stream ;
struct kirkwood_dma_data * data ;
} ;
static struct snd_pcm_hardware kirkwood_dma_snd_hw = {
. info = ( SNDRV_PCM_INFO_INTERLEAVED |
SNDRV_PCM_INFO_MMAP |
SNDRV_PCM_INFO_MMAP_VALID |
SNDRV_PCM_INFO_BLOCK_TRANSFER |
SNDRV_PCM_INFO_PAUSE ) ,
. formats = KIRKWOOD_FORMATS ,
. rates = KIRKWOOD_RATES ,
2012-11-20 12:20:14 +00:00
. rate_min = 8000 ,
. rate_max = 384000 ,
2010-05-31 13:49:14 +02:00
. channels_min = 1 ,
2012-11-20 12:20:55 +00:00
. channels_max = 8 ,
2010-05-31 13:49:14 +02:00
. buffer_bytes_max = KIRKWOOD_SND_MAX_PERIOD_BYTES * KIRKWOOD_SND_MAX_PERIODS ,
. period_bytes_min = KIRKWOOD_SND_MIN_PERIOD_BYTES ,
. period_bytes_max = KIRKWOOD_SND_MAX_PERIOD_BYTES ,
. periods_min = KIRKWOOD_SND_MIN_PERIODS ,
. periods_max = KIRKWOOD_SND_MAX_PERIODS ,
. fifo_size = 0 ,
} ;
2012-01-01 02:43:03 +01:00
static u64 kirkwood_dma_dmamask = DMA_BIT_MASK ( 32 ) ;
2010-05-31 13:49:14 +02:00
static irqreturn_t kirkwood_dma_irq ( int irq , void * dev_id )
{
struct kirkwood_dma_priv * prdata = dev_id ;
struct kirkwood_dma_data * priv = prdata - > data ;
unsigned long mask , status , cause ;
mask = readl ( priv - > io + KIRKWOOD_INT_MASK ) ;
status = readl ( priv - > io + KIRKWOOD_INT_CAUSE ) & mask ;
cause = readl ( priv - > io + KIRKWOOD_ERR_CAUSE ) ;
if ( unlikely ( cause ) ) {
printk ( KERN_WARNING " %s: got err interrupt 0x%lx \n " ,
__func__ , cause ) ;
writel ( cause , priv - > io + KIRKWOOD_ERR_CAUSE ) ;
}
/* we've enabled only bytes interrupts ... */
if ( status & ~ ( KIRKWOOD_INT_CAUSE_PLAY_BYTES | \
KIRKWOOD_INT_CAUSE_REC_BYTES ) ) {
printk ( KERN_WARNING " %s: unexpected interrupt %lx \n " ,
__func__ , status ) ;
return IRQ_NONE ;
}
/* ack int */
writel ( status , priv - > io + KIRKWOOD_INT_CAUSE ) ;
if ( status & KIRKWOOD_INT_CAUSE_PLAY_BYTES )
snd_pcm_period_elapsed ( prdata - > play_stream ) ;
if ( status & KIRKWOOD_INT_CAUSE_REC_BYTES )
snd_pcm_period_elapsed ( prdata - > rec_stream ) ;
return IRQ_HANDLED ;
}
2011-12-07 21:48:07 +01:00
static void
kirkwood_dma_conf_mbus_windows ( void __iomem * base , int win ,
unsigned long dma ,
const struct mbus_dram_target_info * dram )
2010-05-31 13:49:14 +02:00
{
int i ;
/* First disable and clear windows */
writel ( 0 , base + KIRKWOOD_AUDIO_WIN_CTRL_REG ( win ) ) ;
writel ( 0 , base + KIRKWOOD_AUDIO_WIN_BASE_REG ( win ) ) ;
/* try to find matching cs for current dma address */
for ( i = 0 ; i < dram - > num_cs ; i + + ) {
2011-12-07 21:48:07 +01:00
const struct mbus_dram_window * cs = dram - > cs + i ;
2010-05-31 13:49:14 +02:00
if ( ( cs - > base & 0xffff0000 ) < ( dma & 0xffff0000 ) ) {
writel ( cs - > base & 0xffff0000 ,
base + KIRKWOOD_AUDIO_WIN_BASE_REG ( win ) ) ;
writel ( ( ( cs - > size - 1 ) & 0xffff0000 ) |
( cs - > mbus_attr < < 8 ) |
( dram - > mbus_dram_target_id < < 4 ) | 1 ,
base + KIRKWOOD_AUDIO_WIN_CTRL_REG ( win ) ) ;
}
}
}
static int kirkwood_dma_open ( struct snd_pcm_substream * substream )
{
int err ;
struct snd_pcm_runtime * runtime = substream - > runtime ;
struct snd_soc_pcm_runtime * soc_runtime = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_platform * platform = soc_runtime - > platform ;
struct snd_soc_dai * cpu_dai = soc_runtime - > cpu_dai ;
2010-05-31 13:49:14 +02:00
struct kirkwood_dma_data * priv ;
2010-03-17 20:15:21 +00:00
struct kirkwood_dma_priv * prdata = snd_soc_platform_get_drvdata ( platform ) ;
2011-12-07 21:48:07 +01:00
const struct mbus_dram_target_info * dram ;
2010-05-31 13:49:14 +02:00
unsigned long addr ;
priv = snd_soc_dai_get_dma_data ( cpu_dai , substream ) ;
snd_soc_set_runtime_hwparams ( substream , & kirkwood_dma_snd_hw ) ;
2011-03-30 22:57:33 -03:00
/* Ensure that all constraints linked to dma burst are fulfilled */
2010-05-31 13:49:14 +02:00
err = snd_pcm_hw_constraint_minmax ( runtime ,
SNDRV_PCM_HW_PARAM_BUFFER_BYTES ,
priv - > burst * 2 ,
KIRKWOOD_AUDIO_BUF_MAX - 1 ) ;
if ( err < 0 )
return err ;
err = snd_pcm_hw_constraint_step ( runtime , 0 ,
SNDRV_PCM_HW_PARAM_BUFFER_BYTES ,
priv - > burst ) ;
if ( err < 0 )
return err ;
err = snd_pcm_hw_constraint_step ( substream - > runtime , 0 ,
SNDRV_PCM_HW_PARAM_PERIOD_BYTES ,
priv - > burst ) ;
if ( err < 0 )
return err ;
2010-03-17 20:15:21 +00:00
if ( prdata = = NULL ) {
2010-05-31 13:49:14 +02:00
prdata = kzalloc ( sizeof ( struct kirkwood_dma_priv ) , GFP_KERNEL ) ;
if ( prdata = = NULL )
return - ENOMEM ;
prdata - > data = priv ;
err = request_irq ( priv - > irq , kirkwood_dma_irq , IRQF_SHARED ,
" kirkwood-i2s " , prdata ) ;
if ( err ) {
kfree ( prdata ) ;
return - EBUSY ;
}
2010-03-17 20:15:21 +00:00
snd_soc_platform_set_drvdata ( platform , prdata ) ;
2010-05-31 13:49:14 +02:00
/*
* Enable Error interrupts . We ' re only ack ' ing them but
2011-03-30 22:57:33 -03:00
* it ' s useful for diagnostics
2010-05-31 13:49:14 +02:00
*/
writel ( ( unsigned long ) - 1 , priv - > io + KIRKWOOD_ERR_MASK ) ;
}
2011-12-07 21:48:07 +01:00
dram = mv_mbus_dram_info ( ) ;
2012-11-20 12:17:51 +00:00
addr = substream - > dma_buffer . addr ;
2010-05-31 13:49:14 +02:00
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) {
prdata - > play_stream = substream ;
kirkwood_dma_conf_mbus_windows ( priv - > io ,
2011-12-07 21:48:07 +01:00
KIRKWOOD_PLAYBACK_WIN , addr , dram ) ;
2010-05-31 13:49:14 +02:00
} else {
prdata - > rec_stream = substream ;
kirkwood_dma_conf_mbus_windows ( priv - > io ,
2011-12-07 21:48:07 +01:00
KIRKWOOD_RECORD_WIN , addr , dram ) ;
2010-05-31 13:49:14 +02:00
}
return 0 ;
}
static int kirkwood_dma_close ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * soc_runtime = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai = soc_runtime - > cpu_dai ;
struct snd_soc_platform * platform = soc_runtime - > platform ;
struct kirkwood_dma_priv * prdata = snd_soc_platform_get_drvdata ( platform ) ;
2010-05-31 13:49:14 +02:00
struct kirkwood_dma_data * priv ;
priv = snd_soc_dai_get_dma_data ( cpu_dai , substream ) ;
if ( ! prdata | | ! priv )
return 0 ;
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
prdata - > play_stream = NULL ;
else
prdata - > rec_stream = NULL ;
if ( ! prdata - > play_stream & & ! prdata - > rec_stream ) {
writel ( 0 , priv - > io + KIRKWOOD_ERR_MASK ) ;
free_irq ( priv - > irq , prdata ) ;
kfree ( prdata ) ;
2010-03-17 20:15:21 +00:00
snd_soc_platform_set_drvdata ( platform , NULL ) ;
2010-05-31 13:49:14 +02:00
}
return 0 ;
}
static int kirkwood_dma_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
struct snd_pcm_runtime * runtime = substream - > runtime ;
snd_pcm_set_runtime_buffer ( substream , & substream - > dma_buffer ) ;
runtime - > dma_bytes = params_buffer_bytes ( params ) ;
return 0 ;
}
static int kirkwood_dma_hw_free ( struct snd_pcm_substream * substream )
{
snd_pcm_set_runtime_buffer ( substream , NULL ) ;
return 0 ;
}
static int kirkwood_dma_prepare ( struct snd_pcm_substream * substream )
{
struct snd_pcm_runtime * runtime = substream - > runtime ;
struct snd_soc_pcm_runtime * soc_runtime = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai = soc_runtime - > cpu_dai ;
2010-05-31 13:49:14 +02:00
struct kirkwood_dma_data * priv ;
unsigned long size , count ;
priv = snd_soc_dai_get_dma_data ( cpu_dai , substream ) ;
/* compute buffer size in term of "words" as requested in specs */
size = frames_to_bytes ( runtime , runtime - > buffer_size ) ;
size = ( size > > 2 ) - 1 ;
count = snd_pcm_lib_period_bytes ( substream ) ;
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) {
writel ( count , priv - > io + KIRKWOOD_PLAY_BYTE_INT_COUNT ) ;
writel ( runtime - > dma_addr , priv - > io + KIRKWOOD_PLAY_BUF_ADDR ) ;
writel ( size , priv - > io + KIRKWOOD_PLAY_BUF_SIZE ) ;
} else {
writel ( count , priv - > io + KIRKWOOD_REC_BYTE_INT_COUNT ) ;
writel ( runtime - > dma_addr , priv - > io + KIRKWOOD_REC_BUF_ADDR ) ;
writel ( size , priv - > io + KIRKWOOD_REC_BUF_SIZE ) ;
}
return 0 ;
}
static snd_pcm_uframes_t kirkwood_dma_pointer ( struct snd_pcm_substream
* substream )
{
struct snd_soc_pcm_runtime * soc_runtime = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * cpu_dai = soc_runtime - > cpu_dai ;
2010-05-31 13:49:14 +02:00
struct kirkwood_dma_data * priv ;
snd_pcm_uframes_t count ;
priv = snd_soc_dai_get_dma_data ( cpu_dai , substream ) ;
if ( substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK )
count = bytes_to_frames ( substream - > runtime ,
readl ( priv - > io + KIRKWOOD_PLAY_BYTE_COUNT ) ) ;
else
count = bytes_to_frames ( substream - > runtime ,
readl ( priv - > io + KIRKWOOD_REC_BYTE_COUNT ) ) ;
return count ;
}
struct snd_pcm_ops kirkwood_dma_ops = {
. open = kirkwood_dma_open ,
. close = kirkwood_dma_close ,
. ioctl = snd_pcm_lib_ioctl ,
. hw_params = kirkwood_dma_hw_params ,
. hw_free = kirkwood_dma_hw_free ,
. prepare = kirkwood_dma_prepare ,
. pointer = kirkwood_dma_pointer ,
} ;
static int kirkwood_dma_preallocate_dma_buffer ( struct snd_pcm * pcm ,
int stream )
{
struct snd_pcm_substream * substream = pcm - > streams [ stream ] . substream ;
struct snd_dma_buffer * buf = & substream - > dma_buffer ;
size_t size = kirkwood_dma_snd_hw . buffer_bytes_max ;
buf - > dev . type = SNDRV_DMA_TYPE_DEV ;
buf - > dev . dev = pcm - > card - > dev ;
buf - > area = dma_alloc_coherent ( pcm - > card - > dev , size ,
& buf - > addr , GFP_KERNEL ) ;
if ( ! buf - > area )
return - ENOMEM ;
buf - > bytes = size ;
buf - > private_data = NULL ;
return 0 ;
}
2011-06-07 16:08:33 +01:00
static int kirkwood_dma_new ( struct snd_soc_pcm_runtime * rtd )
2010-05-31 13:49:14 +02:00
{
2011-06-07 16:08:33 +01:00
struct snd_card * card = rtd - > card - > snd_card ;
struct snd_pcm * pcm = rtd - > pcm ;
2010-05-31 13:49:14 +02:00
int ret ;
if ( ! card - > dev - > dma_mask )
card - > dev - > dma_mask = & kirkwood_dma_dmamask ;
if ( ! card - > dev - > coherent_dma_mask )
2012-01-01 02:43:03 +01:00
card - > dev - > coherent_dma_mask = DMA_BIT_MASK ( 32 ) ;
2010-05-31 13:49:14 +02:00
2012-01-01 01:58:44 +01:00
if ( pcm - > streams [ SNDRV_PCM_STREAM_PLAYBACK ] . substream ) {
2010-05-31 13:49:14 +02:00
ret = kirkwood_dma_preallocate_dma_buffer ( pcm ,
SNDRV_PCM_STREAM_PLAYBACK ) ;
if ( ret )
return ret ;
}
2012-01-01 01:58:44 +01:00
if ( pcm - > streams [ SNDRV_PCM_STREAM_CAPTURE ] . substream ) {
2010-05-31 13:49:14 +02:00
ret = kirkwood_dma_preallocate_dma_buffer ( pcm ,
SNDRV_PCM_STREAM_CAPTURE ) ;
if ( ret )
return ret ;
}
return 0 ;
}
static void kirkwood_dma_free_dma_buffers ( struct snd_pcm * pcm )
{
struct snd_pcm_substream * substream ;
struct snd_dma_buffer * buf ;
int stream ;
for ( stream = 0 ; stream < 2 ; stream + + ) {
substream = pcm - > streams [ stream ] . substream ;
if ( ! substream )
continue ;
buf = & substream - > dma_buffer ;
if ( ! buf - > area )
continue ;
dma_free_coherent ( pcm - > card - > dev , buf - > bytes ,
buf - > area , buf - > addr ) ;
buf - > area = NULL ;
}
}
2010-03-17 20:15:21 +00:00
static struct snd_soc_platform_driver kirkwood_soc_platform = {
. ops = & kirkwood_dma_ops ,
2010-05-31 13:49:14 +02:00
. pcm_new = kirkwood_dma_new ,
. pcm_free = kirkwood_dma_free_dma_buffers ,
} ;
2012-12-07 09:26:26 -05:00
static int kirkwood_soc_platform_probe ( struct platform_device * pdev )
2010-05-31 13:49:14 +02:00
{
2010-03-17 20:15:21 +00:00
return snd_soc_register_platform ( & pdev - > dev , & kirkwood_soc_platform ) ;
2010-05-31 13:49:14 +02:00
}
2012-12-07 09:26:26 -05:00
static int kirkwood_soc_platform_remove ( struct platform_device * pdev )
2010-05-31 13:49:14 +02:00
{
2010-03-17 20:15:21 +00:00
snd_soc_unregister_platform ( & pdev - > dev ) ;
return 0 ;
}
static struct platform_driver kirkwood_pcm_driver = {
. driver = {
. name = " kirkwood-pcm-audio " ,
. owner = THIS_MODULE ,
} ,
. probe = kirkwood_soc_platform_probe ,
2012-12-07 09:26:26 -05:00
. remove = kirkwood_soc_platform_remove ,
2010-03-17 20:15:21 +00:00
} ;
2011-11-24 11:43:09 +08:00
module_platform_driver ( kirkwood_pcm_driver ) ;
2010-05-31 13:49:14 +02:00
2010-09-09 14:10:33 +02:00
MODULE_AUTHOR ( " Arnaud Patard <arnaud.patard@rtp-net.org> " ) ;
2010-05-31 13:49:14 +02:00
MODULE_DESCRIPTION ( " Marvell Kirkwood Audio DMA module " ) ;
MODULE_LICENSE ( " GPL " ) ;
2010-09-02 10:31:47 +02:00
MODULE_ALIAS ( " platform:kirkwood-pcm-audio " ) ;