18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci// 38c2ecf20Sopenharmony_ci// ASoC audio graph sound card support 48c2ecf20Sopenharmony_ci// 58c2ecf20Sopenharmony_ci// Copyright (C) 2016 Renesas Solutions Corp. 68c2ecf20Sopenharmony_ci// Kuninori Morimoto <kuninori.morimoto.gx@renesas.com> 78c2ecf20Sopenharmony_ci// 88c2ecf20Sopenharmony_ci// based on ${LINUX}/sound/soc/generic/simple-card.c 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <linux/clk.h> 118c2ecf20Sopenharmony_ci#include <linux/device.h> 128c2ecf20Sopenharmony_ci#include <linux/gpio.h> 138c2ecf20Sopenharmony_ci#include <linux/gpio/consumer.h> 148c2ecf20Sopenharmony_ci#include <linux/module.h> 158c2ecf20Sopenharmony_ci#include <linux/of.h> 168c2ecf20Sopenharmony_ci#include <linux/of_device.h> 178c2ecf20Sopenharmony_ci#include <linux/of_gpio.h> 188c2ecf20Sopenharmony_ci#include <linux/of_graph.h> 198c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 208c2ecf20Sopenharmony_ci#include <linux/string.h> 218c2ecf20Sopenharmony_ci#include <sound/simple_card_utils.h> 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_ci#define DPCM_SELECTABLE 1 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_ci#define PREFIX "audio-graph-card," 268c2ecf20Sopenharmony_ci 278c2ecf20Sopenharmony_cistatic int graph_outdrv_event(struct snd_soc_dapm_widget *w, 288c2ecf20Sopenharmony_ci struct snd_kcontrol *kcontrol, 298c2ecf20Sopenharmony_ci int event) 308c2ecf20Sopenharmony_ci{ 318c2ecf20Sopenharmony_ci struct snd_soc_dapm_context *dapm = w->dapm; 328c2ecf20Sopenharmony_ci struct asoc_simple_priv *priv = snd_soc_card_get_drvdata(dapm->card); 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci switch (event) { 358c2ecf20Sopenharmony_ci case SND_SOC_DAPM_POST_PMU: 368c2ecf20Sopenharmony_ci gpiod_set_value_cansleep(priv->pa_gpio, 1); 378c2ecf20Sopenharmony_ci break; 388c2ecf20Sopenharmony_ci case SND_SOC_DAPM_PRE_PMD: 398c2ecf20Sopenharmony_ci gpiod_set_value_cansleep(priv->pa_gpio, 0); 408c2ecf20Sopenharmony_ci break; 418c2ecf20Sopenharmony_ci default: 428c2ecf20Sopenharmony_ci return -EINVAL; 438c2ecf20Sopenharmony_ci } 448c2ecf20Sopenharmony_ci 458c2ecf20Sopenharmony_ci return 0; 468c2ecf20Sopenharmony_ci} 478c2ecf20Sopenharmony_ci 488c2ecf20Sopenharmony_cistatic const struct snd_soc_dapm_widget graph_dapm_widgets[] = { 498c2ecf20Sopenharmony_ci SND_SOC_DAPM_OUT_DRV_E("Amplifier", SND_SOC_NOPM, 508c2ecf20Sopenharmony_ci 0, 0, NULL, 0, graph_outdrv_event, 518c2ecf20Sopenharmony_ci SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD), 528c2ecf20Sopenharmony_ci}; 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_cistatic const struct snd_soc_ops graph_ops = { 558c2ecf20Sopenharmony_ci .startup = asoc_simple_startup, 568c2ecf20Sopenharmony_ci .shutdown = asoc_simple_shutdown, 578c2ecf20Sopenharmony_ci .hw_params = asoc_simple_hw_params, 588c2ecf20Sopenharmony_ci}; 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_cistatic int graph_get_dai_id(struct device_node *ep) 618c2ecf20Sopenharmony_ci{ 628c2ecf20Sopenharmony_ci struct device_node *node; 638c2ecf20Sopenharmony_ci struct device_node *endpoint; 648c2ecf20Sopenharmony_ci struct of_endpoint info; 658c2ecf20Sopenharmony_ci int i, id; 668c2ecf20Sopenharmony_ci const u32 *reg; 678c2ecf20Sopenharmony_ci int ret; 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_ci /* use driver specified DAI ID if exist */ 708c2ecf20Sopenharmony_ci ret = snd_soc_get_dai_id(ep); 718c2ecf20Sopenharmony_ci if (ret != -ENOTSUPP) 728c2ecf20Sopenharmony_ci return ret; 738c2ecf20Sopenharmony_ci 748c2ecf20Sopenharmony_ci /* use endpoint/port reg if exist */ 758c2ecf20Sopenharmony_ci ret = of_graph_parse_endpoint(ep, &info); 768c2ecf20Sopenharmony_ci if (ret == 0) { 778c2ecf20Sopenharmony_ci /* 788c2ecf20Sopenharmony_ci * Because it will count port/endpoint if it doesn't have "reg". 798c2ecf20Sopenharmony_ci * But, we can't judge whether it has "no reg", or "reg = <0>" 808c2ecf20Sopenharmony_ci * only of_graph_parse_endpoint(). 818c2ecf20Sopenharmony_ci * We need to check "reg" property 828c2ecf20Sopenharmony_ci */ 838c2ecf20Sopenharmony_ci if (of_get_property(ep, "reg", NULL)) 848c2ecf20Sopenharmony_ci return info.id; 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci node = of_get_parent(ep); 878c2ecf20Sopenharmony_ci reg = of_get_property(node, "reg", NULL); 888c2ecf20Sopenharmony_ci of_node_put(node); 898c2ecf20Sopenharmony_ci if (reg) 908c2ecf20Sopenharmony_ci return info.port; 918c2ecf20Sopenharmony_ci } 928c2ecf20Sopenharmony_ci node = of_graph_get_port_parent(ep); 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci /* 958c2ecf20Sopenharmony_ci * Non HDMI sound case, counting port/endpoint on its DT 968c2ecf20Sopenharmony_ci * is enough. Let's count it. 978c2ecf20Sopenharmony_ci */ 988c2ecf20Sopenharmony_ci i = 0; 998c2ecf20Sopenharmony_ci id = -1; 1008c2ecf20Sopenharmony_ci for_each_endpoint_of_node(node, endpoint) { 1018c2ecf20Sopenharmony_ci if (endpoint == ep) 1028c2ecf20Sopenharmony_ci id = i; 1038c2ecf20Sopenharmony_ci i++; 1048c2ecf20Sopenharmony_ci } 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_ci of_node_put(node); 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci if (id < 0) 1098c2ecf20Sopenharmony_ci return -ENODEV; 1108c2ecf20Sopenharmony_ci 1118c2ecf20Sopenharmony_ci return id; 1128c2ecf20Sopenharmony_ci} 1138c2ecf20Sopenharmony_ci 1148c2ecf20Sopenharmony_cistatic int asoc_simple_parse_dai(struct device_node *ep, 1158c2ecf20Sopenharmony_ci struct snd_soc_dai_link_component *dlc, 1168c2ecf20Sopenharmony_ci int *is_single_link) 1178c2ecf20Sopenharmony_ci{ 1188c2ecf20Sopenharmony_ci struct device_node *node; 1198c2ecf20Sopenharmony_ci struct of_phandle_args args; 1208c2ecf20Sopenharmony_ci int ret; 1218c2ecf20Sopenharmony_ci 1228c2ecf20Sopenharmony_ci if (!ep) 1238c2ecf20Sopenharmony_ci return 0; 1248c2ecf20Sopenharmony_ci 1258c2ecf20Sopenharmony_ci node = of_graph_get_port_parent(ep); 1268c2ecf20Sopenharmony_ci 1278c2ecf20Sopenharmony_ci /* Get dai->name */ 1288c2ecf20Sopenharmony_ci args.np = node; 1298c2ecf20Sopenharmony_ci args.args[0] = graph_get_dai_id(ep); 1308c2ecf20Sopenharmony_ci args.args_count = (of_graph_get_endpoint_count(node) > 1); 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_ci /* 1338c2ecf20Sopenharmony_ci * FIXME 1348c2ecf20Sopenharmony_ci * 1358c2ecf20Sopenharmony_ci * Here, dlc->dai_name is pointer to CPU/Codec DAI name. 1368c2ecf20Sopenharmony_ci * If user unbinded CPU or Codec driver, but not for Sound Card, 1378c2ecf20Sopenharmony_ci * dlc->dai_name is keeping unbinded CPU or Codec 1388c2ecf20Sopenharmony_ci * driver's pointer. 1398c2ecf20Sopenharmony_ci * 1408c2ecf20Sopenharmony_ci * If user re-bind CPU or Codec driver again, ALSA SoC will try 1418c2ecf20Sopenharmony_ci * to rebind Card via snd_soc_try_rebind_card(), but because of 1428c2ecf20Sopenharmony_ci * above reason, it might can't bind Sound Card. 1438c2ecf20Sopenharmony_ci * Because Sound Card is pointing to released dai_name pointer. 1448c2ecf20Sopenharmony_ci * 1458c2ecf20Sopenharmony_ci * To avoid this rebind Card issue, 1468c2ecf20Sopenharmony_ci * 1) It needs to alloc memory to keep dai_name eventhough 1478c2ecf20Sopenharmony_ci * CPU or Codec driver was unbinded, or 1488c2ecf20Sopenharmony_ci * 2) user need to rebind Sound Card everytime 1498c2ecf20Sopenharmony_ci * if he unbinded CPU or Codec. 1508c2ecf20Sopenharmony_ci */ 1518c2ecf20Sopenharmony_ci ret = snd_soc_get_dai_name(&args, &dlc->dai_name); 1528c2ecf20Sopenharmony_ci if (ret < 0) { 1538c2ecf20Sopenharmony_ci of_node_put(node); 1548c2ecf20Sopenharmony_ci return ret; 1558c2ecf20Sopenharmony_ci } 1568c2ecf20Sopenharmony_ci 1578c2ecf20Sopenharmony_ci dlc->of_node = node; 1588c2ecf20Sopenharmony_ci 1598c2ecf20Sopenharmony_ci if (is_single_link) 1608c2ecf20Sopenharmony_ci *is_single_link = of_graph_get_endpoint_count(node) == 1; 1618c2ecf20Sopenharmony_ci 1628c2ecf20Sopenharmony_ci return 0; 1638c2ecf20Sopenharmony_ci} 1648c2ecf20Sopenharmony_ci 1658c2ecf20Sopenharmony_cistatic void graph_parse_convert(struct device *dev, 1668c2ecf20Sopenharmony_ci struct device_node *ep, 1678c2ecf20Sopenharmony_ci struct asoc_simple_data *adata) 1688c2ecf20Sopenharmony_ci{ 1698c2ecf20Sopenharmony_ci struct device_node *top = dev->of_node; 1708c2ecf20Sopenharmony_ci struct device_node *port = of_get_parent(ep); 1718c2ecf20Sopenharmony_ci struct device_node *ports = of_get_parent(port); 1728c2ecf20Sopenharmony_ci struct device_node *node = of_graph_get_port_parent(ep); 1738c2ecf20Sopenharmony_ci 1748c2ecf20Sopenharmony_ci asoc_simple_parse_convert(dev, top, NULL, adata); 1758c2ecf20Sopenharmony_ci asoc_simple_parse_convert(dev, node, PREFIX, adata); 1768c2ecf20Sopenharmony_ci asoc_simple_parse_convert(dev, ports, NULL, adata); 1778c2ecf20Sopenharmony_ci asoc_simple_parse_convert(dev, port, NULL, adata); 1788c2ecf20Sopenharmony_ci asoc_simple_parse_convert(dev, ep, NULL, adata); 1798c2ecf20Sopenharmony_ci 1808c2ecf20Sopenharmony_ci of_node_put(port); 1818c2ecf20Sopenharmony_ci of_node_put(ports); 1828c2ecf20Sopenharmony_ci of_node_put(node); 1838c2ecf20Sopenharmony_ci} 1848c2ecf20Sopenharmony_ci 1858c2ecf20Sopenharmony_cistatic void graph_parse_mclk_fs(struct device_node *top, 1868c2ecf20Sopenharmony_ci struct device_node *ep, 1878c2ecf20Sopenharmony_ci struct simple_dai_props *props) 1888c2ecf20Sopenharmony_ci{ 1898c2ecf20Sopenharmony_ci struct device_node *port = of_get_parent(ep); 1908c2ecf20Sopenharmony_ci struct device_node *ports = of_get_parent(port); 1918c2ecf20Sopenharmony_ci struct device_node *node = of_graph_get_port_parent(ep); 1928c2ecf20Sopenharmony_ci 1938c2ecf20Sopenharmony_ci of_property_read_u32(top, "mclk-fs", &props->mclk_fs); 1948c2ecf20Sopenharmony_ci of_property_read_u32(ports, "mclk-fs", &props->mclk_fs); 1958c2ecf20Sopenharmony_ci of_property_read_u32(port, "mclk-fs", &props->mclk_fs); 1968c2ecf20Sopenharmony_ci of_property_read_u32(ep, "mclk-fs", &props->mclk_fs); 1978c2ecf20Sopenharmony_ci 1988c2ecf20Sopenharmony_ci of_node_put(port); 1998c2ecf20Sopenharmony_ci of_node_put(ports); 2008c2ecf20Sopenharmony_ci of_node_put(node); 2018c2ecf20Sopenharmony_ci} 2028c2ecf20Sopenharmony_ci 2038c2ecf20Sopenharmony_cistatic int graph_dai_link_of_dpcm(struct asoc_simple_priv *priv, 2048c2ecf20Sopenharmony_ci struct device_node *cpu_ep, 2058c2ecf20Sopenharmony_ci struct device_node *codec_ep, 2068c2ecf20Sopenharmony_ci struct link_info *li, 2078c2ecf20Sopenharmony_ci int dup_codec) 2088c2ecf20Sopenharmony_ci{ 2098c2ecf20Sopenharmony_ci struct device *dev = simple_priv_to_dev(priv); 2108c2ecf20Sopenharmony_ci struct snd_soc_dai_link *dai_link = simple_priv_to_link(priv, li->link); 2118c2ecf20Sopenharmony_ci struct simple_dai_props *dai_props = simple_priv_to_props(priv, li->link); 2128c2ecf20Sopenharmony_ci struct device_node *top = dev->of_node; 2138c2ecf20Sopenharmony_ci struct device_node *ep = li->cpu ? cpu_ep : codec_ep; 2148c2ecf20Sopenharmony_ci struct device_node *port; 2158c2ecf20Sopenharmony_ci struct device_node *ports; 2168c2ecf20Sopenharmony_ci struct device_node *node; 2178c2ecf20Sopenharmony_ci struct asoc_simple_dai *dai; 2188c2ecf20Sopenharmony_ci struct snd_soc_dai_link_component *cpus = dai_link->cpus; 2198c2ecf20Sopenharmony_ci struct snd_soc_dai_link_component *codecs = dai_link->codecs; 2208c2ecf20Sopenharmony_ci int ret; 2218c2ecf20Sopenharmony_ci 2228c2ecf20Sopenharmony_ci /* Do it all CPU endpoint, and 1st Codec endpoint */ 2238c2ecf20Sopenharmony_ci if (!li->cpu && dup_codec) 2248c2ecf20Sopenharmony_ci return 0; 2258c2ecf20Sopenharmony_ci 2268c2ecf20Sopenharmony_ci port = of_get_parent(ep); 2278c2ecf20Sopenharmony_ci ports = of_get_parent(port); 2288c2ecf20Sopenharmony_ci node = of_graph_get_port_parent(ep); 2298c2ecf20Sopenharmony_ci 2308c2ecf20Sopenharmony_ci li->link++; 2318c2ecf20Sopenharmony_ci 2328c2ecf20Sopenharmony_ci dev_dbg(dev, "link_of DPCM (%pOF)\n", ep); 2338c2ecf20Sopenharmony_ci 2348c2ecf20Sopenharmony_ci if (li->cpu) { 2358c2ecf20Sopenharmony_ci int is_single_links = 0; 2368c2ecf20Sopenharmony_ci 2378c2ecf20Sopenharmony_ci /* Codec is dummy */ 2388c2ecf20Sopenharmony_ci codecs->of_node = NULL; 2398c2ecf20Sopenharmony_ci codecs->dai_name = "snd-soc-dummy-dai"; 2408c2ecf20Sopenharmony_ci codecs->name = "snd-soc-dummy"; 2418c2ecf20Sopenharmony_ci 2428c2ecf20Sopenharmony_ci /* FE settings */ 2438c2ecf20Sopenharmony_ci dai_link->dynamic = 1; 2448c2ecf20Sopenharmony_ci dai_link->dpcm_merged_format = 1; 2458c2ecf20Sopenharmony_ci 2468c2ecf20Sopenharmony_ci dai = 2478c2ecf20Sopenharmony_ci dai_props->cpu_dai = &priv->dais[li->dais++]; 2488c2ecf20Sopenharmony_ci 2498c2ecf20Sopenharmony_ci ret = asoc_simple_parse_cpu(ep, dai_link, &is_single_links); 2508c2ecf20Sopenharmony_ci if (ret) 2518c2ecf20Sopenharmony_ci goto out_put_node; 2528c2ecf20Sopenharmony_ci 2538c2ecf20Sopenharmony_ci ret = asoc_simple_parse_clk_cpu(dev, ep, dai_link, dai); 2548c2ecf20Sopenharmony_ci if (ret < 0) 2558c2ecf20Sopenharmony_ci goto out_put_node; 2568c2ecf20Sopenharmony_ci 2578c2ecf20Sopenharmony_ci ret = asoc_simple_set_dailink_name(dev, dai_link, 2588c2ecf20Sopenharmony_ci "fe.%s", 2598c2ecf20Sopenharmony_ci cpus->dai_name); 2608c2ecf20Sopenharmony_ci if (ret < 0) 2618c2ecf20Sopenharmony_ci goto out_put_node; 2628c2ecf20Sopenharmony_ci 2638c2ecf20Sopenharmony_ci /* card->num_links includes Codec */ 2648c2ecf20Sopenharmony_ci asoc_simple_canonicalize_cpu(dai_link, is_single_links); 2658c2ecf20Sopenharmony_ci } else { 2668c2ecf20Sopenharmony_ci struct snd_soc_codec_conf *cconf; 2678c2ecf20Sopenharmony_ci 2688c2ecf20Sopenharmony_ci /* CPU is dummy */ 2698c2ecf20Sopenharmony_ci cpus->of_node = NULL; 2708c2ecf20Sopenharmony_ci cpus->dai_name = "snd-soc-dummy-dai"; 2718c2ecf20Sopenharmony_ci cpus->name = "snd-soc-dummy"; 2728c2ecf20Sopenharmony_ci 2738c2ecf20Sopenharmony_ci /* BE settings */ 2748c2ecf20Sopenharmony_ci dai_link->no_pcm = 1; 2758c2ecf20Sopenharmony_ci dai_link->be_hw_params_fixup = asoc_simple_be_hw_params_fixup; 2768c2ecf20Sopenharmony_ci 2778c2ecf20Sopenharmony_ci dai = 2788c2ecf20Sopenharmony_ci dai_props->codec_dai = &priv->dais[li->dais++]; 2798c2ecf20Sopenharmony_ci 2808c2ecf20Sopenharmony_ci cconf = 2818c2ecf20Sopenharmony_ci dai_props->codec_conf = &priv->codec_conf[li->conf++]; 2828c2ecf20Sopenharmony_ci 2838c2ecf20Sopenharmony_ci ret = asoc_simple_parse_codec(ep, dai_link); 2848c2ecf20Sopenharmony_ci if (ret < 0) 2858c2ecf20Sopenharmony_ci goto out_put_node; 2868c2ecf20Sopenharmony_ci 2878c2ecf20Sopenharmony_ci ret = asoc_simple_parse_clk_codec(dev, ep, dai_link, dai); 2888c2ecf20Sopenharmony_ci if (ret < 0) 2898c2ecf20Sopenharmony_ci goto out_put_node; 2908c2ecf20Sopenharmony_ci 2918c2ecf20Sopenharmony_ci ret = asoc_simple_set_dailink_name(dev, dai_link, 2928c2ecf20Sopenharmony_ci "be.%s", 2938c2ecf20Sopenharmony_ci codecs->dai_name); 2948c2ecf20Sopenharmony_ci if (ret < 0) 2958c2ecf20Sopenharmony_ci goto out_put_node; 2968c2ecf20Sopenharmony_ci 2978c2ecf20Sopenharmony_ci /* check "prefix" from top node */ 2988c2ecf20Sopenharmony_ci snd_soc_of_parse_node_prefix(top, cconf, codecs->of_node, 2998c2ecf20Sopenharmony_ci "prefix"); 3008c2ecf20Sopenharmony_ci snd_soc_of_parse_node_prefix(node, cconf, codecs->of_node, 3018c2ecf20Sopenharmony_ci PREFIX "prefix"); 3028c2ecf20Sopenharmony_ci snd_soc_of_parse_node_prefix(ports, cconf, codecs->of_node, 3038c2ecf20Sopenharmony_ci "prefix"); 3048c2ecf20Sopenharmony_ci snd_soc_of_parse_node_prefix(port, cconf, codecs->of_node, 3058c2ecf20Sopenharmony_ci "prefix"); 3068c2ecf20Sopenharmony_ci } 3078c2ecf20Sopenharmony_ci 3088c2ecf20Sopenharmony_ci graph_parse_convert(dev, ep, &dai_props->adata); 3098c2ecf20Sopenharmony_ci graph_parse_mclk_fs(top, ep, dai_props); 3108c2ecf20Sopenharmony_ci 3118c2ecf20Sopenharmony_ci asoc_simple_canonicalize_platform(dai_link); 3128c2ecf20Sopenharmony_ci 3138c2ecf20Sopenharmony_ci ret = asoc_simple_parse_tdm(ep, dai); 3148c2ecf20Sopenharmony_ci if (ret) 3158c2ecf20Sopenharmony_ci goto out_put_node; 3168c2ecf20Sopenharmony_ci 3178c2ecf20Sopenharmony_ci ret = asoc_simple_parse_daifmt(dev, cpu_ep, codec_ep, 3188c2ecf20Sopenharmony_ci NULL, &dai_link->dai_fmt); 3198c2ecf20Sopenharmony_ci if (ret < 0) 3208c2ecf20Sopenharmony_ci goto out_put_node; 3218c2ecf20Sopenharmony_ci 3228c2ecf20Sopenharmony_ci snd_soc_dai_link_set_capabilities(dai_link); 3238c2ecf20Sopenharmony_ci 3248c2ecf20Sopenharmony_ci dai_link->ops = &graph_ops; 3258c2ecf20Sopenharmony_ci dai_link->init = asoc_simple_dai_init; 3268c2ecf20Sopenharmony_ci 3278c2ecf20Sopenharmony_ciout_put_node: 3288c2ecf20Sopenharmony_ci of_node_put(ports); 3298c2ecf20Sopenharmony_ci of_node_put(port); 3308c2ecf20Sopenharmony_ci of_node_put(node); 3318c2ecf20Sopenharmony_ci return ret; 3328c2ecf20Sopenharmony_ci} 3338c2ecf20Sopenharmony_ci 3348c2ecf20Sopenharmony_cistatic int graph_dai_link_of(struct asoc_simple_priv *priv, 3358c2ecf20Sopenharmony_ci struct device_node *cpu_ep, 3368c2ecf20Sopenharmony_ci struct device_node *codec_ep, 3378c2ecf20Sopenharmony_ci struct link_info *li) 3388c2ecf20Sopenharmony_ci{ 3398c2ecf20Sopenharmony_ci struct device *dev = simple_priv_to_dev(priv); 3408c2ecf20Sopenharmony_ci struct snd_soc_dai_link *dai_link = simple_priv_to_link(priv, li->link); 3418c2ecf20Sopenharmony_ci struct simple_dai_props *dai_props = simple_priv_to_props(priv, li->link); 3428c2ecf20Sopenharmony_ci struct device_node *top = dev->of_node; 3438c2ecf20Sopenharmony_ci struct asoc_simple_dai *cpu_dai; 3448c2ecf20Sopenharmony_ci struct asoc_simple_dai *codec_dai; 3458c2ecf20Sopenharmony_ci int ret, single_cpu = 0; 3468c2ecf20Sopenharmony_ci 3478c2ecf20Sopenharmony_ci /* Do it only CPU turn */ 3488c2ecf20Sopenharmony_ci if (!li->cpu) 3498c2ecf20Sopenharmony_ci return 0; 3508c2ecf20Sopenharmony_ci 3518c2ecf20Sopenharmony_ci dev_dbg(dev, "link_of (%pOF)\n", cpu_ep); 3528c2ecf20Sopenharmony_ci 3538c2ecf20Sopenharmony_ci li->link++; 3548c2ecf20Sopenharmony_ci 3558c2ecf20Sopenharmony_ci cpu_dai = 3568c2ecf20Sopenharmony_ci dai_props->cpu_dai = &priv->dais[li->dais++]; 3578c2ecf20Sopenharmony_ci codec_dai = 3588c2ecf20Sopenharmony_ci dai_props->codec_dai = &priv->dais[li->dais++]; 3598c2ecf20Sopenharmony_ci 3608c2ecf20Sopenharmony_ci /* Factor to mclk, used in hw_params() */ 3618c2ecf20Sopenharmony_ci graph_parse_mclk_fs(top, cpu_ep, dai_props); 3628c2ecf20Sopenharmony_ci graph_parse_mclk_fs(top, codec_ep, dai_props); 3638c2ecf20Sopenharmony_ci 3648c2ecf20Sopenharmony_ci ret = asoc_simple_parse_daifmt(dev, cpu_ep, codec_ep, 3658c2ecf20Sopenharmony_ci NULL, &dai_link->dai_fmt); 3668c2ecf20Sopenharmony_ci if (ret < 0) 3678c2ecf20Sopenharmony_ci return ret; 3688c2ecf20Sopenharmony_ci 3698c2ecf20Sopenharmony_ci ret = asoc_simple_parse_cpu(cpu_ep, dai_link, &single_cpu); 3708c2ecf20Sopenharmony_ci if (ret < 0) 3718c2ecf20Sopenharmony_ci return ret; 3728c2ecf20Sopenharmony_ci 3738c2ecf20Sopenharmony_ci ret = asoc_simple_parse_codec(codec_ep, dai_link); 3748c2ecf20Sopenharmony_ci if (ret < 0) 3758c2ecf20Sopenharmony_ci return ret; 3768c2ecf20Sopenharmony_ci 3778c2ecf20Sopenharmony_ci ret = asoc_simple_parse_tdm(cpu_ep, cpu_dai); 3788c2ecf20Sopenharmony_ci if (ret < 0) 3798c2ecf20Sopenharmony_ci return ret; 3808c2ecf20Sopenharmony_ci 3818c2ecf20Sopenharmony_ci ret = asoc_simple_parse_tdm(codec_ep, codec_dai); 3828c2ecf20Sopenharmony_ci if (ret < 0) 3838c2ecf20Sopenharmony_ci return ret; 3848c2ecf20Sopenharmony_ci 3858c2ecf20Sopenharmony_ci ret = asoc_simple_parse_clk_cpu(dev, cpu_ep, dai_link, cpu_dai); 3868c2ecf20Sopenharmony_ci if (ret < 0) 3878c2ecf20Sopenharmony_ci return ret; 3888c2ecf20Sopenharmony_ci 3898c2ecf20Sopenharmony_ci ret = asoc_simple_parse_clk_codec(dev, codec_ep, dai_link, codec_dai); 3908c2ecf20Sopenharmony_ci if (ret < 0) 3918c2ecf20Sopenharmony_ci return ret; 3928c2ecf20Sopenharmony_ci 3938c2ecf20Sopenharmony_ci ret = asoc_simple_set_dailink_name(dev, dai_link, 3948c2ecf20Sopenharmony_ci "%s-%s", 3958c2ecf20Sopenharmony_ci dai_link->cpus->dai_name, 3968c2ecf20Sopenharmony_ci dai_link->codecs->dai_name); 3978c2ecf20Sopenharmony_ci if (ret < 0) 3988c2ecf20Sopenharmony_ci return ret; 3998c2ecf20Sopenharmony_ci 4008c2ecf20Sopenharmony_ci dai_link->ops = &graph_ops; 4018c2ecf20Sopenharmony_ci dai_link->init = asoc_simple_dai_init; 4028c2ecf20Sopenharmony_ci 4038c2ecf20Sopenharmony_ci asoc_simple_canonicalize_cpu(dai_link, single_cpu); 4048c2ecf20Sopenharmony_ci asoc_simple_canonicalize_platform(dai_link); 4058c2ecf20Sopenharmony_ci 4068c2ecf20Sopenharmony_ci return 0; 4078c2ecf20Sopenharmony_ci} 4088c2ecf20Sopenharmony_ci 4098c2ecf20Sopenharmony_cistatic int graph_for_each_link(struct asoc_simple_priv *priv, 4108c2ecf20Sopenharmony_ci struct link_info *li, 4118c2ecf20Sopenharmony_ci int (*func_noml)(struct asoc_simple_priv *priv, 4128c2ecf20Sopenharmony_ci struct device_node *cpu_ep, 4138c2ecf20Sopenharmony_ci struct device_node *codec_ep, 4148c2ecf20Sopenharmony_ci struct link_info *li), 4158c2ecf20Sopenharmony_ci int (*func_dpcm)(struct asoc_simple_priv *priv, 4168c2ecf20Sopenharmony_ci struct device_node *cpu_ep, 4178c2ecf20Sopenharmony_ci struct device_node *codec_ep, 4188c2ecf20Sopenharmony_ci struct link_info *li, int dup_codec)) 4198c2ecf20Sopenharmony_ci{ 4208c2ecf20Sopenharmony_ci struct of_phandle_iterator it; 4218c2ecf20Sopenharmony_ci struct device *dev = simple_priv_to_dev(priv); 4228c2ecf20Sopenharmony_ci struct device_node *node = dev->of_node; 4238c2ecf20Sopenharmony_ci struct device_node *cpu_port; 4248c2ecf20Sopenharmony_ci struct device_node *cpu_ep; 4258c2ecf20Sopenharmony_ci struct device_node *codec_ep; 4268c2ecf20Sopenharmony_ci struct device_node *codec_port; 4278c2ecf20Sopenharmony_ci struct device_node *codec_port_old = NULL; 4288c2ecf20Sopenharmony_ci struct asoc_simple_data adata; 4298c2ecf20Sopenharmony_ci uintptr_t dpcm_selectable = (uintptr_t)of_device_get_match_data(dev); 4308c2ecf20Sopenharmony_ci int rc, ret; 4318c2ecf20Sopenharmony_ci 4328c2ecf20Sopenharmony_ci /* loop for all listed CPU port */ 4338c2ecf20Sopenharmony_ci of_for_each_phandle(&it, rc, node, "dais", NULL, 0) { 4348c2ecf20Sopenharmony_ci cpu_port = it.node; 4358c2ecf20Sopenharmony_ci cpu_ep = NULL; 4368c2ecf20Sopenharmony_ci 4378c2ecf20Sopenharmony_ci /* loop for all CPU endpoint */ 4388c2ecf20Sopenharmony_ci while (1) { 4398c2ecf20Sopenharmony_ci cpu_ep = of_get_next_child(cpu_port, cpu_ep); 4408c2ecf20Sopenharmony_ci if (!cpu_ep) 4418c2ecf20Sopenharmony_ci break; 4428c2ecf20Sopenharmony_ci 4438c2ecf20Sopenharmony_ci /* get codec */ 4448c2ecf20Sopenharmony_ci codec_ep = of_graph_get_remote_endpoint(cpu_ep); 4458c2ecf20Sopenharmony_ci codec_port = of_get_parent(codec_ep); 4468c2ecf20Sopenharmony_ci 4478c2ecf20Sopenharmony_ci /* get convert-xxx property */ 4488c2ecf20Sopenharmony_ci memset(&adata, 0, sizeof(adata)); 4498c2ecf20Sopenharmony_ci graph_parse_convert(dev, codec_ep, &adata); 4508c2ecf20Sopenharmony_ci graph_parse_convert(dev, cpu_ep, &adata); 4518c2ecf20Sopenharmony_ci 4528c2ecf20Sopenharmony_ci /* 4538c2ecf20Sopenharmony_ci * It is DPCM 4548c2ecf20Sopenharmony_ci * if Codec port has many endpoints, 4558c2ecf20Sopenharmony_ci * or has convert-xxx property 4568c2ecf20Sopenharmony_ci */ 4578c2ecf20Sopenharmony_ci if (dpcm_selectable && 4588c2ecf20Sopenharmony_ci ((of_get_child_count(codec_port) > 1) || 4598c2ecf20Sopenharmony_ci adata.convert_rate || adata.convert_channels)) 4608c2ecf20Sopenharmony_ci ret = func_dpcm(priv, cpu_ep, codec_ep, li, 4618c2ecf20Sopenharmony_ci (codec_port_old == codec_port)); 4628c2ecf20Sopenharmony_ci /* else normal sound */ 4638c2ecf20Sopenharmony_ci else 4648c2ecf20Sopenharmony_ci ret = func_noml(priv, cpu_ep, codec_ep, li); 4658c2ecf20Sopenharmony_ci 4668c2ecf20Sopenharmony_ci of_node_put(codec_ep); 4678c2ecf20Sopenharmony_ci of_node_put(codec_port); 4688c2ecf20Sopenharmony_ci 4698c2ecf20Sopenharmony_ci if (ret < 0) { 4708c2ecf20Sopenharmony_ci of_node_put(cpu_ep); 4718c2ecf20Sopenharmony_ci return ret; 4728c2ecf20Sopenharmony_ci } 4738c2ecf20Sopenharmony_ci 4748c2ecf20Sopenharmony_ci codec_port_old = codec_port; 4758c2ecf20Sopenharmony_ci } 4768c2ecf20Sopenharmony_ci } 4778c2ecf20Sopenharmony_ci 4788c2ecf20Sopenharmony_ci return 0; 4798c2ecf20Sopenharmony_ci} 4808c2ecf20Sopenharmony_ci 4818c2ecf20Sopenharmony_cistatic int graph_parse_of(struct asoc_simple_priv *priv) 4828c2ecf20Sopenharmony_ci{ 4838c2ecf20Sopenharmony_ci struct snd_soc_card *card = simple_priv_to_card(priv); 4848c2ecf20Sopenharmony_ci struct link_info li; 4858c2ecf20Sopenharmony_ci int ret; 4868c2ecf20Sopenharmony_ci 4878c2ecf20Sopenharmony_ci ret = asoc_simple_parse_widgets(card, NULL); 4888c2ecf20Sopenharmony_ci if (ret < 0) 4898c2ecf20Sopenharmony_ci return ret; 4908c2ecf20Sopenharmony_ci 4918c2ecf20Sopenharmony_ci ret = asoc_simple_parse_routing(card, NULL); 4928c2ecf20Sopenharmony_ci if (ret < 0) 4938c2ecf20Sopenharmony_ci return ret; 4948c2ecf20Sopenharmony_ci 4958c2ecf20Sopenharmony_ci memset(&li, 0, sizeof(li)); 4968c2ecf20Sopenharmony_ci for (li.cpu = 1; li.cpu >= 0; li.cpu--) { 4978c2ecf20Sopenharmony_ci /* 4988c2ecf20Sopenharmony_ci * Detect all CPU first, and Detect all Codec 2nd. 4998c2ecf20Sopenharmony_ci * 5008c2ecf20Sopenharmony_ci * In Normal sound case, all DAIs are detected 5018c2ecf20Sopenharmony_ci * as "CPU-Codec". 5028c2ecf20Sopenharmony_ci * 5038c2ecf20Sopenharmony_ci * In DPCM sound case, 5048c2ecf20Sopenharmony_ci * all CPUs are detected as "CPU-dummy", and 5058c2ecf20Sopenharmony_ci * all Codecs are detected as "dummy-Codec". 5068c2ecf20Sopenharmony_ci * To avoid random sub-device numbering, 5078c2ecf20Sopenharmony_ci * detect "dummy-Codec" in last; 5088c2ecf20Sopenharmony_ci */ 5098c2ecf20Sopenharmony_ci ret = graph_for_each_link(priv, &li, 5108c2ecf20Sopenharmony_ci graph_dai_link_of, 5118c2ecf20Sopenharmony_ci graph_dai_link_of_dpcm); 5128c2ecf20Sopenharmony_ci if (ret < 0) 5138c2ecf20Sopenharmony_ci return ret; 5148c2ecf20Sopenharmony_ci } 5158c2ecf20Sopenharmony_ci 5168c2ecf20Sopenharmony_ci return asoc_simple_parse_card_name(card, NULL); 5178c2ecf20Sopenharmony_ci} 5188c2ecf20Sopenharmony_ci 5198c2ecf20Sopenharmony_cistatic int graph_count_noml(struct asoc_simple_priv *priv, 5208c2ecf20Sopenharmony_ci struct device_node *cpu_ep, 5218c2ecf20Sopenharmony_ci struct device_node *codec_ep, 5228c2ecf20Sopenharmony_ci struct link_info *li) 5238c2ecf20Sopenharmony_ci{ 5248c2ecf20Sopenharmony_ci struct device *dev = simple_priv_to_dev(priv); 5258c2ecf20Sopenharmony_ci 5268c2ecf20Sopenharmony_ci li->link += 1; /* 1xCPU-Codec */ 5278c2ecf20Sopenharmony_ci li->dais += 2; /* 1xCPU + 1xCodec */ 5288c2ecf20Sopenharmony_ci 5298c2ecf20Sopenharmony_ci dev_dbg(dev, "Count As Normal\n"); 5308c2ecf20Sopenharmony_ci 5318c2ecf20Sopenharmony_ci return 0; 5328c2ecf20Sopenharmony_ci} 5338c2ecf20Sopenharmony_ci 5348c2ecf20Sopenharmony_cistatic int graph_count_dpcm(struct asoc_simple_priv *priv, 5358c2ecf20Sopenharmony_ci struct device_node *cpu_ep, 5368c2ecf20Sopenharmony_ci struct device_node *codec_ep, 5378c2ecf20Sopenharmony_ci struct link_info *li, 5388c2ecf20Sopenharmony_ci int dup_codec) 5398c2ecf20Sopenharmony_ci{ 5408c2ecf20Sopenharmony_ci struct device *dev = simple_priv_to_dev(priv); 5418c2ecf20Sopenharmony_ci 5428c2ecf20Sopenharmony_ci li->link++; /* 1xCPU-dummy */ 5438c2ecf20Sopenharmony_ci li->dais++; /* 1xCPU */ 5448c2ecf20Sopenharmony_ci 5458c2ecf20Sopenharmony_ci if (!dup_codec) { 5468c2ecf20Sopenharmony_ci li->link++; /* 1xdummy-Codec */ 5478c2ecf20Sopenharmony_ci li->conf++; /* 1xdummy-Codec */ 5488c2ecf20Sopenharmony_ci li->dais++; /* 1xCodec */ 5498c2ecf20Sopenharmony_ci } 5508c2ecf20Sopenharmony_ci 5518c2ecf20Sopenharmony_ci dev_dbg(dev, "Count As DPCM\n"); 5528c2ecf20Sopenharmony_ci 5538c2ecf20Sopenharmony_ci return 0; 5548c2ecf20Sopenharmony_ci} 5558c2ecf20Sopenharmony_ci 5568c2ecf20Sopenharmony_cistatic void graph_get_dais_count(struct asoc_simple_priv *priv, 5578c2ecf20Sopenharmony_ci struct link_info *li) 5588c2ecf20Sopenharmony_ci{ 5598c2ecf20Sopenharmony_ci struct device *dev = simple_priv_to_dev(priv); 5608c2ecf20Sopenharmony_ci 5618c2ecf20Sopenharmony_ci /* 5628c2ecf20Sopenharmony_ci * link_num : number of links. 5638c2ecf20Sopenharmony_ci * CPU-Codec / CPU-dummy / dummy-Codec 5648c2ecf20Sopenharmony_ci * dais_num : number of DAIs 5658c2ecf20Sopenharmony_ci * ccnf_num : number of codec_conf 5668c2ecf20Sopenharmony_ci * same number for "dummy-Codec" 5678c2ecf20Sopenharmony_ci * 5688c2ecf20Sopenharmony_ci * ex1) 5698c2ecf20Sopenharmony_ci * CPU0 --- Codec0 link : 5 5708c2ecf20Sopenharmony_ci * CPU1 --- Codec1 dais : 7 5718c2ecf20Sopenharmony_ci * CPU2 -/ ccnf : 1 5728c2ecf20Sopenharmony_ci * CPU3 --- Codec2 5738c2ecf20Sopenharmony_ci * 5748c2ecf20Sopenharmony_ci * => 5 links = 2xCPU-Codec + 2xCPU-dummy + 1xdummy-Codec 5758c2ecf20Sopenharmony_ci * => 7 DAIs = 4xCPU + 3xCodec 5768c2ecf20Sopenharmony_ci * => 1 ccnf = 1xdummy-Codec 5778c2ecf20Sopenharmony_ci * 5788c2ecf20Sopenharmony_ci * ex2) 5798c2ecf20Sopenharmony_ci * CPU0 --- Codec0 link : 5 5808c2ecf20Sopenharmony_ci * CPU1 --- Codec1 dais : 6 5818c2ecf20Sopenharmony_ci * CPU2 -/ ccnf : 1 5828c2ecf20Sopenharmony_ci * CPU3 -/ 5838c2ecf20Sopenharmony_ci * 5848c2ecf20Sopenharmony_ci * => 5 links = 1xCPU-Codec + 3xCPU-dummy + 1xdummy-Codec 5858c2ecf20Sopenharmony_ci * => 6 DAIs = 4xCPU + 2xCodec 5868c2ecf20Sopenharmony_ci * => 1 ccnf = 1xdummy-Codec 5878c2ecf20Sopenharmony_ci * 5888c2ecf20Sopenharmony_ci * ex3) 5898c2ecf20Sopenharmony_ci * CPU0 --- Codec0 link : 6 5908c2ecf20Sopenharmony_ci * CPU1 -/ dais : 6 5918c2ecf20Sopenharmony_ci * CPU2 --- Codec1 ccnf : 2 5928c2ecf20Sopenharmony_ci * CPU3 -/ 5938c2ecf20Sopenharmony_ci * 5948c2ecf20Sopenharmony_ci * => 6 links = 0xCPU-Codec + 4xCPU-dummy + 2xdummy-Codec 5958c2ecf20Sopenharmony_ci * => 6 DAIs = 4xCPU + 2xCodec 5968c2ecf20Sopenharmony_ci * => 2 ccnf = 2xdummy-Codec 5978c2ecf20Sopenharmony_ci * 5988c2ecf20Sopenharmony_ci * ex4) 5998c2ecf20Sopenharmony_ci * CPU0 --- Codec0 (convert-rate) link : 3 6008c2ecf20Sopenharmony_ci * CPU1 --- Codec1 dais : 4 6018c2ecf20Sopenharmony_ci * ccnf : 1 6028c2ecf20Sopenharmony_ci * 6038c2ecf20Sopenharmony_ci * => 3 links = 1xCPU-Codec + 1xCPU-dummy + 1xdummy-Codec 6048c2ecf20Sopenharmony_ci * => 4 DAIs = 2xCPU + 2xCodec 6058c2ecf20Sopenharmony_ci * => 1 ccnf = 1xdummy-Codec 6068c2ecf20Sopenharmony_ci */ 6078c2ecf20Sopenharmony_ci graph_for_each_link(priv, li, 6088c2ecf20Sopenharmony_ci graph_count_noml, 6098c2ecf20Sopenharmony_ci graph_count_dpcm); 6108c2ecf20Sopenharmony_ci dev_dbg(dev, "link %d, dais %d, ccnf %d\n", 6118c2ecf20Sopenharmony_ci li->link, li->dais, li->conf); 6128c2ecf20Sopenharmony_ci} 6138c2ecf20Sopenharmony_ci 6148c2ecf20Sopenharmony_cistatic int graph_card_probe(struct snd_soc_card *card) 6158c2ecf20Sopenharmony_ci{ 6168c2ecf20Sopenharmony_ci struct asoc_simple_priv *priv = snd_soc_card_get_drvdata(card); 6178c2ecf20Sopenharmony_ci int ret; 6188c2ecf20Sopenharmony_ci 6198c2ecf20Sopenharmony_ci ret = asoc_simple_init_hp(card, &priv->hp_jack, NULL); 6208c2ecf20Sopenharmony_ci if (ret < 0) 6218c2ecf20Sopenharmony_ci return ret; 6228c2ecf20Sopenharmony_ci 6238c2ecf20Sopenharmony_ci ret = asoc_simple_init_mic(card, &priv->mic_jack, NULL); 6248c2ecf20Sopenharmony_ci if (ret < 0) 6258c2ecf20Sopenharmony_ci return ret; 6268c2ecf20Sopenharmony_ci 6278c2ecf20Sopenharmony_ci return 0; 6288c2ecf20Sopenharmony_ci} 6298c2ecf20Sopenharmony_ci 6308c2ecf20Sopenharmony_cistatic int graph_probe(struct platform_device *pdev) 6318c2ecf20Sopenharmony_ci{ 6328c2ecf20Sopenharmony_ci struct asoc_simple_priv *priv; 6338c2ecf20Sopenharmony_ci struct device *dev = &pdev->dev; 6348c2ecf20Sopenharmony_ci struct snd_soc_card *card; 6358c2ecf20Sopenharmony_ci struct link_info li; 6368c2ecf20Sopenharmony_ci int ret; 6378c2ecf20Sopenharmony_ci 6388c2ecf20Sopenharmony_ci /* Allocate the private data and the DAI link array */ 6398c2ecf20Sopenharmony_ci priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); 6408c2ecf20Sopenharmony_ci if (!priv) 6418c2ecf20Sopenharmony_ci return -ENOMEM; 6428c2ecf20Sopenharmony_ci 6438c2ecf20Sopenharmony_ci card = simple_priv_to_card(priv); 6448c2ecf20Sopenharmony_ci card->owner = THIS_MODULE; 6458c2ecf20Sopenharmony_ci card->dev = dev; 6468c2ecf20Sopenharmony_ci card->dapm_widgets = graph_dapm_widgets; 6478c2ecf20Sopenharmony_ci card->num_dapm_widgets = ARRAY_SIZE(graph_dapm_widgets); 6488c2ecf20Sopenharmony_ci card->probe = graph_card_probe; 6498c2ecf20Sopenharmony_ci 6508c2ecf20Sopenharmony_ci memset(&li, 0, sizeof(li)); 6518c2ecf20Sopenharmony_ci graph_get_dais_count(priv, &li); 6528c2ecf20Sopenharmony_ci if (!li.link || !li.dais) 6538c2ecf20Sopenharmony_ci return -EINVAL; 6548c2ecf20Sopenharmony_ci 6558c2ecf20Sopenharmony_ci ret = asoc_simple_init_priv(priv, &li); 6568c2ecf20Sopenharmony_ci if (ret < 0) 6578c2ecf20Sopenharmony_ci return ret; 6588c2ecf20Sopenharmony_ci 6598c2ecf20Sopenharmony_ci priv->pa_gpio = devm_gpiod_get_optional(dev, "pa", GPIOD_OUT_LOW); 6608c2ecf20Sopenharmony_ci if (IS_ERR(priv->pa_gpio)) { 6618c2ecf20Sopenharmony_ci ret = PTR_ERR(priv->pa_gpio); 6628c2ecf20Sopenharmony_ci dev_err(dev, "failed to get amplifier gpio: %d\n", ret); 6638c2ecf20Sopenharmony_ci return ret; 6648c2ecf20Sopenharmony_ci } 6658c2ecf20Sopenharmony_ci 6668c2ecf20Sopenharmony_ci ret = graph_parse_of(priv); 6678c2ecf20Sopenharmony_ci if (ret < 0) { 6688c2ecf20Sopenharmony_ci if (ret != -EPROBE_DEFER) 6698c2ecf20Sopenharmony_ci dev_err(dev, "parse error %d\n", ret); 6708c2ecf20Sopenharmony_ci goto err; 6718c2ecf20Sopenharmony_ci } 6728c2ecf20Sopenharmony_ci 6738c2ecf20Sopenharmony_ci snd_soc_card_set_drvdata(card, priv); 6748c2ecf20Sopenharmony_ci 6758c2ecf20Sopenharmony_ci asoc_simple_debug_info(priv); 6768c2ecf20Sopenharmony_ci 6778c2ecf20Sopenharmony_ci ret = devm_snd_soc_register_card(dev, card); 6788c2ecf20Sopenharmony_ci if (ret < 0) 6798c2ecf20Sopenharmony_ci goto err; 6808c2ecf20Sopenharmony_ci 6818c2ecf20Sopenharmony_ci return 0; 6828c2ecf20Sopenharmony_cierr: 6838c2ecf20Sopenharmony_ci asoc_simple_clean_reference(card); 6848c2ecf20Sopenharmony_ci 6858c2ecf20Sopenharmony_ci return ret; 6868c2ecf20Sopenharmony_ci} 6878c2ecf20Sopenharmony_ci 6888c2ecf20Sopenharmony_cistatic int graph_remove(struct platform_device *pdev) 6898c2ecf20Sopenharmony_ci{ 6908c2ecf20Sopenharmony_ci struct snd_soc_card *card = platform_get_drvdata(pdev); 6918c2ecf20Sopenharmony_ci 6928c2ecf20Sopenharmony_ci return asoc_simple_clean_reference(card); 6938c2ecf20Sopenharmony_ci} 6948c2ecf20Sopenharmony_ci 6958c2ecf20Sopenharmony_cistatic const struct of_device_id graph_of_match[] = { 6968c2ecf20Sopenharmony_ci { .compatible = "audio-graph-card", }, 6978c2ecf20Sopenharmony_ci { .compatible = "audio-graph-scu-card", 6988c2ecf20Sopenharmony_ci .data = (void *)DPCM_SELECTABLE }, 6998c2ecf20Sopenharmony_ci {}, 7008c2ecf20Sopenharmony_ci}; 7018c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, graph_of_match); 7028c2ecf20Sopenharmony_ci 7038c2ecf20Sopenharmony_cistatic struct platform_driver graph_card = { 7048c2ecf20Sopenharmony_ci .driver = { 7058c2ecf20Sopenharmony_ci .name = "asoc-audio-graph-card", 7068c2ecf20Sopenharmony_ci .pm = &snd_soc_pm_ops, 7078c2ecf20Sopenharmony_ci .of_match_table = graph_of_match, 7088c2ecf20Sopenharmony_ci }, 7098c2ecf20Sopenharmony_ci .probe = graph_probe, 7108c2ecf20Sopenharmony_ci .remove = graph_remove, 7118c2ecf20Sopenharmony_ci}; 7128c2ecf20Sopenharmony_cimodule_platform_driver(graph_card); 7138c2ecf20Sopenharmony_ci 7148c2ecf20Sopenharmony_ciMODULE_ALIAS("platform:asoc-audio-graph-card"); 7158c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2"); 7168c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("ASoC Audio Graph Sound Card"); 7178c2ecf20Sopenharmony_ciMODULE_AUTHOR("Kuninori Morimoto <kuninori.morimoto.gx@renesas.com>"); 718