Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
torvalds
GitHub Repository: torvalds/linux
Path: blob/master/sound/hda/codecs/hdmi/nvhdmi.c
26490 views
1
// SPDX-License-Identifier: GPL-2.0-or-later
2
/*
3
* Nvidia HDMI codec support
4
*/
5
6
#include <linux/init.h>
7
#include <linux/slab.h>
8
#include <linux/module.h>
9
#include <sound/core.h>
10
#include <sound/tlv.h>
11
#include <sound/hdaudio.h>
12
#include <sound/hda_codec.h>
13
#include "hda_local.h"
14
#include "hdmi_local.h"
15
16
enum {
17
MODEL_GENERIC,
18
MODEL_LEGACY,
19
};
20
21
/*
22
* NVIDIA codecs ignore ASP mapping for 2ch - confirmed on:
23
* - 0x10de0015
24
* - 0x10de0040
25
*/
26
static int nvhdmi_chmap_cea_alloc_validate_get_type(struct hdac_chmap *chmap,
27
struct hdac_cea_channel_speaker_allocation *cap, int channels)
28
{
29
if (cap->ca_index == 0x00 && channels == 2)
30
return SNDRV_CTL_TLVT_CHMAP_FIXED;
31
32
/* If the speaker allocation matches the channel count, it is OK. */
33
if (cap->channels != channels)
34
return -1;
35
36
/* all channels are remappable freely */
37
return SNDRV_CTL_TLVT_CHMAP_VAR;
38
}
39
40
static int nvhdmi_chmap_validate(struct hdac_chmap *chmap,
41
int ca, int chs, unsigned char *map)
42
{
43
if (ca == 0x00 && (map[0] != SNDRV_CHMAP_FL || map[1] != SNDRV_CHMAP_FR))
44
return -EINVAL;
45
46
return 0;
47
}
48
49
/* map from pin NID to port; port is 0-based */
50
/* for Nvidia: assume widget NID starting from 4, with step 1 (4, 5, 6, ...) */
51
static int nvhdmi_pin2port(void *audio_ptr, int pin_nid)
52
{
53
return pin_nid - 4;
54
}
55
56
/* reverse-map from port to pin NID: see above */
57
static int nvhdmi_port2pin(struct hda_codec *codec, int port)
58
{
59
return port + 4;
60
}
61
62
static const struct drm_audio_component_audio_ops nvhdmi_audio_ops = {
63
.pin2port = nvhdmi_pin2port,
64
.pin_eld_notify = snd_hda_hdmi_acomp_pin_eld_notify,
65
.master_bind = snd_hda_hdmi_acomp_master_bind,
66
.master_unbind = snd_hda_hdmi_acomp_master_unbind,
67
};
68
69
static int probe_generic(struct hda_codec *codec)
70
{
71
struct hdmi_spec *spec;
72
int err;
73
74
err = snd_hda_hdmi_generic_alloc(codec);
75
if (err < 0)
76
return err;
77
codec->dp_mst = true;
78
79
spec = codec->spec;
80
81
err = snd_hda_hdmi_parse_codec(codec);
82
if (err < 0) {
83
snd_hda_hdmi_generic_spec_free(codec);
84
return err;
85
}
86
87
snd_hda_hdmi_generic_init_per_pins(codec);
88
89
spec->dyn_pin_out = true;
90
91
spec->chmap.ops.chmap_cea_alloc_validate_get_type =
92
nvhdmi_chmap_cea_alloc_validate_get_type;
93
spec->chmap.ops.chmap_validate = nvhdmi_chmap_validate;
94
spec->nv_dp_workaround = true;
95
96
codec->link_down_at_suspend = 1;
97
98
snd_hda_hdmi_acomp_init(codec, &nvhdmi_audio_ops, nvhdmi_port2pin);
99
100
return 0;
101
}
102
103
static int probe_legacy(struct hda_codec *codec)
104
{
105
struct hdmi_spec *spec;
106
int err;
107
108
err = snd_hda_hdmi_generic_probe(codec);
109
if (err)
110
return err;
111
112
spec = codec->spec;
113
spec->dyn_pin_out = true;
114
115
spec->chmap.ops.chmap_cea_alloc_validate_get_type =
116
nvhdmi_chmap_cea_alloc_validate_get_type;
117
spec->chmap.ops.chmap_validate = nvhdmi_chmap_validate;
118
spec->nv_dp_workaround = true;
119
120
codec->link_down_at_suspend = 1;
121
122
return 0;
123
}
124
125
static int nvhdmi_probe(struct hda_codec *codec, const struct hda_device_id *id)
126
{
127
if (id->driver_data == MODEL_LEGACY)
128
return probe_legacy(codec);
129
else
130
return probe_generic(codec);
131
}
132
133
static const struct hda_codec_ops nvhdmi_codec_ops = {
134
.probe = nvhdmi_probe,
135
.remove = snd_hda_hdmi_generic_remove,
136
.init = snd_hda_hdmi_generic_init,
137
.build_pcms = snd_hda_hdmi_generic_build_pcms,
138
.build_controls = snd_hda_hdmi_generic_build_controls,
139
.unsol_event = snd_hda_hdmi_generic_unsol_event,
140
.suspend = snd_hda_hdmi_generic_suspend,
141
.resume = snd_hda_hdmi_generic_resume,
142
};
143
144
static const struct hda_device_id snd_hda_id_nvhdmi[] = {
145
HDA_CODEC_ID_MODEL(0x10de0008, "GPU 08 HDMI/DP", MODEL_LEGACY),
146
HDA_CODEC_ID_MODEL(0x10de0009, "GPU 09 HDMI/DP", MODEL_LEGACY),
147
HDA_CODEC_ID_MODEL(0x10de000a, "GPU 0a HDMI/DP", MODEL_LEGACY),
148
HDA_CODEC_ID_MODEL(0x10de000b, "GPU 0b HDMI/DP", MODEL_LEGACY),
149
HDA_CODEC_ID_MODEL(0x10de000c, "MCP89 HDMI", MODEL_LEGACY),
150
HDA_CODEC_ID_MODEL(0x10de000d, "GPU 0d HDMI/DP", MODEL_LEGACY),
151
HDA_CODEC_ID_MODEL(0x10de0010, "GPU 10 HDMI/DP", MODEL_LEGACY),
152
HDA_CODEC_ID_MODEL(0x10de0011, "GPU 11 HDMI/DP", MODEL_LEGACY),
153
HDA_CODEC_ID_MODEL(0x10de0012, "GPU 12 HDMI/DP", MODEL_LEGACY),
154
HDA_CODEC_ID_MODEL(0x10de0013, "GPU 13 HDMI/DP", MODEL_LEGACY),
155
HDA_CODEC_ID_MODEL(0x10de0014, "GPU 14 HDMI/DP", MODEL_LEGACY),
156
HDA_CODEC_ID_MODEL(0x10de0015, "GPU 15 HDMI/DP", MODEL_LEGACY),
157
HDA_CODEC_ID_MODEL(0x10de0016, "GPU 16 HDMI/DP", MODEL_LEGACY),
158
/* 17 is known to be absent */
159
HDA_CODEC_ID_MODEL(0x10de0018, "GPU 18 HDMI/DP", MODEL_LEGACY),
160
HDA_CODEC_ID_MODEL(0x10de0019, "GPU 19 HDMI/DP", MODEL_LEGACY),
161
HDA_CODEC_ID_MODEL(0x10de001a, "GPU 1a HDMI/DP", MODEL_LEGACY),
162
HDA_CODEC_ID_MODEL(0x10de001b, "GPU 1b HDMI/DP", MODEL_LEGACY),
163
HDA_CODEC_ID_MODEL(0x10de001c, "GPU 1c HDMI/DP", MODEL_LEGACY),
164
HDA_CODEC_ID_MODEL(0x10de0040, "GPU 40 HDMI/DP", MODEL_GENERIC),
165
HDA_CODEC_ID_MODEL(0x10de0041, "GPU 41 HDMI/DP", MODEL_GENERIC),
166
HDA_CODEC_ID_MODEL(0x10de0042, "GPU 42 HDMI/DP", MODEL_GENERIC),
167
HDA_CODEC_ID_MODEL(0x10de0043, "GPU 43 HDMI/DP", MODEL_GENERIC),
168
HDA_CODEC_ID_MODEL(0x10de0044, "GPU 44 HDMI/DP", MODEL_GENERIC),
169
HDA_CODEC_ID_MODEL(0x10de0045, "GPU 45 HDMI/DP", MODEL_GENERIC),
170
HDA_CODEC_ID_MODEL(0x10de0050, "GPU 50 HDMI/DP", MODEL_GENERIC),
171
HDA_CODEC_ID_MODEL(0x10de0051, "GPU 51 HDMI/DP", MODEL_GENERIC),
172
HDA_CODEC_ID_MODEL(0x10de0052, "GPU 52 HDMI/DP", MODEL_GENERIC),
173
HDA_CODEC_ID_MODEL(0x10de0060, "GPU 60 HDMI/DP", MODEL_GENERIC),
174
HDA_CODEC_ID_MODEL(0x10de0061, "GPU 61 HDMI/DP", MODEL_GENERIC),
175
HDA_CODEC_ID_MODEL(0x10de0062, "GPU 62 HDMI/DP", MODEL_GENERIC),
176
HDA_CODEC_ID_MODEL(0x10de0070, "GPU 70 HDMI/DP", MODEL_GENERIC),
177
HDA_CODEC_ID_MODEL(0x10de0071, "GPU 71 HDMI/DP", MODEL_GENERIC),
178
HDA_CODEC_ID_MODEL(0x10de0072, "GPU 72 HDMI/DP", MODEL_GENERIC),
179
HDA_CODEC_ID_MODEL(0x10de0073, "GPU 73 HDMI/DP", MODEL_GENERIC),
180
HDA_CODEC_ID_MODEL(0x10de0074, "GPU 74 HDMI/DP", MODEL_GENERIC),
181
HDA_CODEC_ID_MODEL(0x10de0076, "GPU 76 HDMI/DP", MODEL_GENERIC),
182
HDA_CODEC_ID_MODEL(0x10de007b, "GPU 7b HDMI/DP", MODEL_GENERIC),
183
HDA_CODEC_ID_MODEL(0x10de007c, "GPU 7c HDMI/DP", MODEL_GENERIC),
184
HDA_CODEC_ID_MODEL(0x10de007d, "GPU 7d HDMI/DP", MODEL_GENERIC),
185
HDA_CODEC_ID_MODEL(0x10de007e, "GPU 7e HDMI/DP", MODEL_GENERIC),
186
HDA_CODEC_ID_MODEL(0x10de0080, "GPU 80 HDMI/DP", MODEL_GENERIC),
187
HDA_CODEC_ID_MODEL(0x10de0081, "GPU 81 HDMI/DP", MODEL_GENERIC),
188
HDA_CODEC_ID_MODEL(0x10de0082, "GPU 82 HDMI/DP", MODEL_GENERIC),
189
HDA_CODEC_ID_MODEL(0x10de0083, "GPU 83 HDMI/DP", MODEL_GENERIC),
190
HDA_CODEC_ID_MODEL(0x10de0084, "GPU 84 HDMI/DP", MODEL_GENERIC),
191
HDA_CODEC_ID_MODEL(0x10de0090, "GPU 90 HDMI/DP", MODEL_GENERIC),
192
HDA_CODEC_ID_MODEL(0x10de0091, "GPU 91 HDMI/DP", MODEL_GENERIC),
193
HDA_CODEC_ID_MODEL(0x10de0092, "GPU 92 HDMI/DP", MODEL_GENERIC),
194
HDA_CODEC_ID_MODEL(0x10de0093, "GPU 93 HDMI/DP", MODEL_GENERIC),
195
HDA_CODEC_ID_MODEL(0x10de0094, "GPU 94 HDMI/DP", MODEL_GENERIC),
196
HDA_CODEC_ID_MODEL(0x10de0095, "GPU 95 HDMI/DP", MODEL_GENERIC),
197
HDA_CODEC_ID_MODEL(0x10de0097, "GPU 97 HDMI/DP", MODEL_GENERIC),
198
HDA_CODEC_ID_MODEL(0x10de0098, "GPU 98 HDMI/DP", MODEL_GENERIC),
199
HDA_CODEC_ID_MODEL(0x10de0099, "GPU 99 HDMI/DP", MODEL_GENERIC),
200
HDA_CODEC_ID_MODEL(0x10de009a, "GPU 9a HDMI/DP", MODEL_GENERIC),
201
HDA_CODEC_ID_MODEL(0x10de009b, "GPU 9b HDMI/DP", MODEL_GENERIC),
202
HDA_CODEC_ID_MODEL(0x10de009c, "GPU 9c HDMI/DP", MODEL_GENERIC),
203
HDA_CODEC_ID_MODEL(0x10de009d, "GPU 9d HDMI/DP", MODEL_GENERIC),
204
HDA_CODEC_ID_MODEL(0x10de009e, "GPU 9e HDMI/DP", MODEL_GENERIC),
205
HDA_CODEC_ID_MODEL(0x10de009f, "GPU 9f HDMI/DP", MODEL_GENERIC),
206
HDA_CODEC_ID_MODEL(0x10de00a0, "GPU a0 HDMI/DP", MODEL_GENERIC),
207
HDA_CODEC_ID_MODEL(0x10de00a1, "GPU a1 HDMI/DP", MODEL_GENERIC),
208
HDA_CODEC_ID_MODEL(0x10de00a3, "GPU a3 HDMI/DP", MODEL_GENERIC),
209
HDA_CODEC_ID_MODEL(0x10de00a4, "GPU a4 HDMI/DP", MODEL_GENERIC),
210
HDA_CODEC_ID_MODEL(0x10de00a5, "GPU a5 HDMI/DP", MODEL_GENERIC),
211
HDA_CODEC_ID_MODEL(0x10de00a6, "GPU a6 HDMI/DP", MODEL_GENERIC),
212
HDA_CODEC_ID_MODEL(0x10de00a7, "GPU a7 HDMI/DP", MODEL_GENERIC),
213
HDA_CODEC_ID_MODEL(0x10de00a8, "GPU a8 HDMI/DP", MODEL_GENERIC),
214
HDA_CODEC_ID_MODEL(0x10de00a9, "GPU a9 HDMI/DP", MODEL_GENERIC),
215
HDA_CODEC_ID_MODEL(0x10de00aa, "GPU aa HDMI/DP", MODEL_GENERIC),
216
HDA_CODEC_ID_MODEL(0x10de00ab, "GPU ab HDMI/DP", MODEL_GENERIC),
217
HDA_CODEC_ID_MODEL(0x10de00ad, "GPU ad HDMI/DP", MODEL_GENERIC),
218
HDA_CODEC_ID_MODEL(0x10de00ae, "GPU ae HDMI/DP", MODEL_GENERIC),
219
HDA_CODEC_ID_MODEL(0x10de00af, "GPU af HDMI/DP", MODEL_GENERIC),
220
HDA_CODEC_ID_MODEL(0x10de00b0, "GPU b0 HDMI/DP", MODEL_GENERIC),
221
HDA_CODEC_ID_MODEL(0x10de00b1, "GPU b1 HDMI/DP", MODEL_GENERIC),
222
HDA_CODEC_ID_MODEL(0x10de00c0, "GPU c0 HDMI/DP", MODEL_GENERIC),
223
HDA_CODEC_ID_MODEL(0x10de00c1, "GPU c1 HDMI/DP", MODEL_GENERIC),
224
HDA_CODEC_ID_MODEL(0x10de00c3, "GPU c3 HDMI/DP", MODEL_GENERIC),
225
HDA_CODEC_ID_MODEL(0x10de00c4, "GPU c4 HDMI/DP", MODEL_GENERIC),
226
HDA_CODEC_ID_MODEL(0x10de00c5, "GPU c5 HDMI/DP", MODEL_GENERIC),
227
{} /* terminator */
228
};
229
MODULE_DEVICE_TABLE(hdaudio, snd_hda_id_nvhdmi);
230
231
MODULE_LICENSE("GPL");
232
MODULE_DESCRIPTION("Nvidia HDMI HD-audio codec");
233
MODULE_IMPORT_NS("SND_HDA_CODEC_HDMI");
234
235
static struct hda_codec_driver nvhdmi_driver = {
236
.id = snd_hda_id_nvhdmi,
237
.ops = &nvhdmi_codec_ops,
238
};
239
240
module_hda_codec_driver(nvhdmi_driver);
241
242