xref: /rk3399_rockchip-uboot/include/edid.h (revision 21016d27c500da4326bdc59cd3505fcd85d236db)
1d46b5f7dSTom Wai-Hong Tam /*
2d46b5f7dSTom Wai-Hong Tam  * Copyright (c) 2012 The Chromium OS Authors.
3d46b5f7dSTom Wai-Hong Tam  *
4d46b5f7dSTom Wai-Hong Tam  * (C) Copyright 2010
5d46b5f7dSTom Wai-Hong Tam  * Petr Stetiar <ynezz@true.cz>
6d46b5f7dSTom Wai-Hong Tam  *
71a459660SWolfgang Denk  * SPDX-License-Identifier:	GPL-2.0+
8d46b5f7dSTom Wai-Hong Tam  *
9d46b5f7dSTom Wai-Hong Tam  * Contains stolen code from ddcprobe project which is:
10d46b5f7dSTom Wai-Hong Tam  * Copyright (C) Nalin Dahyabhai <bigfun@pobox.com>
11*21016d27SAlgea Cao  * (C) Copyright 2008-2017 Fuzhou Rockchip Electronics Co., Ltd
12d46b5f7dSTom Wai-Hong Tam  */
13d46b5f7dSTom Wai-Hong Tam 
14d46b5f7dSTom Wai-Hong Tam #ifndef __EDID_H_
15d46b5f7dSTom Wai-Hong Tam #define __EDID_H_
16d46b5f7dSTom Wai-Hong Tam 
17*21016d27SAlgea Cao #include <div64.h>
18d46b5f7dSTom Wai-Hong Tam #include <linux/types.h>
19d46b5f7dSTom Wai-Hong Tam 
2000cf1167SSimon Glass /* Size of the EDID data */
2100cf1167SSimon Glass #define EDID_SIZE	128
222dcf1433SSimon Glass #define EDID_EXT_SIZE	256
23*21016d27SAlgea Cao #define MODE_LEN	120
24*21016d27SAlgea Cao 
25*21016d27SAlgea Cao #define CEA_EXT	    0x02
26*21016d27SAlgea Cao #define VTB_EXT	    0x10
27*21016d27SAlgea Cao #define DI_EXT	    0x40
28*21016d27SAlgea Cao #define LS_EXT	    0x50
29*21016d27SAlgea Cao #define MI_EXT	    0x60
30*21016d27SAlgea Cao #define DISPLAYID_EXT 0x70
31*21016d27SAlgea Cao 
32*21016d27SAlgea Cao #define EDID_TIMING_ASPECT_SHIFT 6
33*21016d27SAlgea Cao #define EDID_TIMING_ASPECT_MASK  (0x3 << EDID_TIMING_ASPECT_SHIFT)
34*21016d27SAlgea Cao 
35*21016d27SAlgea Cao /* need to add 60 */
36*21016d27SAlgea Cao #define EDID_TIMING_VFREQ_SHIFT  0
37*21016d27SAlgea Cao #define EDID_TIMING_VFREQ_MASK   (0x3f << EDID_TIMING_VFREQ_SHIFT)
3800cf1167SSimon Glass 
3943c6bdd0SJernej Skrabec /* OUI of HDMI vendor specific data block */
4043c6bdd0SJernej Skrabec #define HDMI_IEEE_OUI 0x000c03
4143c6bdd0SJernej Skrabec 
42*21016d27SAlgea Cao /* drm mode 4k and 3d */
43*21016d27SAlgea Cao #define DRM_MODE_FLAG_420_MASK			(0x03 << 23)
44*21016d27SAlgea Cao #define  DRM_MODE_FLAG_420			BIT(23)
45*21016d27SAlgea Cao #define  DRM_MODE_FLAG_420_ONLY			BIT(24)
46*21016d27SAlgea Cao 
47*21016d27SAlgea Cao #define DRM_MODE_FLAG_3D_MASK			(0x1f << 14)
48*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_NONE			(0 << 14)
49*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_FRAME_PACKING		BIT(14)
50*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_FIELD_ALTERNATIVE	(2 << 14)
51*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_LINE_ALTERNATIVE	(3 << 14)
52*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_SIDE_BY_SIDE_FULL	(4 << 14)
53*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_L_DEPTH		(5 << 14)
54*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_L_DEPTH_GFX_GFX_DEPTH	(6 << 14)
55*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_TOP_AND_BOTTOM	(7 << 14)
56*21016d27SAlgea Cao #define  DRM_MODE_FLAG_3D_SIDE_BY_SIDE_HALF	(8 << 14)
57*21016d27SAlgea Cao 
58*21016d27SAlgea Cao #define BITS_PER_BYTE         8
59*21016d27SAlgea Cao #define BITS_TO_LONGS(nr)     DIV_ROUND_UP(nr, BITS_PER_BYTE * sizeof(long))
60d46b5f7dSTom Wai-Hong Tam #define GET_BIT(_x, _pos) \
61d46b5f7dSTom Wai-Hong Tam 	(((_x) >> (_pos)) & 1)
62d46b5f7dSTom Wai-Hong Tam #define GET_BITS(_x, _pos_msb, _pos_lsb) \
63d46b5f7dSTom Wai-Hong Tam 	(((_x) >> (_pos_lsb)) & ((1 << ((_pos_msb) - (_pos_lsb) + 1)) - 1))
64*21016d27SAlgea Cao #define DRM_MODE(c, hd, hss, hse, ht, vd, vss, vse, vt, vs, f) \
65*21016d27SAlgea Cao 	.clock = (c), \
66*21016d27SAlgea Cao 	.hdisplay = (hd), .hsync_start = (hss), .hsync_end = (hse), \
67*21016d27SAlgea Cao 	.htotal = (ht), .vdisplay = (vd), \
68*21016d27SAlgea Cao 	.vsync_start = (vss), .vsync_end = (vse), .vtotal = (vt), \
69*21016d27SAlgea Cao 	.vscan = (vs), .flags = (f)
70d46b5f7dSTom Wai-Hong Tam 
71d46b5f7dSTom Wai-Hong Tam /* Aspect ratios used in EDID info. */
72d46b5f7dSTom Wai-Hong Tam enum edid_aspect {
73d46b5f7dSTom Wai-Hong Tam 	ASPECT_625 = 0,
74d46b5f7dSTom Wai-Hong Tam 	ASPECT_75,
75d46b5f7dSTom Wai-Hong Tam 	ASPECT_8,
76d46b5f7dSTom Wai-Hong Tam 	ASPECT_5625,
77d46b5f7dSTom Wai-Hong Tam };
78d46b5f7dSTom Wai-Hong Tam 
79*21016d27SAlgea Cao struct est_timings {
80*21016d27SAlgea Cao 	u8 t1;
81*21016d27SAlgea Cao 	u8 t2;
82*21016d27SAlgea Cao 	u8 mfg_rsvd;
83*21016d27SAlgea Cao } __packed;
84*21016d27SAlgea Cao 
85*21016d27SAlgea Cao /* 00=16:10, 01=4:3, 10=5:4, 11=16:9 */
86*21016d27SAlgea Cao #define EDID_TIMING_ASPECT_SHIFT 6
87*21016d27SAlgea Cao #define EDID_TIMING_ASPECT_MASK  (0x3 << EDID_TIMING_ASPECT_SHIFT)
88*21016d27SAlgea Cao 
89*21016d27SAlgea Cao /* need to add 60 */
90*21016d27SAlgea Cao #define EDID_TIMING_VFREQ_SHIFT  0
91*21016d27SAlgea Cao #define EDID_TIMING_VFREQ_MASK   (0x3f << EDID_TIMING_VFREQ_SHIFT)
92*21016d27SAlgea Cao 
93*21016d27SAlgea Cao struct std_timing {
94*21016d27SAlgea Cao 	u8 hsize; /* need to multiply by 8 then add 248 */
95*21016d27SAlgea Cao 	u8 vfreq_aspect;
96*21016d27SAlgea Cao } __packed;
97*21016d27SAlgea Cao 
98*21016d27SAlgea Cao struct detailed_pixel_timing {
99*21016d27SAlgea Cao 	u8 hactive_lo;
100*21016d27SAlgea Cao 	u8 hblank_lo;
101*21016d27SAlgea Cao 	u8 hactive_hblank_hi;
102*21016d27SAlgea Cao 	u8 vactive_lo;
103*21016d27SAlgea Cao 	u8 vblank_lo;
104*21016d27SAlgea Cao 	u8 vactive_vblank_hi;
105*21016d27SAlgea Cao 	u8 hsync_offset_lo;
106*21016d27SAlgea Cao 	u8 hsync_pulse_width_lo;
107*21016d27SAlgea Cao 	u8 vsync_offset_pulse_width_lo;
108*21016d27SAlgea Cao 	u8 hsync_vsync_offset_pulse_width_hi;
109*21016d27SAlgea Cao 	u8 width_mm_lo;
110*21016d27SAlgea Cao 	u8 height_mm_lo;
111*21016d27SAlgea Cao 	u8 width_height_mm_hi;
112*21016d27SAlgea Cao 	u8 hborder;
113*21016d27SAlgea Cao 	u8 vborder;
114*21016d27SAlgea Cao 	u8 misc;
115*21016d27SAlgea Cao } __packed;
116*21016d27SAlgea Cao 
117*21016d27SAlgea Cao /* If it's not pixel timing, it'll be one of the below */
118*21016d27SAlgea Cao struct detailed_data_string {
119*21016d27SAlgea Cao 	u8 str[13];
120*21016d27SAlgea Cao } __packed;
121*21016d27SAlgea Cao 
122*21016d27SAlgea Cao struct detailed_data_monitor_range {
123*21016d27SAlgea Cao 	u8 min_vfreq;
124*21016d27SAlgea Cao 	u8 max_vfreq;
125*21016d27SAlgea Cao 	u8 min_hfreq_khz;
126*21016d27SAlgea Cao 	u8 max_hfreq_khz;
127*21016d27SAlgea Cao 	u8 pixel_clock_mhz; /* need to multiply by 10 */
128*21016d27SAlgea Cao 	u8 flags;
129*21016d27SAlgea Cao 	union {
130*21016d27SAlgea Cao 		struct {
131*21016d27SAlgea Cao 			u8 reserved;
132*21016d27SAlgea Cao 			u8 hfreq_start_khz; /* need to multiply by 2 */
133*21016d27SAlgea Cao 			u8 c; /* need to divide by 2 */
134*21016d27SAlgea Cao 			__le16 m;
135*21016d27SAlgea Cao 			u8 k;
136*21016d27SAlgea Cao 			u8 j; /* need to divide by 2 */
137*21016d27SAlgea Cao 		} __packed gtf2;
138*21016d27SAlgea Cao 		struct {
139*21016d27SAlgea Cao 			u8 version;
140*21016d27SAlgea Cao 			u8 data1; /* high 6 bits: extra clock resolution */
141*21016d27SAlgea Cao 			u8 data2; /* plus low 2 of above: max hactive */
142*21016d27SAlgea Cao 			u8 supported_aspects;
143*21016d27SAlgea Cao 			u8 flags; /* preferred aspect and blanking support */
144*21016d27SAlgea Cao 			u8 supported_scalings;
145*21016d27SAlgea Cao 			u8 preferred_refresh;
146*21016d27SAlgea Cao 		} __packed cvt;
147*21016d27SAlgea Cao 	} formula;
148*21016d27SAlgea Cao } __packed;
149*21016d27SAlgea Cao 
150*21016d27SAlgea Cao struct detailed_data_wpindex {
151*21016d27SAlgea Cao 	u8 white_yx_lo; /* Lower 2 bits each */
152*21016d27SAlgea Cao 	u8 white_x_hi;
153*21016d27SAlgea Cao 	u8 white_y_hi;
154*21016d27SAlgea Cao 	u8 gamma; /* need to divide by 100 then add 1 */
155*21016d27SAlgea Cao } __packed;
156*21016d27SAlgea Cao 
157*21016d27SAlgea Cao struct detailed_data_color_point {
158*21016d27SAlgea Cao 	u8 windex1;
159*21016d27SAlgea Cao 	u8 wpindex1[3];
160*21016d27SAlgea Cao 	u8 windex2;
161*21016d27SAlgea Cao 	u8 wpindex2[3];
162*21016d27SAlgea Cao } __packed;
163*21016d27SAlgea Cao 
164*21016d27SAlgea Cao struct cvt_timing {
165*21016d27SAlgea Cao 	u8 code[3];
166*21016d27SAlgea Cao } __packed;
167*21016d27SAlgea Cao 
168*21016d27SAlgea Cao struct detailed_non_pixel {
169*21016d27SAlgea Cao 	u8 pad1;
170*21016d27SAlgea Cao 	u8 type; /* ff=serial, fe=string, fd=monitor range, fc=monitor name
171*21016d27SAlgea Cao 		  * fb=color point data, fa=standard timing data,
172*21016d27SAlgea Cao 		  * f9=undefined, f8=mfg. reserved
173*21016d27SAlgea Cao 		  */
174*21016d27SAlgea Cao 	u8 pad2;
175*21016d27SAlgea Cao 	union {
176*21016d27SAlgea Cao 		struct detailed_data_string str;
177*21016d27SAlgea Cao 		struct detailed_data_monitor_range range;
178*21016d27SAlgea Cao 		struct detailed_data_wpindex color;
179*21016d27SAlgea Cao 		struct std_timing timings[6];
180*21016d27SAlgea Cao 		struct cvt_timing cvt[4];
181*21016d27SAlgea Cao 	} data;
182*21016d27SAlgea Cao } __packed;
183*21016d27SAlgea Cao 
184*21016d27SAlgea Cao #define EDID_DETAIL_EST_TIMINGS 0xf7
185*21016d27SAlgea Cao #define EDID_DETAIL_CVT_3BYTE 0xf8
186*21016d27SAlgea Cao #define EDID_DETAIL_COLOR_MGMT_DATA 0xf9
187*21016d27SAlgea Cao #define EDID_DETAIL_STD_MODES 0xfa
188*21016d27SAlgea Cao #define EDID_DETAIL_MONITOR_CPDATA 0xfb
189*21016d27SAlgea Cao #define EDID_DETAIL_MONITOR_NAME 0xfc
190*21016d27SAlgea Cao #define EDID_DETAIL_MONITOR_RANGE 0xfd
191*21016d27SAlgea Cao #define EDID_DETAIL_MONITOR_STRING 0xfe
192*21016d27SAlgea Cao #define EDID_DETAIL_MONITOR_SERIAL 0xff
193*21016d27SAlgea Cao 
194*21016d27SAlgea Cao struct detailed_timing {
195*21016d27SAlgea Cao 	__le16 pixel_clock; /* need to multiply by 10 KHz */
196*21016d27SAlgea Cao 	union {
197*21016d27SAlgea Cao 		struct detailed_pixel_timing pixel_data;
198*21016d27SAlgea Cao 		struct detailed_non_pixel other_data;
199*21016d27SAlgea Cao 	} data;
200*21016d27SAlgea Cao } __packed;
201*21016d27SAlgea Cao 
202d46b5f7dSTom Wai-Hong Tam /* Detailed timing information used in EDID v1.x */
203d46b5f7dSTom Wai-Hong Tam struct edid_detailed_timing {
204d46b5f7dSTom Wai-Hong Tam 	unsigned char pixel_clock[2];
205d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_PIXEL_CLOCK(_x) \
206d46b5f7dSTom Wai-Hong Tam 	(((((uint32_t)(_x).pixel_clock[1]) << 8) + \
207d46b5f7dSTom Wai-Hong Tam 	 (_x).pixel_clock[0]) * 10000)
208d46b5f7dSTom Wai-Hong Tam 	unsigned char horizontal_active;
209d46b5f7dSTom Wai-Hong Tam 	unsigned char horizontal_blanking;
210d46b5f7dSTom Wai-Hong Tam 	unsigned char horizontal_active_blanking_hi;
211d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_HORIZONTAL_ACTIVE(_x) \
212d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).horizontal_active_blanking_hi, 7, 4) << 8) + \
213d46b5f7dSTom Wai-Hong Tam 	 (_x).horizontal_active)
214d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_HORIZONTAL_BLANKING(_x) \
215d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).horizontal_active_blanking_hi, 3, 0) << 8) + \
216d46b5f7dSTom Wai-Hong Tam 	 (_x).horizontal_blanking)
217d46b5f7dSTom Wai-Hong Tam 	unsigned char vertical_active;
218d46b5f7dSTom Wai-Hong Tam 	unsigned char vertical_blanking;
219d46b5f7dSTom Wai-Hong Tam 	unsigned char vertical_active_blanking_hi;
220d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_VERTICAL_ACTIVE(_x) \
221d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).vertical_active_blanking_hi, 7, 4) << 8) + \
222d46b5f7dSTom Wai-Hong Tam 	 (_x).vertical_active)
223d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_VERTICAL_BLANKING(_x) \
224d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).vertical_active_blanking_hi, 3, 0) << 8) + \
225d46b5f7dSTom Wai-Hong Tam 	 (_x).vertical_blanking)
226d46b5f7dSTom Wai-Hong Tam 	unsigned char hsync_offset;
227d46b5f7dSTom Wai-Hong Tam 	unsigned char hsync_pulse_width;
2281dc793ddSChristian Gmeiner 	unsigned char vsync_offset_pulse_width;
229d46b5f7dSTom Wai-Hong Tam 	unsigned char hsync_vsync_offset_pulse_width_hi;
230d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_HSYNC_OFFSET(_x) \
231d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).hsync_vsync_offset_pulse_width_hi, 7, 6) << 8) + \
232d46b5f7dSTom Wai-Hong Tam 	 (_x).hsync_offset)
233d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_HSYNC_PULSE_WIDTH(_x) \
234d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).hsync_vsync_offset_pulse_width_hi, 5, 4) << 8) + \
235d46b5f7dSTom Wai-Hong Tam 	 (_x).hsync_pulse_width)
236d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_VSYNC_OFFSET(_x) \
237d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).hsync_vsync_offset_pulse_width_hi, 3, 2) << 4) + \
238d46b5f7dSTom Wai-Hong Tam 	 GET_BITS((_x).vsync_offset_pulse_width, 7, 4))
239d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_VSYNC_PULSE_WIDTH(_x) \
240d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).hsync_vsync_offset_pulse_width_hi, 1, 0) << 4) + \
241d46b5f7dSTom Wai-Hong Tam 	 GET_BITS((_x).vsync_offset_pulse_width, 3, 0))
242d46b5f7dSTom Wai-Hong Tam 	unsigned char himage_size;
243d46b5f7dSTom Wai-Hong Tam 	unsigned char vimage_size;
244d46b5f7dSTom Wai-Hong Tam 	unsigned char himage_vimage_size_hi;
245d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_HIMAGE_SIZE(_x) \
246d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).himage_vimage_size_hi, 7, 4) << 8) + (_x).himage_size)
247d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_VIMAGE_SIZE(_x) \
248d46b5f7dSTom Wai-Hong Tam 	((GET_BITS((_x).himage_vimage_size_hi, 3, 0) << 8) + (_x).vimage_size)
249d46b5f7dSTom Wai-Hong Tam 	unsigned char hborder;
250d46b5f7dSTom Wai-Hong Tam 	unsigned char vborder;
251d46b5f7dSTom Wai-Hong Tam 	unsigned char flags;
252d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_FLAG_INTERLACED(_x) \
253d46b5f7dSTom Wai-Hong Tam 	GET_BIT((_x).flags, 7)
254d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_FLAG_STEREO(_x) \
255d46b5f7dSTom Wai-Hong Tam 	GET_BITS((_x).flags, 6, 5)
256d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_FLAG_DIGITAL_COMPOSITE(_x) \
257d46b5f7dSTom Wai-Hong Tam 	GET_BITS((_x).flags, 4, 3)
258d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_FLAG_POLARITY(_x) \
259d46b5f7dSTom Wai-Hong Tam 	GET_BITS((_x).flags, 2, 1)
260b7ce12ddSHans de Goede #define EDID_DETAILED_TIMING_FLAG_VSYNC_POLARITY(_x) \
261b7ce12ddSHans de Goede 	GET_BIT((_x).flags, 2)
262b7ce12ddSHans de Goede #define EDID_DETAILED_TIMING_FLAG_HSYNC_POLARITY(_x) \
263b7ce12ddSHans de Goede 	GET_BIT((_x).flags, 1)
264d46b5f7dSTom Wai-Hong Tam #define EDID_DETAILED_TIMING_FLAG_INTERLEAVED(_x) \
265d46b5f7dSTom Wai-Hong Tam 	GET_BIT((_x).flags, 0)
266d46b5f7dSTom Wai-Hong Tam } __attribute__ ((__packed__));
267d46b5f7dSTom Wai-Hong Tam 
268d46b5f7dSTom Wai-Hong Tam enum edid_monitor_descriptor_types {
269d46b5f7dSTom Wai-Hong Tam 	EDID_MONITOR_DESCRIPTOR_SERIAL = 0xff,
270d46b5f7dSTom Wai-Hong Tam 	EDID_MONITOR_DESCRIPTOR_ASCII = 0xfe,
271d46b5f7dSTom Wai-Hong Tam 	EDID_MONITOR_DESCRIPTOR_RANGE = 0xfd,
272d46b5f7dSTom Wai-Hong Tam 	EDID_MONITOR_DESCRIPTOR_NAME = 0xfc,
273d46b5f7dSTom Wai-Hong Tam };
274d46b5f7dSTom Wai-Hong Tam 
275d46b5f7dSTom Wai-Hong Tam struct edid_monitor_descriptor {
276d46b5f7dSTom Wai-Hong Tam 	uint16_t zero_flag_1;
277d46b5f7dSTom Wai-Hong Tam 	unsigned char zero_flag_2;
278d46b5f7dSTom Wai-Hong Tam 	unsigned char type;
279d46b5f7dSTom Wai-Hong Tam 	unsigned char zero_flag_3;
280d46b5f7dSTom Wai-Hong Tam 	union {
281d46b5f7dSTom Wai-Hong Tam 		char string[13];
282d46b5f7dSTom Wai-Hong Tam 		struct {
283d46b5f7dSTom Wai-Hong Tam 			unsigned char vertical_min;
284d46b5f7dSTom Wai-Hong Tam 			unsigned char vertical_max;
285d46b5f7dSTom Wai-Hong Tam 			unsigned char horizontal_min;
286d46b5f7dSTom Wai-Hong Tam 			unsigned char horizontal_max;
287d46b5f7dSTom Wai-Hong Tam 			unsigned char pixel_clock_max;
288d46b5f7dSTom Wai-Hong Tam 			unsigned char gtf_data[8];
289d46b5f7dSTom Wai-Hong Tam 		} range_data;
290d46b5f7dSTom Wai-Hong Tam 	} data;
291d46b5f7dSTom Wai-Hong Tam } __attribute__ ((__packed__));
292d46b5f7dSTom Wai-Hong Tam 
293*21016d27SAlgea Cao #define DRM_EDID_INPUT_SERRATION_VSYNC (1 << 0)
294*21016d27SAlgea Cao #define DRM_EDID_INPUT_SYNC_ON_GREEN   (1 << 1)
295*21016d27SAlgea Cao #define DRM_EDID_INPUT_COMPOSITE_SYNC  (1 << 2)
296*21016d27SAlgea Cao #define DRM_EDID_INPUT_SEPARATE_SYNCS  (1 << 3)
297*21016d27SAlgea Cao #define DRM_EDID_INPUT_BLANK_TO_BLACK  (1 << 4)
298*21016d27SAlgea Cao #define DRM_EDID_INPUT_VIDEO_LEVEL     (3 << 5)
299*21016d27SAlgea Cao #define DRM_EDID_INPUT_DIGITAL         (1 << 7)
300*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_MASK    (7 << 4)
301*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_UNDEF   (0 << 4)
302*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_6       (1 << 4)
303*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_8       (2 << 4)
304*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_10      (3 << 4)
305*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_12      (4 << 4)
306*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_14      (5 << 4)
307*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_16      (6 << 4)
308*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_DEPTH_RSVD    (7 << 4)
309*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_TYPE_UNDEF    (0)
310*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_TYPE_DVI      (1)
311*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_TYPE_HDMI_A   (2)
312*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_TYPE_HDMI_B   (3)
313*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_TYPE_MDDI     (4)
314*21016d27SAlgea Cao #define DRM_EDID_DIGITAL_TYPE_DP       (5)
315*21016d27SAlgea Cao 
316*21016d27SAlgea Cao #define DRM_EDID_FEATURE_DEFAULT_GTF      (1 << 0)
317*21016d27SAlgea Cao #define DRM_EDID_FEATURE_PREFERRED_TIMING (1 << 1)
318*21016d27SAlgea Cao #define DRM_EDID_FEATURE_STANDARD_COLOR   (1 << 2)
319*21016d27SAlgea Cao /* If analog */
320*21016d27SAlgea Cao #define DRM_EDID_FEATURE_DISPLAY_TYPE     (3 << 3) /* 00=mono, 01=rgb, 10=non-rgb, 11=unknown */
321*21016d27SAlgea Cao /* If digital */
322*21016d27SAlgea Cao #define DRM_EDID_FEATURE_COLOR_MASK	  (3 << 3)
323*21016d27SAlgea Cao #define DRM_EDID_FEATURE_RGB		  (0 << 3)
324*21016d27SAlgea Cao #define DRM_EDID_FEATURE_RGB_YCRCB444	  (1 << 3)
325*21016d27SAlgea Cao #define DRM_EDID_FEATURE_RGB_YCRCB422	  (2 << 3)
326*21016d27SAlgea Cao #define DRM_EDID_FEATURE_RGB_YCRCB	  (3 << 3) /* both 4:4:4 and 4:2:2 */
327*21016d27SAlgea Cao 
328*21016d27SAlgea Cao #define DRM_EDID_FEATURE_PM_ACTIVE_OFF    (1 << 5)
329*21016d27SAlgea Cao #define DRM_EDID_FEATURE_PM_SUSPEND       (1 << 6)
330*21016d27SAlgea Cao #define DRM_EDID_FEATURE_PM_STANDBY       (1 << 7)
331*21016d27SAlgea Cao 
332*21016d27SAlgea Cao #define DRM_EDID_HDMI_DC_48               (1 << 6)
333*21016d27SAlgea Cao #define DRM_EDID_HDMI_DC_36               (1 << 5)
334*21016d27SAlgea Cao #define DRM_EDID_HDMI_DC_30               (1 << 4)
335*21016d27SAlgea Cao #define DRM_EDID_HDMI_DC_Y444             (1 << 3)
336*21016d27SAlgea Cao 
337*21016d27SAlgea Cao /* YCBCR 420 deep color modes */
338*21016d27SAlgea Cao #define DRM_EDID_YCBCR420_DC_48		  (1 << 2)
339*21016d27SAlgea Cao #define DRM_EDID_YCBCR420_DC_36		  (1 << 1)
340*21016d27SAlgea Cao #define DRM_EDID_YCBCR420_DC_30		  (1 << 0)
341*21016d27SAlgea Cao #define DRM_EDID_YCBCR420_DC_MASK (DRM_EDID_YCBCR420_DC_48 | \
342*21016d27SAlgea Cao 				    DRM_EDID_YCBCR420_DC_36 | \
343*21016d27SAlgea Cao 				    DRM_EDID_YCBCR420_DC_30)
344*21016d27SAlgea Cao 
345d46b5f7dSTom Wai-Hong Tam struct edid1_info {
346d46b5f7dSTom Wai-Hong Tam 	unsigned char header[8];
347d46b5f7dSTom Wai-Hong Tam 	unsigned char manufacturer_name[2];
348d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_MANUFACTURER_NAME_ZERO(_x) \
349d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).manufacturer_name[0]), 7)
350d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_MANUFACTURER_NAME_CHAR1(_x) \
351d46b5f7dSTom Wai-Hong Tam 	GET_BITS(((_x).manufacturer_name[0]), 6, 2)
352d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_MANUFACTURER_NAME_CHAR2(_x) \
353d46b5f7dSTom Wai-Hong Tam 	((GET_BITS(((_x).manufacturer_name[0]), 1, 0) << 3) + \
354d46b5f7dSTom Wai-Hong Tam 	 GET_BITS(((_x).manufacturer_name[1]), 7, 5))
355d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_MANUFACTURER_NAME_CHAR3(_x) \
356d46b5f7dSTom Wai-Hong Tam 	GET_BITS(((_x).manufacturer_name[1]), 4, 0)
357d46b5f7dSTom Wai-Hong Tam 	unsigned char product_code[2];
358d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_PRODUCT_CODE(_x) \
359d46b5f7dSTom Wai-Hong Tam 	(((uint16_t)(_x).product_code[1] << 8) + (_x).product_code[0])
360d46b5f7dSTom Wai-Hong Tam 	unsigned char serial_number[4];
361d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_SERIAL_NUMBER(_x) \
362d46b5f7dSTom Wai-Hong Tam 	(((uint32_t)(_x).serial_number[3] << 24) + \
363d46b5f7dSTom Wai-Hong Tam 	 ((_x).serial_number[2] << 16) + ((_x).serial_number[1] << 8) + \
364d46b5f7dSTom Wai-Hong Tam 	 (_x).serial_number[0])
365d46b5f7dSTom Wai-Hong Tam 	unsigned char week;
366d46b5f7dSTom Wai-Hong Tam 	unsigned char year;
367d46b5f7dSTom Wai-Hong Tam 	unsigned char version;
368d46b5f7dSTom Wai-Hong Tam 	unsigned char revision;
369d46b5f7dSTom Wai-Hong Tam 	unsigned char video_input_definition;
370d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_VIDEO_INPUT_DIGITAL(_x) \
371d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).video_input_definition), 7)
372d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_VIDEO_INPUT_VOLTAGE_LEVEL(_x) \
373d46b5f7dSTom Wai-Hong Tam 	GET_BITS(((_x).video_input_definition), 6, 5)
374d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_VIDEO_INPUT_BLANK_TO_BLACK(_x) \
375d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).video_input_definition), 4)
376d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_VIDEO_INPUT_SEPARATE_SYNC(_x) \
377d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).video_input_definition), 3)
378d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_VIDEO_INPUT_COMPOSITE_SYNC(_x) \
379d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).video_input_definition), 2)
380d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_VIDEO_INPUT_SYNC_ON_GREEN(_x) \
381d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).video_input_definition), 1)
382d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_VIDEO_INPUT_SERRATION_V(_x) \
383d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).video_input_definition), 0)
384d46b5f7dSTom Wai-Hong Tam 	unsigned char max_size_horizontal;
385d46b5f7dSTom Wai-Hong Tam 	unsigned char max_size_vertical;
386d46b5f7dSTom Wai-Hong Tam 	unsigned char gamma;
387d46b5f7dSTom Wai-Hong Tam 	unsigned char feature_support;
388d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_FEATURE_STANDBY(_x) \
389d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).feature_support), 7)
390d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_FEATURE_SUSPEND(_x) \
391d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).feature_support), 6)
392d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_FEATURE_ACTIVE_OFF(_x) \
393d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).feature_support), 5)
394d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_FEATURE_DISPLAY_TYPE(_x) \
395d46b5f7dSTom Wai-Hong Tam 	GET_BITS(((_x).feature_support), 4, 3)
396d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_FEATURE_RGB(_x) \
397d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).feature_support), 2)
398d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_FEATURE_PREFERRED_TIMING_MODE(_x) \
399d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).feature_support), 1)
400d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_FEATURE_DEFAULT_GTF_SUPPORT(_x) \
401d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).feature_support), 0)
402d46b5f7dSTom Wai-Hong Tam 	unsigned char color_characteristics[10];
403d46b5f7dSTom Wai-Hong Tam 	unsigned char established_timings[3];
404d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_720X400_70(_x) \
405d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 7)
406d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_720X400_88(_x) \
407d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 6)
408d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_640X480_60(_x) \
409d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 5)
410d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_640X480_67(_x) \
411d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 4)
412d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_640X480_72(_x) \
413d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 3)
414d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_640X480_75(_x) \
415d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 2)
416d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_800X600_56(_x) \
417d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 1)
418d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_800X600_60(_x) \
419d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[0]), 0)
420d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_800X600_72(_x) \
421d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 7)
422d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_800X600_75(_x) \
423d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 6)
424d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_832X624_75(_x) \
425d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 5)
426d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_1024X768_87I(_x) \
427d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 4)
428d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_1024X768_60(_x) \
429d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 3)
430d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_1024X768_70(_x) \
431d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 2)
432d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_1024X768_75(_x) \
433d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 1)
434d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_1280X1024_75(_x) \
435d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[1]), 0)
436d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_ESTABLISHED_TIMING_1152X870_75(_x) \
437d46b5f7dSTom Wai-Hong Tam 	GET_BIT(((_x).established_timings[2]), 7)
438d46b5f7dSTom Wai-Hong Tam 	struct {
439d46b5f7dSTom Wai-Hong Tam 		unsigned char xresolution;
440d46b5f7dSTom Wai-Hong Tam 		unsigned char aspect_vfreq;
441d46b5f7dSTom Wai-Hong Tam 	} __attribute__((__packed__)) standard_timings[8];
442d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_STANDARD_TIMING_XRESOLUTION(_x, _i) \
443d46b5f7dSTom Wai-Hong Tam 	(((_x).standard_timings[_i]).xresolution)
444d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_STANDARD_TIMING_ASPECT(_x, _i) \
445d46b5f7dSTom Wai-Hong Tam 	GET_BITS(((_x).standard_timings[_i].aspect_vfreq), 7, 6)
446d46b5f7dSTom Wai-Hong Tam #define EDID1_INFO_STANDARD_TIMING_VFREQ(_x, _i) \
447d46b5f7dSTom Wai-Hong Tam 	GET_BITS(((_x).standard_timings[_i].aspect_vfreq), 5, 0)
448d46b5f7dSTom Wai-Hong Tam 	union {
449d46b5f7dSTom Wai-Hong Tam 		unsigned char timing[72];
450d46b5f7dSTom Wai-Hong Tam 		struct edid_monitor_descriptor descriptor[4];
451d46b5f7dSTom Wai-Hong Tam 	} monitor_details;
452d46b5f7dSTom Wai-Hong Tam 	unsigned char extension_flag;
453d46b5f7dSTom Wai-Hong Tam 	unsigned char checksum;
454d46b5f7dSTom Wai-Hong Tam } __attribute__ ((__packed__));
455d46b5f7dSTom Wai-Hong Tam 
45643c6bdd0SJernej Skrabec enum edid_cea861_db_types {
45743c6bdd0SJernej Skrabec 	EDID_CEA861_DB_AUDIO = 0x01,
45843c6bdd0SJernej Skrabec 	EDID_CEA861_DB_VIDEO = 0x02,
45943c6bdd0SJernej Skrabec 	EDID_CEA861_DB_VENDOR = 0x03,
46043c6bdd0SJernej Skrabec 	EDID_CEA861_DB_SPEAKER = 0x04,
461*21016d27SAlgea Cao 	EDID_CEA861_DB_USE_EXTENDED = 0x07,
46243c6bdd0SJernej Skrabec };
46343c6bdd0SJernej Skrabec 
464*21016d27SAlgea Cao #define EXT_VIDEO_CAPABILITY_BLOCK 0x00
465*21016d27SAlgea Cao #define EXT_VIDEO_DATA_BLOCK_420        0x0E
466*21016d27SAlgea Cao #define EXT_VIDEO_CAP_BLOCK_Y420CMDB 0x0F
467*21016d27SAlgea Cao #define EDID_BASIC_AUDIO        BIT(6)
468*21016d27SAlgea Cao #define EDID_CEA_YCRCB444       BIT(5)
469*21016d27SAlgea Cao #define EDID_CEA_YCRCB422       BIT(4)
470*21016d27SAlgea Cao #define EDID_CEA_VCDB_QS        BIT(6)
471*21016d27SAlgea Cao 
472*21016d27SAlgea Cao #define EXT_VIDEO_DATA_BLOCK_420 0x0E
473*21016d27SAlgea Cao 
474d2fabdc7SHans de Goede struct edid_cea861_info {
475d2fabdc7SHans de Goede 	unsigned char extension_tag;
476d2fabdc7SHans de Goede #define EDID_CEA861_EXTENSION_TAG	0x02
477d2fabdc7SHans de Goede 	unsigned char revision;
478d2fabdc7SHans de Goede 	unsigned char dtd_offset;
479d2fabdc7SHans de Goede 	unsigned char dtd_count;
480d2fabdc7SHans de Goede #define EDID_CEA861_SUPPORTS_UNDERSCAN(_x) \
481d2fabdc7SHans de Goede 	GET_BIT(((_x).dtd_count), 7)
482d2fabdc7SHans de Goede #define EDID_CEA861_SUPPORTS_BASIC_AUDIO(_x) \
483d2fabdc7SHans de Goede 	GET_BIT(((_x).dtd_count), 6)
484d2fabdc7SHans de Goede #define EDID_CEA861_SUPPORTS_YUV444(_x) \
485d2fabdc7SHans de Goede 	GET_BIT(((_x).dtd_count), 5)
486d2fabdc7SHans de Goede #define EDID_CEA861_SUPPORTS_YUV422(_x) \
487d2fabdc7SHans de Goede 	GET_BIT(((_x).dtd_count), 4)
488d2fabdc7SHans de Goede #define EDID_CEA861_DTD_COUNT(_x) \
489d2fabdc7SHans de Goede 	GET_BITS(((_x).dtd_count), 3, 0)
490d2fabdc7SHans de Goede 	unsigned char data[124];
49143c6bdd0SJernej Skrabec #define EDID_CEA861_DB_TYPE(_x, offset) \
49243c6bdd0SJernej Skrabec 	GET_BITS((_x).data[offset], 7, 5)
49343c6bdd0SJernej Skrabec #define EDID_CEA861_DB_LEN(_x, offset) \
49443c6bdd0SJernej Skrabec 	GET_BITS((_x).data[offset], 4, 0)
495d2fabdc7SHans de Goede } __attribute__ ((__packed__));
496d2fabdc7SHans de Goede 
497*21016d27SAlgea Cao #define DATA_BLOCK_PRODUCT_ID 0x00
498*21016d27SAlgea Cao #define DATA_BLOCK_DISPLAY_PARAMETERS 0x01
499*21016d27SAlgea Cao #define DATA_BLOCK_COLOR_CHARACTERISTICS 0x02
500*21016d27SAlgea Cao #define DATA_BLOCK_TYPE_1_DETAILED_TIMING 0x03
501*21016d27SAlgea Cao #define DATA_BLOCK_TYPE_2_DETAILED_TIMING 0x04
502*21016d27SAlgea Cao #define DATA_BLOCK_TYPE_3_SHORT_TIMING 0x05
503*21016d27SAlgea Cao #define DATA_BLOCK_TYPE_4_DMT_TIMING 0x06
504*21016d27SAlgea Cao #define DATA_BLOCK_VESA_TIMING 0x07
505*21016d27SAlgea Cao #define DATA_BLOCK_CEA_TIMING 0x08
506*21016d27SAlgea Cao #define DATA_BLOCK_VIDEO_TIMING_RANGE 0x09
507*21016d27SAlgea Cao #define DATA_BLOCK_PRODUCT_SERIAL_NUMBER 0x0a
508*21016d27SAlgea Cao #define DATA_BLOCK_GP_ASCII_STRING 0x0b
509*21016d27SAlgea Cao #define DATA_BLOCK_DISPLAY_DEVICE_DATA 0x0c
510*21016d27SAlgea Cao #define DATA_BLOCK_INTERFACE_POWER_SEQUENCING 0x0d
511*21016d27SAlgea Cao #define DATA_BLOCK_TRANSFER_CHARACTERISTICS 0x0e
512*21016d27SAlgea Cao #define DATA_BLOCK_DISPLAY_INTERFACE 0x0f
513*21016d27SAlgea Cao #define DATA_BLOCK_STEREO_DISPLAY_INTERFACE 0x10
514*21016d27SAlgea Cao #define DATA_BLOCK_TILED_DISPLAY 0x12
515*21016d27SAlgea Cao 
516*21016d27SAlgea Cao struct displayid_hdr {
517*21016d27SAlgea Cao 	u8 rev;
518*21016d27SAlgea Cao 	u8 bytes;
519*21016d27SAlgea Cao 	u8 prod_id;
520*21016d27SAlgea Cao 	u8 ext_count;
521*21016d27SAlgea Cao } __packed;
522*21016d27SAlgea Cao 
523*21016d27SAlgea Cao struct displayid_block {
524*21016d27SAlgea Cao 	u8 tag;
525*21016d27SAlgea Cao 	u8 rev;
526*21016d27SAlgea Cao 	u8 num_bytes;
527*21016d27SAlgea Cao } __packed;
528*21016d27SAlgea Cao 
529*21016d27SAlgea Cao struct displayid_detailed_timings_1 {
530*21016d27SAlgea Cao 	u8 pixel_clock[3];
531*21016d27SAlgea Cao 	u8 flags;
532*21016d27SAlgea Cao 	u8 hactive[2];
533*21016d27SAlgea Cao 	u8 hblank[2];
534*21016d27SAlgea Cao 	u8 hsync[2];
535*21016d27SAlgea Cao 	u8 hsw[2];
536*21016d27SAlgea Cao 	u8 vactive[2];
537*21016d27SAlgea Cao 	u8 vblank[2];
538*21016d27SAlgea Cao 	u8 vsync[2];
539*21016d27SAlgea Cao 	u8 vsw[2];
540*21016d27SAlgea Cao } __packed;
541*21016d27SAlgea Cao 
542*21016d27SAlgea Cao struct displayid_detailed_timing_block {
543*21016d27SAlgea Cao 	struct displayid_block base;
544*21016d27SAlgea Cao 	struct displayid_detailed_timings_1 timings[0];
545*21016d27SAlgea Cao };
546*21016d27SAlgea Cao 
547*21016d27SAlgea Cao /**
548*21016d27SAlgea Cao  * struct drm_scrambling: sink's scrambling support.
549*21016d27SAlgea Cao  */
550*21016d27SAlgea Cao struct drm_scrambling {
551*21016d27SAlgea Cao 	/**
552*21016d27SAlgea Cao 	 * @supported: scrambling supported for rates > 340 Mhz.
553*21016d27SAlgea Cao 	 */
554*21016d27SAlgea Cao 	bool supported;
555*21016d27SAlgea Cao 	/**
556*21016d27SAlgea Cao 	 * @low_rates: scrambling supported for rates <= 340 Mhz.
557*21016d27SAlgea Cao 	 */
558*21016d27SAlgea Cao 	bool low_rates;
559*21016d27SAlgea Cao };
560*21016d27SAlgea Cao 
561*21016d27SAlgea Cao /**
562*21016d27SAlgea Cao  * struct drm_scdc - Information about scdc capabilities of a HDMI 2.0 sink
563*21016d27SAlgea Cao  *
564*21016d27SAlgea Cao  * Provides SCDC register support and capabilities related information on a
565*21016d27SAlgea Cao  * HDMI 2.0 sink. In case of a HDMI 1.4 sink, all parameter must be 0.
566*21016d27SAlgea Cao  */
567*21016d27SAlgea Cao 
568*21016d27SAlgea Cao struct drm_scdc {
569*21016d27SAlgea Cao 	/**
570*21016d27SAlgea Cao 	 * @supported: status control & data channel present.
571*21016d27SAlgea Cao 	 */
572*21016d27SAlgea Cao 	bool supported;
573*21016d27SAlgea Cao 	/**
574*21016d27SAlgea Cao 	 * @read_request: sink is capable of generating scdc read request.
575*21016d27SAlgea Cao 	 */
576*21016d27SAlgea Cao 	bool read_request;
577*21016d27SAlgea Cao 	/**
578*21016d27SAlgea Cao 	 * @scrambling: sink's scrambling capabilities
579*21016d27SAlgea Cao 	 */
580*21016d27SAlgea Cao 	struct drm_scrambling scrambling;
581*21016d27SAlgea Cao };
582*21016d27SAlgea Cao 
583*21016d27SAlgea Cao /**
584*21016d27SAlgea Cao  * struct drm_hdmi_info - runtime information about the connected HDMI sink
585*21016d27SAlgea Cao  *
586*21016d27SAlgea Cao  * Describes if a given display supports advanced HDMI 2.0 features.
587*21016d27SAlgea Cao  * This information is available in CEA-861-F extension blocks (like HF-VSDB).
588*21016d27SAlgea Cao  */
589*21016d27SAlgea Cao struct drm_hdmi_info {
590*21016d27SAlgea Cao 	struct drm_scdc scdc;
591*21016d27SAlgea Cao 
592*21016d27SAlgea Cao 	/**
593*21016d27SAlgea Cao 	 * @y420_vdb_modes: bitmap of modes which can support ycbcr420
594*21016d27SAlgea Cao 	 * output only (not normal RGB/YCBCR444/422 outputs). There are total
595*21016d27SAlgea Cao 	 * 107 VICs defined by CEA-861-F spec, so the size is 128 bits to map
596*21016d27SAlgea Cao 	 * upto 128 VICs;
597*21016d27SAlgea Cao 	 */
598*21016d27SAlgea Cao 	unsigned long y420_vdb_modes[BITS_TO_LONGS(128)];
599*21016d27SAlgea Cao 
600*21016d27SAlgea Cao 	/**
601*21016d27SAlgea Cao 	 * @y420_cmdb_modes: bitmap of modes which can support ycbcr420
602*21016d27SAlgea Cao 	 * output also, along with normal HDMI outputs. There are total 107
603*21016d27SAlgea Cao 	 * VICs defined by CEA-861-F spec, so the size is 128 bits to map upto
604*21016d27SAlgea Cao 	 * 128 VICs;
605*21016d27SAlgea Cao 	 */
606*21016d27SAlgea Cao 	unsigned long y420_cmdb_modes[BITS_TO_LONGS(128)];
607*21016d27SAlgea Cao 
608*21016d27SAlgea Cao 	/** @y420_cmdb_map: bitmap of SVD index, to extraxt vcb modes */
609*21016d27SAlgea Cao 	u64 y420_cmdb_map;
610*21016d27SAlgea Cao 
611*21016d27SAlgea Cao 	/** @y420_dc_modes: bitmap of deep color support index */
612*21016d27SAlgea Cao 	u8 y420_dc_modes;
613*21016d27SAlgea Cao };
614*21016d27SAlgea Cao 
615*21016d27SAlgea Cao enum subpixel_order {
616*21016d27SAlgea Cao 	subpixelunknown = 0,
617*21016d27SAlgea Cao 	subpixelhorizontalrgb,
618*21016d27SAlgea Cao 	subpixelhorizontalbgr,
619*21016d27SAlgea Cao 	subpixelverticalrgb,
620*21016d27SAlgea Cao 	subpixelverticalbgr,
621*21016d27SAlgea Cao 	subpixelnone,
622*21016d27SAlgea Cao };
623*21016d27SAlgea Cao 
624*21016d27SAlgea Cao #define DRM_COLOR_FORMAT_RGB444         BIT(0)
625*21016d27SAlgea Cao #define DRM_COLOR_FORMAT_YCRCB444       BIT(1)
626*21016d27SAlgea Cao #define DRM_COLOR_FORMAT_YCRCB422       BIT(2)
627*21016d27SAlgea Cao #define DRM_COLOR_FORMAT_YCRCB420       BIT(3)
628*21016d27SAlgea Cao 
629*21016d27SAlgea Cao /*
630*21016d27SAlgea Cao  * Describes a given display (e.g. CRT or flat panel) and its limitations.
631*21016d27SAlgea Cao  */
632*21016d27SAlgea Cao struct drm_display_info {
633*21016d27SAlgea Cao 	char name[32];
634*21016d27SAlgea Cao 
635*21016d27SAlgea Cao 	/* Physical size */
636*21016d27SAlgea Cao 	unsigned int width_mm;
637*21016d27SAlgea Cao 	unsigned int height_mm;
638*21016d27SAlgea Cao 
639*21016d27SAlgea Cao 	/* Clock limits FIXME: storage format */
640*21016d27SAlgea Cao 	unsigned int min_vfreq, max_vfreq;
641*21016d27SAlgea Cao 	unsigned int min_hfreq, max_hfreq;
642*21016d27SAlgea Cao 	unsigned int pixel_clock;
643*21016d27SAlgea Cao 	unsigned int bpc;
644*21016d27SAlgea Cao 
645*21016d27SAlgea Cao 	enum subpixel_order subpixel_order;
646*21016d27SAlgea Cao 	u32 color_formats;
647*21016d27SAlgea Cao 
648*21016d27SAlgea Cao 	const u32 *bus_formats;
649*21016d27SAlgea Cao 	unsigned int num_bus_formats;
650*21016d27SAlgea Cao 
651*21016d27SAlgea Cao 	/**
652*21016d27SAlgea Cao 	 * @max_tmds_clock: Maximum TMDS clock rate supported by the
653*21016d27SAlgea Cao 	 * sink in kHz. 0 means undefined.
654*21016d27SAlgea Cao 	 */
655*21016d27SAlgea Cao 	int max_tmds_clock;
656*21016d27SAlgea Cao 
657*21016d27SAlgea Cao 	/**
658*21016d27SAlgea Cao 	 * @dvi_dual: Dual-link DVI sink?
659*21016d27SAlgea Cao 	 */
660*21016d27SAlgea Cao 	bool dvi_dual;
661*21016d27SAlgea Cao 
662*21016d27SAlgea Cao 	/* Mask of supported hdmi deep color modes */
663*21016d27SAlgea Cao 	u8 edid_hdmi_dc_modes;
664*21016d27SAlgea Cao 
665*21016d27SAlgea Cao 	u8 cea_rev;
666*21016d27SAlgea Cao 
667*21016d27SAlgea Cao 	/**
668*21016d27SAlgea Cao 	 * @hdmi: advance features of a HDMI sink.
669*21016d27SAlgea Cao 	 */
670*21016d27SAlgea Cao 	struct drm_hdmi_info hdmi;
671*21016d27SAlgea Cao };
672*21016d27SAlgea Cao 
673*21016d27SAlgea Cao struct edid {
674*21016d27SAlgea Cao 	u8 header[8];
675*21016d27SAlgea Cao 	/* Vendor & product info */
676*21016d27SAlgea Cao 	u8 mfg_id[2];
677*21016d27SAlgea Cao 	u8 prod_code[2];
678*21016d27SAlgea Cao 	u32 serial; /* FIXME: byte order */
679*21016d27SAlgea Cao 	u8 mfg_week;
680*21016d27SAlgea Cao 	u8 mfg_year;
681*21016d27SAlgea Cao 	/* EDID version */
682*21016d27SAlgea Cao 	u8 version;
683*21016d27SAlgea Cao 	u8 revision;
684*21016d27SAlgea Cao 	/* Display info: */
685*21016d27SAlgea Cao 	u8 input;
686*21016d27SAlgea Cao 	u8 width_cm;
687*21016d27SAlgea Cao 	u8 height_cm;
688*21016d27SAlgea Cao 	u8 gamma;
689*21016d27SAlgea Cao 	u8 features;
690*21016d27SAlgea Cao 	/* Color characteristics */
691*21016d27SAlgea Cao 	u8 red_green_lo;
692*21016d27SAlgea Cao 	u8 black_white_lo;
693*21016d27SAlgea Cao 	u8 red_x;
694*21016d27SAlgea Cao 	u8 red_y;
695*21016d27SAlgea Cao 	u8 green_x;
696*21016d27SAlgea Cao 	u8 green_y;
697*21016d27SAlgea Cao 	u8 blue_x;
698*21016d27SAlgea Cao 	u8 blue_y;
699*21016d27SAlgea Cao 	u8 white_x;
700*21016d27SAlgea Cao 	u8 white_y;
701*21016d27SAlgea Cao 	/* Est. timings and mfg rsvd timings*/
702*21016d27SAlgea Cao 	struct est_timings established_timings;
703*21016d27SAlgea Cao 	/* Standard timings 1-8*/
704*21016d27SAlgea Cao 	struct std_timing standard_timings[8];
705*21016d27SAlgea Cao 	/* Detailing timings 1-4 */
706*21016d27SAlgea Cao 	struct detailed_timing detailed_timings[4];
707*21016d27SAlgea Cao 	/* Number of 128 byte ext. blocks */
708*21016d27SAlgea Cao 	u8 extensions;
709*21016d27SAlgea Cao 	/* Checksum */
710*21016d27SAlgea Cao 	u8 checksum;
711*21016d27SAlgea Cao } __packed;
712*21016d27SAlgea Cao 
713d46b5f7dSTom Wai-Hong Tam /**
714d46b5f7dSTom Wai-Hong Tam  * Print the EDID info.
715d46b5f7dSTom Wai-Hong Tam  *
716d46b5f7dSTom Wai-Hong Tam  * @param edid_info	The EDID info to be printed
717d46b5f7dSTom Wai-Hong Tam  */
718d46b5f7dSTom Wai-Hong Tam void edid_print_info(struct edid1_info *edid_info);
719d46b5f7dSTom Wai-Hong Tam 
720d46b5f7dSTom Wai-Hong Tam /**
721d46b5f7dSTom Wai-Hong Tam  * Check the EDID info.
722d46b5f7dSTom Wai-Hong Tam  *
723d46b5f7dSTom Wai-Hong Tam  * @param info  The EDID info to be checked
724d46b5f7dSTom Wai-Hong Tam  * @return 0 on valid, or -1 on invalid
725d46b5f7dSTom Wai-Hong Tam  */
726d46b5f7dSTom Wai-Hong Tam int edid_check_info(struct edid1_info *info);
727d46b5f7dSTom Wai-Hong Tam 
728d46b5f7dSTom Wai-Hong Tam /**
729e745d064SHans de Goede  * Check checksum of a 128 bytes EDID data block
730e745d064SHans de Goede  *
731e745d064SHans de Goede  * @param edid_block	EDID block data
732e745d064SHans de Goede  *
733e745d064SHans de Goede  * @return 0 on success, or a negative errno on error
734e745d064SHans de Goede  */
735e745d064SHans de Goede int edid_check_checksum(u8 *edid_block);
736e745d064SHans de Goede 
737e745d064SHans de Goede /**
738d46b5f7dSTom Wai-Hong Tam  * Get the horizontal and vertical rate ranges of the monitor.
739d46b5f7dSTom Wai-Hong Tam  *
740d46b5f7dSTom Wai-Hong Tam  * @param edid	The EDID info
741d46b5f7dSTom Wai-Hong Tam  * @param hmin	Returns the minimum horizontal rate
742*21016d27SAlgea Cao  * @param hmax	Returns the maximum horizontal rate
743d46b5f7dSTom Wai-Hong Tam  * @param vmin	Returns the minimum vertical rate
744*21016d27SAlgea Cao  * @param vmax	Returns the maximum vertical rate
745d46b5f7dSTom Wai-Hong Tam  * @return 0 on success, or -1 on error
746d46b5f7dSTom Wai-Hong Tam  */
747d46b5f7dSTom Wai-Hong Tam int edid_get_ranges(struct edid1_info *edid, unsigned int *hmin,
748d46b5f7dSTom Wai-Hong Tam 		    unsigned int *hmax, unsigned int *vmin,
749d46b5f7dSTom Wai-Hong Tam 		    unsigned int *vmax);
750d46b5f7dSTom Wai-Hong Tam 
751b9e63a96SMark Yao struct drm_display_mode;
75200cf1167SSimon Glass struct display_timing;
75300cf1167SSimon Glass 
754*21016d27SAlgea Cao struct hdmi_edid_data {
755*21016d27SAlgea Cao 	struct drm_display_mode *preferred_mode;
756*21016d27SAlgea Cao 	int modes;
757*21016d27SAlgea Cao 	struct drm_hdmi_info hdmi_info;
758*21016d27SAlgea Cao 	struct drm_display_mode *mode_buf;
759*21016d27SAlgea Cao 	struct drm_display_info display_info;
760*21016d27SAlgea Cao };
761*21016d27SAlgea Cao 
76200cf1167SSimon Glass /**
76300cf1167SSimon Glass  * edid_get_timing() - Get basic digital display parameters
76400cf1167SSimon Glass  *
76500cf1167SSimon Glass  * @param buf		Buffer containing EDID data
76600cf1167SSimon Glass  * @param buf_size	Size of buffer in bytes
76700cf1167SSimon Glass  * @param timing	Place to put preferring timing information
76800cf1167SSimon Glass  * @param panel_bits_per_colourp	Place to put the number of bits per
76900cf1167SSimon Glass  *			colour supported by the panel. This will be set to
77000cf1167SSimon Glass  *			-1 if not available
77100cf1167SSimon Glass  * @return 0 if timings are OK, -ve on error
77200cf1167SSimon Glass  */
77300cf1167SSimon Glass int edid_get_timing(u8 *buf, int buf_size, struct display_timing *timing,
77400cf1167SSimon Glass 		    int *panel_bits_per_colourp);
775b9e63a96SMark Yao int edid_get_drm_mode(u8 *buf, int buf_size, struct drm_display_mode *mode,
776b9e63a96SMark Yao 		      int *panel_bits_per_colourp);
777*21016d27SAlgea Cao int drm_add_edid_modes(struct hdmi_edid_data *data, u8 *edid);
778*21016d27SAlgea Cao bool drm_detect_hdmi_monitor(struct edid *edid);
779*21016d27SAlgea Cao bool drm_detect_monitor_audio(struct edid *edid);
78000cf1167SSimon Glass 
781d46b5f7dSTom Wai-Hong Tam #endif /* __EDID_H_ */
782