// SPDX-License-Identifier: GPL-2.0-or-later // // Realtek ALC861-VD codec // Based on ALC882 // In addition, an independent DAC // #include #include #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);