efl/src/lib/ecore_drm/ecore_drm_device.c

740 lines
19 KiB
C

/* Portions of this code have been derived from Weston
*
* Copyright © 2008-2012 Kristian Høgsberg
* Copyright © 2010-2012 Intel Corporation
* Copyright © 2010-2011 Benjamin Franzke
* Copyright © 2011-2012 Collabora, Ltd.
* Copyright © 2010 Red Hat <mjg@redhat.com>
*
* 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, sublicense,
* 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 NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS 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 "ecore_drm_private.h"
#include <dlfcn.h>
#define INSIDE(x, y, xx, yy, ww, hh) \
(((x) < ((xx) + (ww))) && ((y) < ((yy) + (hh))) && \
((x) >= (xx)) && ((y) >= (yy)))
static Eina_List *drm_devices;
static int ticking = 0;
struct xkb_keymap *cached_keymap;
struct xkb_context *cached_context;
static void _ecore_drm_tick_source_set(Ecore_Drm_Device *dev);
static void
_ecore_drm_tick_schedule(Ecore_Drm_Device *dev)
{
drmVBlank vbl;
if (!ticking) return;
vbl.request.type = (DRM_VBLANK_RELATIVE | DRM_VBLANK_EVENT);
vbl.request.sequence = 1;
vbl.request.signal = (unsigned long)dev;
if (drmWaitVBlank(dev->drm.fd, &vbl) < 0)
{
WRN("Vblank failed, disabling custom ticks");
_ecore_drm_tick_source_set(NULL);
}
}
static void
_ecore_drm_tick_begin(void *data)
{
ticking = 1;
_ecore_drm_tick_schedule(data);
}
static void
_ecore_drm_tick_end(void *data EINA_UNUSED)
{
ticking = 0;
}
static void
_ecore_drm_tick_source_set(Ecore_Drm_Device *dev)
{
if (!dev)
{
ecore_animator_custom_source_tick_begin_callback_set(NULL, NULL);
ecore_animator_custom_source_tick_end_callback_set(NULL, NULL);
ecore_animator_source_set(ECORE_ANIMATOR_SOURCE_TIMER);
return;
}
ecore_animator_custom_source_tick_begin_callback_set
(_ecore_drm_tick_begin, dev);
ecore_animator_custom_source_tick_end_callback_set
(_ecore_drm_tick_end, dev);
ecore_animator_source_set(ECORE_ANIMATOR_SOURCE_CUSTOM);
}
static void
_ecore_drm_device_cb_page_flip(int fd EINA_UNUSED, unsigned int frame EINA_UNUSED, unsigned int sec EINA_UNUSED, unsigned int usec EINA_UNUSED, void *data)
{
Ecore_Drm_Output *output = data;
Ecore_Drm_Fb *next;
if (output->pending_destroy)
{
ecore_drm_output_free(output);
return;
}
output->dev->current = output->current;
/* We were unable to queue a page on the last flip attempt, so we'll
* try again now. */
next = output->next;
if (next)
{
output->next = NULL;
_ecore_drm_output_fb_send(output->dev, next, output);
}
}
static void
_ecore_drm_device_cb_vblank(int fd EINA_UNUSED, unsigned int frame EINA_UNUSED, unsigned int sec EINA_UNUSED, unsigned int usec EINA_UNUSED, void *data)
{
ecore_animator_custom_tick();
if (ticking) _ecore_drm_tick_schedule(data);
}
static Eina_Bool
_cb_drm_event_handle(void *data, Ecore_Fd_Handler *hdlr EINA_UNUSED)
{
Ecore_Drm_Device *dev = data;
int err;
err = drmHandleEvent(dev->drm.fd, &dev->drm_ctx);
if (err)
{
ERR("drmHandleEvent failed to read an event");
return EINA_FALSE;
}
return EINA_TRUE;
}
static void
_ecore_drm_device_cb_output_event(const char *device EINA_UNUSED, Eeze_Udev_Event event EINA_UNUSED, void *data, Eeze_Udev_Watch *watch EINA_UNUSED)
{
Ecore_Drm_Device *dev;
if (!(dev = data)) return;
_ecore_drm_outputs_update(dev);
}
struct xkb_context *
_ecore_drm_device_cached_context_get(enum xkb_context_flags flags)
{
if (!cached_context)
return xkb_context_new(flags);
else
return xkb_context_ref(cached_context);
}
struct xkb_keymap *
_ecore_drm_device_cached_keymap_get(struct xkb_context *ctx, const struct xkb_rule_names *names, enum xkb_keymap_compile_flags flags)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(ctx, NULL);
if (!cached_keymap)
cached_keymap = xkb_map_new_from_names(ctx, names, flags);
return xkb_map_ref(cached_keymap);
}
void
_ecore_drm_device_cached_context_update(struct xkb_context *ctx)
{
Eina_List *l;
Ecore_Drm_Device *dev;
EINA_LIST_FOREACH(drm_devices, l, dev)
{
xkb_context_unref(dev->xkb_ctx);
dev->xkb_ctx = xkb_context_ref(ctx);
}
}
void
_ecore_drm_device_cached_keymap_update(struct xkb_keymap *map)
{
Eina_List *l, *l2, *l3;
Ecore_Drm_Device *dev;
Ecore_Drm_Seat *seat;
Ecore_Drm_Evdev *edev;
EINA_LIST_FOREACH(drm_devices, l, dev)
EINA_LIST_FOREACH(dev->seats, l2, seat)
EINA_LIST_FOREACH(seat->devices, l3, edev)
{
xkb_keymap_unref(edev->xkb.keymap);
edev->xkb.keymap = xkb_keymap_ref(map);
}
}
/**
* @defgroup Ecore_Drm_Device_Group Device manipulation functions
*
* Functions that deal with finding, opening, closing, and otherwise using
* the DRM device itself.
*/
EAPI Ecore_Drm_Device *
ecore_drm_device_find(const char *name, const char *seat)
{
Ecore_Drm_Device *dev = NULL;
Eina_Bool found = EINA_FALSE;
Eina_Bool platform = EINA_FALSE;
Eina_List *devs, *l;
const char *device;
/* try to get a list of drm devics */
if (!(devs = eeze_udev_find_by_type(EEZE_UDEV_TYPE_DRM, name)))
return NULL;
DBG("Find Drm Device: %s", name);
EINA_LIST_FOREACH(devs, l, device)
{
const char *devpath;
const char *devseat;
const char *devparent;
if (!(devpath = eeze_udev_syspath_get_devpath(device)))
continue;
DBG("Found Drm Device");
DBG("\tDevice: %s", device);
DBG("\tDevpath: %s", devpath);
if ((name) && (strcmp(name, devpath))) goto cont;
if (!(devseat = eeze_udev_syspath_get_property(device, "ID_SEAT")))
devseat = eina_stringshare_add("seat0");
if ((seat) && (strcmp(seat, devseat)))
goto cont;
else if (strcmp(devseat, "seat0"))
goto cont;
devparent = eeze_udev_syspath_get_parent_filtered(device, "pci", NULL);
if (!devparent)
{
devparent =
eeze_udev_syspath_get_parent_filtered(device, "platform", NULL);
platform = EINA_TRUE;
}
if (devparent)
{
if (!platform)
{
const char *id;
if ((id = eeze_udev_syspath_get_sysattr(devparent, "boot_vga")))
{
if (!strcmp(id, "1")) found = EINA_TRUE;
eina_stringshare_del(id);
}
}
else
found = EINA_TRUE;
eina_stringshare_del(devparent);
}
cont:
eina_stringshare_del(devpath);
if (found) break;
}
if (!found) goto out;
if ((dev = calloc(1, sizeof(Ecore_Drm_Device))))
{
dev->drm.fd = -1;
dev->drm.name = eeze_udev_syspath_get_devpath(device);
dev->drm.path = eina_stringshare_add(device);
dev->id = eeze_udev_syspath_get_sysnum(device);
dev->seat = eeze_udev_syspath_get_property(device, "ID_SEAT");
if (!dev->seat) dev->seat = eina_stringshare_add("seat0");
dev->vt = 0;
dev->format = 0;
dev->use_hw_accel = EINA_FALSE;
dev->session = NULL;
DBG("Using Drm Device: %s", dev->drm.name);
drm_devices = eina_list_append(drm_devices, dev);
}
out:
EINA_LIST_FREE(devs, device)
eina_stringshare_del(device);
return dev;
}
EAPI void
ecore_drm_device_free(Ecore_Drm_Device *dev)
{
unsigned int i = 0;
/* check for valid device */
if (!dev) return;
for (; i < ALEN(dev->dumb); i++)
{
if (dev->dumb[i]) ecore_drm_fb_destroy(dev->dumb[i]);
dev->dumb[i] = NULL;
}
if (dev->watch) eeze_udev_watch_del(dev->watch);
/* free crtcs */
if (dev->crtcs) free(dev->crtcs);
/* free device name */
if (dev->drm.name) eina_stringshare_del(dev->drm.name);
/* free device path */
if (dev->drm.path) eina_stringshare_del(dev->drm.path);
/* free device seat */
if (dev->seat) eina_stringshare_del(dev->seat);
/* free session */
free(dev->session);
drm_devices = eina_list_remove(drm_devices, dev);
/* free structure */
free(dev);
}
EAPI Eina_Bool
ecore_drm_device_open(Ecore_Drm_Device *dev)
{
uint64_t caps;
int events = 0;
drmVersionPtr ver;
/* check for valid device */
if ((!dev) || (!dev->drm.name)) return EINA_FALSE;
/* check if device is already opened */
if (dev->drm.fd != -1)
{
ERR("Device is already opened");
return EINA_FALSE;
}
/* DRM device node is needed immediately to keep going. */
dev->drm.fd =
_ecore_drm_launcher_device_open_no_pending(dev->drm.name, O_RDWR);
if (dev->drm.fd < 0) return EINA_FALSE;
DBG("Opened Device %s : %d", dev->drm.name, dev->drm.fd);
ver = drmGetVersion(dev->drm.fd);
if (ver)
{
DBG("\tDriver Name: %s", ver->name);
DBG("\tDriver Date: %s", ver->date);
DBG("\tDriver Description: %s", ver->desc);
DBG("\tDriver Version: %d.%d.%d",
ver->version_major, ver->version_minor,
ver->version_patchlevel);
drmFreeVersion(ver);
}
/* set client capabilities to 'universal planes' so drm core will expose
* the full universal plane list (including primary & cursor planes) */
drmSetClientCap(dev->drm.fd, DRM_CLIENT_CAP_UNIVERSAL_PLANES, 1);
if (!drmGetCap(dev->drm.fd, DRM_CAP_TIMESTAMP_MONOTONIC, &caps))
{
if (caps == 1)
dev->drm.clock = CLOCK_MONOTONIC;
else
dev->drm.clock = CLOCK_REALTIME;
}
else
{
ERR("Could not get TIMESTAMP_MONOTONIC device capabilities: %m");
}
/* Without DUMB_BUFFER we can't do software rendering on DRM. Fail without it
* until we have rock solid hardware accelerated DRM on all drivers */
if (drmGetCap(dev->drm.fd, DRM_CAP_DUMB_BUFFER, &caps) < 0 || !caps)
{
ERR("Could not get DUMB_BUFFER device capabilities: %m");
return EINA_FALSE;
}
/* try to create xkb context */
if (!(dev->xkb_ctx = _ecore_drm_device_cached_context_get(0)))
{
ERR("Failed to create xkb context");
return EINA_FALSE;
}
memset(&dev->drm_ctx, 0, sizeof(dev->drm_ctx));
dev->drm_ctx.version = DRM_EVENT_CONTEXT_VERSION;
dev->drm_ctx.page_flip_handler = _ecore_drm_device_cb_page_flip;
dev->drm_ctx.vblank_handler = _ecore_drm_device_cb_vblank;
events = (EEZE_UDEV_EVENT_ADD | EEZE_UDEV_EVENT_REMOVE |
EEZE_UDEV_EVENT_CHANGE);
dev->watch =
eeze_udev_watch_add(EEZE_UDEV_TYPE_DRM, events,
_ecore_drm_device_cb_output_event, dev);
dev->drm.hdlr =
ecore_main_fd_handler_add(dev->drm.fd, ECORE_FD_READ,
_cb_drm_event_handle, dev, NULL, NULL);
/* dev->drm.idler = */
/* ecore_idle_enterer_add(_ecore_drm_device_cb_idle, dev); */
return EINA_TRUE;
}
EAPI Eina_Bool
ecore_drm_device_close(Ecore_Drm_Device *dev)
{
/* check for valid device */
EINA_SAFETY_ON_NULL_RETURN_VAL(dev, EINA_FALSE);
/* check if device is opened */
if (dev->drm.fd == -1) return EINA_FALSE;
/* delete udev watch */
if (dev->watch) eeze_udev_watch_del(dev->watch);
dev->watch = NULL;
/* close xkb context */
if (dev->xkb_ctx) xkb_context_unref(dev->xkb_ctx);
dev->xkb_ctx = NULL;
_ecore_drm_launcher_device_close(dev->drm.name, dev->drm.fd);
/* reset device fd */
dev->drm.fd = -1;
return EINA_TRUE;
}
EAPI const Eina_List *
ecore_drm_devices_get(void)
{
return drm_devices;
}
EAPI Eina_Bool
ecore_drm_device_master_get(Ecore_Drm_Device *dev)
{
drm_magic_t mag;
/* check for valid device */
if ((!dev) || (dev->drm.fd < 0)) return EINA_FALSE;
/* get if we are master or not */
if ((drmGetMagic(dev->drm.fd, &mag) == 0) &&
(drmAuthMagic(dev->drm.fd, mag) == 0))
return EINA_TRUE;
return EINA_FALSE;
}
EAPI Eina_Bool
ecore_drm_device_master_set(Ecore_Drm_Device *dev)
{
/* check for valid device */
if ((!dev) || (dev->drm.fd < 0)) return EINA_FALSE;
DBG("Set Master On Fd: %d", dev->drm.fd);
drmSetMaster(dev->drm.fd);
return EINA_TRUE;
}
EAPI Eina_Bool
ecore_drm_device_master_drop(Ecore_Drm_Device *dev)
{
/* check for valid device */
if ((!dev) || (dev->drm.fd < 0)) return EINA_FALSE;
DBG("Drop Master On Fd: %d", dev->drm.fd);
drmDropMaster(dev->drm.fd);
return EINA_TRUE;
}
EAPI int
ecore_drm_device_fd_get(Ecore_Drm_Device *dev)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(dev, -1);
return dev->drm.fd;
}
EAPI void
ecore_drm_device_window_set(Ecore_Drm_Device *dev, unsigned int window)
{
/* check for valid device */
EINA_SAFETY_ON_TRUE_RETURN((!dev) || (dev->drm.fd < 0));
dev->window = window;
}
EAPI const char *
ecore_drm_device_name_get(Ecore_Drm_Device *dev)
{
/* check for valid device */
EINA_SAFETY_ON_TRUE_RETURN_VAL((!dev) || (dev->drm.fd < 0), NULL);
return dev->drm.name;
}
EAPI void
ecore_drm_device_pointer_xy_get(Ecore_Drm_Device *dev, int *x, int *y)
{
Ecore_Drm_Seat *seat;
Ecore_Drm_Evdev *edev;
Eina_List *l, *ll;
if (x) *x = 0;
if (y) *y = 0;
/* check for valid device */
EINA_SAFETY_ON_TRUE_RETURN((!dev) || (dev->drm.fd < 0));
EINA_LIST_FOREACH(dev->seats, l, seat)
{
EINA_LIST_FOREACH(seat->devices, ll, edev)
{
if (!libinput_device_has_capability(edev->device,
LIBINPUT_DEVICE_CAP_POINTER))
continue;
if (x) *x = seat->ptr.dx;
if (y) *y = seat->ptr.dy;
return;
}
}
}
EAPI void
ecore_drm_device_pointer_warp(Ecore_Drm_Device *dev, int x, int y)
{
Ecore_Drm_Seat *seat;
Ecore_Drm_Evdev *edev;
Eina_List *l, *ll;
/* check for valid device */
EINA_SAFETY_ON_TRUE_RETURN((!dev) || (dev->drm.fd < 0));
EINA_LIST_FOREACH(dev->seats, l, seat)
{
EINA_LIST_FOREACH(seat->devices, ll, edev)
{
if (!libinput_device_has_capability(edev->device,
LIBINPUT_DEVICE_CAP_POINTER))
continue;
seat->ptr.dx = seat->ptr.ix = x;
seat->ptr.dy = seat->ptr.iy = y;
_ecore_drm_pointer_motion_post(edev);
}
}
}
EAPI Eina_Bool
ecore_drm_device_software_setup(Ecore_Drm_Device *dev)
{
unsigned int i = 0;
int w = 0, h = 0;
EINA_SAFETY_ON_NULL_RETURN_VAL(dev, EINA_FALSE);
/* destroy any old buffers */
for (; i < ALEN(dev->dumb); i++)
{
if (dev->dumb[i]) ecore_drm_fb_destroy(dev->dumb[i]);
dev->dumb[i] = NULL;
}
/* get screen size */
ecore_drm_outputs_geometry_get(dev, NULL, NULL, &w, &h);
/* create new buffers */
for (i = 0; i < ALEN(dev->dumb); i++)
{
if (!(dev->dumb[i] = ecore_drm_fb_create(dev, w, h)))
{
ERR("Could not create dumb framebuffer");
goto err;
}
DBG("Ecore_Drm_Device Created Dumb Buffer");
DBG("\tFb: %d", dev->dumb[i]->id);
DBG("\tHandle: %d", dev->dumb[i]->hdl);
DBG("\tStride: %d", dev->dumb[i]->stride);
DBG("\tSize: %d", dev->dumb[i]->size);
DBG("\tW: %d\tH: %d", dev->dumb[i]->w, dev->dumb[i]->h);
}
_ecore_drm_tick_source_set(dev);
return EINA_TRUE;
err:
for (i = 0; i < ALEN(dev->dumb); i++)
{
if (dev->dumb[i]) ecore_drm_fb_destroy(dev->dumb[i]);
dev->dumb[i] = NULL;
}
return EINA_FALSE;
}
EAPI Ecore_Drm_Output *
ecore_drm_device_output_find(Ecore_Drm_Device *dev, int x, int y)
{
Ecore_Drm_Output *output;
Eina_List *l;
EINA_SAFETY_ON_NULL_RETURN_VAL(dev, NULL);
EINA_SAFETY_ON_TRUE_RETURN_VAL((x < 0) || (y < 0), NULL);
EINA_LIST_FOREACH(dev->outputs, l, output)
{
int ox = 0, oy = 0;
int ow = 0, oh = 0;
if (!output->cloned)
{
ox = output->x;
oy = output->y;
}
ow = output->current_mode->width;
oh = output->current_mode->height;
if (INSIDE(x, y, ox, oy, ow, oh))
return output;
}
return NULL;
}
EAPI void
ecore_drm_screen_size_range_get(Ecore_Drm_Device *dev, int *minw, int *minh, int *maxw, int *maxh)
{
EINA_SAFETY_ON_NULL_RETURN(dev);
if (minw) *minw = dev->min_width;
if (minh) *minh = dev->min_height;
if (maxw) *maxw = dev->max_width;
if (maxh) *maxh = dev->max_height;
}
EAPI Ecore_Drm_Output *
ecore_drm_device_output_name_find(Ecore_Drm_Device *dev, const char *name)
{
Ecore_Drm_Output *output;
Eina_List *l;
EINA_SAFETY_ON_NULL_RETURN_VAL(dev, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(name, NULL);
EINA_LIST_FOREACH(dev->outputs, l, output)
if ((output->name) && (!strcmp(name, output->name)))
return output;
return NULL;
}
EAPI Eina_Bool
ecore_drm_device_pointer_left_handed_set(Ecore_Drm_Device *dev, Eina_Bool left_handed)
{
Ecore_Drm_Seat *seat = NULL;
Ecore_Drm_Evdev *edev = NULL;
Eina_List *l = NULL, *l2 = NULL;
EINA_SAFETY_ON_NULL_RETURN_VAL(dev, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(dev->seats, EINA_FALSE);
if (dev->left_handed == left_handed)
return EINA_TRUE;
dev->left_handed = !!left_handed;
EINA_LIST_FOREACH(dev->seats, l, seat)
{
EINA_LIST_FOREACH(seat->devices, l2, edev)
{
if (libinput_device_has_capability(edev->device,
LIBINPUT_DEVICE_CAP_POINTER))
{
if (libinput_device_config_left_handed_set(edev->device, (int)left_handed) !=
LIBINPUT_CONFIG_STATUS_SUCCESS)
{
WRN("Failed to set left hand mode about device: %s\n",
libinput_device_get_name(edev->device));
continue;
}
}
}
}
return EINA_TRUE;
}
EAPI void
ecore_drm_device_keyboard_cached_context_set(struct xkb_context *ctx)
{
EINA_SAFETY_ON_NULL_RETURN(ctx);
if (cached_context == ctx) return;
if (cached_context)
_ecore_drm_device_cached_context_update(ctx);
cached_context = ctx;
}
EAPI void
ecore_drm_device_keyboard_cached_keymap_set(struct xkb_keymap *map)
{
EINA_SAFETY_ON_NULL_RETURN(map);
if (cached_keymap == map) return;
if (cached_keymap)
_ecore_drm_device_cached_keymap_update(map);
cached_keymap = map;
}