xref: /openbmc/linux/sound/soc/codecs/audio-iio-aux.c (revision 1d298ad8)
1*1d298ad8SHerve Codina // SPDX-License-Identifier: GPL-2.0-only
2*1d298ad8SHerve Codina //
3*1d298ad8SHerve Codina // ALSA SoC glue to use IIO devices as audio components
4*1d298ad8SHerve Codina //
5*1d298ad8SHerve Codina // Copyright 2023 CS GROUP France
6*1d298ad8SHerve Codina //
7*1d298ad8SHerve Codina // Author: Herve Codina <herve.codina@bootlin.com>
8*1d298ad8SHerve Codina 
9*1d298ad8SHerve Codina #include <linux/iio/consumer.h>
10*1d298ad8SHerve Codina #include <linux/minmax.h>
11*1d298ad8SHerve Codina #include <linux/mod_devicetable.h>
12*1d298ad8SHerve Codina #include <linux/platform_device.h>
13*1d298ad8SHerve Codina #include <linux/slab.h>
14*1d298ad8SHerve Codina #include <linux/string_helpers.h>
15*1d298ad8SHerve Codina 
16*1d298ad8SHerve Codina #include <sound/soc.h>
17*1d298ad8SHerve Codina #include <sound/tlv.h>
18*1d298ad8SHerve Codina 
19*1d298ad8SHerve Codina struct audio_iio_aux_chan {
20*1d298ad8SHerve Codina 	struct iio_channel *iio_chan;
21*1d298ad8SHerve Codina 	const char *name;
22*1d298ad8SHerve Codina 	int max;
23*1d298ad8SHerve Codina 	int min;
24*1d298ad8SHerve Codina 	bool is_invert_range;
25*1d298ad8SHerve Codina };
26*1d298ad8SHerve Codina 
27*1d298ad8SHerve Codina struct audio_iio_aux {
28*1d298ad8SHerve Codina 	struct device *dev;
29*1d298ad8SHerve Codina 	struct audio_iio_aux_chan *chans;
30*1d298ad8SHerve Codina 	unsigned int num_chans;
31*1d298ad8SHerve Codina };
32*1d298ad8SHerve Codina 
audio_iio_aux_info_volsw(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_info * uinfo)33*1d298ad8SHerve Codina static int audio_iio_aux_info_volsw(struct snd_kcontrol *kcontrol,
34*1d298ad8SHerve Codina 				    struct snd_ctl_elem_info *uinfo)
35*1d298ad8SHerve Codina {
36*1d298ad8SHerve Codina 	struct audio_iio_aux_chan *chan = (struct audio_iio_aux_chan *)kcontrol->private_value;
37*1d298ad8SHerve Codina 
38*1d298ad8SHerve Codina 	uinfo->count = 1;
39*1d298ad8SHerve Codina 	uinfo->value.integer.min = 0;
40*1d298ad8SHerve Codina 	uinfo->value.integer.max = chan->max - chan->min;
41*1d298ad8SHerve Codina 	uinfo->type = (uinfo->value.integer.max == 1) ?
42*1d298ad8SHerve Codina 			SNDRV_CTL_ELEM_TYPE_BOOLEAN : SNDRV_CTL_ELEM_TYPE_INTEGER;
43*1d298ad8SHerve Codina 	return 0;
44*1d298ad8SHerve Codina }
45*1d298ad8SHerve Codina 
audio_iio_aux_get_volsw(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)46*1d298ad8SHerve Codina static int audio_iio_aux_get_volsw(struct snd_kcontrol *kcontrol,
47*1d298ad8SHerve Codina 				   struct snd_ctl_elem_value *ucontrol)
48*1d298ad8SHerve Codina {
49*1d298ad8SHerve Codina 	struct audio_iio_aux_chan *chan = (struct audio_iio_aux_chan *)kcontrol->private_value;
50*1d298ad8SHerve Codina 	int max = chan->max;
51*1d298ad8SHerve Codina 	int min = chan->min;
52*1d298ad8SHerve Codina 	bool invert_range = chan->is_invert_range;
53*1d298ad8SHerve Codina 	int ret;
54*1d298ad8SHerve Codina 	int val;
55*1d298ad8SHerve Codina 
56*1d298ad8SHerve Codina 	ret = iio_read_channel_raw(chan->iio_chan, &val);
57*1d298ad8SHerve Codina 	if (ret < 0)
58*1d298ad8SHerve Codina 		return ret;
59*1d298ad8SHerve Codina 
60*1d298ad8SHerve Codina 	ucontrol->value.integer.value[0] = val - min;
61*1d298ad8SHerve Codina 	if (invert_range)
62*1d298ad8SHerve Codina 		ucontrol->value.integer.value[0] = max - ucontrol->value.integer.value[0];
63*1d298ad8SHerve Codina 
64*1d298ad8SHerve Codina 	return 0;
65*1d298ad8SHerve Codina }
66*1d298ad8SHerve Codina 
audio_iio_aux_put_volsw(struct snd_kcontrol * kcontrol,struct snd_ctl_elem_value * ucontrol)67*1d298ad8SHerve Codina static int audio_iio_aux_put_volsw(struct snd_kcontrol *kcontrol,
68*1d298ad8SHerve Codina 				   struct snd_ctl_elem_value *ucontrol)
69*1d298ad8SHerve Codina {
70*1d298ad8SHerve Codina 	struct audio_iio_aux_chan *chan = (struct audio_iio_aux_chan *)kcontrol->private_value;
71*1d298ad8SHerve Codina 	int max = chan->max;
72*1d298ad8SHerve Codina 	int min = chan->min;
73*1d298ad8SHerve Codina 	bool invert_range = chan->is_invert_range;
74*1d298ad8SHerve Codina 	int val;
75*1d298ad8SHerve Codina 	int ret;
76*1d298ad8SHerve Codina 	int tmp;
77*1d298ad8SHerve Codina 
78*1d298ad8SHerve Codina 	val = ucontrol->value.integer.value[0];
79*1d298ad8SHerve Codina 	if (val < 0)
80*1d298ad8SHerve Codina 		return -EINVAL;
81*1d298ad8SHerve Codina 	if (val > max - min)
82*1d298ad8SHerve Codina 		return -EINVAL;
83*1d298ad8SHerve Codina 
84*1d298ad8SHerve Codina 	val = val + min;
85*1d298ad8SHerve Codina 	if (invert_range)
86*1d298ad8SHerve Codina 		val = max - val;
87*1d298ad8SHerve Codina 
88*1d298ad8SHerve Codina 	ret = iio_read_channel_raw(chan->iio_chan, &tmp);
89*1d298ad8SHerve Codina 	if (ret < 0)
90*1d298ad8SHerve Codina 		return ret;
91*1d298ad8SHerve Codina 
92*1d298ad8SHerve Codina 	if (tmp == val)
93*1d298ad8SHerve Codina 		return 0;
94*1d298ad8SHerve Codina 
95*1d298ad8SHerve Codina 	ret = iio_write_channel_raw(chan->iio_chan, val);
96*1d298ad8SHerve Codina 	if (ret)
97*1d298ad8SHerve Codina 		return ret;
98*1d298ad8SHerve Codina 
99*1d298ad8SHerve Codina 	return 1; /* The value changed */
100*1d298ad8SHerve Codina }
101*1d298ad8SHerve Codina 
audio_iio_aux_add_controls(struct snd_soc_component * component,struct audio_iio_aux_chan * chan)102*1d298ad8SHerve Codina static int audio_iio_aux_add_controls(struct snd_soc_component *component,
103*1d298ad8SHerve Codina 				      struct audio_iio_aux_chan *chan)
104*1d298ad8SHerve Codina {
105*1d298ad8SHerve Codina 	struct snd_kcontrol_new control = {
106*1d298ad8SHerve Codina 		.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
107*1d298ad8SHerve Codina 		.name = chan->name,
108*1d298ad8SHerve Codina 		.info = audio_iio_aux_info_volsw,
109*1d298ad8SHerve Codina 		.get = audio_iio_aux_get_volsw,
110*1d298ad8SHerve Codina 		.put = audio_iio_aux_put_volsw,
111*1d298ad8SHerve Codina 		.private_value = (unsigned long)chan,
112*1d298ad8SHerve Codina 	};
113*1d298ad8SHerve Codina 
114*1d298ad8SHerve Codina 	return snd_soc_add_component_controls(component, &control, 1);
115*1d298ad8SHerve Codina }
116*1d298ad8SHerve Codina 
117*1d298ad8SHerve Codina /*
118*1d298ad8SHerve Codina  * These data could be on stack but they are pretty big.
119*1d298ad8SHerve Codina  * As ASoC internally copy them and protect them against concurrent accesses
120*1d298ad8SHerve Codina  * (snd_soc_bind_card() protects using client_mutex), keep them in the global
121*1d298ad8SHerve Codina  * data area.
122*1d298ad8SHerve Codina  */
123*1d298ad8SHerve Codina static struct snd_soc_dapm_widget widgets[3];
124*1d298ad8SHerve Codina static struct snd_soc_dapm_route routes[2];
125*1d298ad8SHerve Codina 
126*1d298ad8SHerve Codina /* Be sure sizes are correct (need 3 widgets and 2 routes) */
127*1d298ad8SHerve Codina static_assert(ARRAY_SIZE(widgets) >= 3, "3 widgets are needed");
128*1d298ad8SHerve Codina static_assert(ARRAY_SIZE(routes) >= 2, "2 routes are needed");
129*1d298ad8SHerve Codina 
audio_iio_aux_add_dapms(struct snd_soc_component * component,struct audio_iio_aux_chan * chan)130*1d298ad8SHerve Codina static int audio_iio_aux_add_dapms(struct snd_soc_component *component,
131*1d298ad8SHerve Codina 				   struct audio_iio_aux_chan *chan)
132*1d298ad8SHerve Codina {
133*1d298ad8SHerve Codina 	struct snd_soc_dapm_context *dapm = snd_soc_component_get_dapm(component);
134*1d298ad8SHerve Codina 	char *output_name;
135*1d298ad8SHerve Codina 	char *input_name;
136*1d298ad8SHerve Codina 	char *pga_name;
137*1d298ad8SHerve Codina 	int ret;
138*1d298ad8SHerve Codina 
139*1d298ad8SHerve Codina 	input_name = kasprintf(GFP_KERNEL, "%s IN", chan->name);
140*1d298ad8SHerve Codina 	if (!input_name)
141*1d298ad8SHerve Codina 		return -ENOMEM;
142*1d298ad8SHerve Codina 
143*1d298ad8SHerve Codina 	output_name = kasprintf(GFP_KERNEL, "%s OUT", chan->name);
144*1d298ad8SHerve Codina 	if (!output_name) {
145*1d298ad8SHerve Codina 		ret = -ENOMEM;
146*1d298ad8SHerve Codina 		goto out_free_input_name;
147*1d298ad8SHerve Codina 	}
148*1d298ad8SHerve Codina 
149*1d298ad8SHerve Codina 	pga_name = kasprintf(GFP_KERNEL, "%s PGA", chan->name);
150*1d298ad8SHerve Codina 	if (!pga_name) {
151*1d298ad8SHerve Codina 		ret = -ENOMEM;
152*1d298ad8SHerve Codina 		goto out_free_output_name;
153*1d298ad8SHerve Codina 	}
154*1d298ad8SHerve Codina 
155*1d298ad8SHerve Codina 	widgets[0] = SND_SOC_DAPM_INPUT(input_name);
156*1d298ad8SHerve Codina 	widgets[1] = SND_SOC_DAPM_OUTPUT(output_name);
157*1d298ad8SHerve Codina 	widgets[2] = SND_SOC_DAPM_PGA(pga_name, SND_SOC_NOPM, 0, 0, NULL, 0);
158*1d298ad8SHerve Codina 	ret = snd_soc_dapm_new_controls(dapm, widgets, 3);
159*1d298ad8SHerve Codina 	if (ret)
160*1d298ad8SHerve Codina 		goto out_free_pga_name;
161*1d298ad8SHerve Codina 
162*1d298ad8SHerve Codina 	routes[0].sink = pga_name;
163*1d298ad8SHerve Codina 	routes[0].control = NULL;
164*1d298ad8SHerve Codina 	routes[0].source = input_name;
165*1d298ad8SHerve Codina 	routes[1].sink = output_name;
166*1d298ad8SHerve Codina 	routes[1].control = NULL;
167*1d298ad8SHerve Codina 	routes[1].source = pga_name;
168*1d298ad8SHerve Codina 	ret = snd_soc_dapm_add_routes(dapm, routes, 2);
169*1d298ad8SHerve Codina 
170*1d298ad8SHerve Codina 	/* Allocated names are no more needed (duplicated in ASoC internals) */
171*1d298ad8SHerve Codina 
172*1d298ad8SHerve Codina out_free_pga_name:
173*1d298ad8SHerve Codina 	kfree(pga_name);
174*1d298ad8SHerve Codina out_free_output_name:
175*1d298ad8SHerve Codina 	kfree(output_name);
176*1d298ad8SHerve Codina out_free_input_name:
177*1d298ad8SHerve Codina 	kfree(input_name);
178*1d298ad8SHerve Codina 	return ret;
179*1d298ad8SHerve Codina }
180*1d298ad8SHerve Codina 
audio_iio_aux_component_probe(struct snd_soc_component * component)181*1d298ad8SHerve Codina static int audio_iio_aux_component_probe(struct snd_soc_component *component)
182*1d298ad8SHerve Codina {
183*1d298ad8SHerve Codina 	struct audio_iio_aux *iio_aux = snd_soc_component_get_drvdata(component);
184*1d298ad8SHerve Codina 	struct audio_iio_aux_chan *chan;
185*1d298ad8SHerve Codina 	int ret;
186*1d298ad8SHerve Codina 	int i;
187*1d298ad8SHerve Codina 
188*1d298ad8SHerve Codina 	for (i = 0; i < iio_aux->num_chans; i++) {
189*1d298ad8SHerve Codina 		chan = iio_aux->chans + i;
190*1d298ad8SHerve Codina 
191*1d298ad8SHerve Codina 		ret = iio_read_max_channel_raw(chan->iio_chan, &chan->max);
192*1d298ad8SHerve Codina 		if (ret)
193*1d298ad8SHerve Codina 			return dev_err_probe(component->dev, ret,
194*1d298ad8SHerve Codina 					     "chan[%d] %s: Cannot get max raw value\n",
195*1d298ad8SHerve Codina 					     i, chan->name);
196*1d298ad8SHerve Codina 
197*1d298ad8SHerve Codina 		ret = iio_read_min_channel_raw(chan->iio_chan, &chan->min);
198*1d298ad8SHerve Codina 		if (ret)
199*1d298ad8SHerve Codina 			return dev_err_probe(component->dev, ret,
200*1d298ad8SHerve Codina 					     "chan[%d] %s: Cannot get min raw value\n",
201*1d298ad8SHerve Codina 					     i, chan->name);
202*1d298ad8SHerve Codina 
203*1d298ad8SHerve Codina 		if (chan->min > chan->max) {
204*1d298ad8SHerve Codina 			/*
205*1d298ad8SHerve Codina 			 * This should never happen but to avoid any check
206*1d298ad8SHerve Codina 			 * later, just swap values here to ensure that the
207*1d298ad8SHerve Codina 			 * minimum value is lower than the maximum value.
208*1d298ad8SHerve Codina 			 */
209*1d298ad8SHerve Codina 			dev_dbg(component->dev, "chan[%d] %s: Swap min and max\n",
210*1d298ad8SHerve Codina 				i, chan->name);
211*1d298ad8SHerve Codina 			swap(chan->min, chan->max);
212*1d298ad8SHerve Codina 		}
213*1d298ad8SHerve Codina 
214*1d298ad8SHerve Codina 		/* Set initial value */
215*1d298ad8SHerve Codina 		ret = iio_write_channel_raw(chan->iio_chan,
216*1d298ad8SHerve Codina 					    chan->is_invert_range ? chan->max : chan->min);
217*1d298ad8SHerve Codina 		if (ret)
218*1d298ad8SHerve Codina 			return dev_err_probe(component->dev, ret,
219*1d298ad8SHerve Codina 					     "chan[%d] %s: Cannot set initial value\n",
220*1d298ad8SHerve Codina 					     i, chan->name);
221*1d298ad8SHerve Codina 
222*1d298ad8SHerve Codina 		ret = audio_iio_aux_add_controls(component, chan);
223*1d298ad8SHerve Codina 		if (ret)
224*1d298ad8SHerve Codina 			return ret;
225*1d298ad8SHerve Codina 
226*1d298ad8SHerve Codina 		ret = audio_iio_aux_add_dapms(component, chan);
227*1d298ad8SHerve Codina 		if (ret)
228*1d298ad8SHerve Codina 			return ret;
229*1d298ad8SHerve Codina 
230*1d298ad8SHerve Codina 		dev_dbg(component->dev, "chan[%d]: Added %s (min=%d, max=%d, invert=%s)\n",
231*1d298ad8SHerve Codina 			i, chan->name, chan->min, chan->max,
232*1d298ad8SHerve Codina 			str_on_off(chan->is_invert_range));
233*1d298ad8SHerve Codina 	}
234*1d298ad8SHerve Codina 
235*1d298ad8SHerve Codina 	return 0;
236*1d298ad8SHerve Codina }
237*1d298ad8SHerve Codina 
238*1d298ad8SHerve Codina static const struct snd_soc_component_driver audio_iio_aux_component_driver = {
239*1d298ad8SHerve Codina 	.probe = audio_iio_aux_component_probe,
240*1d298ad8SHerve Codina };
241*1d298ad8SHerve Codina 
audio_iio_aux_probe(struct platform_device * pdev)242*1d298ad8SHerve Codina static int audio_iio_aux_probe(struct platform_device *pdev)
243*1d298ad8SHerve Codina {
244*1d298ad8SHerve Codina 	struct audio_iio_aux_chan *iio_aux_chan;
245*1d298ad8SHerve Codina 	struct device *dev = &pdev->dev;
246*1d298ad8SHerve Codina 	struct audio_iio_aux *iio_aux;
247*1d298ad8SHerve Codina 	const char **names;
248*1d298ad8SHerve Codina 	u32 *invert_ranges;
249*1d298ad8SHerve Codina 	int count;
250*1d298ad8SHerve Codina 	int ret;
251*1d298ad8SHerve Codina 	int i;
252*1d298ad8SHerve Codina 
253*1d298ad8SHerve Codina 	iio_aux = devm_kzalloc(dev, sizeof(*iio_aux), GFP_KERNEL);
254*1d298ad8SHerve Codina 	if (!iio_aux)
255*1d298ad8SHerve Codina 		return -ENOMEM;
256*1d298ad8SHerve Codina 
257*1d298ad8SHerve Codina 	iio_aux->dev = dev;
258*1d298ad8SHerve Codina 
259*1d298ad8SHerve Codina 	count = device_property_string_array_count(dev, "io-channel-names");
260*1d298ad8SHerve Codina 	if (count < 0)
261*1d298ad8SHerve Codina 		return dev_err_probe(dev, count, "failed to count io-channel-names\n");
262*1d298ad8SHerve Codina 
263*1d298ad8SHerve Codina 	iio_aux->num_chans = count;
264*1d298ad8SHerve Codina 
265*1d298ad8SHerve Codina 	iio_aux->chans = devm_kmalloc_array(dev, iio_aux->num_chans,
266*1d298ad8SHerve Codina 					    sizeof(*iio_aux->chans), GFP_KERNEL);
267*1d298ad8SHerve Codina 	if (!iio_aux->chans)
268*1d298ad8SHerve Codina 		return -ENOMEM;
269*1d298ad8SHerve Codina 
270*1d298ad8SHerve Codina 	names = kcalloc(iio_aux->num_chans, sizeof(*names), GFP_KERNEL);
271*1d298ad8SHerve Codina 	if (!names)
272*1d298ad8SHerve Codina 		return -ENOMEM;
273*1d298ad8SHerve Codina 
274*1d298ad8SHerve Codina 	invert_ranges = kcalloc(iio_aux->num_chans, sizeof(*invert_ranges), GFP_KERNEL);
275*1d298ad8SHerve Codina 	if (!invert_ranges) {
276*1d298ad8SHerve Codina 		ret = -ENOMEM;
277*1d298ad8SHerve Codina 		goto out_free_names;
278*1d298ad8SHerve Codina 	}
279*1d298ad8SHerve Codina 
280*1d298ad8SHerve Codina 	ret = device_property_read_string_array(dev, "io-channel-names",
281*1d298ad8SHerve Codina 						names, iio_aux->num_chans);
282*1d298ad8SHerve Codina 	if (ret < 0) {
283*1d298ad8SHerve Codina 		dev_err_probe(dev, ret, "failed to read io-channel-names\n");
284*1d298ad8SHerve Codina 		goto out_free_invert_ranges;
285*1d298ad8SHerve Codina 	}
286*1d298ad8SHerve Codina 
287*1d298ad8SHerve Codina 	/*
288*1d298ad8SHerve Codina 	 * snd-control-invert-range is optional and can contain fewer items
289*1d298ad8SHerve Codina 	 * than the number of channels. Unset values default to 0.
290*1d298ad8SHerve Codina 	 */
291*1d298ad8SHerve Codina 	count = device_property_count_u32(dev, "snd-control-invert-range");
292*1d298ad8SHerve Codina 	if (count > 0) {
293*1d298ad8SHerve Codina 		count = min_t(unsigned int, count, iio_aux->num_chans);
294*1d298ad8SHerve Codina 		ret = device_property_read_u32_array(dev, "snd-control-invert-range",
295*1d298ad8SHerve Codina 						     invert_ranges, count);
296*1d298ad8SHerve Codina 		if (ret < 0) {
297*1d298ad8SHerve Codina 			dev_err_probe(dev, ret, "failed to read snd-control-invert-range\n");
298*1d298ad8SHerve Codina 			goto out_free_invert_ranges;
299*1d298ad8SHerve Codina 		}
300*1d298ad8SHerve Codina 	}
301*1d298ad8SHerve Codina 
302*1d298ad8SHerve Codina 	for (i = 0; i < iio_aux->num_chans; i++) {
303*1d298ad8SHerve Codina 		iio_aux_chan = iio_aux->chans + i;
304*1d298ad8SHerve Codina 		iio_aux_chan->name = names[i];
305*1d298ad8SHerve Codina 		iio_aux_chan->is_invert_range = invert_ranges[i];
306*1d298ad8SHerve Codina 
307*1d298ad8SHerve Codina 		iio_aux_chan->iio_chan = devm_iio_channel_get(dev, iio_aux_chan->name);
308*1d298ad8SHerve Codina 		if (IS_ERR(iio_aux_chan->iio_chan)) {
309*1d298ad8SHerve Codina 			ret = PTR_ERR(iio_aux_chan->iio_chan);
310*1d298ad8SHerve Codina 			dev_err_probe(dev, ret, "get IIO channel '%s' failed\n",
311*1d298ad8SHerve Codina 				      iio_aux_chan->name);
312*1d298ad8SHerve Codina 			goto out_free_invert_ranges;
313*1d298ad8SHerve Codina 		}
314*1d298ad8SHerve Codina 	}
315*1d298ad8SHerve Codina 
316*1d298ad8SHerve Codina 	platform_set_drvdata(pdev, iio_aux);
317*1d298ad8SHerve Codina 
318*1d298ad8SHerve Codina 	ret = devm_snd_soc_register_component(dev, &audio_iio_aux_component_driver,
319*1d298ad8SHerve Codina 					      NULL, 0);
320*1d298ad8SHerve Codina out_free_invert_ranges:
321*1d298ad8SHerve Codina 	kfree(invert_ranges);
322*1d298ad8SHerve Codina out_free_names:
323*1d298ad8SHerve Codina 	kfree(names);
324*1d298ad8SHerve Codina 	return ret;
325*1d298ad8SHerve Codina }
326*1d298ad8SHerve Codina 
327*1d298ad8SHerve Codina static const struct of_device_id audio_iio_aux_ids[] = {
328*1d298ad8SHerve Codina 	{ .compatible = "audio-iio-aux" },
329*1d298ad8SHerve Codina 	{ }
330*1d298ad8SHerve Codina };
331*1d298ad8SHerve Codina MODULE_DEVICE_TABLE(of, audio_iio_aux_ids);
332*1d298ad8SHerve Codina 
333*1d298ad8SHerve Codina static struct platform_driver audio_iio_aux_driver = {
334*1d298ad8SHerve Codina 	.driver = {
335*1d298ad8SHerve Codina 		.name = "audio-iio-aux",
336*1d298ad8SHerve Codina 		.of_match_table = audio_iio_aux_ids,
337*1d298ad8SHerve Codina 	},
338*1d298ad8SHerve Codina 	.probe = audio_iio_aux_probe,
339*1d298ad8SHerve Codina };
340*1d298ad8SHerve Codina module_platform_driver(audio_iio_aux_driver);
341*1d298ad8SHerve Codina 
342*1d298ad8SHerve Codina MODULE_AUTHOR("Herve Codina <herve.codina@bootlin.com>");
343*1d298ad8SHerve Codina MODULE_DESCRIPTION("IIO ALSA SoC aux driver");
344*1d298ad8SHerve Codina MODULE_LICENSE("GPL");
345