18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
28c2ecf20Sopenharmony_ci// Copyright (c) 2018, Linaro Limited.
38c2ecf20Sopenharmony_ci// Copyright (c) 2018, The Linux Foundation. All rights reserved.
48c2ecf20Sopenharmony_ci
58c2ecf20Sopenharmony_ci#include <linux/module.h>
68c2ecf20Sopenharmony_ci#include "common.h"
78c2ecf20Sopenharmony_ci
88c2ecf20Sopenharmony_ciint qcom_snd_parse_of(struct snd_soc_card *card)
98c2ecf20Sopenharmony_ci{
108c2ecf20Sopenharmony_ci	struct device_node *np;
118c2ecf20Sopenharmony_ci	struct device_node *codec = NULL;
128c2ecf20Sopenharmony_ci	struct device_node *platform = NULL;
138c2ecf20Sopenharmony_ci	struct device_node *cpu = NULL;
148c2ecf20Sopenharmony_ci	struct device *dev = card->dev;
158c2ecf20Sopenharmony_ci	struct snd_soc_dai_link *link;
168c2ecf20Sopenharmony_ci	struct of_phandle_args args;
178c2ecf20Sopenharmony_ci	struct snd_soc_dai_link_component *dlc;
188c2ecf20Sopenharmony_ci	int ret, num_links;
198c2ecf20Sopenharmony_ci
208c2ecf20Sopenharmony_ci	ret = snd_soc_of_parse_card_name(card, "model");
218c2ecf20Sopenharmony_ci	if (ret == 0 && !card->name)
228c2ecf20Sopenharmony_ci		/* Deprecated, only for compatibility with old device trees */
238c2ecf20Sopenharmony_ci		ret = snd_soc_of_parse_card_name(card, "qcom,model");
248c2ecf20Sopenharmony_ci	if (ret) {
258c2ecf20Sopenharmony_ci		dev_err(dev, "Error parsing card name: %d\n", ret);
268c2ecf20Sopenharmony_ci		return ret;
278c2ecf20Sopenharmony_ci	}
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_ci	/* DAPM routes */
308c2ecf20Sopenharmony_ci	if (of_property_read_bool(dev->of_node, "audio-routing")) {
318c2ecf20Sopenharmony_ci		ret = snd_soc_of_parse_audio_routing(card, "audio-routing");
328c2ecf20Sopenharmony_ci		if (ret)
338c2ecf20Sopenharmony_ci			return ret;
348c2ecf20Sopenharmony_ci	}
358c2ecf20Sopenharmony_ci	/* Deprecated, only for compatibility with old device trees */
368c2ecf20Sopenharmony_ci	if (of_property_read_bool(dev->of_node, "qcom,audio-routing")) {
378c2ecf20Sopenharmony_ci		ret = snd_soc_of_parse_audio_routing(card, "qcom,audio-routing");
388c2ecf20Sopenharmony_ci		if (ret)
398c2ecf20Sopenharmony_ci			return ret;
408c2ecf20Sopenharmony_ci	}
418c2ecf20Sopenharmony_ci
428c2ecf20Sopenharmony_ci	ret = snd_soc_of_parse_aux_devs(card, "aux-devs");
438c2ecf20Sopenharmony_ci	if (ret)
448c2ecf20Sopenharmony_ci		return ret;
458c2ecf20Sopenharmony_ci
468c2ecf20Sopenharmony_ci	/* Populate links */
478c2ecf20Sopenharmony_ci	num_links = of_get_child_count(dev->of_node);
488c2ecf20Sopenharmony_ci
498c2ecf20Sopenharmony_ci	/* Allocate the DAI link array */
508c2ecf20Sopenharmony_ci	card->dai_link = devm_kcalloc(dev, num_links, sizeof(*link), GFP_KERNEL);
518c2ecf20Sopenharmony_ci	if (!card->dai_link)
528c2ecf20Sopenharmony_ci		return -ENOMEM;
538c2ecf20Sopenharmony_ci
548c2ecf20Sopenharmony_ci	card->num_links = num_links;
558c2ecf20Sopenharmony_ci	link = card->dai_link;
568c2ecf20Sopenharmony_ci
578c2ecf20Sopenharmony_ci	for_each_child_of_node(dev->of_node, np) {
588c2ecf20Sopenharmony_ci		dlc = devm_kzalloc(dev, 2 * sizeof(*dlc), GFP_KERNEL);
598c2ecf20Sopenharmony_ci		if (!dlc) {
608c2ecf20Sopenharmony_ci			ret = -ENOMEM;
618c2ecf20Sopenharmony_ci			goto err_put_np;
628c2ecf20Sopenharmony_ci		}
638c2ecf20Sopenharmony_ci
648c2ecf20Sopenharmony_ci		link->cpus	= &dlc[0];
658c2ecf20Sopenharmony_ci		link->platforms	= &dlc[1];
668c2ecf20Sopenharmony_ci
678c2ecf20Sopenharmony_ci		link->num_cpus		= 1;
688c2ecf20Sopenharmony_ci		link->num_platforms	= 1;
698c2ecf20Sopenharmony_ci
708c2ecf20Sopenharmony_ci		ret = of_property_read_string(np, "link-name", &link->name);
718c2ecf20Sopenharmony_ci		if (ret) {
728c2ecf20Sopenharmony_ci			dev_err(card->dev, "error getting codec dai_link name\n");
738c2ecf20Sopenharmony_ci			goto err_put_np;
748c2ecf20Sopenharmony_ci		}
758c2ecf20Sopenharmony_ci
768c2ecf20Sopenharmony_ci		cpu = of_get_child_by_name(np, "cpu");
778c2ecf20Sopenharmony_ci		platform = of_get_child_by_name(np, "platform");
788c2ecf20Sopenharmony_ci		codec = of_get_child_by_name(np, "codec");
798c2ecf20Sopenharmony_ci
808c2ecf20Sopenharmony_ci		if (!cpu) {
818c2ecf20Sopenharmony_ci			dev_err(dev, "%s: Can't find cpu DT node\n", link->name);
828c2ecf20Sopenharmony_ci			ret = -EINVAL;
838c2ecf20Sopenharmony_ci			goto err;
848c2ecf20Sopenharmony_ci		}
858c2ecf20Sopenharmony_ci
868c2ecf20Sopenharmony_ci		ret = of_parse_phandle_with_args(cpu, "sound-dai",
878c2ecf20Sopenharmony_ci					"#sound-dai-cells", 0, &args);
888c2ecf20Sopenharmony_ci		if (ret) {
898c2ecf20Sopenharmony_ci			dev_err(card->dev, "%s: error getting cpu phandle\n", link->name);
908c2ecf20Sopenharmony_ci			goto err;
918c2ecf20Sopenharmony_ci		}
928c2ecf20Sopenharmony_ci		link->cpus->of_node = args.np;
938c2ecf20Sopenharmony_ci		link->id = args.args[0];
948c2ecf20Sopenharmony_ci
958c2ecf20Sopenharmony_ci		ret = snd_soc_of_get_dai_name(cpu, &link->cpus->dai_name);
968c2ecf20Sopenharmony_ci		if (ret) {
978c2ecf20Sopenharmony_ci			if (ret != -EPROBE_DEFER)
988c2ecf20Sopenharmony_ci				dev_err(card->dev, "%s: error getting cpu dai name: %d\n",
998c2ecf20Sopenharmony_ci					link->name, ret);
1008c2ecf20Sopenharmony_ci			goto err;
1018c2ecf20Sopenharmony_ci		}
1028c2ecf20Sopenharmony_ci
1038c2ecf20Sopenharmony_ci		if (platform) {
1048c2ecf20Sopenharmony_ci			link->platforms->of_node = of_parse_phandle(platform,
1058c2ecf20Sopenharmony_ci					"sound-dai",
1068c2ecf20Sopenharmony_ci					0);
1078c2ecf20Sopenharmony_ci			if (!link->platforms->of_node) {
1088c2ecf20Sopenharmony_ci				dev_err(card->dev, "%s: platform dai not found\n", link->name);
1098c2ecf20Sopenharmony_ci				ret = -EINVAL;
1108c2ecf20Sopenharmony_ci				goto err;
1118c2ecf20Sopenharmony_ci			}
1128c2ecf20Sopenharmony_ci		} else {
1138c2ecf20Sopenharmony_ci			link->platforms->of_node = link->cpus->of_node;
1148c2ecf20Sopenharmony_ci		}
1158c2ecf20Sopenharmony_ci
1168c2ecf20Sopenharmony_ci		if (codec) {
1178c2ecf20Sopenharmony_ci			ret = snd_soc_of_get_dai_link_codecs(dev, codec, link);
1188c2ecf20Sopenharmony_ci			if (ret < 0) {
1198c2ecf20Sopenharmony_ci				if (ret != -EPROBE_DEFER)
1208c2ecf20Sopenharmony_ci					dev_err(card->dev, "%s: codec dai not found: %d\n",
1218c2ecf20Sopenharmony_ci						link->name, ret);
1228c2ecf20Sopenharmony_ci				goto err;
1238c2ecf20Sopenharmony_ci			}
1248c2ecf20Sopenharmony_ci
1258c2ecf20Sopenharmony_ci			if (platform) {
1268c2ecf20Sopenharmony_ci				/* DPCM backend */
1278c2ecf20Sopenharmony_ci				link->no_pcm = 1;
1288c2ecf20Sopenharmony_ci				link->ignore_pmdown_time = 1;
1298c2ecf20Sopenharmony_ci			}
1308c2ecf20Sopenharmony_ci		} else {
1318c2ecf20Sopenharmony_ci			/* DPCM frontend */
1328c2ecf20Sopenharmony_ci			dlc = devm_kzalloc(dev, sizeof(*dlc), GFP_KERNEL);
1338c2ecf20Sopenharmony_ci			if (!dlc) {
1348c2ecf20Sopenharmony_ci				ret = -ENOMEM;
1358c2ecf20Sopenharmony_ci				goto err;
1368c2ecf20Sopenharmony_ci			}
1378c2ecf20Sopenharmony_ci
1388c2ecf20Sopenharmony_ci			link->codecs	 = dlc;
1398c2ecf20Sopenharmony_ci			link->num_codecs = 1;
1408c2ecf20Sopenharmony_ci
1418c2ecf20Sopenharmony_ci			link->codecs->dai_name = "snd-soc-dummy-dai";
1428c2ecf20Sopenharmony_ci			link->codecs->name = "snd-soc-dummy";
1438c2ecf20Sopenharmony_ci			link->dynamic = 1;
1448c2ecf20Sopenharmony_ci		}
1458c2ecf20Sopenharmony_ci
1468c2ecf20Sopenharmony_ci		if (platform || !codec) {
1478c2ecf20Sopenharmony_ci			/* DPCM */
1488c2ecf20Sopenharmony_ci			snd_soc_dai_link_set_capabilities(link);
1498c2ecf20Sopenharmony_ci			link->ignore_suspend = 1;
1508c2ecf20Sopenharmony_ci			link->nonatomic = 1;
1518c2ecf20Sopenharmony_ci		}
1528c2ecf20Sopenharmony_ci
1538c2ecf20Sopenharmony_ci		link->stream_name = link->name;
1548c2ecf20Sopenharmony_ci		link++;
1558c2ecf20Sopenharmony_ci
1568c2ecf20Sopenharmony_ci		of_node_put(cpu);
1578c2ecf20Sopenharmony_ci		of_node_put(codec);
1588c2ecf20Sopenharmony_ci		of_node_put(platform);
1598c2ecf20Sopenharmony_ci	}
1608c2ecf20Sopenharmony_ci
1618c2ecf20Sopenharmony_ci	return 0;
1628c2ecf20Sopenharmony_cierr:
1638c2ecf20Sopenharmony_ci	of_node_put(cpu);
1648c2ecf20Sopenharmony_ci	of_node_put(codec);
1658c2ecf20Sopenharmony_ci	of_node_put(platform);
1668c2ecf20Sopenharmony_cierr_put_np:
1678c2ecf20Sopenharmony_ci	of_node_put(np);
1688c2ecf20Sopenharmony_ci	return ret;
1698c2ecf20Sopenharmony_ci}
1708c2ecf20Sopenharmony_ciEXPORT_SYMBOL(qcom_snd_parse_of);
1718c2ecf20Sopenharmony_ci
1728c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2");
173