xref: /linux/drivers/gpu/drm/i915/display/intel_display_device.c (revision 3e0bc2855b573bcffa2a52955a878f537f5ac0cd)
1 // SPDX-License-Identifier: MIT
2 /*
3  * Copyright © 2023 Intel Corporation
4  */
5 
6 #include <drm/i915_pciids.h>
7 #include <drm/drm_color_mgmt.h>
8 #include <linux/pci.h>
9 
10 #include "i915_drv.h"
11 #include "i915_reg.h"
12 #include "intel_de.h"
13 #include "intel_display.h"
14 #include "intel_display_device.h"
15 #include "intel_display_params.h"
16 #include "intel_display_power.h"
17 #include "intel_display_reg_defs.h"
18 #include "intel_fbc.h"
19 
20 static const struct intel_display_device_info no_display = {};
21 
22 #define PIPE_A_OFFSET		0x70000
23 #define PIPE_B_OFFSET		0x71000
24 #define PIPE_C_OFFSET		0x72000
25 #define PIPE_D_OFFSET		0x73000
26 #define CHV_PIPE_C_OFFSET	0x74000
27 /*
28  * There's actually no pipe EDP. Some pipe registers have
29  * simply shifted from the pipe to the transcoder, while
30  * keeping their original offset. Thus we need PIPE_EDP_OFFSET
31  * to access such registers in transcoder EDP.
32  */
33 #define PIPE_EDP_OFFSET	0x7f000
34 
35 /* ICL DSI 0 and 1 */
36 #define PIPE_DSI0_OFFSET	0x7b000
37 #define PIPE_DSI1_OFFSET	0x7b800
38 
39 #define TRANSCODER_A_OFFSET 0x60000
40 #define TRANSCODER_B_OFFSET 0x61000
41 #define TRANSCODER_C_OFFSET 0x62000
42 #define CHV_TRANSCODER_C_OFFSET 0x63000
43 #define TRANSCODER_D_OFFSET 0x63000
44 #define TRANSCODER_EDP_OFFSET 0x6f000
45 #define TRANSCODER_DSI0_OFFSET	0x6b000
46 #define TRANSCODER_DSI1_OFFSET	0x6b800
47 
48 #define CURSOR_A_OFFSET 0x70080
49 #define CURSOR_B_OFFSET 0x700c0
50 #define CHV_CURSOR_C_OFFSET 0x700e0
51 #define IVB_CURSOR_B_OFFSET 0x71080
52 #define IVB_CURSOR_C_OFFSET 0x72080
53 #define TGL_CURSOR_D_OFFSET 0x73080
54 
55 #define I845_PIPE_OFFSETS \
56 	.pipe_offsets = { \
57 		[TRANSCODER_A] = PIPE_A_OFFSET,	\
58 	}, \
59 	.trans_offsets = { \
60 		[TRANSCODER_A] = TRANSCODER_A_OFFSET, \
61 	}
62 
63 #define I9XX_PIPE_OFFSETS \
64 	.pipe_offsets = { \
65 		[TRANSCODER_A] = PIPE_A_OFFSET,	\
66 		[TRANSCODER_B] = PIPE_B_OFFSET, \
67 	}, \
68 	.trans_offsets = { \
69 		[TRANSCODER_A] = TRANSCODER_A_OFFSET, \
70 		[TRANSCODER_B] = TRANSCODER_B_OFFSET, \
71 	}
72 
73 #define IVB_PIPE_OFFSETS \
74 	.pipe_offsets = { \
75 		[TRANSCODER_A] = PIPE_A_OFFSET,	\
76 		[TRANSCODER_B] = PIPE_B_OFFSET, \
77 		[TRANSCODER_C] = PIPE_C_OFFSET, \
78 	}, \
79 	.trans_offsets = { \
80 		[TRANSCODER_A] = TRANSCODER_A_OFFSET, \
81 		[TRANSCODER_B] = TRANSCODER_B_OFFSET, \
82 		[TRANSCODER_C] = TRANSCODER_C_OFFSET, \
83 	}
84 
85 #define HSW_PIPE_OFFSETS \
86 	.pipe_offsets = { \
87 		[TRANSCODER_A] = PIPE_A_OFFSET,	\
88 		[TRANSCODER_B] = PIPE_B_OFFSET, \
89 		[TRANSCODER_C] = PIPE_C_OFFSET, \
90 		[TRANSCODER_EDP] = PIPE_EDP_OFFSET, \
91 	}, \
92 	.trans_offsets = { \
93 		[TRANSCODER_A] = TRANSCODER_A_OFFSET, \
94 		[TRANSCODER_B] = TRANSCODER_B_OFFSET, \
95 		[TRANSCODER_C] = TRANSCODER_C_OFFSET, \
96 		[TRANSCODER_EDP] = TRANSCODER_EDP_OFFSET, \
97 	}
98 
99 #define CHV_PIPE_OFFSETS \
100 	.pipe_offsets = { \
101 		[TRANSCODER_A] = PIPE_A_OFFSET, \
102 		[TRANSCODER_B] = PIPE_B_OFFSET, \
103 		[TRANSCODER_C] = CHV_PIPE_C_OFFSET, \
104 	}, \
105 	.trans_offsets = { \
106 		[TRANSCODER_A] = TRANSCODER_A_OFFSET, \
107 		[TRANSCODER_B] = TRANSCODER_B_OFFSET, \
108 		[TRANSCODER_C] = CHV_TRANSCODER_C_OFFSET, \
109 	}
110 
111 #define I845_CURSOR_OFFSETS \
112 	.cursor_offsets = { \
113 		[PIPE_A] = CURSOR_A_OFFSET, \
114 	}
115 
116 #define I9XX_CURSOR_OFFSETS \
117 	.cursor_offsets = { \
118 		[PIPE_A] = CURSOR_A_OFFSET, \
119 		[PIPE_B] = CURSOR_B_OFFSET, \
120 	}
121 
122 #define CHV_CURSOR_OFFSETS \
123 	.cursor_offsets = { \
124 		[PIPE_A] = CURSOR_A_OFFSET, \
125 		[PIPE_B] = CURSOR_B_OFFSET, \
126 		[PIPE_C] = CHV_CURSOR_C_OFFSET, \
127 	}
128 
129 #define IVB_CURSOR_OFFSETS \
130 	.cursor_offsets = { \
131 		[PIPE_A] = CURSOR_A_OFFSET, \
132 		[PIPE_B] = IVB_CURSOR_B_OFFSET, \
133 		[PIPE_C] = IVB_CURSOR_C_OFFSET, \
134 	}
135 
136 #define TGL_CURSOR_OFFSETS \
137 	.cursor_offsets = { \
138 		[PIPE_A] = CURSOR_A_OFFSET, \
139 		[PIPE_B] = IVB_CURSOR_B_OFFSET, \
140 		[PIPE_C] = IVB_CURSOR_C_OFFSET, \
141 		[PIPE_D] = TGL_CURSOR_D_OFFSET, \
142 	}
143 
144 #define I845_COLORS \
145 	.color = { .gamma_lut_size = 256 }
146 #define I9XX_COLORS \
147 	.color = { .gamma_lut_size = 129, \
148 		   .gamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING, \
149 	}
150 #define ILK_COLORS \
151 	.color = { .gamma_lut_size = 1024 }
152 #define IVB_COLORS \
153 	.color = { .degamma_lut_size = 1024, .gamma_lut_size = 1024 }
154 #define CHV_COLORS \
155 	.color = { \
156 		.degamma_lut_size = 65, .gamma_lut_size = 257, \
157 		.degamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING, \
158 		.gamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING, \
159 	}
160 #define GLK_COLORS \
161 	.color = { \
162 		.degamma_lut_size = 33, .gamma_lut_size = 1024, \
163 		.degamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING | \
164 				     DRM_COLOR_LUT_EQUAL_CHANNELS, \
165 	}
166 #define ICL_COLORS \
167 	.color = { \
168 		.degamma_lut_size = 33, .gamma_lut_size = 262145, \
169 		.degamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING | \
170 				     DRM_COLOR_LUT_EQUAL_CHANNELS, \
171 		.gamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING, \
172 	}
173 
174 #define I830_DISPLAY \
175 	.has_overlay = 1, \
176 	.cursor_needs_physical = 1, \
177 	.overlay_needs_physical = 1, \
178 	.has_gmch = 1, \
179 	I9XX_PIPE_OFFSETS, \
180 	I9XX_CURSOR_OFFSETS, \
181 	I9XX_COLORS, \
182 	\
183 	.__runtime_defaults.ip.ver = 2, \
184 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B), \
185 	.__runtime_defaults.cpu_transcoder_mask = \
186 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B)
187 
188 #define I845_DISPLAY \
189 	.has_overlay = 1, \
190 	.overlay_needs_physical = 1, \
191 	.has_gmch = 1, \
192 	I845_PIPE_OFFSETS, \
193 	I845_CURSOR_OFFSETS, \
194 	I845_COLORS, \
195 	\
196 	.__runtime_defaults.ip.ver = 2, \
197 	.__runtime_defaults.pipe_mask = BIT(PIPE_A), \
198 	.__runtime_defaults.cpu_transcoder_mask = BIT(TRANSCODER_A)
199 
200 static const struct intel_display_device_info i830_display = {
201 	I830_DISPLAY,
202 
203 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C), /* DVO A/B/C */
204 };
205 
206 static const struct intel_display_device_info i845_display = {
207 	I845_DISPLAY,
208 
209 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C), /* DVO B/C */
210 };
211 
212 static const struct intel_display_device_info i85x_display = {
213 	I830_DISPLAY,
214 
215 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C), /* DVO B/C */
216 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
217 };
218 
219 static const struct intel_display_device_info i865g_display = {
220 	I845_DISPLAY,
221 
222 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C), /* DVO B/C */
223 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
224 };
225 
226 #define GEN3_DISPLAY \
227 	.has_gmch = 1, \
228 	.has_overlay = 1, \
229 	I9XX_PIPE_OFFSETS, \
230 	I9XX_CURSOR_OFFSETS, \
231 	\
232 	.__runtime_defaults.ip.ver = 3, \
233 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B), \
234 	.__runtime_defaults.cpu_transcoder_mask = \
235 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B), \
236 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C) /* SDVO B/C */
237 
238 static const struct intel_display_device_info i915g_display = {
239 	GEN3_DISPLAY,
240 	I845_COLORS,
241 	.cursor_needs_physical = 1,
242 	.overlay_needs_physical = 1,
243 };
244 
245 static const struct intel_display_device_info i915gm_display = {
246 	GEN3_DISPLAY,
247 	I9XX_COLORS,
248 	.cursor_needs_physical = 1,
249 	.overlay_needs_physical = 1,
250 	.supports_tv = 1,
251 
252 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
253 };
254 
255 static const struct intel_display_device_info i945g_display = {
256 	GEN3_DISPLAY,
257 	I845_COLORS,
258 	.has_hotplug = 1,
259 	.cursor_needs_physical = 1,
260 	.overlay_needs_physical = 1,
261 };
262 
263 static const struct intel_display_device_info i945gm_display = {
264 	GEN3_DISPLAY,
265 	I9XX_COLORS,
266 	.has_hotplug = 1,
267 	.cursor_needs_physical = 1,
268 	.overlay_needs_physical = 1,
269 	.supports_tv = 1,
270 
271 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
272 };
273 
274 static const struct intel_display_device_info g33_display = {
275 	GEN3_DISPLAY,
276 	I845_COLORS,
277 	.has_hotplug = 1,
278 };
279 
280 static const struct intel_display_device_info pnv_display = {
281 	GEN3_DISPLAY,
282 	I9XX_COLORS,
283 	.has_hotplug = 1,
284 };
285 
286 #define GEN4_DISPLAY \
287 	.has_hotplug = 1, \
288 	.has_gmch = 1, \
289 	I9XX_PIPE_OFFSETS, \
290 	I9XX_CURSOR_OFFSETS, \
291 	I9XX_COLORS, \
292 	\
293 	.__runtime_defaults.ip.ver = 4, \
294 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B), \
295 	.__runtime_defaults.cpu_transcoder_mask = \
296 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B)
297 
298 static const struct intel_display_device_info i965g_display = {
299 	GEN4_DISPLAY,
300 	.has_overlay = 1,
301 
302 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C), /* SDVO B/C */
303 };
304 
305 static const struct intel_display_device_info i965gm_display = {
306 	GEN4_DISPLAY,
307 	.has_overlay = 1,
308 	.supports_tv = 1,
309 
310 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C), /* SDVO B/C */
311 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
312 };
313 
314 static const struct intel_display_device_info g45_display = {
315 	GEN4_DISPLAY,
316 
317 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D), /* SDVO/HDMI/DP B/C, DP D */
318 };
319 
320 static const struct intel_display_device_info gm45_display = {
321 	GEN4_DISPLAY,
322 	.supports_tv = 1,
323 
324 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D), /* SDVO/HDMI/DP B/C, DP D */
325 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
326 };
327 
328 #define ILK_DISPLAY \
329 	.has_hotplug = 1, \
330 	I9XX_PIPE_OFFSETS, \
331 	I9XX_CURSOR_OFFSETS, \
332 	ILK_COLORS, \
333 	\
334 	.__runtime_defaults.ip.ver = 5, \
335 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B), \
336 	.__runtime_defaults.cpu_transcoder_mask = \
337 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B), \
338 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D) /* DP A, SDVO/HDMI/DP B, HDMI/DP C/D */
339 
340 static const struct intel_display_device_info ilk_d_display = {
341 	ILK_DISPLAY,
342 };
343 
344 static const struct intel_display_device_info ilk_m_display = {
345 	ILK_DISPLAY,
346 
347 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
348 };
349 
350 static const struct intel_display_device_info snb_display = {
351 	.has_hotplug = 1,
352 	I9XX_PIPE_OFFSETS,
353 	I9XX_CURSOR_OFFSETS,
354 	ILK_COLORS,
355 
356 	.__runtime_defaults.ip.ver = 6,
357 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B),
358 	.__runtime_defaults.cpu_transcoder_mask =
359 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B),
360 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D), /* DP A, SDVO/HDMI/DP B, HDMI/DP C/D */
361 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
362 };
363 
364 static const struct intel_display_device_info ivb_display = {
365 	.has_hotplug = 1,
366 	IVB_PIPE_OFFSETS,
367 	IVB_CURSOR_OFFSETS,
368 	IVB_COLORS,
369 
370 	.__runtime_defaults.ip.ver = 7,
371 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C),
372 	.__runtime_defaults.cpu_transcoder_mask =
373 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) | BIT(TRANSCODER_C),
374 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D), /* DP A, SDVO/HDMI/DP B, HDMI/DP C/D */
375 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
376 };
377 
378 static const struct intel_display_device_info vlv_display = {
379 	.has_gmch = 1,
380 	.has_hotplug = 1,
381 	.mmio_offset = VLV_DISPLAY_BASE,
382 	I9XX_PIPE_OFFSETS,
383 	I9XX_CURSOR_OFFSETS,
384 	I9XX_COLORS,
385 
386 	.__runtime_defaults.ip.ver = 7,
387 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B),
388 	.__runtime_defaults.cpu_transcoder_mask =
389 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B),
390 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C), /* HDMI/DP B/C */
391 };
392 
393 static const struct intel_display_device_info hsw_display = {
394 	.has_ddi = 1,
395 	.has_dp_mst = 1,
396 	.has_fpga_dbg = 1,
397 	.has_hotplug = 1,
398 	.has_psr = 1,
399 	.has_psr_hw_tracking = 1,
400 	HSW_PIPE_OFFSETS,
401 	IVB_CURSOR_OFFSETS,
402 	IVB_COLORS,
403 
404 	.__runtime_defaults.ip.ver = 7,
405 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C),
406 	.__runtime_defaults.cpu_transcoder_mask =
407 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) |
408 		BIT(TRANSCODER_C) | BIT(TRANSCODER_EDP),
409 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D) | BIT(PORT_E),
410 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
411 };
412 
413 static const struct intel_display_device_info bdw_display = {
414 	.has_ddi = 1,
415 	.has_dp_mst = 1,
416 	.has_fpga_dbg = 1,
417 	.has_hotplug = 1,
418 	.has_psr = 1,
419 	.has_psr_hw_tracking = 1,
420 	HSW_PIPE_OFFSETS,
421 	IVB_CURSOR_OFFSETS,
422 	IVB_COLORS,
423 
424 	.__runtime_defaults.ip.ver = 8,
425 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C),
426 	.__runtime_defaults.cpu_transcoder_mask =
427 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) |
428 		BIT(TRANSCODER_C) | BIT(TRANSCODER_EDP),
429 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D) | BIT(PORT_E),
430 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
431 };
432 
433 static const struct intel_display_device_info chv_display = {
434 	.has_hotplug = 1,
435 	.has_gmch = 1,
436 	.mmio_offset = VLV_DISPLAY_BASE,
437 	CHV_PIPE_OFFSETS,
438 	CHV_CURSOR_OFFSETS,
439 	CHV_COLORS,
440 
441 	.__runtime_defaults.ip.ver = 8,
442 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C),
443 	.__runtime_defaults.cpu_transcoder_mask =
444 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) | BIT(TRANSCODER_C),
445 	.__runtime_defaults.port_mask = BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D), /* HDMI/DP B/C/D */
446 };
447 
448 static const struct intel_display_device_info skl_display = {
449 	.dbuf.size = 896 - 4, /* 4 blocks for bypass path allocation */
450 	.dbuf.slice_mask = BIT(DBUF_S1),
451 	.has_ddi = 1,
452 	.has_dp_mst = 1,
453 	.has_fpga_dbg = 1,
454 	.has_hotplug = 1,
455 	.has_ipc = 1,
456 	.has_psr = 1,
457 	.has_psr_hw_tracking = 1,
458 	HSW_PIPE_OFFSETS,
459 	IVB_CURSOR_OFFSETS,
460 	IVB_COLORS,
461 
462 	.__runtime_defaults.ip.ver = 9,
463 	.__runtime_defaults.has_dmc = 1,
464 	.__runtime_defaults.has_hdcp = 1,
465 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C),
466 	.__runtime_defaults.cpu_transcoder_mask =
467 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) |
468 		BIT(TRANSCODER_C) | BIT(TRANSCODER_EDP),
469 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D) | BIT(PORT_E),
470 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),
471 };
472 
473 #define GEN9_LP_DISPLAY \
474 	.dbuf.slice_mask = BIT(DBUF_S1), \
475 	.has_dp_mst = 1, \
476 	.has_ddi = 1, \
477 	.has_fpga_dbg = 1, \
478 	.has_hotplug = 1, \
479 	.has_ipc = 1, \
480 	.has_psr = 1, \
481 	.has_psr_hw_tracking = 1, \
482 	HSW_PIPE_OFFSETS, \
483 	IVB_CURSOR_OFFSETS, \
484 	IVB_COLORS, \
485 	\
486 	.__runtime_defaults.has_dmc = 1, \
487 	.__runtime_defaults.has_hdcp = 1, \
488 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A), \
489 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C), \
490 	.__runtime_defaults.cpu_transcoder_mask = \
491 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) | \
492 		BIT(TRANSCODER_C) | BIT(TRANSCODER_EDP) | \
493 		BIT(TRANSCODER_DSI_A) | BIT(TRANSCODER_DSI_C), \
494 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C)
495 
496 static const struct intel_display_device_info bxt_display = {
497 	GEN9_LP_DISPLAY,
498 	.dbuf.size = 512 - 4, /* 4 blocks for bypass path allocation */
499 
500 	.__runtime_defaults.ip.ver = 9,
501 };
502 
503 static const struct intel_display_device_info glk_display = {
504 	GEN9_LP_DISPLAY,
505 	.dbuf.size = 1024 - 4, /* 4 blocks for bypass path allocation */
506 	GLK_COLORS,
507 
508 	.__runtime_defaults.ip.ver = 10,
509 };
510 
511 #define ICL_DISPLAY \
512 	.abox_mask = BIT(0), \
513 	.dbuf.size = 2048, \
514 	.dbuf.slice_mask = BIT(DBUF_S1) | BIT(DBUF_S2), \
515 	.has_ddi = 1, \
516 	.has_dp_mst = 1, \
517 	.has_fpga_dbg = 1, \
518 	.has_hotplug = 1, \
519 	.has_ipc = 1, \
520 	.has_psr = 1, \
521 	.has_psr_hw_tracking = 1, \
522 	.pipe_offsets = { \
523 		[TRANSCODER_A] = PIPE_A_OFFSET, \
524 		[TRANSCODER_B] = PIPE_B_OFFSET, \
525 		[TRANSCODER_C] = PIPE_C_OFFSET, \
526 		[TRANSCODER_EDP] = PIPE_EDP_OFFSET, \
527 		[TRANSCODER_DSI_0] = PIPE_DSI0_OFFSET, \
528 		[TRANSCODER_DSI_1] = PIPE_DSI1_OFFSET, \
529 	}, \
530 	.trans_offsets = { \
531 		[TRANSCODER_A] = TRANSCODER_A_OFFSET, \
532 		[TRANSCODER_B] = TRANSCODER_B_OFFSET, \
533 		[TRANSCODER_C] = TRANSCODER_C_OFFSET, \
534 		[TRANSCODER_EDP] = TRANSCODER_EDP_OFFSET, \
535 		[TRANSCODER_DSI_0] = TRANSCODER_DSI0_OFFSET, \
536 		[TRANSCODER_DSI_1] = TRANSCODER_DSI1_OFFSET, \
537 	}, \
538 	IVB_CURSOR_OFFSETS, \
539 	ICL_COLORS, \
540 	\
541 	.__runtime_defaults.ip.ver = 11, \
542 	.__runtime_defaults.has_dmc = 1, \
543 	.__runtime_defaults.has_dsc = 1, \
544 	.__runtime_defaults.has_hdcp = 1, \
545 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C), \
546 	.__runtime_defaults.cpu_transcoder_mask = \
547 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) | \
548 		BIT(TRANSCODER_C) | BIT(TRANSCODER_EDP) | \
549 		BIT(TRANSCODER_DSI_0) | BIT(TRANSCODER_DSI_1), \
550 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A)
551 
552 static const struct intel_display_device_info icl_display = {
553 	ICL_DISPLAY,
554 
555 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D) | BIT(PORT_E),
556 };
557 
558 static const struct intel_display_device_info jsl_ehl_display = {
559 	ICL_DISPLAY,
560 
561 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D),
562 };
563 
564 #define XE_D_DISPLAY \
565 	.abox_mask = GENMASK(2, 1), \
566 	.dbuf.size = 2048, \
567 	.dbuf.slice_mask = BIT(DBUF_S1) | BIT(DBUF_S2), \
568 	.has_ddi = 1, \
569 	.has_dp_mst = 1, \
570 	.has_dsb = 1, \
571 	.has_fpga_dbg = 1, \
572 	.has_hotplug = 1, \
573 	.has_ipc = 1, \
574 	.has_psr = 1, \
575 	.has_psr_hw_tracking = 1, \
576 	.pipe_offsets = { \
577 		[TRANSCODER_A] = PIPE_A_OFFSET, \
578 		[TRANSCODER_B] = PIPE_B_OFFSET, \
579 		[TRANSCODER_C] = PIPE_C_OFFSET, \
580 		[TRANSCODER_D] = PIPE_D_OFFSET, \
581 		[TRANSCODER_DSI_0] = PIPE_DSI0_OFFSET, \
582 		[TRANSCODER_DSI_1] = PIPE_DSI1_OFFSET, \
583 	}, \
584 	.trans_offsets = { \
585 		[TRANSCODER_A] = TRANSCODER_A_OFFSET, \
586 		[TRANSCODER_B] = TRANSCODER_B_OFFSET, \
587 		[TRANSCODER_C] = TRANSCODER_C_OFFSET, \
588 		[TRANSCODER_D] = TRANSCODER_D_OFFSET, \
589 		[TRANSCODER_DSI_0] = TRANSCODER_DSI0_OFFSET, \
590 		[TRANSCODER_DSI_1] = TRANSCODER_DSI1_OFFSET, \
591 	}, \
592 	TGL_CURSOR_OFFSETS, \
593 	ICL_COLORS, \
594 	\
595 	.__runtime_defaults.ip.ver = 12, \
596 	.__runtime_defaults.has_dmc = 1, \
597 	.__runtime_defaults.has_dsc = 1, \
598 	.__runtime_defaults.has_hdcp = 1, \
599 	.__runtime_defaults.pipe_mask = \
600 		BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C) | BIT(PIPE_D), \
601 	.__runtime_defaults.cpu_transcoder_mask = \
602 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) | \
603 		BIT(TRANSCODER_C) | BIT(TRANSCODER_D) | \
604 		BIT(TRANSCODER_DSI_0) | BIT(TRANSCODER_DSI_1), \
605 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A)
606 
607 static const struct intel_display_device_info tgl_display = {
608 	XE_D_DISPLAY,
609 
610 	/*
611 	 * FIXME DDI C/combo PHY C missing due to combo PHY
612 	 * code making a mess on SKUs where the PHY is missing.
613 	 */
614 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) |
615 		BIT(PORT_TC1) | BIT(PORT_TC2) | BIT(PORT_TC3) | BIT(PORT_TC4) | BIT(PORT_TC5) | BIT(PORT_TC6),
616 };
617 
618 static const struct intel_display_device_info dg1_display = {
619 	XE_D_DISPLAY,
620 
621 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) |
622 		BIT(PORT_TC1) | BIT(PORT_TC2),
623 };
624 
625 static const struct intel_display_device_info rkl_display = {
626 	XE_D_DISPLAY,
627 	.abox_mask = BIT(0),
628 	.has_hti = 1,
629 	.has_psr_hw_tracking = 0,
630 
631 	.__runtime_defaults.pipe_mask = BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C),
632 	.__runtime_defaults.cpu_transcoder_mask =
633 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) | BIT(TRANSCODER_C),
634 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) |
635 		BIT(PORT_TC1) | BIT(PORT_TC2),
636 };
637 
638 static const struct intel_display_device_info adl_s_display = {
639 	XE_D_DISPLAY,
640 	.has_hti = 1,
641 	.has_psr_hw_tracking = 0,
642 
643 	.__runtime_defaults.port_mask = BIT(PORT_A) |
644 		BIT(PORT_TC1) | BIT(PORT_TC2) | BIT(PORT_TC3) | BIT(PORT_TC4),
645 };
646 
647 #define XE_LPD_FEATURES \
648 	.abox_mask = GENMASK(1, 0),						\
649 	.color = {								\
650 		.degamma_lut_size = 129, .gamma_lut_size = 1024,		\
651 		.degamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING |		\
652 		DRM_COLOR_LUT_EQUAL_CHANNELS,					\
653 	},									\
654 	.dbuf.size = 4096,							\
655 	.dbuf.slice_mask = BIT(DBUF_S1) | BIT(DBUF_S2) | BIT(DBUF_S3) |		\
656 		BIT(DBUF_S4),							\
657 	.has_ddi = 1,								\
658 	.has_dp_mst = 1,							\
659 	.has_dsb = 1,								\
660 	.has_fpga_dbg = 1,							\
661 	.has_hotplug = 1,							\
662 	.has_ipc = 1,								\
663 	.has_psr = 1,								\
664 	.pipe_offsets = {							\
665 		[TRANSCODER_A] = PIPE_A_OFFSET,					\
666 		[TRANSCODER_B] = PIPE_B_OFFSET,					\
667 		[TRANSCODER_C] = PIPE_C_OFFSET,					\
668 		[TRANSCODER_D] = PIPE_D_OFFSET,					\
669 		[TRANSCODER_DSI_0] = PIPE_DSI0_OFFSET,				\
670 		[TRANSCODER_DSI_1] = PIPE_DSI1_OFFSET,				\
671 	},									\
672 	.trans_offsets = {							\
673 		[TRANSCODER_A] = TRANSCODER_A_OFFSET,				\
674 		[TRANSCODER_B] = TRANSCODER_B_OFFSET,				\
675 		[TRANSCODER_C] = TRANSCODER_C_OFFSET,				\
676 		[TRANSCODER_D] = TRANSCODER_D_OFFSET,				\
677 		[TRANSCODER_DSI_0] = TRANSCODER_DSI0_OFFSET,			\
678 		[TRANSCODER_DSI_1] = TRANSCODER_DSI1_OFFSET,			\
679 	},									\
680 	TGL_CURSOR_OFFSETS,							\
681 										\
682 	.__runtime_defaults.ip.ver = 13,					\
683 	.__runtime_defaults.has_dmc = 1,					\
684 	.__runtime_defaults.has_dsc = 1,					\
685 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A),			\
686 	.__runtime_defaults.has_hdcp = 1,					\
687 	.__runtime_defaults.pipe_mask =						\
688 		BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C) | BIT(PIPE_D)
689 
690 static const struct intel_display_device_info xe_lpd_display = {
691 	XE_LPD_FEATURES,
692 	.has_cdclk_crawl = 1,
693 	.has_psr_hw_tracking = 0,
694 
695 	.__runtime_defaults.cpu_transcoder_mask =
696 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) |
697 		BIT(TRANSCODER_C) | BIT(TRANSCODER_D) |
698 		BIT(TRANSCODER_DSI_0) | BIT(TRANSCODER_DSI_1),
699 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) |
700 		BIT(PORT_TC1) | BIT(PORT_TC2) | BIT(PORT_TC3) | BIT(PORT_TC4),
701 };
702 
703 static const struct intel_display_device_info xe_hpd_display = {
704 	XE_LPD_FEATURES,
705 	.has_cdclk_squash = 1,
706 
707 	.__runtime_defaults.cpu_transcoder_mask =
708 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) |
709 		BIT(TRANSCODER_C) | BIT(TRANSCODER_D),
710 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) | BIT(PORT_C) | BIT(PORT_D_XELPD) |
711 		BIT(PORT_TC1),
712 };
713 
714 #define XE_LPDP_FEATURES							\
715 	.abox_mask = GENMASK(1, 0),						\
716 	.color = {								\
717 		.degamma_lut_size = 129, .gamma_lut_size = 1024,		\
718 		.degamma_lut_tests = DRM_COLOR_LUT_NON_DECREASING |		\
719 		DRM_COLOR_LUT_EQUAL_CHANNELS,					\
720 	},									\
721 	.dbuf.size = 4096,							\
722 	.dbuf.slice_mask = BIT(DBUF_S1) | BIT(DBUF_S2) | BIT(DBUF_S3) |		\
723 		BIT(DBUF_S4),							\
724 	.has_cdclk_crawl = 1,							\
725 	.has_cdclk_squash = 1,							\
726 	.has_ddi = 1,								\
727 	.has_dp_mst = 1,							\
728 	.has_dsb = 1,								\
729 	.has_fpga_dbg = 1,							\
730 	.has_hotplug = 1,							\
731 	.has_ipc = 1,								\
732 	.has_psr = 1,								\
733 	.pipe_offsets = {							\
734 		[TRANSCODER_A] = PIPE_A_OFFSET,					\
735 		[TRANSCODER_B] = PIPE_B_OFFSET,					\
736 		[TRANSCODER_C] = PIPE_C_OFFSET,					\
737 		[TRANSCODER_D] = PIPE_D_OFFSET,					\
738 	},									\
739 	.trans_offsets = {							\
740 		[TRANSCODER_A] = TRANSCODER_A_OFFSET,				\
741 		[TRANSCODER_B] = TRANSCODER_B_OFFSET,				\
742 		[TRANSCODER_C] = TRANSCODER_C_OFFSET,				\
743 		[TRANSCODER_D] = TRANSCODER_D_OFFSET,				\
744 	},									\
745 	TGL_CURSOR_OFFSETS,							\
746 										\
747 	.__runtime_defaults.cpu_transcoder_mask =				\
748 		BIT(TRANSCODER_A) | BIT(TRANSCODER_B) |				\
749 		BIT(TRANSCODER_C) | BIT(TRANSCODER_D),				\
750 	.__runtime_defaults.fbc_mask = BIT(INTEL_FBC_A) | BIT(INTEL_FBC_B),	\
751 	.__runtime_defaults.has_dmc = 1,					\
752 	.__runtime_defaults.has_dsc = 1,					\
753 	.__runtime_defaults.has_hdcp = 1,					\
754 	.__runtime_defaults.pipe_mask =						\
755 		BIT(PIPE_A) | BIT(PIPE_B) | BIT(PIPE_C) | BIT(PIPE_D),		\
756 	.__runtime_defaults.port_mask = BIT(PORT_A) | BIT(PORT_B) |		\
757 		BIT(PORT_TC1) | BIT(PORT_TC2) | BIT(PORT_TC3) | BIT(PORT_TC4)
758 
759 static const struct intel_display_device_info xe_lpdp_display = {
760 	XE_LPDP_FEATURES,
761 };
762 
763 static const struct intel_display_device_info xe2_lpd_display = {
764 	XE_LPDP_FEATURES,
765 
766 	.__runtime_defaults.fbc_mask =
767 		BIT(INTEL_FBC_A) | BIT(INTEL_FBC_B) |
768 		BIT(INTEL_FBC_C) | BIT(INTEL_FBC_D),
769 };
770 
771 /*
772  * Separate detection for no display cases to keep the display id array simple.
773  *
774  * IVB Q requires subvendor and subdevice matching to differentiate from IVB D
775  * GT2 server.
776  */
777 static bool has_no_display(struct pci_dev *pdev)
778 {
779 	static const struct pci_device_id ids[] = {
780 		INTEL_IVB_Q_IDS(0),
781 		{}
782 	};
783 
784 	return pci_match_id(ids, pdev);
785 }
786 
787 #undef INTEL_VGA_DEVICE
788 #define INTEL_VGA_DEVICE(id, info) { id, info }
789 
790 static const struct {
791 	u32 devid;
792 	const struct intel_display_device_info *info;
793 } intel_display_ids[] = {
794 	INTEL_I830_IDS(&i830_display),
795 	INTEL_I845G_IDS(&i845_display),
796 	INTEL_I85X_IDS(&i85x_display),
797 	INTEL_I865G_IDS(&i865g_display),
798 	INTEL_I915G_IDS(&i915g_display),
799 	INTEL_I915GM_IDS(&i915gm_display),
800 	INTEL_I945G_IDS(&i945g_display),
801 	INTEL_I945GM_IDS(&i945gm_display),
802 	INTEL_I965G_IDS(&i965g_display),
803 	INTEL_G33_IDS(&g33_display),
804 	INTEL_I965GM_IDS(&i965gm_display),
805 	INTEL_GM45_IDS(&gm45_display),
806 	INTEL_G45_IDS(&g45_display),
807 	INTEL_PINEVIEW_G_IDS(&pnv_display),
808 	INTEL_PINEVIEW_M_IDS(&pnv_display),
809 	INTEL_IRONLAKE_D_IDS(&ilk_d_display),
810 	INTEL_IRONLAKE_M_IDS(&ilk_m_display),
811 	INTEL_SNB_D_IDS(&snb_display),
812 	INTEL_SNB_M_IDS(&snb_display),
813 	INTEL_IVB_M_IDS(&ivb_display),
814 	INTEL_IVB_D_IDS(&ivb_display),
815 	INTEL_HSW_IDS(&hsw_display),
816 	INTEL_VLV_IDS(&vlv_display),
817 	INTEL_BDW_IDS(&bdw_display),
818 	INTEL_CHV_IDS(&chv_display),
819 	INTEL_SKL_IDS(&skl_display),
820 	INTEL_BXT_IDS(&bxt_display),
821 	INTEL_GLK_IDS(&glk_display),
822 	INTEL_KBL_IDS(&skl_display),
823 	INTEL_CFL_IDS(&skl_display),
824 	INTEL_ICL_11_IDS(&icl_display),
825 	INTEL_EHL_IDS(&jsl_ehl_display),
826 	INTEL_JSL_IDS(&jsl_ehl_display),
827 	INTEL_TGL_12_IDS(&tgl_display),
828 	INTEL_DG1_IDS(&dg1_display),
829 	INTEL_RKL_IDS(&rkl_display),
830 	INTEL_ADLS_IDS(&adl_s_display),
831 	INTEL_RPLS_IDS(&adl_s_display),
832 	INTEL_ADLP_IDS(&xe_lpd_display),
833 	INTEL_ADLN_IDS(&xe_lpd_display),
834 	INTEL_RPLP_IDS(&xe_lpd_display),
835 	INTEL_DG2_IDS(&xe_hpd_display),
836 
837 	/*
838 	 * Do not add any GMD_ID-based platforms to this list.  They will
839 	 * be probed automatically based on the IP version reported by
840 	 * the hardware.
841 	 */
842 };
843 
844 static const struct {
845 	u16 ver;
846 	u16 rel;
847 	const struct intel_display_device_info *display;
848 } gmdid_display_map[] = {
849 	{ 14,  0, &xe_lpdp_display },
850 	{ 20,  0, &xe2_lpd_display },
851 };
852 
853 static const struct intel_display_device_info *
854 probe_gmdid_display(struct drm_i915_private *i915, u16 *ver, u16 *rel, u16 *step)
855 {
856 	struct pci_dev *pdev = to_pci_dev(i915->drm.dev);
857 	void __iomem *addr;
858 	u32 val;
859 	int i;
860 
861 	/* The caller expects to ver, rel and step to be initialized
862 	 * here, and there's no good way to check when there was a
863 	 * failure and no_display was returned.  So initialize all these
864 	 * values here zero, to be sure.
865 	 */
866 	*ver = 0;
867 	*rel = 0;
868 	*step = 0;
869 
870 	addr = pci_iomap_range(pdev, 0, i915_mmio_reg_offset(GMD_ID_DISPLAY), sizeof(u32));
871 	if (!addr) {
872 		drm_err(&i915->drm, "Cannot map MMIO BAR to read display GMD_ID\n");
873 		return &no_display;
874 	}
875 
876 	val = ioread32(addr);
877 	pci_iounmap(pdev, addr);
878 
879 	if (val == 0) {
880 		drm_dbg_kms(&i915->drm, "Device doesn't have display\n");
881 		return &no_display;
882 	}
883 
884 	*ver = REG_FIELD_GET(GMD_ID_ARCH_MASK, val);
885 	*rel = REG_FIELD_GET(GMD_ID_RELEASE_MASK, val);
886 	*step = REG_FIELD_GET(GMD_ID_STEP, val);
887 
888 	for (i = 0; i < ARRAY_SIZE(gmdid_display_map); i++)
889 		if (*ver == gmdid_display_map[i].ver &&
890 		    *rel == gmdid_display_map[i].rel)
891 			return gmdid_display_map[i].display;
892 
893 	drm_err(&i915->drm, "Unrecognized display IP version %d.%02d; disabling display.\n",
894 		*ver, *rel);
895 	return &no_display;
896 }
897 
898 static const struct intel_display_device_info *
899 probe_display(struct drm_i915_private *i915)
900 {
901 	struct pci_dev *pdev = to_pci_dev(i915->drm.dev);
902 	int i;
903 
904 	if (has_no_display(pdev)) {
905 		drm_dbg_kms(&i915->drm, "Device doesn't have display\n");
906 		return &no_display;
907 	}
908 
909 	for (i = 0; i < ARRAY_SIZE(intel_display_ids); i++) {
910 		if (intel_display_ids[i].devid == pdev->device)
911 			return intel_display_ids[i].info;
912 	}
913 
914 	drm_dbg(&i915->drm, "No display ID found for device ID %04x; disabling display.\n",
915 		pdev->device);
916 
917 	return &no_display;
918 }
919 
920 void intel_display_device_probe(struct drm_i915_private *i915)
921 {
922 	const struct intel_display_device_info *info;
923 	u16 ver, rel, step;
924 
925 	if (HAS_GMD_ID(i915))
926 		info = probe_gmdid_display(i915, &ver, &rel, &step);
927 	else
928 		info = probe_display(i915);
929 
930 	DISPLAY_INFO(i915) = info;
931 
932 	memcpy(DISPLAY_RUNTIME_INFO(i915),
933 	       &DISPLAY_INFO(i915)->__runtime_defaults,
934 	       sizeof(*DISPLAY_RUNTIME_INFO(i915)));
935 
936 	if (HAS_GMD_ID(i915)) {
937 		DISPLAY_RUNTIME_INFO(i915)->ip.ver = ver;
938 		DISPLAY_RUNTIME_INFO(i915)->ip.rel = rel;
939 		DISPLAY_RUNTIME_INFO(i915)->ip.step = step;
940 	}
941 
942 	intel_display_params_copy(&i915->display.params);
943 }
944 
945 void intel_display_device_remove(struct drm_i915_private *i915)
946 {
947 	intel_display_params_free(&i915->display.params);
948 }
949 
950 static void __intel_display_device_info_runtime_init(struct drm_i915_private *i915)
951 {
952 	struct intel_display_runtime_info *display_runtime = DISPLAY_RUNTIME_INFO(i915);
953 	enum pipe pipe;
954 
955 	BUILD_BUG_ON(BITS_PER_TYPE(display_runtime->pipe_mask) < I915_MAX_PIPES);
956 	BUILD_BUG_ON(BITS_PER_TYPE(display_runtime->cpu_transcoder_mask) < I915_MAX_TRANSCODERS);
957 	BUILD_BUG_ON(BITS_PER_TYPE(display_runtime->port_mask) < I915_MAX_PORTS);
958 
959 	/* This covers both ULT and ULX */
960 	if (IS_HASWELL_ULT(i915) || IS_BROADWELL_ULT(i915))
961 		display_runtime->port_mask &= ~BIT(PORT_D);
962 
963 	if (IS_ICL_WITH_PORT_F(i915))
964 		display_runtime->port_mask |= BIT(PORT_F);
965 
966 	/* Wa_14011765242: adl-s A0,A1 */
967 	if (IS_ALDERLAKE_S(i915) && IS_DISPLAY_STEP(i915, STEP_A0, STEP_A2))
968 		for_each_pipe(i915, pipe)
969 			display_runtime->num_scalers[pipe] = 0;
970 	else if (DISPLAY_VER(i915) >= 11) {
971 		for_each_pipe(i915, pipe)
972 			display_runtime->num_scalers[pipe] = 2;
973 	} else if (DISPLAY_VER(i915) >= 9) {
974 		display_runtime->num_scalers[PIPE_A] = 2;
975 		display_runtime->num_scalers[PIPE_B] = 2;
976 		display_runtime->num_scalers[PIPE_C] = 1;
977 	}
978 
979 	if (DISPLAY_VER(i915) >= 13 || HAS_D12_PLANE_MINIMIZATION(i915))
980 		for_each_pipe(i915, pipe)
981 			display_runtime->num_sprites[pipe] = 4;
982 	else if (DISPLAY_VER(i915) >= 11)
983 		for_each_pipe(i915, pipe)
984 			display_runtime->num_sprites[pipe] = 6;
985 	else if (DISPLAY_VER(i915) == 10)
986 		for_each_pipe(i915, pipe)
987 			display_runtime->num_sprites[pipe] = 3;
988 	else if (IS_BROXTON(i915)) {
989 		/*
990 		 * Skylake and Broxton currently don't expose the topmost plane as its
991 		 * use is exclusive with the legacy cursor and we only want to expose
992 		 * one of those, not both. Until we can safely expose the topmost plane
993 		 * as a DRM_PLANE_TYPE_CURSOR with all the features exposed/supported,
994 		 * we don't expose the topmost plane at all to prevent ABI breakage
995 		 * down the line.
996 		 */
997 
998 		display_runtime->num_sprites[PIPE_A] = 2;
999 		display_runtime->num_sprites[PIPE_B] = 2;
1000 		display_runtime->num_sprites[PIPE_C] = 1;
1001 	} else if (IS_VALLEYVIEW(i915) || IS_CHERRYVIEW(i915)) {
1002 		for_each_pipe(i915, pipe)
1003 			display_runtime->num_sprites[pipe] = 2;
1004 	} else if (DISPLAY_VER(i915) >= 5 || IS_G4X(i915)) {
1005 		for_each_pipe(i915, pipe)
1006 			display_runtime->num_sprites[pipe] = 1;
1007 	}
1008 
1009 	if ((IS_DGFX(i915) || DISPLAY_VER(i915) >= 14) &&
1010 	    !(intel_de_read(i915, GU_CNTL_PROTECTED) & DEPRESENT)) {
1011 		drm_info(&i915->drm, "Display not present, disabling\n");
1012 		goto display_fused_off;
1013 	}
1014 
1015 	if (IS_GRAPHICS_VER(i915, 7, 8) && HAS_PCH_SPLIT(i915)) {
1016 		u32 fuse_strap = intel_de_read(i915, FUSE_STRAP);
1017 		u32 sfuse_strap = intel_de_read(i915, SFUSE_STRAP);
1018 
1019 		/*
1020 		 * SFUSE_STRAP is supposed to have a bit signalling the display
1021 		 * is fused off. Unfortunately it seems that, at least in
1022 		 * certain cases, fused off display means that PCH display
1023 		 * reads don't land anywhere. In that case, we read 0s.
1024 		 *
1025 		 * On CPT/PPT, we can detect this case as SFUSE_STRAP_FUSE_LOCK
1026 		 * should be set when taking over after the firmware.
1027 		 */
1028 		if (fuse_strap & ILK_INTERNAL_DISPLAY_DISABLE ||
1029 		    sfuse_strap & SFUSE_STRAP_DISPLAY_DISABLED ||
1030 		    (HAS_PCH_CPT(i915) &&
1031 		     !(sfuse_strap & SFUSE_STRAP_FUSE_LOCK))) {
1032 			drm_info(&i915->drm,
1033 				 "Display fused off, disabling\n");
1034 			goto display_fused_off;
1035 		} else if (fuse_strap & IVB_PIPE_C_DISABLE) {
1036 			drm_info(&i915->drm, "PipeC fused off\n");
1037 			display_runtime->pipe_mask &= ~BIT(PIPE_C);
1038 			display_runtime->cpu_transcoder_mask &= ~BIT(TRANSCODER_C);
1039 		}
1040 	} else if (DISPLAY_VER(i915) >= 9) {
1041 		u32 dfsm = intel_de_read(i915, SKL_DFSM);
1042 
1043 		if (dfsm & SKL_DFSM_PIPE_A_DISABLE) {
1044 			display_runtime->pipe_mask &= ~BIT(PIPE_A);
1045 			display_runtime->cpu_transcoder_mask &= ~BIT(TRANSCODER_A);
1046 			display_runtime->fbc_mask &= ~BIT(INTEL_FBC_A);
1047 		}
1048 		if (dfsm & SKL_DFSM_PIPE_B_DISABLE) {
1049 			display_runtime->pipe_mask &= ~BIT(PIPE_B);
1050 			display_runtime->cpu_transcoder_mask &= ~BIT(TRANSCODER_B);
1051 			display_runtime->fbc_mask &= ~BIT(INTEL_FBC_B);
1052 		}
1053 		if (dfsm & SKL_DFSM_PIPE_C_DISABLE) {
1054 			display_runtime->pipe_mask &= ~BIT(PIPE_C);
1055 			display_runtime->cpu_transcoder_mask &= ~BIT(TRANSCODER_C);
1056 			display_runtime->fbc_mask &= ~BIT(INTEL_FBC_C);
1057 		}
1058 
1059 		if (DISPLAY_VER(i915) >= 12 &&
1060 		    (dfsm & TGL_DFSM_PIPE_D_DISABLE)) {
1061 			display_runtime->pipe_mask &= ~BIT(PIPE_D);
1062 			display_runtime->cpu_transcoder_mask &= ~BIT(TRANSCODER_D);
1063 			display_runtime->fbc_mask &= ~BIT(INTEL_FBC_D);
1064 		}
1065 
1066 		if (!display_runtime->pipe_mask)
1067 			goto display_fused_off;
1068 
1069 		if (dfsm & SKL_DFSM_DISPLAY_HDCP_DISABLE)
1070 			display_runtime->has_hdcp = 0;
1071 
1072 		if (dfsm & SKL_DFSM_DISPLAY_PM_DISABLE)
1073 			display_runtime->fbc_mask = 0;
1074 
1075 		if (DISPLAY_VER(i915) >= 11 && (dfsm & ICL_DFSM_DMC_DISABLE))
1076 			display_runtime->has_dmc = 0;
1077 
1078 		if (IS_DISPLAY_VER(i915, 10, 12) &&
1079 		    (dfsm & GLK_DFSM_DISPLAY_DSC_DISABLE))
1080 			display_runtime->has_dsc = 0;
1081 	}
1082 
1083 	if (DISPLAY_VER(i915) >= 20) {
1084 		u32 cap = intel_de_read(i915, XE2LPD_DE_CAP);
1085 
1086 		if (REG_FIELD_GET(XE2LPD_DE_CAP_DSC_MASK, cap) ==
1087 		    XE2LPD_DE_CAP_DSC_REMOVED)
1088 			display_runtime->has_dsc = 0;
1089 
1090 		if (REG_FIELD_GET(XE2LPD_DE_CAP_SCALER_MASK, cap) ==
1091 		    XE2LPD_DE_CAP_SCALER_SINGLE) {
1092 			for_each_pipe(i915, pipe)
1093 				if (display_runtime->num_scalers[pipe])
1094 					display_runtime->num_scalers[pipe] = 1;
1095 		}
1096 	}
1097 
1098 	return;
1099 
1100 display_fused_off:
1101 	memset(display_runtime, 0, sizeof(*display_runtime));
1102 }
1103 
1104 void intel_display_device_info_runtime_init(struct drm_i915_private *i915)
1105 {
1106 	if (HAS_DISPLAY(i915))
1107 		__intel_display_device_info_runtime_init(i915);
1108 
1109 	/* Display may have been disabled by runtime init */
1110 	if (!HAS_DISPLAY(i915)) {
1111 		i915->drm.driver_features &= ~(DRIVER_MODESET | DRIVER_ATOMIC);
1112 		i915->display.info.__device_info = &no_display;
1113 	}
1114 
1115 	/* Disable nuclear pageflip by default on pre-g4x */
1116 	if (!i915->display.params.nuclear_pageflip &&
1117 	    DISPLAY_VER(i915) < 5 && !IS_G4X(i915))
1118 		i915->drm.driver_features &= ~DRIVER_ATOMIC;
1119 }
1120 
1121 void intel_display_device_info_print(const struct intel_display_device_info *info,
1122 				     const struct intel_display_runtime_info *runtime,
1123 				     struct drm_printer *p)
1124 {
1125 	if (runtime->ip.rel)
1126 		drm_printf(p, "display version: %u.%02u\n",
1127 			   runtime->ip.ver,
1128 			   runtime->ip.rel);
1129 	else
1130 		drm_printf(p, "display version: %u\n",
1131 			   runtime->ip.ver);
1132 
1133 #define PRINT_FLAG(name) drm_printf(p, "%s: %s\n", #name, str_yes_no(info->name))
1134 	DEV_INFO_DISPLAY_FOR_EACH_FLAG(PRINT_FLAG);
1135 #undef PRINT_FLAG
1136 
1137 	drm_printf(p, "has_hdcp: %s\n", str_yes_no(runtime->has_hdcp));
1138 	drm_printf(p, "has_dmc: %s\n", str_yes_no(runtime->has_dmc));
1139 	drm_printf(p, "has_dsc: %s\n", str_yes_no(runtime->has_dsc));
1140 }
1141 
1142 /*
1143  * Assuming the device has display hardware, should it be enabled?
1144  *
1145  * It's an error to call this function if the device does not have display
1146  * hardware.
1147  *
1148  * Disabling display means taking over the display hardware, putting it to
1149  * sleep, and preventing connectors from being connected via any means.
1150  */
1151 bool intel_display_device_enabled(struct drm_i915_private *i915)
1152 {
1153 	/* Only valid when HAS_DISPLAY() is true */
1154 	drm_WARN_ON(&i915->drm, !HAS_DISPLAY(i915));
1155 
1156 	return !i915->display.params.disable_display &&
1157 		!intel_opregion_headless_sku(i915);
1158 }
1159