0,0 → 1,223 |
/* |
* va_wayland_drm.c - Wayland/DRM helpers |
* |
* Copyright (c) 2012 Intel Corporation. All Rights Reserved. |
* |
* Permission is hereby granted, free of charge, to any person obtaining a |
* copy of this software and associated documentation files (the |
* "Software"), to deal in the Software without restriction, including |
* without limitation the rights to use, copy, modify, merge, publish, |
* distribute, sub license, and/or sell copies of the Software, and to |
* permit persons to whom the Software is furnished to do so, subject to |
* the following conditions: |
* |
* The above copyright notice and this permission notice (including the |
* next paragraph) shall be included in all copies or substantial portions |
* of the Software. |
* |
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS |
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF |
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. |
* IN NO EVENT SHALL INTEL AND/OR ITS SUPPLIERS BE LIABLE FOR |
* ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, |
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE |
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
*/ |
|
#include "sysdeps.h" |
#include <unistd.h> |
#include <errno.h> |
#include <fcntl.h> |
#include <dlfcn.h> |
#include <sys/stat.h> |
#include <xf86drm.h> |
#include "va_drmcommon.h" |
#include "drm/va_drm_utils.h" |
#include "va_wayland_drm.h" |
#include "va_wayland_private.h" |
#include "wayland-drm-client-protocol.h" |
|
/* XXX: Wayland/DRM support currently lives in Mesa libEGL.so.* library */ |
#define LIBWAYLAND_DRM_NAME "libEGL.so.1" |
|
typedef struct va_wayland_drm_context { |
struct va_wayland_context base; |
void *handle; |
struct wl_drm *drm; |
struct wl_registry *registry; |
void *drm_interface; |
unsigned int is_authenticated : 1; |
} VADisplayContextWaylandDRM; |
|
static void |
drm_handle_device(void *data, struct wl_drm *drm, const char *device) |
{ |
VADisplayContextP const pDisplayContext = data; |
VADriverContextP const ctx = pDisplayContext->pDriverContext; |
VADisplayContextWaylandDRM * const wl_drm_ctx = pDisplayContext->opaque; |
struct drm_state * const drm_state = ctx->drm_state; |
drm_magic_t magic; |
struct stat st; |
|
if (stat(device, &st) < 0) { |
va_wayland_error("failed to identify %s: %s (errno %d)", |
device, strerror(errno), errno); |
return; |
} |
|
if (!S_ISCHR(st.st_mode)) { |
va_wayland_error("%s is not a device", device); |
return; |
} |
|
drm_state->fd = open(device, O_RDWR); |
if (drm_state->fd < 0) { |
va_wayland_error("failed to open %s: %s (errno %d)", |
device, strerror(errno), errno); |
return; |
} |
|
drmGetMagic(drm_state->fd, &magic); |
wl_drm_authenticate(wl_drm_ctx->drm, magic); |
} |
|
static void |
drm_handle_format(void *data, struct wl_drm *drm, uint32_t format) |
{ |
} |
|
static void |
drm_handle_authenticated(void *data, struct wl_drm *drm) |
{ |
VADisplayContextP const pDisplayContext = data; |
VADriverContextP const ctx = pDisplayContext->pDriverContext; |
VADisplayContextWaylandDRM * const wl_drm_ctx = pDisplayContext->opaque; |
struct drm_state * const drm_state = ctx->drm_state; |
|
wl_drm_ctx->is_authenticated = 1; |
drm_state->auth_type = VA_DRM_AUTH_CUSTOM; |
} |
|
static const struct wl_drm_listener drm_listener = { |
drm_handle_device, |
drm_handle_format, |
drm_handle_authenticated |
}; |
|
static VAStatus |
va_DisplayContextGetDriverName( |
VADisplayContextP pDisplayContext, |
char **driver_name_ptr |
) |
{ |
VADriverContextP const ctx = pDisplayContext->pDriverContext; |
|
return VA_DRM_GetDriverName(ctx, driver_name_ptr); |
} |
|
void |
va_wayland_drm_destroy(VADisplayContextP pDisplayContext) |
{ |
VADriverContextP const ctx = pDisplayContext->pDriverContext; |
struct va_wayland_drm_context * const wl_drm_ctx = pDisplayContext->opaque; |
struct drm_state * const drm_state = ctx->drm_state; |
|
if (wl_drm_ctx->drm) { |
wl_drm_destroy(wl_drm_ctx->drm); |
wl_drm_ctx->drm = NULL; |
} |
wl_drm_ctx->is_authenticated = 0; |
|
if (wl_drm_ctx->handle) { |
dlclose(wl_drm_ctx->handle); |
wl_drm_ctx->handle = NULL; |
} |
|
if (drm_state) { |
if (drm_state->fd >= 0) { |
close(drm_state->fd); |
drm_state->fd = -1; |
} |
free(ctx->drm_state); |
ctx->drm_state = NULL; |
} |
} |
|
static void |
registry_handle_global( |
void *data, |
struct wl_registry *registry, |
uint32_t id, |
const char *interface, |
uint32_t version |
) |
{ |
struct va_wayland_drm_context *wl_drm_ctx = data; |
|
if (strcmp(interface, "wl_drm") == 0) { |
wl_drm_ctx->drm = |
wl_registry_bind(wl_drm_ctx->registry, id, wl_drm_ctx->drm_interface, 1); |
} |
} |
|
static const struct wl_registry_listener registry_listener = { |
registry_handle_global, |
NULL, |
}; |
|
bool |
va_wayland_drm_create(VADisplayContextP pDisplayContext) |
{ |
VADriverContextP const ctx = pDisplayContext->pDriverContext; |
struct va_wayland_drm_context *wl_drm_ctx; |
struct drm_state *drm_state; |
uint32_t id; |
|
wl_drm_ctx = malloc(sizeof(*wl_drm_ctx)); |
if (!wl_drm_ctx) |
return false; |
wl_drm_ctx->base.destroy = va_wayland_drm_destroy; |
wl_drm_ctx->handle = NULL; |
wl_drm_ctx->drm = NULL; |
wl_drm_ctx->drm_interface = NULL; |
wl_drm_ctx->is_authenticated = 0; |
pDisplayContext->opaque = wl_drm_ctx; |
pDisplayContext->vaGetDriverName = va_DisplayContextGetDriverName; |
|
drm_state = calloc(1, sizeof(struct drm_state)); |
if (!drm_state) |
return false; |
drm_state->fd = -1; |
drm_state->auth_type = 0; |
ctx->drm_state = drm_state; |
|
wl_drm_ctx->handle = dlopen(LIBWAYLAND_DRM_NAME, RTLD_LAZY|RTLD_LOCAL); |
if (!wl_drm_ctx->handle) |
return false; |
|
wl_drm_ctx->drm_interface = |
dlsym(wl_drm_ctx->handle, "wl_drm_interface"); |
if (!wl_drm_ctx->drm_interface) |
return false; |
|
wl_drm_ctx->registry = wl_display_get_registry(ctx->native_dpy); |
wl_registry_add_listener(wl_drm_ctx->registry, ®istry_listener, wl_drm_ctx); |
wl_display_roundtrip(ctx->native_dpy); |
|
/* registry_handle_global should have been called by the |
* wl_display_roundtrip above |
*/ |
|
if (!wl_drm_ctx->drm) |
return false; |
|
wl_drm_add_listener(wl_drm_ctx->drm, &drm_listener, pDisplayContext); |
wl_display_roundtrip(ctx->native_dpy); |
if (drm_state->fd < 0) |
return false; |
|
wl_display_roundtrip(ctx->native_dpy); |
if (!wl_drm_ctx->is_authenticated) |
return false; |
return true; |
} |