172b0b8b2SWesley Cheng // SPDX-License-Identifier: GPL-2.0
272b0b8b2SWesley Cheng /*
372b0b8b2SWesley Cheng * Copyright (c) 2022-2025 Qualcomm Innovation Center, Inc. All rights reserved.
472b0b8b2SWesley Cheng */
572b0b8b2SWesley Cheng
672b0b8b2SWesley Cheng #include <linux/auxiliary_bus.h>
772b0b8b2SWesley Cheng #include <linux/device.h>
872b0b8b2SWesley Cheng #include <linux/dma-mapping.h>
972b0b8b2SWesley Cheng #include <linux/dma-map-ops.h>
1072b0b8b2SWesley Cheng #include <linux/err.h>
1172b0b8b2SWesley Cheng #include <linux/init.h>
1272b0b8b2SWesley Cheng #include <linux/iommu.h>
1372b0b8b2SWesley Cheng #include <linux/module.h>
1472b0b8b2SWesley Cheng #include <linux/platform_device.h>
1572b0b8b2SWesley Cheng #include <linux/slab.h>
1672b0b8b2SWesley Cheng
1772b0b8b2SWesley Cheng #include <sound/asound.h>
181b8d0d87SWesley Cheng #include <sound/jack.h>
1972b0b8b2SWesley Cheng #include <sound/pcm.h>
2072b0b8b2SWesley Cheng #include <sound/pcm_params.h>
2172b0b8b2SWesley Cheng #include <sound/q6usboffload.h>
2272b0b8b2SWesley Cheng #include <sound/soc.h>
2372b0b8b2SWesley Cheng #include <sound/soc-usb.h>
2472b0b8b2SWesley Cheng
2572b0b8b2SWesley Cheng #include <dt-bindings/sound/qcom,q6afe.h>
2672b0b8b2SWesley Cheng
2772b0b8b2SWesley Cheng #include "q6afe.h"
2872b0b8b2SWesley Cheng #include "q6dsp-lpass-ports.h"
2972b0b8b2SWesley Cheng
3072b0b8b2SWesley Cheng #define Q6_USB_SID_MASK 0xF
3172b0b8b2SWesley Cheng
3272b0b8b2SWesley Cheng struct q6usb_port_data {
3372b0b8b2SWesley Cheng struct auxiliary_device uauxdev;
3472b0b8b2SWesley Cheng struct q6afe_usb_cfg usb_cfg;
3572b0b8b2SWesley Cheng struct snd_soc_usb *usb;
361b8d0d87SWesley Cheng struct snd_soc_jack *hs_jack;
3772b0b8b2SWesley Cheng struct q6usb_offload priv;
3872b0b8b2SWesley Cheng
3972b0b8b2SWesley Cheng /* Protects against operations between SOC USB and ASoC */
4072b0b8b2SWesley Cheng struct mutex mutex;
4172b0b8b2SWesley Cheng struct list_head devices;
4272b0b8b2SWesley Cheng };
4372b0b8b2SWesley Cheng
4472b0b8b2SWesley Cheng static const struct snd_soc_dapm_widget q6usb_dai_widgets[] = {
4572b0b8b2SWesley Cheng SND_SOC_DAPM_HP("USB_RX_BE", NULL),
4672b0b8b2SWesley Cheng };
4772b0b8b2SWesley Cheng
4872b0b8b2SWesley Cheng static const struct snd_soc_dapm_route q6usb_dapm_routes[] = {
4972b0b8b2SWesley Cheng {"USB Playback", NULL, "USB_RX_BE"},
5072b0b8b2SWesley Cheng };
5172b0b8b2SWesley Cheng
q6usb_hw_params(struct snd_pcm_substream * substream,struct snd_pcm_hw_params * params,struct snd_soc_dai * dai)5272b0b8b2SWesley Cheng static int q6usb_hw_params(struct snd_pcm_substream *substream,
5372b0b8b2SWesley Cheng struct snd_pcm_hw_params *params,
5472b0b8b2SWesley Cheng struct snd_soc_dai *dai)
5572b0b8b2SWesley Cheng {
5672b0b8b2SWesley Cheng struct q6usb_port_data *data = dev_get_drvdata(dai->dev);
5772b0b8b2SWesley Cheng struct snd_soc_pcm_runtime *rtd = substream->private_data;
5872b0b8b2SWesley Cheng struct snd_soc_dai *cpu_dai = snd_soc_rtd_to_cpu(rtd, 0);
5972b0b8b2SWesley Cheng int direction = substream->stream;
6072b0b8b2SWesley Cheng struct q6afe_port *q6usb_afe;
6172b0b8b2SWesley Cheng struct snd_soc_usb_device *sdev;
6272b0b8b2SWesley Cheng int ret = -EINVAL;
6372b0b8b2SWesley Cheng
6472b0b8b2SWesley Cheng mutex_lock(&data->mutex);
6572b0b8b2SWesley Cheng
6672b0b8b2SWesley Cheng /* No active chip index */
6772b0b8b2SWesley Cheng if (list_empty(&data->devices))
6872b0b8b2SWesley Cheng goto out;
6972b0b8b2SWesley Cheng
7072b0b8b2SWesley Cheng sdev = list_last_entry(&data->devices, struct snd_soc_usb_device, list);
7172b0b8b2SWesley Cheng
7272b0b8b2SWesley Cheng ret = snd_soc_usb_find_supported_format(sdev->chip_idx, params, direction);
7372b0b8b2SWesley Cheng if (ret < 0)
7472b0b8b2SWesley Cheng goto out;
7572b0b8b2SWesley Cheng
7672b0b8b2SWesley Cheng q6usb_afe = q6afe_port_get_from_id(cpu_dai->dev, USB_RX);
77ba6474f1SDan Carpenter if (IS_ERR(q6usb_afe)) {
78ba6474f1SDan Carpenter ret = PTR_ERR(q6usb_afe);
7972b0b8b2SWesley Cheng goto out;
80ba6474f1SDan Carpenter }
8172b0b8b2SWesley Cheng
8272b0b8b2SWesley Cheng /* Notify audio DSP about the devices being offloaded */
8372b0b8b2SWesley Cheng ret = afe_port_send_usb_dev_param(q6usb_afe, sdev->card_idx,
8472b0b8b2SWesley Cheng sdev->ppcm_idx[sdev->num_playback - 1]);
8572b0b8b2SWesley Cheng
8672b0b8b2SWesley Cheng out:
8772b0b8b2SWesley Cheng mutex_unlock(&data->mutex);
8872b0b8b2SWesley Cheng
8972b0b8b2SWesley Cheng return ret;
9072b0b8b2SWesley Cheng }
9172b0b8b2SWesley Cheng
9272b0b8b2SWesley Cheng static const struct snd_soc_dai_ops q6usb_ops = {
9372b0b8b2SWesley Cheng .hw_params = q6usb_hw_params,
9472b0b8b2SWesley Cheng };
9572b0b8b2SWesley Cheng
9672b0b8b2SWesley Cheng static struct snd_soc_dai_driver q6usb_be_dais[] = {
9772b0b8b2SWesley Cheng {
9872b0b8b2SWesley Cheng .playback = {
9972b0b8b2SWesley Cheng .stream_name = "USB BE RX",
10072b0b8b2SWesley Cheng .rates = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_11025 |
10172b0b8b2SWesley Cheng SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_22050 |
10272b0b8b2SWesley Cheng SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 |
10372b0b8b2SWesley Cheng SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_96000 |
10472b0b8b2SWesley Cheng SNDRV_PCM_RATE_192000,
10572b0b8b2SWesley Cheng .formats = SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_S16_BE |
10672b0b8b2SWesley Cheng SNDRV_PCM_FMTBIT_U16_LE | SNDRV_PCM_FMTBIT_U16_BE |
10772b0b8b2SWesley Cheng SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_S24_BE |
10872b0b8b2SWesley Cheng SNDRV_PCM_FMTBIT_U24_LE | SNDRV_PCM_FMTBIT_U24_BE,
10972b0b8b2SWesley Cheng .channels_min = 1,
11072b0b8b2SWesley Cheng .channels_max = 2,
11172b0b8b2SWesley Cheng .rate_max = 192000,
11272b0b8b2SWesley Cheng .rate_min = 8000,
11372b0b8b2SWesley Cheng },
11472b0b8b2SWesley Cheng .id = USB_RX,
11572b0b8b2SWesley Cheng .name = "USB_RX_BE",
11672b0b8b2SWesley Cheng .ops = &q6usb_ops,
11772b0b8b2SWesley Cheng },
11872b0b8b2SWesley Cheng };
11972b0b8b2SWesley Cheng
q6usb_audio_ports_of_xlate_dai_name(struct snd_soc_component * component,const struct of_phandle_args * args,const char ** dai_name)12072b0b8b2SWesley Cheng static int q6usb_audio_ports_of_xlate_dai_name(struct snd_soc_component *component,
12172b0b8b2SWesley Cheng const struct of_phandle_args *args,
12272b0b8b2SWesley Cheng const char **dai_name)
12372b0b8b2SWesley Cheng {
12472b0b8b2SWesley Cheng int id = args->args[0];
12572b0b8b2SWesley Cheng int ret = -EINVAL;
12672b0b8b2SWesley Cheng int i;
12772b0b8b2SWesley Cheng
12872b0b8b2SWesley Cheng for (i = 0; i < ARRAY_SIZE(q6usb_be_dais); i++) {
12972b0b8b2SWesley Cheng if (q6usb_be_dais[i].id == id) {
13072b0b8b2SWesley Cheng *dai_name = q6usb_be_dais[i].name;
13172b0b8b2SWesley Cheng ret = 0;
13272b0b8b2SWesley Cheng break;
13372b0b8b2SWesley Cheng }
13472b0b8b2SWesley Cheng }
13572b0b8b2SWesley Cheng
13672b0b8b2SWesley Cheng return ret;
13772b0b8b2SWesley Cheng }
13872b0b8b2SWesley Cheng
q6usb_get_pcm_id_from_widget(struct snd_soc_dapm_widget * w)139e0dd9240SWesley Cheng static int q6usb_get_pcm_id_from_widget(struct snd_soc_dapm_widget *w)
140e0dd9240SWesley Cheng {
141e0dd9240SWesley Cheng struct snd_soc_pcm_runtime *rtd;
142e0dd9240SWesley Cheng struct snd_soc_dai *dai;
143e0dd9240SWesley Cheng
144e0dd9240SWesley Cheng for_each_card_rtds(w->dapm->card, rtd) {
145e0dd9240SWesley Cheng dai = snd_soc_rtd_to_cpu(rtd, 0);
146e0dd9240SWesley Cheng /*
147e0dd9240SWesley Cheng * Only look for playback widget. RTD number carries the assigned
148e0dd9240SWesley Cheng * PCM index.
149e0dd9240SWesley Cheng */
150e0dd9240SWesley Cheng if (dai->stream[0].widget == w)
151e0dd9240SWesley Cheng return rtd->id;
152e0dd9240SWesley Cheng }
153e0dd9240SWesley Cheng
154e0dd9240SWesley Cheng return -1;
155e0dd9240SWesley Cheng }
156e0dd9240SWesley Cheng
q6usb_usb_mixer_enabled(struct snd_soc_dapm_widget * w)157e0dd9240SWesley Cheng static int q6usb_usb_mixer_enabled(struct snd_soc_dapm_widget *w)
158e0dd9240SWesley Cheng {
159e0dd9240SWesley Cheng struct snd_soc_dapm_path *p;
160e0dd9240SWesley Cheng
161e0dd9240SWesley Cheng /* Checks to ensure USB path is enabled/connected */
162e0dd9240SWesley Cheng snd_soc_dapm_widget_for_each_sink_path(w, p)
163e0dd9240SWesley Cheng if (!strcmp(p->sink->name, "USB Mixer") && p->connect)
164e0dd9240SWesley Cheng return 1;
165e0dd9240SWesley Cheng
166e0dd9240SWesley Cheng return 0;
167e0dd9240SWesley Cheng }
168e0dd9240SWesley Cheng
q6usb_get_pcm_id(struct snd_soc_component * component)169e0dd9240SWesley Cheng static int q6usb_get_pcm_id(struct snd_soc_component *component)
170e0dd9240SWesley Cheng {
171e0dd9240SWesley Cheng struct snd_soc_dapm_widget *w;
172e0dd9240SWesley Cheng struct snd_soc_dapm_path *p;
173e0dd9240SWesley Cheng int pidx;
174e0dd9240SWesley Cheng
175e0dd9240SWesley Cheng /*
176e0dd9240SWesley Cheng * Traverse widgets to find corresponding FE widget. The DAI links are
177e0dd9240SWesley Cheng * built like the following:
178e0dd9240SWesley Cheng * MultiMedia* <-> MM_DL* <-> USB Mixer*
179e0dd9240SWesley Cheng */
180e0dd9240SWesley Cheng for_each_card_widgets(component->card, w) {
181e0dd9240SWesley Cheng if (!strncmp(w->name, "MultiMedia", 10)) {
182e0dd9240SWesley Cheng /*
183e0dd9240SWesley Cheng * Look up all paths associated with the FE widget to see if
184e0dd9240SWesley Cheng * the USB BE is enabled. The sink widget is responsible to
185e0dd9240SWesley Cheng * link with the USB mixers.
186e0dd9240SWesley Cheng */
187e0dd9240SWesley Cheng snd_soc_dapm_widget_for_each_sink_path(w, p) {
188e0dd9240SWesley Cheng if (q6usb_usb_mixer_enabled(p->sink)) {
189e0dd9240SWesley Cheng pidx = q6usb_get_pcm_id_from_widget(w);
190e0dd9240SWesley Cheng return pidx;
191e0dd9240SWesley Cheng }
192e0dd9240SWesley Cheng }
193e0dd9240SWesley Cheng }
194e0dd9240SWesley Cheng }
195e0dd9240SWesley Cheng
196e0dd9240SWesley Cheng return -1;
197e0dd9240SWesley Cheng }
198e0dd9240SWesley Cheng
q6usb_update_offload_route(struct snd_soc_component * component,int card,int pcm,int direction,enum snd_soc_usb_kctl path,long * route)199e0dd9240SWesley Cheng static int q6usb_update_offload_route(struct snd_soc_component *component, int card,
200e0dd9240SWesley Cheng int pcm, int direction, enum snd_soc_usb_kctl path,
201e0dd9240SWesley Cheng long *route)
202e0dd9240SWesley Cheng {
203e0dd9240SWesley Cheng struct q6usb_port_data *data = dev_get_drvdata(component->dev);
204e0dd9240SWesley Cheng struct snd_soc_usb_device *sdev;
205e0dd9240SWesley Cheng int ret = 0;
206e0dd9240SWesley Cheng int idx = -1;
207e0dd9240SWesley Cheng
208e0dd9240SWesley Cheng mutex_lock(&data->mutex);
209e0dd9240SWesley Cheng
210e0dd9240SWesley Cheng if (list_empty(&data->devices) ||
211e0dd9240SWesley Cheng direction == SNDRV_PCM_STREAM_CAPTURE) {
212e0dd9240SWesley Cheng ret = -ENODEV;
213e0dd9240SWesley Cheng goto out;
214e0dd9240SWesley Cheng }
215e0dd9240SWesley Cheng
216e0dd9240SWesley Cheng sdev = list_last_entry(&data->devices, struct snd_soc_usb_device, list);
217e0dd9240SWesley Cheng
218e0dd9240SWesley Cheng /*
219e0dd9240SWesley Cheng * Will always look for last PCM device discovered/probed as the
220e0dd9240SWesley Cheng * active offload index.
221e0dd9240SWesley Cheng */
222e0dd9240SWesley Cheng if (card == sdev->card_idx &&
223e0dd9240SWesley Cheng pcm == sdev->ppcm_idx[sdev->num_playback - 1]) {
224e0dd9240SWesley Cheng idx = path == SND_SOC_USB_KCTL_CARD_ROUTE ?
225e0dd9240SWesley Cheng component->card->snd_card->number :
226e0dd9240SWesley Cheng q6usb_get_pcm_id(component);
227e0dd9240SWesley Cheng }
228e0dd9240SWesley Cheng
229e0dd9240SWesley Cheng out:
230e0dd9240SWesley Cheng route[0] = idx;
231e0dd9240SWesley Cheng mutex_unlock(&data->mutex);
232e0dd9240SWesley Cheng
233e0dd9240SWesley Cheng return ret;
234e0dd9240SWesley Cheng }
235e0dd9240SWesley Cheng
q6usb_alsa_connection_cb(struct snd_soc_usb * usb,struct snd_soc_usb_device * sdev,bool connected)23672b0b8b2SWesley Cheng static int q6usb_alsa_connection_cb(struct snd_soc_usb *usb,
23772b0b8b2SWesley Cheng struct snd_soc_usb_device *sdev, bool connected)
23872b0b8b2SWesley Cheng {
23972b0b8b2SWesley Cheng struct q6usb_port_data *data;
24072b0b8b2SWesley Cheng
24172b0b8b2SWesley Cheng if (!usb->component)
24272b0b8b2SWesley Cheng return -ENODEV;
24372b0b8b2SWesley Cheng
24472b0b8b2SWesley Cheng data = dev_get_drvdata(usb->component->dev);
24572b0b8b2SWesley Cheng
24672b0b8b2SWesley Cheng mutex_lock(&data->mutex);
24772b0b8b2SWesley Cheng if (connected) {
2481b8d0d87SWesley Cheng if (data->hs_jack)
2491b8d0d87SWesley Cheng snd_jack_report(data->hs_jack->jack, SND_JACK_USB);
2501b8d0d87SWesley Cheng
25172b0b8b2SWesley Cheng /* Selects the latest USB headset plugged in for offloading */
25272b0b8b2SWesley Cheng list_add_tail(&sdev->list, &data->devices);
25372b0b8b2SWesley Cheng } else {
25472b0b8b2SWesley Cheng list_del(&sdev->list);
2551b8d0d87SWesley Cheng
2561b8d0d87SWesley Cheng if (data->hs_jack)
2571b8d0d87SWesley Cheng snd_jack_report(data->hs_jack->jack, 0);
25872b0b8b2SWesley Cheng }
25972b0b8b2SWesley Cheng mutex_unlock(&data->mutex);
26072b0b8b2SWesley Cheng
26172b0b8b2SWesley Cheng return 0;
26272b0b8b2SWesley Cheng }
26372b0b8b2SWesley Cheng
q6usb_component_disable_jack(struct q6usb_port_data * data)2641b8d0d87SWesley Cheng static void q6usb_component_disable_jack(struct q6usb_port_data *data)
2651b8d0d87SWesley Cheng {
2661b8d0d87SWesley Cheng /* Offload jack has already been disabled */
2671b8d0d87SWesley Cheng if (!data->hs_jack)
2681b8d0d87SWesley Cheng return;
2691b8d0d87SWesley Cheng
2701b8d0d87SWesley Cheng snd_jack_report(data->hs_jack->jack, 0);
2711b8d0d87SWesley Cheng data->hs_jack = NULL;
2721b8d0d87SWesley Cheng }
2731b8d0d87SWesley Cheng
q6usb_component_enable_jack(struct q6usb_port_data * data,struct snd_soc_jack * jack)2741b8d0d87SWesley Cheng static void q6usb_component_enable_jack(struct q6usb_port_data *data,
2751b8d0d87SWesley Cheng struct snd_soc_jack *jack)
2761b8d0d87SWesley Cheng {
2771b8d0d87SWesley Cheng snd_jack_report(jack->jack, !list_empty(&data->devices) ? SND_JACK_USB : 0);
2781b8d0d87SWesley Cheng data->hs_jack = jack;
2791b8d0d87SWesley Cheng }
2801b8d0d87SWesley Cheng
q6usb_component_set_jack(struct snd_soc_component * component,struct snd_soc_jack * jack,void * priv)2811b8d0d87SWesley Cheng static int q6usb_component_set_jack(struct snd_soc_component *component,
2821b8d0d87SWesley Cheng struct snd_soc_jack *jack, void *priv)
2831b8d0d87SWesley Cheng {
2841b8d0d87SWesley Cheng struct q6usb_port_data *data = dev_get_drvdata(component->dev);
2851b8d0d87SWesley Cheng
2861b8d0d87SWesley Cheng mutex_lock(&data->mutex);
2871b8d0d87SWesley Cheng if (jack)
2881b8d0d87SWesley Cheng q6usb_component_enable_jack(data, jack);
2891b8d0d87SWesley Cheng else
2901b8d0d87SWesley Cheng q6usb_component_disable_jack(data);
2911b8d0d87SWesley Cheng mutex_unlock(&data->mutex);
2921b8d0d87SWesley Cheng
2931b8d0d87SWesley Cheng return 0;
2941b8d0d87SWesley Cheng }
2951b8d0d87SWesley Cheng
q6usb_dai_aux_release(struct device * dev)29672b0b8b2SWesley Cheng static void q6usb_dai_aux_release(struct device *dev) {}
29772b0b8b2SWesley Cheng
q6usb_dai_add_aux_device(struct q6usb_port_data * data,struct auxiliary_device * auxdev)29872b0b8b2SWesley Cheng static int q6usb_dai_add_aux_device(struct q6usb_port_data *data,
29972b0b8b2SWesley Cheng struct auxiliary_device *auxdev)
30072b0b8b2SWesley Cheng {
30172b0b8b2SWesley Cheng int ret;
30272b0b8b2SWesley Cheng
30372b0b8b2SWesley Cheng auxdev->dev.parent = data->priv.dev;
30472b0b8b2SWesley Cheng auxdev->dev.release = q6usb_dai_aux_release;
30572b0b8b2SWesley Cheng auxdev->name = "qc-usb-audio-offload";
30672b0b8b2SWesley Cheng
30772b0b8b2SWesley Cheng ret = auxiliary_device_init(auxdev);
30872b0b8b2SWesley Cheng if (ret)
30972b0b8b2SWesley Cheng return ret;
31072b0b8b2SWesley Cheng
31172b0b8b2SWesley Cheng ret = auxiliary_device_add(auxdev);
31272b0b8b2SWesley Cheng if (ret)
31372b0b8b2SWesley Cheng auxiliary_device_uninit(auxdev);
31472b0b8b2SWesley Cheng
31572b0b8b2SWesley Cheng return ret;
31672b0b8b2SWesley Cheng }
31772b0b8b2SWesley Cheng
q6usb_component_probe(struct snd_soc_component * component)31872b0b8b2SWesley Cheng static int q6usb_component_probe(struct snd_soc_component *component)
31972b0b8b2SWesley Cheng {
32072b0b8b2SWesley Cheng struct q6usb_port_data *data = dev_get_drvdata(component->dev);
32172b0b8b2SWesley Cheng struct snd_soc_usb *usb;
32272b0b8b2SWesley Cheng int ret;
32372b0b8b2SWesley Cheng
32472b0b8b2SWesley Cheng /* Add the QC USB SND aux device */
32572b0b8b2SWesley Cheng ret = q6usb_dai_add_aux_device(data, &data->uauxdev);
32672b0b8b2SWesley Cheng if (ret < 0)
32772b0b8b2SWesley Cheng return ret;
32872b0b8b2SWesley Cheng
32972b0b8b2SWesley Cheng usb = snd_soc_usb_allocate_port(component, &data->priv);
33072b0b8b2SWesley Cheng if (IS_ERR(usb))
33172b0b8b2SWesley Cheng return -ENOMEM;
33272b0b8b2SWesley Cheng
33372b0b8b2SWesley Cheng usb->connection_status_cb = q6usb_alsa_connection_cb;
334e0dd9240SWesley Cheng usb->update_offload_route_info = q6usb_update_offload_route;
33572b0b8b2SWesley Cheng
33672b0b8b2SWesley Cheng snd_soc_usb_add_port(usb);
33772b0b8b2SWesley Cheng data->usb = usb;
33872b0b8b2SWesley Cheng
33972b0b8b2SWesley Cheng return 0;
34072b0b8b2SWesley Cheng }
34172b0b8b2SWesley Cheng
q6usb_component_remove(struct snd_soc_component * component)34272b0b8b2SWesley Cheng static void q6usb_component_remove(struct snd_soc_component *component)
34372b0b8b2SWesley Cheng {
34472b0b8b2SWesley Cheng struct q6usb_port_data *data = dev_get_drvdata(component->dev);
34572b0b8b2SWesley Cheng
34672b0b8b2SWesley Cheng snd_soc_usb_remove_port(data->usb);
34772b0b8b2SWesley Cheng auxiliary_device_delete(&data->uauxdev);
34872b0b8b2SWesley Cheng auxiliary_device_uninit(&data->uauxdev);
34972b0b8b2SWesley Cheng snd_soc_usb_free_port(data->usb);
35072b0b8b2SWesley Cheng }
35172b0b8b2SWesley Cheng
35272b0b8b2SWesley Cheng static const struct snd_soc_component_driver q6usb_dai_component = {
35372b0b8b2SWesley Cheng .probe = q6usb_component_probe,
3541b8d0d87SWesley Cheng .set_jack = q6usb_component_set_jack,
35572b0b8b2SWesley Cheng .remove = q6usb_component_remove,
35672b0b8b2SWesley Cheng .name = "q6usb-dai-component",
35772b0b8b2SWesley Cheng .dapm_widgets = q6usb_dai_widgets,
35872b0b8b2SWesley Cheng .num_dapm_widgets = ARRAY_SIZE(q6usb_dai_widgets),
35972b0b8b2SWesley Cheng .dapm_routes = q6usb_dapm_routes,
36072b0b8b2SWesley Cheng .num_dapm_routes = ARRAY_SIZE(q6usb_dapm_routes),
36172b0b8b2SWesley Cheng .of_xlate_dai_name = q6usb_audio_ports_of_xlate_dai_name,
36272b0b8b2SWesley Cheng };
36372b0b8b2SWesley Cheng
q6usb_dai_dev_probe(struct platform_device * pdev)36472b0b8b2SWesley Cheng static int q6usb_dai_dev_probe(struct platform_device *pdev)
36572b0b8b2SWesley Cheng {
36672b0b8b2SWesley Cheng struct device_node *node = pdev->dev.of_node;
36772b0b8b2SWesley Cheng struct q6usb_port_data *data;
36872b0b8b2SWesley Cheng struct device *dev = &pdev->dev;
36972b0b8b2SWesley Cheng struct of_phandle_args args;
37072b0b8b2SWesley Cheng int ret;
37172b0b8b2SWesley Cheng
37272b0b8b2SWesley Cheng data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);
37372b0b8b2SWesley Cheng if (!data)
37472b0b8b2SWesley Cheng return -ENOMEM;
37572b0b8b2SWesley Cheng
37672b0b8b2SWesley Cheng ret = of_property_read_u16(node, "qcom,usb-audio-intr-idx",
37772b0b8b2SWesley Cheng &data->priv.intr_num);
37872b0b8b2SWesley Cheng if (ret) {
37972b0b8b2SWesley Cheng dev_err(&pdev->dev, "failed to read intr idx.\n");
38072b0b8b2SWesley Cheng return ret;
38172b0b8b2SWesley Cheng }
38272b0b8b2SWesley Cheng
38372b0b8b2SWesley Cheng ret = of_parse_phandle_with_fixed_args(node, "iommus", 1, 0, &args);
38472b0b8b2SWesley Cheng if (!ret)
38572b0b8b2SWesley Cheng data->priv.sid = args.args[0] & Q6_USB_SID_MASK;
38672b0b8b2SWesley Cheng
38772b0b8b2SWesley Cheng ret = devm_mutex_init(dev, &data->mutex);
38872b0b8b2SWesley Cheng if (ret < 0)
38972b0b8b2SWesley Cheng return ret;
39072b0b8b2SWesley Cheng
39172b0b8b2SWesley Cheng data->priv.domain = iommu_get_domain_for_dev(&pdev->dev);
39272b0b8b2SWesley Cheng
39372b0b8b2SWesley Cheng data->priv.dev = dev;
39472b0b8b2SWesley Cheng INIT_LIST_HEAD(&data->devices);
39572b0b8b2SWesley Cheng dev_set_drvdata(dev, data);
39672b0b8b2SWesley Cheng
39772b0b8b2SWesley Cheng return devm_snd_soc_register_component(dev, &q6usb_dai_component,
39872b0b8b2SWesley Cheng q6usb_be_dais, ARRAY_SIZE(q6usb_be_dais));
39972b0b8b2SWesley Cheng }
40072b0b8b2SWesley Cheng
40172b0b8b2SWesley Cheng static const struct of_device_id q6usb_dai_device_id[] = {
40272b0b8b2SWesley Cheng { .compatible = "qcom,q6usb" },
40372b0b8b2SWesley Cheng {},
40472b0b8b2SWesley Cheng };
40572b0b8b2SWesley Cheng MODULE_DEVICE_TABLE(of, q6usb_dai_device_id);
40672b0b8b2SWesley Cheng
40772b0b8b2SWesley Cheng static struct platform_driver q6usb_dai_platform_driver = {
40872b0b8b2SWesley Cheng .driver = {
40972b0b8b2SWesley Cheng .name = "q6usb-dai",
410*e2d8ae89SArnd Bergmann .of_match_table = q6usb_dai_device_id,
41172b0b8b2SWesley Cheng },
41272b0b8b2SWesley Cheng .probe = q6usb_dai_dev_probe,
41372b0b8b2SWesley Cheng /*
41472b0b8b2SWesley Cheng * Remove not required as resources are cleaned up as part of
41572b0b8b2SWesley Cheng * component removal. Others are device managed resources.
41672b0b8b2SWesley Cheng */
41772b0b8b2SWesley Cheng };
41872b0b8b2SWesley Cheng module_platform_driver(q6usb_dai_platform_driver);
41972b0b8b2SWesley Cheng
42072b0b8b2SWesley Cheng MODULE_DESCRIPTION("Q6 USB backend dai driver");
42172b0b8b2SWesley Cheng MODULE_LICENSE("GPL");
422