diff options
Diffstat (limited to 'sound/soc/codecs/cs42l51.c')
-rw-r--r-- | sound/soc/codecs/cs42l51.c | 302 |
1 files changed, 96 insertions, 206 deletions
diff --git a/sound/soc/codecs/cs42l51.c b/sound/soc/codecs/cs42l51.c index dd9b8550c402..8fb7070108dd 100644 --- a/sound/soc/codecs/cs42l51.c +++ b/sound/soc/codecs/cs42l51.c | |||
@@ -26,7 +26,6 @@ | |||
26 | #include <linux/slab.h> | 26 | #include <linux/slab.h> |
27 | #include <sound/core.h> | 27 | #include <sound/core.h> |
28 | #include <sound/soc.h> | 28 | #include <sound/soc.h> |
29 | #include <sound/soc-dapm.h> | ||
30 | #include <sound/tlv.h> | 29 | #include <sound/tlv.h> |
31 | #include <sound/initval.h> | 30 | #include <sound/initval.h> |
32 | #include <sound/pcm_params.h> | 31 | #include <sound/pcm_params.h> |
@@ -42,15 +41,13 @@ enum master_slave_mode { | |||
42 | }; | 41 | }; |
43 | 42 | ||
44 | struct cs42l51_private { | 43 | struct cs42l51_private { |
44 | enum snd_soc_control_type control_type; | ||
45 | void *control_data; | ||
45 | unsigned int mclk; | 46 | unsigned int mclk; |
46 | unsigned int audio_mode; /* The mode (I2S or left-justified) */ | 47 | unsigned int audio_mode; /* The mode (I2S or left-justified) */ |
47 | enum master_slave_mode func; | 48 | enum master_slave_mode func; |
48 | struct snd_soc_codec codec; | ||
49 | u8 reg_cache[CS42L51_NUMREGS]; | ||
50 | }; | 49 | }; |
51 | 50 | ||
52 | static struct snd_soc_codec *cs42l51_codec; | ||
53 | |||
54 | #define CS42L51_FORMATS ( \ | 51 | #define CS42L51_FORMATS ( \ |
55 | SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S16_BE | \ | 52 | SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S16_BE | \ |
56 | SNDRV_PCM_FMTBIT_S18_3LE | SNDRV_PCM_FMTBIT_S18_3BE | \ | 53 | SNDRV_PCM_FMTBIT_S18_3LE | SNDRV_PCM_FMTBIT_S18_3BE | \ |
@@ -75,134 +72,6 @@ static int cs42l51_fill_cache(struct snd_soc_codec *codec) | |||
75 | return 0; | 72 | return 0; |
76 | } | 73 | } |
77 | 74 | ||
78 | static int cs42l51_i2c_probe(struct i2c_client *i2c_client, | ||
79 | const struct i2c_device_id *id) | ||
80 | { | ||
81 | struct snd_soc_codec *codec; | ||
82 | struct cs42l51_private *cs42l51; | ||
83 | int ret = 0; | ||
84 | int reg; | ||
85 | |||
86 | if (cs42l51_codec) | ||
87 | return -EBUSY; | ||
88 | |||
89 | /* Verify that we have a CS42L51 */ | ||
90 | ret = i2c_smbus_read_byte_data(i2c_client, CS42L51_CHIP_REV_ID); | ||
91 | if (ret < 0) { | ||
92 | dev_err(&i2c_client->dev, "failed to read I2C\n"); | ||
93 | goto error; | ||
94 | } | ||
95 | |||
96 | if ((ret != CS42L51_MK_CHIP_REV(CS42L51_CHIP_ID, CS42L51_CHIP_REV_A)) && | ||
97 | (ret != CS42L51_MK_CHIP_REV(CS42L51_CHIP_ID, CS42L51_CHIP_REV_B))) { | ||
98 | dev_err(&i2c_client->dev, "Invalid chip id\n"); | ||
99 | ret = -ENODEV; | ||
100 | goto error; | ||
101 | } | ||
102 | |||
103 | dev_info(&i2c_client->dev, "found device cs42l51 rev %d\n", | ||
104 | ret & 7); | ||
105 | |||
106 | cs42l51 = kzalloc(sizeof(struct cs42l51_private), GFP_KERNEL); | ||
107 | if (!cs42l51) { | ||
108 | dev_err(&i2c_client->dev, "could not allocate codec\n"); | ||
109 | return -ENOMEM; | ||
110 | } | ||
111 | codec = &cs42l51->codec; | ||
112 | |||
113 | mutex_init(&codec->mutex); | ||
114 | INIT_LIST_HEAD(&codec->dapm_widgets); | ||
115 | INIT_LIST_HEAD(&codec->dapm_paths); | ||
116 | |||
117 | codec->dev = &i2c_client->dev; | ||
118 | codec->name = "CS42L51"; | ||
119 | codec->owner = THIS_MODULE; | ||
120 | codec->dai = &cs42l51_dai; | ||
121 | codec->num_dai = 1; | ||
122 | snd_soc_codec_set_drvdata(codec, cs42l51); | ||
123 | |||
124 | codec->control_data = i2c_client; | ||
125 | codec->reg_cache = cs42l51->reg_cache; | ||
126 | codec->reg_cache_size = CS42L51_NUMREGS; | ||
127 | i2c_set_clientdata(i2c_client, codec); | ||
128 | |||
129 | ret = cs42l51_fill_cache(codec); | ||
130 | if (ret < 0) { | ||
131 | dev_err(&i2c_client->dev, "failed to fill register cache\n"); | ||
132 | goto error_alloc; | ||
133 | } | ||
134 | |||
135 | ret = snd_soc_codec_set_cache_io(codec, 8, 8, SND_SOC_I2C); | ||
136 | if (ret < 0) { | ||
137 | dev_err(&i2c_client->dev, "Failed to set cache I/O: %d\n", ret); | ||
138 | goto error_alloc; | ||
139 | } | ||
140 | |||
141 | /* | ||
142 | * DAC configuration | ||
143 | * - Use signal processor | ||
144 | * - auto mute | ||
145 | * - vol changes immediate | ||
146 | * - no de-emphasize | ||
147 | */ | ||
148 | reg = CS42L51_DAC_CTL_DATA_SEL(1) | ||
149 | | CS42L51_DAC_CTL_AMUTE | CS42L51_DAC_CTL_DACSZ(0); | ||
150 | ret = snd_soc_write(codec, CS42L51_DAC_CTL, reg); | ||
151 | if (ret < 0) | ||
152 | goto error_alloc; | ||
153 | |||
154 | cs42l51_dai.dev = codec->dev; | ||
155 | cs42l51_codec = codec; | ||
156 | |||
157 | ret = snd_soc_register_codec(codec); | ||
158 | if (ret != 0) { | ||
159 | dev_err(codec->dev, "Failed to register codec: %d\n", ret); | ||
160 | goto error_alloc; | ||
161 | } | ||
162 | |||
163 | ret = snd_soc_register_dai(&cs42l51_dai); | ||
164 | if (ret < 0) { | ||
165 | dev_err(&i2c_client->dev, "failed to register DAIe\n"); | ||
166 | goto error_reg; | ||
167 | } | ||
168 | |||
169 | return 0; | ||
170 | |||
171 | error_reg: | ||
172 | snd_soc_unregister_codec(codec); | ||
173 | error_alloc: | ||
174 | kfree(cs42l51); | ||
175 | error: | ||
176 | return ret; | ||
177 | } | ||
178 | |||
179 | static int cs42l51_i2c_remove(struct i2c_client *client) | ||
180 | { | ||
181 | struct cs42l51_private *cs42l51 = i2c_get_clientdata(client); | ||
182 | snd_soc_unregister_dai(&cs42l51_dai); | ||
183 | snd_soc_unregister_codec(cs42l51_codec); | ||
184 | cs42l51_codec = NULL; | ||
185 | kfree(cs42l51); | ||
186 | return 0; | ||
187 | } | ||
188 | |||
189 | |||
190 | static const struct i2c_device_id cs42l51_id[] = { | ||
191 | {"cs42l51", 0}, | ||
192 | {} | ||
193 | }; | ||
194 | MODULE_DEVICE_TABLE(i2c, cs42l51_id); | ||
195 | |||
196 | static struct i2c_driver cs42l51_i2c_driver = { | ||
197 | .driver = { | ||
198 | .name = "CS42L51 I2C", | ||
199 | .owner = THIS_MODULE, | ||
200 | }, | ||
201 | .id_table = cs42l51_id, | ||
202 | .probe = cs42l51_i2c_probe, | ||
203 | .remove = cs42l51_i2c_remove, | ||
204 | }; | ||
205 | |||
206 | static int cs42l51_get_chan_mix(struct snd_kcontrol *kcontrol, | 75 | static int cs42l51_get_chan_mix(struct snd_kcontrol *kcontrol, |
207 | struct snd_ctl_elem_value *ucontrol) | 76 | struct snd_ctl_elem_value *ucontrol) |
208 | { | 77 | { |
@@ -484,51 +353,8 @@ static int cs42l51_set_dai_sysclk(struct snd_soc_dai *codec_dai, | |||
484 | { | 353 | { |
485 | struct snd_soc_codec *codec = codec_dai->codec; | 354 | struct snd_soc_codec *codec = codec_dai->codec; |
486 | struct cs42l51_private *cs42l51 = snd_soc_codec_get_drvdata(codec); | 355 | struct cs42l51_private *cs42l51 = snd_soc_codec_get_drvdata(codec); |
487 | struct cs42l51_ratios *ratios = NULL; | ||
488 | int nr_ratios = 0; | ||
489 | unsigned int rates = 0; | ||
490 | unsigned int rate_min = -1; | ||
491 | unsigned int rate_max = 0; | ||
492 | int i; | ||
493 | 356 | ||
494 | cs42l51->mclk = freq; | 357 | cs42l51->mclk = freq; |
495 | |||
496 | switch (cs42l51->func) { | ||
497 | case MODE_MASTER: | ||
498 | return -EINVAL; | ||
499 | case MODE_SLAVE: | ||
500 | ratios = slave_ratios; | ||
501 | nr_ratios = ARRAY_SIZE(slave_ratios); | ||
502 | break; | ||
503 | case MODE_SLAVE_AUTO: | ||
504 | ratios = slave_auto_ratios; | ||
505 | nr_ratios = ARRAY_SIZE(slave_auto_ratios); | ||
506 | break; | ||
507 | } | ||
508 | |||
509 | for (i = 0; i < nr_ratios; i++) { | ||
510 | unsigned int rate = freq / ratios[i].ratio; | ||
511 | rates |= snd_pcm_rate_to_rate_bit(rate); | ||
512 | if (rate < rate_min) | ||
513 | rate_min = rate; | ||
514 | if (rate > rate_max) | ||
515 | rate_max = rate; | ||
516 | } | ||
517 | rates &= ~SNDRV_PCM_RATE_KNOT; | ||
518 | |||
519 | if (!rates) { | ||
520 | dev_err(codec->dev, "could not find a valid sample rate\n"); | ||
521 | return -EINVAL; | ||
522 | } | ||
523 | |||
524 | codec_dai->playback.rates = rates; | ||
525 | codec_dai->playback.rate_min = rate_min; | ||
526 | codec_dai->playback.rate_max = rate_max; | ||
527 | |||
528 | codec_dai->capture.rates = rates; | ||
529 | codec_dai->capture.rate_min = rate_min; | ||
530 | codec_dai->capture.rate_max = rate_max; | ||
531 | |||
532 | return 0; | 358 | return 0; |
533 | } | 359 | } |
534 | 360 | ||
@@ -537,8 +363,7 @@ static int cs42l51_hw_params(struct snd_pcm_substream *substream, | |||
537 | struct snd_soc_dai *dai) | 363 | struct snd_soc_dai *dai) |
538 | { | 364 | { |
539 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 365 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
540 | struct snd_soc_device *socdev = rtd->socdev; | 366 | struct snd_soc_codec *codec = rtd->codec; |
541 | struct snd_soc_codec *codec = socdev->card->codec; | ||
542 | struct cs42l51_private *cs42l51 = snd_soc_codec_get_drvdata(codec); | 367 | struct cs42l51_private *cs42l51 = snd_soc_codec_get_drvdata(codec); |
543 | int ret; | 368 | int ret; |
544 | unsigned int i; | 369 | unsigned int i; |
@@ -670,8 +495,8 @@ static struct snd_soc_dai_ops cs42l51_dai_ops = { | |||
670 | .digital_mute = cs42l51_dai_mute, | 495 | .digital_mute = cs42l51_dai_mute, |
671 | }; | 496 | }; |
672 | 497 | ||
673 | struct snd_soc_dai cs42l51_dai = { | 498 | static struct snd_soc_dai_driver cs42l51_dai = { |
674 | .name = "CS42L51 HiFi", | 499 | .name = "cs42l51-hifi", |
675 | .playback = { | 500 | .playback = { |
676 | .stream_name = "Playback", | 501 | .stream_name = "Playback", |
677 | .channels_min = 1, | 502 | .channels_min = 1, |
@@ -688,56 +513,121 @@ struct snd_soc_dai cs42l51_dai = { | |||
688 | }, | 513 | }, |
689 | .ops = &cs42l51_dai_ops, | 514 | .ops = &cs42l51_dai_ops, |
690 | }; | 515 | }; |
691 | EXPORT_SYMBOL_GPL(cs42l51_dai); | ||
692 | |||
693 | 516 | ||
694 | static int cs42l51_probe(struct platform_device *pdev) | 517 | static int cs42l51_probe(struct snd_soc_codec *codec) |
695 | { | 518 | { |
696 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | 519 | struct cs42l51_private *cs42l51 = snd_soc_codec_get_drvdata(codec); |
697 | struct snd_soc_codec *codec; | 520 | struct snd_soc_dapm_context *dapm = &codec->dapm; |
698 | int ret = 0; | 521 | int ret, reg; |
699 | 522 | ||
700 | if (!cs42l51_codec) { | 523 | codec->control_data = cs42l51->control_data; |
701 | dev_err(&pdev->dev, "CS42L51 codec not yet registered\n"); | ||
702 | return -EINVAL; | ||
703 | } | ||
704 | 524 | ||
705 | socdev->card->codec = cs42l51_codec; | 525 | ret = cs42l51_fill_cache(codec); |
706 | codec = socdev->card->codec; | 526 | if (ret < 0) { |
527 | dev_err(codec->dev, "failed to fill register cache\n"); | ||
528 | return ret; | ||
529 | } | ||
707 | 530 | ||
708 | /* Register PCMs */ | 531 | ret = snd_soc_codec_set_cache_io(codec, 8, 8, cs42l51->control_type); |
709 | ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); | ||
710 | if (ret < 0) { | 532 | if (ret < 0) { |
711 | dev_err(&pdev->dev, "failed to create PCMs\n"); | 533 | dev_err(codec->dev, "Failed to set cache I/O: %d\n", ret); |
712 | return ret; | 534 | return ret; |
713 | } | 535 | } |
714 | 536 | ||
537 | /* | ||
538 | * DAC configuration | ||
539 | * - Use signal processor | ||
540 | * - auto mute | ||
541 | * - vol changes immediate | ||
542 | * - no de-emphasize | ||
543 | */ | ||
544 | reg = CS42L51_DAC_CTL_DATA_SEL(1) | ||
545 | | CS42L51_DAC_CTL_AMUTE | CS42L51_DAC_CTL_DACSZ(0); | ||
546 | ret = snd_soc_write(codec, CS42L51_DAC_CTL, reg); | ||
547 | if (ret < 0) | ||
548 | return ret; | ||
549 | |||
715 | snd_soc_add_controls(codec, cs42l51_snd_controls, | 550 | snd_soc_add_controls(codec, cs42l51_snd_controls, |
716 | ARRAY_SIZE(cs42l51_snd_controls)); | 551 | ARRAY_SIZE(cs42l51_snd_controls)); |
717 | snd_soc_dapm_new_controls(codec, cs42l51_dapm_widgets, | 552 | snd_soc_dapm_new_controls(dapm, cs42l51_dapm_widgets, |
718 | ARRAY_SIZE(cs42l51_dapm_widgets)); | 553 | ARRAY_SIZE(cs42l51_dapm_widgets)); |
719 | snd_soc_dapm_add_routes(codec, cs42l51_routes, | 554 | snd_soc_dapm_add_routes(dapm, cs42l51_routes, |
720 | ARRAY_SIZE(cs42l51_routes)); | 555 | ARRAY_SIZE(cs42l51_routes)); |
721 | 556 | ||
722 | return 0; | 557 | return 0; |
723 | } | 558 | } |
724 | 559 | ||
560 | static struct snd_soc_codec_driver soc_codec_device_cs42l51 = { | ||
561 | .probe = cs42l51_probe, | ||
562 | .reg_cache_size = CS42L51_NUMREGS, | ||
563 | .reg_word_size = sizeof(u8), | ||
564 | }; | ||
725 | 565 | ||
726 | static int cs42l51_remove(struct platform_device *pdev) | 566 | static int cs42l51_i2c_probe(struct i2c_client *i2c_client, |
567 | const struct i2c_device_id *id) | ||
727 | { | 568 | { |
728 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | 569 | struct cs42l51_private *cs42l51; |
570 | int ret; | ||
729 | 571 | ||
730 | snd_soc_free_pcms(socdev); | 572 | /* Verify that we have a CS42L51 */ |
731 | snd_soc_dapm_free(socdev); | 573 | ret = i2c_smbus_read_byte_data(i2c_client, CS42L51_CHIP_REV_ID); |
574 | if (ret < 0) { | ||
575 | dev_err(&i2c_client->dev, "failed to read I2C\n"); | ||
576 | goto error; | ||
577 | } | ||
578 | |||
579 | if ((ret != CS42L51_MK_CHIP_REV(CS42L51_CHIP_ID, CS42L51_CHIP_REV_A)) && | ||
580 | (ret != CS42L51_MK_CHIP_REV(CS42L51_CHIP_ID, CS42L51_CHIP_REV_B))) { | ||
581 | dev_err(&i2c_client->dev, "Invalid chip id\n"); | ||
582 | ret = -ENODEV; | ||
583 | goto error; | ||
584 | } | ||
585 | |||
586 | dev_info(&i2c_client->dev, "found device cs42l51 rev %d\n", | ||
587 | ret & 7); | ||
588 | |||
589 | cs42l51 = kzalloc(sizeof(struct cs42l51_private), GFP_KERNEL); | ||
590 | if (!cs42l51) { | ||
591 | dev_err(&i2c_client->dev, "could not allocate codec\n"); | ||
592 | return -ENOMEM; | ||
593 | } | ||
594 | |||
595 | i2c_set_clientdata(i2c_client, cs42l51); | ||
596 | cs42l51->control_data = i2c_client; | ||
597 | cs42l51->control_type = SND_SOC_I2C; | ||
732 | 598 | ||
599 | ret = snd_soc_register_codec(&i2c_client->dev, | ||
600 | &soc_codec_device_cs42l51, &cs42l51_dai, 1); | ||
601 | if (ret < 0) | ||
602 | kfree(cs42l51); | ||
603 | error: | ||
604 | return ret; | ||
605 | } | ||
606 | |||
607 | static int cs42l51_i2c_remove(struct i2c_client *client) | ||
608 | { | ||
609 | struct cs42l51_private *cs42l51 = i2c_get_clientdata(client); | ||
610 | |||
611 | snd_soc_unregister_codec(&client->dev); | ||
612 | kfree(cs42l51); | ||
733 | return 0; | 613 | return 0; |
734 | } | 614 | } |
735 | 615 | ||
736 | struct snd_soc_codec_device soc_codec_device_cs42l51 = { | 616 | static const struct i2c_device_id cs42l51_id[] = { |
737 | .probe = cs42l51_probe, | 617 | {"cs42l51", 0}, |
738 | .remove = cs42l51_remove | 618 | {} |
619 | }; | ||
620 | MODULE_DEVICE_TABLE(i2c, cs42l51_id); | ||
621 | |||
622 | static struct i2c_driver cs42l51_i2c_driver = { | ||
623 | .driver = { | ||
624 | .name = "cs42l51-codec", | ||
625 | .owner = THIS_MODULE, | ||
626 | }, | ||
627 | .id_table = cs42l51_id, | ||
628 | .probe = cs42l51_i2c_probe, | ||
629 | .remove = cs42l51_i2c_remove, | ||
739 | }; | 630 | }; |
740 | EXPORT_SYMBOL_GPL(soc_codec_device_cs42l51); | ||
741 | 631 | ||
742 | static int __init cs42l51_init(void) | 632 | static int __init cs42l51_init(void) |
743 | { | 633 | { |
@@ -758,6 +648,6 @@ static void __exit cs42l51_exit(void) | |||
758 | } | 648 | } |
759 | module_exit(cs42l51_exit); | 649 | module_exit(cs42l51_exit); |
760 | 650 | ||
761 | MODULE_AUTHOR("Arnaud Patard <apatard@mandriva.com>"); | 651 | MODULE_AUTHOR("Arnaud Patard <arnaud.patard@rtp-net.org>"); |
762 | MODULE_DESCRIPTION("Cirrus Logic CS42L51 ALSA SoC Codec Driver"); | 652 | MODULE_DESCRIPTION("Cirrus Logic CS42L51 ALSA SoC Codec Driver"); |
763 | MODULE_LICENSE("GPL"); | 653 | MODULE_LICENSE("GPL"); |