162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci// 362306a36Sopenharmony_ci// Copyright(c) 2019 Intel Corporation. All rights reserved. 462306a36Sopenharmony_ci 562306a36Sopenharmony_ci#include <linux/module.h> 662306a36Sopenharmony_ci#include <sound/pcm.h> 762306a36Sopenharmony_ci#include <sound/soc.h> 862306a36Sopenharmony_ci#include <sound/hda_codec.h> 962306a36Sopenharmony_ci#include <sound/hda_i915.h> 1062306a36Sopenharmony_ci#include "../../codecs/hdac_hda.h" 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#include "hda_dsp_common.h" 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci#if IS_ENABLED(CONFIG_SND_SOC_SOF_HDA_AUDIO_CODEC) 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci/* 1762306a36Sopenharmony_ci * Search card topology and return PCM device number 1862306a36Sopenharmony_ci * matching Nth HDMI device (zero-based index). 1962306a36Sopenharmony_ci */ 2062306a36Sopenharmony_cistatic struct snd_pcm *hda_dsp_hdmi_pcm_handle(struct snd_soc_card *card, 2162306a36Sopenharmony_ci int hdmi_idx) 2262306a36Sopenharmony_ci{ 2362306a36Sopenharmony_ci struct snd_soc_pcm_runtime *rtd; 2462306a36Sopenharmony_ci struct snd_pcm *spcm; 2562306a36Sopenharmony_ci int i = 0; 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci for_each_card_rtds(card, rtd) { 2862306a36Sopenharmony_ci spcm = rtd->pcm ? 2962306a36Sopenharmony_ci rtd->pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].pcm : NULL; 3062306a36Sopenharmony_ci if (spcm && strstr(spcm->id, "HDMI")) { 3162306a36Sopenharmony_ci if (i == hdmi_idx) 3262306a36Sopenharmony_ci return rtd->pcm; 3362306a36Sopenharmony_ci ++i; 3462306a36Sopenharmony_ci } 3562306a36Sopenharmony_ci } 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci return NULL; 3862306a36Sopenharmony_ci} 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci/* 4162306a36Sopenharmony_ci * Search card topology and register HDMI PCM related controls 4262306a36Sopenharmony_ci * to codec driver. 4362306a36Sopenharmony_ci */ 4462306a36Sopenharmony_ciint hda_dsp_hdmi_build_controls(struct snd_soc_card *card, 4562306a36Sopenharmony_ci struct snd_soc_component *comp) 4662306a36Sopenharmony_ci{ 4762306a36Sopenharmony_ci struct hdac_hda_priv *hda_pvt; 4862306a36Sopenharmony_ci struct hda_codec *hcodec; 4962306a36Sopenharmony_ci struct snd_pcm *spcm; 5062306a36Sopenharmony_ci struct hda_pcm *hpcm; 5162306a36Sopenharmony_ci int err = 0, i = 0; 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci if (!comp) 5462306a36Sopenharmony_ci return -EINVAL; 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci hda_pvt = snd_soc_component_get_drvdata(comp); 5762306a36Sopenharmony_ci hcodec = hda_pvt->codec; 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci list_for_each_entry(hpcm, &hcodec->pcm_list_head, list) { 6062306a36Sopenharmony_ci spcm = hda_dsp_hdmi_pcm_handle(card, i); 6162306a36Sopenharmony_ci if (spcm) { 6262306a36Sopenharmony_ci hpcm->pcm = spcm; 6362306a36Sopenharmony_ci hpcm->device = spcm->device; 6462306a36Sopenharmony_ci dev_dbg(card->dev, 6562306a36Sopenharmony_ci "mapping HDMI converter %d to PCM %d (%p)\n", 6662306a36Sopenharmony_ci i, hpcm->device, spcm); 6762306a36Sopenharmony_ci } else { 6862306a36Sopenharmony_ci hpcm->pcm = NULL; 6962306a36Sopenharmony_ci hpcm->device = SNDRV_PCM_INVALID_DEVICE; 7062306a36Sopenharmony_ci dev_warn(card->dev, 7162306a36Sopenharmony_ci "%s: no PCM in topology for HDMI converter %d\n", 7262306a36Sopenharmony_ci __func__, i); 7362306a36Sopenharmony_ci } 7462306a36Sopenharmony_ci i++; 7562306a36Sopenharmony_ci } 7662306a36Sopenharmony_ci snd_hdac_display_power(hcodec->core.bus, 7762306a36Sopenharmony_ci HDA_CODEC_IDX_CONTROLLER, true); 7862306a36Sopenharmony_ci err = snd_hda_codec_build_controls(hcodec); 7962306a36Sopenharmony_ci if (err < 0) 8062306a36Sopenharmony_ci dev_err(card->dev, "unable to create controls %d\n", err); 8162306a36Sopenharmony_ci snd_hdac_display_power(hcodec->core.bus, 8262306a36Sopenharmony_ci HDA_CODEC_IDX_CONTROLLER, false); 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_ci return err; 8562306a36Sopenharmony_ci} 8662306a36Sopenharmony_ciEXPORT_SYMBOL_NS(hda_dsp_hdmi_build_controls, SND_SOC_INTEL_HDA_DSP_COMMON); 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci#endif 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_ciMODULE_DESCRIPTION("ASoC Intel HDMI helpers"); 9162306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 92