diff options
Diffstat (limited to 'sound/soc/codecs/wm8350.c')
-rw-r--r-- | sound/soc/codecs/wm8350.c | 343 |
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 | ||
55 | struct wm8350_jack_data { | 55 | struct 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 | */ |
231 | static void wm8350_pga_work(struct work_struct *work) | 232 | static 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 | ||
787 | static int wm8350_add_widgets(struct snd_soc_codec *codec) | 789 | static 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 | ||
1324 | static int wm8350_suspend(struct platform_device *pdev, pm_message_t state) | 1327 | static 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 | ||
1333 | static int wm8350_resume(struct platform_device *pdev) | 1333 | static 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 | ||
1343 | static irqreturn_t wm8350_hp_jack_handler(int irq, void *data) | 1340 | static 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 | |||
1358 | static 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 | |||
1366 | static 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 | |||
1374 | static 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 | } |
1490 | EXPORT_SYMBOL_GPL(wm8350_mic_jack_detect); | 1515 | EXPORT_SYMBOL_GPL(wm8350_mic_jack_detect); |
1491 | 1516 | ||
1492 | static 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 | |||
1523 | static 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 | |||
1533 | static 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 | ||
1494 | static int wm8350_probe(struct platform_device *pdev) | 1552 | static 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 | |||
1653 | err_priv: | ||
1654 | kfree(priv); | ||
1655 | return ret; | ||
1573 | } | 1656 | } |
1574 | 1657 | ||
1575 | static int wm8350_remove(struct platform_device *pdev) | 1658 | static 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) | 1692 | static 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 | |||
1619 | static 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 | |||
1629 | struct 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 | }; | ||
1647 | EXPORT_SYMBOL_GPL(wm8350_dai); | ||
1648 | |||
1649 | struct 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 | }; |
1655 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm8350); | ||
1656 | 1701 | ||
1657 | static __devinit int wm8350_codec_probe(struct platform_device *pdev) | 1702 | static 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 | |||
1717 | err_codec: | ||
1718 | snd_soc_unregister_codec(codec); | ||
1719 | err_supply: | ||
1720 | regulator_bulk_free(ARRAY_SIZE(priv->supplies), priv->supplies); | ||
1721 | err_priv: | ||
1722 | kfree(priv); | ||
1723 | wm8350_codec = NULL; | ||
1724 | return ret; | ||
1725 | } | 1706 | } |
1726 | 1707 | ||
1727 | static int __devexit wm8350_codec_remove(struct platform_device *pdev) | 1708 | static 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 | ||
1750 | static __init int wm8350_init(void) | 1723 | static __init int wm8350_init(void) |