aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm8350.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm8350.c')
-rw-r--r--sound/soc/codecs/wm8350.c343
1 files changed, 158 insertions, 185 deletions
diff --git a/sound/soc/codecs/wm8350.c b/sound/soc/codecs/wm8350.c
index 0221ca79b3ae..6d6dc9efe914 100644
--- a/sound/soc/codecs/wm8350.c
+++ b/sound/soc/codecs/wm8350.c
@@ -24,9 +24,9 @@
24#include <sound/pcm.h> 24#include <sound/pcm.h>
25#include <sound/pcm_params.h> 25#include <sound/pcm_params.h>
26#include <sound/soc.h> 26#include <sound/soc.h>
27#include <sound/soc-dapm.h>
28#include <sound/initval.h> 27#include <sound/initval.h>
29#include <sound/tlv.h> 28#include <sound/tlv.h>
29#include <trace/events/asoc.h>
30 30
31#include "wm8350.h" 31#include "wm8350.h"
32 32
@@ -54,6 +54,7 @@ struct wm8350_output {
54 54
55struct wm8350_jack_data { 55struct wm8350_jack_data {
56 struct snd_soc_jack *jack; 56 struct snd_soc_jack *jack;
57 struct delayed_work work;
57 int report; 58 int report;
58 int short_report; 59 int short_report;
59}; 60};
@@ -230,8 +231,9 @@ static inline int wm8350_out2_ramp_step(struct snd_soc_codec *codec)
230 */ 231 */
231static void wm8350_pga_work(struct work_struct *work) 232static void wm8350_pga_work(struct work_struct *work)
232{ 233{
233 struct snd_soc_codec *codec = 234 struct snd_soc_dapm_context *dapm =
234 container_of(work, struct snd_soc_codec, delayed_work.work); 235 container_of(work, struct snd_soc_dapm_context, delayed_work.work);
236 struct snd_soc_codec *codec = dapm->codec;
235 struct wm8350_data *wm8350_data = snd_soc_codec_get_drvdata(codec); 237 struct wm8350_data *wm8350_data = snd_soc_codec_get_drvdata(codec);
236 struct wm8350_output *out1 = &wm8350_data->out1, 238 struct wm8350_output *out1 = &wm8350_data->out1,
237 *out2 = &wm8350_data->out2; 239 *out2 = &wm8350_data->out2;
@@ -302,8 +304,8 @@ static int pga_event(struct snd_soc_dapm_widget *w,
302 out->ramp = WM8350_RAMP_UP; 304 out->ramp = WM8350_RAMP_UP;
303 out->active = 1; 305 out->active = 1;
304 306
305 if (!delayed_work_pending(&codec->delayed_work)) 307 if (!delayed_work_pending(&codec->dapm.delayed_work))
306 schedule_delayed_work(&codec->delayed_work, 308 schedule_delayed_work(&codec->dapm.delayed_work,
307 msecs_to_jiffies(1)); 309 msecs_to_jiffies(1));
308 break; 310 break;
309 311
@@ -311,8 +313,8 @@ static int pga_event(struct snd_soc_dapm_widget *w,
311 out->ramp = WM8350_RAMP_DOWN; 313 out->ramp = WM8350_RAMP_DOWN;
312 out->active = 0; 314 out->active = 0;
313 315
314 if (!delayed_work_pending(&codec->delayed_work)) 316 if (!delayed_work_pending(&codec->dapm.delayed_work))
315 schedule_delayed_work(&codec->delayed_work, 317 schedule_delayed_work(&codec->dapm.delayed_work,
316 msecs_to_jiffies(1)); 318 msecs_to_jiffies(1));
317 break; 319 break;
318 } 320 }
@@ -786,9 +788,10 @@ static const struct snd_soc_dapm_route audio_map[] = {
786 788
787static int wm8350_add_widgets(struct snd_soc_codec *codec) 789static int wm8350_add_widgets(struct snd_soc_codec *codec)
788{ 790{
791 struct snd_soc_dapm_context *dapm = &codec->dapm;
789 int ret; 792 int ret;
790 793
791 ret = snd_soc_dapm_new_controls(codec, 794 ret = snd_soc_dapm_new_controls(dapm,
792 wm8350_dapm_widgets, 795 wm8350_dapm_widgets,
793 ARRAY_SIZE(wm8350_dapm_widgets)); 796 ARRAY_SIZE(wm8350_dapm_widgets));
794 if (ret != 0) { 797 if (ret != 0) {
@@ -797,7 +800,7 @@ static int wm8350_add_widgets(struct snd_soc_codec *codec)
797 } 800 }
798 801
799 /* set up audio paths */ 802 /* set up audio paths */
800 ret = snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); 803 ret = snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map));
801 if (ret != 0) { 804 if (ret != 0) {
802 dev_err(codec->dev, "DAPM route register failed\n"); 805 dev_err(codec->dev, "DAPM route register failed\n");
803 return ret; 806 return ret;
@@ -831,7 +834,7 @@ static int wm8350_set_dai_sysclk(struct snd_soc_dai *codec_dai,
831 } 834 }
832 835
833 /* MCLK direction */ 836 /* MCLK direction */
834 if (dir == WM8350_MCLK_DIR_OUT) 837 if (dir == SND_SOC_CLOCK_OUT)
835 wm8350_set_bits(wm8350, WM8350_CLOCK_CONTROL_2, 838 wm8350_set_bits(wm8350, WM8350_CLOCK_CONTROL_2,
836 WM8350_MCLK_DIR); 839 WM8350_MCLK_DIR);
837 else 840 else
@@ -1184,7 +1187,7 @@ static int wm8350_set_bias_level(struct snd_soc_codec *codec,
1184 break; 1187 break;
1185 1188
1186 case SND_SOC_BIAS_STANDBY: 1189 case SND_SOC_BIAS_STANDBY:
1187 if (codec->bias_level == SND_SOC_BIAS_OFF) { 1190 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
1188 ret = regulator_bulk_enable(ARRAY_SIZE(priv->supplies), 1191 ret = regulator_bulk_enable(ARRAY_SIZE(priv->supplies),
1189 priv->supplies); 1192 priv->supplies);
1190 if (ret != 0) 1193 if (ret != 0)
@@ -1317,68 +1320,86 @@ static int wm8350_set_bias_level(struct snd_soc_codec *codec,
1317 priv->supplies); 1320 priv->supplies);
1318 break; 1321 break;
1319 } 1322 }
1320 codec->bias_level = level; 1323 codec->dapm.bias_level = level;
1321 return 0; 1324 return 0;
1322} 1325}
1323 1326
1324static int wm8350_suspend(struct platform_device *pdev, pm_message_t state) 1327static int wm8350_suspend(struct snd_soc_codec *codec, pm_message_t state)
1325{ 1328{
1326 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1327 struct snd_soc_codec *codec = socdev->card->codec;
1328
1329 wm8350_set_bias_level(codec, SND_SOC_BIAS_OFF); 1329 wm8350_set_bias_level(codec, SND_SOC_BIAS_OFF);
1330 return 0; 1330 return 0;
1331} 1331}
1332 1332
1333static int wm8350_resume(struct platform_device *pdev) 1333static int wm8350_resume(struct snd_soc_codec *codec)
1334{ 1334{
1335 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1336 struct snd_soc_codec *codec = socdev->card->codec;
1337
1338 wm8350_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 1335 wm8350_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1339 1336
1340 return 0; 1337 return 0;
1341} 1338}
1342 1339
1343static irqreturn_t wm8350_hp_jack_handler(int irq, void *data) 1340static void wm8350_hp_work(struct wm8350_data *priv,
1341 struct wm8350_jack_data *jack,
1342 u16 mask)
1344{ 1343{
1345 struct wm8350_data *priv = data;
1346 struct wm8350 *wm8350 = priv->codec.control_data; 1344 struct wm8350 *wm8350 = priv->codec.control_data;
1347 u16 reg; 1345 u16 reg;
1348 int report; 1346 int report;
1349 int mask; 1347
1348 reg = wm8350_reg_read(wm8350, WM8350_JACK_PIN_STATUS);
1349 if (reg & mask)
1350 report = jack->report;
1351 else
1352 report = 0;
1353
1354 snd_soc_jack_report(jack->jack, report, jack->report);
1355
1356}
1357
1358static void wm8350_hpl_work(struct work_struct *work)
1359{
1360 struct wm8350_data *priv =
1361 container_of(work, struct wm8350_data, hpl.work.work);
1362
1363 wm8350_hp_work(priv, &priv->hpl, WM8350_JACK_L_LVL);
1364}
1365
1366static void wm8350_hpr_work(struct work_struct *work)
1367{
1368 struct wm8350_data *priv =
1369 container_of(work, struct wm8350_data, hpr.work.work);
1370
1371 wm8350_hp_work(priv, &priv->hpr, WM8350_JACK_R_LVL);
1372}
1373
1374static irqreturn_t wm8350_hp_jack_handler(int irq, void *data)
1375{
1376 struct wm8350_data *priv = data;
1377 struct wm8350 *wm8350 = priv->codec.control_data;
1350 struct wm8350_jack_data *jack = NULL; 1378 struct wm8350_jack_data *jack = NULL;
1351 1379
1352 switch (irq - wm8350->irq_base) { 1380 switch (irq - wm8350->irq_base) {
1353 case WM8350_IRQ_CODEC_JCK_DET_L: 1381 case WM8350_IRQ_CODEC_JCK_DET_L:
1382#ifndef CONFIG_SND_SOC_WM8350_MODULE
1383 trace_snd_soc_jack_irq("WM8350 HPL");
1384#endif
1354 jack = &priv->hpl; 1385 jack = &priv->hpl;
1355 mask = WM8350_JACK_L_LVL;
1356 break; 1386 break;
1357 1387
1358 case WM8350_IRQ_CODEC_JCK_DET_R: 1388 case WM8350_IRQ_CODEC_JCK_DET_R:
1389#ifndef CONFIG_SND_SOC_WM8350_MODULE
1390 trace_snd_soc_jack_irq("WM8350 HPR");
1391#endif
1359 jack = &priv->hpr; 1392 jack = &priv->hpr;
1360 mask = WM8350_JACK_R_LVL;
1361 break; 1393 break;
1362 1394
1363 default: 1395 default:
1364 BUG(); 1396 BUG();
1365 } 1397 }
1366 1398
1367 if (!jack->jack) { 1399 if (device_may_wakeup(wm8350->dev))
1368 dev_warn(wm8350->dev, "Jack interrupt called with no jack\n"); 1400 pm_wakeup_event(wm8350->dev, 250);
1369 return IRQ_NONE;
1370 }
1371
1372 /* Debounce */
1373 msleep(200);
1374
1375 reg = wm8350_reg_read(wm8350, WM8350_JACK_PIN_STATUS);
1376 if (reg & mask)
1377 report = jack->report;
1378 else
1379 report = 0;
1380 1401
1381 snd_soc_jack_report(jack->jack, report, jack->report); 1402 schedule_delayed_work(&jack->work, 200);
1382 1403
1383 return IRQ_HANDLED; 1404 return IRQ_HANDLED;
1384} 1405}
@@ -1442,6 +1463,10 @@ static irqreturn_t wm8350_mic_handler(int irq, void *data)
1442 u16 reg; 1463 u16 reg;
1443 int report = 0; 1464 int report = 0;
1444 1465
1466#ifndef CONFIG_SND_SOC_WM8350_MODULE
1467 trace_snd_soc_jack_irq("WM8350 mic");
1468#endif
1469
1445 reg = wm8350_reg_read(wm8350, WM8350_JACK_PIN_STATUS); 1470 reg = wm8350_reg_read(wm8350, WM8350_JACK_PIN_STATUS);
1446 if (reg & WM8350_JACK_MICSCD_LVL) 1471 if (reg & WM8350_JACK_MICSCD_LVL)
1447 report |= priv->mic.short_report; 1472 report |= priv->mic.short_report;
@@ -1489,24 +1514,76 @@ int wm8350_mic_jack_detect(struct snd_soc_codec *codec,
1489} 1514}
1490EXPORT_SYMBOL_GPL(wm8350_mic_jack_detect); 1515EXPORT_SYMBOL_GPL(wm8350_mic_jack_detect);
1491 1516
1492static struct snd_soc_codec *wm8350_codec; 1517#define WM8350_RATES (SNDRV_PCM_RATE_8000_96000)
1518
1519#define WM8350_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\
1520 SNDRV_PCM_FMTBIT_S20_3LE |\
1521 SNDRV_PCM_FMTBIT_S24_LE)
1522
1523static struct snd_soc_dai_ops wm8350_dai_ops = {
1524 .hw_params = wm8350_pcm_hw_params,
1525 .digital_mute = wm8350_mute,
1526 .trigger = wm8350_pcm_trigger,
1527 .set_fmt = wm8350_set_dai_fmt,
1528 .set_sysclk = wm8350_set_dai_sysclk,
1529 .set_pll = wm8350_set_fll,
1530 .set_clkdiv = wm8350_set_clkdiv,
1531};
1532
1533static struct snd_soc_dai_driver wm8350_dai = {
1534 .name = "wm8350-hifi",
1535 .playback = {
1536 .stream_name = "Playback",
1537 .channels_min = 1,
1538 .channels_max = 2,
1539 .rates = WM8350_RATES,
1540 .formats = WM8350_FORMATS,
1541 },
1542 .capture = {
1543 .stream_name = "Capture",
1544 .channels_min = 1,
1545 .channels_max = 2,
1546 .rates = WM8350_RATES,
1547 .formats = WM8350_FORMATS,
1548 },
1549 .ops = &wm8350_dai_ops,
1550};
1493 1551
1494static int wm8350_probe(struct platform_device *pdev) 1552static int wm8350_codec_probe(struct snd_soc_codec *codec)
1495{ 1553{
1496 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 1554 struct wm8350 *wm8350 = dev_get_platdata(codec->dev);
1497 struct snd_soc_codec *codec;
1498 struct wm8350 *wm8350;
1499 struct wm8350_data *priv; 1555 struct wm8350_data *priv;
1500 int ret;
1501 struct wm8350_output *out1; 1556 struct wm8350_output *out1;
1502 struct wm8350_output *out2; 1557 struct wm8350_output *out2;
1558 int ret, i;
1559
1560 if (wm8350->codec.platform_data == NULL) {
1561 dev_err(codec->dev, "No audio platform data supplied\n");
1562 return -EINVAL;
1563 }
1564
1565 priv = kzalloc(sizeof(struct wm8350_data), GFP_KERNEL);
1566 if (priv == NULL)
1567 return -ENOMEM;
1568 snd_soc_codec_set_drvdata(codec, priv);
1503 1569
1504 BUG_ON(!wm8350_codec); 1570 for (i = 0; i < ARRAY_SIZE(supply_names); i++)
1571 priv->supplies[i].supply = supply_names[i];
1505 1572
1506 socdev->card->codec = wm8350_codec; 1573 ret = regulator_bulk_get(wm8350->dev, ARRAY_SIZE(priv->supplies),
1507 codec = socdev->card->codec; 1574 priv->supplies);
1508 wm8350 = codec->control_data; 1575 if (ret != 0)
1509 priv = snd_soc_codec_get_drvdata(codec); 1576 goto err_priv;
1577
1578 wm8350->codec.codec = codec;
1579 codec->control_data = wm8350;
1580
1581 /* Put the codec into reset if it wasn't already */
1582 wm8350_clear_bits(wm8350, WM8350_POWER_MGMT_5, WM8350_CODEC_ENA);
1583
1584 INIT_DELAYED_WORK(&codec->dapm.delayed_work, wm8350_pga_work);
1585 INIT_DELAYED_WORK(&priv->hpl.work, wm8350_hpl_work);
1586 INIT_DELAYED_WORK(&priv->hpr.work, wm8350_hpr_work);
1510 1587
1511 /* Enable the codec */ 1588 /* Enable the codec */
1512 wm8350_set_bits(wm8350, WM8350_POWER_MGMT_5, WM8350_CODEC_ENA); 1589 wm8350_set_bits(wm8350, WM8350_POWER_MGMT_5, WM8350_CODEC_ENA);
@@ -1542,6 +1619,13 @@ static int wm8350_probe(struct platform_device *pdev)
1542 wm8350_set_bits(wm8350, WM8350_ROUT2_VOLUME, 1619 wm8350_set_bits(wm8350, WM8350_ROUT2_VOLUME,
1543 WM8350_OUT2_VU | WM8350_OUT2R_MUTE); 1620 WM8350_OUT2_VU | WM8350_OUT2R_MUTE);
1544 1621
1622 /* Make sure AIF tristating is disabled by default */
1623 wm8350_clear_bits(wm8350, WM8350_AI_FORMATING, WM8350_AIF_TRI);
1624
1625 /* Make sure we've got a sane companding setup too */
1626 wm8350_clear_bits(wm8350, WM8350_ADC_DAC_COMP,
1627 WM8350_DAC_COMP | WM8350_LOOPBACK);
1628
1545 /* Make sure jack detect is disabled to start off with */ 1629 /* Make sure jack detect is disabled to start off with */
1546 wm8350_clear_bits(wm8350, WM8350_JACK_DETECT, 1630 wm8350_clear_bits(wm8350, WM8350_JACK_DETECT,
1547 WM8350_JDL_ENA | WM8350_JDR_ENA); 1631 WM8350_JDL_ENA | WM8350_JDR_ENA);
@@ -1557,11 +1641,6 @@ static int wm8350_probe(struct platform_device *pdev)
1557 wm8350_register_irq(wm8350, WM8350_IRQ_CODEC_MICD, 1641 wm8350_register_irq(wm8350, WM8350_IRQ_CODEC_MICD,
1558 wm8350_mic_handler, 0, "Microphone detect", priv); 1642 wm8350_mic_handler, 0, "Microphone detect", priv);
1559 1643
1560 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1);
1561 if (ret < 0) {
1562 dev_err(&pdev->dev, "failed to create pcms\n");
1563 return ret;
1564 }
1565 1644
1566 snd_soc_add_controls(codec, wm8350_snd_controls, 1645 snd_soc_add_controls(codec, wm8350_snd_controls,
1567 ARRAY_SIZE(wm8350_snd_controls)); 1646 ARRAY_SIZE(wm8350_snd_controls));
@@ -1570,15 +1649,16 @@ static int wm8350_probe(struct platform_device *pdev)
1570 wm8350_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 1649 wm8350_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
1571 1650
1572 return 0; 1651 return 0;
1652
1653err_priv:
1654 kfree(priv);
1655 return ret;
1573} 1656}
1574 1657
1575static int wm8350_remove(struct platform_device *pdev) 1658static int wm8350_codec_remove(struct snd_soc_codec *codec)
1576{ 1659{
1577 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
1578 struct snd_soc_codec *codec = socdev->card->codec;
1579 struct wm8350 *wm8350 = codec->control_data;
1580 struct wm8350_data *priv = snd_soc_codec_get_drvdata(codec); 1660 struct wm8350_data *priv = snd_soc_codec_get_drvdata(codec);
1581 int ret; 1661 struct wm8350 *wm8350 = dev_get_platdata(codec->dev);
1582 1662
1583 wm8350_clear_bits(wm8350, WM8350_JACK_DETECT, 1663 wm8350_clear_bits(wm8350, WM8350_JACK_DETECT,
1584 WM8350_JDL_ENA | WM8350_JDR_ENA); 1664 WM8350_JDL_ENA | WM8350_JDR_ENA);
@@ -1593,148 +1673,41 @@ static int wm8350_remove(struct platform_device *pdev)
1593 priv->hpr.jack = NULL; 1673 priv->hpr.jack = NULL;
1594 priv->mic.jack = NULL; 1674 priv->mic.jack = NULL;
1595 1675
1596 /* cancel any work waiting to be queued. */ 1676 cancel_delayed_work_sync(&priv->hpl.work);
1597 ret = cancel_delayed_work(&codec->delayed_work); 1677 cancel_delayed_work_sync(&priv->hpr.work);
1598 1678
1599 /* if there was any work waiting then we run it now and 1679 /* if there was any work waiting then we run it now and
1600 * wait for its completion */ 1680 * wait for its completion */
1601 if (ret) { 1681 flush_delayed_work_sync(&codec->dapm.delayed_work);
1602 schedule_delayed_work(&codec->delayed_work, 0);
1603 flush_scheduled_work();
1604 }
1605 1682
1606 wm8350_set_bias_level(codec, SND_SOC_BIAS_OFF); 1683 wm8350_set_bias_level(codec, SND_SOC_BIAS_OFF);
1607 1684
1608 wm8350_clear_bits(wm8350, WM8350_POWER_MGMT_5, WM8350_CODEC_ENA); 1685 wm8350_clear_bits(wm8350, WM8350_POWER_MGMT_5, WM8350_CODEC_ENA);
1609 1686
1687 regulator_bulk_free(ARRAY_SIZE(priv->supplies), priv->supplies);
1688 kfree(priv);
1610 return 0; 1689 return 0;
1611} 1690}
1612 1691
1613#define WM8350_RATES (SNDRV_PCM_RATE_8000_96000) 1692static struct snd_soc_codec_driver soc_codec_dev_wm8350 = {
1614 1693 .probe = wm8350_codec_probe,
1615#define WM8350_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\ 1694 .remove = wm8350_codec_remove,
1616 SNDRV_PCM_FMTBIT_S20_3LE |\
1617 SNDRV_PCM_FMTBIT_S24_LE)
1618
1619static struct snd_soc_dai_ops wm8350_dai_ops = {
1620 .hw_params = wm8350_pcm_hw_params,
1621 .digital_mute = wm8350_mute,
1622 .trigger = wm8350_pcm_trigger,
1623 .set_fmt = wm8350_set_dai_fmt,
1624 .set_sysclk = wm8350_set_dai_sysclk,
1625 .set_pll = wm8350_set_fll,
1626 .set_clkdiv = wm8350_set_clkdiv,
1627};
1628
1629struct snd_soc_dai wm8350_dai = {
1630 .name = "WM8350",
1631 .playback = {
1632 .stream_name = "Playback",
1633 .channels_min = 1,
1634 .channels_max = 2,
1635 .rates = WM8350_RATES,
1636 .formats = WM8350_FORMATS,
1637 },
1638 .capture = {
1639 .stream_name = "Capture",
1640 .channels_min = 1,
1641 .channels_max = 2,
1642 .rates = WM8350_RATES,
1643 .formats = WM8350_FORMATS,
1644 },
1645 .ops = &wm8350_dai_ops,
1646};
1647EXPORT_SYMBOL_GPL(wm8350_dai);
1648
1649struct snd_soc_codec_device soc_codec_dev_wm8350 = {
1650 .probe = wm8350_probe,
1651 .remove = wm8350_remove,
1652 .suspend = wm8350_suspend, 1695 .suspend = wm8350_suspend,
1653 .resume = wm8350_resume, 1696 .resume = wm8350_resume,
1697 .read = wm8350_codec_read,
1698 .write = wm8350_codec_write,
1699 .set_bias_level = wm8350_set_bias_level,
1654}; 1700};
1655EXPORT_SYMBOL_GPL(soc_codec_dev_wm8350);
1656 1701
1657static __devinit int wm8350_codec_probe(struct platform_device *pdev) 1702static int __devinit wm8350_probe(struct platform_device *pdev)
1658{ 1703{
1659 struct wm8350 *wm8350 = platform_get_drvdata(pdev); 1704 return snd_soc_register_codec(&pdev->dev, &soc_codec_dev_wm8350,
1660 struct wm8350_data *priv; 1705 &wm8350_dai, 1);
1661 struct snd_soc_codec *codec;
1662 int ret, i;
1663
1664 if (wm8350->codec.platform_data == NULL) {
1665 dev_err(&pdev->dev, "No audio platform data supplied\n");
1666 return -EINVAL;
1667 }
1668
1669 priv = kzalloc(sizeof(struct wm8350_data), GFP_KERNEL);
1670 if (priv == NULL)
1671 return -ENOMEM;
1672
1673 for (i = 0; i < ARRAY_SIZE(supply_names); i++)
1674 priv->supplies[i].supply = supply_names[i];
1675
1676 ret = regulator_bulk_get(wm8350->dev, ARRAY_SIZE(priv->supplies),
1677 priv->supplies);
1678 if (ret != 0)
1679 goto err_priv;
1680
1681 codec = &priv->codec;
1682 wm8350->codec.codec = codec;
1683
1684 wm8350_dai.dev = &pdev->dev;
1685
1686 mutex_init(&codec->mutex);
1687 INIT_LIST_HEAD(&codec->dapm_widgets);
1688 INIT_LIST_HEAD(&codec->dapm_paths);
1689 codec->dev = &pdev->dev;
1690 codec->name = "WM8350";
1691 codec->owner = THIS_MODULE;
1692 codec->read = wm8350_codec_read;
1693 codec->write = wm8350_codec_write;
1694 codec->bias_level = SND_SOC_BIAS_OFF;
1695 codec->set_bias_level = wm8350_set_bias_level;
1696 codec->dai = &wm8350_dai;
1697 codec->num_dai = 1;
1698 codec->reg_cache_size = WM8350_MAX_REGISTER;
1699 snd_soc_codec_set_drvdata(codec, priv);
1700 codec->control_data = wm8350;
1701
1702 /* Put the codec into reset if it wasn't already */
1703 wm8350_clear_bits(wm8350, WM8350_POWER_MGMT_5, WM8350_CODEC_ENA);
1704
1705 INIT_DELAYED_WORK(&codec->delayed_work, wm8350_pga_work);
1706 ret = snd_soc_register_codec(codec);
1707 if (ret != 0)
1708 goto err_supply;
1709
1710 wm8350_codec = codec;
1711
1712 ret = snd_soc_register_dai(&wm8350_dai);
1713 if (ret != 0)
1714 goto err_codec;
1715 return 0;
1716
1717err_codec:
1718 snd_soc_unregister_codec(codec);
1719err_supply:
1720 regulator_bulk_free(ARRAY_SIZE(priv->supplies), priv->supplies);
1721err_priv:
1722 kfree(priv);
1723 wm8350_codec = NULL;
1724 return ret;
1725} 1706}
1726 1707
1727static int __devexit wm8350_codec_remove(struct platform_device *pdev) 1708static int __devexit wm8350_remove(struct platform_device *pdev)
1728{ 1709{
1729 struct wm8350 *wm8350 = platform_get_drvdata(pdev); 1710 snd_soc_unregister_codec(&pdev->dev);
1730 struct snd_soc_codec *codec = wm8350->codec.codec;
1731 struct wm8350_data *priv = snd_soc_codec_get_drvdata(codec);
1732
1733 snd_soc_unregister_dai(&wm8350_dai);
1734 snd_soc_unregister_codec(codec);
1735 regulator_bulk_free(ARRAY_SIZE(priv->supplies), priv->supplies);
1736 kfree(priv);
1737 wm8350_codec = NULL;
1738 return 0; 1711 return 0;
1739} 1712}
1740 1713
@@ -1743,8 +1716,8 @@ static struct platform_driver wm8350_codec_driver = {
1743 .name = "wm8350-codec", 1716 .name = "wm8350-codec",
1744 .owner = THIS_MODULE, 1717 .owner = THIS_MODULE,
1745 }, 1718 },
1746 .probe = wm8350_codec_probe, 1719 .probe = wm8350_probe,
1747 .remove = __devexit_p(wm8350_codec_remove), 1720 .remove = __devexit_p(wm8350_remove),
1748}; 1721};
1749 1722
1750static __init int wm8350_init(void) 1723static __init int wm8350_init(void)