Path: blob/master/sound/soc/mediatek/common/mtk-soundcard-driver.c
26450 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 device_node *sub_node;92struct snd_soc_dai_link *dai_link;93const char *dai_link_name;94int ret, i;9596/* Loop over all the dai link sub nodes */97for_each_available_child_of_node(dev->of_node, sub_node) {98if (of_property_read_string(sub_node, "link-name",99&dai_link_name)) {100of_node_put(sub_node);101return -EINVAL;102}103104for_each_card_prelinks(card, i, dai_link) {105if (!strcmp(dai_link_name, dai_link->name))106break;107}108109if (i >= card->num_links) {110of_node_put(sub_node);111return -EINVAL;112}113114ret = set_card_codec_info(card, sub_node, dai_link);115if (ret < 0) {116of_node_put(sub_node);117return ret;118}119120ret = set_dailink_daifmt(card, sub_node, dai_link);121if (ret < 0) {122of_node_put(sub_node);123return ret;124}125}126127return 0;128}129EXPORT_SYMBOL_GPL(parse_dai_link_info);130131void clean_card_reference(struct snd_soc_card *card)132{133struct snd_soc_dai_link *dai_link;134int i;135136/* release codec reference gotten by set_card_codec_info */137for_each_card_prelinks(card, i, dai_link)138snd_soc_of_put_dai_link_codecs(dai_link);139}140EXPORT_SYMBOL_GPL(clean_card_reference);141142int mtk_soundcard_startup(struct snd_pcm_substream *substream,143enum mtk_pcm_constraint_type ctype)144{145struct snd_soc_pcm_runtime *rtd = snd_soc_substream_to_rtd(substream);146struct mtk_soc_card_data *soc_card = snd_soc_card_get_drvdata(rtd->card);147const struct mtk_pcm_constraints_data *mpc = &soc_card->card_data->pcm_constraints[ctype];148int ret;149150if (unlikely(!mpc))151return -EINVAL;152153ret = snd_pcm_hw_constraint_list(substream->runtime, 0,154SNDRV_PCM_HW_PARAM_RATE,155mpc->rates);156if (ret < 0) {157dev_err(rtd->dev, "hw_constraint_list rate failed\n");158return ret;159}160161ret = snd_pcm_hw_constraint_list(substream->runtime, 0,162SNDRV_PCM_HW_PARAM_CHANNELS,163mpc->channels);164if (ret < 0) {165dev_err(rtd->dev, "hw_constraint_list channel failed\n");166return ret;167}168169return 0;170}171EXPORT_SYMBOL_GPL(mtk_soundcard_startup);172173static int mtk_soundcard_playback_startup(struct snd_pcm_substream *substream)174{175return mtk_soundcard_startup(substream, MTK_CONSTRAINT_PLAYBACK);176}177178const struct snd_soc_ops mtk_soundcard_common_playback_ops = {179.startup = mtk_soundcard_playback_startup,180};181EXPORT_SYMBOL_GPL(mtk_soundcard_common_playback_ops);182183static int mtk_soundcard_capture_startup(struct snd_pcm_substream *substream)184{185return mtk_soundcard_startup(substream, MTK_CONSTRAINT_CAPTURE);186}187188const struct snd_soc_ops mtk_soundcard_common_capture_ops = {189.startup = mtk_soundcard_capture_startup,190};191EXPORT_SYMBOL_GPL(mtk_soundcard_common_capture_ops);192193int mtk_soundcard_common_probe(struct platform_device *pdev)194{195struct device_node *platform_node, *adsp_node, *accdet_node;196struct snd_soc_component *accdet_comp;197struct platform_device *accdet_pdev;198const struct mtk_soundcard_pdata *pdata;199struct mtk_soc_card_data *soc_card_data;200struct snd_soc_dai_link *orig_dai_link, *dai_link;201struct snd_soc_jack *jacks;202struct snd_soc_card *card;203int i, orig_num_links, ret;204bool needs_legacy_probe;205206pdata = device_get_match_data(&pdev->dev);207if (!pdata)208return -EINVAL;209210card = pdata->card_data->card;211card->dev = &pdev->dev;212orig_dai_link = card->dai_link;213orig_num_links = card->num_links;214215ret = snd_soc_of_parse_card_name(card, "model");216if (ret)217return ret;218219if (!card->name) {220if (!pdata->card_name)221return -EINVAL;222223card->name = pdata->card_name;224}225226needs_legacy_probe = !of_property_present(pdev->dev.of_node, "audio-routing");227if (needs_legacy_probe) {228/*229* If we have no .soc_probe() callback there's no way of using230* any legacy probe mechanism, as that cannot not be generic.231*/232if (!pdata->soc_probe)233return -EINVAL;234235dev_info_once(&pdev->dev, "audio-routing not found: using legacy probe\n");236} else {237ret = snd_soc_of_parse_audio_routing(card, "audio-routing");238if (ret)239return ret;240}241242soc_card_data = devm_kzalloc(&pdev->dev, sizeof(*soc_card_data), GFP_KERNEL);243if (!soc_card_data)244return -ENOMEM;245246soc_card_data->card_data = pdata->card_data;247248jacks = devm_kcalloc(card->dev, soc_card_data->card_data->num_jacks,249sizeof(*jacks), GFP_KERNEL);250if (!jacks)251return -ENOMEM;252253soc_card_data->card_data->jacks = jacks;254255accdet_node = of_parse_phandle(pdev->dev.of_node, "mediatek,accdet", 0);256if (accdet_node) {257accdet_pdev = of_find_device_by_node(accdet_node);258if (accdet_pdev) {259accdet_comp = snd_soc_lookup_component(&accdet_pdev->dev, NULL);260if (accdet_comp)261soc_card_data->accdet = accdet_comp;262else263dev_err(&pdev->dev, "No sound component found from mediatek,accdet property\n");264265put_device(&accdet_pdev->dev);266} else {267dev_err(&pdev->dev, "No device found from mediatek,accdet property\n");268}269270of_node_put(accdet_node);271}272273platform_node = of_parse_phandle(pdev->dev.of_node, "mediatek,platform", 0);274if (!platform_node)275return dev_err_probe(&pdev->dev, -EINVAL,276"Property mediatek,platform missing or invalid\n");277278/* Check if this SoC has an Audio DSP */279if (pdata->sof_priv)280adsp_node = of_parse_phandle(pdev->dev.of_node, "mediatek,adsp", 0);281else282adsp_node = NULL;283284if (adsp_node) {285if (of_property_present(pdev->dev.of_node, "mediatek,dai-link")) {286ret = mtk_sof_dailink_parse_of(card, pdev->dev.of_node,287"mediatek,dai-link",288card->dai_link, card->num_links);289if (ret) {290of_node_put(adsp_node);291of_node_put(platform_node);292return dev_err_probe(&pdev->dev, ret,293"Cannot parse mediatek,dai-link\n");294}295}296297soc_card_data->sof_priv = pdata->sof_priv;298card->probe = mtk_sof_card_probe;299card->late_probe = mtk_sof_card_late_probe;300if (!card->topology_shortname_created) {301snprintf(card->topology_shortname, 32, "sof-%s", card->name);302card->topology_shortname_created = true;303}304card->name = card->topology_shortname;305}306307/*308* Regardless of whether the ADSP is wanted and/or present in a machine309* specific device tree or not and regardless of whether any AFE_SOF310* link is present, we have to make sure that the platforms->of_node311* is not NULL, and set to either ADSP (adsp_node) or AFE (platform_node).312*/313for_each_card_prelinks(card, i, dai_link) {314if (adsp_node && !strncmp(dai_link->name, "AFE_SOF", strlen("AFE_SOF")))315dai_link->platforms->of_node = adsp_node;316else if (!dai_link->platforms->name && !dai_link->platforms->of_node)317dai_link->platforms->of_node = platform_node;318}319320if (!needs_legacy_probe) {321ret = parse_dai_link_info(card);322if (ret)323goto err_restore_dais;324} else {325if (adsp_node)326of_node_put(adsp_node);327of_node_put(platform_node);328}329330if (pdata->soc_probe) {331ret = pdata->soc_probe(soc_card_data, needs_legacy_probe);332if (ret) {333if (!needs_legacy_probe)334clean_card_reference(card);335goto err_restore_dais;336}337}338snd_soc_card_set_drvdata(card, soc_card_data);339340ret = devm_snd_soc_register_card(&pdev->dev, card);341342if (!needs_legacy_probe)343clean_card_reference(card);344345if (ret) {346dev_err_probe(&pdev->dev, ret, "Cannot register card\n");347goto err_restore_dais;348}349350return 0;351352err_restore_dais:353card->dai_link = orig_dai_link;354card->num_links = orig_num_links;355return ret;356}357EXPORT_SYMBOL_GPL(mtk_soundcard_common_probe);358359360