diff options
Diffstat (limited to 'sound/soc/codecs/twl4030.c')
-rw-r--r-- | sound/soc/codecs/twl4030.c | 284 |
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 | ||
235 | static 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 | |||
227 | static void twl4030_codec_enable(struct snd_soc_codec *codec, int enable) | 245 | static 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 | ||
280 | static void twl4030_init_chip(struct platform_device *pdev) | 298 | static 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 | ||
719 | static void headset_ramp(struct snd_soc_codec *codec, int ramp) | 733 | static 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 | ||
1623 | static int twl4030_add_widgets(struct snd_soc_codec *codec) | 1636 | static 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 | ||
2178 | static struct snd_soc_dai_ops twl4030_dai_ops = { | 2187 | static 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 | ||
2196 | struct snd_soc_dai twl4030_dai[] = { | 2205 | static 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 | }; |
2230 | EXPORT_SYMBOL_GPL(twl4030_dai); | ||
2231 | 2239 | ||
2232 | static int twl4030_soc_suspend(struct platform_device *pdev, pm_message_t state) | 2240 | static 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 | ||
2242 | static int twl4030_soc_resume(struct platform_device *pdev) | 2246 | static 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 | ||
2251 | static struct snd_soc_codec *twl4030_codec; | 2252 | static int twl4030_soc_probe(struct snd_soc_codec *codec) |
2252 | |||
2253 | static 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 | ||
2280 | static int twl4030_soc_remove(struct platform_device *pdev) | 2274 | static 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 | ||
2285 | static 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 | |||
2294 | static int __devinit twl4030_codec_probe(struct platform_device *pdev) | 2298 | static 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 | |||
2360 | error_codec: | ||
2361 | twl4030_codec_enable(codec, 0); | ||
2362 | kfree(codec->reg_cache); | ||
2363 | error_cache: | ||
2364 | kfree(twl4030); | ||
2365 | return ret; | ||
2366 | } | 2309 | } |
2367 | 2310 | ||
2368 | static int __devexit twl4030_codec_remove(struct platform_device *pdev) | 2311 | static 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 | ||
2381 | MODULE_ALIAS("platform:twl4030_codec_audio"); | 2317 | MODULE_ALIAS("platform:twl4030-codec"); |
2382 | 2318 | ||
2383 | static struct platform_driver twl4030_codec_driver = { | 2319 | static 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 | } |
2402 | module_exit(twl4030_exit); | 2338 | module_exit(twl4030_exit); |
2403 | 2339 | ||
2404 | struct 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 | }; | ||
2410 | EXPORT_SYMBOL_GPL(soc_codec_dev_twl4030); | ||
2411 | |||
2412 | MODULE_DESCRIPTION("ASoC TWL4030 codec driver"); | 2340 | MODULE_DESCRIPTION("ASoC TWL4030 codec driver"); |
2413 | MODULE_AUTHOR("Steve Sakoman"); | 2341 | MODULE_AUTHOR("Steve Sakoman"); |
2414 | MODULE_LICENSE("GPL"); | 2342 | MODULE_LICENSE("GPL"); |