Lines Matching +full:audio +full:- +full:graph +full:- +full:card

1 // SPDX-License-Identifier: GPL-2.0
3 // ASoC audio graph sound card support
8 // based on ${LINUX}/sound/soc/generic/simple-card.c
46 struct snd_soc_dapm_context *dapm = w->dapm; in graph_outdrv_event()
47 struct simple_util_priv *priv = snd_soc_card_get_drvdata(dapm->card); in graph_outdrv_event()
51 gpiod_set_value_cansleep(priv->pa_gpio, 1); in graph_outdrv_event()
54 gpiod_set_value_cansleep(priv->pa_gpio, 0); in graph_outdrv_event()
57 return -EINVAL; in graph_outdrv_event()
79 if (dai && (dai->component->driver->pcm_construct || in soc_component_is_pcm()
80 (dai->driver->ops && dai->driver->ops->pcm_new))) in soc_component_is_pcm()
90 struct device_node *top = dev->of_node; in graph_parse_convert()
107 struct snd_soc_dai_link *dai_link = simple_priv_to_link(priv, li->link); in graph_parse_node()
108 struct simple_dai_props *dai_props = simple_priv_to_props(priv, li->link); in graph_parse_node()
141 struct device_node *top = dev->of_node; in graph_link_init()
142 struct snd_soc_dai_link *dai_link = simple_priv_to_link(priv, li->link); in graph_link_init()
143 struct simple_dai_props *dai_props = simple_priv_to_props(priv, li->link); in graph_link_init()
154 NULL, &dai_link->dai_fmt); in graph_link_init()
164 of_property_read_u32(top, "mclk-fs", &dai_props->mclk_fs); in graph_link_init()
165 of_property_read_u32(ports_cpu, "mclk-fs", &dai_props->mclk_fs); in graph_link_init()
166 of_property_read_u32(ports_codec, "mclk-fs", &dai_props->mclk_fs); in graph_link_init()
167 of_property_read_u32(port_cpu, "mclk-fs", &dai_props->mclk_fs); in graph_link_init()
168 of_property_read_u32(port_codec, "mclk-fs", &dai_props->mclk_fs); in graph_link_init()
169 of_property_read_u32(ep_cpu, "mclk-fs", &dai_props->mclk_fs); in graph_link_init()
170 of_property_read_u32(ep_codec, "mclk-fs", &dai_props->mclk_fs); in graph_link_init()
180 dai_link->playback_only = playback_only; in graph_link_init()
181 dai_link->capture_only = capture_only; in graph_link_init()
183 dai_link->trigger_start = trigger_start; in graph_link_init()
184 dai_link->trigger_stop = trigger_stop; in graph_link_init()
186 dai_link->init = simple_util_dai_init; in graph_link_init()
187 dai_link->ops = &graph_ops; in graph_link_init()
188 if (priv->ops) in graph_link_init()
189 dai_link->ops = priv->ops; in graph_link_init()
202 struct snd_soc_dai_link *dai_link = simple_priv_to_link(priv, li->link); in graph_dai_link_of_dpcm()
203 struct simple_dai_props *dai_props = simple_priv_to_props(priv, li->link); in graph_dai_link_of_dpcm()
204 struct device_node *top = dev->of_node; in graph_dai_link_of_dpcm()
205 struct device_node *ep = li->cpu ? cpu_ep : codec_ep; in graph_dai_link_of_dpcm()
211 if (li->cpu) { in graph_dai_link_of_dpcm()
212 struct snd_soc_card *card = simple_priv_to_card(priv); in graph_dai_link_of_dpcm() local
220 dai_link->dynamic = 1; in graph_dai_link_of_dpcm()
221 dai_link->dpcm_merged_format = 1; in graph_dai_link_of_dpcm()
228 "fe.%pOFP.%s", cpus->of_node, cpus->dai_name); in graph_dai_link_of_dpcm()
230 * In BE<->BE connections it is not required to create in graph_dai_link_of_dpcm()
234 * form a valid audio path. in graph_dai_link_of_dpcm()
236 * For example: FE <-> BE1 <-> BE2 <-> ... <-> BEn where in graph_dai_link_of_dpcm()
239 if (card->component_chaining && !soc_component_is_pcm(cpus)) { in graph_dai_link_of_dpcm()
240 dai_link->no_pcm = 1; in graph_dai_link_of_dpcm()
241 dai_link->be_hw_params_fixup = simple_util_be_hw_params_fixup; in graph_dai_link_of_dpcm()
253 dai_link->no_pcm = 1; in graph_dai_link_of_dpcm()
254 dai_link->be_hw_params_fixup = simple_util_be_hw_params_fixup; in graph_dai_link_of_dpcm()
261 "be.%pOFP.%s", codecs->of_node, codecs->dai_name); in graph_dai_link_of_dpcm()
267 snd_soc_of_parse_node_prefix(top, cconf, codecs->of_node, "prefix"); in graph_dai_link_of_dpcm()
268 snd_soc_of_parse_node_prefix(ports, cconf, codecs->of_node, "prefix"); in graph_dai_link_of_dpcm()
269 snd_soc_of_parse_node_prefix(port, cconf, codecs->of_node, "prefix"); in graph_dai_link_of_dpcm()
272 graph_parse_convert(dev, ep, &dai_props->adata); in graph_dai_link_of_dpcm()
276 li->link++; in graph_dai_link_of_dpcm()
287 struct snd_soc_dai_link *dai_link = simple_priv_to_link(priv, li->link); in graph_dai_link_of()
305 "%s-%s", cpus->dai_name, codecs->dai_name); in graph_dai_link_of()
314 li->link++; in graph_dai_link_of()
323 if (priv->force_dpcm) in parse_as_dpcm_link()
326 if (!priv->dpcm_selectable) in parse_as_dpcm_link()
332 * or has convert-xxx property in parse_as_dpcm_link()
354 struct device_node *node = dev->of_node; in __graph_for_each_link()
370 /* get convert-xxx property */ in __graph_for_each_link()
378 * Codec endpoint can be NULL for pluggable audio HW. in __graph_for_each_link()
383 if (li->cpu || in __graph_for_each_link()
388 if (li->cpu) in __graph_for_each_link()
418 * as "CPU-Codec". in graph_for_each_link()
421 * all CPUs are detected as "CPU-dummy", and in graph_for_each_link()
422 * all Codecs are detected as "dummy-Codec". in graph_for_each_link()
423 * To avoid random sub-device numbering, in graph_for_each_link()
424 * detect "dummy-Codec" in last; in graph_for_each_link()
426 for (li->cpu = 1; li->cpu >= 0; li->cpu--) { in graph_for_each_link()
441 int ret = -EINVAL; in graph_count_noml()
443 if (li->link >= SNDRV_MAX_LINKS) in graph_count_noml()
449 * simple-card.c :: simple_count_noml() in graph_count_noml()
451 li->num[li->link].cpus = 1; in graph_count_noml()
452 li->num[li->link].platforms = 1; in graph_count_noml()
454 li->num[li->link].codecs = 1; in graph_count_noml()
456 li->link += 1; /* 1xCPU-Codec */ in graph_count_noml()
470 int ret = -EINVAL; in graph_count_dpcm()
472 if (li->link >= SNDRV_MAX_LINKS) in graph_count_dpcm()
475 if (li->cpu) { in graph_count_dpcm()
479 * simple-card.c :: simple_count_noml() in graph_count_dpcm()
481 li->num[li->link].cpus = 1; in graph_count_dpcm()
482 li->num[li->link].platforms = 1; in graph_count_dpcm()
484 li->link++; /* 1xCPU-dummy */ in graph_count_dpcm()
486 li->num[li->link].codecs = 1; in graph_count_dpcm()
488 li->link++; /* 1xdummy-Codec */ in graph_count_dpcm()
502 * CPU-Codec / CPU-dummy / dummy-Codec in graph_get_dais_count()
505 * same number for "dummy-Codec" in graph_get_dais_count()
508 * CPU0 --- Codec0 link : 5 in graph_get_dais_count()
509 * CPU1 --- Codec1 dais : 7 in graph_get_dais_count()
510 * CPU2 -/ ccnf : 1 in graph_get_dais_count()
511 * CPU3 --- Codec2 in graph_get_dais_count()
513 * => 5 links = 2xCPU-Codec + 2xCPU-dummy + 1xdummy-Codec in graph_get_dais_count()
515 * => 1 ccnf = 1xdummy-Codec in graph_get_dais_count()
518 * CPU0 --- Codec0 link : 5 in graph_get_dais_count()
519 * CPU1 --- Codec1 dais : 6 in graph_get_dais_count()
520 * CPU2 -/ ccnf : 1 in graph_get_dais_count()
521 * CPU3 -/ in graph_get_dais_count()
523 * => 5 links = 1xCPU-Codec + 3xCPU-dummy + 1xdummy-Codec in graph_get_dais_count()
525 * => 1 ccnf = 1xdummy-Codec in graph_get_dais_count()
528 * CPU0 --- Codec0 link : 6 in graph_get_dais_count()
529 * CPU1 -/ dais : 6 in graph_get_dais_count()
530 * CPU2 --- Codec1 ccnf : 2 in graph_get_dais_count()
531 * CPU3 -/ in graph_get_dais_count()
533 * => 6 links = 0xCPU-Codec + 4xCPU-dummy + 2xdummy-Codec in graph_get_dais_count()
535 * => 2 ccnf = 2xdummy-Codec in graph_get_dais_count()
538 * CPU0 --- Codec0 (convert-rate) link : 3 in graph_get_dais_count()
539 * CPU1 --- Codec1 dais : 4 in graph_get_dais_count()
542 * => 3 links = 1xCPU-Codec + 1xCPU-dummy + 1xdummy-Codec in graph_get_dais_count()
544 * => 1 ccnf = 1xdummy-Codec in graph_get_dais_count()
553 struct snd_soc_card *card = simple_priv_to_card(priv); in audio_graph_parse_of() local
554 int ret = -ENOMEM; in audio_graph_parse_of()
560 card->owner = THIS_MODULE; in audio_graph_parse_of()
561 card->dev = dev; in audio_graph_parse_of()
567 ret = -EINVAL; in audio_graph_parse_of()
568 if (!li->link) in audio_graph_parse_of()
575 priv->pa_gpio = devm_gpiod_get_optional(dev, "pa", GPIOD_OUT_LOW); in audio_graph_parse_of()
576 if (IS_ERR(priv->pa_gpio)) { in audio_graph_parse_of()
577 ret = PTR_ERR(priv->pa_gpio); in audio_graph_parse_of()
582 ret = simple_util_parse_widgets(card, NULL); in audio_graph_parse_of()
586 ret = simple_util_parse_routing(card, NULL); in audio_graph_parse_of()
601 snd_soc_card_set_drvdata(card, priv); in audio_graph_parse_of()
605 ret = devm_snd_soc_register_card(dev, card); in audio_graph_parse_of()
611 simple_util_clean_reference(card); in audio_graph_parse_of()
620 struct device *dev = &pdev->dev; in graph_probe()
621 struct snd_soc_card *card; in graph_probe() local
626 return -ENOMEM; in graph_probe()
628 card = simple_priv_to_card(priv); in graph_probe()
629 card->dapm_widgets = graph_dapm_widgets; in graph_probe()
630 card->num_dapm_widgets = ARRAY_SIZE(graph_dapm_widgets); in graph_probe()
631 card->probe = graph_util_card_probe; in graph_probe()
634 priv->dpcm_selectable = 1; in graph_probe()
640 { .compatible = "audio-graph-card", },
641 { .compatible = "audio-graph-scu-card",
649 .name = "asoc-audio-graph-card",
658 MODULE_ALIAS("platform:asoc-audio-graph-card");
660 MODULE_DESCRIPTION("ASoC Audio Graph Sound Card");