2 * Mesa 3-D graphics library
4 * Copyright (C) 2011 Benjamin Franzke <benjaminfranzke@googlemail.com>
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the "Software"),
8 * to deal in the Software without restriction, including without limitation
9 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10 * and/or sell copies of the Software, and to permit persons to whom the
11 * Software is furnished to do so, subject to the following conditions:
13 * The above copyright notice and this permission notice shall be included
14 * in all copies or substantial portions of the Software.
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22 * DEALINGS IN THE SOFTWARE.
25 #include "util/u_memory.h"
26 #include "util/u_inlines.h"
28 #include "pipe/p_compiler.h"
29 #include "pipe/p_screen.h"
30 #include "pipe/p_context.h"
31 #include "pipe/p_state.h"
32 #include "state_tracker/drm_driver.h"
37 #include "native_wayland.h"
39 #include <wayland-client.h>
40 #include "wayland-drm-client-protocol.h"
41 #include "wayland-egl-priv.h"
43 #include "common/native_wayland_drm_bufmgr_helper.h"
46 #include <sys/types.h>
50 struct wayland_drm_display {
51 struct wayland_display base;
53 const struct native_event_handler *event_handler;
55 struct wl_drm *wl_drm;
56 struct wl_drm *wl_server_drm; /* for EGL_WL_bind_wayland_display */
59 boolean authenticated;
62 static INLINE struct wayland_drm_display *
63 wayland_drm_display(const struct native_display *ndpy)
65 return (struct wayland_drm_display *) ndpy;
69 wayland_drm_display_destroy(struct native_display *ndpy)
71 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
74 wl_drm_destroy(drmdpy->wl_drm);
75 FREE(drmdpy->device_name);
76 FREE(drmdpy->base.configs);
77 if (drmdpy->base.own_dpy)
78 wl_display_disconnect(drmdpy->base.dpy);
88 static struct wl_buffer *
89 wayland_create_drm_buffer(struct wayland_display *display,
90 struct wayland_surface *surface,
91 enum native_attachment attachment)
93 struct wayland_drm_display *drmdpy = (struct wayland_drm_display *) display;
94 struct pipe_screen *screen = drmdpy->base.base.screen;
95 struct pipe_resource *resource;
96 struct winsys_handle wsh;
98 enum wl_drm_format format;
100 resource = resource_surface_get_single_resource(surface->rsurf, attachment);
101 resource_surface_get_size(surface->rsurf, &width, &height);
103 wsh.type = DRM_API_HANDLE_TYPE_SHARED;
104 screen->resource_get_handle(screen, resource, &wsh);
106 pipe_resource_reference(&resource, NULL);
108 switch (surface->color_format) {
109 case PIPE_FORMAT_B8G8R8A8_UNORM:
110 format = WL_DRM_FORMAT_ARGB8888;
112 case PIPE_FORMAT_B8G8R8X8_UNORM:
113 format = WL_DRM_FORMAT_XRGB8888;
120 return wl_drm_create_buffer(drmdpy->wl_drm, wsh.handle,
121 width, height, wsh.stride, format);
125 drm_handle_device(void *data, struct wl_drm *drm, const char *device)
127 struct wayland_drm_display *drmdpy = data;
130 drmdpy->device_name = strdup(device);
131 if (!drmdpy->device_name)
135 drmdpy->fd = open(drmdpy->device_name, O_RDWR | O_CLOEXEC);
136 if (drmdpy->fd == -1 && errno == EINVAL)
139 drmdpy->fd = open(drmdpy->device_name, O_RDWR);
140 if (drmdpy->fd != -1)
141 fcntl(drmdpy->fd, F_SETFD, fcntl(drmdpy->fd, F_GETFD) | FD_CLOEXEC);
143 if (drmdpy->fd == -1) {
144 _eglLog(_EGL_WARNING, "wayland-egl: could not open %s (%s)",
145 drmdpy->device_name, strerror(errno));
149 drmGetMagic(drmdpy->fd, &magic);
150 wl_drm_authenticate(drmdpy->wl_drm, magic);
154 drm_handle_format(void *data, struct wl_drm *drm, uint32_t format)
156 struct wayland_drm_display *drmdpy = data;
159 case WL_DRM_FORMAT_ARGB8888:
160 drmdpy->base.formats |= HAS_ARGB8888;
162 case WL_DRM_FORMAT_XRGB8888:
163 drmdpy->base.formats |= HAS_XRGB8888;
169 drm_handle_authenticated(void *data, struct wl_drm *drm)
171 struct wayland_drm_display *drmdpy = data;
173 drmdpy->authenticated = true;
176 static const struct wl_drm_listener drm_listener = {
179 drm_handle_authenticated
183 registry_handle_global(void *data, struct wl_registry *registry, uint32_t name,
184 const char *interface, uint32_t version)
186 struct wayland_drm_display *drmdpy = data;
188 if (strcmp(interface, "wl_drm") == 0) {
189 drmdpy->wl_drm = wl_registry_bind(registry, name, &wl_drm_interface, 1);
190 wl_drm_add_listener(drmdpy->wl_drm, &drm_listener, drmdpy);
194 static const struct wl_registry_listener registry_listener = {
195 registry_handle_global
199 wayland_drm_display_init_screen(struct native_display *ndpy)
201 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
203 drmdpy->base.queue = wl_display_create_queue(drmdpy->base.dpy);
204 drmdpy->base.registry = wl_display_get_registry(drmdpy->base.dpy);
205 wl_proxy_set_queue((struct wl_proxy *) drmdpy->base.registry,
207 wl_registry_add_listener(drmdpy->base.registry, ®istry_listener, drmdpy);
208 if (wayland_roundtrip(&drmdpy->base) < 0 || drmdpy->wl_drm == NULL)
211 wl_drm_add_listener(drmdpy->wl_drm, &drm_listener, drmdpy);
212 if (wayland_roundtrip(&drmdpy->base) < 0 || drmdpy->fd == -1)
215 if (wayland_roundtrip(&drmdpy->base) < 0 || !drmdpy->authenticated)
218 if (drmdpy->base.formats == 0)
221 drmdpy->base.base.screen =
222 drmdpy->event_handler->new_drm_screen(&drmdpy->base.base,
224 if (!drmdpy->base.base.screen) {
225 _eglLog(_EGL_WARNING, "failed to create DRM screen");
232 static struct native_display_buffer wayland_drm_display_buffer = {
233 /* use the helpers */
234 drm_display_import_native_buffer,
235 drm_display_export_native_buffer
239 wayland_drm_display_authenticate(void *user_data, uint32_t magic)
241 struct native_display *ndpy = user_data;
242 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
243 boolean current_authenticate, authenticated;
245 current_authenticate = drmdpy->authenticated;
247 wl_drm_authenticate(drmdpy->wl_drm, magic);
248 wl_display_roundtrip(drmdpy->base.dpy);
249 authenticated = drmdpy->authenticated;
251 drmdpy->authenticated = current_authenticate;
253 return authenticated ? 0 : -1;
256 static struct wayland_drm_callbacks wl_drm_callbacks = {
257 wayland_drm_display_authenticate,
258 egl_g3d_wl_drm_helper_reference_buffer,
259 egl_g3d_wl_drm_helper_unreference_buffer
263 wayland_drm_display_bind_wayland_display(struct native_display *ndpy,
264 struct wl_display *wl_dpy)
266 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
268 if (drmdpy->wl_server_drm)
271 drmdpy->wl_server_drm =
272 wayland_drm_init(wl_dpy, drmdpy->device_name,
273 &wl_drm_callbacks, ndpy, 0);
275 if (!drmdpy->wl_server_drm)
282 wayland_drm_display_unbind_wayland_display(struct native_display *ndpy,
283 struct wl_display *wl_dpy)
285 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
287 if (!drmdpy->wl_server_drm)
290 wayland_drm_uninit(drmdpy->wl_server_drm);
291 drmdpy->wl_server_drm = NULL;
296 static struct native_display_wayland_bufmgr wayland_drm_display_wayland_bufmgr = {
297 wayland_drm_display_bind_wayland_display,
298 wayland_drm_display_unbind_wayland_display,
299 egl_g3d_wl_drm_common_wl_buffer_get_resource,
300 egl_g3d_wl_drm_common_query_buffer
304 struct wayland_display *
305 wayland_create_drm_display(struct wl_display *dpy,
306 const struct native_event_handler *event_handler)
308 struct wayland_drm_display *drmdpy;
310 drmdpy = CALLOC_STRUCT(wayland_drm_display);
314 drmdpy->event_handler = event_handler;
316 drmdpy->base.dpy = dpy;
317 if (!drmdpy->base.dpy) {
318 wayland_drm_display_destroy(&drmdpy->base.base);
322 drmdpy->base.base.init_screen = wayland_drm_display_init_screen;
323 drmdpy->base.base.destroy = wayland_drm_display_destroy;
324 drmdpy->base.base.buffer = &wayland_drm_display_buffer;
325 drmdpy->base.base.wayland_bufmgr = &wayland_drm_display_wayland_bufmgr;
327 drmdpy->base.create_buffer = wayland_create_drm_buffer;
329 return &drmdpy->base;
332 /* vim: set sw=3 ts=8 sts=3 expandtab: */