2009-11-27 13:47:10 +01:00
/*
* raumfeld_audio . c - - SoC audio for Raumfeld audio devices
*
* Copyright ( c ) 2009 Daniel Mack < daniel @ caiaq . de >
*
* based on code from :
*
* Wolfson Microelectronics PLC .
* Openedhand Ltd .
* Liam Girdwood < lrg @ slimlogic . co . uk >
* Richard Purdie < richard @ openedhand . com >
*
* 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/module.h>
# include <linux/i2c.h>
# include <linux/delay.h>
# include <linux/gpio.h>
# include <sound/pcm.h>
# include <sound/soc.h>
# include <asm/mach-types.h>
# include "pxa-ssp.h"
# define GPIO_SPDIF_RESET (38)
# define GPIO_MCLK_RESET (111)
# define GPIO_CODEC_RESET (120)
static struct i2c_client * max9486_client ;
static struct i2c_board_info max9486_hwmon_info = {
I2C_BOARD_INFO ( " max9485 " , 0x63 ) ,
} ;
# define MAX9485_MCLK_FREQ_112896 0x22
2010-01-15 17:36:49 +01:00
# define MAX9485_MCLK_FREQ_122880 0x23
# define MAX9485_MCLK_FREQ_225792 0x32
# define MAX9485_MCLK_FREQ_245760 0x33
2009-11-27 13:47:10 +01:00
static void set_max9485_clk ( char clk )
{
i2c_master_send ( max9486_client , & clk , 1 ) ;
}
static void raumfeld_enable_audio ( bool en )
{
if ( en ) {
gpio_set_value ( GPIO_MCLK_RESET , 1 ) ;
/* wait some time to let the clocks become stable */
msleep ( 100 ) ;
gpio_set_value ( GPIO_SPDIF_RESET , 1 ) ;
gpio_set_value ( GPIO_CODEC_RESET , 1 ) ;
} else {
gpio_set_value ( GPIO_MCLK_RESET , 0 ) ;
gpio_set_value ( GPIO_SPDIF_RESET , 0 ) ;
gpio_set_value ( GPIO_CODEC_RESET , 0 ) ;
}
}
/* CS4270 */
static int raumfeld_cs4270_startup ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
2009-11-27 13:47:10 +01:00
2010-01-15 17:36:49 +01:00
/* set freq to 0 to enable all possible codec sample rates */
return snd_soc_dai_set_sysclk ( codec_dai , 0 , 0 , 0 ) ;
}
2009-11-27 13:47:10 +01:00
2010-01-15 17:36:49 +01:00
static void raumfeld_cs4270_shutdown ( struct snd_pcm_substream * substream )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
2010-01-15 17:36:49 +01:00
/* set freq to 0 to enable all possible codec sample rates */
snd_soc_dai_set_sysclk ( codec_dai , 0 , 0 , 0 ) ;
2009-11-27 13:47:10 +01:00
}
static int raumfeld_cs4270_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
struct snd_soc_dai * cpu_dai = rtd - > cpu_dai ;
2009-11-27 13:47:10 +01:00
unsigned int fmt , clk = 0 ;
int ret = 0 ;
switch ( params_rate ( params ) ) {
2010-01-15 17:36:49 +01:00
case 44100 :
set_max9485_clk ( MAX9485_MCLK_FREQ_112896 ) ;
clk = 11289600 ;
break ;
2009-11-27 13:47:10 +01:00
case 48000 :
set_max9485_clk ( MAX9485_MCLK_FREQ_122880 ) ;
clk = 12288000 ;
break ;
case 88200 :
2010-01-15 17:36:49 +01:00
set_max9485_clk ( MAX9485_MCLK_FREQ_225792 ) ;
clk = 22579200 ;
2009-11-27 13:47:10 +01:00
break ;
2010-01-15 17:36:49 +01:00
case 96000 :
set_max9485_clk ( MAX9485_MCLK_FREQ_245760 ) ;
clk = 24576000 ;
break ;
default :
return - EINVAL ;
2009-11-27 13:47:10 +01:00
}
fmt = SND_SOC_DAIFMT_I2S |
SND_SOC_DAIFMT_NB_NF |
SND_SOC_DAIFMT_CBS_CFS ;
/* setup the CODEC DAI */
ret = snd_soc_dai_set_fmt ( codec_dai , fmt ) ;
if ( ret < 0 )
return ret ;
ret = snd_soc_dai_set_sysclk ( codec_dai , 0 , clk , 0 ) ;
if ( ret < 0 )
return ret ;
/* setup the CPU DAI */
2009-11-30 14:06:37 +01:00
ret = snd_soc_dai_set_pll ( cpu_dai , 0 , 0 , 0 , clk ) ;
2009-11-27 13:47:10 +01:00
if ( ret < 0 )
return ret ;
ret = snd_soc_dai_set_fmt ( cpu_dai , fmt ) ;
if ( ret < 0 )
return ret ;
ret = snd_soc_dai_set_clkdiv ( cpu_dai , PXA_SSP_DIV_SCR , 4 ) ;
if ( ret < 0 )
return ret ;
2010-01-15 17:36:49 +01:00
ret = snd_soc_dai_set_sysclk ( cpu_dai , PXA_SSP_CLK_EXT , clk , 1 ) ;
2009-11-27 13:47:10 +01:00
if ( ret < 0 )
return ret ;
return 0 ;
}
static struct snd_soc_ops raumfeld_cs4270_ops = {
. startup = raumfeld_cs4270_startup ,
2010-01-15 17:36:49 +01:00
. shutdown = raumfeld_cs4270_shutdown ,
2009-11-27 13:47:10 +01:00
. hw_params = raumfeld_cs4270_hw_params ,
} ;
2011-01-26 14:05:25 +00:00
static int raumfeld_line_suspend ( struct snd_soc_card * card )
2009-11-27 13:47:10 +01:00
{
raumfeld_enable_audio ( false ) ;
return 0 ;
}
2011-01-26 14:05:25 +00:00
static int raumfeld_line_resume ( struct snd_soc_card * card )
2009-11-27 13:47:10 +01:00
{
raumfeld_enable_audio ( true ) ;
return 0 ;
}
/* AK4104 */
static int raumfeld_ak4104_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * params )
{
struct snd_soc_pcm_runtime * rtd = substream - > private_data ;
2010-03-17 20:15:21 +00:00
struct snd_soc_dai * codec_dai = rtd - > codec_dai ;
struct snd_soc_dai * cpu_dai = rtd - > cpu_dai ;
2009-11-27 13:47:10 +01:00
int fmt , ret = 0 , clk = 0 ;
switch ( params_rate ( params ) ) {
2010-01-15 17:36:49 +01:00
case 44100 :
set_max9485_clk ( MAX9485_MCLK_FREQ_112896 ) ;
clk = 11289600 ;
break ;
2009-11-27 13:47:10 +01:00
case 48000 :
set_max9485_clk ( MAX9485_MCLK_FREQ_122880 ) ;
clk = 12288000 ;
break ;
case 88200 :
2010-01-15 17:36:49 +01:00
set_max9485_clk ( MAX9485_MCLK_FREQ_225792 ) ;
clk = 22579200 ;
break ;
case 96000 :
set_max9485_clk ( MAX9485_MCLK_FREQ_245760 ) ;
clk = 24576000 ;
2009-11-27 13:47:10 +01:00
break ;
2010-01-15 17:36:49 +01:00
default :
return - EINVAL ;
2009-11-27 13:47:10 +01:00
}
fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF ;
/* setup the CODEC DAI */
ret = snd_soc_dai_set_fmt ( codec_dai , fmt | SND_SOC_DAIFMT_CBS_CFS ) ;
if ( ret < 0 )
return ret ;
/* setup the CPU DAI */
2009-11-30 14:06:37 +01:00
ret = snd_soc_dai_set_pll ( cpu_dai , 0 , 0 , 0 , clk ) ;
2009-11-27 13:47:10 +01:00
if ( ret < 0 )
return ret ;
ret = snd_soc_dai_set_fmt ( cpu_dai , fmt | SND_SOC_DAIFMT_CBS_CFS ) ;
if ( ret < 0 )
return ret ;
ret = snd_soc_dai_set_clkdiv ( cpu_dai , PXA_SSP_DIV_SCR , 4 ) ;
if ( ret < 0 )
return ret ;
2010-01-15 17:36:49 +01:00
ret = snd_soc_dai_set_sysclk ( cpu_dai , PXA_SSP_CLK_EXT , clk , 1 ) ;
2009-11-27 13:47:10 +01:00
if ( ret < 0 )
return ret ;
return 0 ;
}
static struct snd_soc_ops raumfeld_ak4104_ops = {
. hw_params = raumfeld_ak4104_hw_params ,
} ;
2010-03-17 20:15:21 +00:00
static struct snd_soc_dai_link raumfeld_dai [ ] = {
{
2009-11-27 13:47:10 +01:00
. name = " ak4104 " ,
. stream_name = " Playback " ,
2011-01-31 16:10:50 +01:00
. cpu_dai_name = " pxa-ssp-dai.1 " ,
. codec_dai_name = " ak4104-hifi " ,
. platform_name = " pxa-pcm-audio " ,
2009-11-27 13:47:10 +01:00
. ops = & raumfeld_ak4104_ops ,
2011-01-31 16:10:50 +01:00
. codec_name = " ak4104-codec.0 " ,
2010-03-17 20:15:21 +00:00
} ,
{
. name = " CS4270 " ,
. stream_name = " CS4270 " ,
2011-01-31 16:10:50 +01:00
. cpu_dai_name = " pxa-ssp-dai.0 " ,
. platform_name = " pxa-pcm-audio " ,
. codec_dai_name = " cs4270-hifi " ,
. codec_name = " cs4270-codec.0-0048 " ,
2010-03-17 20:15:21 +00:00
. ops = & raumfeld_cs4270_ops ,
} , } ;
2009-11-27 13:47:10 +01:00
2010-03-17 20:15:21 +00:00
static struct snd_soc_card snd_soc_raumfeld = {
. name = " Raumfeld " ,
. dai_link = raumfeld_dai ,
. suspend_post = raumfeld_line_suspend ,
. resume_pre = raumfeld_line_resume ,
. num_links = ARRAY_SIZE ( raumfeld_dai ) ,
2009-11-27 13:47:10 +01:00
} ;
2010-03-17 20:15:21 +00:00
static struct platform_device * raumfeld_audio_device ;
2009-11-27 13:47:10 +01:00
static int __init raumfeld_audio_init ( void )
{
int ret ;
if ( ! machine_is_raumfeld_speaker ( ) & &
! machine_is_raumfeld_connector ( ) )
return 0 ;
max9486_client = i2c_new_device ( i2c_get_adapter ( 0 ) ,
& max9486_hwmon_info ) ;
if ( ! max9486_client )
return - ENOMEM ;
set_max9485_clk ( MAX9485_MCLK_FREQ_122880 ) ;
2010-03-17 20:15:21 +00:00
/* Register LINE and SPDIF */
raumfeld_audio_device = platform_device_alloc ( " soc-audio " , 0 ) ;
if ( ! raumfeld_audio_device )
2009-11-27 13:47:10 +01:00
return - ENOMEM ;
2010-03-17 20:15:21 +00:00
platform_set_drvdata ( raumfeld_audio_device ,
& snd_soc_raumfeld ) ;
ret = platform_device_add ( raumfeld_audio_device ) ;
2009-11-27 13:47:10 +01:00
/* no S/PDIF on Speakers */
if ( machine_is_raumfeld_speaker ( ) )
return ret ;
raumfeld_enable_audio ( true ) ;
return ret ;
}
static void __exit raumfeld_audio_exit ( void )
{
raumfeld_enable_audio ( false ) ;
2010-03-17 20:15:21 +00:00
platform_device_unregister ( raumfeld_audio_device ) ;
2009-11-27 13:47:10 +01:00
i2c_unregister_device ( max9486_client ) ;
gpio_free ( GPIO_MCLK_RESET ) ;
gpio_free ( GPIO_CODEC_RESET ) ;
gpio_free ( GPIO_SPDIF_RESET ) ;
}
module_init ( raumfeld_audio_init ) ;
module_exit ( raumfeld_audio_exit ) ;
/* Module information */
MODULE_AUTHOR ( " Daniel Mack <daniel@caiaq.de> " ) ;
MODULE_DESCRIPTION ( " Raumfeld audio SoC " ) ;
MODULE_LICENSE ( " GPL " ) ;