root/sound/hda/codecs/hdmi/simplehdmi.c
// SPDX-License-Identifier: GPL-2.0-or-later
/*
 * Non-generic simple HDMI codec support
 */

#include <linux/slab.h>
#include <linux/module.h>
#include "hdmi_local.h"
#include "hda_jack.h"

int snd_hda_hdmi_simple_build_pcms(struct hda_codec *codec)
{
        struct hdmi_spec *spec = codec->spec;
        struct hda_pcm *info;
        unsigned int chans;
        struct hda_pcm_stream *pstr;
        struct hdmi_spec_per_cvt *per_cvt;

        per_cvt = get_cvt(spec, 0);
        chans = get_wcaps(codec, per_cvt->cvt_nid);
        chans = get_wcaps_channels(chans);

        info = snd_hda_codec_pcm_new(codec, "HDMI 0");
        if (!info)
                return -ENOMEM;
        spec->pcm_rec[0].pcm = info;
        info->pcm_type = HDA_PCM_TYPE_HDMI;
        pstr = &info->stream[SNDRV_PCM_STREAM_PLAYBACK];
        *pstr = spec->pcm_playback;
        pstr->nid = per_cvt->cvt_nid;
        if (pstr->channels_max <= 2 && chans && chans <= 16)
                pstr->channels_max = chans;

        return 0;
}
EXPORT_SYMBOL_NS_GPL(snd_hda_hdmi_simple_build_pcms, "SND_HDA_CODEC_HDMI");

/* unsolicited event for jack sensing */
void snd_hda_hdmi_simple_unsol_event(struct hda_codec *codec,
                                     unsigned int res)
{
        snd_hda_jack_set_dirty_all(codec);
        snd_hda_jack_report_sync(codec);
}
EXPORT_SYMBOL_NS_GPL(snd_hda_hdmi_simple_unsol_event, "SND_HDA_CODEC_HDMI");

static void free_hdmi_jack_priv(struct snd_jack *jack)
{
        struct hdmi_pcm *pcm = jack->private_data;

        pcm->jack = NULL;
}

static int simple_hdmi_build_jack(struct hda_codec *codec)
{
        char hdmi_str[32] = "HDMI/DP";
        struct hdmi_spec *spec = codec->spec;
        struct snd_jack *jack;
        struct hdmi_pcm *pcmp = get_hdmi_pcm(spec, 0);
        int pcmdev = pcmp->pcm->device;
        int err;

        if (pcmdev > 0)
                sprintf(hdmi_str + strlen(hdmi_str), ",pcm=%d", pcmdev);

        err = snd_jack_new(codec->card, hdmi_str, SND_JACK_AVOUT, &jack,
                           true, false);
        if (err < 0)
                return err;

        pcmp->jack = jack;
        jack->private_data = pcmp;
        jack->private_free = free_hdmi_jack_priv;
        return 0;
}

int snd_hda_hdmi_simple_build_controls(struct hda_codec *codec)
{
        struct hdmi_spec *spec = codec->spec;
        struct hdmi_spec_per_cvt *per_cvt;
        int err;

        per_cvt = get_cvt(spec, 0);
        err = snd_hda_create_dig_out_ctls(codec, per_cvt->cvt_nid,
                                          per_cvt->cvt_nid,
                                          HDA_PCM_TYPE_HDMI);
        if (err < 0)
                return err;
        return simple_hdmi_build_jack(codec);
}
EXPORT_SYMBOL_NS_GPL(snd_hda_hdmi_simple_build_controls, "SND_HDA_CODEC_HDMI");

int snd_hda_hdmi_simple_init(struct hda_codec *codec)
{
        struct hdmi_spec *spec = codec->spec;
        struct hdmi_spec_per_pin *per_pin = get_pin(spec, 0);
        hda_nid_t pin = per_pin->pin_nid;

        snd_hda_codec_write(codec, pin, 0,
                            AC_VERB_SET_PIN_WIDGET_CONTROL, PIN_OUT);
        /* some codecs require to unmute the pin */
        if (get_wcaps(codec, pin) & AC_WCAP_OUT_AMP)
                snd_hda_codec_write(codec, pin, 0, AC_VERB_SET_AMP_GAIN_MUTE,
                                    AMP_OUT_UNMUTE);
        snd_hda_jack_detect_enable(codec, pin, per_pin->dev_id);
        return 0;
}
EXPORT_SYMBOL_NS_GPL(snd_hda_hdmi_simple_init, "SND_HDA_CODEC_HDMI");

void snd_hda_hdmi_simple_remove(struct hda_codec *codec)
{
        struct hdmi_spec *spec = codec->spec;

        snd_array_free(&spec->pins);
        snd_array_free(&spec->cvts);
        kfree(spec);
}
EXPORT_SYMBOL_NS_GPL(snd_hda_hdmi_simple_remove, "SND_HDA_CODEC_HDMI");

