1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * Handles the Mitac mioa701 SoC system
4 *
5 * Copyright (C) 2008 Robert Jarzmik
6 *
7 * This is a little schema of the sound interconnections :
8 *
9 * Sagem X200 Wolfson WM9713
10 * +--------+ +-------------------+ Rear Speaker
11 * | | | | /-+
12 * | +--->----->---+MONOIN SPKL+--->----+-+ |
13 * | GSM | | | | | |
14 * | +--->----->---+PCBEEP SPKR+--->----+-+ |
15 * | CHIP | | | \-+
16 * | +---<-----<---+MONO |
17 * | | | | Front Speaker
18 * +--------+ | | /-+
19 * | HPL+--->----+-+ |
20 * | | | | |
21 * | OUT3+--->----+-+ |
22 * | | \-+
23 * | |
24 * | | Front Micro
25 * | | +
26 * | MIC1+-----<--+o+
27 * | | +
28 * +-------------------+ ---
29 */
30
31 #include <linux/module.h>
32 #include <linux/moduleparam.h>
33 #include <linux/platform_device.h>
34
35 #include <asm/mach-types.h>
36 #include <linux/platform_data/asoc-pxa.h>
37
38 #include <sound/core.h>
39 #include <sound/pcm.h>
40 #include <sound/soc.h>
41 #include <sound/initval.h>
42 #include <sound/ac97_codec.h>
43
44 #include "../codecs/wm9713.h"
45
46 #define AC97_GPIO_PULL 0x58
47
48 /* Use GPIO8 for rear speaker amplifier */
rear_amp_power(struct snd_soc_component * component,int power)49 static int rear_amp_power(struct snd_soc_component *component, int power)
50 {
51 unsigned short reg;
52
53 if (power) {
54 reg = snd_soc_component_read(component, AC97_GPIO_CFG);
55 snd_soc_component_write(component, AC97_GPIO_CFG, reg | 0x0100);
56 reg = snd_soc_component_read(component, AC97_GPIO_PULL);
57 snd_soc_component_write(component, AC97_GPIO_PULL, reg | (1<<15));
58 } else {
59 reg = snd_soc_component_read(component, AC97_GPIO_CFG);
60 snd_soc_component_write(component, AC97_GPIO_CFG, reg & ~0x0100);
61 reg = snd_soc_component_read(component, AC97_GPIO_PULL);
62 snd_soc_component_write(component, AC97_GPIO_PULL, reg & ~(1<<15));
63 }
64
65 return 0;
66 }
67
rear_amp_event(struct snd_soc_dapm_widget * widget,struct snd_kcontrol * kctl,int event)68 static int rear_amp_event(struct snd_soc_dapm_widget *widget,
69 struct snd_kcontrol *kctl, int event)
70 {
71 struct snd_soc_card *card = widget->dapm->card;
72 struct snd_soc_pcm_runtime *rtd;
73 struct snd_soc_component *component;
74
75 rtd = snd_soc_get_pcm_runtime(card, &card->dai_link[0]);
76 component = asoc_rtd_to_codec(rtd, 0)->component;
77 return rear_amp_power(component, SND_SOC_DAPM_EVENT_ON(event));
78 }
79
80 /* mioa701 machine dapm widgets */
81 static const struct snd_soc_dapm_widget mioa701_dapm_widgets[] = {
82 SND_SOC_DAPM_SPK("Front Speaker", NULL),
83 SND_SOC_DAPM_SPK("Rear Speaker", rear_amp_event),
84 SND_SOC_DAPM_MIC("Headset", NULL),
85 SND_SOC_DAPM_LINE("GSM Line Out", NULL),
86 SND_SOC_DAPM_LINE("GSM Line In", NULL),
87 SND_SOC_DAPM_MIC("Headset Mic", NULL),
88 SND_SOC_DAPM_MIC("Front Mic", NULL),
89 };
90
91 static const struct snd_soc_dapm_route audio_map[] = {
92 /* Call Mic */
93 {"Mic Bias", NULL, "Front Mic"},
94 {"MIC1", NULL, "Mic Bias"},
95
96 /* Headset Mic */
97 {"LINEL", NULL, "Headset Mic"},
98 {"LINER", NULL, "Headset Mic"},
99
100 /* GSM Module */
101 {"MONOIN", NULL, "GSM Line Out"},
102 {"PCBEEP", NULL, "GSM Line Out"},
103 {"GSM Line In", NULL, "MONO"},
104
105 /* headphone connected to HPL, HPR */
106 {"Headset", NULL, "HPL"},
107 {"Headset", NULL, "HPR"},
108
109 /* front speaker connected to HPL, OUT3 */
110 {"Front Speaker", NULL, "HPL"},
111 {"Front Speaker", NULL, "OUT3"},
112
113 /* rear speaker connected to SPKL, SPKR */
114 {"Rear Speaker", NULL, "SPKL"},
115 {"Rear Speaker", NULL, "SPKR"},
116 };
117
mioa701_wm9713_init(struct snd_soc_pcm_runtime * rtd)118 static int mioa701_wm9713_init(struct snd_soc_pcm_runtime *rtd)
119 {
120 struct snd_soc_component *component = asoc_rtd_to_codec(rtd, 0)->component;
121
122 /* Prepare GPIO8 for rear speaker amplifier */
123 snd_soc_component_update_bits(component, AC97_GPIO_CFG, 0x100, 0x100);
124
125 /* Prepare MIC input */
126 snd_soc_component_update_bits(component, AC97_3D_CONTROL, 0xc000, 0xc000);
127
128 return 0;
129 }
130
131 static struct snd_soc_ops mioa701_ops;
132
133 SND_SOC_DAILINK_DEFS(ac97,
134 DAILINK_COMP_ARRAY(COMP_CPU("pxa2xx-ac97")),
135 DAILINK_COMP_ARRAY(COMP_CODEC("wm9713-codec", "wm9713-hifi")),
136 DAILINK_COMP_ARRAY(COMP_PLATFORM("pxa-pcm-audio")));
137
138 SND_SOC_DAILINK_DEFS(ac97_aux,
139 DAILINK_COMP_ARRAY(COMP_CPU("pxa2xx-ac97-aux")),
140 DAILINK_COMP_ARRAY(COMP_CODEC("wm9713-codec", "wm9713-aux")),
141 DAILINK_COMP_ARRAY(COMP_PLATFORM("pxa-pcm-audio")));
142
143 static struct snd_soc_dai_link mioa701_dai[] = {
144 {
145 .name = "AC97",
146 .stream_name = "AC97 HiFi",
147 .init = mioa701_wm9713_init,
148 .ops = &mioa701_ops,
149 SND_SOC_DAILINK_REG(ac97),
150 },
151 {
152 .name = "AC97 Aux",
153 .stream_name = "AC97 Aux",
154 .ops = &mioa701_ops,
155 SND_SOC_DAILINK_REG(ac97_aux),
156 },
157 };
158
159 static struct snd_soc_card mioa701 = {
160 .name = "MioA701",
161 .owner = THIS_MODULE,
162 .dai_link = mioa701_dai,
163 .num_links = ARRAY_SIZE(mioa701_dai),
164
165 .dapm_widgets = mioa701_dapm_widgets,
166 .num_dapm_widgets = ARRAY_SIZE(mioa701_dapm_widgets),
167 .dapm_routes = audio_map,
168 .num_dapm_routes = ARRAY_SIZE(audio_map),
169 };
170
mioa701_wm9713_probe(struct platform_device * pdev)171 static int mioa701_wm9713_probe(struct platform_device *pdev)
172 {
173 int rc;
174
175 if (!machine_is_mioa701())
176 return -ENODEV;
177
178 mioa701.dev = &pdev->dev;
179 rc = devm_snd_soc_register_card(&pdev->dev, &mioa701);
180 if (!rc)
181 dev_warn(&pdev->dev, "Be warned that incorrect mixers/muxes setup will "
182 "lead to overheating and possible destruction of your device."
183 " Do not use without a good knowledge of mio's board design!\n");
184 return rc;
185 }
186
187 static struct platform_driver mioa701_wm9713_driver = {
188 .probe = mioa701_wm9713_probe,
189 .driver = {
190 .name = "mioa701-wm9713",
191 .pm = &snd_soc_pm_ops,
192 },
193 };
194
195 module_platform_driver(mioa701_wm9713_driver);
196
197 /* Module information */
198 MODULE_AUTHOR("Robert Jarzmik (rjarzmik@free.fr)");
199 MODULE_DESCRIPTION("ALSA SoC WM9713 MIO A701");
200 MODULE_LICENSE("GPL");
201 MODULE_ALIAS("platform:mioa701-wm9713");
202