diff options
Diffstat (limited to 'sound/soc/codecs/wm8974.c')
-rw-r--r-- | sound/soc/codecs/wm8974.c | 180 |
1 files changed, 49 insertions, 131 deletions
diff --git a/sound/soc/codecs/wm8974.c b/sound/soc/codecs/wm8974.c index 1468fe10cbbe..ca646a822444 100644 --- a/sound/soc/codecs/wm8974.c +++ b/sound/soc/codecs/wm8974.c | |||
@@ -23,7 +23,6 @@ | |||
23 | #include <sound/pcm.h> | 23 | #include <sound/pcm.h> |
24 | #include <sound/pcm_params.h> | 24 | #include <sound/pcm_params.h> |
25 | #include <sound/soc.h> | 25 | #include <sound/soc.h> |
26 | #include <sound/soc-dapm.h> | ||
27 | #include <sound/initval.h> | 26 | #include <sound/initval.h> |
28 | #include <sound/tlv.h> | 27 | #include <sound/tlv.h> |
29 | 28 | ||
@@ -51,12 +50,9 @@ static const u16 wm8974_reg[WM8974_CACHEREGNUM] = { | |||
51 | #define WM8974_POWER1_BUFIOEN 0x04 | 50 | #define WM8974_POWER1_BUFIOEN 0x04 |
52 | 51 | ||
53 | struct wm8974_priv { | 52 | struct wm8974_priv { |
54 | struct snd_soc_codec codec; | 53 | enum snd_soc_control_type control_type; |
55 | u16 reg_cache[WM8974_CACHEREGNUM]; | ||
56 | }; | 54 | }; |
57 | 55 | ||
58 | static struct snd_soc_codec *wm8974_codec; | ||
59 | |||
60 | #define wm8974_reset(c) snd_soc_write(c, WM8974_RESET, 0) | 56 | #define wm8974_reset(c) snd_soc_write(c, WM8974_RESET, 0) |
61 | 57 | ||
62 | static const char *wm8974_companding[] = {"Off", "NC", "u-law", "A-law" }; | 58 | static const char *wm8974_companding[] = {"Off", "NC", "u-law", "A-law" }; |
@@ -276,10 +272,11 @@ static const struct snd_soc_dapm_route audio_map[] = { | |||
276 | 272 | ||
277 | static int wm8974_add_widgets(struct snd_soc_codec *codec) | 273 | static int wm8974_add_widgets(struct snd_soc_codec *codec) |
278 | { | 274 | { |
279 | snd_soc_dapm_new_controls(codec, wm8974_dapm_widgets, | 275 | struct snd_soc_dapm_context *dapm = &codec->dapm; |
280 | ARRAY_SIZE(wm8974_dapm_widgets)); | ||
281 | 276 | ||
282 | snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); | 277 | snd_soc_dapm_new_controls(dapm, wm8974_dapm_widgets, |
278 | ARRAY_SIZE(wm8974_dapm_widgets)); | ||
279 | snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map)); | ||
283 | 280 | ||
284 | return 0; | 281 | return 0; |
285 | } | 282 | } |
@@ -532,7 +529,7 @@ static int wm8974_set_bias_level(struct snd_soc_codec *codec, | |||
532 | case SND_SOC_BIAS_STANDBY: | 529 | case SND_SOC_BIAS_STANDBY: |
533 | power1 |= WM8974_POWER1_BIASEN | WM8974_POWER1_BUFIOEN; | 530 | power1 |= WM8974_POWER1_BIASEN | WM8974_POWER1_BUFIOEN; |
534 | 531 | ||
535 | if (codec->bias_level == SND_SOC_BIAS_OFF) { | 532 | if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) { |
536 | /* Initial cap charge at VMID 5k */ | 533 | /* Initial cap charge at VMID 5k */ |
537 | snd_soc_write(codec, WM8974_POWER1, power1 | 0x3); | 534 | snd_soc_write(codec, WM8974_POWER1, power1 | 0x3); |
538 | mdelay(100); | 535 | mdelay(100); |
@@ -549,7 +546,7 @@ static int wm8974_set_bias_level(struct snd_soc_codec *codec, | |||
549 | break; | 546 | break; |
550 | } | 547 | } |
551 | 548 | ||
552 | codec->bias_level = level; | 549 | codec->dapm.bias_level = level; |
553 | return 0; | 550 | return 0; |
554 | } | 551 | } |
555 | 552 | ||
@@ -566,8 +563,8 @@ static struct snd_soc_dai_ops wm8974_ops = { | |||
566 | .set_pll = wm8974_set_dai_pll, | 563 | .set_pll = wm8974_set_dai_pll, |
567 | }; | 564 | }; |
568 | 565 | ||
569 | struct snd_soc_dai wm8974_dai = { | 566 | static struct snd_soc_dai_driver wm8974_dai = { |
570 | .name = "WM8974 HiFi", | 567 | .name = "wm8974-hifi", |
571 | .playback = { | 568 | .playback = { |
572 | .stream_name = "Playback", | 569 | .stream_name = "Playback", |
573 | .channels_min = 1, | 570 | .channels_min = 1, |
@@ -583,21 +580,15 @@ struct snd_soc_dai wm8974_dai = { | |||
583 | .ops = &wm8974_ops, | 580 | .ops = &wm8974_ops, |
584 | .symmetric_rates = 1, | 581 | .symmetric_rates = 1, |
585 | }; | 582 | }; |
586 | EXPORT_SYMBOL_GPL(wm8974_dai); | ||
587 | 583 | ||
588 | static int wm8974_suspend(struct platform_device *pdev, pm_message_t state) | 584 | static int wm8974_suspend(struct snd_soc_codec *codec, pm_message_t state) |
589 | { | 585 | { |
590 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
591 | struct snd_soc_codec *codec = socdev->card->codec; | ||
592 | |||
593 | wm8974_set_bias_level(codec, SND_SOC_BIAS_OFF); | 586 | wm8974_set_bias_level(codec, SND_SOC_BIAS_OFF); |
594 | return 0; | 587 | return 0; |
595 | } | 588 | } |
596 | 589 | ||
597 | static int wm8974_resume(struct platform_device *pdev) | 590 | static int wm8974_resume(struct snd_soc_codec *codec) |
598 | { | 591 | { |
599 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
600 | struct snd_soc_codec *codec = socdev->card->codec; | ||
601 | int i; | 592 | int i; |
602 | u8 data[2]; | 593 | u8 data[2]; |
603 | u16 *cache = codec->reg_cache; | 594 | u16 *cache = codec->reg_cache; |
@@ -613,156 +604,72 @@ static int wm8974_resume(struct platform_device *pdev) | |||
613 | return 0; | 604 | return 0; |
614 | } | 605 | } |
615 | 606 | ||
616 | static int wm8974_probe(struct platform_device *pdev) | 607 | static int wm8974_probe(struct snd_soc_codec *codec) |
617 | { | 608 | { |
618 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
619 | struct snd_soc_codec *codec; | ||
620 | int ret = 0; | 609 | int ret = 0; |
621 | 610 | ||
622 | if (wm8974_codec == NULL) { | 611 | ret = snd_soc_codec_set_cache_io(codec, 7, 9, SND_SOC_I2C); |
623 | dev_err(&pdev->dev, "Codec device not registered\n"); | 612 | if (ret < 0) { |
624 | return -ENODEV; | 613 | dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); |
614 | return ret; | ||
625 | } | 615 | } |
626 | 616 | ||
627 | socdev->card->codec = wm8974_codec; | 617 | ret = wm8974_reset(codec); |
628 | codec = wm8974_codec; | ||
629 | |||
630 | /* register pcms */ | ||
631 | ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); | ||
632 | if (ret < 0) { | 618 | if (ret < 0) { |
633 | dev_err(codec->dev, "failed to create pcms: %d\n", ret); | 619 | dev_err(codec->dev, "Failed to issue reset\n"); |
634 | goto pcm_err; | 620 | return ret; |
635 | } | 621 | } |
636 | 622 | ||
623 | wm8974_set_bias_level(codec, SND_SOC_BIAS_STANDBY); | ||
637 | snd_soc_add_controls(codec, wm8974_snd_controls, | 624 | snd_soc_add_controls(codec, wm8974_snd_controls, |
638 | ARRAY_SIZE(wm8974_snd_controls)); | 625 | ARRAY_SIZE(wm8974_snd_controls)); |
639 | wm8974_add_widgets(codec); | 626 | wm8974_add_widgets(codec); |
640 | 627 | ||
641 | return ret; | 628 | return ret; |
642 | |||
643 | pcm_err: | ||
644 | return ret; | ||
645 | } | 629 | } |
646 | 630 | ||
647 | /* power down chip */ | 631 | /* power down chip */ |
648 | static int wm8974_remove(struct platform_device *pdev) | 632 | static int wm8974_remove(struct snd_soc_codec *codec) |
649 | { | 633 | { |
650 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | 634 | wm8974_set_bias_level(codec, SND_SOC_BIAS_OFF); |
651 | |||
652 | snd_soc_free_pcms(socdev); | ||
653 | snd_soc_dapm_free(socdev); | ||
654 | |||
655 | return 0; | 635 | return 0; |
656 | } | 636 | } |
657 | 637 | ||
658 | struct snd_soc_codec_device soc_codec_dev_wm8974 = { | 638 | static struct snd_soc_codec_driver soc_codec_dev_wm8974 = { |
659 | .probe = wm8974_probe, | 639 | .probe = wm8974_probe, |
660 | .remove = wm8974_remove, | 640 | .remove = wm8974_remove, |
661 | .suspend = wm8974_suspend, | 641 | .suspend = wm8974_suspend, |
662 | .resume = wm8974_resume, | 642 | .resume = wm8974_resume, |
643 | .set_bias_level = wm8974_set_bias_level, | ||
644 | .reg_cache_size = ARRAY_SIZE(wm8974_reg), | ||
645 | .reg_word_size = sizeof(u16), | ||
646 | .reg_cache_default = wm8974_reg, | ||
663 | }; | 647 | }; |
664 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm8974); | ||
665 | |||
666 | static __devinit int wm8974_register(struct wm8974_priv *wm8974) | ||
667 | { | ||
668 | int ret; | ||
669 | struct snd_soc_codec *codec = &wm8974->codec; | ||
670 | |||
671 | if (wm8974_codec) { | ||
672 | dev_err(codec->dev, "Another WM8974 is registered\n"); | ||
673 | ret = -EINVAL; | ||
674 | goto err; | ||
675 | } | ||
676 | |||
677 | mutex_init(&codec->mutex); | ||
678 | INIT_LIST_HEAD(&codec->dapm_widgets); | ||
679 | INIT_LIST_HEAD(&codec->dapm_paths); | ||
680 | |||
681 | snd_soc_codec_set_drvdata(codec, wm8974); | ||
682 | codec->name = "WM8974"; | ||
683 | codec->owner = THIS_MODULE; | ||
684 | codec->bias_level = SND_SOC_BIAS_OFF; | ||
685 | codec->set_bias_level = wm8974_set_bias_level; | ||
686 | codec->dai = &wm8974_dai; | ||
687 | codec->num_dai = 1; | ||
688 | codec->reg_cache_size = WM8974_CACHEREGNUM; | ||
689 | codec->reg_cache = &wm8974->reg_cache; | ||
690 | |||
691 | ret = snd_soc_codec_set_cache_io(codec, 7, 9, SND_SOC_I2C); | ||
692 | if (ret < 0) { | ||
693 | dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); | ||
694 | goto err; | ||
695 | } | ||
696 | |||
697 | memcpy(codec->reg_cache, wm8974_reg, sizeof(wm8974_reg)); | ||
698 | |||
699 | ret = wm8974_reset(codec); | ||
700 | if (ret < 0) { | ||
701 | dev_err(codec->dev, "Failed to issue reset\n"); | ||
702 | goto err; | ||
703 | } | ||
704 | |||
705 | wm8974_dai.dev = codec->dev; | ||
706 | |||
707 | wm8974_set_bias_level(codec, SND_SOC_BIAS_STANDBY); | ||
708 | |||
709 | wm8974_codec = codec; | ||
710 | |||
711 | ret = snd_soc_register_codec(codec); | ||
712 | if (ret != 0) { | ||
713 | dev_err(codec->dev, "Failed to register codec: %d\n", ret); | ||
714 | goto err; | ||
715 | } | ||
716 | |||
717 | ret = snd_soc_register_dai(&wm8974_dai); | ||
718 | if (ret != 0) { | ||
719 | dev_err(codec->dev, "Failed to register DAI: %d\n", ret); | ||
720 | goto err_codec; | ||
721 | } | ||
722 | |||
723 | return 0; | ||
724 | |||
725 | err_codec: | ||
726 | snd_soc_unregister_codec(codec); | ||
727 | err: | ||
728 | kfree(wm8974); | ||
729 | return ret; | ||
730 | } | ||
731 | |||
732 | static __devexit void wm8974_unregister(struct wm8974_priv *wm8974) | ||
733 | { | ||
734 | wm8974_set_bias_level(&wm8974->codec, SND_SOC_BIAS_OFF); | ||
735 | snd_soc_unregister_dai(&wm8974_dai); | ||
736 | snd_soc_unregister_codec(&wm8974->codec); | ||
737 | kfree(wm8974); | ||
738 | wm8974_codec = NULL; | ||
739 | } | ||
740 | 648 | ||
649 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | ||
741 | static __devinit int wm8974_i2c_probe(struct i2c_client *i2c, | 650 | static __devinit int wm8974_i2c_probe(struct i2c_client *i2c, |
742 | const struct i2c_device_id *id) | 651 | const struct i2c_device_id *id) |
743 | { | 652 | { |
744 | struct wm8974_priv *wm8974; | 653 | struct wm8974_priv *wm8974; |
745 | struct snd_soc_codec *codec; | 654 | int ret; |
746 | 655 | ||
747 | wm8974 = kzalloc(sizeof(struct wm8974_priv), GFP_KERNEL); | 656 | wm8974 = kzalloc(sizeof(struct wm8974_priv), GFP_KERNEL); |
748 | if (wm8974 == NULL) | 657 | if (wm8974 == NULL) |
749 | return -ENOMEM; | 658 | return -ENOMEM; |
750 | 659 | ||
751 | codec = &wm8974->codec; | ||
752 | codec->hw_write = (hw_write_t)i2c_master_send; | ||
753 | |||
754 | i2c_set_clientdata(i2c, wm8974); | 660 | i2c_set_clientdata(i2c, wm8974); |
755 | codec->control_data = i2c; | ||
756 | |||
757 | codec->dev = &i2c->dev; | ||
758 | 661 | ||
759 | return wm8974_register(wm8974); | 662 | ret = snd_soc_register_codec(&i2c->dev, |
663 | &soc_codec_dev_wm8974, &wm8974_dai, 1); | ||
664 | if (ret < 0) | ||
665 | kfree(wm8974); | ||
666 | return ret; | ||
760 | } | 667 | } |
761 | 668 | ||
762 | static __devexit int wm8974_i2c_remove(struct i2c_client *client) | 669 | static __devexit int wm8974_i2c_remove(struct i2c_client *client) |
763 | { | 670 | { |
764 | struct wm8974_priv *wm8974 = i2c_get_clientdata(client); | 671 | snd_soc_unregister_codec(&client->dev); |
765 | wm8974_unregister(wm8974); | 672 | kfree(i2c_get_clientdata(client)); |
766 | return 0; | 673 | return 0; |
767 | } | 674 | } |
768 | 675 | ||
@@ -774,23 +681,34 @@ MODULE_DEVICE_TABLE(i2c, wm8974_i2c_id); | |||
774 | 681 | ||
775 | static struct i2c_driver wm8974_i2c_driver = { | 682 | static struct i2c_driver wm8974_i2c_driver = { |
776 | .driver = { | 683 | .driver = { |
777 | .name = "WM8974", | 684 | .name = "wm8974-codec", |
778 | .owner = THIS_MODULE, | 685 | .owner = THIS_MODULE, |
779 | }, | 686 | }, |
780 | .probe = wm8974_i2c_probe, | 687 | .probe = wm8974_i2c_probe, |
781 | .remove = __devexit_p(wm8974_i2c_remove), | 688 | .remove = __devexit_p(wm8974_i2c_remove), |
782 | .id_table = wm8974_i2c_id, | 689 | .id_table = wm8974_i2c_id, |
783 | }; | 690 | }; |
691 | #endif | ||
784 | 692 | ||
785 | static int __init wm8974_modinit(void) | 693 | static int __init wm8974_modinit(void) |
786 | { | 694 | { |
787 | return i2c_add_driver(&wm8974_i2c_driver); | 695 | int ret = 0; |
696 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | ||
697 | ret = i2c_add_driver(&wm8974_i2c_driver); | ||
698 | if (ret != 0) { | ||
699 | printk(KERN_ERR "Failed to register wm8974 I2C driver: %d\n", | ||
700 | ret); | ||
701 | } | ||
702 | #endif | ||
703 | return ret; | ||
788 | } | 704 | } |
789 | module_init(wm8974_modinit); | 705 | module_init(wm8974_modinit); |
790 | 706 | ||
791 | static void __exit wm8974_exit(void) | 707 | static void __exit wm8974_exit(void) |
792 | { | 708 | { |
709 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | ||
793 | i2c_del_driver(&wm8974_i2c_driver); | 710 | i2c_del_driver(&wm8974_i2c_driver); |
711 | #endif | ||
794 | } | 712 | } |
795 | module_exit(wm8974_exit); | 713 | module_exit(wm8974_exit); |
796 | 714 | ||