native_drm.c revision b89bca6d8b84e7922f44258c41e37997775f98c9
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   const 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
70wayland_drm_display_destroy(struct native_display *ndpy)
71{
72   struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
73
74   if (drmdpy->fd)
75      close(drmdpy->fd);
76   if (drmdpy->wl_drm)
77      wl_drm_destroy(drmdpy->wl_drm);
78   if (drmdpy->device_name)
79      FREE(drmdpy->device_name);
80   if (drmdpy->base.config)
81      FREE(drmdpy->base.config);
82   if (drmdpy->base.own_dpy)
83      wl_display_destroy(drmdpy->base.dpy);
84
85   ndpy_uninit(ndpy);
86
87   FREE(drmdpy);
88}
89
90static struct wl_buffer *
91wayland_create_drm_buffer(struct wayland_display *display,
92                          struct wayland_surface *surface,
93                          enum native_attachment attachment)
94{
95   struct wayland_drm_display *drmdpy = (struct wayland_drm_display *) display;
96   struct pipe_screen *screen = drmdpy->base.base.screen;
97   struct pipe_resource *resource;
98   struct winsys_handle wsh;
99   uint width, height;
100   struct wl_visual *visual;
101
102   resource = resource_surface_get_single_resource(surface->rsurf, attachment);
103   resource_surface_get_size(surface->rsurf, &width, &height);
104
105   wsh.type = DRM_API_HANDLE_TYPE_SHARED;
106   screen->resource_get_handle(screen, resource, &wsh);
107
108   pipe_resource_reference(&resource, NULL);
109
110   switch (surface->type) {
111   case WL_WINDOW_SURFACE:
112      visual = surface->win->visual;
113      break;
114   case WL_PIXMAP_SURFACE:
115      visual = surface->pix->visual;
116      break;
117   default:
118      return NULL;
119   }
120
121   return wl_drm_create_buffer(drmdpy->wl_drm, wsh.handle,
122                               width, height, wsh.stride, visual);
123}
124
125static void
126drm_handle_device(void *data, struct wl_drm *drm, const char *device)
127{
128   struct wayland_drm_display *drmdpy = data;
129   drm_magic_t magic;
130
131   drmdpy->device_name = strdup(device);
132   if (!drmdpy->device_name)
133      return;
134
135   drmdpy->fd = open(drmdpy->device_name, O_RDWR);
136   if (drmdpy->fd == -1) {
137      _eglLog(_EGL_WARNING, "wayland-egl: could not open %s (%s)",
138              drmdpy->device_name, strerror(errno));
139      return;
140   }
141
142   drmGetMagic(drmdpy->fd, &magic);
143   wl_drm_authenticate(drmdpy->wl_drm, magic);
144}
145
146static void
147drm_handle_authenticated(void *data, struct wl_drm *drm)
148{
149   struct wayland_drm_display *drmdpy = data;
150
151   drmdpy->authenticated = true;
152}
153
154static const struct wl_drm_listener drm_listener = {
155   drm_handle_device,
156   drm_handle_authenticated
157};
158
159static boolean
160wayland_drm_display_init_screen(struct native_display *ndpy)
161{
162   struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
163   uint32_t id;
164
165   id = wl_display_get_global(drmdpy->base.dpy, "wl_drm", 1);
166   if (id == 0)
167      wl_display_roundtrip(drmdpy->base.dpy);
168   id = wl_display_get_global(drmdpy->base.dpy, "wl_drm", 1);
169   if (id == 0)
170      return FALSE;
171
172   drmdpy->wl_drm = wl_display_bind(drmdpy->base.dpy, id, &wl_drm_interface);
173   if (!drmdpy->wl_drm)
174      return FALSE;
175
176   wl_drm_add_listener(drmdpy->wl_drm, &drm_listener, drmdpy);
177   wl_display_roundtrip(drmdpy->base.dpy);
178   if (drmdpy->fd == -1)
179      return FALSE;
180
181   wl_display_roundtrip(drmdpy->base.dpy);
182   if (!drmdpy->authenticated)
183      return FALSE;
184
185   drmdpy->base.base.screen =
186      drmdpy->event_handler->new_drm_screen(&drmdpy->base.base,
187                                            NULL, drmdpy->fd);
188   if (!drmdpy->base.base.screen) {
189      _eglLog(_EGL_WARNING, "failed to create DRM screen");
190      return FALSE;
191   }
192
193   return TRUE;
194}
195
196static struct native_display_buffer wayland_drm_display_buffer = {
197   /* use the helpers */
198   drm_display_import_native_buffer,
199   drm_display_export_native_buffer
200};
201
202static int
203wayland_drm_display_authenticate(void *user_data, uint32_t magic)
204{
205   struct native_display *ndpy = user_data;
206   struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
207   boolean current_authenticate, authenticated;
208
209   current_authenticate = drmdpy->authenticated;
210
211   wl_drm_authenticate(drmdpy->wl_drm, magic);
212   wl_display_roundtrip(drmdpy->base.dpy);
213   authenticated = drmdpy->authenticated;
214
215   drmdpy->authenticated = current_authenticate;
216
217   return authenticated ? 0 : -1;
218}
219
220static struct wayland_drm_callbacks wl_drm_callbacks = {
221   wayland_drm_display_authenticate,
222   egl_g3d_wl_drm_helper_reference_buffer,
223   egl_g3d_wl_drm_helper_unreference_buffer
224};
225
226static boolean
227wayland_drm_display_bind_wayland_display(struct native_display *ndpy,
228                                         struct wl_display *wl_dpy)
229{
230   struct wayland_drm_display *drmdpy = wayland_drm_display(ndpy);
231
232   if (drmdpy->wl_server_drm)
233      return FALSE;
234
235   drmdpy->wl_server_drm =
236      wayland_drm_init(wl_dpy, drmdpy->device_name,
237                       &wl_drm_callbacks, ndpy);
238
239   if (!drmdpy->wl_server_drm)
240      return FALSE;
241
242   return TRUE;
243}
244
245static boolean
246wayland_drm_display_unbind_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   wayland_drm_uninit(drmdpy->wl_server_drm);
255   drmdpy->wl_server_drm = NULL;
256
257   return TRUE;
258}
259
260static struct native_display_wayland_bufmgr wayland_drm_display_wayland_bufmgr = {
261   wayland_drm_display_bind_wayland_display,
262   wayland_drm_display_unbind_wayland_display,
263   egl_g3d_wl_drm_common_wl_buffer_get_resource
264};
265
266
267struct wayland_display *
268wayland_create_drm_display(struct wl_display *dpy,
269                           const struct native_event_handler *event_handler)
270{
271   struct wayland_drm_display *drmdpy;
272
273   drmdpy = CALLOC_STRUCT(wayland_drm_display);
274   if (!drmdpy)
275      return NULL;
276
277   drmdpy->event_handler = event_handler;
278
279   drmdpy->base.dpy = dpy;
280   if (!drmdpy->base.dpy) {
281      wayland_drm_display_destroy(&drmdpy->base.base);
282      return NULL;
283   }
284
285   drmdpy->base.base.init_screen = wayland_drm_display_init_screen;
286   drmdpy->base.base.destroy = wayland_drm_display_destroy;
287   drmdpy->base.base.buffer = &wayland_drm_display_buffer;
288   drmdpy->base.base.wayland_bufmgr = &wayland_drm_display_wayland_bufmgr;
289
290   drmdpy->base.create_buffer = wayland_create_drm_buffer;
291
292   return &drmdpy->base;
293}
294
295/* vim: set sw=3 ts=8 sts=3 expandtab: */
296