native_drm.c revision febf5e4147612641fc23f202d5813958bee3af13
1/* 2 * Mesa 3-D graphics library 3 * Version: 7.11 4 * 5 * Copyright (C) 2011 Benjamin Franzke <benjaminfranzke@googlemail.com> 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 "util/u_memory.h" 27#include "util/u_inlines.h" 28 29#include "pipe/p_compiler.h" 30#include "pipe/p_screen.h" 31#include "pipe/p_context.h" 32#include "pipe/p_state.h" 33#include "state_tracker/drm_driver.h" 34 35#include "egllog.h" 36#include <errno.h> 37 38#include "native_wayland.h" 39 40#include <wayland-client.h> 41#include "wayland-drm-client-protocol.h" 42#include "wayland-egl-priv.h" 43 44#include "common/native_wayland_drm_bufmgr_helper.h" 45 46#include <xf86drm.h> 47#include <sys/types.h> 48#include <sys/stat.h> 49#include <fcntl.h> 50 51struct wayland_drm_display { 52 struct wayland_display base; 53 54 struct native_event_handler *event_handler; 55 56 struct wl_drm *wl_drm; 57 struct wl_drm *wl_server_drm; /* for EGL_WL_bind_wayland_display */ 58 int fd; 59 char *device_name; 60 boolean authenticated; 61}; 62 63static INLINE struct wayland_drm_display * 64wayland_drm_display(const struct native_display *ndpy) 65{ 66 return (struct wayland_drm_display *) ndpy; 67} 68 69static void 70sync_callback(void *data) 71{ 72 int *done = data; 73 74 *done = 1; 75} 76 77static void 78force_roundtrip(struct wl_display *display) 79{ 80 int done = 0; 81 82 wl_display_sync_callback(display, sync_callback, &done); 83 wl_display_iterate(display, WL_DISPLAY_WRITABLE); 84 while (!done) 85 wl_display_iterate(display, WL_DISPLAY_READABLE); 86} 87 88static void 89wayland_drm_display_destroy(struct native_display *ndpy) 90{ 91 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy); 92 93 if (drmdpy->fd) 94 close(drmdpy->fd); 95 if (drmdpy->wl_drm) 96 wl_drm_destroy(drmdpy->wl_drm); 97 if (drmdpy->device_name) 98 FREE(drmdpy->device_name); 99 if (drmdpy->base.config) 100 FREE(drmdpy->base.config); 101 if (drmdpy->base.own_dpy) 102 wl_display_destroy(drmdpy->base.dpy); 103 104 ndpy_uninit(ndpy); 105 106 FREE(drmdpy); 107} 108 109static struct wl_buffer * 110wayland_create_drm_buffer(struct wayland_display *display, 111 struct wayland_surface *surface, 112 enum native_attachment attachment) 113{ 114 struct wayland_drm_display *drmdpy = (struct wayland_drm_display *) display; 115 struct pipe_screen *screen = drmdpy->base.base.screen; 116 struct pipe_resource *resource; 117 struct winsys_handle wsh; 118 uint width, height; 119 struct wl_visual *visual; 120 121 resource = resource_surface_get_single_resource(surface->rsurf, attachment); 122 resource_surface_get_size(surface->rsurf, &width, &height); 123 124 wsh.type = DRM_API_HANDLE_TYPE_SHARED; 125 screen->resource_get_handle(screen, resource, &wsh); 126 127 pipe_resource_reference(&resource, NULL); 128 129 switch (surface->type) { 130 case WL_WINDOW_SURFACE: 131 visual = surface->win->visual; 132 break; 133 case WL_PIXMAP_SURFACE: 134 visual = surface->pix->visual; 135 break; 136 default: 137 return NULL; 138 } 139 140 return wl_drm_create_buffer(drmdpy->wl_drm, wsh.handle, 141 width, height, wsh.stride, visual); 142} 143 144static void 145drm_handle_device(void *data, struct wl_drm *drm, const char *device) 146{ 147 struct wayland_drm_display *drmdpy = data; 148 drm_magic_t magic; 149 150 drmdpy->device_name = strdup(device); 151 if (!drmdpy->device_name) 152 return; 153 154 drmdpy->fd = open(drmdpy->device_name, O_RDWR); 155 if (drmdpy->fd == -1) { 156 _eglLog(_EGL_WARNING, "wayland-egl: could not open %s (%s)", 157 drmdpy->device_name, strerror(errno)); 158 return; 159 } 160 161 drmGetMagic(drmdpy->fd, &magic); 162 wl_drm_authenticate(drmdpy->wl_drm, magic); 163} 164 165static void 166drm_handle_authenticated(void *data, struct wl_drm *drm) 167{ 168 struct wayland_drm_display *drmdpy = data; 169 170 drmdpy->authenticated = true; 171} 172 173static const struct wl_drm_listener drm_listener = { 174 drm_handle_device, 175 drm_handle_authenticated 176}; 177 178static boolean 179wayland_drm_display_init_screen(struct native_display *ndpy) 180{ 181 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy); 182 uint32_t id; 183 184 id = wl_display_get_global(drmdpy->base.dpy, "wl_drm", 1); 185 if (id == 0) 186 force_roundtrip(drmdpy->base.dpy); 187 id = wl_display_get_global(drmdpy->base.dpy, "wl_drm", 1); 188 if (id == 0) 189 return FALSE; 190 191 drmdpy->wl_drm = wl_drm_create(drmdpy->base.dpy, id, 1); 192 if (!drmdpy->wl_drm) 193 return FALSE; 194 195 wl_drm_add_listener(drmdpy->wl_drm, &drm_listener, drmdpy); 196 force_roundtrip(drmdpy->base.dpy); 197 if (drmdpy->fd == -1) 198 return FALSE; 199 200 force_roundtrip(drmdpy->base.dpy); 201 if (!drmdpy->authenticated) 202 return FALSE; 203 204 drmdpy->base.base.screen = 205 drmdpy->event_handler->new_drm_screen(&drmdpy->base.base, 206 NULL, drmdpy->fd); 207 if (!drmdpy->base.base.screen) { 208 _eglLog(_EGL_WARNING, "failed to create DRM screen"); 209 return FALSE; 210 } 211 212 return TRUE; 213} 214 215static struct native_display_buffer wayland_drm_display_buffer = { 216 /* use the helpers */ 217 drm_display_import_native_buffer, 218 drm_display_export_native_buffer 219}; 220 221static int 222wayland_drm_display_authenticate(void *user_data, uint32_t magic) 223{ 224 struct native_display *ndpy = user_data; 225 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy); 226 boolean current_authenticate, authenticated; 227 228 current_authenticate = drmdpy->authenticated; 229 230 wl_drm_authenticate(drmdpy->wl_drm, magic); 231 force_roundtrip(drmdpy->base.dpy); 232 authenticated = drmdpy->authenticated; 233 234 drmdpy->authenticated = current_authenticate; 235 236 return authenticated ? 0 : -1; 237} 238 239static struct wayland_drm_callbacks wl_drm_callbacks = { 240 wayland_drm_display_authenticate, 241 egl_g3d_wl_drm_helper_reference_buffer, 242 egl_g3d_wl_drm_helper_unreference_buffer 243}; 244 245static boolean 246wayland_drm_display_bind_wayland_display(struct native_display *ndpy, 247 struct wl_display *wl_dpy) 248{ 249 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy); 250 251 if (drmdpy->wl_server_drm) 252 return FALSE; 253 254 drmdpy->wl_server_drm = 255 wayland_drm_init(wl_dpy, drmdpy->device_name, 256 &wl_drm_callbacks, ndpy); 257 258 if (!drmdpy->wl_server_drm) 259 return FALSE; 260 261 return TRUE; 262} 263 264static boolean 265wayland_drm_display_unbind_wayland_display(struct native_display *ndpy, 266 struct wl_display *wl_dpy) 267{ 268 struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy); 269 270 if (!drmdpy->wl_server_drm) 271 return FALSE; 272 273 wayland_drm_uninit(drmdpy->wl_server_drm); 274 drmdpy->wl_server_drm = NULL; 275 276 return TRUE; 277} 278 279static struct native_display_wayland_bufmgr wayland_drm_display_wayland_bufmgr = { 280 wayland_drm_display_bind_wayland_display, 281 wayland_drm_display_unbind_wayland_display, 282 egl_g3d_wl_drm_common_wl_buffer_get_resource 283}; 284 285 286struct wayland_display * 287wayland_create_drm_display(struct wl_display *dpy, 288 struct native_event_handler *event_handler, 289 void *user_data) 290{ 291 struct wayland_drm_display *drmdpy; 292 293 drmdpy = CALLOC_STRUCT(wayland_drm_display); 294 if (!drmdpy) 295 return NULL; 296 297 drmdpy->event_handler = event_handler; 298 drmdpy->base.base.user_data = user_data; 299 300 drmdpy->base.dpy = dpy; 301 if (!drmdpy->base.dpy) { 302 wayland_drm_display_destroy(&drmdpy->base.base); 303 return NULL; 304 } 305 306 if (!wayland_drm_display_init_screen(&drmdpy->base.base)) { 307 wayland_drm_display_destroy(&drmdpy->base.base); 308 return NULL; 309 } 310 drmdpy->base.base.destroy = wayland_drm_display_destroy; 311 drmdpy->base.base.buffer = &wayland_drm_display_buffer; 312 drmdpy->base.base.wayland_bufmgr = &wayland_drm_display_wayland_bufmgr; 313 314 drmdpy->base.create_buffer = wayland_create_drm_buffer; 315 316 return &drmdpy->base; 317} 318 319/* vim: set sw=3 ts=8 sts=3 expandtab: */ 320