backend/drm: add wlr_drm_backend_monitor
This helper is responsible for listening for new DRM devices and create new child DRM backends as necessary.
This commit is contained in:
parent
f6d3efbf4b
commit
d8ca494558
|
@ -21,6 +21,7 @@
|
||||||
|
|
||||||
#if WLR_HAS_DRM_BACKEND
|
#if WLR_HAS_DRM_BACKEND
|
||||||
#include <wlr/backend/drm.h>
|
#include <wlr/backend/drm.h>
|
||||||
|
#include "backend/drm/monitor.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if WLR_HAS_LIBINPUT_BACKEND
|
#if WLR_HAS_LIBINPUT_BACKEND
|
||||||
|
@ -375,6 +376,8 @@ struct wlr_backend *wlr_backend_autocreate(struct wl_display *display) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
drm_backend_monitor_create(backend, primary_drm, multi->session);
|
||||||
|
|
||||||
return backend;
|
return backend;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,7 @@ wlr_files += files(
|
||||||
'cvt.c',
|
'cvt.c',
|
||||||
'drm.c',
|
'drm.c',
|
||||||
'legacy.c',
|
'legacy.c',
|
||||||
|
'monitor.c',
|
||||||
'properties.c',
|
'properties.c',
|
||||||
'renderer.c',
|
'renderer.c',
|
||||||
'util.c',
|
'util.c',
|
||||||
|
|
|
@ -0,0 +1,94 @@
|
||||||
|
#include <wlr/util/log.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include "backend/drm/monitor.h"
|
||||||
|
#include "backend/multi.h"
|
||||||
|
#include "backend/session/session.h"
|
||||||
|
|
||||||
|
static void drm_backend_monitor_destroy(struct wlr_drm_backend_monitor* monitor) {
|
||||||
|
wl_list_remove(&monitor->session_add_drm_card.link);
|
||||||
|
wl_list_remove(&monitor->session_destroy.link);
|
||||||
|
wl_list_remove(&monitor->primary_drm_destroy.link);
|
||||||
|
wl_list_remove(&monitor->multi_destroy.link);
|
||||||
|
free(monitor);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_add_drm_card(struct wl_listener *listener, void *data) {
|
||||||
|
struct wlr_session_add_event *event = data;
|
||||||
|
struct wlr_drm_backend_monitor *backend_monitor =
|
||||||
|
wl_container_of(listener, backend_monitor, session_add_drm_card);
|
||||||
|
|
||||||
|
struct wlr_device *dev =
|
||||||
|
session_open_if_kms(backend_monitor->session, event->path);
|
||||||
|
if (!dev) {
|
||||||
|
wlr_log(WLR_ERROR, "Unable to open %s as DRM device", event->path);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
wlr_log(WLR_DEBUG, "Creating DRM backend for %s after hotplug", event->path);
|
||||||
|
struct wlr_backend *child_drm = wlr_drm_backend_create(
|
||||||
|
backend_monitor->session->display, backend_monitor->session,
|
||||||
|
dev, backend_monitor->primary_drm);
|
||||||
|
if (!child_drm) {
|
||||||
|
wlr_log(WLR_ERROR, "Failed to create DRM backend after hotplug");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!wlr_multi_backend_add(backend_monitor->multi, child_drm)) {
|
||||||
|
wlr_log(WLR_ERROR, "Failed to add new drm backend to multi backend");
|
||||||
|
wlr_backend_destroy(child_drm);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!wlr_backend_start(child_drm)) {
|
||||||
|
wlr_log(WLR_ERROR, "Failed to start new child DRM backend");
|
||||||
|
wlr_backend_destroy(child_drm);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_session_destroy(struct wl_listener *listener, void *data) {
|
||||||
|
struct wlr_drm_backend_monitor *backend_monitor =
|
||||||
|
wl_container_of(listener, backend_monitor, session_destroy);
|
||||||
|
drm_backend_monitor_destroy(backend_monitor);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_primary_drm_destroy(struct wl_listener *listener, void *data) {
|
||||||
|
struct wlr_drm_backend_monitor *backend_monitor =
|
||||||
|
wl_container_of(listener, backend_monitor, primary_drm_destroy);
|
||||||
|
drm_backend_monitor_destroy(backend_monitor);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_multi_destroy(struct wl_listener *listener, void *data) {
|
||||||
|
struct wlr_drm_backend_monitor *backend_monitor =
|
||||||
|
wl_container_of(listener, backend_monitor, multi_destroy);
|
||||||
|
drm_backend_monitor_destroy(backend_monitor);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wlr_drm_backend_monitor *drm_backend_monitor_create(
|
||||||
|
struct wlr_backend *multi,
|
||||||
|
struct wlr_backend *primary_drm,
|
||||||
|
struct wlr_session *session) {
|
||||||
|
struct wlr_drm_backend_monitor *monitor =
|
||||||
|
calloc(1, sizeof(struct wlr_drm_backend_monitor));
|
||||||
|
if (!monitor) {
|
||||||
|
wlr_log_errno(WLR_ERROR, "Allocation failed");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
monitor->multi = multi;
|
||||||
|
monitor->primary_drm = primary_drm;
|
||||||
|
monitor->session = session;
|
||||||
|
|
||||||
|
monitor->session_add_drm_card.notify = handle_add_drm_card;
|
||||||
|
wl_signal_add(&session->events.add_drm_card, &monitor->session_add_drm_card);
|
||||||
|
|
||||||
|
monitor->session_destroy.notify = handle_session_destroy;
|
||||||
|
wl_signal_add(&session->events.destroy, &monitor->session_destroy);
|
||||||
|
|
||||||
|
monitor->primary_drm_destroy.notify = handle_primary_drm_destroy;
|
||||||
|
wl_signal_add(&primary_drm->events.destroy, &monitor->primary_drm_destroy);
|
||||||
|
|
||||||
|
monitor->multi_destroy.notify = handle_multi_destroy;
|
||||||
|
wl_signal_add(&multi->events.destroy, &monitor->multi_destroy);
|
||||||
|
|
||||||
|
return monitor;
|
||||||
|
}
|
|
@ -370,7 +370,7 @@ bool wlr_session_change_vt(struct wlr_session *session, unsigned vt) {
|
||||||
|
|
||||||
/* Tests if 'path' is KMS compatible by trying to open it. Returns the opened
|
/* Tests if 'path' is KMS compatible by trying to open it. Returns the opened
|
||||||
* device on success. */
|
* device on success. */
|
||||||
static struct wlr_device *open_if_kms(struct wlr_session *restrict session,
|
struct wlr_device *session_open_if_kms(struct wlr_session *restrict session,
|
||||||
const char *restrict path) {
|
const char *restrict path) {
|
||||||
if (!path) {
|
if (!path) {
|
||||||
return NULL;
|
return NULL;
|
||||||
|
@ -406,7 +406,7 @@ static ssize_t explicit_find_gpus(struct wlr_session *session,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret[i] = open_if_kms(session, ptr);
|
ret[i] = session_open_if_kms(session, ptr);
|
||||||
if (!ret[i]) {
|
if (!ret[i]) {
|
||||||
wlr_log(WLR_ERROR, "Unable to open %s as DRM device", ptr);
|
wlr_log(WLR_ERROR, "Unable to open %s as DRM device", ptr);
|
||||||
} else {
|
} else {
|
||||||
|
@ -542,7 +542,7 @@ ssize_t wlr_session_find_gpus(struct wlr_session *session,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct wlr_device *wlr_dev =
|
struct wlr_device *wlr_dev =
|
||||||
open_if_kms(session, udev_device_get_devnode(dev));
|
session_open_if_kms(session, udev_device_get_devnode(dev));
|
||||||
if (!wlr_dev) {
|
if (!wlr_dev) {
|
||||||
udev_device_unref(dev);
|
udev_device_unref(dev);
|
||||||
continue;
|
continue;
|
||||||
|
|
|
@ -0,0 +1,24 @@
|
||||||
|
#ifndef BACKEND_DRM_MONITOR_H
|
||||||
|
#define BACKEND_DRM_MONITOR_H
|
||||||
|
|
||||||
|
#include <wlr/backend/drm.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper to create new DRM sub-backends on GPU hotplug.
|
||||||
|
*/
|
||||||
|
struct wlr_drm_backend_monitor {
|
||||||
|
struct wlr_backend *multi;
|
||||||
|
struct wlr_backend *primary_drm;
|
||||||
|
struct wlr_session *session;
|
||||||
|
|
||||||
|
struct wl_listener multi_destroy;
|
||||||
|
struct wl_listener primary_drm_destroy;
|
||||||
|
struct wl_listener session_destroy;
|
||||||
|
struct wl_listener session_add_drm_card;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct wlr_drm_backend_monitor *drm_backend_monitor_create(
|
||||||
|
struct wlr_backend *multi, struct wlr_backend *primary_drm,
|
||||||
|
struct wlr_session *session);
|
||||||
|
|
||||||
|
#endif
|
|
@ -11,4 +11,7 @@ bool libseat_change_vt(struct wlr_session *base, unsigned vt);
|
||||||
|
|
||||||
void session_init(struct wlr_session *session);
|
void session_init(struct wlr_session *session);
|
||||||
|
|
||||||
|
struct wlr_device *session_open_if_kms(struct wlr_session *restrict session,
|
||||||
|
const char *restrict path);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in New Issue