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