OSDN Git Service

6ac03cf0cfb45f24858f59426c69fdd5e10d56f6
[android-x86/external-mesa.git] / src / glx / dri2_glx.c
1 /*
2  * Copyright © 2008 Red Hat, Inc.
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the "Soft-
6  * ware"), to deal in the Software without restriction, including without
7  * limitation the rights to use, copy, modify, merge, publish, distribute,
8  * and/or sell copies of the Software, and to permit persons to whom the
9  * Software is furnished to do so, provided that the above copyright
10  * notice(s) and this permission notice appear in all copies of the Soft-
11  * ware and that both the above copyright notice(s) and this permission
12  * notice appear in supporting documentation.
13  *
14  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABIL-
16  * ITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY
17  * RIGHTS. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR HOLDERS INCLUDED IN
18  * THIS NOTICE BE LIABLE FOR ANY CLAIM, OR ANY SPECIAL INDIRECT OR CONSE-
19  * QUENTIAL DAMAGES, OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
20  * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
21  * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFOR-
22  * MANCE OF THIS SOFTWARE.
23  *
24  * Except as contained in this notice, the name of a copyright holder shall
25  * not be used in advertising or otherwise to promote the sale, use or
26  * other dealings in this Software without prior written authorization of
27  * the copyright holder.
28  *
29  * Authors:
30  *   Kristian Høgsberg (krh@redhat.com)
31  */
32
33 #if defined(GLX_DIRECT_RENDERING) && !defined(GLX_USE_APPLEGL)
34
35 #include <X11/Xlib.h>
36 #include <X11/extensions/Xfixes.h>
37 #include <X11/extensions/Xdamage.h>
38 #include "glapi.h"
39 #include "glxclient.h"
40 #include <X11/extensions/dri2proto.h>
41 #include "xf86dri.h"
42 #include <dlfcn.h>
43 #include <fcntl.h>
44 #include <unistd.h>
45 #include <sys/types.h>
46 #include <sys/mman.h>
47 #include "xf86drm.h"
48 #include "dri2.h"
49 #include "dri_common.h"
50
51 /* From xmlpool/options.h, user exposed so should be stable */
52 #define DRI_CONF_VBLANK_NEVER 0
53 #define DRI_CONF_VBLANK_DEF_INTERVAL_0 1
54 #define DRI_CONF_VBLANK_DEF_INTERVAL_1 2
55 #define DRI_CONF_VBLANK_ALWAYS_SYNC 3
56
57 #undef DRI2_MINOR
58 #define DRI2_MINOR 1
59
60 struct dri2_display
61 {
62    __GLXDRIdisplay base;
63
64    /*
65     ** XFree86-DRI version information
66     */
67    int driMajor;
68    int driMinor;
69    int driPatch;
70    int swapAvailable;
71    int invalidateAvailable;
72
73    __glxHashTable *dri2Hash;
74
75    const __DRIextension *loader_extensions[4];
76 };
77
78 struct dri2_screen {
79    __GLXscreenConfigs base;
80
81    __DRIscreen *driScreen;
82    __GLXDRIscreen vtable;
83    const __DRIdri2Extension *dri2;
84    const __DRIcoreExtension *core;
85
86    const __DRI2flushExtension *f;
87    const __DRI2configQueryExtension *config;
88    const __DRItexBufferExtension *texBuffer;
89
90    void *driver;
91    int fd;
92 };
93
94 struct dri2_context
95 {
96    __GLXDRIcontext base;
97    __DRIcontext *driContext;
98    __GLXscreenConfigs *psc;
99 };
100
101 struct dri2_drawable
102 {
103    __GLXDRIdrawable base;
104    __DRIdrawable *driDrawable;
105    __DRIbuffer buffers[5];
106    int bufferCount;
107    int width, height;
108    int have_back;
109    int have_fake_front;
110    int swap_interval;
111 };
112
113 static void
114 dri2DestroyContext(__GLXDRIcontext *context,
115                    __GLXscreenConfigs *base, Display *dpy)
116 {
117    struct dri2_context *pcp = (struct dri2_context *) context;
118    struct dri2_screen *psc = (struct dri2_screen *) base;
119
120    (*psc->core->destroyContext) (pcp->driContext);
121
122    Xfree(pcp);
123 }
124
125 static Bool
126 dri2BindContext(__GLXDRIcontext *context,
127                 __GLXDRIdrawable *draw, __GLXDRIdrawable *read)
128 {
129    struct dri2_context *pcp = (struct dri2_context *) context;
130    struct dri2_screen *psc = (struct dri2_screen *) pcp->psc;
131    struct dri2_drawable *pdr = (struct dri2_drawable *) draw;
132    struct dri2_drawable *prd = (struct dri2_drawable *) read;
133
134    return (*psc->core->bindContext) (pcp->driContext,
135                                      pdr->driDrawable, prd->driDrawable);
136 }
137
138 static void
139 dri2UnbindContext(__GLXDRIcontext *context)
140 {
141    struct dri2_context *pcp = (struct dri2_context *) context;
142    struct dri2_screen *psc = (struct dri2_screen *) pcp->psc;
143
144    (*psc->core->unbindContext) (pcp->driContext);
145 }
146
147 static __GLXDRIcontext *
148 dri2CreateContext(__GLXscreenConfigs *base,
149                   const __GLcontextModes * mode,
150                   GLXContext gc, GLXContext shareList, int renderType)
151 {
152    struct dri2_context *pcp, *pcp_shared;
153    struct dri2_screen *psc = (struct dri2_screen *) base;
154    __GLXDRIconfigPrivate *config = (__GLXDRIconfigPrivate *) mode;
155    __DRIcontext *shared = NULL;
156
157    if (shareList) {
158       pcp_shared = (struct dri2_context *) shareList->driContext;
159       shared = pcp_shared->driContext;
160    }
161
162    pcp = Xmalloc(sizeof *pcp);
163    if (pcp == NULL)
164       return NULL;
165
166    pcp->psc = &psc->base;
167    pcp->driContext =
168       (*psc->dri2->createNewContext) (psc->driScreen,
169                                       config->driConfig, shared, pcp);
170    gc->__driContext = pcp->driContext;
171
172    if (pcp->driContext == NULL) {
173       Xfree(pcp);
174       return NULL;
175    }
176
177    pcp->base.destroyContext = dri2DestroyContext;
178    pcp->base.bindContext = dri2BindContext;
179    pcp->base.unbindContext = dri2UnbindContext;
180
181    return &pcp->base;
182 }
183
184 static void
185 dri2DestroyDrawable(__GLXDRIdrawable *base)
186 {
187    struct dri2_screen *psc = (struct dri2_screen *) base->psc;
188    struct dri2_drawable *pdraw = (struct dri2_drawable *) base;
189    __GLXdisplayPrivate *dpyPriv = psc->base.display;
190    struct dri2_display *pdp = (struct dri2_display *)dpyPriv->dri2Display;
191
192    __glxHashDelete(pdp->dri2Hash, pdraw->base.xDrawable);
193    (*psc->core->destroyDrawable) (pdraw->driDrawable);
194    DRI2DestroyDrawable(psc->base.dpy, pdraw->base.xDrawable);
195    Xfree(pdraw);
196 }
197
198 static __GLXDRIdrawable *
199 dri2CreateDrawable(__GLXscreenConfigs *base, XID xDrawable,
200                    GLXDrawable drawable, const __GLcontextModes * modes)
201 {
202    struct dri2_drawable *pdraw;
203    struct dri2_screen *psc = (struct dri2_screen *) base;
204    __GLXDRIconfigPrivate *config = (__GLXDRIconfigPrivate *) modes;
205    __GLXdisplayPrivate *dpyPriv;
206    struct dri2_display *pdp;
207    GLint vblank_mode = DRI_CONF_VBLANK_DEF_INTERVAL_1;
208
209    pdraw = Xmalloc(sizeof(*pdraw));
210    if (!pdraw)
211       return NULL;
212
213    pdraw->base.destroyDrawable = dri2DestroyDrawable;
214    pdraw->base.xDrawable = xDrawable;
215    pdraw->base.drawable = drawable;
216    pdraw->base.psc = &psc->base;
217    pdraw->bufferCount = 0;
218    pdraw->swap_interval = 1; /* default may be overridden below */
219    pdraw->have_back = 0;
220
221    if (psc->config)
222       psc->config->configQueryi(psc->driScreen,
223                                 "vblank_mode", &vblank_mode);
224
225    switch (vblank_mode) {
226    case DRI_CONF_VBLANK_NEVER:
227    case DRI_CONF_VBLANK_DEF_INTERVAL_0:
228       pdraw->swap_interval = 0;
229       break;
230    case DRI_CONF_VBLANK_DEF_INTERVAL_1:
231    case DRI_CONF_VBLANK_ALWAYS_SYNC:
232    default:
233       pdraw->swap_interval = 1;
234       break;
235    }
236
237    DRI2CreateDrawable(psc->base.dpy, xDrawable);
238
239    dpyPriv = __glXInitialize(psc->base.dpy);
240    pdp = (struct dri2_display *)dpyPriv->dri2Display;;
241    /* Create a new drawable */
242    pdraw->driDrawable =
243       (*psc->dri2->createNewDrawable) (psc->driScreen,
244                                        config->driConfig, pdraw);
245
246    if (!pdraw->driDrawable) {
247       DRI2DestroyDrawable(psc->base.dpy, xDrawable);
248       Xfree(pdraw);
249       return NULL;
250    }
251
252    if (__glxHashInsert(pdp->dri2Hash, xDrawable, pdraw)) {
253       (*psc->core->destroyDrawable) (pdraw->driDrawable);
254       DRI2DestroyDrawable(psc->base.dpy, xDrawable);
255       Xfree(pdraw);
256       return None;
257    }
258
259
260 #ifdef X_DRI2SwapInterval
261    /*
262     * Make sure server has the same swap interval we do for the new
263     * drawable.
264     */
265    if (pdp->swapAvailable)
266       DRI2SwapInterval(psc->base.dpy, xDrawable, pdraw->swap_interval);
267 #endif
268
269    return &pdraw->base;
270 }
271
272 #ifdef X_DRI2GetMSC
273
274 static int
275 dri2DrawableGetMSC(__GLXscreenConfigs *psc, __GLXDRIdrawable *pdraw,
276                    int64_t *ust, int64_t *msc, int64_t *sbc)
277 {
278    CARD64 dri2_ust, dri2_msc, dri2_sbc;
279    int ret;
280
281    ret = DRI2GetMSC(psc->dpy, pdraw->xDrawable,
282                     &dri2_ust, &dri2_msc, &dri2_sbc);
283    *ust = dri2_ust;
284    *msc = dri2_msc;
285    *sbc = dri2_sbc;
286
287    return ret;
288 }
289
290 #endif
291
292
293 #ifdef X_DRI2WaitMSC
294
295 static int
296 dri2WaitForMSC(__GLXDRIdrawable *pdraw, int64_t target_msc, int64_t divisor,
297                int64_t remainder, int64_t *ust, int64_t *msc, int64_t *sbc)
298 {
299    CARD64 dri2_ust, dri2_msc, dri2_sbc;
300    int ret;
301
302    ret = DRI2WaitMSC(pdraw->psc->dpy, pdraw->xDrawable, target_msc, divisor,
303                      remainder, &dri2_ust, &dri2_msc, &dri2_sbc);
304    *ust = dri2_ust;
305    *msc = dri2_msc;
306    *sbc = dri2_sbc;
307
308    return ret;
309 }
310
311 static int
312 dri2WaitForSBC(__GLXDRIdrawable *pdraw, int64_t target_sbc, int64_t *ust,
313                int64_t *msc, int64_t *sbc)
314 {
315    CARD64 dri2_ust, dri2_msc, dri2_sbc;
316    int ret;
317
318    ret = DRI2WaitSBC(pdraw->psc->dpy, pdraw->xDrawable,
319                      target_sbc, &dri2_ust, &dri2_msc, &dri2_sbc);
320    *ust = dri2_ust;
321    *msc = dri2_msc;
322    *sbc = dri2_sbc;
323
324    return ret;
325 }
326
327 #endif /* X_DRI2WaitMSC */
328
329 static void
330 dri2CopySubBuffer(__GLXDRIdrawable *pdraw, int x, int y, int width, int height)
331 {
332    struct dri2_drawable *priv = (struct dri2_drawable *) pdraw;
333    struct dri2_screen *psc = (struct dri2_screen *) pdraw->psc;
334    XRectangle xrect;
335    XserverRegion region;
336
337    /* Check we have the right attachments */
338    if (!priv->have_back)
339       return;
340
341    xrect.x = x;
342    xrect.y = priv->height - y - height;
343    xrect.width = width;
344    xrect.height = height;
345
346 #ifdef __DRI2_FLUSH
347    if (psc->f)
348       (*psc->f->flush) (priv->driDrawable);
349 #endif
350
351    region = XFixesCreateRegion(psc->base.dpy, &xrect, 1);
352    DRI2CopyRegion(psc->base.dpy, pdraw->xDrawable, region,
353                   DRI2BufferFrontLeft, DRI2BufferBackLeft);
354    XFixesDestroyRegion(psc->base.dpy, region);
355
356    /* Refresh the fake front (if present) after we just damaged the real
357     * front.
358     */
359    DRI2CopyRegion(psc->base.dpy, pdraw->xDrawable, region,
360                   DRI2BufferFakeFrontLeft, DRI2BufferFrontLeft);
361    XFixesDestroyRegion(psc->base.dpy, region);
362 }
363
364 static void
365 dri2_copy_drawable(struct dri2_drawable *priv, int dest, int src)
366 {
367    XRectangle xrect;
368    XserverRegion region;
369    struct dri2_screen *psc = (struct dri2_screen *) priv->base.psc;
370
371    xrect.x = 0;
372    xrect.y = 0;
373    xrect.width = priv->width;
374    xrect.height = priv->height;
375
376 #ifdef __DRI2_FLUSH
377    if (psc->f)
378       (*psc->f->flush) (priv->driDrawable);
379 #endif
380
381    region = XFixesCreateRegion(psc->base.dpy, &xrect, 1);
382    DRI2CopyRegion(psc->base.dpy, priv->base.xDrawable, region, dest, src);
383    XFixesDestroyRegion(psc->base.dpy, region);
384
385 }
386
387 static void
388 dri2WaitX(__GLXDRIdrawable *pdraw)
389 {
390    struct dri2_drawable *priv = (struct dri2_drawable *) pdraw;
391
392    if (!priv->have_fake_front)
393       return;
394
395    dri2_copy_drawable(priv, DRI2BufferFakeFrontLeft, DRI2BufferFrontLeft);
396 }
397
398 static void
399 dri2WaitGL(__GLXDRIdrawable * pdraw)
400 {
401    struct dri2_drawable *priv = (struct dri2_drawable *) pdraw;
402
403    if (!priv->have_fake_front)
404       return;
405
406    dri2_copy_drawable(priv, DRI2BufferFrontLeft, DRI2BufferFakeFrontLeft);
407 }
408
409 static void
410 dri2FlushFrontBuffer(__DRIdrawable *driDrawable, void *loaderPrivate)
411 {
412    struct dri2_drawable *pdraw = loaderPrivate;
413    __GLXdisplayPrivate *priv = __glXInitialize(pdraw->base.psc->dpy);
414    struct dri2_display *pdp = (struct dri2_display *)priv->dri2Display;
415
416    /* Old servers don't send invalidate events */
417    if (!pdp->invalidateAvailable)
418        dri2InvalidateBuffers(priv->dpy, pdraw->base.drawable);
419
420    dri2WaitGL(loaderPrivate);
421 }
422
423
424 static void
425 dri2DestroyScreen(__GLXscreenConfigs *base)
426 {
427    struct dri2_screen *psc = (struct dri2_screen *) base;
428
429    /* Free the direct rendering per screen data */
430    (*psc->core->destroyScreen) (psc->driScreen);
431    close(psc->fd);
432    Xfree(psc);
433 }
434
435 /**
436  * Process list of buffer received from the server
437  *
438  * Processes the list of buffers received in a reply from the server to either
439  * \c DRI2GetBuffers or \c DRI2GetBuffersWithFormat.
440  */
441 static void
442 process_buffers(struct dri2_drawable * pdraw, DRI2Buffer * buffers,
443                 unsigned count)
444 {
445    int i;
446
447    pdraw->bufferCount = count;
448    pdraw->have_fake_front = 0;
449    pdraw->have_back = 0;
450
451    /* This assumes the DRI2 buffer attachment tokens matches the
452     * __DRIbuffer tokens. */
453    for (i = 0; i < count; i++) {
454       pdraw->buffers[i].attachment = buffers[i].attachment;
455       pdraw->buffers[i].name = buffers[i].name;
456       pdraw->buffers[i].pitch = buffers[i].pitch;
457       pdraw->buffers[i].cpp = buffers[i].cpp;
458       pdraw->buffers[i].flags = buffers[i].flags;
459       if (pdraw->buffers[i].attachment == __DRI_BUFFER_FAKE_FRONT_LEFT)
460          pdraw->have_fake_front = 1;
461       if (pdraw->buffers[i].attachment == __DRI_BUFFER_BACK_LEFT)
462          pdraw->have_back = 1;
463    }
464
465 }
466
467 static int64_t
468 dri2SwapBuffers(__GLXDRIdrawable *pdraw, int64_t target_msc, int64_t divisor,
469                 int64_t remainder)
470 {
471     struct dri2_drawable *priv = (struct dri2_drawable *) pdraw;
472     __GLXdisplayPrivate *dpyPriv = __glXInitialize(priv->base.psc->dpy);
473     struct dri2_screen *psc = (struct dri2_screen *) priv->base.psc;
474     struct dri2_display *pdp =
475         (struct dri2_display *)dpyPriv->dri2Display;
476     CARD64 ret;
477
478 #ifdef __DRI2_FLUSH
479     if (psc->f)
480         (*psc->f->flush)(priv->driDrawable);
481 #endif
482
483     /* Old servers don't send invalidate events */
484     if (!pdp->invalidateAvailable)
485        dri2InvalidateBuffers(dpyPriv->dpy, pdraw->drawable);
486
487     /* Old servers can't handle swapbuffers */
488     if (!pdp->swapAvailable) {
489        dri2CopySubBuffer(pdraw, 0, 0, priv->width, priv->height);
490        return 0;
491     }
492
493 #ifdef X_DRI2SwapBuffers
494     DRI2SwapBuffers(psc->base.dpy, pdraw->xDrawable, target_msc, divisor,
495                     remainder, &ret);
496 #endif
497
498     return ret;
499 }
500
501 static __DRIbuffer *
502 dri2GetBuffers(__DRIdrawable * driDrawable,
503                int *width, int *height,
504                unsigned int *attachments, int count,
505                int *out_count, void *loaderPrivate)
506 {
507    struct dri2_drawable *pdraw = loaderPrivate;
508    DRI2Buffer *buffers;
509
510    buffers = DRI2GetBuffers(pdraw->base.psc->dpy, pdraw->base.xDrawable,
511                             width, height, attachments, count, out_count);
512    if (buffers == NULL)
513       return NULL;
514
515    pdraw->width = *width;
516    pdraw->height = *height;
517    process_buffers(pdraw, buffers, *out_count);
518
519    Xfree(buffers);
520
521    return pdraw->buffers;
522 }
523
524 static __DRIbuffer *
525 dri2GetBuffersWithFormat(__DRIdrawable * driDrawable,
526                          int *width, int *height,
527                          unsigned int *attachments, int count,
528                          int *out_count, void *loaderPrivate)
529 {
530    struct dri2_drawable *pdraw = loaderPrivate;
531    DRI2Buffer *buffers;
532
533    buffers = DRI2GetBuffersWithFormat(pdraw->base.psc->dpy,
534                                       pdraw->base.xDrawable,
535                                       width, height, attachments,
536                                       count, out_count);
537    if (buffers == NULL)
538       return NULL;
539
540    pdraw->width = *width;
541    pdraw->height = *height;
542    process_buffers(pdraw, buffers, *out_count);
543
544    Xfree(buffers);
545
546    return pdraw->buffers;
547 }
548
549 #ifdef X_DRI2SwapInterval
550
551 static int
552 dri2SetSwapInterval(__GLXDRIdrawable *pdraw, int interval)
553 {
554    struct dri2_drawable *priv =  (struct dri2_drawable *) pdraw;
555    GLint vblank_mode = DRI_CONF_VBLANK_DEF_INTERVAL_1;
556    struct dri2_screen *psc = (struct dri2_screen *) priv->base.psc;
557
558    if (psc->config)
559       psc->config->configQueryi(psc->driScreen,
560                                 "vblank_mode", &vblank_mode);
561
562    switch (vblank_mode) {
563    case DRI_CONF_VBLANK_NEVER:
564       return GLX_BAD_VALUE;
565    case DRI_CONF_VBLANK_ALWAYS_SYNC:
566       if (interval <= 0)
567          return GLX_BAD_VALUE;
568       break;
569    default:
570       break;
571    }
572
573    DRI2SwapInterval(priv->base.psc->dpy, priv->base.xDrawable, interval);
574    priv->swap_interval = interval;
575
576    return 0;
577 }
578
579 static int
580 dri2GetSwapInterval(__GLXDRIdrawable *pdraw)
581 {
582    struct dri2_drawable *priv =  (struct dri2_drawable *) pdraw;
583
584   return priv->swap_interval;
585 }
586
587 #endif /* X_DRI2SwapInterval */
588
589 static const __DRIdri2LoaderExtension dri2LoaderExtension = {
590    {__DRI_DRI2_LOADER, __DRI_DRI2_LOADER_VERSION},
591    dri2GetBuffers,
592    dri2FlushFrontBuffer,
593    dri2GetBuffersWithFormat,
594 };
595
596 static const __DRIdri2LoaderExtension dri2LoaderExtension_old = {
597    {__DRI_DRI2_LOADER, __DRI_DRI2_LOADER_VERSION},
598    dri2GetBuffers,
599    dri2FlushFrontBuffer,
600    NULL,
601 };
602
603 #ifdef __DRI_USE_INVALIDATE
604 static const __DRIuseInvalidateExtension dri2UseInvalidate = {
605    { __DRI_USE_INVALIDATE, __DRI_USE_INVALIDATE_VERSION }
606 };
607 #endif
608
609 _X_HIDDEN void
610 dri2InvalidateBuffers(Display *dpy, XID drawable)
611 {
612    __GLXDRIdrawable *pdraw =
613       dri2GetGlxDrawableFromXDrawableId(dpy, drawable);
614    struct dri2_screen *psc = (struct dri2_screen *) pdraw->psc;
615    struct dri2_drawable *pdp = (struct dri2_drawable *) pdraw;
616
617 #if __DRI2_FLUSH_VERSION >= 3
618    if (pdraw && psc->f)
619        psc->f->invalidate(pdp->driDrawable);
620 #endif
621 }
622
623 static void
624 dri2_bind_tex_image(Display * dpy,
625                     GLXDrawable drawable,
626                     int buffer, const int *attrib_list)
627 {
628    GLXContext gc = __glXGetCurrentContext();
629    __GLXDRIdrawable *base = GetGLXDRIDrawable(dpy, drawable, NULL);
630    __GLXdisplayPrivate *dpyPriv = __glXInitialize(dpy);
631    struct dri2_drawable *pdraw = (struct dri2_drawable *) base;
632    struct dri2_display *pdp =
633       (struct dri2_display *) dpyPriv->dri2Display;
634    struct dri2_screen *psc = (struct dri2_screen *) base->psc;
635
636    if (pdraw != NULL) {
637
638 #if __DRI2_FLUSH_VERSION >= 3
639       if (!pdp->invalidateAvailable && psc->f)
640          psc->f->invalidate(pdraw->driDrawable);
641 #endif
642
643       if (psc->texBuffer->base.version >= 2 &&
644           psc->texBuffer->setTexBuffer2 != NULL) {
645          (*psc->texBuffer->setTexBuffer2) (gc->__driContext,
646                                            pdraw->base.textureTarget,
647                                            pdraw->base.textureFormat,
648                                            pdraw->driDrawable);
649       }
650       else {
651          (*psc->texBuffer->setTexBuffer) (gc->__driContext,
652                                           pdraw->base.textureTarget,
653                                           pdraw->driDrawable);
654       }
655    }
656 }
657
658 static void
659 dri2_release_tex_image(Display * dpy, GLXDrawable drawable, int buffer)
660 {
661 }
662
663 static const struct glx_context_vtable dri2_context_vtable = {
664    dri2_bind_tex_image,
665    dri2_release_tex_image,
666 };
667
668 static void
669 dri2BindExtensions(struct dri2_screen *psc, const __DRIextension **extensions)
670 {
671    int i;
672
673    __glXEnableDirectExtension(&psc->base, "GLX_SGI_video_sync");
674    __glXEnableDirectExtension(&psc->base, "GLX_SGI_swap_control");
675    __glXEnableDirectExtension(&psc->base, "GLX_MESA_swap_control");
676
677    /* FIXME: if DRI2 version supports it... */
678    __glXEnableDirectExtension(&psc->base, "INTEL_swap_event");
679
680    for (i = 0; extensions[i]; i++) {
681       if ((strcmp(extensions[i]->name, __DRI_TEX_BUFFER) == 0)) {
682          psc->texBuffer = (__DRItexBufferExtension *) extensions[i];
683          __glXEnableDirectExtension(&psc->base, "GLX_EXT_texture_from_pixmap");
684       }
685
686       if ((strcmp(extensions[i]->name, __DRI2_FLUSH) == 0)) {
687          psc->f = (__DRI2flushExtension *) extensions[i];
688          /* internal driver extension, no GL extension exposed */
689       }
690
691       if ((strcmp(extensions[i]->name, __DRI2_CONFIG_QUERY) == 0))
692          psc->config = (__DRI2configQueryExtension *) extensions[i];
693    }
694 }
695
696
697 static __GLXscreenConfigs *
698 dri2CreateScreen(int screen, __GLXdisplayPrivate * priv)
699 {
700    const __DRIconfig **driver_configs;
701    const __DRIextension **extensions;
702    const struct dri2_display *const pdp = (struct dri2_display *)
703       priv->dri2Display;
704    struct dri2_screen *psc;
705    __GLXDRIscreen *psp;
706    char *driverName, *deviceName;
707    drm_magic_t magic;
708    int i;
709
710    psc = Xmalloc(sizeof *psc);
711    if (psc == NULL)
712       return NULL;
713
714    memset(psc, 0, sizeof *psc);
715    if (!glx_screen_init(&psc->base, screen, priv))
716        return NULL;
717
718    if (!DRI2Connect(priv->dpy, RootWindow(priv->dpy, screen),
719                     &driverName, &deviceName)) {
720       XFree(psc);
721       return NULL;
722    }
723
724    psc->driver = driOpenDriver(driverName);
725    if (psc->driver == NULL) {
726       ErrorMessageF("driver pointer missing\n");
727       goto handle_error;
728    }
729
730    extensions = dlsym(psc->driver, __DRI_DRIVER_EXTENSIONS);
731    if (extensions == NULL) {
732       ErrorMessageF("driver exports no extensions (%s)\n", dlerror());
733       goto handle_error;
734    }
735
736    for (i = 0; extensions[i]; i++) {
737       if (strcmp(extensions[i]->name, __DRI_CORE) == 0)
738          psc->core = (__DRIcoreExtension *) extensions[i];
739       if (strcmp(extensions[i]->name, __DRI_DRI2) == 0)
740          psc->dri2 = (__DRIdri2Extension *) extensions[i];
741    }
742
743    if (psc->core == NULL || psc->dri2 == NULL) {
744       ErrorMessageF("core dri or dri2 extension not found\n");
745       goto handle_error;
746    }
747
748    psc->fd = open(deviceName, O_RDWR);
749    if (psc->fd < 0) {
750       ErrorMessageF("failed to open drm device: %s\n", strerror(errno));
751       goto handle_error;
752    }
753
754    if (drmGetMagic(psc->fd, &magic)) {
755       ErrorMessageF("failed to get magic\n");
756       goto handle_error;
757    }
758
759    if (!DRI2Authenticate(priv->dpy, RootWindow(priv->dpy, screen), magic)) {
760       ErrorMessageF("failed to authenticate magic %d\n", magic);
761       goto handle_error;
762    }
763
764    
765    /* If the server does not support the protocol for
766     * DRI2GetBuffersWithFormat, don't supply that interface to the driver.
767     */
768    psc->driScreen =
769       psc->dri2->createNewScreen(screen, psc->fd,
770                                  (const __DRIextension **)
771                                  &pdp->loader_extensions[0],
772                                  &driver_configs, psc);
773
774    if (psc->driScreen == NULL) {
775       ErrorMessageF("failed to create dri screen\n");
776       goto handle_error;
777    }
778
779    extensions = psc->core->getExtensions(psc->driScreen);
780    driBindCommonExtensions(&psc->base, extensions);
781    dri2BindExtensions(psc, extensions);
782
783    psc->base.configs =
784       driConvertConfigs(psc->core, psc->base.configs, driver_configs);
785    psc->base.visuals =
786       driConvertConfigs(psc->core, psc->base.visuals, driver_configs);
787
788    psc->base.driver_configs = driver_configs;
789
790    psp = &psc->vtable;
791    psc->base.driScreen = psp;
792    psp->destroyScreen = dri2DestroyScreen;
793    psp->createContext = dri2CreateContext;
794    psp->createDrawable = dri2CreateDrawable;
795    psp->swapBuffers = dri2SwapBuffers;
796    psp->waitGL = dri2WaitGL;
797    psp->waitX = dri2WaitX;
798    psp->getDrawableMSC = NULL;
799    psp->waitForMSC = NULL;
800    psp->waitForSBC = NULL;
801    psp->setSwapInterval = NULL;
802    psp->getSwapInterval = NULL;
803
804    if (pdp->driMinor >= 2) {
805 #ifdef X_DRI2GetMSC
806       psp->getDrawableMSC = dri2DrawableGetMSC;
807 #endif
808 #ifdef X_DRI2WaitMSC
809       psp->waitForMSC = dri2WaitForMSC;
810       psp->waitForSBC = dri2WaitForSBC;
811 #endif
812 #ifdef X_DRI2SwapInterval
813       psp->setSwapInterval = dri2SetSwapInterval;
814       psp->getSwapInterval = dri2GetSwapInterval;
815 #endif
816 #if defined(X_DRI2GetMSC) && defined(X_DRI2WaitMSC) && defined(X_DRI2SwapInterval)
817       __glXEnableDirectExtension(&psc->base, "GLX_OML_sync_control");
818 #endif
819    }
820
821    /* DRI2 suports SubBuffer through DRI2CopyRegion, so it's always
822     * available.*/
823    psp->copySubBuffer = dri2CopySubBuffer;
824    __glXEnableDirectExtension(&psc->base, "GLX_MESA_copy_sub_buffer");
825
826    psc->base.direct_context_vtable = &dri2_context_vtable;
827
828    Xfree(driverName);
829    Xfree(deviceName);
830
831    return &psc->base;
832
833 handle_error:
834    Xfree(driverName);
835    Xfree(deviceName);
836    XFree(psc);
837
838    /* FIXME: clean up here */
839
840    return NULL;
841 }
842
843 /* Called from __glXFreeDisplayPrivate.
844  */
845 static void
846 dri2DestroyDisplay(__GLXDRIdisplay * dpy)
847 {
848    Xfree(dpy);
849 }
850
851 _X_HIDDEN __GLXDRIdrawable *
852 dri2GetGlxDrawableFromXDrawableId(Display *dpy, XID id)
853 {
854    __GLXdisplayPrivate *d = __glXInitialize(dpy);
855    struct dri2_display *pdp = (struct dri2_display *) d->dri2Display;
856    __GLXDRIdrawable *pdraw;
857
858    if (__glxHashLookup(pdp->dri2Hash, id, (void *) &pdraw) == 0)
859       return pdraw;
860
861    return NULL;
862 }
863
864 /*
865  * Allocate, initialize and return a __DRIdisplayPrivate object.
866  * This is called from __glXInitialize() when we are given a new
867  * display pointer.
868  */
869 _X_HIDDEN __GLXDRIdisplay *
870 dri2CreateDisplay(Display * dpy)
871 {
872    struct dri2_display *pdp;
873    int eventBase, errorBase, i;
874
875    if (!DRI2QueryExtension(dpy, &eventBase, &errorBase))
876       return NULL;
877
878    pdp = Xmalloc(sizeof *pdp);
879    if (pdp == NULL)
880       return NULL;
881
882    if (!DRI2QueryVersion(dpy, &pdp->driMajor, &pdp->driMinor)) {
883       Xfree(pdp);
884       return NULL;
885    }
886
887    pdp->driPatch = 0;
888    pdp->swapAvailable = (pdp->driMinor >= 2);
889    pdp->invalidateAvailable = (pdp->driMinor >= 3);
890
891    pdp->base.destroyDisplay = dri2DestroyDisplay;
892    pdp->base.createScreen = dri2CreateScreen;
893
894    i = 0;
895    if (pdp->driMinor < 1)
896       pdp->loader_extensions[i++] = &dri2LoaderExtension_old.base;
897    else
898       pdp->loader_extensions[i++] = &dri2LoaderExtension.base;
899    
900    pdp->loader_extensions[i++] = &systemTimeExtension.base;
901
902 #ifdef __DRI_USE_INVALIDATE
903    pdp->loader_extensions[i++] = &dri2UseInvalidate.base;
904 #endif
905    pdp->loader_extensions[i++] = NULL;
906
907    pdp->dri2Hash = __glxHashCreate();
908    if (pdp->dri2Hash == NULL) {
909       Xfree(pdp);
910       return NULL;
911    }
912
913    return &pdp->base;
914 }
915
916 #endif /* GLX_DIRECT_RENDERING */