173cd0490STakashi Iwai // SPDX-License-Identifier: GPL-2.0-or-later
273cd0490STakashi Iwai /*
373cd0490STakashi Iwai * Nvidia HDMI codec support
473cd0490STakashi Iwai */
573cd0490STakashi Iwai
673cd0490STakashi Iwai #include <linux/init.h>
773cd0490STakashi Iwai #include <linux/slab.h>
873cd0490STakashi Iwai #include <linux/module.h>
973cd0490STakashi Iwai #include <sound/core.h>
1073cd0490STakashi Iwai #include <sound/tlv.h>
1173cd0490STakashi Iwai #include <sound/hdaudio.h>
1273cd0490STakashi Iwai #include <sound/hda_codec.h>
1373cd0490STakashi Iwai #include "hda_local.h"
1473cd0490STakashi Iwai #include "hdmi_local.h"
1573cd0490STakashi Iwai
16ad781b55STakashi Iwai enum {
17ad781b55STakashi Iwai MODEL_GENERIC,
18ad781b55STakashi Iwai MODEL_LEGACY,
19ad781b55STakashi Iwai };
20ad781b55STakashi Iwai
2173cd0490STakashi Iwai /*
2273cd0490STakashi Iwai * NVIDIA codecs ignore ASP mapping for 2ch - confirmed on:
2373cd0490STakashi Iwai * - 0x10de0015
2473cd0490STakashi Iwai * - 0x10de0040
2573cd0490STakashi Iwai */
nvhdmi_chmap_cea_alloc_validate_get_type(struct hdac_chmap * chmap,struct hdac_cea_channel_speaker_allocation * cap,int channels)2673cd0490STakashi Iwai static int nvhdmi_chmap_cea_alloc_validate_get_type(struct hdac_chmap *chmap,
2773cd0490STakashi Iwai struct hdac_cea_channel_speaker_allocation *cap, int channels)
2873cd0490STakashi Iwai {
2973cd0490STakashi Iwai if (cap->ca_index == 0x00 && channels == 2)
3073cd0490STakashi Iwai return SNDRV_CTL_TLVT_CHMAP_FIXED;
3173cd0490STakashi Iwai
3273cd0490STakashi Iwai /* If the speaker allocation matches the channel count, it is OK. */
3373cd0490STakashi Iwai if (cap->channels != channels)
3473cd0490STakashi Iwai return -1;
3573cd0490STakashi Iwai
3673cd0490STakashi Iwai /* all channels are remappable freely */
3773cd0490STakashi Iwai return SNDRV_CTL_TLVT_CHMAP_VAR;
3873cd0490STakashi Iwai }
3973cd0490STakashi Iwai
nvhdmi_chmap_validate(struct hdac_chmap * chmap,int ca,int chs,unsigned char * map)4073cd0490STakashi Iwai static int nvhdmi_chmap_validate(struct hdac_chmap *chmap,
4173cd0490STakashi Iwai int ca, int chs, unsigned char *map)
4273cd0490STakashi Iwai {
4373cd0490STakashi Iwai if (ca == 0x00 && (map[0] != SNDRV_CHMAP_FL || map[1] != SNDRV_CHMAP_FR))
4473cd0490STakashi Iwai return -EINVAL;
4573cd0490STakashi Iwai
4673cd0490STakashi Iwai return 0;
4773cd0490STakashi Iwai }
4873cd0490STakashi Iwai
4973cd0490STakashi Iwai /* map from pin NID to port; port is 0-based */
5073cd0490STakashi Iwai /* for Nvidia: assume widget NID starting from 4, with step 1 (4, 5, 6, ...) */
nvhdmi_pin2port(void * audio_ptr,int pin_nid)5173cd0490STakashi Iwai static int nvhdmi_pin2port(void *audio_ptr, int pin_nid)
5273cd0490STakashi Iwai {
5373cd0490STakashi Iwai return pin_nid - 4;
5473cd0490STakashi Iwai }
5573cd0490STakashi Iwai
5673cd0490STakashi Iwai /* reverse-map from port to pin NID: see above */
nvhdmi_port2pin(struct hda_codec * codec,int port)5773cd0490STakashi Iwai static int nvhdmi_port2pin(struct hda_codec *codec, int port)
5873cd0490STakashi Iwai {
5973cd0490STakashi Iwai return port + 4;
6073cd0490STakashi Iwai }
6173cd0490STakashi Iwai
6273cd0490STakashi Iwai static const struct drm_audio_component_audio_ops nvhdmi_audio_ops = {
6373cd0490STakashi Iwai .pin2port = nvhdmi_pin2port,
6473cd0490STakashi Iwai .pin_eld_notify = snd_hda_hdmi_acomp_pin_eld_notify,
6573cd0490STakashi Iwai .master_bind = snd_hda_hdmi_acomp_master_bind,
6673cd0490STakashi Iwai .master_unbind = snd_hda_hdmi_acomp_master_unbind,
6773cd0490STakashi Iwai };
6873cd0490STakashi Iwai
probe_generic(struct hda_codec * codec)69ad781b55STakashi Iwai static int probe_generic(struct hda_codec *codec)
7073cd0490STakashi Iwai {
7173cd0490STakashi Iwai struct hdmi_spec *spec;
7273cd0490STakashi Iwai int err;
7373cd0490STakashi Iwai
7473cd0490STakashi Iwai err = snd_hda_hdmi_generic_alloc(codec);
7573cd0490STakashi Iwai if (err < 0)
7673cd0490STakashi Iwai return err;
7773cd0490STakashi Iwai codec->dp_mst = true;
7873cd0490STakashi Iwai
7973cd0490STakashi Iwai spec = codec->spec;
8073cd0490STakashi Iwai
8173cd0490STakashi Iwai err = snd_hda_hdmi_parse_codec(codec);
8273cd0490STakashi Iwai if (err < 0) {
8373cd0490STakashi Iwai snd_hda_hdmi_generic_spec_free(codec);
8473cd0490STakashi Iwai return err;
8573cd0490STakashi Iwai }
8673cd0490STakashi Iwai
8773cd0490STakashi Iwai snd_hda_hdmi_generic_init_per_pins(codec);
8873cd0490STakashi Iwai
8973cd0490STakashi Iwai spec->dyn_pin_out = true;
9073cd0490STakashi Iwai
9173cd0490STakashi Iwai spec->chmap.ops.chmap_cea_alloc_validate_get_type =
9273cd0490STakashi Iwai nvhdmi_chmap_cea_alloc_validate_get_type;
9373cd0490STakashi Iwai spec->chmap.ops.chmap_validate = nvhdmi_chmap_validate;
9473cd0490STakashi Iwai spec->nv_dp_workaround = true;
9573cd0490STakashi Iwai
9673cd0490STakashi Iwai codec->link_down_at_suspend = 1;
9773cd0490STakashi Iwai
9873cd0490STakashi Iwai snd_hda_hdmi_acomp_init(codec, &nvhdmi_audio_ops, nvhdmi_port2pin);
9973cd0490STakashi Iwai
10073cd0490STakashi Iwai return 0;
10173cd0490STakashi Iwai }
10273cd0490STakashi Iwai
probe_legacy(struct hda_codec * codec)103ad781b55STakashi Iwai static int probe_legacy(struct hda_codec *codec)
10473cd0490STakashi Iwai {
10573cd0490STakashi Iwai struct hdmi_spec *spec;
10673cd0490STakashi Iwai int err;
10773cd0490STakashi Iwai
108ad781b55STakashi Iwai err = snd_hda_hdmi_generic_probe(codec);
10973cd0490STakashi Iwai if (err)
11073cd0490STakashi Iwai return err;
11173cd0490STakashi Iwai
11273cd0490STakashi Iwai spec = codec->spec;
11373cd0490STakashi Iwai spec->dyn_pin_out = true;
11473cd0490STakashi Iwai
11573cd0490STakashi Iwai spec->chmap.ops.chmap_cea_alloc_validate_get_type =
11673cd0490STakashi Iwai nvhdmi_chmap_cea_alloc_validate_get_type;
11773cd0490STakashi Iwai spec->chmap.ops.chmap_validate = nvhdmi_chmap_validate;
11873cd0490STakashi Iwai spec->nv_dp_workaround = true;
11973cd0490STakashi Iwai
12073cd0490STakashi Iwai codec->link_down_at_suspend = 1;
12173cd0490STakashi Iwai
12273cd0490STakashi Iwai return 0;
12373cd0490STakashi Iwai }
12473cd0490STakashi Iwai
nvhdmi_probe(struct hda_codec * codec,const struct hda_device_id * id)125ad781b55STakashi Iwai static int nvhdmi_probe(struct hda_codec *codec, const struct hda_device_id *id)
126ad781b55STakashi Iwai {
127ad781b55STakashi Iwai if (id->driver_data == MODEL_LEGACY)
128ad781b55STakashi Iwai return probe_legacy(codec);
129ad781b55STakashi Iwai else
130ad781b55STakashi Iwai return probe_generic(codec);
131ad781b55STakashi Iwai }
132ad781b55STakashi Iwai
133ad781b55STakashi Iwai static const struct hda_codec_ops nvhdmi_codec_ops = {
134ad781b55STakashi Iwai .probe = nvhdmi_probe,
135ad781b55STakashi Iwai .remove = snd_hda_hdmi_generic_remove,
136ad781b55STakashi Iwai .init = snd_hda_hdmi_generic_init,
137ad781b55STakashi Iwai .build_pcms = snd_hda_hdmi_generic_build_pcms,
138ad781b55STakashi Iwai .build_controls = snd_hda_hdmi_generic_build_controls,
139ad781b55STakashi Iwai .unsol_event = snd_hda_hdmi_generic_unsol_event,
140ad781b55STakashi Iwai .suspend = snd_hda_hdmi_generic_suspend,
141ad781b55STakashi Iwai .resume = snd_hda_hdmi_generic_resume,
142ad781b55STakashi Iwai };
143ad781b55STakashi Iwai
14473cd0490STakashi Iwai static const struct hda_device_id snd_hda_id_nvhdmi[] = {
145ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0008, "GPU 08 HDMI/DP", MODEL_LEGACY),
146ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0009, "GPU 09 HDMI/DP", MODEL_LEGACY),
147ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de000a, "GPU 0a HDMI/DP", MODEL_LEGACY),
148ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de000b, "GPU 0b HDMI/DP", MODEL_LEGACY),
149ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de000c, "MCP89 HDMI", MODEL_LEGACY),
150ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de000d, "GPU 0d HDMI/DP", MODEL_LEGACY),
151ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0010, "GPU 10 HDMI/DP", MODEL_LEGACY),
152ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0011, "GPU 11 HDMI/DP", MODEL_LEGACY),
153ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0012, "GPU 12 HDMI/DP", MODEL_LEGACY),
154ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0013, "GPU 13 HDMI/DP", MODEL_LEGACY),
155ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0014, "GPU 14 HDMI/DP", MODEL_LEGACY),
156ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0015, "GPU 15 HDMI/DP", MODEL_LEGACY),
157ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0016, "GPU 16 HDMI/DP", MODEL_LEGACY),
15873cd0490STakashi Iwai /* 17 is known to be absent */
159ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0018, "GPU 18 HDMI/DP", MODEL_LEGACY),
160ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0019, "GPU 19 HDMI/DP", MODEL_LEGACY),
161ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de001a, "GPU 1a HDMI/DP", MODEL_LEGACY),
162ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de001b, "GPU 1b HDMI/DP", MODEL_LEGACY),
163ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de001c, "GPU 1c HDMI/DP", MODEL_LEGACY),
164ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0040, "GPU 40 HDMI/DP", MODEL_GENERIC),
165ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0041, "GPU 41 HDMI/DP", MODEL_GENERIC),
166ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0042, "GPU 42 HDMI/DP", MODEL_GENERIC),
167ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0043, "GPU 43 HDMI/DP", MODEL_GENERIC),
168ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0044, "GPU 44 HDMI/DP", MODEL_GENERIC),
169ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0045, "GPU 45 HDMI/DP", MODEL_GENERIC),
170ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0050, "GPU 50 HDMI/DP", MODEL_GENERIC),
171ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0051, "GPU 51 HDMI/DP", MODEL_GENERIC),
172ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0052, "GPU 52 HDMI/DP", MODEL_GENERIC),
173ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0060, "GPU 60 HDMI/DP", MODEL_GENERIC),
174ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0061, "GPU 61 HDMI/DP", MODEL_GENERIC),
175ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0062, "GPU 62 HDMI/DP", MODEL_GENERIC),
176ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0070, "GPU 70 HDMI/DP", MODEL_GENERIC),
177ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0071, "GPU 71 HDMI/DP", MODEL_GENERIC),
178ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0072, "GPU 72 HDMI/DP", MODEL_GENERIC),
179ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0073, "GPU 73 HDMI/DP", MODEL_GENERIC),
180ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0074, "GPU 74 HDMI/DP", MODEL_GENERIC),
181ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0076, "GPU 76 HDMI/DP", MODEL_GENERIC),
182ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de007b, "GPU 7b HDMI/DP", MODEL_GENERIC),
183ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de007c, "GPU 7c HDMI/DP", MODEL_GENERIC),
184ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de007d, "GPU 7d HDMI/DP", MODEL_GENERIC),
185ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de007e, "GPU 7e HDMI/DP", MODEL_GENERIC),
186ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0080, "GPU 80 HDMI/DP", MODEL_GENERIC),
187ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0081, "GPU 81 HDMI/DP", MODEL_GENERIC),
188ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0082, "GPU 82 HDMI/DP", MODEL_GENERIC),
189ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0083, "GPU 83 HDMI/DP", MODEL_GENERIC),
190ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0084, "GPU 84 HDMI/DP", MODEL_GENERIC),
191ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0090, "GPU 90 HDMI/DP", MODEL_GENERIC),
192ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0091, "GPU 91 HDMI/DP", MODEL_GENERIC),
193ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0092, "GPU 92 HDMI/DP", MODEL_GENERIC),
194ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0093, "GPU 93 HDMI/DP", MODEL_GENERIC),
195ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0094, "GPU 94 HDMI/DP", MODEL_GENERIC),
196ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0095, "GPU 95 HDMI/DP", MODEL_GENERIC),
197ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0097, "GPU 97 HDMI/DP", MODEL_GENERIC),
198ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0098, "GPU 98 HDMI/DP", MODEL_GENERIC),
199ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de0099, "GPU 99 HDMI/DP", MODEL_GENERIC),
200ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de009a, "GPU 9a HDMI/DP", MODEL_GENERIC),
201*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de009b, "GPU 9b HDMI/DP", MODEL_GENERIC),
202*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de009c, "GPU 9c HDMI/DP", MODEL_GENERIC),
203ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de009d, "GPU 9d HDMI/DP", MODEL_GENERIC),
204ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de009e, "GPU 9e HDMI/DP", MODEL_GENERIC),
205ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de009f, "GPU 9f HDMI/DP", MODEL_GENERIC),
206ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de00a0, "GPU a0 HDMI/DP", MODEL_GENERIC),
207*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00a1, "GPU a1 HDMI/DP", MODEL_GENERIC),
208ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de00a3, "GPU a3 HDMI/DP", MODEL_GENERIC),
209ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de00a4, "GPU a4 HDMI/DP", MODEL_GENERIC),
210ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de00a5, "GPU a5 HDMI/DP", MODEL_GENERIC),
211ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de00a6, "GPU a6 HDMI/DP", MODEL_GENERIC),
212ad781b55STakashi Iwai HDA_CODEC_ID_MODEL(0x10de00a7, "GPU a7 HDMI/DP", MODEL_GENERIC),
213*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00a8, "GPU a8 HDMI/DP", MODEL_GENERIC),
214*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00a9, "GPU a9 HDMI/DP", MODEL_GENERIC),
215*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00aa, "GPU aa HDMI/DP", MODEL_GENERIC),
216*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00ab, "GPU ab HDMI/DP", MODEL_GENERIC),
217*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00ad, "GPU ad HDMI/DP", MODEL_GENERIC),
218*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00ae, "GPU ae HDMI/DP", MODEL_GENERIC),
219*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00af, "GPU af HDMI/DP", MODEL_GENERIC),
220*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00b0, "GPU b0 HDMI/DP", MODEL_GENERIC),
221*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00b1, "GPU b1 HDMI/DP", MODEL_GENERIC),
222*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00c0, "GPU c0 HDMI/DP", MODEL_GENERIC),
223*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00c1, "GPU c1 HDMI/DP", MODEL_GENERIC),
224*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00c3, "GPU c3 HDMI/DP", MODEL_GENERIC),
225*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00c4, "GPU c4 HDMI/DP", MODEL_GENERIC),
226*1148bb0cSDaniel Dadap HDA_CODEC_ID_MODEL(0x10de00c5, "GPU c5 HDMI/DP", MODEL_GENERIC),
22773cd0490STakashi Iwai {} /* terminator */
22873cd0490STakashi Iwai };
22973cd0490STakashi Iwai MODULE_DEVICE_TABLE(hdaudio, snd_hda_id_nvhdmi);
23073cd0490STakashi Iwai
23173cd0490STakashi Iwai MODULE_LICENSE("GPL");
23273cd0490STakashi Iwai MODULE_DESCRIPTION("Nvidia HDMI HD-audio codec");
23373cd0490STakashi Iwai MODULE_IMPORT_NS("SND_HDA_CODEC_HDMI");
23473cd0490STakashi Iwai
23573cd0490STakashi Iwai static struct hda_codec_driver nvhdmi_driver = {
23673cd0490STakashi Iwai .id = snd_hda_id_nvhdmi,
237ad781b55STakashi Iwai .ops = &nvhdmi_codec_ops,
23873cd0490STakashi Iwai };
23973cd0490STakashi Iwai
24073cd0490STakashi Iwai module_hda_codec_driver(nvhdmi_driver);
241