1 /*
2  * Copyright 2008 George Sapountzis
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the "Software"),
6  * to deal in the Software without restriction, including without limitation
7  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8  * and/or sell copies of the Software, and to permit persons to whom the
9  * Software is furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice (including the next
12  * paragraph) shall be included in all copies or substantial portions of the
13  * Software.
14  *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21  * SOFTWARE.
22  */
23 
24 #if defined(GLX_DIRECT_RENDERING) && !defined(GLX_USE_APPLEGL)
25 
26 #include <X11/Xlib.h>
27 #include "glxclient.h"
28 #include <dlfcn.h>
29 #include "dri_common.h"
30 #include "drisw_priv.h"
31 
32 static Bool
XCreateDrawable(struct drisw_drawable * pdp,Display * dpy,XID drawable,int visualid)33 XCreateDrawable(struct drisw_drawable * pdp,
34                 Display * dpy, XID drawable, int visualid)
35 {
36    XGCValues gcvalues;
37    long visMask;
38    XVisualInfo visTemp;
39    int num_visuals;
40 
41    /* create GC's */
42    pdp->gc = XCreateGC(dpy, drawable, 0, NULL);
43    pdp->swapgc = XCreateGC(dpy, drawable, 0, NULL);
44 
45    gcvalues.function = GXcopy;
46    gcvalues.graphics_exposures = False;
47    XChangeGC(dpy, pdp->gc, GCFunction, &gcvalues);
48    XChangeGC(dpy, pdp->swapgc, GCFunction, &gcvalues);
49    XChangeGC(dpy, pdp->swapgc, GCGraphicsExposures, &gcvalues);
50 
51    /* visual */
52    visTemp.visualid = visualid;
53    visMask = VisualIDMask;
54    pdp->visinfo = XGetVisualInfo(dpy, visMask, &visTemp, &num_visuals);
55 
56    if (!pdp->visinfo || num_visuals == 0)
57       return False;
58 
59    /* create XImage */
60    pdp->ximage = XCreateImage(dpy,
61                               pdp->visinfo->visual,
62                               pdp->visinfo->depth,
63                               ZPixmap, 0,             /* format, offset */
64                               NULL,                   /* data */
65                               0, 0,                   /* width, height */
66                               32,                     /* bitmap_pad */
67                               0);                     /* bytes_per_line */
68 
69   /**
70    * swrast does not handle 24-bit depth with 24 bpp, so let X do the
71    * the conversion for us.
72    */
73   if (pdp->ximage->bits_per_pixel == 24)
74      pdp->ximage->bits_per_pixel = 32;
75 
76    return True;
77 }
78 
79 static void
XDestroyDrawable(struct drisw_drawable * pdp,Display * dpy,XID drawable)80 XDestroyDrawable(struct drisw_drawable * pdp, Display * dpy, XID drawable)
81 {
82    XDestroyImage(pdp->ximage);
83    free(pdp->visinfo);
84 
85    XFreeGC(dpy, pdp->gc);
86    XFreeGC(dpy, pdp->swapgc);
87 }
88 
89 /**
90  * swrast loader functions
91  */
92 
93 static void
swrastGetDrawableInfo(__DRIdrawable * draw,int * x,int * y,int * w,int * h,void * loaderPrivate)94 swrastGetDrawableInfo(__DRIdrawable * draw,
95                       int *x, int *y, int *w, int *h,
96                       void *loaderPrivate)
97 {
98    struct drisw_drawable *pdp = loaderPrivate;
99    __GLXDRIdrawable *pdraw = &(pdp->base);
100    Display *dpy = pdraw->psc->dpy;
101    Drawable drawable;
102 
103    Window root;
104    unsigned uw, uh, bw, depth;
105 
106    drawable = pdraw->xDrawable;
107 
108    XGetGeometry(dpy, drawable, &root, x, y, &uw, &uh, &bw, &depth);
109    *w = uw;
110    *h = uh;
111 }
112 
113 /**
114  * Align renderbuffer pitch.
115  *
116  * This should be chosen by the driver and the loader (libGL, xserver/glx)
117  * should use the driver provided pitch.
118  *
119  * It seems that the xorg loader (that is the xserver loading swrast_dri for
120  * indirect rendering, not client-side libGL) requires that the pitch is
121  * exactly the image width padded to 32 bits. XXX
122  *
123  * The above restriction can probably be overcome by using ScratchPixmap and
124  * CopyArea in the xserver, similar to ShmPutImage, and setting the width of
125  * the scratch pixmap to 'pitch / cpp'.
126  */
127 static inline int
bytes_per_line(unsigned pitch_bits,unsigned mul)128 bytes_per_line(unsigned pitch_bits, unsigned mul)
129 {
130    unsigned mask = mul - 1;
131 
132    return ((pitch_bits + mask) & ~mask) / 8;
133 }
134 
135 static void
swrastPutImage2(__DRIdrawable * draw,int op,int x,int y,int w,int h,int stride,char * data,void * loaderPrivate)136 swrastPutImage2(__DRIdrawable * draw, int op,
137                 int x, int y, int w, int h, int stride,
138                 char *data, void *loaderPrivate)
139 {
140    struct drisw_drawable *pdp = loaderPrivate;
141    __GLXDRIdrawable *pdraw = &(pdp->base);
142    Display *dpy = pdraw->psc->dpy;
143    Drawable drawable;
144    XImage *ximage;
145    GC gc;
146 
147    switch (op) {
148    case __DRI_SWRAST_IMAGE_OP_DRAW:
149       gc = pdp->gc;
150       break;
151    case __DRI_SWRAST_IMAGE_OP_SWAP:
152       gc = pdp->swapgc;
153       break;
154    default:
155       return;
156    }
157 
158    drawable = pdraw->xDrawable;
159 
160    ximage = pdp->ximage;
161    ximage->data = data;
162    ximage->width = w;
163    ximage->height = h;
164    ximage->bytes_per_line = stride ? stride : bytes_per_line(w * ximage->bits_per_pixel, 32);
165 
166    XPutImage(dpy, drawable, gc, ximage, 0, 0, x, y, w, h);
167 
168    ximage->data = NULL;
169 }
170 
171 static void
swrastPutImage(__DRIdrawable * draw,int op,int x,int y,int w,int h,char * data,void * loaderPrivate)172 swrastPutImage(__DRIdrawable * draw, int op,
173                int x, int y, int w, int h,
174                char *data, void *loaderPrivate)
175 {
176    swrastPutImage2(draw, op, x, y, w, h, 0, data, loaderPrivate);
177 }
178 
179 static void
swrastGetImage2(__DRIdrawable * read,int x,int y,int w,int h,int stride,char * data,void * loaderPrivate)180 swrastGetImage2(__DRIdrawable * read,
181                 int x, int y, int w, int h, int stride,
182                 char *data, void *loaderPrivate)
183 {
184    struct drisw_drawable *prp = loaderPrivate;
185    __GLXDRIdrawable *pread = &(prp->base);
186    Display *dpy = pread->psc->dpy;
187    Drawable readable;
188    XImage *ximage;
189 
190    readable = pread->xDrawable;
191 
192    ximage = prp->ximage;
193    ximage->data = data;
194    ximage->width = w;
195    ximage->height = h;
196    ximage->bytes_per_line = stride ? stride : bytes_per_line(w * ximage->bits_per_pixel, 32);
197 
198    XGetSubImage(dpy, readable, x, y, w, h, ~0L, ZPixmap, ximage, 0, 0);
199 
200    ximage->data = NULL;
201 }
202 
203 static void
swrastGetImage(__DRIdrawable * read,int x,int y,int w,int h,char * data,void * loaderPrivate)204 swrastGetImage(__DRIdrawable * read,
205                int x, int y, int w, int h,
206                char *data, void *loaderPrivate)
207 {
208    swrastGetImage2(read, x, y, w, h, 0, data, loaderPrivate);
209 }
210 
211 static const __DRIswrastLoaderExtension swrastLoaderExtension = {
212    .base = {__DRI_SWRAST_LOADER, 3 },
213 
214    .getDrawableInfo     = swrastGetDrawableInfo,
215    .putImage            = swrastPutImage,
216    .getImage            = swrastGetImage,
217    .putImage2           = swrastPutImage2,
218    .getImage2           = swrastGetImage2,
219 };
220 
221 static const __DRIextension *loader_extensions[] = {
222    &swrastLoaderExtension.base,
223    NULL
224 };
225 
226 /**
227  * GLXDRI functions
228  */
229 
230 static void
drisw_destroy_context(struct glx_context * context)231 drisw_destroy_context(struct glx_context *context)
232 {
233    struct drisw_context *pcp = (struct drisw_context *) context;
234    struct drisw_screen *psc = (struct drisw_screen *) context->psc;
235 
236    driReleaseDrawables(&pcp->base);
237 
238    free((char *) context->extensions);
239 
240    (*psc->core->destroyContext) (pcp->driContext);
241 
242    free(pcp);
243 }
244 
245 static int
drisw_bind_context(struct glx_context * context,struct glx_context * old,GLXDrawable draw,GLXDrawable read)246 drisw_bind_context(struct glx_context *context, struct glx_context *old,
247 		   GLXDrawable draw, GLXDrawable read)
248 {
249    struct drisw_context *pcp = (struct drisw_context *) context;
250    struct drisw_screen *psc = (struct drisw_screen *) pcp->base.psc;
251    struct drisw_drawable *pdraw, *pread;
252 
253    pdraw = (struct drisw_drawable *) driFetchDrawable(context, draw);
254    pread = (struct drisw_drawable *) driFetchDrawable(context, read);
255 
256    driReleaseDrawables(&pcp->base);
257 
258    if ((*psc->core->bindContext) (pcp->driContext,
259                                   pdraw ? pdraw->driDrawable : NULL,
260                                   pread ? pread->driDrawable : NULL))
261       return Success;
262 
263    return GLXBadContext;
264 }
265 
266 static void
drisw_unbind_context(struct glx_context * context,struct glx_context * new)267 drisw_unbind_context(struct glx_context *context, struct glx_context *new)
268 {
269    struct drisw_context *pcp = (struct drisw_context *) context;
270    struct drisw_screen *psc = (struct drisw_screen *) pcp->base.psc;
271 
272    (*psc->core->unbindContext) (pcp->driContext);
273 }
274 
275 static void
drisw_bind_tex_image(Display * dpy,GLXDrawable drawable,int buffer,const int * attrib_list)276 drisw_bind_tex_image(Display * dpy,
277 		    GLXDrawable drawable,
278 		    int buffer, const int *attrib_list)
279 {
280    struct glx_context *gc = __glXGetCurrentContext();
281    struct drisw_context *pcp = (struct drisw_context *) gc;
282    __GLXDRIdrawable *base = GetGLXDRIDrawable(dpy, drawable);
283    struct drisw_drawable *pdraw = (struct drisw_drawable *) base;
284    struct drisw_screen *psc;
285 
286    __glXInitialize(dpy);
287 
288    if (pdraw != NULL) {
289       psc = (struct drisw_screen *) base->psc;
290 
291       if (!psc->texBuffer)
292          return;
293 
294       if (psc->texBuffer->base.version >= 2 &&
295         psc->texBuffer->setTexBuffer2 != NULL) {
296 	      (*psc->texBuffer->setTexBuffer2) (pcp->driContext,
297 					   pdraw->base.textureTarget,
298 					   pdraw->base.textureFormat,
299 					   pdraw->driDrawable);
300       }
301       else {
302 	      (*psc->texBuffer->setTexBuffer) (pcp->driContext,
303 					  pdraw->base.textureTarget,
304 					  pdraw->driDrawable);
305       }
306    }
307 }
308 
309 static void
drisw_release_tex_image(Display * dpy,GLXDrawable drawable,int buffer)310 drisw_release_tex_image(Display * dpy, GLXDrawable drawable, int buffer)
311 {
312    struct glx_context *gc = __glXGetCurrentContext();
313    struct drisw_context *pcp = (struct drisw_context *) gc;
314    __GLXDRIdrawable *base = GetGLXDRIDrawable(dpy, drawable);
315    struct glx_display *dpyPriv = __glXInitialize(dpy);
316    struct drisw_drawable *pdraw = (struct drisw_drawable *) base;
317    struct drisw_screen *psc;
318 
319    if (dpyPriv != NULL && pdraw != NULL) {
320       psc = (struct drisw_screen *) base->psc;
321 
322       if (!psc->texBuffer)
323          return;
324 
325       if (psc->texBuffer->base.version >= 3 &&
326           psc->texBuffer->releaseTexBuffer != NULL) {
327          (*psc->texBuffer->releaseTexBuffer) (pcp->driContext,
328                                            pdraw->base.textureTarget,
329                                            pdraw->driDrawable);
330       }
331    }
332 }
333 
334 static const struct glx_context_vtable drisw_context_vtable = {
335    .destroy             = drisw_destroy_context,
336    .bind                = drisw_bind_context,
337    .unbind              = drisw_unbind_context,
338    .wait_gl             = NULL,
339    .wait_x              = NULL,
340    .use_x_font          = DRI_glXUseXFont,
341    .bind_tex_image      = drisw_bind_tex_image,
342    .release_tex_image   = drisw_release_tex_image,
343    .get_proc_address    = NULL,
344 };
345 
346 static struct glx_context *
drisw_create_context(struct glx_screen * base,struct glx_config * config_base,struct glx_context * shareList,int renderType)347 drisw_create_context(struct glx_screen *base,
348 		     struct glx_config *config_base,
349 		     struct glx_context *shareList, int renderType)
350 {
351    struct drisw_context *pcp, *pcp_shared;
352    __GLXDRIconfigPrivate *config = (__GLXDRIconfigPrivate *) config_base;
353    struct drisw_screen *psc = (struct drisw_screen *) base;
354    __DRIcontext *shared = NULL;
355 
356    if (!psc->base.driScreen)
357       return NULL;
358 
359    /* Check the renderType value */
360    if (!validate_renderType_against_config(config_base, renderType))
361        return NULL;
362 
363    if (shareList) {
364       /* If the shareList context is not a DRISW context, we cannot possibly
365        * create a DRISW context that shares it.
366        */
367       if (shareList->vtable->destroy != drisw_destroy_context) {
368 	 return NULL;
369       }
370 
371       pcp_shared = (struct drisw_context *) shareList;
372       shared = pcp_shared->driContext;
373    }
374 
375    pcp = calloc(1, sizeof *pcp);
376    if (pcp == NULL)
377       return NULL;
378 
379    if (!glx_context_init(&pcp->base, &psc->base, &config->base)) {
380       free(pcp);
381       return NULL;
382    }
383 
384    pcp->base.renderType = renderType;
385 
386    pcp->driContext =
387       (*psc->core->createNewContext) (psc->driScreen,
388 				      config->driConfig, shared, pcp);
389    if (pcp->driContext == NULL) {
390       free(pcp);
391       return NULL;
392    }
393 
394    pcp->base.vtable = &drisw_context_vtable;
395 
396    return &pcp->base;
397 }
398 
399 static struct glx_context *
drisw_create_context_attribs(struct glx_screen * base,struct glx_config * config_base,struct glx_context * shareList,unsigned num_attribs,const uint32_t * attribs,unsigned * error)400 drisw_create_context_attribs(struct glx_screen *base,
401 			     struct glx_config *config_base,
402 			     struct glx_context *shareList,
403 			     unsigned num_attribs,
404 			     const uint32_t *attribs,
405 			     unsigned *error)
406 {
407    struct drisw_context *pcp, *pcp_shared;
408    __GLXDRIconfigPrivate *config = (__GLXDRIconfigPrivate *) config_base;
409    struct drisw_screen *psc = (struct drisw_screen *) base;
410    __DRIcontext *shared = NULL;
411 
412    uint32_t minor_ver;
413    uint32_t major_ver;
414    uint32_t renderType;
415    uint32_t flags;
416    unsigned api;
417    int reset;
418    int release;
419    uint32_t ctx_attribs[2 * 5];
420    unsigned num_ctx_attribs = 0;
421 
422    if (!psc->base.driScreen)
423       return NULL;
424 
425    if (psc->swrast->base.version < 3)
426       return NULL;
427 
428    /* Remap the GLX tokens to DRI2 tokens.
429     */
430    if (!dri2_convert_glx_attribs(num_attribs, attribs,
431                                  &major_ver, &minor_ver, &renderType, &flags,
432                                  &api, &reset, &release, error))
433       return NULL;
434 
435    /* Check the renderType value */
436    if (!validate_renderType_against_config(config_base, renderType)) {
437        return NULL;
438    }
439 
440    if (reset != __DRI_CTX_RESET_NO_NOTIFICATION)
441       return NULL;
442 
443    if (release != __DRI_CTX_RELEASE_BEHAVIOR_FLUSH &&
444        release != __DRI_CTX_RELEASE_BEHAVIOR_NONE)
445       return NULL;
446 
447    if (shareList) {
448       pcp_shared = (struct drisw_context *) shareList;
449       shared = pcp_shared->driContext;
450    }
451 
452    pcp = calloc(1, sizeof *pcp);
453    if (pcp == NULL)
454       return NULL;
455 
456    if (!glx_context_init(&pcp->base, &psc->base, config_base)) {
457       free(pcp);
458       return NULL;
459    }
460 
461    ctx_attribs[num_ctx_attribs++] = __DRI_CTX_ATTRIB_MAJOR_VERSION;
462    ctx_attribs[num_ctx_attribs++] = major_ver;
463    ctx_attribs[num_ctx_attribs++] = __DRI_CTX_ATTRIB_MINOR_VERSION;
464    ctx_attribs[num_ctx_attribs++] = minor_ver;
465    if (release != __DRI_CTX_RELEASE_BEHAVIOR_FLUSH) {
466        ctx_attribs[num_ctx_attribs++] = __DRI_CTX_ATTRIB_RELEASE_BEHAVIOR;
467        ctx_attribs[num_ctx_attribs++] = release;
468    }
469 
470    if (flags != 0) {
471       ctx_attribs[num_ctx_attribs++] = __DRI_CTX_ATTRIB_FLAGS;
472 
473       /* The current __DRI_CTX_FLAG_* values are identical to the
474        * GLX_CONTEXT_*_BIT values.
475        */
476       ctx_attribs[num_ctx_attribs++] = flags;
477    }
478 
479    pcp->base.renderType = renderType;
480 
481    pcp->driContext =
482       (*psc->swrast->createContextAttribs) (psc->driScreen,
483 					    api,
484 					    config ? config->driConfig : 0,
485 					    shared,
486 					    num_ctx_attribs / 2,
487 					    ctx_attribs,
488 					    error,
489 					    pcp);
490    if (pcp->driContext == NULL) {
491       free(pcp);
492       return NULL;
493    }
494 
495    pcp->base.vtable = &drisw_context_vtable;
496 
497    return &pcp->base;
498 }
499 
500 static void
driswDestroyDrawable(__GLXDRIdrawable * pdraw)501 driswDestroyDrawable(__GLXDRIdrawable * pdraw)
502 {
503    struct drisw_drawable *pdp = (struct drisw_drawable *) pdraw;
504    struct drisw_screen *psc = (struct drisw_screen *) pdp->base.psc;
505 
506    (*psc->core->destroyDrawable) (pdp->driDrawable);
507 
508    XDestroyDrawable(pdp, pdraw->psc->dpy, pdraw->drawable);
509    free(pdp);
510 }
511 
512 static __GLXDRIdrawable *
driswCreateDrawable(struct glx_screen * base,XID xDrawable,GLXDrawable drawable,struct glx_config * modes)513 driswCreateDrawable(struct glx_screen *base, XID xDrawable,
514 		    GLXDrawable drawable, struct glx_config *modes)
515 {
516    struct drisw_drawable *pdp;
517    __GLXDRIconfigPrivate *config = (__GLXDRIconfigPrivate *) modes;
518    struct drisw_screen *psc = (struct drisw_screen *) base;
519    Bool ret;
520    const __DRIswrastExtension *swrast = psc->swrast;
521 
522    pdp = calloc(1, sizeof(*pdp));
523    if (!pdp)
524       return NULL;
525 
526    pdp->base.xDrawable = xDrawable;
527    pdp->base.drawable = drawable;
528    pdp->base.psc = &psc->base;
529 
530    ret = XCreateDrawable(pdp, psc->base.dpy, xDrawable, modes->visualID);
531    if (!ret) {
532       free(pdp);
533       return NULL;
534    }
535 
536    /* Create a new drawable */
537    pdp->driDrawable =
538       (*swrast->createNewDrawable) (psc->driScreen, config->driConfig, pdp);
539 
540    if (!pdp->driDrawable) {
541       XDestroyDrawable(pdp, psc->base.dpy, xDrawable);
542       free(pdp);
543       return NULL;
544    }
545 
546    pdp->base.destroyDrawable = driswDestroyDrawable;
547 
548    return &pdp->base;
549 }
550 
551 static int64_t
driswSwapBuffers(__GLXDRIdrawable * pdraw,int64_t target_msc,int64_t divisor,int64_t remainder,Bool flush)552 driswSwapBuffers(__GLXDRIdrawable * pdraw,
553                  int64_t target_msc, int64_t divisor, int64_t remainder,
554                  Bool flush)
555 {
556    struct drisw_drawable *pdp = (struct drisw_drawable *) pdraw;
557    struct drisw_screen *psc = (struct drisw_screen *) pdp->base.psc;
558 
559    (void) target_msc;
560    (void) divisor;
561    (void) remainder;
562 
563    if (flush) {
564       glFlush();
565    }
566 
567    (*psc->core->swapBuffers) (pdp->driDrawable);
568 
569    return 0;
570 }
571 
572 static void
driswCopySubBuffer(__GLXDRIdrawable * pdraw,int x,int y,int width,int height,Bool flush)573 driswCopySubBuffer(__GLXDRIdrawable * pdraw,
574                    int x, int y, int width, int height, Bool flush)
575 {
576    struct drisw_drawable *pdp = (struct drisw_drawable *) pdraw;
577    struct drisw_screen *psc = (struct drisw_screen *) pdp->base.psc;
578 
579    if (flush) {
580       glFlush();
581    }
582 
583    (*psc->copySubBuffer->copySubBuffer) (pdp->driDrawable,
584 					    x, y, width, height);
585 }
586 
587 static void
driswDestroyScreen(struct glx_screen * base)588 driswDestroyScreen(struct glx_screen *base)
589 {
590    struct drisw_screen *psc = (struct drisw_screen *) base;
591 
592    /* Free the direct rendering per screen data */
593    (*psc->core->destroyScreen) (psc->driScreen);
594    driDestroyConfigs(psc->driver_configs);
595    psc->driScreen = NULL;
596    if (psc->driver)
597       dlclose(psc->driver);
598    free(psc);
599 }
600 
601 #define SWRAST_DRIVER_NAME "swrast"
602 
603 static void *
driOpenSwrast(void)604 driOpenSwrast(void)
605 {
606    void *driver = NULL;
607 
608    if (driver == NULL)
609       driver = driOpenDriver(SWRAST_DRIVER_NAME);
610 
611    return driver;
612 }
613 
614 static const struct glx_screen_vtable drisw_screen_vtable = {
615    .create_context         = drisw_create_context,
616    .create_context_attribs = drisw_create_context_attribs,
617    .query_renderer_integer = drisw_query_renderer_integer,
618    .query_renderer_string  = drisw_query_renderer_string,
619 };
620 
621 static void
driswBindExtensions(struct drisw_screen * psc,const __DRIextension ** extensions)622 driswBindExtensions(struct drisw_screen *psc, const __DRIextension **extensions)
623 {
624    int i;
625 
626    __glXEnableDirectExtension(&psc->base, "GLX_SGI_make_current_read");
627 
628    if (psc->swrast->base.version >= 3) {
629       __glXEnableDirectExtension(&psc->base, "GLX_ARB_create_context");
630       __glXEnableDirectExtension(&psc->base, "GLX_ARB_create_context_profile");
631 
632       /* DRISW version >= 2 implies support for OpenGL ES.
633        */
634       __glXEnableDirectExtension(&psc->base,
635 				 "GLX_EXT_create_context_es_profile");
636       __glXEnableDirectExtension(&psc->base,
637 				 "GLX_EXT_create_context_es2_profile");
638    }
639 
640    if (psc->copySubBuffer)
641       __glXEnableDirectExtension(&psc->base, "GLX_MESA_copy_sub_buffer");
642 
643    /* FIXME: Figure out what other extensions can be ported here from dri2. */
644    for (i = 0; extensions[i]; i++) {
645       if ((strcmp(extensions[i]->name, __DRI_TEX_BUFFER) == 0)) {
646 	 psc->texBuffer = (__DRItexBufferExtension *) extensions[i];
647 	 __glXEnableDirectExtension(&psc->base, "GLX_EXT_texture_from_pixmap");
648       }
649       /* DRISW version 3 is also required because GLX_MESA_query_renderer
650        * requires GLX_ARB_create_context_profile.
651        */
652       if (psc->swrast->base.version >= 3
653           && strcmp(extensions[i]->name, __DRI2_RENDERER_QUERY) == 0) {
654          psc->rendererQuery = (__DRI2rendererQueryExtension *) extensions[i];
655          __glXEnableDirectExtension(&psc->base, "GLX_MESA_query_renderer");
656       }
657       if (strcmp(extensions[i]->name, __DRI2_FLUSH_CONTROL) == 0) {
658 	  __glXEnableDirectExtension(&psc->base,
659 				     "GLX_ARB_context_flush_control");
660       }
661    }
662 }
663 
664 static struct glx_screen *
driswCreateScreen(int screen,struct glx_display * priv)665 driswCreateScreen(int screen, struct glx_display *priv)
666 {
667    __GLXDRIscreen *psp;
668    const __DRIconfig **driver_configs;
669    const __DRIextension **extensions;
670    struct drisw_screen *psc;
671    struct glx_config *configs = NULL, *visuals = NULL;
672    int i;
673 
674    psc = calloc(1, sizeof *psc);
675    if (psc == NULL)
676       return NULL;
677 
678    if (!glx_screen_init(&psc->base, screen, priv)) {
679       free(psc);
680       return NULL;
681    }
682 
683    psc->driver = driOpenSwrast();
684    if (psc->driver == NULL)
685       goto handle_error;
686 
687    extensions = driGetDriverExtensions(psc->driver, SWRAST_DRIVER_NAME);
688    if (extensions == NULL)
689       goto handle_error;
690 
691    for (i = 0; extensions[i]; i++) {
692       if (strcmp(extensions[i]->name, __DRI_CORE) == 0)
693 	 psc->core = (__DRIcoreExtension *) extensions[i];
694       if (strcmp(extensions[i]->name, __DRI_SWRAST) == 0)
695 	 psc->swrast = (__DRIswrastExtension *) extensions[i];
696       if (strcmp(extensions[i]->name, __DRI_COPY_SUB_BUFFER) == 0)
697 	 psc->copySubBuffer = (__DRIcopySubBufferExtension *) extensions[i];
698    }
699 
700    if (psc->core == NULL || psc->swrast == NULL) {
701       ErrorMessageF("core dri extension not found\n");
702       goto handle_error;
703    }
704 
705    if (psc->swrast->base.version >= 4) {
706       psc->driScreen =
707          psc->swrast->createNewScreen2(screen, loader_extensions,
708                                        extensions,
709                                        &driver_configs, psc);
710    } else {
711       psc->driScreen =
712          psc->swrast->createNewScreen(screen, loader_extensions,
713                                       &driver_configs, psc);
714    }
715    if (psc->driScreen == NULL) {
716       ErrorMessageF("failed to create dri screen\n");
717       goto handle_error;
718    }
719 
720    extensions = psc->core->getExtensions(psc->driScreen);
721    driswBindExtensions(psc, extensions);
722 
723    configs = driConvertConfigs(psc->core, psc->base.configs, driver_configs);
724    visuals = driConvertConfigs(psc->core, psc->base.visuals, driver_configs);
725 
726    if (!configs || !visuals) {
727        ErrorMessageF("No matching fbConfigs or visuals found\n");
728        goto handle_error;
729    }
730 
731    glx_config_destroy_list(psc->base.configs);
732    psc->base.configs = configs;
733    glx_config_destroy_list(psc->base.visuals);
734    psc->base.visuals = visuals;
735 
736    psc->driver_configs = driver_configs;
737 
738    psc->base.vtable = &drisw_screen_vtable;
739    psp = &psc->vtable;
740    psc->base.driScreen = psp;
741    psp->destroyScreen = driswDestroyScreen;
742    psp->createDrawable = driswCreateDrawable;
743    psp->swapBuffers = driswSwapBuffers;
744 
745    if (psc->copySubBuffer)
746       psp->copySubBuffer = driswCopySubBuffer;
747 
748    return &psc->base;
749 
750  handle_error:
751    if (configs)
752        glx_config_destroy_list(configs);
753    if (visuals)
754        glx_config_destroy_list(visuals);
755    if (psc->driScreen)
756        psc->core->destroyScreen(psc->driScreen);
757    psc->driScreen = NULL;
758 
759    if (psc->driver)
760       dlclose(psc->driver);
761    glx_screen_cleanup(&psc->base);
762    free(psc);
763 
764    CriticalErrorMessageF("failed to load driver: %s\n", SWRAST_DRIVER_NAME);
765 
766    return NULL;
767 }
768 
769 /* Called from __glXFreeDisplayPrivate.
770  */
771 static void
driswDestroyDisplay(__GLXDRIdisplay * dpy)772 driswDestroyDisplay(__GLXDRIdisplay * dpy)
773 {
774    free(dpy);
775 }
776 
777 /*
778  * Allocate, initialize and return a __DRIdisplayPrivate object.
779  * This is called from __glXInitialize() when we are given a new
780  * display pointer.
781  */
782 _X_HIDDEN __GLXDRIdisplay *
driswCreateDisplay(Display * dpy)783 driswCreateDisplay(Display * dpy)
784 {
785    struct drisw_display *pdpyp;
786 
787    pdpyp = malloc(sizeof *pdpyp);
788    if (pdpyp == NULL)
789       return NULL;
790 
791    pdpyp->base.destroyDisplay = driswDestroyDisplay;
792    pdpyp->base.createScreen = driswCreateScreen;
793 
794    return &pdpyp->base;
795 }
796 
797 #endif /* GLX_DIRECT_RENDERING */
798