diff options
Diffstat (limited to 'sound/soc/codecs/wm9090.c')
-rw-r--r-- | sound/soc/codecs/wm9090.c | 216 |
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 | ||
37 | static struct snd_soc_codec *wm9090_codec; | ||
38 | |||
39 | static const u16 wm9090_reg_defaults[] = { | 36 | static 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 */ |
144 | struct wm9090_priv { | 141 | struct 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 | ||
156 | static int wm9090_volatile(unsigned int reg) | 147 | static 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[] = { | |||
450 | static int wm9090_add_controls(struct snd_soc_codec *codec) | 441 | static 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 | ||
559 | static int wm9090_probe(struct platform_device *pdev) | 551 | static 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 | |||
584 | pcm_err: | ||
585 | return ret; | ||
586 | } | 608 | } |
587 | 609 | ||
588 | #ifdef CONFIG_PM | 610 | #ifdef CONFIG_PM |
589 | static int wm9090_suspend(struct platform_device *pdev, pm_message_t state) | 611 | static 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 | ||
599 | static int wm9090_resume(struct platform_device *pdev) | 618 | static 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 | ||
613 | static int wm9090_remove(struct platform_device *pdev) | 629 | static 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 | ||
623 | struct snd_soc_codec_device soc_codec_dev_wm9090 = { | 636 | static 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 | }; |
629 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm9090); | ||
630 | 647 | ||
631 | static int wm9090_i2c_probe(struct i2c_client *i2c, | 648 | static 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 | |||
720 | err_bias: | ||
721 | wm9090_set_bias_level(codec, SND_SOC_BIAS_OFF); | ||
722 | err: | ||
723 | kfree(wm9090); | ||
724 | i2c_set_clientdata(i2c, NULL); | ||
725 | wm9090_codec = NULL; | ||
726 | |||
727 | return ret; | 672 | return ret; |
728 | } | 673 | } |
729 | 674 | ||
730 | static int wm9090_i2c_remove(struct i2c_client *i2c) | 675 | static 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 | ||
749 | static struct i2c_driver wm9090_i2c_driver = { | 691 | static 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, |