1 /*
2  * Mesa 3-D graphics library
3  * Version:  7.8
4  *
5  * Copyright (C) 2009-2010 Chia-I Wu <olv@0xlab.org>
6  *
7  * Permission is hereby granted, free of charge, to any person obtaining a
8  * copy of this software and associated documentation files (the "Software"),
9  * to deal in the Software without restriction, including without limitation
10  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
11  * and/or sell copies of the Software, and to permit persons to whom the
12  * Software is furnished to do so, subject to the following conditions:
13  *
14  * The above copyright notice and this permission notice shall be included
15  * in all copies or substantial portions of the Software.
16  *
17  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
20  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
22  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
23  * DEALINGS IN THE SOFTWARE.
24  */
25 
26 #include <unistd.h>
27 #include <fcntl.h>
28 #include <sys/types.h>
29 #include <sys/stat.h>
30 #include <xf86drm.h>
31 #include <X11/Xlibint.h>
32 #include <X11/extensions/XShm.h>
33 
34 #include "util/u_memory.h"
35 #include "egllog.h"
36 
37 #include "x11_screen.h"
38 #include "dri2.h"
39 #include "glxinit.h"
40 
41 struct x11_screen {
42    Display *dpy;
43    int number;
44 
45    /*
46     * This is used to fetch GLX visuals/fbconfigs.  It steals code from GLX.
47     * It might be better to rewrite the part in Xlib or XCB.
48     */
49    __GLXdisplayPrivate *glx_dpy;
50 
51    int dri_major, dri_minor;
52    char *dri_driver;
53    char *dri_device;
54    int dri_fd;
55 
56    x11_drawable_invalidate_buffers dri_invalidate_buffers;
57    void *dri_user_data;
58 
59    XVisualInfo *visuals;
60    int num_visuals;
61 
62    /* cached values for x11_drawable_get_depth */
63    Drawable last_drawable;
64    unsigned int last_depth;
65 };
66 
67 
68 /**
69  * Create a X11 screen.
70  */
71 struct x11_screen *
x11_screen_create(Display * dpy,int screen)72 x11_screen_create(Display *dpy, int screen)
73 {
74    struct x11_screen *xscr;
75 
76    if (screen >= ScreenCount(dpy))
77       return NULL;
78 
79    xscr = CALLOC_STRUCT(x11_screen);
80    if (xscr) {
81       xscr->dpy = dpy;
82       xscr->number = screen;
83 
84       xscr->dri_major = -1;
85       xscr->dri_fd = -1;
86    }
87    return xscr;
88 }
89 
90 /**
91  * Destroy a X11 screen.
92  */
93 void
x11_screen_destroy(struct x11_screen * xscr)94 x11_screen_destroy(struct x11_screen *xscr)
95 {
96    if (xscr->dri_fd >= 0)
97       close(xscr->dri_fd);
98    if (xscr->dri_driver)
99       Xfree(xscr->dri_driver);
100    if (xscr->dri_device)
101       Xfree(xscr->dri_device);
102 
103 #ifdef GLX_DIRECT_RENDERING
104    /* xscr->glx_dpy will be destroyed with the X display */
105    if (xscr->glx_dpy)
106       xscr->glx_dpy->xscr = NULL;
107 #endif
108 
109    if (xscr->visuals)
110       XFree(xscr->visuals);
111    FREE(xscr);
112 }
113 
114 #ifdef GLX_DIRECT_RENDERING
115 
116 static boolean
x11_screen_init_dri2(struct x11_screen * xscr)117 x11_screen_init_dri2(struct x11_screen *xscr)
118 {
119    if (xscr->dri_major < 0) {
120       int eventBase, errorBase;
121 
122       if (!DRI2QueryExtension(xscr->dpy, &eventBase, &errorBase) ||
123           !DRI2QueryVersion(xscr->dpy, &xscr->dri_major, &xscr->dri_minor))
124          xscr->dri_major = -1;
125    }
126    return (xscr->dri_major >= 0);
127 }
128 
129 static boolean
x11_screen_init_glx(struct x11_screen * xscr)130 x11_screen_init_glx(struct x11_screen *xscr)
131 {
132    if (!xscr->glx_dpy)
133       xscr->glx_dpy = __glXInitialize(xscr->dpy);
134    return (xscr->glx_dpy != NULL);
135 }
136 
137 #endif /* GLX_DIRECT_RENDERING */
138 
139 /**
140  * Return true if the screen supports the extension.
141  */
142 boolean
x11_screen_support(struct x11_screen * xscr,enum x11_screen_extension ext)143 x11_screen_support(struct x11_screen *xscr, enum x11_screen_extension ext)
144 {
145    boolean supported = FALSE;
146 
147    switch (ext) {
148    case X11_SCREEN_EXTENSION_XSHM:
149       supported = XShmQueryExtension(xscr->dpy);
150       break;
151 #ifdef GLX_DIRECT_RENDERING
152    case X11_SCREEN_EXTENSION_GLX:
153       supported = x11_screen_init_glx(xscr);
154       break;
155    case X11_SCREEN_EXTENSION_DRI2:
156       supported = x11_screen_init_dri2(xscr);
157       break;
158 #endif
159    default:
160       break;
161    }
162 
163    return supported;
164 }
165 
166 /**
167  * Return the X visuals.
168  */
169 const XVisualInfo *
x11_screen_get_visuals(struct x11_screen * xscr,int * num_visuals)170 x11_screen_get_visuals(struct x11_screen *xscr, int *num_visuals)
171 {
172    if (!xscr->visuals) {
173       XVisualInfo vinfo_template;
174       vinfo_template.screen = xscr->number;
175       xscr->visuals = XGetVisualInfo(xscr->dpy, VisualScreenMask,
176             &vinfo_template, &xscr->num_visuals);
177    }
178 
179    if (num_visuals)
180       *num_visuals = xscr->num_visuals;
181    return xscr->visuals;
182 }
183 
184 /**
185  * Return the depth of a drawable.
186  *
187  * Unlike other drawable functions, the drawable needs not be a DRI2 drawable.
188  */
189 uint
x11_drawable_get_depth(struct x11_screen * xscr,Drawable drawable)190 x11_drawable_get_depth(struct x11_screen *xscr, Drawable drawable)
191 {
192    unsigned int depth;
193 
194    if (drawable != xscr->last_drawable) {
195       Window root;
196       int x, y;
197       unsigned int w, h, border;
198       Status ok;
199 
200       ok = XGetGeometry(xscr->dpy, drawable, &root,
201             &x, &y, &w, &h, &border, &depth);
202       if (!ok)
203          depth = 0;
204 
205       xscr->last_drawable = drawable;
206       xscr->last_depth = depth;
207    }
208    else {
209       depth = xscr->last_depth;
210    }
211 
212    return depth;
213 }
214 
215 #ifdef GLX_DIRECT_RENDERING
216 
217 /**
218  * Return the GLX fbconfigs.
219  */
220 const __GLcontextModes *
x11_screen_get_glx_configs(struct x11_screen * xscr)221 x11_screen_get_glx_configs(struct x11_screen *xscr)
222 {
223    return (x11_screen_init_glx(xscr))
224       ? xscr->glx_dpy->screenConfigs[xscr->number]->configs
225       : NULL;
226 }
227 
228 /**
229  * Probe the screen for the DRI2 driver name.
230  */
231 const char *
x11_screen_probe_dri2(struct x11_screen * xscr,int * major,int * minor)232 x11_screen_probe_dri2(struct x11_screen *xscr, int *major, int *minor)
233 {
234    if (!x11_screen_init_dri2(xscr))
235       return NULL;
236 
237    /* get the driver name and the device name */
238    if (!xscr->dri_driver) {
239       if (!DRI2Connect(xscr->dpy, RootWindow(xscr->dpy, xscr->number),
240                &xscr->dri_driver, &xscr->dri_device))
241          xscr->dri_driver = xscr->dri_device = NULL;
242    }
243    if (major)
244       *major = xscr->dri_major;
245    if (minor)
246       *minor = xscr->dri_minor;
247 
248    return xscr->dri_driver;
249 }
250 
251 /**
252  * Enable DRI2 and returns the file descriptor of the DRM device.  The file
253  * descriptor will be closed automatically when the screen is destoryed.
254  */
255 int
x11_screen_enable_dri2(struct x11_screen * xscr,x11_drawable_invalidate_buffers invalidate_buffers,void * user_data)256 x11_screen_enable_dri2(struct x11_screen *xscr,
257                        x11_drawable_invalidate_buffers invalidate_buffers,
258                        void *user_data)
259 {
260    if (xscr->dri_fd < 0) {
261       int fd;
262       drm_magic_t magic;
263 
264       /* get the driver name and the device name first */
265       if (!x11_screen_probe_dri2(xscr, NULL, NULL))
266          return -1;
267 
268 #ifdef O_CLOEXEC
269       fd = open(xscr->dri_device, O_RDWR | O_CLOEXEC);
270       if (fd == -1 && errno == EINVAL)
271 #endif
272       {
273          fd = open(xscr->dri_device, O_RDWR);
274          if (fd != -1)
275             fcntl(fd, F_SETFD, fcntl(fd, F_GETFD) | FD_CLOEXEC);
276       }
277       if (fd < 0) {
278          _eglLog(_EGL_WARNING, "failed to open %s", xscr->dri_device);
279          return -1;
280       }
281 
282       memset(&magic, 0, sizeof(magic));
283       if (drmGetMagic(fd, &magic)) {
284          _eglLog(_EGL_WARNING, "failed to get magic");
285          close(fd);
286          return -1;
287       }
288 
289       if (!DRI2Authenticate(xscr->dpy,
290                RootWindow(xscr->dpy, xscr->number), magic)) {
291          _eglLog(_EGL_WARNING, "failed to authenticate magic");
292          close(fd);
293          return -1;
294       }
295 
296       if (!x11_screen_init_glx(xscr)) {
297          _eglLog(_EGL_WARNING, "failed to initialize GLX");
298          close(fd);
299          return -1;
300       }
301       if (xscr->glx_dpy->xscr) {
302          _eglLog(_EGL_WARNING,
303                "display is already managed by another x11 screen");
304          close(fd);
305          return -1;
306       }
307 
308       xscr->glx_dpy->xscr = xscr;
309       xscr->dri_invalidate_buffers = invalidate_buffers;
310       xscr->dri_user_data = user_data;
311 
312       xscr->dri_fd = fd;
313    }
314 
315    return xscr->dri_fd;
316 }
317 
318 char *
x11_screen_get_device_name(struct x11_screen * xscr)319 x11_screen_get_device_name(struct x11_screen *xscr)
320 {
321    return xscr->dri_device;
322 }
323 
324 int
x11_screen_authenticate(struct x11_screen * xscr,uint32_t id)325 x11_screen_authenticate(struct x11_screen *xscr, uint32_t id)
326 {
327    boolean authenticated;
328 
329    authenticated = DRI2Authenticate(xscr->dpy,
330          RootWindow(xscr->dpy, xscr->number), id);
331 
332    return authenticated ? 0 : -1;
333 }
334 
335 /**
336  * Create/Destroy the DRI drawable.
337  */
338 void
x11_drawable_enable_dri2(struct x11_screen * xscr,Drawable drawable,boolean on)339 x11_drawable_enable_dri2(struct x11_screen *xscr,
340                          Drawable drawable, boolean on)
341 {
342    if (on)
343       DRI2CreateDrawable(xscr->dpy, drawable);
344    else
345       DRI2DestroyDrawable(xscr->dpy, drawable);
346 }
347 
348 /**
349  * Copy between buffers of the DRI2 drawable.
350  */
351 void
x11_drawable_copy_buffers_region(struct x11_screen * xscr,Drawable drawable,int num_rects,const int * rects,int src_buf,int dst_buf)352 x11_drawable_copy_buffers_region(struct x11_screen *xscr, Drawable drawable,
353                                  int num_rects, const int *rects,
354                                  int src_buf, int dst_buf)
355 {
356    XserverRegion region;
357    XRectangle *rectangles = CALLOC(num_rects, sizeof(XRectangle));
358 
359    for (int i = 0; i < num_rects; i++) {
360       rectangles[i].x = rects[i * 4 + 0];
361       rectangles[i].y = rects[i * 4 + 1];
362       rectangles[i].width = rects[i * 4 + 2];
363       rectangles[i].height = rects[i * 4 + 3];
364    }
365 
366    region = XFixesCreateRegion(xscr->dpy, rectangles, num_rects);
367    DRI2CopyRegion(xscr->dpy, drawable, region, dst_buf, src_buf);
368    XFixesDestroyRegion(xscr->dpy, region);
369    FREE(rectangles);
370 }
371 
372 /**
373  * Get the buffers of the DRI2 drawable.  The returned array should be freed.
374  */
375 struct x11_drawable_buffer *
x11_drawable_get_buffers(struct x11_screen * xscr,Drawable drawable,int * width,int * height,unsigned int * attachments,boolean with_format,int num_ins,int * num_outs)376 x11_drawable_get_buffers(struct x11_screen *xscr, Drawable drawable,
377                          int *width, int *height, unsigned int *attachments,
378                          boolean with_format, int num_ins, int *num_outs)
379 {
380    DRI2Buffer *dri2bufs;
381 
382    if (with_format)
383       dri2bufs = DRI2GetBuffersWithFormat(xscr->dpy, drawable, width, height,
384             attachments, num_ins, num_outs);
385    else
386       dri2bufs = DRI2GetBuffers(xscr->dpy, drawable, width, height,
387             attachments, num_ins, num_outs);
388 
389    return (struct x11_drawable_buffer *) dri2bufs;
390 }
391 
392 /**
393  * Create a mode list of the given size.
394  */
395 __GLcontextModes *
x11_context_modes_create(unsigned count)396 x11_context_modes_create(unsigned count)
397 {
398    const size_t size = sizeof(__GLcontextModes);
399    __GLcontextModes *base = NULL;
400    __GLcontextModes **next;
401    unsigned i;
402 
403    next = &base;
404    for (i = 0; i < count; i++) {
405       *next = (__GLcontextModes *) CALLOC(1, size);
406       if (*next == NULL) {
407          x11_context_modes_destroy(base);
408          base = NULL;
409          break;
410       }
411       next = &((*next)->next);
412    }
413 
414    return base;
415 }
416 
417 /**
418  * Destroy a mode list.
419  */
420 void
x11_context_modes_destroy(__GLcontextModes * modes)421 x11_context_modes_destroy(__GLcontextModes *modes)
422 {
423    while (modes != NULL) {
424       __GLcontextModes *next = modes->next;
425       FREE(modes);
426       modes = next;
427    }
428 }
429 
430 /**
431  * Return the number of the modes in the mode list.
432  */
433 unsigned
x11_context_modes_count(const __GLcontextModes * modes)434 x11_context_modes_count(const __GLcontextModes *modes)
435 {
436    const __GLcontextModes *mode;
437    int count = 0;
438    for (mode = modes; mode; mode = mode->next)
439       count++;
440    return count;
441 }
442 
443 extern void
444 dri2InvalidateBuffers(Display *dpy, XID drawable);
445 
446 /**
447  * This is called from src/glx/dri2.c.
448  */
449 void
dri2InvalidateBuffers(Display * dpy,XID drawable)450 dri2InvalidateBuffers(Display *dpy, XID drawable)
451 {
452    __GLXdisplayPrivate *priv = __glXInitialize(dpy);
453    struct x11_screen *xscr = NULL;
454 
455    if (priv && priv->xscr)
456       xscr = priv->xscr;
457    if (!xscr || !xscr->dri_invalidate_buffers)
458       return;
459 
460    xscr->dri_invalidate_buffers(xscr, drawable, xscr->dri_user_data);
461 }
462 
463 extern unsigned
464 dri2GetSwapEventType(Display *dpy, XID drawable);
465 
466 extern void *
467 dri2GetGlxDrawableFromXDrawableId(Display *dpy, XID id);
468 
469 extern void *
470 GetGLXDrawable(Display *dpy, XID drawable);
471 
472 /**
473  * This is also called from src/glx/dri2.c.
474  */
dri2GetSwapEventType(Display * dpy,XID drawable)475 unsigned dri2GetSwapEventType(Display *dpy, XID drawable)
476 {
477    return 0;
478 }
479 
480 void *
dri2GetGlxDrawableFromXDrawableId(Display * dpy,XID id)481 dri2GetGlxDrawableFromXDrawableId(Display *dpy, XID id)
482 {
483    return NULL;
484 }
485 
486 void *
GetGLXDrawable(Display * dpy,XID drawable)487 GetGLXDrawable(Display *dpy, XID drawable)
488 {
489    return NULL;
490 }
491 
492 #endif /* GLX_DIRECT_RENDERING */
493