int snd_hda_hdmi_simple_pcm_open(struct hda_pcm_stream *hinfo,
                                 struct hda_codec *codec,
                                 struct snd_pcm_substream *substream)
{
        struct hdmi_spec *spec = codec->spec;

        if (spec->hw_constraints_channels) {
                snd_pcm_hw_constraint_list(substream->runtime, 0,
                                SNDRV_PCM_HW_PARAM_CHANNELS,
                                spec->hw_constraints_channels);
        } else {
                snd_pcm_hw_constraint_step(substream->runtime, 0,
                                           SNDRV_PCM_HW_PARAM_CHANNELS, 2);
        }

        return snd_hda_multi_out_dig_open(codec, &spec->multiout);
}
EXPORT_SYMBOL_NS_GPL(snd_hda_hdmi_simple_pcm_open, "SND_HDA_CODEC_HDMI");

static int simple_playback_pcm_close(struct hda_pcm_stream *hinfo,
                                     struct hda_codec *codec,
                                     struct snd_pcm_substream *substream)
{
        struct hdmi_spec *spec = codec->spec;

        return snd_hda_multi_out_dig_close(codec, &spec->multiout);
}

static int simple_playback_pcm_prepare(struct hda_pcm_stream *hinfo,
                                       struct hda_codec *codec,
                                       unsigned int stream_tag,
                                       unsigned int format,
                                       struct snd_pcm_substream *substream)
{
        struct hdmi_spec *spec = codec->spec;

        return snd_hda_multi_out_dig_prepare(codec, &spec->multiout,
                                             stream_tag, format, substream);
}

static const struct hda_pcm_stream simple_pcm_playback = {
        .substreams = 1,
        .channels_min = 2,
        .channels_max = 2,
        .ops = {
                .open = snd_hda_hdmi_simple_pcm_open,
                .close = simple_playback_pcm_close,
                .prepare = simple_playback_pcm_prepare
        },
};

int snd_hda_hdmi_simple_probe(struct hda_codec *codec,
                              hda_nid_t cvt_nid, hda_nid_t pin_nid)
{
        struct hdmi_spec *spec;
        struct hdmi_spec_per_cvt *per_cvt;
        struct hdmi_spec_per_pin *per_pin;

        spec = kzalloc_obj(*spec);
        if (!spec)
                return -ENOMEM;

        spec->codec = codec;
        codec->spec = spec;
        snd_array_init(&spec->pins, sizeof(struct hdmi_spec_per_pin), 1);
        snd_array_init(&spec->cvts, sizeof(struct hdmi_spec_per_cvt), 1);

        spec->multiout.num_dacs = 0;  /* no analog */
        spec->multiout.max_channels = 2;
        spec->multiout.dig_out_nid = cvt_nid;
        spec->num_cvts = 1;
        spec->num_pins = 1;
        per_pin = snd_array_new(&spec->pins);
        per_cvt = snd_array_new(&spec->cvts);
        if (!per_pin || !per_cvt) {
                snd_hda_hdmi_simple_remove(codec);
                return -ENOMEM;
        }
        per_cvt->cvt_nid = cvt_nid;
        per_pin->pin_nid = pin_nid;
        spec->pcm_playback = simple_pcm_playback;

        return 0;
}
EXPORT_SYMBOL_NS_GPL(snd_hda_hdmi_simple_probe, "SND_HDA_CODEC_HDMI");

/*
 * driver entries
 */

enum { MODEL_VIA };

/* VIA HDMI Implementation */
#define VIAHDMI_CVT_NID 0x02    /* audio converter1 */
#define VIAHDMI_PIN_NID 0x03    /* HDMI output pin1 */

static int simplehdmi_probe(struct hda_codec *codec,
                            const struct hda_device_id *id)
{
        switch (id->driver_data) {
        case MODEL_VIA:
                return snd_hda_hdmi_simple_probe(codec, VIAHDMI_CVT_NID,
                                                 VIAHDMI_PIN_NID);
        default:
                return -EINVAL;
        }
}

static const struct hda_codec_ops simplehdmi_codec_ops = {
        .probe = simplehdmi_probe,
        .remove = snd_hda_hdmi_simple_remove,
        .build_controls = snd_hda_hdmi_simple_build_controls,
        .build_pcms = snd_hda_hdmi_simple_build_pcms,
        .init = snd_hda_hdmi_simple_init,
        .unsol_event = snd_hda_hdmi_simple_unsol_event,
};

static const struct hda_device_id snd_hda_id_simplehdmi[] = {
        HDA_CODEC_ID_MODEL(0x11069f80, "VX900 HDMI/DP", MODEL_VIA),
        HDA_CODEC_ID_MODEL(0x11069f81, "VX900 HDMI/DP", MODEL_VIA),
        {} /* terminator */
};

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("Simple HDMI HD-audio codec support");

static struct hda_codec_driver simplehdmi_driver = {
        .id = snd_hda_id_simplehdmi,
        .ops = &simplehdmi_codec_ops,
};

module_hda_codec_driver(simplehdmi_driver);