aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/codecs/wm9090.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/codecs/wm9090.c')
-rw-r--r--sound/soc/codecs/wm9090.c216
1 files changed, 79 insertions, 137 deletions
diff --git a/sound/soc/codecs/wm9090.c b/sound/soc/codecs/wm9090.c
index 1592250daec0..4de12203e611 100644
--- a/sound/soc/codecs/wm9090.c
+++ b/sound/soc/codecs/wm9090.c
@@ -28,14 +28,11 @@
28#include <linux/slab.h> 28#include <linux/slab.h>
29#include <sound/initval.h> 29#include <sound/initval.h>
30#include <sound/soc.h> 30#include <sound/soc.h>
31#include <sound/soc-dapm.h>
32#include <sound/tlv.h> 31#include <sound/tlv.h>
33#include <sound/wm9090.h> 32#include <sound/wm9090.h>
34 33
35#include "wm9090.h" 34#include "wm9090.h"
36 35
37static struct snd_soc_codec *wm9090_codec;
38
39static const u16 wm9090_reg_defaults[] = { 36static const u16 wm9090_reg_defaults[] = {
40 0x9093, /* R0 - Software Reset */ 37 0x9093, /* R0 - Software Reset */
41 0x0006, /* R1 - Power Management (1) */ 38 0x0006, /* R1 - Power Management (1) */
@@ -142,18 +139,12 @@ static const u16 wm9090_reg_defaults[] = {
142 139
143/* This struct is used to save the context */ 140/* This struct is used to save the context */
144struct wm9090_priv { 141struct wm9090_priv {
145 /* We're not really registering as a CODEC since ASoC core
146 * does not yet support multiple CODECs but having the CODEC
147 * structure means we can reuse some of the ASoC core
148 * features.
149 */
150 struct snd_soc_codec codec;
151 struct mutex mutex; 142 struct mutex mutex;
152 u16 reg_cache[WM9090_MAX_REGISTER + 1];
153 struct wm9090_platform_data pdata; 143 struct wm9090_platform_data pdata;
144 void *control_data;
154}; 145};
155 146
156static int wm9090_volatile(unsigned int reg) 147static int wm9090_volatile(struct snd_soc_codec *codec, unsigned int reg)
157{ 148{
158 switch (reg) { 149 switch (reg) {
159 case WM9090_SOFTWARE_RESET: 150 case WM9090_SOFTWARE_RESET:
@@ -450,31 +441,32 @@ static const struct snd_soc_dapm_route audio_map_in2_diff[] = {
450static int wm9090_add_controls(struct snd_soc_codec *codec) 441static int wm9090_add_controls(struct snd_soc_codec *codec)
451{ 442{
452 struct wm9090_priv *wm9090 = snd_soc_codec_get_drvdata(codec); 443 struct wm9090_priv *wm9090 = snd_soc_codec_get_drvdata(codec);
444 struct snd_soc_dapm_context *dapm = &codec->dapm;
453 int i; 445 int i;
454 446
455 snd_soc_dapm_new_controls(codec, wm9090_dapm_widgets, 447 snd_soc_dapm_new_controls(dapm, wm9090_dapm_widgets,
456 ARRAY_SIZE(wm9090_dapm_widgets)); 448 ARRAY_SIZE(wm9090_dapm_widgets));
457 449
458 snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); 450 snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map));
459 451
460 snd_soc_add_controls(codec, wm9090_controls, 452 snd_soc_add_controls(codec, wm9090_controls,
461 ARRAY_SIZE(wm9090_controls)); 453 ARRAY_SIZE(wm9090_controls));
462 454
463 if (wm9090->pdata.lin1_diff) { 455 if (wm9090->pdata.lin1_diff) {
464 snd_soc_dapm_add_routes(codec, audio_map_in1_diff, 456 snd_soc_dapm_add_routes(dapm, audio_map_in1_diff,
465 ARRAY_SIZE(audio_map_in1_diff)); 457 ARRAY_SIZE(audio_map_in1_diff));
466 } else { 458 } else {
467 snd_soc_dapm_add_routes(codec, audio_map_in1_se, 459 snd_soc_dapm_add_routes(dapm, audio_map_in1_se,
468 ARRAY_SIZE(audio_map_in1_se)); 460 ARRAY_SIZE(audio_map_in1_se));
469 snd_soc_add_controls(codec, wm9090_in1_se_controls, 461 snd_soc_add_controls(codec, wm9090_in1_se_controls,
470 ARRAY_SIZE(wm9090_in1_se_controls)); 462 ARRAY_SIZE(wm9090_in1_se_controls));
471 } 463 }
472 464
473 if (wm9090->pdata.lin2_diff) { 465 if (wm9090->pdata.lin2_diff) {
474 snd_soc_dapm_add_routes(codec, audio_map_in2_diff, 466 snd_soc_dapm_add_routes(dapm, audio_map_in2_diff,
475 ARRAY_SIZE(audio_map_in2_diff)); 467 ARRAY_SIZE(audio_map_in2_diff));
476 } else { 468 } else {
477 snd_soc_dapm_add_routes(codec, audio_map_in2_se, 469 snd_soc_dapm_add_routes(dapm, audio_map_in2_se,
478 ARRAY_SIZE(audio_map_in2_se)); 470 ARRAY_SIZE(audio_map_in2_se));
479 snd_soc_add_controls(codec, wm9090_in2_se_controls, 471 snd_soc_add_controls(codec, wm9090_in2_se_controls,
480 ARRAY_SIZE(wm9090_in2_se_controls)); 472 ARRAY_SIZE(wm9090_in2_se_controls));
@@ -521,12 +513,12 @@ static int wm9090_set_bias_level(struct snd_soc_codec *codec,
521 break; 513 break;
522 514
523 case SND_SOC_BIAS_STANDBY: 515 case SND_SOC_BIAS_STANDBY:
524 if (codec->bias_level == SND_SOC_BIAS_OFF) { 516 if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) {
525 /* Restore the register cache */ 517 /* Restore the register cache */
526 for (i = 1; i < codec->reg_cache_size; i++) { 518 for (i = 1; i < codec->driver->reg_cache_size; i++) {
527 if (reg_cache[i] == wm9090_reg_defaults[i]) 519 if (reg_cache[i] == wm9090_reg_defaults[i])
528 continue; 520 continue;
529 if (wm9090_volatile(i)) 521 if (wm9090_volatile(codec, i))
530 continue; 522 continue;
531 523
532 ret = snd_soc_write(codec, i, reg_cache[i]); 524 ret = snd_soc_write(codec, i, reg_cache[i]);
@@ -551,56 +543,80 @@ static int wm9090_set_bias_level(struct snd_soc_codec *codec,
551 break; 543 break;
552 } 544 }
553 545
554 codec->bias_level = level; 546 codec->dapm.bias_level = level;
555 547
556 return 0; 548 return 0;
557} 549}
558 550
559static int wm9090_probe(struct platform_device *pdev) 551static int wm9090_probe(struct snd_soc_codec *codec)
560{ 552{
561 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 553 struct wm9090_priv *wm9090 = snd_soc_codec_get_drvdata(codec);
562 struct snd_soc_codec *codec; 554 int ret;
563 int ret = 0;
564 555
565 if (wm9090_codec == NULL) { 556 codec->control_data = wm9090->control_data;
566 dev_err(&pdev->dev, "Codec device not registered\n"); 557 ret = snd_soc_codec_set_cache_io(codec, 8, 16, SND_SOC_I2C);
567 return -ENODEV; 558 if (ret != 0) {
559 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
560 return ret;
568 } 561 }
569 562
570 socdev->card->codec = wm9090_codec; 563 ret = snd_soc_read(codec, WM9090_SOFTWARE_RESET);
571 codec = wm9090_codec; 564 if (ret < 0)
572 565 return ret;
573 /* register pcms */ 566 if (ret != wm9090_reg_defaults[WM9090_SOFTWARE_RESET]) {
574 ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); 567 dev_err(codec->dev, "Device is not a WM9090, ID=%x\n", ret);
575 if (ret < 0) { 568 return -EINVAL;
576 dev_err(codec->dev, "failed to create pcms: %d\n", ret);
577 goto pcm_err;
578 } 569 }
579 570
571 ret = snd_soc_write(codec, WM9090_SOFTWARE_RESET, 0);
572 if (ret < 0)
573 return ret;
574
575 /* Configure some defaults; they will be written out when we
576 * bring the bias up.
577 */
578 snd_soc_update_bits(codec, WM9090_IN1_LINE_INPUT_A_VOLUME,
579 WM9090_IN1_VU | WM9090_IN1A_ZC,
580 WM9090_IN1_VU | WM9090_IN1A_ZC);
581 snd_soc_update_bits(codec, WM9090_IN1_LINE_INPUT_B_VOLUME,
582 WM9090_IN1_VU | WM9090_IN1B_ZC,
583 WM9090_IN1_VU | WM9090_IN1B_ZC);
584 snd_soc_update_bits(codec, WM9090_IN2_LINE_INPUT_A_VOLUME,
585 WM9090_IN2_VU | WM9090_IN2A_ZC,
586 WM9090_IN2_VU | WM9090_IN2A_ZC);
587 snd_soc_update_bits(codec, WM9090_IN2_LINE_INPUT_B_VOLUME,
588 WM9090_IN2_VU | WM9090_IN2B_ZC,
589 WM9090_IN2_VU | WM9090_IN2B_ZC);
590 snd_soc_update_bits(codec, WM9090_SPEAKER_VOLUME_LEFT,
591 WM9090_SPKOUT_VU | WM9090_SPKOUTL_ZC,
592 WM9090_SPKOUT_VU | WM9090_SPKOUTL_ZC);
593 snd_soc_update_bits(codec, WM9090_LEFT_OUTPUT_VOLUME,
594 WM9090_HPOUT1_VU | WM9090_HPOUT1L_ZC,
595 WM9090_HPOUT1_VU | WM9090_HPOUT1L_ZC);
596 snd_soc_update_bits(codec, WM9090_RIGHT_OUTPUT_VOLUME,
597 WM9090_HPOUT1_VU | WM9090_HPOUT1R_ZC,
598 WM9090_HPOUT1_VU | WM9090_HPOUT1R_ZC);
599
600 snd_soc_update_bits(codec, WM9090_CLOCKING_1,
601 WM9090_TOCLK_ENA, WM9090_TOCLK_ENA);
602
603 wm9090_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
604
580 wm9090_add_controls(codec); 605 wm9090_add_controls(codec);
581 606
582 return 0; 607 return 0;
583
584pcm_err:
585 return ret;
586} 608}
587 609
588#ifdef CONFIG_PM 610#ifdef CONFIG_PM
589static int wm9090_suspend(struct platform_device *pdev, pm_message_t state) 611static int wm9090_suspend(struct snd_soc_codec *codec, pm_message_t state)
590{ 612{
591 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
592 struct snd_soc_codec *codec = socdev->card->codec;
593
594 wm9090_set_bias_level(codec, SND_SOC_BIAS_OFF); 613 wm9090_set_bias_level(codec, SND_SOC_BIAS_OFF);
595 614
596 return 0; 615 return 0;
597} 616}
598 617
599static int wm9090_resume(struct platform_device *pdev) 618static int wm9090_resume(struct snd_soc_codec *codec)
600{ 619{
601 struct snd_soc_device *socdev = platform_get_drvdata(pdev);
602 struct snd_soc_codec *codec = socdev->card->codec;
603
604 wm9090_set_bias_level(codec, SND_SOC_BIAS_STANDBY); 620 wm9090_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
605 621
606 return 0; 622 return 0;
@@ -610,29 +626,29 @@ static int wm9090_resume(struct platform_device *pdev)
610#define wm9090_resume NULL 626#define wm9090_resume NULL
611#endif 627#endif
612 628
613static int wm9090_remove(struct platform_device *pdev) 629static int wm9090_remove(struct snd_soc_codec *codec)
614{ 630{
615 struct snd_soc_device *socdev = platform_get_drvdata(pdev); 631 wm9090_set_bias_level(codec, SND_SOC_BIAS_OFF);
616
617 snd_soc_free_pcms(socdev);
618 snd_soc_dapm_free(socdev);
619 632
620 return 0; 633 return 0;
621} 634}
622 635
623struct snd_soc_codec_device soc_codec_dev_wm9090 = { 636static struct snd_soc_codec_driver soc_codec_dev_wm9090 = {
624 .probe = wm9090_probe, 637 .probe = wm9090_probe,
625 .remove = wm9090_remove, 638 .remove = wm9090_remove,
626 .suspend = wm9090_suspend, 639 .suspend = wm9090_suspend,
627 .resume = wm9090_resume, 640 .resume = wm9090_resume,
641 .set_bias_level = wm9090_set_bias_level,
642 .reg_cache_size = (WM9090_MAX_REGISTER + 1),
643 .reg_word_size = sizeof(u16),
644 .reg_cache_default = wm9090_reg_defaults,
645 .volatile_register = wm9090_volatile,
628}; 646};
629EXPORT_SYMBOL_GPL(soc_codec_dev_wm9090);
630 647
631static int wm9090_i2c_probe(struct i2c_client *i2c, 648static int wm9090_i2c_probe(struct i2c_client *i2c,
632 const struct i2c_device_id *id) 649 const struct i2c_device_id *id)
633{ 650{
634 struct wm9090_priv *wm9090; 651 struct wm9090_priv *wm9090;
635 struct snd_soc_codec *codec;
636 int ret; 652 int ret;
637 653
638 wm9090 = kzalloc(sizeof(*wm9090), GFP_KERNEL); 654 wm9090 = kzalloc(sizeof(*wm9090), GFP_KERNEL);
@@ -640,102 +656,28 @@ static int wm9090_i2c_probe(struct i2c_client *i2c,
640 dev_err(&i2c->dev, "Can not allocate memory\n"); 656 dev_err(&i2c->dev, "Can not allocate memory\n");
641 return -ENOMEM; 657 return -ENOMEM;
642 } 658 }
643 codec = &wm9090->codec;
644 659
645 if (i2c->dev.platform_data) 660 if (i2c->dev.platform_data)
646 memcpy(&wm9090->pdata, i2c->dev.platform_data, 661 memcpy(&wm9090->pdata, i2c->dev.platform_data,
647 sizeof(wm9090->pdata)); 662 sizeof(wm9090->pdata));
648 663
649 wm9090_codec = codec;
650
651 i2c_set_clientdata(i2c, wm9090); 664 i2c_set_clientdata(i2c, wm9090);
665 wm9090->control_data = i2c;
666 mutex_init(&wm9090->mutex);
652 667
653 mutex_init(&codec->mutex); 668 ret = snd_soc_register_codec(&i2c->dev,
654 INIT_LIST_HEAD(&codec->dapm_widgets); 669 &soc_codec_dev_wm9090, NULL, 0);
655 INIT_LIST_HEAD(&codec->dapm_paths);
656
657 codec->control_data = i2c;
658 snd_soc_codec_set_drvdata(codec, wm9090);
659 codec->dev = &i2c->dev;
660 codec->name = "WM9090";
661 codec->owner = THIS_MODULE;
662 codec->bias_level = SND_SOC_BIAS_OFF;
663 codec->set_bias_level = wm9090_set_bias_level,
664 codec->reg_cache_size = WM9090_MAX_REGISTER + 1;
665 codec->reg_cache = &wm9090->reg_cache;
666 codec->volatile_register = wm9090_volatile;
667
668 ret = snd_soc_codec_set_cache_io(codec, 8, 16, SND_SOC_I2C);
669 if (ret != 0) {
670 dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret);
671 goto err;
672 }
673
674 memcpy(&wm9090->reg_cache, wm9090_reg_defaults,
675 sizeof(wm9090->reg_cache));
676
677 ret = snd_soc_read(codec, WM9090_SOFTWARE_RESET);
678 if (ret < 0) 670 if (ret < 0)
679 goto err; 671 kfree(wm9090);
680 if (ret != wm9090_reg_defaults[WM9090_SOFTWARE_RESET]) {
681 dev_err(&i2c->dev, "Device is not a WM9090, ID=%x\n", ret);
682 ret = -EINVAL;
683 goto err;
684 }
685
686 ret = snd_soc_write(codec, WM9090_SOFTWARE_RESET, 0);
687 if (ret < 0)
688 goto err;
689
690 /* Configure some defaults; they will be written out when we
691 * bring the bias up.
692 */
693 wm9090->reg_cache[WM9090_IN1_LINE_INPUT_A_VOLUME] |= WM9090_IN1_VU
694 | WM9090_IN1A_ZC;
695 wm9090->reg_cache[WM9090_IN1_LINE_INPUT_B_VOLUME] |= WM9090_IN1_VU
696 | WM9090_IN1B_ZC;
697 wm9090->reg_cache[WM9090_IN2_LINE_INPUT_A_VOLUME] |= WM9090_IN2_VU
698 | WM9090_IN2A_ZC;
699 wm9090->reg_cache[WM9090_IN2_LINE_INPUT_B_VOLUME] |= WM9090_IN2_VU
700 | WM9090_IN2B_ZC;
701 wm9090->reg_cache[WM9090_SPEAKER_VOLUME_LEFT] |=
702 WM9090_SPKOUT_VU | WM9090_SPKOUTL_ZC;
703 wm9090->reg_cache[WM9090_LEFT_OUTPUT_VOLUME] |=
704 WM9090_HPOUT1_VU | WM9090_HPOUT1L_ZC;
705 wm9090->reg_cache[WM9090_RIGHT_OUTPUT_VOLUME] |=
706 WM9090_HPOUT1_VU | WM9090_HPOUT1R_ZC;
707
708 wm9090->reg_cache[WM9090_CLOCKING_1] |= WM9090_TOCLK_ENA;
709
710 wm9090_set_bias_level(codec, SND_SOC_BIAS_STANDBY);
711
712 ret = snd_soc_register_codec(codec);
713 if (ret != 0) {
714 dev_err(&i2c->dev, "Failed to register CODEC: %d\n", ret);
715 goto err_bias;
716 }
717
718 return 0;
719
720err_bias:
721 wm9090_set_bias_level(codec, SND_SOC_BIAS_OFF);
722err:
723 kfree(wm9090);
724 i2c_set_clientdata(i2c, NULL);
725 wm9090_codec = NULL;
726
727 return ret; 672 return ret;
728} 673}
729 674
730static int wm9090_i2c_remove(struct i2c_client *i2c) 675static int __devexit wm9090_i2c_remove(struct i2c_client *i2c)
731{ 676{
732 struct wm9090_priv *wm9090 = i2c_get_clientdata(i2c); 677 struct wm9090_priv *wm9090 = i2c_get_clientdata(i2c);
733 struct snd_soc_codec *codec = &wm9090->codec;
734 678
735 snd_soc_unregister_codec(codec); 679 snd_soc_unregister_codec(&i2c->dev);
736 wm9090_set_bias_level(codec, SND_SOC_BIAS_OFF);
737 kfree(wm9090); 680 kfree(wm9090);
738 wm9090_codec = NULL;
739 681
740 return 0; 682 return 0;
741} 683}
@@ -748,7 +690,7 @@ MODULE_DEVICE_TABLE(i2c, wm9090_id);
748 690
749static struct i2c_driver wm9090_i2c_driver = { 691static struct i2c_driver wm9090_i2c_driver = {
750 .driver = { 692 .driver = {
751 .name = "wm9090", 693 .name = "wm9090-codec",
752 .owner = THIS_MODULE, 694 .owner = THIS_MODULE,
753 }, 695 },
754 .probe = wm9090_i2c_probe, 696 .probe = wm9090_i2c_probe,