diff options
Diffstat (limited to 'sound/soc/codecs/wm8510.c')
-rw-r--r-- | sound/soc/codecs/wm8510.c | 300 |
1 files changed, 98 insertions, 202 deletions
diff --git a/sound/soc/codecs/wm8510.c b/sound/soc/codecs/wm8510.c index 0f7bcb61071a..db0dced74843 100644 --- a/sound/soc/codecs/wm8510.c +++ b/sound/soc/codecs/wm8510.c | |||
@@ -24,15 +24,10 @@ | |||
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 | 28 | ||
30 | #include "wm8510.h" | 29 | #include "wm8510.h" |
31 | 30 | ||
32 | #define WM8510_VERSION "0.6" | ||
33 | |||
34 | struct snd_soc_codec_device soc_codec_dev_wm8510; | ||
35 | |||
36 | /* | 31 | /* |
37 | * wm8510 register cache | 32 | * wm8510 register cache |
38 | * We can't read the WM8510 register space when we are | 33 | * We can't read the WM8510 register space when we are |
@@ -61,6 +56,11 @@ static const u16 wm8510_reg[WM8510_CACHEREGNUM] = { | |||
61 | 56 | ||
62 | #define wm8510_reset(c) snd_soc_write(c, WM8510_RESET, 0) | 57 | #define wm8510_reset(c) snd_soc_write(c, WM8510_RESET, 0) |
63 | 58 | ||
59 | /* codec private data */ | ||
60 | struct wm8510_priv { | ||
61 | enum snd_soc_control_type control_type; | ||
62 | }; | ||
63 | |||
64 | static const char *wm8510_companding[] = { "Off", "NC", "u-law", "A-law" }; | 64 | static const char *wm8510_companding[] = { "Off", "NC", "u-law", "A-law" }; |
65 | static const char *wm8510_deemp[] = { "None", "32kHz", "44.1kHz", "48kHz" }; | 65 | static const char *wm8510_deemp[] = { "None", "32kHz", "44.1kHz", "48kHz" }; |
66 | static const char *wm8510_alc[] = { "ALC", "Limiter" }; | 66 | static const char *wm8510_alc[] = { "ALC", "Limiter" }; |
@@ -215,10 +215,11 @@ static const struct snd_soc_dapm_route audio_map[] = { | |||
215 | 215 | ||
216 | static int wm8510_add_widgets(struct snd_soc_codec *codec) | 216 | static int wm8510_add_widgets(struct snd_soc_codec *codec) |
217 | { | 217 | { |
218 | snd_soc_dapm_new_controls(codec, wm8510_dapm_widgets, | 218 | struct snd_soc_dapm_context *dapm = &codec->dapm; |
219 | ARRAY_SIZE(wm8510_dapm_widgets)); | ||
220 | 219 | ||
221 | snd_soc_dapm_add_routes(codec, audio_map, ARRAY_SIZE(audio_map)); | 220 | snd_soc_dapm_new_controls(dapm, wm8510_dapm_widgets, |
221 | ARRAY_SIZE(wm8510_dapm_widgets)); | ||
222 | snd_soc_dapm_add_routes(dapm, audio_map, ARRAY_SIZE(audio_map)); | ||
222 | 223 | ||
223 | return 0; | 224 | return 0; |
224 | } | 225 | } |
@@ -403,8 +404,7 @@ static int wm8510_pcm_hw_params(struct snd_pcm_substream *substream, | |||
403 | struct snd_soc_dai *dai) | 404 | struct snd_soc_dai *dai) |
404 | { | 405 | { |
405 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 406 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
406 | struct snd_soc_device *socdev = rtd->socdev; | 407 | struct snd_soc_codec *codec = rtd->codec; |
407 | struct snd_soc_codec *codec = socdev->card->codec; | ||
408 | u16 iface = snd_soc_read(codec, WM8510_IFACE) & 0x19f; | 408 | u16 iface = snd_soc_read(codec, WM8510_IFACE) & 0x19f; |
409 | u16 adn = snd_soc_read(codec, WM8510_ADD) & 0x1f1; | 409 | u16 adn = snd_soc_read(codec, WM8510_ADD) & 0x1f1; |
410 | 410 | ||
@@ -478,7 +478,7 @@ static int wm8510_set_bias_level(struct snd_soc_codec *codec, | |||
478 | case SND_SOC_BIAS_STANDBY: | 478 | case SND_SOC_BIAS_STANDBY: |
479 | power1 |= WM8510_POWER1_BIASEN | WM8510_POWER1_BUFIOEN; | 479 | power1 |= WM8510_POWER1_BIASEN | WM8510_POWER1_BUFIOEN; |
480 | 480 | ||
481 | if (codec->bias_level == SND_SOC_BIAS_OFF) { | 481 | if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) { |
482 | /* Initial cap charge at VMID 5k */ | 482 | /* Initial cap charge at VMID 5k */ |
483 | snd_soc_write(codec, WM8510_POWER1, power1 | 0x3); | 483 | snd_soc_write(codec, WM8510_POWER1, power1 | 0x3); |
484 | mdelay(100); | 484 | mdelay(100); |
@@ -495,7 +495,7 @@ static int wm8510_set_bias_level(struct snd_soc_codec *codec, | |||
495 | break; | 495 | break; |
496 | } | 496 | } |
497 | 497 | ||
498 | codec->bias_level = level; | 498 | codec->dapm.bias_level = level; |
499 | return 0; | 499 | return 0; |
500 | } | 500 | } |
501 | 501 | ||
@@ -514,8 +514,8 @@ static struct snd_soc_dai_ops wm8510_dai_ops = { | |||
514 | .set_pll = wm8510_set_dai_pll, | 514 | .set_pll = wm8510_set_dai_pll, |
515 | }; | 515 | }; |
516 | 516 | ||
517 | struct snd_soc_dai wm8510_dai = { | 517 | static struct snd_soc_dai_driver wm8510_dai = { |
518 | .name = "WM8510 HiFi", | 518 | .name = "wm8510-hifi", |
519 | .playback = { | 519 | .playback = { |
520 | .stream_name = "Playback", | 520 | .stream_name = "Playback", |
521 | .channels_min = 2, | 521 | .channels_min = 2, |
@@ -531,21 +531,15 @@ struct snd_soc_dai wm8510_dai = { | |||
531 | .ops = &wm8510_dai_ops, | 531 | .ops = &wm8510_dai_ops, |
532 | .symmetric_rates = 1, | 532 | .symmetric_rates = 1, |
533 | }; | 533 | }; |
534 | EXPORT_SYMBOL_GPL(wm8510_dai); | ||
535 | 534 | ||
536 | static int wm8510_suspend(struct platform_device *pdev, pm_message_t state) | 535 | static int wm8510_suspend(struct snd_soc_codec *codec, pm_message_t state) |
537 | { | 536 | { |
538 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
539 | struct snd_soc_codec *codec = socdev->card->codec; | ||
540 | |||
541 | wm8510_set_bias_level(codec, SND_SOC_BIAS_OFF); | 537 | wm8510_set_bias_level(codec, SND_SOC_BIAS_OFF); |
542 | return 0; | 538 | return 0; |
543 | } | 539 | } |
544 | 540 | ||
545 | static int wm8510_resume(struct platform_device *pdev) | 541 | static int wm8510_resume(struct snd_soc_codec *codec) |
546 | { | 542 | { |
547 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
548 | struct snd_soc_codec *codec = socdev->card->codec; | ||
549 | int i; | 543 | int i; |
550 | u8 data[2]; | 544 | u8 data[2]; |
551 | u16 *cache = codec->reg_cache; | 545 | u16 *cache = codec->reg_cache; |
@@ -561,256 +555,158 @@ static int wm8510_resume(struct platform_device *pdev) | |||
561 | return 0; | 555 | return 0; |
562 | } | 556 | } |
563 | 557 | ||
564 | /* | 558 | static int wm8510_probe(struct snd_soc_codec *codec) |
565 | * initialise the WM8510 driver | ||
566 | * register the mixer and dsp interfaces with the kernel | ||
567 | */ | ||
568 | static int wm8510_init(struct snd_soc_device *socdev, | ||
569 | enum snd_soc_control_type control) | ||
570 | { | 559 | { |
571 | struct snd_soc_codec *codec = socdev->card->codec; | 560 | struct wm8510_priv *wm8510 = snd_soc_codec_get_drvdata(codec); |
572 | int ret = 0; | 561 | int ret; |
573 | |||
574 | codec->name = "WM8510"; | ||
575 | codec->owner = THIS_MODULE; | ||
576 | codec->set_bias_level = wm8510_set_bias_level; | ||
577 | codec->dai = &wm8510_dai; | ||
578 | codec->num_dai = 1; | ||
579 | codec->reg_cache_size = ARRAY_SIZE(wm8510_reg); | ||
580 | codec->reg_cache = kmemdup(wm8510_reg, sizeof(wm8510_reg), GFP_KERNEL); | ||
581 | |||
582 | if (codec->reg_cache == NULL) | ||
583 | return -ENOMEM; | ||
584 | 562 | ||
585 | ret = snd_soc_codec_set_cache_io(codec, 7, 9, control); | 563 | ret = snd_soc_codec_set_cache_io(codec, 7, 9, wm8510->control_type); |
586 | if (ret < 0) { | 564 | if (ret < 0) { |
587 | printk(KERN_ERR "wm8510: failed to set cache I/O: %d\n", | 565 | printk(KERN_ERR "wm8510: failed to set cache I/O: %d\n", ret); |
588 | ret); | 566 | return ret; |
589 | goto err; | ||
590 | } | 567 | } |
591 | 568 | ||
592 | wm8510_reset(codec); | 569 | wm8510_reset(codec); |
593 | 570 | ||
594 | /* register pcms */ | ||
595 | ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); | ||
596 | if (ret < 0) { | ||
597 | printk(KERN_ERR "wm8510: failed to create pcms\n"); | ||
598 | goto err; | ||
599 | } | ||
600 | |||
601 | /* power on device */ | 571 | /* power on device */ |
602 | codec->bias_level = SND_SOC_BIAS_OFF; | ||
603 | wm8510_set_bias_level(codec, SND_SOC_BIAS_STANDBY); | 572 | wm8510_set_bias_level(codec, SND_SOC_BIAS_STANDBY); |
604 | snd_soc_add_controls(codec, wm8510_snd_controls, | 573 | snd_soc_add_controls(codec, wm8510_snd_controls, |
605 | ARRAY_SIZE(wm8510_snd_controls)); | 574 | ARRAY_SIZE(wm8510_snd_controls)); |
606 | wm8510_add_widgets(codec); | 575 | wm8510_add_widgets(codec); |
607 | 576 | ||
608 | return ret; | 577 | return ret; |
609 | |||
610 | err: | ||
611 | kfree(codec->reg_cache); | ||
612 | return ret; | ||
613 | } | 578 | } |
614 | 579 | ||
615 | static struct snd_soc_device *wm8510_socdev; | 580 | /* power down chip */ |
581 | static int wm8510_remove(struct snd_soc_codec *codec) | ||
582 | { | ||
583 | struct wm8510_priv *wm8510 = snd_soc_codec_get_drvdata(codec); | ||
616 | 584 | ||
617 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | 585 | wm8510_set_bias_level(codec, SND_SOC_BIAS_OFF); |
586 | kfree(wm8510); | ||
587 | return 0; | ||
588 | } | ||
618 | 589 | ||
619 | /* | 590 | static struct snd_soc_codec_driver soc_codec_dev_wm8510 = { |
620 | * WM8510 2 wire address is 0x1a | 591 | .probe = wm8510_probe, |
621 | */ | 592 | .remove = wm8510_remove, |
593 | .suspend = wm8510_suspend, | ||
594 | .resume = wm8510_resume, | ||
595 | .set_bias_level = wm8510_set_bias_level, | ||
596 | .reg_cache_size = ARRAY_SIZE(wm8510_reg), | ||
597 | .reg_word_size = sizeof(u16), | ||
598 | .reg_cache_default =wm8510_reg, | ||
599 | }; | ||
622 | 600 | ||
623 | static int wm8510_i2c_probe(struct i2c_client *i2c, | 601 | #if defined(CONFIG_SPI_MASTER) |
624 | const struct i2c_device_id *id) | 602 | static int __devinit wm8510_spi_probe(struct spi_device *spi) |
625 | { | 603 | { |
626 | struct snd_soc_device *socdev = wm8510_socdev; | 604 | struct wm8510_priv *wm8510; |
627 | struct snd_soc_codec *codec = socdev->card->codec; | ||
628 | int ret; | 605 | int ret; |
629 | 606 | ||
630 | i2c_set_clientdata(i2c, codec); | 607 | wm8510 = kzalloc(sizeof(struct wm8510_priv), GFP_KERNEL); |
631 | codec->control_data = i2c; | 608 | if (wm8510 == NULL) |
609 | return -ENOMEM; | ||
610 | |||
611 | wm8510->control_type = SND_SOC_SPI; | ||
612 | spi_set_drvdata(spi, wm8510); | ||
632 | 613 | ||
633 | ret = wm8510_init(socdev, SND_SOC_I2C); | 614 | ret = snd_soc_register_codec(&spi->dev, |
615 | &soc_codec_dev_wm8510, &wm8510_dai, 1); | ||
634 | if (ret < 0) | 616 | if (ret < 0) |
635 | pr_err("failed to initialise WM8510\n"); | 617 | kfree(wm8510); |
636 | |||
637 | return ret; | 618 | return ret; |
638 | } | 619 | } |
639 | 620 | ||
640 | static int wm8510_i2c_remove(struct i2c_client *client) | 621 | static int __devexit wm8510_spi_remove(struct spi_device *spi) |
641 | { | 622 | { |
642 | struct snd_soc_codec *codec = i2c_get_clientdata(client); | 623 | snd_soc_unregister_codec(&spi->dev); |
643 | kfree(codec->reg_cache); | ||
644 | return 0; | 624 | return 0; |
645 | } | 625 | } |
646 | 626 | ||
647 | static const struct i2c_device_id wm8510_i2c_id[] = { | 627 | static struct spi_driver wm8510_spi_driver = { |
648 | { "wm8510", 0 }, | ||
649 | { } | ||
650 | }; | ||
651 | MODULE_DEVICE_TABLE(i2c, wm8510_i2c_id); | ||
652 | |||
653 | static struct i2c_driver wm8510_i2c_driver = { | ||
654 | .driver = { | 628 | .driver = { |
655 | .name = "WM8510 I2C Codec", | 629 | .name = "wm8510", |
656 | .owner = THIS_MODULE, | 630 | .owner = THIS_MODULE, |
657 | }, | 631 | }, |
658 | .probe = wm8510_i2c_probe, | 632 | .probe = wm8510_spi_probe, |
659 | .remove = wm8510_i2c_remove, | 633 | .remove = __devexit_p(wm8510_spi_remove), |
660 | .id_table = wm8510_i2c_id, | ||
661 | }; | 634 | }; |
635 | #endif /* CONFIG_SPI_MASTER */ | ||
662 | 636 | ||
663 | static int wm8510_add_i2c_device(struct platform_device *pdev, | 637 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) |
664 | const struct wm8510_setup_data *setup) | 638 | static __devinit int wm8510_i2c_probe(struct i2c_client *i2c, |
639 | const struct i2c_device_id *id) | ||
665 | { | 640 | { |
666 | struct i2c_board_info info; | 641 | struct wm8510_priv *wm8510; |
667 | struct i2c_adapter *adapter; | ||
668 | struct i2c_client *client; | ||
669 | int ret; | 642 | int ret; |
670 | 643 | ||
671 | ret = i2c_add_driver(&wm8510_i2c_driver); | 644 | wm8510 = kzalloc(sizeof(struct wm8510_priv), GFP_KERNEL); |
672 | if (ret != 0) { | 645 | if (wm8510 == NULL) |
673 | dev_err(&pdev->dev, "can't add i2c driver\n"); | 646 | return -ENOMEM; |
674 | return ret; | ||
675 | } | ||
676 | |||
677 | memset(&info, 0, sizeof(struct i2c_board_info)); | ||
678 | info.addr = setup->i2c_address; | ||
679 | strlcpy(info.type, "wm8510", I2C_NAME_SIZE); | ||
680 | |||
681 | adapter = i2c_get_adapter(setup->i2c_bus); | ||
682 | if (!adapter) { | ||
683 | dev_err(&pdev->dev, "can't get i2c adapter %d\n", | ||
684 | setup->i2c_bus); | ||
685 | goto err_driver; | ||
686 | } | ||
687 | |||
688 | client = i2c_new_device(adapter, &info); | ||
689 | i2c_put_adapter(adapter); | ||
690 | if (!client) { | ||
691 | dev_err(&pdev->dev, "can't add i2c device at 0x%x\n", | ||
692 | (unsigned int)info.addr); | ||
693 | goto err_driver; | ||
694 | } | ||
695 | |||
696 | return 0; | ||
697 | |||
698 | err_driver: | ||
699 | i2c_del_driver(&wm8510_i2c_driver); | ||
700 | return -ENODEV; | ||
701 | } | ||
702 | #endif | ||
703 | |||
704 | #if defined(CONFIG_SPI_MASTER) | ||
705 | static int __devinit wm8510_spi_probe(struct spi_device *spi) | ||
706 | { | ||
707 | struct snd_soc_device *socdev = wm8510_socdev; | ||
708 | struct snd_soc_codec *codec = socdev->card->codec; | ||
709 | int ret; | ||
710 | 647 | ||
711 | codec->control_data = spi; | 648 | i2c_set_clientdata(i2c, wm8510); |
649 | wm8510->control_type = SND_SOC_I2C; | ||
712 | 650 | ||
713 | ret = wm8510_init(socdev, SND_SOC_SPI); | 651 | ret = snd_soc_register_codec(&i2c->dev, |
652 | &soc_codec_dev_wm8510, &wm8510_dai, 1); | ||
714 | if (ret < 0) | 653 | if (ret < 0) |
715 | dev_err(&spi->dev, "failed to initialise WM8510\n"); | 654 | kfree(wm8510); |
716 | |||
717 | return ret; | 655 | return ret; |
718 | } | 656 | } |
719 | 657 | ||
720 | static int __devexit wm8510_spi_remove(struct spi_device *spi) | 658 | static __devexit int wm8510_i2c_remove(struct i2c_client *client) |
721 | { | 659 | { |
660 | snd_soc_unregister_codec(&client->dev); | ||
722 | return 0; | 661 | return 0; |
723 | } | 662 | } |
724 | 663 | ||
725 | static struct spi_driver wm8510_spi_driver = { | 664 | static const struct i2c_device_id wm8510_i2c_id[] = { |
665 | { "wm8510", 0 }, | ||
666 | { } | ||
667 | }; | ||
668 | MODULE_DEVICE_TABLE(i2c, wm8510_i2c_id); | ||
669 | |||
670 | static struct i2c_driver wm8510_i2c_driver = { | ||
726 | .driver = { | 671 | .driver = { |
727 | .name = "wm8510", | 672 | .name = "wm8510-codec", |
728 | .bus = &spi_bus_type, | 673 | .owner = THIS_MODULE, |
729 | .owner = THIS_MODULE, | ||
730 | }, | 674 | }, |
731 | .probe = wm8510_spi_probe, | 675 | .probe = wm8510_i2c_probe, |
732 | .remove = __devexit_p(wm8510_spi_remove), | 676 | .remove = __devexit_p(wm8510_i2c_remove), |
677 | .id_table = wm8510_i2c_id, | ||
733 | }; | 678 | }; |
734 | #endif /* CONFIG_SPI_MASTER */ | 679 | #endif |
735 | 680 | ||
736 | static int wm8510_probe(struct platform_device *pdev) | 681 | static int __init wm8510_modinit(void) |
737 | { | 682 | { |
738 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
739 | struct wm8510_setup_data *setup; | ||
740 | struct snd_soc_codec *codec; | ||
741 | int ret = 0; | 683 | int ret = 0; |
742 | |||
743 | pr_info("WM8510 Audio Codec %s", WM8510_VERSION); | ||
744 | |||
745 | setup = socdev->codec_data; | ||
746 | codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); | ||
747 | if (codec == NULL) | ||
748 | return -ENOMEM; | ||
749 | |||
750 | socdev->card->codec = codec; | ||
751 | mutex_init(&codec->mutex); | ||
752 | INIT_LIST_HEAD(&codec->dapm_widgets); | ||
753 | INIT_LIST_HEAD(&codec->dapm_paths); | ||
754 | |||
755 | wm8510_socdev = socdev; | ||
756 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | 684 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) |
757 | if (setup->i2c_address) { | 685 | ret = i2c_add_driver(&wm8510_i2c_driver); |
758 | ret = wm8510_add_i2c_device(pdev, setup); | 686 | if (ret != 0) { |
687 | printk(KERN_ERR "Failed to register WM8510 I2C driver: %d\n", | ||
688 | ret); | ||
759 | } | 689 | } |
760 | #endif | 690 | #endif |
761 | #if defined(CONFIG_SPI_MASTER) | 691 | #if defined(CONFIG_SPI_MASTER) |
762 | if (setup->spi) { | 692 | ret = spi_register_driver(&wm8510_spi_driver); |
763 | ret = spi_register_driver(&wm8510_spi_driver); | 693 | if (ret != 0) { |
764 | if (ret != 0) | 694 | printk(KERN_ERR "Failed to register WM8510 SPI driver: %d\n", |
765 | printk(KERN_ERR "can't add spi driver"); | 695 | ret); |
766 | } | 696 | } |
767 | #endif | 697 | #endif |
768 | |||
769 | if (ret != 0) | ||
770 | kfree(codec); | ||
771 | return ret; | 698 | return ret; |
772 | } | 699 | } |
700 | module_init(wm8510_modinit); | ||
773 | 701 | ||
774 | /* power down chip */ | 702 | static void __exit wm8510_exit(void) |
775 | static int wm8510_remove(struct platform_device *pdev) | ||
776 | { | 703 | { |
777 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
778 | struct snd_soc_codec *codec = socdev->card->codec; | ||
779 | |||
780 | if (codec->control_data) | ||
781 | wm8510_set_bias_level(codec, SND_SOC_BIAS_OFF); | ||
782 | |||
783 | snd_soc_free_pcms(socdev); | ||
784 | snd_soc_dapm_free(socdev); | ||
785 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | 704 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) |
786 | i2c_unregister_device(codec->control_data); | ||
787 | i2c_del_driver(&wm8510_i2c_driver); | 705 | i2c_del_driver(&wm8510_i2c_driver); |
788 | #endif | 706 | #endif |
789 | #if defined(CONFIG_SPI_MASTER) | 707 | #if defined(CONFIG_SPI_MASTER) |
790 | spi_unregister_driver(&wm8510_spi_driver); | 708 | spi_unregister_driver(&wm8510_spi_driver); |
791 | #endif | 709 | #endif |
792 | kfree(codec); | ||
793 | |||
794 | return 0; | ||
795 | } | ||
796 | |||
797 | struct snd_soc_codec_device soc_codec_dev_wm8510 = { | ||
798 | .probe = wm8510_probe, | ||
799 | .remove = wm8510_remove, | ||
800 | .suspend = wm8510_suspend, | ||
801 | .resume = wm8510_resume, | ||
802 | }; | ||
803 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm8510); | ||
804 | |||
805 | static int __init wm8510_modinit(void) | ||
806 | { | ||
807 | return snd_soc_register_dai(&wm8510_dai); | ||
808 | } | ||
809 | module_init(wm8510_modinit); | ||
810 | |||
811 | static void __exit wm8510_exit(void) | ||
812 | { | ||
813 | snd_soc_unregister_dai(&wm8510_dai); | ||
814 | } | 710 | } |
815 | module_exit(wm8510_exit); | 711 | module_exit(wm8510_exit); |
816 | 712 | ||