root/sound/hda/codecs/realtek/alc861vd.c
// SPDX-License-Identifier: GPL-2.0-or-later
//
// Realtek ALC861-VD codec
// Based on ALC882
// In addition, an independent DAC
//

#include <linux/init.h>
#include <linux/module.h>
#include "realtek.h"

static int alc861vd_parse_auto_config(struct hda_codec *codec)
{
        static const hda_nid_t alc861vd_ignore[] = { 0x1d, 0 };
        static const hda_nid_t alc861vd_ssids[] = { 0x15, 0x1b, 0x14, 0 };
        return alc_parse_auto_config(codec, alc861vd_ignore, alc861vd_ssids);
}

enum {
        ALC660VD_FIX_ASUS_GPIO1,
        ALC861VD_FIX_DALLAS,
};

/* exclude VREF80 */
static void alc861vd_fixup_dallas(struct hda_codec *codec,
                                  const struct hda_fixup *fix, int action)
{
        if (action == HDA_FIXUP_ACT_PRE_PROBE) {
                snd_hda_override_pin_caps(codec, 0x18, 0x00000734);
                snd_hda_override_pin_caps(codec, 0x19, 0x0000073c);
        }
}

/* reset GPIO1 */
static void alc660vd_fixup_asus_gpio1(struct hda_codec *codec,
                                      const struct hda_fixup *fix, int action)
{
        struct alc_spec *spec = codec->spec;

        if (action == HDA_FIXUP_ACT_PRE_PROBE)
                spec->gpio_mask |= 0x02;
        alc_fixup_gpio(codec, action, 0x01);
}

static const struct hda_fixup alc861vd_fixups[] = {
        [ALC660VD_FIX_ASUS_GPIO1] = {
                .type = HDA_FIXUP_FUNC,
                .v.func = alc660vd_fixup_asus_gpio1,
        },
        [ALC861VD_FIX_DALLAS] = {
                .type = HDA_FIXUP_FUNC,
                .v.func = alc861vd_fixup_dallas,
        },
};

static const struct hda_quirk alc861vd_fixup_tbl[] = {
        SND_PCI_QUIRK(0x103c, 0x30bf, "HP TX1000", ALC861VD_FIX_DALLAS),
        SND_PCI_QUIRK(0x1043, 0x1339, "ASUS A7-K", ALC660VD_FIX_ASUS_GPIO1),
        SND_PCI_QUIRK(0x1179, 0xff31, "Toshiba L30-149", ALC861VD_FIX_DALLAS),
        {}
};

/*
 */
static int alc861vd_probe(struct hda_codec *codec, const struct hda_device_id *id)
{
        struct alc_spec *spec;
        int err;

        err = alc_alloc_spec(codec, 0x0b);
        if (err < 0)
                return err;

        spec = codec->spec;
        if (has_cdefine_beep(codec))
                spec->gen.beep_nid = 0x23;

        spec->shutup = alc_eapd_shutup;

        alc_pre_init(codec);

        snd_hda_pick_fixup(codec, NULL, alc861vd_fixup_tbl, alc861vd_fixups);
        snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE);

        /* automatic parse from the BIOS config */
        err = alc861vd_parse_auto_config(codec);
        if (err < 0)
                goto error;

        if (!spec->gen.no_analog) {
                err = set_beep_amp(spec, 0x0b, 0x05, HDA_INPUT);
                if (err < 0)
                        goto error;
        }

        snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE);

        return 0;

 error:
        snd_hda_gen_remove(codec);
        return err;
}

static const struct hda_codec_ops alc861vd_codec_ops = {
        .probe = alc861vd_probe,
        .remove = snd_hda_gen_remove,
        .build_controls = alc_build_controls,
        .build_pcms = snd_hda_gen_build_pcms,
        .init = alc_init,
        .unsol_event = snd_hda_jack_unsol_event,
        .resume = alc_resume,
        .suspend = alc_suspend,
        .check_power_status = snd_hda_gen_check_power_status,
        .stream_pm = snd_hda_gen_stream_pm,
};

/*
 * driver entries
 */
static const struct hda_device_id snd_hda_id_alc861vd[] = {
        HDA_CODEC_ID(0x10ec0660, "ALC660-VD"),
        HDA_CODEC_ID(0x10ec0862, "ALC861-VD"),
        {} /* terminator */
};
MODULE_DEVICE_TABLE(hdaudio, snd_hda_id_alc861vd);

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("Realtek ALC861-VD HD-audio codec");
MODULE_IMPORT_NS("SND_HDA_CODEC_REALTEK");

static struct hda_codec_driver alc861vd_driver = {
        .id = snd_hda_id_alc861vd,
        .ops = &alc861vd_codec_ops,
};

module_hda_codec_driver(alc861vd_driver);