xref: /OK3568_Linux_fs/external/xserver/hw/xfree86/ddc/ddc.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /* xf86DDC.c
2*4882a593Smuzhiyun  *
3*4882a593Smuzhiyun  * Copyright 1998,1999 by Egbert Eich <Egbert.Eich@Physik.TU-Darmstadt.DE>
4*4882a593Smuzhiyun  */
5*4882a593Smuzhiyun 
6*4882a593Smuzhiyun /*
7*4882a593Smuzhiyun  * A note on terminology.  DDC1 is the original dumb serial protocol, and
8*4882a593Smuzhiyun  * can only do up to 128 bytes of EDID.  DDC2 is I2C-encapsulated and
9*4882a593Smuzhiyun  * introduces extension blocks.  EDID is the old display identification
10*4882a593Smuzhiyun  * block, DisplayID is the new one.
11*4882a593Smuzhiyun  */
12*4882a593Smuzhiyun 
13*4882a593Smuzhiyun #ifdef HAVE_XORG_CONFIG_H
14*4882a593Smuzhiyun #include <xorg-config.h>
15*4882a593Smuzhiyun #endif
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun #include "misc.h"
18*4882a593Smuzhiyun #include "xf86.h"
19*4882a593Smuzhiyun #include "xf86_OSproc.h"
20*4882a593Smuzhiyun #include "xf86DDC.h"
21*4882a593Smuzhiyun #include <string.h>
22*4882a593Smuzhiyun 
23*4882a593Smuzhiyun #define RETRIES 4
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun typedef enum {
26*4882a593Smuzhiyun     DDCOPT_NODDC1,
27*4882a593Smuzhiyun     DDCOPT_NODDC2,
28*4882a593Smuzhiyun     DDCOPT_NODDC
29*4882a593Smuzhiyun } DDCOpts;
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun static const OptionInfoRec DDCOptions[] = {
32*4882a593Smuzhiyun     {DDCOPT_NODDC1, "NoDDC1", OPTV_BOOLEAN, {0}, FALSE},
33*4882a593Smuzhiyun     {DDCOPT_NODDC2, "NoDDC2", OPTV_BOOLEAN, {0}, FALSE},
34*4882a593Smuzhiyun     {DDCOPT_NODDC, "NoDDC", OPTV_BOOLEAN, {0}, FALSE},
35*4882a593Smuzhiyun     {-1, NULL, OPTV_NONE, {0}, FALSE},
36*4882a593Smuzhiyun };
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun /* DDC1 */
39*4882a593Smuzhiyun 
40*4882a593Smuzhiyun static int
find_start(unsigned int * ptr)41*4882a593Smuzhiyun find_start(unsigned int *ptr)
42*4882a593Smuzhiyun {
43*4882a593Smuzhiyun     unsigned int comp[9], test[9];
44*4882a593Smuzhiyun     int i, j;
45*4882a593Smuzhiyun 
46*4882a593Smuzhiyun     for (i = 0; i < 9; i++) {
47*4882a593Smuzhiyun         comp[i] = *(ptr++);
48*4882a593Smuzhiyun         test[i] = 1;
49*4882a593Smuzhiyun     }
50*4882a593Smuzhiyun     for (i = 0; i < 127; i++) {
51*4882a593Smuzhiyun         for (j = 0; j < 9; j++) {
52*4882a593Smuzhiyun             test[j] = test[j] & !(comp[j] ^ *(ptr++));
53*4882a593Smuzhiyun         }
54*4882a593Smuzhiyun     }
55*4882a593Smuzhiyun     for (i = 0; i < 9; i++)
56*4882a593Smuzhiyun         if (test[i])
57*4882a593Smuzhiyun             return i + 1;
58*4882a593Smuzhiyun     return -1;
59*4882a593Smuzhiyun }
60*4882a593Smuzhiyun 
61*4882a593Smuzhiyun static unsigned char *
find_header(unsigned char * block)62*4882a593Smuzhiyun find_header(unsigned char *block)
63*4882a593Smuzhiyun {
64*4882a593Smuzhiyun     unsigned char *ptr, *head_ptr, *end;
65*4882a593Smuzhiyun     unsigned char header[] = { 0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00 };
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun     ptr = block;
68*4882a593Smuzhiyun     end = block + EDID1_LEN;
69*4882a593Smuzhiyun     while (ptr < end) {
70*4882a593Smuzhiyun         int i;
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun         head_ptr = ptr;
73*4882a593Smuzhiyun         for (i = 0; i < 8; i++) {
74*4882a593Smuzhiyun             if (header[i] != *(head_ptr++))
75*4882a593Smuzhiyun                 break;
76*4882a593Smuzhiyun             if (head_ptr == end)
77*4882a593Smuzhiyun                 head_ptr = block;
78*4882a593Smuzhiyun         }
79*4882a593Smuzhiyun         if (i == 8)
80*4882a593Smuzhiyun             break;
81*4882a593Smuzhiyun         ptr++;
82*4882a593Smuzhiyun     }
83*4882a593Smuzhiyun     if (ptr == end)
84*4882a593Smuzhiyun         return NULL;
85*4882a593Smuzhiyun     return ptr;
86*4882a593Smuzhiyun }
87*4882a593Smuzhiyun 
88*4882a593Smuzhiyun static unsigned char *
resort(unsigned char * s_block)89*4882a593Smuzhiyun resort(unsigned char *s_block)
90*4882a593Smuzhiyun {
91*4882a593Smuzhiyun     unsigned char *d_new, *d_ptr, *d_end, *s_ptr, *s_end;
92*4882a593Smuzhiyun     unsigned char tmp;
93*4882a593Smuzhiyun 
94*4882a593Smuzhiyun     s_ptr = find_header(s_block);
95*4882a593Smuzhiyun     if (!s_ptr)
96*4882a593Smuzhiyun         return NULL;
97*4882a593Smuzhiyun     s_end = s_block + EDID1_LEN;
98*4882a593Smuzhiyun 
99*4882a593Smuzhiyun     d_new = malloc(EDID1_LEN);
100*4882a593Smuzhiyun     if (!d_new)
101*4882a593Smuzhiyun         return NULL;
102*4882a593Smuzhiyun     d_end = d_new + EDID1_LEN;
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun     for (d_ptr = d_new; d_ptr < d_end; d_ptr++) {
105*4882a593Smuzhiyun         tmp = *(s_ptr++);
106*4882a593Smuzhiyun         *d_ptr = tmp;
107*4882a593Smuzhiyun         if (s_ptr == s_end)
108*4882a593Smuzhiyun             s_ptr = s_block;
109*4882a593Smuzhiyun     }
110*4882a593Smuzhiyun     free(s_block);
111*4882a593Smuzhiyun     return d_new;
112*4882a593Smuzhiyun }
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun static int
DDC_checksum(const unsigned char * block,int len)115*4882a593Smuzhiyun DDC_checksum(const unsigned char *block, int len)
116*4882a593Smuzhiyun {
117*4882a593Smuzhiyun     int i, result = 0;
118*4882a593Smuzhiyun     int not_null = 0;
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun     for (i = 0; i < len; i++) {
121*4882a593Smuzhiyun         not_null |= block[i];
122*4882a593Smuzhiyun         result += block[i];
123*4882a593Smuzhiyun     }
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun #ifdef DEBUG
126*4882a593Smuzhiyun     if (result & 0xFF)
127*4882a593Smuzhiyun         ErrorF("DDC checksum not correct\n");
128*4882a593Smuzhiyun     if (!not_null)
129*4882a593Smuzhiyun         ErrorF("DDC read all Null\n");
130*4882a593Smuzhiyun #endif
131*4882a593Smuzhiyun 
132*4882a593Smuzhiyun     /* catch the trivial case where all bytes are 0 */
133*4882a593Smuzhiyun     if (!not_null)
134*4882a593Smuzhiyun         return 1;
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun     return result & 0xFF;
137*4882a593Smuzhiyun }
138*4882a593Smuzhiyun 
139*4882a593Smuzhiyun static unsigned char *
GetEDID_DDC1(unsigned int * s_ptr)140*4882a593Smuzhiyun GetEDID_DDC1(unsigned int *s_ptr)
141*4882a593Smuzhiyun {
142*4882a593Smuzhiyun     unsigned char *d_block, *d_pos;
143*4882a593Smuzhiyun     unsigned int *s_pos, *s_end;
144*4882a593Smuzhiyun     int s_start;
145*4882a593Smuzhiyun     int i, j;
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun     s_start = find_start(s_ptr);
148*4882a593Smuzhiyun     if (s_start == -1)
149*4882a593Smuzhiyun         return NULL;
150*4882a593Smuzhiyun     s_end = s_ptr + NUM;
151*4882a593Smuzhiyun     s_pos = s_ptr + s_start;
152*4882a593Smuzhiyun     d_block = calloc(1, EDID1_LEN);
153*4882a593Smuzhiyun     if (!d_block)
154*4882a593Smuzhiyun         return NULL;
155*4882a593Smuzhiyun     d_pos = d_block;
156*4882a593Smuzhiyun     for (i = 0; i < EDID1_LEN; i++) {
157*4882a593Smuzhiyun         for (j = 0; j < 8; j++) {
158*4882a593Smuzhiyun             *d_pos <<= 1;
159*4882a593Smuzhiyun             if (*s_pos) {
160*4882a593Smuzhiyun                 *d_pos |= 0x01;
161*4882a593Smuzhiyun             }
162*4882a593Smuzhiyun             s_pos++;
163*4882a593Smuzhiyun             if (s_pos == s_end)
164*4882a593Smuzhiyun                 s_pos = s_ptr;
165*4882a593Smuzhiyun         };
166*4882a593Smuzhiyun         s_pos++;
167*4882a593Smuzhiyun         if (s_pos == s_end)
168*4882a593Smuzhiyun             s_pos = s_ptr;
169*4882a593Smuzhiyun         d_pos++;
170*4882a593Smuzhiyun     }
171*4882a593Smuzhiyun     free(s_ptr);
172*4882a593Smuzhiyun     if (d_block && DDC_checksum(d_block, EDID1_LEN)) {
173*4882a593Smuzhiyun         free(d_block);
174*4882a593Smuzhiyun         return NULL;
175*4882a593Smuzhiyun     }
176*4882a593Smuzhiyun     return (resort(d_block));
177*4882a593Smuzhiyun }
178*4882a593Smuzhiyun 
179*4882a593Smuzhiyun /* fetch entire EDID record; DDC bit needs to be masked */
180*4882a593Smuzhiyun static unsigned int *
FetchEDID_DDC1(register ScrnInfoPtr pScrn,register unsigned int (* read_DDC)(ScrnInfoPtr))181*4882a593Smuzhiyun FetchEDID_DDC1(register ScrnInfoPtr pScrn,
182*4882a593Smuzhiyun                register unsigned int (*read_DDC) (ScrnInfoPtr))
183*4882a593Smuzhiyun {
184*4882a593Smuzhiyun     int count = NUM;
185*4882a593Smuzhiyun     unsigned int *ptr, *xp;
186*4882a593Smuzhiyun 
187*4882a593Smuzhiyun     ptr = xp = malloc(sizeof(int) * NUM);
188*4882a593Smuzhiyun 
189*4882a593Smuzhiyun     if (!ptr)
190*4882a593Smuzhiyun         return NULL;
191*4882a593Smuzhiyun     do {
192*4882a593Smuzhiyun         /* wait for next retrace */
193*4882a593Smuzhiyun         *xp = read_DDC(pScrn);
194*4882a593Smuzhiyun         xp++;
195*4882a593Smuzhiyun     } while (--count);
196*4882a593Smuzhiyun     return ptr;
197*4882a593Smuzhiyun }
198*4882a593Smuzhiyun 
199*4882a593Smuzhiyun /* test if DDC1  return 0 if not */
200*4882a593Smuzhiyun static Bool
TestDDC1(ScrnInfoPtr pScrn,unsigned int (* read_DDC)(ScrnInfoPtr))201*4882a593Smuzhiyun TestDDC1(ScrnInfoPtr pScrn, unsigned int (*read_DDC) (ScrnInfoPtr))
202*4882a593Smuzhiyun {
203*4882a593Smuzhiyun     int old, count;
204*4882a593Smuzhiyun 
205*4882a593Smuzhiyun     old = read_DDC(pScrn);
206*4882a593Smuzhiyun     count = HEADER * BITS_PER_BYTE;
207*4882a593Smuzhiyun     do {
208*4882a593Smuzhiyun         /* wait for next retrace */
209*4882a593Smuzhiyun         if (old != read_DDC(pScrn))
210*4882a593Smuzhiyun             break;
211*4882a593Smuzhiyun     } while (count--);
212*4882a593Smuzhiyun     return count;
213*4882a593Smuzhiyun }
214*4882a593Smuzhiyun 
215*4882a593Smuzhiyun /*
216*4882a593Smuzhiyun  * read EDID record , pass it to callback function to interpret.
217*4882a593Smuzhiyun  * callback function will store it for further use by calling
218*4882a593Smuzhiyun  * function; it will also decide if we need to reread it
219*4882a593Smuzhiyun  */
220*4882a593Smuzhiyun static unsigned char *
EDIDRead_DDC1(ScrnInfoPtr pScrn,DDC1SetSpeedProc DDCSpeed,unsigned int (* read_DDC)(ScrnInfoPtr))221*4882a593Smuzhiyun EDIDRead_DDC1(ScrnInfoPtr pScrn, DDC1SetSpeedProc DDCSpeed,
222*4882a593Smuzhiyun               unsigned int (*read_DDC) (ScrnInfoPtr))
223*4882a593Smuzhiyun {
224*4882a593Smuzhiyun     unsigned char *EDID_block = NULL;
225*4882a593Smuzhiyun     int count = RETRIES;
226*4882a593Smuzhiyun 
227*4882a593Smuzhiyun     if (!read_DDC) {
228*4882a593Smuzhiyun         xf86DrvMsg(pScrn->scrnIndex, X_PROBED,
229*4882a593Smuzhiyun                    "chipset doesn't support DDC1\n");
230*4882a593Smuzhiyun         return NULL;
231*4882a593Smuzhiyun     };
232*4882a593Smuzhiyun 
233*4882a593Smuzhiyun     if (TestDDC1(pScrn, read_DDC) == -1) {
234*4882a593Smuzhiyun         xf86DrvMsg(pScrn->scrnIndex, X_PROBED, "No DDC signal\n");
235*4882a593Smuzhiyun         return NULL;
236*4882a593Smuzhiyun     };
237*4882a593Smuzhiyun 
238*4882a593Smuzhiyun     if (DDCSpeed)
239*4882a593Smuzhiyun         DDCSpeed(pScrn, DDC_FAST);
240*4882a593Smuzhiyun     do {
241*4882a593Smuzhiyun         EDID_block = GetEDID_DDC1(FetchEDID_DDC1(pScrn, read_DDC));
242*4882a593Smuzhiyun         count--;
243*4882a593Smuzhiyun     } while (!EDID_block && count);
244*4882a593Smuzhiyun     if (DDCSpeed)
245*4882a593Smuzhiyun         DDCSpeed(pScrn, DDC_SLOW);
246*4882a593Smuzhiyun 
247*4882a593Smuzhiyun     return EDID_block;
248*4882a593Smuzhiyun }
249*4882a593Smuzhiyun 
250*4882a593Smuzhiyun /**
251*4882a593Smuzhiyun  * Attempts to probe the monitor for EDID information, if NoDDC and NoDDC1 are
252*4882a593Smuzhiyun  * unset.  EDID information blocks are interpreted and the results returned in
253*4882a593Smuzhiyun  * an xf86MonPtr.
254*4882a593Smuzhiyun  *
255*4882a593Smuzhiyun  * This function does not affect the list of modes used by drivers -- it is up
256*4882a593Smuzhiyun  * to the driver to decide policy on what to do with EDID information.
257*4882a593Smuzhiyun  *
258*4882a593Smuzhiyun  * @return pointer to a new xf86MonPtr containing the EDID information.
259*4882a593Smuzhiyun  * @return NULL if no monitor attached or failure to interpret the EDID.
260*4882a593Smuzhiyun  */
261*4882a593Smuzhiyun xf86MonPtr
xf86DoEDID_DDC1(ScrnInfoPtr pScrn,DDC1SetSpeedProc DDC1SetSpeed,unsigned int (* DDC1Read)(ScrnInfoPtr))262*4882a593Smuzhiyun xf86DoEDID_DDC1(ScrnInfoPtr pScrn, DDC1SetSpeedProc DDC1SetSpeed,
263*4882a593Smuzhiyun                 unsigned int (*DDC1Read) (ScrnInfoPtr))
264*4882a593Smuzhiyun {
265*4882a593Smuzhiyun     unsigned char *EDID_block = NULL;
266*4882a593Smuzhiyun     xf86MonPtr tmp = NULL;
267*4882a593Smuzhiyun 
268*4882a593Smuzhiyun     /* Default DDC and DDC1 to enabled. */
269*4882a593Smuzhiyun     Bool noddc = FALSE, noddc1 = FALSE;
270*4882a593Smuzhiyun     OptionInfoPtr options;
271*4882a593Smuzhiyun 
272*4882a593Smuzhiyun     options = xnfalloc(sizeof(DDCOptions));
273*4882a593Smuzhiyun     (void) memcpy(options, DDCOptions, sizeof(DDCOptions));
274*4882a593Smuzhiyun     xf86ProcessOptions(pScrn->scrnIndex, pScrn->options, options);
275*4882a593Smuzhiyun 
276*4882a593Smuzhiyun     xf86GetOptValBool(options, DDCOPT_NODDC, &noddc);
277*4882a593Smuzhiyun     xf86GetOptValBool(options, DDCOPT_NODDC1, &noddc1);
278*4882a593Smuzhiyun     free(options);
279*4882a593Smuzhiyun 
280*4882a593Smuzhiyun     if (noddc || noddc1)
281*4882a593Smuzhiyun         return NULL;
282*4882a593Smuzhiyun 
283*4882a593Smuzhiyun     OsBlockSignals();
284*4882a593Smuzhiyun     EDID_block = EDIDRead_DDC1(pScrn, DDC1SetSpeed, DDC1Read);
285*4882a593Smuzhiyun     OsReleaseSignals();
286*4882a593Smuzhiyun 
287*4882a593Smuzhiyun     if (EDID_block) {
288*4882a593Smuzhiyun         tmp = xf86InterpretEDID(pScrn->scrnIndex, EDID_block);
289*4882a593Smuzhiyun     }
290*4882a593Smuzhiyun #ifdef DEBUG
291*4882a593Smuzhiyun     else
292*4882a593Smuzhiyun         ErrorF("No EDID block returned\n");
293*4882a593Smuzhiyun     if (!tmp)
294*4882a593Smuzhiyun         ErrorF("Cannot interpret EDID block\n");
295*4882a593Smuzhiyun #endif
296*4882a593Smuzhiyun     return tmp;
297*4882a593Smuzhiyun }
298*4882a593Smuzhiyun 
299*4882a593Smuzhiyun /* DDC2 */
300*4882a593Smuzhiyun 
301*4882a593Smuzhiyun static I2CDevPtr
DDC2MakeDevice(I2CBusPtr pBus,int address,const char * name)302*4882a593Smuzhiyun DDC2MakeDevice(I2CBusPtr pBus, int address, const char *name)
303*4882a593Smuzhiyun {
304*4882a593Smuzhiyun     I2CDevPtr dev = NULL;
305*4882a593Smuzhiyun 
306*4882a593Smuzhiyun     if (!(dev = xf86I2CFindDev(pBus, address))) {
307*4882a593Smuzhiyun         dev = xf86CreateI2CDevRec();
308*4882a593Smuzhiyun         dev->DevName = name;
309*4882a593Smuzhiyun         dev->SlaveAddr = address;
310*4882a593Smuzhiyun         dev->ByteTimeout = 2200;        /* VESA DDC spec 3 p. 43 (+10 %) */
311*4882a593Smuzhiyun         dev->StartTimeout = 550;
312*4882a593Smuzhiyun         dev->BitTimeout = 40;
313*4882a593Smuzhiyun         dev->AcknTimeout = 40;
314*4882a593Smuzhiyun 
315*4882a593Smuzhiyun         dev->pI2CBus = pBus;
316*4882a593Smuzhiyun         if (!xf86I2CDevInit(dev)) {
317*4882a593Smuzhiyun             xf86DrvMsg(pBus->scrnIndex, X_PROBED, "No DDC2 device\n");
318*4882a593Smuzhiyun             return NULL;
319*4882a593Smuzhiyun         }
320*4882a593Smuzhiyun     }
321*4882a593Smuzhiyun 
322*4882a593Smuzhiyun     return dev;
323*4882a593Smuzhiyun }
324*4882a593Smuzhiyun 
325*4882a593Smuzhiyun static I2CDevPtr
DDC2Init(I2CBusPtr pBus)326*4882a593Smuzhiyun DDC2Init(I2CBusPtr pBus)
327*4882a593Smuzhiyun {
328*4882a593Smuzhiyun     I2CDevPtr dev = NULL;
329*4882a593Smuzhiyun 
330*4882a593Smuzhiyun     /*
331*4882a593Smuzhiyun      * Slow down the bus so that older monitors don't
332*4882a593Smuzhiyun      * miss things.
333*4882a593Smuzhiyun      */
334*4882a593Smuzhiyun     pBus->RiseFallTime = 20;
335*4882a593Smuzhiyun 
336*4882a593Smuzhiyun     dev = DDC2MakeDevice(pBus, 0x00A0, "ddc2");
337*4882a593Smuzhiyun     if (xf86I2CProbeAddress(pBus, 0x0060))
338*4882a593Smuzhiyun         DDC2MakeDevice(pBus, 0x0060, "E-EDID segment register");
339*4882a593Smuzhiyun 
340*4882a593Smuzhiyun     return dev;
341*4882a593Smuzhiyun }
342*4882a593Smuzhiyun 
343*4882a593Smuzhiyun /* Mmmm, smell the hacks */
344*4882a593Smuzhiyun static void
EEDIDStop(I2CDevPtr d)345*4882a593Smuzhiyun EEDIDStop(I2CDevPtr d)
346*4882a593Smuzhiyun {
347*4882a593Smuzhiyun }
348*4882a593Smuzhiyun 
349*4882a593Smuzhiyun /* block is the EDID block number.  a segment is two blocks. */
350*4882a593Smuzhiyun static Bool
DDC2Read(I2CDevPtr dev,int block,unsigned char * R_Buffer)351*4882a593Smuzhiyun DDC2Read(I2CDevPtr dev, int block, unsigned char *R_Buffer)
352*4882a593Smuzhiyun {
353*4882a593Smuzhiyun     unsigned char W_Buffer[1];
354*4882a593Smuzhiyun     int i, segment;
355*4882a593Smuzhiyun     I2CDevPtr seg;
356*4882a593Smuzhiyun     void (*stop) (I2CDevPtr);
357*4882a593Smuzhiyun 
358*4882a593Smuzhiyun     for (i = 0; i < RETRIES; i++) {
359*4882a593Smuzhiyun         /* Stop bits reset the segment pointer to 0, so be careful here. */
360*4882a593Smuzhiyun         segment = block >> 1;
361*4882a593Smuzhiyun         if (segment) {
362*4882a593Smuzhiyun             Bool b;
363*4882a593Smuzhiyun 
364*4882a593Smuzhiyun             if (!(seg = xf86I2CFindDev(dev->pI2CBus, 0x0060)))
365*4882a593Smuzhiyun                 return FALSE;
366*4882a593Smuzhiyun 
367*4882a593Smuzhiyun             W_Buffer[0] = segment;
368*4882a593Smuzhiyun 
369*4882a593Smuzhiyun             stop = dev->pI2CBus->I2CStop;
370*4882a593Smuzhiyun             dev->pI2CBus->I2CStop = EEDIDStop;
371*4882a593Smuzhiyun 
372*4882a593Smuzhiyun             b = xf86I2CWriteRead(seg, W_Buffer, 1, NULL, 0);
373*4882a593Smuzhiyun 
374*4882a593Smuzhiyun             dev->pI2CBus->I2CStop = stop;
375*4882a593Smuzhiyun             if (!b) {
376*4882a593Smuzhiyun                 dev->pI2CBus->I2CStop(dev);
377*4882a593Smuzhiyun                 continue;
378*4882a593Smuzhiyun             }
379*4882a593Smuzhiyun         }
380*4882a593Smuzhiyun 
381*4882a593Smuzhiyun         W_Buffer[0] = (block & 0x01) * EDID1_LEN;
382*4882a593Smuzhiyun 
383*4882a593Smuzhiyun         if (xf86I2CWriteRead(dev, W_Buffer, 1, R_Buffer, EDID1_LEN)) {
384*4882a593Smuzhiyun             if (!DDC_checksum(R_Buffer, EDID1_LEN))
385*4882a593Smuzhiyun                 return TRUE;
386*4882a593Smuzhiyun         }
387*4882a593Smuzhiyun     }
388*4882a593Smuzhiyun 
389*4882a593Smuzhiyun     return FALSE;
390*4882a593Smuzhiyun }
391*4882a593Smuzhiyun 
392*4882a593Smuzhiyun /**
393*4882a593Smuzhiyun  * Attempts to probe the monitor for EDID information, if NoDDC and NoDDC2 are
394*4882a593Smuzhiyun  * unset.  EDID information blocks are interpreted and the results returned in
395*4882a593Smuzhiyun  * an xf86MonPtr.  Unlike xf86DoEDID_DDC[12](), this function will return
396*4882a593Smuzhiyun  * the complete EDID data, including all extension blocks, if the 'complete'
397*4882a593Smuzhiyun  * parameter is TRUE;
398*4882a593Smuzhiyun  *
399*4882a593Smuzhiyun  * This function does not affect the list of modes used by drivers -- it is up
400*4882a593Smuzhiyun  * to the driver to decide policy on what to do with EDID information.
401*4882a593Smuzhiyun  *
402*4882a593Smuzhiyun  * @return pointer to a new xf86MonPtr containing the EDID information.
403*4882a593Smuzhiyun  * @return NULL if no monitor attached or failure to interpret the EDID.
404*4882a593Smuzhiyun  */
405*4882a593Smuzhiyun xf86MonPtr
xf86DoEEDID(ScrnInfoPtr pScrn,I2CBusPtr pBus,Bool complete)406*4882a593Smuzhiyun xf86DoEEDID(ScrnInfoPtr pScrn, I2CBusPtr pBus, Bool complete)
407*4882a593Smuzhiyun {
408*4882a593Smuzhiyun     unsigned char *EDID_block = NULL;
409*4882a593Smuzhiyun     xf86MonPtr tmp = NULL;
410*4882a593Smuzhiyun     I2CDevPtr dev = NULL;
411*4882a593Smuzhiyun 
412*4882a593Smuzhiyun     /* Default DDC and DDC2 to enabled. */
413*4882a593Smuzhiyun     Bool noddc = FALSE, noddc2 = FALSE;
414*4882a593Smuzhiyun     OptionInfoPtr options;
415*4882a593Smuzhiyun 
416*4882a593Smuzhiyun     options = malloc(sizeof(DDCOptions));
417*4882a593Smuzhiyun     if (!options)
418*4882a593Smuzhiyun         return NULL;
419*4882a593Smuzhiyun     memcpy(options, DDCOptions, sizeof(DDCOptions));
420*4882a593Smuzhiyun     xf86ProcessOptions(pScrn->scrnIndex, pScrn->options, options);
421*4882a593Smuzhiyun 
422*4882a593Smuzhiyun     xf86GetOptValBool(options, DDCOPT_NODDC, &noddc);
423*4882a593Smuzhiyun     xf86GetOptValBool(options, DDCOPT_NODDC2, &noddc2);
424*4882a593Smuzhiyun     free(options);
425*4882a593Smuzhiyun 
426*4882a593Smuzhiyun     if (noddc || noddc2)
427*4882a593Smuzhiyun         return NULL;
428*4882a593Smuzhiyun 
429*4882a593Smuzhiyun     if (!(dev = DDC2Init(pBus)))
430*4882a593Smuzhiyun         return NULL;
431*4882a593Smuzhiyun 
432*4882a593Smuzhiyun     EDID_block = calloc(1, EDID1_LEN);
433*4882a593Smuzhiyun     if (!EDID_block)
434*4882a593Smuzhiyun         return NULL;
435*4882a593Smuzhiyun 
436*4882a593Smuzhiyun     if (DDC2Read(dev, 0, EDID_block)) {
437*4882a593Smuzhiyun         int i, n = EDID_block[0x7e];
438*4882a593Smuzhiyun 
439*4882a593Smuzhiyun         if (complete && n) {
440*4882a593Smuzhiyun             EDID_block = reallocarray(EDID_block, 1 + n, EDID1_LEN);
441*4882a593Smuzhiyun 
442*4882a593Smuzhiyun             for (i = 0; i < n; i++)
443*4882a593Smuzhiyun                 DDC2Read(dev, i + 1, EDID_block + (EDID1_LEN * (1 + i)));
444*4882a593Smuzhiyun         }
445*4882a593Smuzhiyun 
446*4882a593Smuzhiyun         tmp = xf86InterpretEEDID(pScrn->scrnIndex, EDID_block);
447*4882a593Smuzhiyun     }
448*4882a593Smuzhiyun 
449*4882a593Smuzhiyun     if (tmp && complete)
450*4882a593Smuzhiyun         tmp->flags |= MONITOR_EDID_COMPLETE_RAWDATA;
451*4882a593Smuzhiyun 
452*4882a593Smuzhiyun     return tmp;
453*4882a593Smuzhiyun }
454*4882a593Smuzhiyun 
455*4882a593Smuzhiyun /**
456*4882a593Smuzhiyun  * Attempts to probe the monitor for EDID information, if NoDDC and NoDDC2 are
457*4882a593Smuzhiyun  * unset.  EDID information blocks are interpreted and the results returned in
458*4882a593Smuzhiyun  * an xf86MonPtr.
459*4882a593Smuzhiyun  *
460*4882a593Smuzhiyun  * This function does not affect the list of modes used by drivers -- it is up
461*4882a593Smuzhiyun  * to the driver to decide policy on what to do with EDID information.
462*4882a593Smuzhiyun  *
463*4882a593Smuzhiyun  * @return pointer to a new xf86MonPtr containing the EDID information.
464*4882a593Smuzhiyun  * @return NULL if no monitor attached or failure to interpret the EDID.
465*4882a593Smuzhiyun  */
466*4882a593Smuzhiyun xf86MonPtr
xf86DoEDID_DDC2(ScrnInfoPtr pScrn,I2CBusPtr pBus)467*4882a593Smuzhiyun xf86DoEDID_DDC2(ScrnInfoPtr pScrn, I2CBusPtr pBus)
468*4882a593Smuzhiyun {
469*4882a593Smuzhiyun     return xf86DoEEDID(pScrn, pBus, FALSE);
470*4882a593Smuzhiyun }
471