Path: blob/master/sound/soc/mediatek/common/mtk-soundcard-driver.c
29268 views
// SPDX-License-Identifier: GPL-2.01/*2* mtk-soundcard-driver.c -- MediaTek soundcard driver common3*4* Copyright (c) 2022 MediaTek Inc.5* Author: Trevor Wu <[email protected]>6*/78#include <linux/module.h>9#include <linux/of.h>10#include <linux/of_platform.h>11#include <sound/soc.h>1213#include "mtk-dsp-sof-common.h"14#include "mtk-soc-card.h"15#include "mtk-soundcard-driver.h"1617static int set_card_codec_info(struct snd_soc_card *card,18struct device_node *sub_node,19struct snd_soc_dai_link *dai_link)20{21struct device *dev = card->dev;22struct device_node *codec_node;23int ret;2425codec_node = of_get_child_by_name(sub_node, "codec");26if (!codec_node) {27dev_dbg(dev, "%s no specified codec: setting dummy.\n", dai_link->name);2829dai_link->codecs = &snd_soc_dummy_dlc;30dai_link->num_codecs = 1;31dai_link->dynamic = 1;32return 0;33}3435/* set card codec info */36ret = snd_soc_of_get_dai_link_codecs(dev, codec_node, dai_link);3738of_node_put(codec_node);3940if (ret < 0)41return dev_err_probe(dev, ret, "%s: codec dai not found\n",42dai_link->name);4344return 0;45}4647static int set_dailink_daifmt(struct snd_soc_card *card,48struct device_node *sub_node,49struct snd_soc_dai_link *dai_link)50{51unsigned int daifmt;52const char *str;53int ret;54struct {55char *name;56unsigned int val;57} of_clk_table[] = {58{ "cpu", SND_SOC_DAIFMT_CBC_CFC },59{ "codec", SND_SOC_DAIFMT_CBP_CFP },60};6162daifmt = snd_soc_daifmt_parse_format(sub_node, NULL);63if (daifmt) {64dai_link->dai_fmt &= SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK;65dai_link->dai_fmt |= daifmt;66}6768/*69* check "mediatek,clk-provider = xxx"70* SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK area71*/72ret = of_property_read_string(sub_node, "mediatek,clk-provider", &str);73if (ret == 0) {74int i;7576for (i = 0; i < ARRAY_SIZE(of_clk_table); i++) {77if (strcmp(str, of_clk_table[i].name) == 0) {78dai_link->dai_fmt &= ~SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK;79dai_link->dai_fmt |= of_clk_table[i].val;80break;81}82}83}8485return 0;86}8788int parse_dai_link_info(struct snd_soc_card *card)89{90struct device *dev = card->dev;91struct snd_soc_dai_link *dai_link;92const char *dai_link_name;93int ret, i;9495/* Loop over all the dai link sub nodes */96for_each_available_child_of_node_scoped(dev->of_node, sub_node) {97if (of_property_read_string(sub_node, "link-name",98&dai_link_name))99return -EINVAL;100101for_each_card_prelinks(card, i, dai_link) {102if (!strcmp(dai_link_name, dai_link->name))103break;104}105106if (i >= card->num_links)107return -EINVAL;108109ret = set_card_codec_info(card, sub_node, dai_link);110if (ret < 0)111return ret;112113ret = set_dailink_daifmt(card, sub_node, dai_link);114if (ret < 0)115return ret;116}117118return 0;119}120EXPORT_SYMBOL_GPL(parse_dai_link_info);121122void clean_card_reference(struct snd_soc_card *card)123{124struct snd_soc_dai_link *dai_link;125int i;126127/* release codec reference gotten by set_card_codec_info */128for_each_card_prelinks(card, i, dai_link)129snd_soc_of_put_dai_link_codecs(dai_link);130}131EXPORT_SYMBOL_GPL(clean_card_reference);132133int mtk_soundcard_startup(struct snd_pcm_substream *substream,134enum mtk_pcm_constraint_type ctype)135{136struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);137struct mtk_soc_card_data *soc_card = snd_soc_card_get_drvdata(rtd->card);138const struct mtk_pcm_constraints_data *mpc = &soc_card->card_data->pcm_constraints[ctype];139int ret;140141if (unlikely(!mpc))142return -EINVAL;143144ret = snd_pcm_hw_constraint_list(substream->runtime, 0,145SNDRV_PCM_HW_PARAM_RATE,146mpc->rates);147if (ret < 0) {148dev_err(rtd->dev, "hw_constraint_list rate failed\n");149return ret;150}151152ret = snd_pcm_hw_constraint_list(substream->runtime, 0,153SNDRV_PCM_HW_PARAM_CHANNELS,154mpc->channels);155if (ret < 0) {156dev_err(rtd->dev, "hw_constraint_list channel failed\n");157return ret;158}159160return 0;161}162EXPORT_SYMBOL_GPL(mtk_soundcard_startup);163164static int mtk_soundcard_playback_startup(struct snd_pcm_substream *substream)165{166return mtk_soundcard_startup(substream, MTK_CONSTRAINT_PLAYBACK);167}168169const struct snd_soc_ops mtk_soundcard_common_playback_ops = {170.startup = mtk_soundcard_playback_startup,171};172EXPORT_SYMBOL_GPL(mtk_soundcard_common_playback_ops);173174static int mtk_soundcard_capture_startup(struct snd_pcm_substream *substream)175{176return mtk_soundcard_startup(substream, MTK_CONSTRAINT_CAPTURE);177}178179const struct snd_soc_ops mtk_soundcard_common_capture_ops = {180.startup = mtk_soundcard_capture_startup,181};182EXPORT_SYMBOL_GPL(mtk_soundcard_common_capture_ops);183184int mtk_soundcard_common_probe(struct platform_device *pdev)185{186struct device_node *platform_node, *adsp_node, *accdet_node;187struct snd_soc_component *accdet_comp;188struct platform_device *accdet_pdev;189const struct mtk_soundcard_pdata *pdata;190struct mtk_soc_card_data *soc_card_data;191struct snd_soc_dai_link *orig_dai_link, *dai_link;192struct snd_soc_jack *jacks;193struct snd_soc_card *card;194int i, orig_num_links, ret;195bool needs_legacy_probe;196197pdata = device_get_match_data(&pdev->dev);198if (!pdata)199return -EINVAL;200201card = pdata->card_data->card;202card->dev = &pdev->dev;203orig_dai_link = card->dai_link;204orig_num_links = card->num_links;205206ret = snd_soc_of_parse_card_name(card, "model");207if (ret)208return ret;209210if (!card->name) {211if (!pdata->card_name)212return -EINVAL;213214card->name = pdata->card_name;215}216217needs_legacy_probe = !of_property_present(pdev->dev.of_node, "audio-routing");218if (needs_legacy_probe) {219/*220* If we have no .soc_probe() callback there's no way of using221* any legacy probe mechanism, as that cannot not be generic.222*/223if (!pdata->soc_probe)224return -EINVAL;225226dev_info_once(&pdev->dev, "audio-routing not found: using legacy probe\n");227} else {228ret = snd_soc_of_parse_audio_routing(card, "audio-routing");229if (ret)230return ret;231}232233soc_card_data = devm_kzalloc(&pdev->dev, sizeof(*soc_card_data), GFP_KERNEL);234if (!soc_card_data)235return -ENOMEM;236237soc_card_data->card_data = pdata->card_data;238239jacks = devm_kcalloc(card->dev, soc_card_data->card_data->num_jacks,240sizeof(*jacks), GFP_KERNEL);241if (!jacks)242return -ENOMEM;243244soc_card_data->card_data->jacks = jacks;245246accdet_node = of_parse_phandle(pdev->dev.of_node, "mediatek,accdet", 0);247if (accdet_node) {248accdet_pdev = of_find_device_by_node(accdet_node);249if (accdet_pdev) {250accdet_comp = snd_soc_lookup_component(&accdet_pdev->dev, NULL);251if (accdet_comp)252soc_card_data->accdet = accdet_comp;253else254dev_err(&pdev->dev, "No sound component found from mediatek,accdet property\n");255256put_device(&accdet_pdev->dev);257} else {258dev_err(&pdev->dev, "No device found from mediatek,accdet property\n");259}260261of_node_put(accdet_node);262}263264platform_node = of_parse_phandle(pdev->dev.of_node, "mediatek,platform", 0);265if (!platform_node)266return dev_err_probe(&pdev->dev, -EINVAL,267"Property mediatek,platform missing or invalid\n");268269/* Check if this SoC has an Audio DSP */270if (pdata->sof_priv)271adsp_node = of_parse_phandle(pdev->dev.of_node, "mediatek,adsp", 0);272else273adsp_node = NULL;274275if (adsp_node) {276if (of_property_present(pdev->dev.of_node, "mediatek,dai-link")) {277ret = mtk_sof_dailink_parse_of(card, pdev->dev.of_node,278"mediatek,dai-link",279card->dai_link, card->num_links);280if (ret) {281of_node_put(adsp_node);282of_node_put(platform_node);283return dev_err_probe(&pdev->dev, ret,284"Cannot parse mediatek,dai-link\n");285}286}287288soc_card_data->sof_priv = pdata->sof_priv;289card->probe = mtk_sof_card_probe;290card->late_probe = mtk_sof_card_late_probe;291if (!card->topology_shortname_created) {292snprintf(card->topology_shortname, 32, "sof-%s", card->name);293card->topology_shortname_created = true;294}295card->name = card->topology_shortname;296}297298/*299* Regardless of whether the ADSP is wanted and/or present in a machine300* specific device tree or not and regardless of whether any AFE_SOF301* link is present, we have to make sure that the platforms->of_node302* is not NULL, and set to either ADSP (adsp_node) or AFE (platform_node).303*/304for_each_card_prelinks(card, i, dai_link) {305if (adsp_node && !strncmp(dai_link->name, "AFE_SOF", strlen("AFE_SOF")))306dai_link->platforms->of_node = adsp_node;307else if (!dai_link->platforms->name && !dai_link->platforms->of_node)308dai_link->platforms->of_node = platform_node;309}310311if (!needs_legacy_probe) {312ret = parse_dai_link_info(card);313if (ret)314goto err_restore_dais;315} else {316if (adsp_node)317of_node_put(adsp_node);318of_node_put(platform_node);319}320321if (pdata->soc_probe) {322ret = pdata->soc_probe(soc_card_data, needs_legacy_probe);323if (ret) {324if (!needs_legacy_probe)325clean_card_reference(card);326goto err_restore_dais;327}328}329snd_soc_card_set_drvdata(card, soc_card_data);330331ret = devm_snd_soc_register_card(&pdev->dev, card);332333if (!needs_legacy_probe)334clean_card_reference(card);335336if (ret) {337dev_err_probe(&pdev->dev, ret, "Cannot register card\n");338goto err_restore_dais;339}340341return 0;342343err_restore_dais:344card->dai_link = orig_dai_link;345card->num_links = orig_num_links;346return ret;347}348EXPORT_SYMBOL_GPL(mtk_soundcard_common_probe);349350351