aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/twl4030.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/twl4030.c')
-rw-r--r--sound/soc/codecs/twl4030.c284
1 files changed, 106 insertions, 178 deletions
diff --git a/sound/soc/codecs/twl4030.c b/sound/soc/codecs/twl4030.c
index 7b618bbff884..bec788b12613 100644
--- a/sound/soc/codecs/twl4030.c
+++ b/sound/soc/codecs/twl4030.c
@@ -32,11 +32,19 @@
32#include <sound/pcm.h> 32#include <sound/pcm.h>
33#include <sound/pcm_params.h> 33#include <sound/pcm_params.h>
34#include <sound/soc.h> 34#include <sound/soc.h>
35#include <sound/soc-dapm.h>
36#include <sound/initval.h> 35#include <sound/initval.h>
37#include <sound/tlv.h> 36#include <sound/tlv.h>
38 37
39#include "twl4030.h" 38/* Register descriptions are here */
39#include <linux/mfd/twl4030-codec.h>
40
41/* Shadow register used by the audio driver */
42#define TWL4030_REG_SW_SHADOW 0x4A
43#define TWL4030_CACHEREGNUM (TWL4030_REG_SW_SHADOW + 1)
44
45/* TWL4030_REG_SW_SHADOW (0x4A) Fields */
46#define TWL4030_HFL_EN 0x01
47#define TWL4030_HFR_EN 0x02
40 48
41/* 49/*
42 * twl4030 register cache & default register settings 50 * twl4030 register cache & default register settings
@@ -224,6 +232,16 @@ static int twl4030_write(struct snd_soc_codec *codec,
224 return 0; 232 return 0;
225} 233}
226 234
235static inline void twl4030_wait_ms(int time)
236{
237 if (time < 60) {
238 time *= 1000;
239 usleep_range(time, time + 500);
240 } else {
241 msleep(time);
242 }
243}
244
227static void twl4030_codec_enable(struct snd_soc_codec *codec, int enable) 245static void twl4030_codec_enable(struct snd_soc_codec *codec, int enable)
228{ 246{
229 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 247 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
@@ -262,7 +280,7 @@ static inline void twl4030_check_defaults(struct snd_soc_codec *codec)
262 i, val, twl4030_reg[i]); 280 i, val, twl4030_reg[i]);
263 } 281 }
264 } 282 }
265 dev_dbg(codec->dev, "Found %d non maching registers. %s\n", 283 dev_dbg(codec->dev, "Found %d non-matching registers. %s\n",
266 difference, difference ? "Not OK" : "OK"); 284 difference, difference ? "Not OK" : "OK");
267} 285}
268 286
@@ -277,21 +295,19 @@ static inline void twl4030_reset_registers(struct snd_soc_codec *codec)
277 295
278} 296}
279 297
280static void twl4030_init_chip(struct platform_device *pdev) 298static void twl4030_init_chip(struct snd_soc_codec *codec)
281{ 299{
282 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 300 struct twl4030_codec_audio_data *pdata = dev_get_platdata(codec->dev);
283 struct twl4030_setup_data *setup = socdev->codec_data;
284 struct snd_soc_codec *codec = socdev->card->codec;
285 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 301 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
286 u8 reg, byte; 302 u8 reg, byte;
287 int i = 0; 303 int i = 0;
288 304
289 /* Check defaults, if instructed before anything else */ 305 /* Check defaults, if instructed before anything else */
290 if (setup && setup->check_defaults) 306 if (pdata && pdata->check_defaults)
291 twl4030_check_defaults(codec); 307 twl4030_check_defaults(codec);
292 308
293 /* Reset registers, if no setup data or if instructed to do so */ 309 /* Reset registers, if no setup data or if instructed to do so */
294 if (!setup || (setup && setup->reset_registers)) 310 if (!pdata || (pdata && pdata->reset_registers))
295 twl4030_reset_registers(codec); 311 twl4030_reset_registers(codec);
296 312
297 /* Refresh APLL_CTL register from HW */ 313 /* Refresh APLL_CTL register from HW */
@@ -312,20 +328,14 @@ static void twl4030_init_chip(struct platform_device *pdev)
312 twl4030_write(codec, TWL4030_REG_ARXR2_APGA_CTL, 0x32); 328 twl4030_write(codec, TWL4030_REG_ARXR2_APGA_CTL, 0x32);
313 329
314 /* Machine dependent setup */ 330 /* Machine dependent setup */
315 if (!setup) 331 if (!pdata)
316 return; 332 return;
317 333
318 twl4030->digimic_delay = setup->digimic_delay; 334 twl4030->digimic_delay = pdata->digimic_delay;
319
320 /* Configuration for headset ramp delay from setup data */
321 if (setup->sysclk != twl4030->sysclk)
322 dev_warn(codec->dev,
323 "Mismatch in APLL mclk: %u (configured: %u)\n",
324 setup->sysclk, twl4030->sysclk);
325 335
326 reg = twl4030_read_reg_cache(codec, TWL4030_REG_HS_POPN_SET); 336 reg = twl4030_read_reg_cache(codec, TWL4030_REG_HS_POPN_SET);
327 reg &= ~TWL4030_RAMP_DELAY; 337 reg &= ~TWL4030_RAMP_DELAY;
328 reg |= (setup->ramp_delay_value << 2); 338 reg |= (pdata->ramp_delay_value << 2);
329 twl4030_write_reg_cache(codec, TWL4030_REG_HS_POPN_SET, reg); 339 twl4030_write_reg_cache(codec, TWL4030_REG_HS_POPN_SET, reg);
330 340
331 /* initiate offset cancellation */ 341 /* initiate offset cancellation */
@@ -333,14 +343,18 @@ static void twl4030_init_chip(struct platform_device *pdev)
333 343
334 reg = twl4030_read_reg_cache(codec, TWL4030_REG_ANAMICL); 344 reg = twl4030_read_reg_cache(codec, TWL4030_REG_ANAMICL);
335 reg &= ~TWL4030_OFFSET_CNCL_SEL; 345 reg &= ~TWL4030_OFFSET_CNCL_SEL;
336 reg |= setup->offset_cncl_path; 346 reg |= pdata->offset_cncl_path;
337 twl4030_write(codec, TWL4030_REG_ANAMICL, 347 twl4030_write(codec, TWL4030_REG_ANAMICL,
338 reg | TWL4030_CNCL_OFFSET_START); 348 reg | TWL4030_CNCL_OFFSET_START);
339 349
340 /* wait for offset cancellation to complete */ 350 /*
351 * Wait for offset cancellation to complete.
352 * Since this takes a while, do not slam the i2c.
353 * Start polling the status after ~20ms.
354 */
355 msleep(20);
341 do { 356 do {
342 /* this takes a little while, so don't slam i2c */ 357 usleep_range(1000, 2000);
343 udelay(2000);
344 twl_i2c_read_u8(TWL4030_MODULE_AUDIO_VOICE, &byte, 358 twl_i2c_read_u8(TWL4030_MODULE_AUDIO_VOICE, &byte,
345 TWL4030_REG_ANAMICL); 359 TWL4030_REG_ANAMICL);
346 } while ((i++ < 100) && 360 } while ((i++ < 100) &&
@@ -718,23 +732,24 @@ static int aif_event(struct snd_soc_dapm_widget *w,
718 732
719static void headset_ramp(struct snd_soc_codec *codec, int ramp) 733static void headset_ramp(struct snd_soc_codec *codec, int ramp)
720{ 734{
721 struct snd_soc_device *socdev = codec->socdev; 735 struct twl4030_codec_audio_data *pdata = codec->dev->platform_data;
722 struct twl4030_setup_data *setup = socdev->codec_data;
723
724 unsigned char hs_gain, hs_pop; 736 unsigned char hs_gain, hs_pop;
725 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 737 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
726 /* Base values for ramp delay calculation: 2^19 - 2^26 */ 738 /* Base values for ramp delay calculation: 2^19 - 2^26 */
727 unsigned int ramp_base[] = {524288, 1048576, 2097152, 4194304, 739 unsigned int ramp_base[] = {524288, 1048576, 2097152, 4194304,
728 8388608, 16777216, 33554432, 67108864}; 740 8388608, 16777216, 33554432, 67108864};
741 unsigned int delay;
729 742
730 hs_gain = twl4030_read_reg_cache(codec, TWL4030_REG_HS_GAIN_SET); 743 hs_gain = twl4030_read_reg_cache(codec, TWL4030_REG_HS_GAIN_SET);
731 hs_pop = twl4030_read_reg_cache(codec, TWL4030_REG_HS_POPN_SET); 744 hs_pop = twl4030_read_reg_cache(codec, TWL4030_REG_HS_POPN_SET);
745 delay = (ramp_base[(hs_pop & TWL4030_RAMP_DELAY) >> 2] /
746 twl4030->sysclk) + 1;
732 747
733 /* Enable external mute control, this dramatically reduces 748 /* Enable external mute control, this dramatically reduces
734 * the pop-noise */ 749 * the pop-noise */
735 if (setup && setup->hs_extmute) { 750 if (pdata && pdata->hs_extmute) {
736 if (setup->set_hs_extmute) { 751 if (pdata->set_hs_extmute) {
737 setup->set_hs_extmute(1); 752 pdata->set_hs_extmute(1);
738 } else { 753 } else {
739 hs_pop |= TWL4030_EXTMUTE; 754 hs_pop |= TWL4030_EXTMUTE;
740 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop); 755 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop);
@@ -752,16 +767,14 @@ static void headset_ramp(struct snd_soc_codec *codec, int ramp)
752 hs_pop |= TWL4030_RAMP_EN; 767 hs_pop |= TWL4030_RAMP_EN;
753 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop); 768 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop);
754 /* Wait ramp delay time + 1, so the VMID can settle */ 769 /* Wait ramp delay time + 1, so the VMID can settle */
755 mdelay((ramp_base[(hs_pop & TWL4030_RAMP_DELAY) >> 2] / 770 twl4030_wait_ms(delay);
756 twl4030->sysclk) + 1);
757 } else { 771 } else {
758 /* Headset ramp-down _not_ according to 772 /* Headset ramp-down _not_ according to
759 * the TRM, but in a way that it is working */ 773 * the TRM, but in a way that it is working */
760 hs_pop &= ~TWL4030_RAMP_EN; 774 hs_pop &= ~TWL4030_RAMP_EN;
761 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop); 775 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop);
762 /* Wait ramp delay time + 1, so the VMID can settle */ 776 /* Wait ramp delay time + 1, so the VMID can settle */
763 mdelay((ramp_base[(hs_pop & TWL4030_RAMP_DELAY) >> 2] / 777 twl4030_wait_ms(delay);
764 twl4030->sysclk) + 1);
765 /* Bypass the reg_cache to mute the headset */ 778 /* Bypass the reg_cache to mute the headset */
766 twl_i2c_write_u8(TWL4030_MODULE_AUDIO_VOICE, 779 twl_i2c_write_u8(TWL4030_MODULE_AUDIO_VOICE,
767 hs_gain & (~0x0f), 780 hs_gain & (~0x0f),
@@ -772,9 +785,9 @@ static void headset_ramp(struct snd_soc_codec *codec, int ramp)
772 } 785 }
773 786
774 /* Disable external mute */ 787 /* Disable external mute */
775 if (setup && setup->hs_extmute) { 788 if (pdata && pdata->hs_extmute) {
776 if (setup->set_hs_extmute) { 789 if (pdata->set_hs_extmute) {
777 setup->set_hs_extmute(0); 790 pdata->set_hs_extmute(0);
778 } else { 791 } else {
779 hs_pop &= ~TWL4030_EXTMUTE; 792 hs_pop &= ~TWL4030_EXTMUTE;
780 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop); 793 twl4030_write(codec, TWL4030_REG_HS_POPN_SET, hs_pop);
@@ -836,7 +849,7 @@ static int digimic_event(struct snd_soc_dapm_widget *w,
836 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(w->codec); 849 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(w->codec);
837 850
838 if (twl4030->digimic_delay) 851 if (twl4030->digimic_delay)
839 mdelay(twl4030->digimic_delay); 852 twl4030_wait_ms(twl4030->digimic_delay);
840 return 0; 853 return 0;
841} 854}
842 855
@@ -1622,10 +1635,11 @@ static const struct snd_soc_dapm_route intercon[] = {
1622 1635
1623static int twl4030_add_widgets(struct snd_soc_codec *codec) 1636static int twl4030_add_widgets(struct snd_soc_codec *codec)
1624{ 1637{
1625 snd_soc_dapm_new_controls(codec, twl4030_dapm_widgets, 1638 struct snd_soc_dapm_context *dapm = &codec->dapm;
1626 ARRAY_SIZE(twl4030_dapm_widgets));
1627 1639
1628 snd_soc_dapm_add_routes(codec, intercon, ARRAY_SIZE(intercon)); 1640 snd_soc_dapm_new_controls(dapm, twl4030_dapm_widgets,
1641 ARRAY_SIZE(twl4030_dapm_widgets));
1642 snd_soc_dapm_add_routes(dapm, intercon, ARRAY_SIZE(intercon));
1629 1643
1630 return 0; 1644 return 0;
1631} 1645}
@@ -1639,14 +1653,14 @@ static int twl4030_set_bias_level(struct snd_soc_codec *codec,
1639 case SND_SOC_BIAS_PREPARE: 1653 case SND_SOC_BIAS_PREPARE:
1640 break; 1654 break;
1641 case SND_SOC_BIAS_STANDBY: 1655 case SND_SOC_BIAS_STANDBY:
1642 if (codec->bias_level == SND_SOC_BIAS_OFF) 1656 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF)
1643 twl4030_codec_enable(codec, 1); 1657 twl4030_codec_enable(codec, 1);
1644 break; 1658 break;
1645 case SND_SOC_BIAS_OFF: 1659 case SND_SOC_BIAS_OFF:
1646 twl4030_codec_enable(codec, 0); 1660 twl4030_codec_enable(codec, 0);
1647 break; 1661 break;
1648 } 1662 }
1649 codec->bias_level = level; 1663 codec->dapm.bias_level = level;
1650 1664
1651 return 0; 1665 return 0;
1652} 1666}
@@ -1707,10 +1721,10 @@ static int twl4030_startup(struct snd_pcm_substream *substream,
1707 struct snd_soc_dai *dai) 1721 struct snd_soc_dai *dai)
1708{ 1722{
1709 struct snd_soc_pcm_runtime *rtd = substream->private_data; 1723 struct snd_soc_pcm_runtime *rtd = substream->private_data;
1710 struct snd_soc_device *socdev = rtd->socdev; 1724 struct snd_soc_codec *codec = rtd->codec;
1711 struct snd_soc_codec *codec = socdev->card->codec;
1712 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 1725 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
1713 1726
1727 snd_pcm_hw_constraint_msbits(substream->runtime, 0, 32, 24);
1714 if (twl4030->master_substream) { 1728 if (twl4030->master_substream) {
1715 twl4030->slave_substream = substream; 1729 twl4030->slave_substream = substream;
1716 /* The DAI has one configuration for playback and capture, so 1730 /* The DAI has one configuration for playback and capture, so
@@ -1738,8 +1752,7 @@ static void twl4030_shutdown(struct snd_pcm_substream *substream,
1738 struct snd_soc_dai *dai) 1752 struct snd_soc_dai *dai)
1739{ 1753{
1740 struct snd_soc_pcm_runtime *rtd = substream->private_data; 1754 struct snd_soc_pcm_runtime *rtd = substream->private_data;
1741 struct snd_soc_device *socdev = rtd->socdev; 1755 struct snd_soc_codec *codec = rtd->codec;
1742 struct snd_soc_codec *codec = socdev->card->codec;
1743 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 1756 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
1744 1757
1745 if (twl4030->master_substream == substream) 1758 if (twl4030->master_substream == substream)
@@ -1764,8 +1777,7 @@ static int twl4030_hw_params(struct snd_pcm_substream *substream,
1764 struct snd_soc_dai *dai) 1777 struct snd_soc_dai *dai)
1765{ 1778{
1766 struct snd_soc_pcm_runtime *rtd = substream->private_data; 1779 struct snd_soc_pcm_runtime *rtd = substream->private_data;
1767 struct snd_soc_device *socdev = rtd->socdev; 1780 struct snd_soc_codec *codec = rtd->codec;
1768 struct snd_soc_codec *codec = socdev->card->codec;
1769 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 1781 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
1770 u8 mode, old_mode, format, old_format; 1782 u8 mode, old_mode, format, old_format;
1771 1783
@@ -1837,7 +1849,7 @@ static int twl4030_hw_params(struct snd_pcm_substream *substream,
1837 case SNDRV_PCM_FORMAT_S16_LE: 1849 case SNDRV_PCM_FORMAT_S16_LE:
1838 format |= TWL4030_DATA_WIDTH_16S_16W; 1850 format |= TWL4030_DATA_WIDTH_16S_16W;
1839 break; 1851 break;
1840 case SNDRV_PCM_FORMAT_S24_LE: 1852 case SNDRV_PCM_FORMAT_S32_LE:
1841 format |= TWL4030_DATA_WIDTH_32S_24W; 1853 format |= TWL4030_DATA_WIDTH_32S_24W;
1842 break; 1854 break;
1843 default: 1855 default:
@@ -1999,13 +2011,12 @@ static int twl4030_voice_startup(struct snd_pcm_substream *substream,
1999 struct snd_soc_dai *dai) 2011 struct snd_soc_dai *dai)
2000{ 2012{
2001 struct snd_soc_pcm_runtime *rtd = substream->private_data; 2013 struct snd_soc_pcm_runtime *rtd = substream->private_data;
2002 struct snd_soc_device *socdev = rtd->socdev; 2014 struct snd_soc_codec *codec = rtd->codec;
2003 struct snd_soc_codec *codec = socdev->card->codec;
2004 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 2015 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
2005 u8 mode; 2016 u8 mode;
2006 2017
2007 /* If the system master clock is not 26MHz, the voice PCM interface is 2018 /* If the system master clock is not 26MHz, the voice PCM interface is
2008 * not avilable. 2019 * not available.
2009 */ 2020 */
2010 if (twl4030->sysclk != 26000) { 2021 if (twl4030->sysclk != 26000) {
2011 dev_err(codec->dev, "The board is configured for %u Hz, while" 2022 dev_err(codec->dev, "The board is configured for %u Hz, while"
@@ -2015,7 +2026,7 @@ static int twl4030_voice_startup(struct snd_pcm_substream *substream,
2015 } 2026 }
2016 2027
2017 /* If the codec mode is not option2, the voice PCM interface is not 2028 /* If the codec mode is not option2, the voice PCM interface is not
2018 * avilable. 2029 * available.
2019 */ 2030 */
2020 mode = twl4030_read_reg_cache(codec, TWL4030_REG_CODEC_MODE) 2031 mode = twl4030_read_reg_cache(codec, TWL4030_REG_CODEC_MODE)
2021 & TWL4030_OPT_MODE; 2032 & TWL4030_OPT_MODE;
@@ -2033,8 +2044,7 @@ static void twl4030_voice_shutdown(struct snd_pcm_substream *substream,
2033 struct snd_soc_dai *dai) 2044 struct snd_soc_dai *dai)
2034{ 2045{
2035 struct snd_soc_pcm_runtime *rtd = substream->private_data; 2046 struct snd_soc_pcm_runtime *rtd = substream->private_data;
2036 struct snd_soc_device *socdev = rtd->socdev; 2047 struct snd_soc_codec *codec = rtd->codec;
2037 struct snd_soc_codec *codec = socdev->card->codec;
2038 2048
2039 /* Enable voice digital filters */ 2049 /* Enable voice digital filters */
2040 twl4030_voice_enable(codec, substream->stream, 0); 2050 twl4030_voice_enable(codec, substream->stream, 0);
@@ -2044,8 +2054,7 @@ static int twl4030_voice_hw_params(struct snd_pcm_substream *substream,
2044 struct snd_pcm_hw_params *params, struct snd_soc_dai *dai) 2054 struct snd_pcm_hw_params *params, struct snd_soc_dai *dai)
2045{ 2055{
2046 struct snd_soc_pcm_runtime *rtd = substream->private_data; 2056 struct snd_soc_pcm_runtime *rtd = substream->private_data;
2047 struct snd_soc_device *socdev = rtd->socdev; 2057 struct snd_soc_codec *codec = rtd->codec;
2048 struct snd_soc_codec *codec = socdev->card->codec;
2049 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec); 2058 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
2050 u8 old_mode, mode; 2059 u8 old_mode, mode;
2051 2060
@@ -2173,9 +2182,9 @@ static int twl4030_voice_set_tristate(struct snd_soc_dai *dai, int tristate)
2173} 2182}
2174 2183
2175#define TWL4030_RATES (SNDRV_PCM_RATE_8000_48000) 2184#define TWL4030_RATES (SNDRV_PCM_RATE_8000_48000)
2176#define TWL4030_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FORMAT_S24_LE) 2185#define TWL4030_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S32_LE)
2177 2186
2178static struct snd_soc_dai_ops twl4030_dai_ops = { 2187static struct snd_soc_dai_ops twl4030_dai_hifi_ops = {
2179 .startup = twl4030_startup, 2188 .startup = twl4030_startup,
2180 .shutdown = twl4030_shutdown, 2189 .shutdown = twl4030_shutdown,
2181 .hw_params = twl4030_hw_params, 2190 .hw_params = twl4030_hw_params,
@@ -2193,9 +2202,9 @@ static struct snd_soc_dai_ops twl4030_dai_voice_ops = {
2193 .set_tristate = twl4030_voice_set_tristate, 2202 .set_tristate = twl4030_voice_set_tristate,
2194}; 2203};
2195 2204
2196struct snd_soc_dai twl4030_dai[] = { 2205static struct snd_soc_dai_driver twl4030_dai[] = {
2197{ 2206{
2198 .name = "twl4030", 2207 .name = "twl4030-hifi",
2199 .playback = { 2208 .playback = {
2200 .stream_name = "HiFi Playback", 2209 .stream_name = "HiFi Playback",
2201 .channels_min = 2, 2210 .channels_min = 2,
@@ -2208,10 +2217,10 @@ struct snd_soc_dai twl4030_dai[] = {
2208 .channels_max = 4, 2217 .channels_max = 4,
2209 .rates = TWL4030_RATES, 2218 .rates = TWL4030_RATES,
2210 .formats = TWL4030_FORMATS,}, 2219 .formats = TWL4030_FORMATS,},
2211 .ops = &twl4030_dai_ops, 2220 .ops = &twl4030_dai_hifi_ops,
2212}, 2221},
2213{ 2222{
2214 .name = "twl4030 Voice", 2223 .name = "twl4030-voice",
2215 .playback = { 2224 .playback = {
2216 .stream_name = "Voice Playback", 2225 .stream_name = "Voice Playback",
2217 .channels_min = 1, 2226 .channels_min = 1,
@@ -2227,164 +2236,91 @@ struct snd_soc_dai twl4030_dai[] = {
2227 .ops = &twl4030_dai_voice_ops, 2236 .ops = &twl4030_dai_voice_ops,
2228}, 2237},
2229}; 2238};
2230EXPORT_SYMBOL_GPL(twl4030_dai);
2231 2239
2232static int twl4030_soc_suspend(struct platform_device *pdev, pm_message_t state) 2240static int twl4030_soc_suspend(struct snd_soc_codec *codec, pm_message_t state)
2233{ 2241{
2234 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
2235 struct snd_soc_codec *codec = socdev->card->codec;
2236
2237 twl4030_set_bias_level(codec, SND_SOC_BIAS_OFF); 2242 twl4030_set_bias_level(codec, SND_SOC_BIAS_OFF);
2238
2239 return 0; 2243 return 0;
2240} 2244}
2241 2245
2242static int twl4030_soc_resume(struct platform_device *pdev) 2246static int twl4030_soc_resume(struct snd_soc_codec *codec)
2243{ 2247{
2244 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
2245 struct snd_soc_codec *codec = socdev->card->codec;
2246
2247 twl4030_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 2248 twl4030_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
2248 return 0; 2249 return 0;
2249} 2250}
2250 2251
2251static struct snd_soc_codec *twl4030_codec; 2252static int twl4030_soc_probe(struct snd_soc_codec *codec)
2252
2253static int twl4030_soc_probe(struct platform_device *pdev)
2254{ 2253{
2255 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 2254 struct twl4030_priv *twl4030;
2256 struct snd_soc_codec *codec;
2257 int ret;
2258
2259 BUG_ON(!twl4030_codec);
2260
2261 codec = twl4030_codec;
2262 socdev->card->codec = codec;
2263
2264 twl4030_init_chip(pdev);
2265 2255
2266 /* register pcms */ 2256 twl4030 = kzalloc(sizeof(struct twl4030_priv), GFP_KERNEL);
2267 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); 2257 if (twl4030 == NULL) {
2268 if (ret < 0) { 2258 printk("Can not allocate memroy\n");
2269 dev_err(&pdev->dev, "failed to create pcms\n"); 2259 return -ENOMEM;
2270 return ret;
2271 } 2260 }
2261 snd_soc_codec_set_drvdata(codec, twl4030);
2262 /* Set the defaults, and power up the codec */
2263 twl4030->sysclk = twl4030_codec_get_mclk() / 1000;
2264 codec->dapm.idle_bias_off = 1;
2265
2266 twl4030_init_chip(codec);
2272 2267
2273 snd_soc_add_controls(codec, twl4030_snd_controls, 2268 snd_soc_add_controls(codec, twl4030_snd_controls,
2274 ARRAY_SIZE(twl4030_snd_controls)); 2269 ARRAY_SIZE(twl4030_snd_controls));
2275 twl4030_add_widgets(codec); 2270 twl4030_add_widgets(codec);
2276
2277 return 0; 2271 return 0;
2278} 2272}
2279 2273
2280static int twl4030_soc_remove(struct platform_device *pdev) 2274static int twl4030_soc_remove(struct snd_soc_codec *codec)
2281{ 2275{
2282 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 2276 struct twl4030_priv *twl4030 = snd_soc_codec_get_drvdata(codec);
2283 struct snd_soc_codec *codec = socdev->card->codec;
2284 2277
2285 /* Reset registers to their chip default before leaving */ 2278 /* Reset registers to their chip default before leaving */
2286 twl4030_reset_registers(codec); 2279 twl4030_reset_registers(codec);
2287 twl4030_set_bias_level(codec, SND_SOC_BIAS_OFF); 2280 twl4030_set_bias_level(codec, SND_SOC_BIAS_OFF);
2288 snd_soc_free_pcms(socdev); 2281 kfree(twl4030);
2289 snd_soc_dapm_free(socdev);
2290
2291 return 0; 2282 return 0;
2292} 2283}
2293 2284
2285static struct snd_soc_codec_driver soc_codec_dev_twl4030 = {
2286 .probe = twl4030_soc_probe,
2287 .remove = twl4030_soc_remove,
2288 .suspend = twl4030_soc_suspend,
2289 .resume = twl4030_soc_resume,
2290 .read = twl4030_read_reg_cache,
2291 .write = twl4030_write,
2292 .set_bias_level = twl4030_set_bias_level,
2293 .reg_cache_size = sizeof(twl4030_reg),
2294 .reg_word_size = sizeof(u8),
2295 .reg_cache_default = twl4030_reg,
2296};
2297
2294static int __devinit twl4030_codec_probe(struct platform_device *pdev) 2298static int __devinit twl4030_codec_probe(struct platform_device *pdev)
2295{ 2299{
2296 struct twl4030_codec_audio_data *pdata = pdev->dev.platform_data; 2300 struct twl4030_codec_audio_data *pdata = pdev->dev.platform_data;
2297 struct snd_soc_codec *codec;
2298 struct twl4030_priv *twl4030;
2299 int ret;
2300 2301
2301 if (!pdata) { 2302 if (!pdata) {
2302 dev_err(&pdev->dev, "platform_data is missing\n"); 2303 dev_err(&pdev->dev, "platform_data is missing\n");
2303 return -EINVAL; 2304 return -EINVAL;
2304 } 2305 }
2305 2306
2306 twl4030 = kzalloc(sizeof(struct twl4030_priv), GFP_KERNEL); 2307 return snd_soc_register_codec(&pdev->dev, &soc_codec_dev_twl4030,
2307 if (twl4030 == NULL) { 2308 twl4030_dai, ARRAY_SIZE(twl4030_dai));
2308 dev_err(&pdev->dev, "Can not allocate memroy\n");
2309 return -ENOMEM;
2310 }
2311
2312 codec = &twl4030->codec;
2313 snd_soc_codec_set_drvdata(codec, twl4030);
2314 codec->dev = &pdev->dev;
2315 twl4030_dai[0].dev = &pdev->dev;
2316 twl4030_dai[1].dev = &pdev->dev;
2317
2318 mutex_init(&codec->mutex);
2319 INIT_LIST_HEAD(&codec->dapm_widgets);
2320 INIT_LIST_HEAD(&codec->dapm_paths);
2321
2322 codec->name = "twl4030";
2323 codec->owner = THIS_MODULE;
2324 codec->read = twl4030_read_reg_cache;
2325 codec->write = twl4030_write;
2326 codec->set_bias_level = twl4030_set_bias_level;
2327 codec->idle_bias_off = 1;
2328 codec->dai = twl4030_dai;
2329 codec->num_dai = ARRAY_SIZE(twl4030_dai);
2330 codec->reg_cache_size = sizeof(twl4030_reg);
2331 codec->reg_cache = kmemdup(twl4030_reg, sizeof(twl4030_reg),
2332 GFP_KERNEL);
2333 if (codec->reg_cache == NULL) {
2334 ret = -ENOMEM;
2335 goto error_cache;
2336 }
2337
2338 platform_set_drvdata(pdev, twl4030);
2339 twl4030_codec = codec;
2340
2341 /* Set the defaults, and power up the codec */
2342 twl4030->sysclk = twl4030_codec_get_mclk() / 1000;
2343 codec->bias_level = SND_SOC_BIAS_OFF;
2344
2345 ret = snd_soc_register_codec(codec);
2346 if (ret != 0) {
2347 dev_err(codec->dev, "Failed to register codec: %d\n", ret);
2348 goto error_codec;
2349 }
2350
2351 ret = snd_soc_register_dais(&twl4030_dai[0], ARRAY_SIZE(twl4030_dai));
2352 if (ret != 0) {
2353 dev_err(codec->dev, "Failed to register DAIs: %d\n", ret);
2354 snd_soc_unregister_codec(codec);
2355 goto error_codec;
2356 }
2357
2358 return 0;
2359
2360error_codec:
2361 twl4030_codec_enable(codec, 0);
2362 kfree(codec->reg_cache);
2363error_cache:
2364 kfree(twl4030);
2365 return ret;
2366} 2309}
2367 2310
2368static int __devexit twl4030_codec_remove(struct platform_device *pdev) 2311static int __devexit twl4030_codec_remove(struct platform_device *pdev)
2369{ 2312{
2370 struct twl4030_priv *twl4030 = platform_get_drvdata(pdev); 2313 snd_soc_unregister_codec(&pdev->dev);
2371
2372 snd_soc_unregister_dais(&twl4030_dai[0], ARRAY_SIZE(twl4030_dai));
2373 snd_soc_unregister_codec(&twl4030->codec);
2374 kfree(twl4030->codec.reg_cache);
2375 kfree(twl4030);
2376
2377 twl4030_codec = NULL;
2378 return 0; 2314 return 0;
2379} 2315}
2380 2316
2381MODULE_ALIAS("platform:twl4030_codec_audio"); 2317MODULE_ALIAS("platform:twl4030-codec");
2382 2318
2383static struct platform_driver twl4030_codec_driver = { 2319static struct platform_driver twl4030_codec_driver = {
2384 .probe = twl4030_codec_probe, 2320 .probe = twl4030_codec_probe,
2385 .remove = __devexit_p(twl4030_codec_remove), 2321 .remove = __devexit_p(twl4030_codec_remove),
2386 .driver = { 2322 .driver = {
2387 .name = "twl4030_codec_audio", 2323 .name = "twl4030-codec",
2388 .owner = THIS_MODULE, 2324 .owner = THIS_MODULE,
2389 }, 2325 },
2390}; 2326};
@@ -2401,14 +2337,6 @@ static void __exit twl4030_exit(void)
2401} 2337}
2402module_exit(twl4030_exit); 2338module_exit(twl4030_exit);
2403 2339
2404struct snd_soc_codec_device soc_codec_dev_twl4030 = {
2405 .probe = twl4030_soc_probe,
2406 .remove = twl4030_soc_remove,
2407 .suspend = twl4030_soc_suspend,
2408 .resume = twl4030_soc_resume,
2409};
2410EXPORT_SYMBOL_GPL(soc_codec_dev_twl4030);
2411
2412MODULE_DESCRIPTION("ASoC TWL4030 codec driver"); 2340MODULE_DESCRIPTION("ASoC TWL4030 codec driver");
2413MODULE_AUTHOR("Steve Sakoman"); 2341MODULE_AUTHOR("Steve Sakoman");
2414MODULE_LICENSE("GPL"); 2342MODULE_LICENSE("GPL");