mirror of
https://github.com/swaywm/sway.git
synced 2026-04-21 06:46:22 -04:00
Add support for toplevel capture
References: https://gitlab.freedesktop.org/wlroots/wlroots/-/merge_requests/5078 Rebased from origin/master.
This commit is contained in:
parent
b56f1930f7
commit
8c4439e98e
6 changed files with 297 additions and 225 deletions
|
|
@ -116,6 +116,9 @@ struct sway_server {
|
||||||
struct wlr_export_dmabuf_manager_v1 *export_dmabuf_manager_v1;
|
struct wlr_export_dmabuf_manager_v1 *export_dmabuf_manager_v1;
|
||||||
struct wlr_security_context_manager_v1 *security_context_manager_v1;
|
struct wlr_security_context_manager_v1 *security_context_manager_v1;
|
||||||
|
|
||||||
|
struct wlr_ext_foreign_toplevel_image_capture_source_manager_v1 *ext_foreign_toplevel_image_capture_source_manager_v1;
|
||||||
|
struct wl_listener new_foreign_toplevel_capture_request;
|
||||||
|
|
||||||
struct wlr_xdg_activation_v1 *xdg_activation_v1;
|
struct wlr_xdg_activation_v1 *xdg_activation_v1;
|
||||||
struct wl_listener xdg_activation_v1_request_activate;
|
struct wl_listener xdg_activation_v1_request_activate;
|
||||||
struct wl_listener xdg_activation_v1_new_token;
|
struct wl_listener xdg_activation_v1_new_token;
|
||||||
|
|
|
||||||
|
|
@ -69,6 +69,9 @@ struct sway_view {
|
||||||
struct wlr_scene_tree *content_tree;
|
struct wlr_scene_tree *content_tree;
|
||||||
struct wlr_scene_tree *saved_surface_tree;
|
struct wlr_scene_tree *saved_surface_tree;
|
||||||
|
|
||||||
|
struct wlr_scene *image_capture_scene;
|
||||||
|
struct wlr_ext_image_capture_source_v1 *image_capture_source;
|
||||||
|
|
||||||
struct sway_container *container; // NULL if unmapped and transactions finished
|
struct sway_container *container; // NULL if unmapped and transactions finished
|
||||||
struct wlr_surface *surface; // NULL for unmapped views
|
struct wlr_surface *surface; // NULL for unmapped views
|
||||||
struct sway_xdg_decoration *xdg_decoration;
|
struct sway_xdg_decoration *xdg_decoration;
|
||||||
|
|
@ -124,6 +127,8 @@ struct sway_view {
|
||||||
struct sway_xdg_shell_view {
|
struct sway_xdg_shell_view {
|
||||||
struct sway_view view;
|
struct sway_view view;
|
||||||
|
|
||||||
|
struct wlr_scene_tree *image_capture_tree;
|
||||||
|
|
||||||
struct wl_listener commit;
|
struct wl_listener commit;
|
||||||
struct wl_listener request_move;
|
struct wl_listener request_move;
|
||||||
struct wl_listener request_resize;
|
struct wl_listener request_resize;
|
||||||
|
|
@ -142,6 +147,8 @@ struct sway_xwayland_view {
|
||||||
|
|
||||||
struct wlr_scene_tree *surface_tree;
|
struct wlr_scene_tree *surface_tree;
|
||||||
|
|
||||||
|
struct wlr_scene_surface *image_capture_scene_surface;
|
||||||
|
|
||||||
struct wl_listener commit;
|
struct wl_listener commit;
|
||||||
struct wl_listener request_move;
|
struct wl_listener request_move;
|
||||||
struct wl_listener request_resize;
|
struct wl_listener request_resize;
|
||||||
|
|
@ -192,10 +199,12 @@ struct sway_popup_desc {
|
||||||
|
|
||||||
struct sway_xdg_popup {
|
struct sway_xdg_popup {
|
||||||
struct sway_view *view;
|
struct sway_view *view;
|
||||||
|
struct wlr_xdg_popup *wlr_xdg_popup;
|
||||||
|
|
||||||
struct wlr_scene_tree *scene_tree;
|
struct wlr_scene_tree *scene_tree;
|
||||||
struct wlr_scene_tree *xdg_surface_tree;
|
struct wlr_scene_tree *xdg_surface_tree;
|
||||||
struct wlr_xdg_popup *wlr_xdg_popup;
|
|
||||||
|
struct wlr_scene_tree *image_capture_tree;
|
||||||
|
|
||||||
struct sway_popup_desc desc;
|
struct sway_popup_desc desc;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -20,13 +20,13 @@
|
||||||
|
|
||||||
static struct sway_xdg_popup *popup_create(
|
static struct sway_xdg_popup *popup_create(
|
||||||
struct wlr_xdg_popup *wlr_popup, struct sway_view *view,
|
struct wlr_xdg_popup *wlr_popup, struct sway_view *view,
|
||||||
struct wlr_scene_tree *parent);
|
struct wlr_scene_tree *parent, struct wlr_scene_tree *image_capture_parent);
|
||||||
|
|
||||||
static void popup_handle_new_popup(struct wl_listener *listener, void *data) {
|
static void popup_handle_new_popup(struct wl_listener *listener, void *data) {
|
||||||
struct sway_xdg_popup *popup =
|
struct sway_xdg_popup *popup =
|
||||||
wl_container_of(listener, popup, new_popup);
|
wl_container_of(listener, popup, new_popup);
|
||||||
struct wlr_xdg_popup *wlr_popup = data;
|
struct wlr_xdg_popup *wlr_popup = data;
|
||||||
popup_create(wlr_popup, popup->view, popup->xdg_surface_tree);
|
popup_create(wlr_popup, popup->view, popup->xdg_surface_tree, popup->image_capture_tree);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void popup_handle_destroy(struct wl_listener *listener, void *data) {
|
static void popup_handle_destroy(struct wl_listener *listener, void *data) {
|
||||||
|
|
@ -77,7 +77,8 @@ static void popup_handle_reposition(struct wl_listener *listener, void *data) {
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct sway_xdg_popup *popup_create(struct wlr_xdg_popup *wlr_popup,
|
static struct sway_xdg_popup *popup_create(struct wlr_xdg_popup *wlr_popup,
|
||||||
struct sway_view *view, struct wlr_scene_tree *parent) {
|
struct sway_view *view, struct wlr_scene_tree *parent,
|
||||||
|
struct wlr_scene_tree *image_capture_parent) {
|
||||||
struct wlr_xdg_surface *xdg_surface = wlr_popup->base;
|
struct wlr_xdg_surface *xdg_surface = wlr_popup->base;
|
||||||
|
|
||||||
struct sway_xdg_popup *popup = calloc(1, sizeof(struct sway_xdg_popup));
|
struct sway_xdg_popup *popup = calloc(1, sizeof(struct sway_xdg_popup));
|
||||||
|
|
@ -113,6 +114,11 @@ static struct sway_xdg_popup *popup_create(struct wlr_xdg_popup *wlr_popup,
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
popup->image_capture_tree = wlr_scene_xdg_surface_create(image_capture_parent, xdg_surface);
|
||||||
|
if (popup->image_capture_tree == NULL) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
popup->wlr_xdg_popup = xdg_surface->popup;
|
popup->wlr_xdg_popup = xdg_surface->popup;
|
||||||
struct sway_xdg_shell_view *shell_view =
|
struct sway_xdg_shell_view *shell_view =
|
||||||
wl_container_of(view, shell_view, view);
|
wl_container_of(view, shell_view, view);
|
||||||
|
|
@ -359,7 +365,7 @@ static void handle_new_popup(struct wl_listener *listener, void *data) {
|
||||||
struct wlr_xdg_popup *wlr_popup = data;
|
struct wlr_xdg_popup *wlr_popup = data;
|
||||||
|
|
||||||
struct sway_xdg_popup *popup = popup_create(wlr_popup,
|
struct sway_xdg_popup *popup = popup_create(wlr_popup,
|
||||||
&xdg_shell_view->view, root->layers.popup);
|
&xdg_shell_view->view, root->layers.popup, xdg_shell_view->image_capture_tree);
|
||||||
if (!popup) {
|
if (!popup) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
@ -570,6 +576,8 @@ void handle_xdg_shell_toplevel(struct wl_listener *listener, void *data) {
|
||||||
wl_signal_add(&xdg_toplevel->events.destroy, &xdg_shell_view->destroy);
|
wl_signal_add(&xdg_toplevel->events.destroy, &xdg_shell_view->destroy);
|
||||||
|
|
||||||
wlr_scene_xdg_surface_create(xdg_shell_view->view.content_tree, xdg_toplevel->base);
|
wlr_scene_xdg_surface_create(xdg_shell_view->view.content_tree, xdg_toplevel->base);
|
||||||
|
xdg_shell_view->image_capture_tree =
|
||||||
|
wlr_scene_xdg_surface_create(&xdg_shell_view->view.image_capture_scene->tree, xdg_toplevel->base);
|
||||||
|
|
||||||
xdg_toplevel->base->data = xdg_shell_view;
|
xdg_toplevel->base->data = xdg_shell_view;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -497,6 +497,9 @@ static void handle_unmap(struct wl_listener *listener, void *data) {
|
||||||
wl_list_remove(&xwayland_view->commit.link);
|
wl_list_remove(&xwayland_view->commit.link);
|
||||||
wl_list_remove(&xwayland_view->surface_tree_destroy.link);
|
wl_list_remove(&xwayland_view->surface_tree_destroy.link);
|
||||||
|
|
||||||
|
wlr_scene_node_destroy(&xwayland_view->image_capture_scene_surface->buffer->node);
|
||||||
|
xwayland_view->image_capture_scene_surface = NULL;
|
||||||
|
|
||||||
if (xwayland_view->surface_tree) {
|
if (xwayland_view->surface_tree) {
|
||||||
wlr_scene_node_destroy(&xwayland_view->surface_tree->node);
|
wlr_scene_node_destroy(&xwayland_view->surface_tree->node);
|
||||||
xwayland_view->surface_tree = NULL;
|
xwayland_view->surface_tree = NULL;
|
||||||
|
|
@ -537,6 +540,9 @@ static void handle_map(struct wl_listener *listener, void *data) {
|
||||||
&xwayland_view->surface_tree_destroy);
|
&xwayland_view->surface_tree_destroy);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
xwayland_view->image_capture_scene_surface =
|
||||||
|
wlr_scene_surface_create(&xwayland_view->view.image_capture_scene->tree, xsurface->surface);
|
||||||
|
|
||||||
transaction_commit_dirty();
|
transaction_commit_dirty();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
479
sway/server.c
479
sway/server.c
|
|
@ -1,3 +1,13 @@
|
||||||
|
#include "sway/server.h"
|
||||||
|
#include "config.h"
|
||||||
|
#include "list.h"
|
||||||
|
#include "log.h"
|
||||||
|
#include "sway/config.h"
|
||||||
|
#include "sway/desktop/idle_inhibit_v1.h"
|
||||||
|
#include "sway/input/cursor.h"
|
||||||
|
#include "sway/input/input-manager.h"
|
||||||
|
#include "sway/output.h"
|
||||||
|
#include "sway/tree/root.h"
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
@ -14,13 +24,13 @@
|
||||||
#include <wlr/types/wlr_content_type_v1.h>
|
#include <wlr/types/wlr_content_type_v1.h>
|
||||||
#include <wlr/types/wlr_cursor_shape_v1.h>
|
#include <wlr/types/wlr_cursor_shape_v1.h>
|
||||||
#include <wlr/types/wlr_data_control_v1.h>
|
#include <wlr/types/wlr_data_control_v1.h>
|
||||||
#include <wlr/types/wlr_ext_data_control_v1.h>
|
|
||||||
#include <wlr/types/wlr_data_device.h>
|
#include <wlr/types/wlr_data_device.h>
|
||||||
#include <wlr/types/wlr_export_dmabuf_v1.h>
|
#include <wlr/types/wlr_export_dmabuf_v1.h>
|
||||||
|
#include <wlr/types/wlr_ext_data_control_v1.h>
|
||||||
#include <wlr/types/wlr_ext_foreign_toplevel_list_v1.h>
|
#include <wlr/types/wlr_ext_foreign_toplevel_list_v1.h>
|
||||||
#include <wlr/types/wlr_foreign_toplevel_management_v1.h>
|
|
||||||
#include <wlr/types/wlr_ext_image_capture_source_v1.h>
|
#include <wlr/types/wlr_ext_image_capture_source_v1.h>
|
||||||
#include <wlr/types/wlr_ext_image_copy_capture_v1.h>
|
#include <wlr/types/wlr_ext_image_copy_capture_v1.h>
|
||||||
|
#include <wlr/types/wlr_foreign_toplevel_management_v1.h>
|
||||||
#include <wlr/types/wlr_fractional_scale_v1.h>
|
#include <wlr/types/wlr_fractional_scale_v1.h>
|
||||||
#include <wlr/types/wlr_gamma_control_v1.h>
|
#include <wlr/types/wlr_gamma_control_v1.h>
|
||||||
#include <wlr/types/wlr_idle_notify_v1.h>
|
#include <wlr/types/wlr_idle_notify_v1.h>
|
||||||
|
|
@ -49,20 +59,10 @@
|
||||||
#include <wlr/types/wlr_xdg_foreign_v2.h>
|
#include <wlr/types/wlr_xdg_foreign_v2.h>
|
||||||
#include <wlr/types/wlr_xdg_output_v1.h>
|
#include <wlr/types/wlr_xdg_output_v1.h>
|
||||||
#include <xf86drm.h>
|
#include <xf86drm.h>
|
||||||
#include "config.h"
|
|
||||||
#include "list.h"
|
|
||||||
#include "log.h"
|
|
||||||
#include "sway/config.h"
|
|
||||||
#include "sway/desktop/idle_inhibit_v1.h"
|
|
||||||
#include "sway/input/input-manager.h"
|
|
||||||
#include "sway/output.h"
|
|
||||||
#include "sway/server.h"
|
|
||||||
#include "sway/input/cursor.h"
|
|
||||||
#include "sway/tree/root.h"
|
|
||||||
|
|
||||||
#if WLR_HAS_XWAYLAND
|
#if WLR_HAS_XWAYLAND
|
||||||
#include <wlr/xwayland/shell.h>
|
|
||||||
#include "sway/xwayland.h"
|
#include "sway/xwayland.h"
|
||||||
|
#include <wlr/xwayland/shell.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if WLR_HAS_DRM_BACKEND
|
#if WLR_HAS_DRM_BACKEND
|
||||||
|
|
@ -82,278 +82,300 @@ bool allow_unsupported_gpu = false;
|
||||||
|
|
||||||
#if WLR_HAS_DRM_BACKEND
|
#if WLR_HAS_DRM_BACKEND
|
||||||
static void handle_drm_lease_request(struct wl_listener *listener, void *data) {
|
static void handle_drm_lease_request(struct wl_listener *listener, void *data) {
|
||||||
/* We only offer non-desktop outputs, but in the future we might want to do
|
/* We only offer non-desktop outputs, but in the future we might want to do
|
||||||
* more logic here. */
|
* more logic here. */
|
||||||
|
|
||||||
struct wlr_drm_lease_request_v1 *req = data;
|
struct wlr_drm_lease_request_v1 *req = data;
|
||||||
struct wlr_drm_lease_v1 *lease = wlr_drm_lease_request_v1_grant(req);
|
struct wlr_drm_lease_v1 *lease = wlr_drm_lease_request_v1_grant(req);
|
||||||
if (!lease) {
|
if (!lease) {
|
||||||
sway_log(SWAY_ERROR, "Failed to grant lease request");
|
sway_log(SWAY_ERROR, "Failed to grant lease request");
|
||||||
wlr_drm_lease_request_v1_reject(req);
|
wlr_drm_lease_request_v1_reject(req);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static bool is_privileged(const struct wl_global *global) {
|
static bool is_privileged(const struct wl_global *global) {
|
||||||
#if WLR_HAS_DRM_BACKEND
|
#if WLR_HAS_DRM_BACKEND
|
||||||
if (server.drm_lease_manager != NULL) {
|
if (server.drm_lease_manager != NULL) {
|
||||||
struct wlr_drm_lease_device_v1 *drm_lease_dev;
|
struct wlr_drm_lease_device_v1 *drm_lease_dev;
|
||||||
wl_list_for_each(drm_lease_dev, &server.drm_lease_manager->devices, link) {
|
wl_list_for_each(drm_lease_dev, &server.drm_lease_manager->devices, link) {
|
||||||
if (drm_lease_dev->global == global) {
|
if (drm_lease_dev->global == global) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return
|
return global == server.output_manager_v1->global ||
|
||||||
global == server.output_manager_v1->global ||
|
global == server.output_power_manager_v1->global ||
|
||||||
global == server.output_power_manager_v1->global ||
|
global == server.input_method->global ||
|
||||||
global == server.input_method->global ||
|
global == server.foreign_toplevel_list->global ||
|
||||||
global == server.foreign_toplevel_list->global ||
|
global == server.foreign_toplevel_manager->global ||
|
||||||
global == server.foreign_toplevel_manager->global ||
|
global == server.wlr_data_control_manager_v1->global ||
|
||||||
global == server.wlr_data_control_manager_v1->global ||
|
global == server.ext_data_control_manager_v1->global ||
|
||||||
global == server.ext_data_control_manager_v1->global ||
|
global == server.screencopy_manager_v1->global ||
|
||||||
global == server.screencopy_manager_v1->global ||
|
global == server.ext_image_copy_capture_manager_v1->global ||
|
||||||
global == server.ext_image_copy_capture_manager_v1->global ||
|
global == server.export_dmabuf_manager_v1->global ||
|
||||||
global == server.export_dmabuf_manager_v1->global ||
|
global == server.security_context_manager_v1->global ||
|
||||||
global == server.security_context_manager_v1->global ||
|
global == server.gamma_control_manager_v1->global ||
|
||||||
global == server.gamma_control_manager_v1->global ||
|
global == server.layer_shell->global ||
|
||||||
global == server.layer_shell->global ||
|
global == server.session_lock.manager->global ||
|
||||||
global == server.session_lock.manager->global ||
|
global == server.input->keyboard_shortcuts_inhibit->global ||
|
||||||
global == server.input->keyboard_shortcuts_inhibit->global ||
|
global == server.input->virtual_keyboard->global ||
|
||||||
global == server.input->virtual_keyboard->global ||
|
global == server.input->virtual_pointer->global ||
|
||||||
global == server.input->virtual_pointer->global ||
|
global == server.input->transient_seat_manager->global ||
|
||||||
global == server.input->transient_seat_manager->global ||
|
global == server.xdg_output_manager_v1->global;
|
||||||
global == server.xdg_output_manager_v1->global;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool filter_global(const struct wl_client *client,
|
static bool filter_global(const struct wl_client *client,
|
||||||
const struct wl_global *global, void *data) {
|
const struct wl_global *global, void *data) {
|
||||||
#if WLR_HAS_XWAYLAND
|
#if WLR_HAS_XWAYLAND
|
||||||
struct wlr_xwayland *xwayland = server.xwayland.wlr_xwayland;
|
struct wlr_xwayland *xwayland = server.xwayland.wlr_xwayland;
|
||||||
if (xwayland && global == xwayland->shell_v1->global) {
|
if (xwayland && global == xwayland->shell_v1->global) {
|
||||||
return xwayland->server != NULL && client == xwayland->server->client;
|
return xwayland->server != NULL && client == xwayland->server->client;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Restrict usage of privileged protocols to unsandboxed clients
|
// Restrict usage of privileged protocols to unsandboxed clients
|
||||||
// TODO: add a way for users to configure an allow-list
|
// TODO: add a way for users to configure an allow-list
|
||||||
const struct wlr_security_context_v1_state *security_context =
|
const struct wlr_security_context_v1_state *security_context =
|
||||||
wlr_security_context_manager_v1_lookup_client(
|
wlr_security_context_manager_v1_lookup_client(
|
||||||
server.security_context_manager_v1, (struct wl_client *)client);
|
server.security_context_manager_v1, (struct wl_client *)client);
|
||||||
if (is_privileged(global)) {
|
if (is_privileged(global)) {
|
||||||
return security_context == NULL;
|
return security_context == NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void detect_proprietary(struct wlr_backend *backend, void *data) {
|
static void detect_proprietary(struct wlr_backend *backend, void *data) {
|
||||||
int drm_fd = wlr_backend_get_drm_fd(backend);
|
int drm_fd = wlr_backend_get_drm_fd(backend);
|
||||||
if (drm_fd < 0) {
|
if (drm_fd < 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
drmVersion *version = drmGetVersion(drm_fd);
|
drmVersion *version = drmGetVersion(drm_fd);
|
||||||
if (version == NULL) {
|
if (version == NULL) {
|
||||||
sway_log(SWAY_ERROR, "drmGetVersion() failed");
|
sway_log(SWAY_ERROR, "drmGetVersion() failed");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool is_unsupported = false;
|
bool is_unsupported = false;
|
||||||
if (strcmp(version->name, "nvidia-drm") == 0) {
|
if (strcmp(version->name, "nvidia-drm") == 0) {
|
||||||
is_unsupported = true;
|
is_unsupported = true;
|
||||||
sway_log(SWAY_ERROR, "!!! Proprietary Nvidia drivers are in use !!!");
|
sway_log(SWAY_ERROR, "!!! Proprietary Nvidia drivers are in use !!!");
|
||||||
if (!allow_unsupported_gpu) {
|
if (!allow_unsupported_gpu) {
|
||||||
sway_log(SWAY_ERROR, "Use Nouveau instead");
|
sway_log(SWAY_ERROR, "Use Nouveau instead");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strcmp(version->name, "evdi") == 0) {
|
if (strcmp(version->name, "evdi") == 0) {
|
||||||
is_unsupported = true;
|
is_unsupported = true;
|
||||||
sway_log(SWAY_ERROR, "!!! Proprietary DisplayLink drivers are in use !!!");
|
sway_log(SWAY_ERROR, "!!! Proprietary DisplayLink drivers are in use !!!");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!allow_unsupported_gpu && is_unsupported) {
|
if (!allow_unsupported_gpu && is_unsupported) {
|
||||||
sway_log(SWAY_ERROR,
|
sway_log(SWAY_ERROR,
|
||||||
"Proprietary drivers are NOT supported. To launch sway anyway, "
|
"Proprietary drivers are NOT supported. To launch sway anyway, "
|
||||||
"launch with --unsupported-gpu and DO NOT report issues.");
|
"launch with --unsupported-gpu and DO NOT report issues.");
|
||||||
exit(EXIT_FAILURE);
|
exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
drmFreeVersion(version);
|
drmFreeVersion(version);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void do_renderer_recreate(void *data) {
|
static void do_renderer_recreate(void *data) {
|
||||||
struct sway_server *server = data;
|
struct sway_server *server = data;
|
||||||
server->recreating_renderer = NULL;
|
server->recreating_renderer = NULL;
|
||||||
|
|
||||||
sway_log(SWAY_INFO, "Re-creating renderer after GPU reset");
|
sway_log(SWAY_INFO, "Re-creating renderer after GPU reset");
|
||||||
struct wlr_renderer *renderer = wlr_renderer_autocreate(server->backend);
|
struct wlr_renderer *renderer = wlr_renderer_autocreate(server->backend);
|
||||||
if (renderer == NULL) {
|
if (renderer == NULL) {
|
||||||
sway_log(SWAY_ERROR, "Unable to create renderer");
|
sway_log(SWAY_ERROR, "Unable to create renderer");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct wlr_allocator *allocator =
|
struct wlr_allocator *allocator =
|
||||||
wlr_allocator_autocreate(server->backend, renderer);
|
wlr_allocator_autocreate(server->backend, renderer);
|
||||||
if (allocator == NULL) {
|
if (allocator == NULL) {
|
||||||
sway_log(SWAY_ERROR, "Unable to create allocator");
|
sway_log(SWAY_ERROR, "Unable to create allocator");
|
||||||
wlr_renderer_destroy(renderer);
|
wlr_renderer_destroy(renderer);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct wlr_renderer *old_renderer = server->renderer;
|
struct wlr_renderer *old_renderer = server->renderer;
|
||||||
struct wlr_allocator *old_allocator = server->allocator;
|
struct wlr_allocator *old_allocator = server->allocator;
|
||||||
server->renderer = renderer;
|
server->renderer = renderer;
|
||||||
server->allocator = allocator;
|
server->allocator = allocator;
|
||||||
|
|
||||||
wl_list_remove(&server->renderer_lost.link);
|
wl_list_remove(&server->renderer_lost.link);
|
||||||
wl_signal_add(&server->renderer->events.lost, &server->renderer_lost);
|
wl_signal_add(&server->renderer->events.lost, &server->renderer_lost);
|
||||||
|
|
||||||
wlr_compositor_set_renderer(server->compositor, renderer);
|
wlr_compositor_set_renderer(server->compositor, renderer);
|
||||||
|
|
||||||
struct sway_output *output;
|
struct sway_output *output;
|
||||||
wl_list_for_each(output, &root->all_outputs, link) {
|
wl_list_for_each(output, &root->all_outputs, link) {
|
||||||
wlr_output_init_render(output->wlr_output,
|
wlr_output_init_render(output->wlr_output, server->allocator,
|
||||||
server->allocator, server->renderer);
|
server->renderer);
|
||||||
}
|
}
|
||||||
|
|
||||||
wlr_allocator_destroy(old_allocator);
|
wlr_allocator_destroy(old_allocator);
|
||||||
wlr_renderer_destroy(old_renderer);
|
wlr_renderer_destroy(old_renderer);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_renderer_lost(struct wl_listener *listener, void *data) {
|
static void handle_renderer_lost(struct wl_listener *listener, void *data) {
|
||||||
struct sway_server *server = wl_container_of(listener, server, renderer_lost);
|
struct sway_server *server = wl_container_of(listener, server, renderer_lost);
|
||||||
|
|
||||||
if (server->recreating_renderer != NULL) {
|
if (server->recreating_renderer != NULL) {
|
||||||
sway_log(SWAY_DEBUG, "Re-creation of renderer already scheduled");
|
sway_log(SWAY_DEBUG, "Re-creation of renderer already scheduled");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
sway_log(SWAY_INFO, "Scheduling re-creation of renderer after GPU reset");
|
sway_log(SWAY_INFO, "Scheduling re-creation of renderer after GPU reset");
|
||||||
server->recreating_renderer = wl_event_loop_add_idle(server->wl_event_loop, do_renderer_recreate, server);
|
server->recreating_renderer = wl_event_loop_add_idle(
|
||||||
|
server->wl_event_loop, do_renderer_recreate, server);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
handle_new_foreign_toplevel_capture_request(struct wl_listener *listener,
|
||||||
|
void *data) {
|
||||||
|
struct wlr_ext_foreign_toplevel_image_capture_source_manager_v1_request
|
||||||
|
*request = data;
|
||||||
|
struct sway_view *view = request->toplevel_handle->data;
|
||||||
|
|
||||||
|
if (view->image_capture_source == NULL) {
|
||||||
|
view->image_capture_source =
|
||||||
|
wlr_ext_image_capture_source_v1_create_with_scene_node(
|
||||||
|
&view->image_capture_scene->tree.node, server.wl_event_loop,
|
||||||
|
server.allocator, server.renderer);
|
||||||
|
if (view->image_capture_source == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wlr_ext_foreign_toplevel_image_capture_source_manager_v1_request_accept(
|
||||||
|
request, view->image_capture_source);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool server_init(struct sway_server *server) {
|
bool server_init(struct sway_server *server) {
|
||||||
sway_log(SWAY_DEBUG, "Initializing Wayland server");
|
sway_log(SWAY_DEBUG, "Initializing Wayland server");
|
||||||
server->wl_display = wl_display_create();
|
server->wl_display = wl_display_create();
|
||||||
server->wl_event_loop = wl_display_get_event_loop(server->wl_display);
|
server->wl_event_loop = wl_display_get_event_loop(server->wl_display);
|
||||||
|
|
||||||
wl_display_set_global_filter(server->wl_display, filter_global, NULL);
|
wl_display_set_global_filter(server->wl_display, filter_global, NULL);
|
||||||
wl_display_set_default_max_buffer_size(server->wl_display, 1024 * 1024);
|
wl_display_set_default_max_buffer_size(server->wl_display, 1024 * 1024);
|
||||||
|
|
||||||
root = root_create(server->wl_display);
|
root = root_create(server->wl_display);
|
||||||
|
|
||||||
server->backend = wlr_backend_autocreate(server->wl_event_loop, &server->session);
|
server->backend =
|
||||||
if (!server->backend) {
|
wlr_backend_autocreate(server->wl_event_loop, &server->session);
|
||||||
sway_log(SWAY_ERROR, "Unable to create backend");
|
if (!server->backend) {
|
||||||
return false;
|
sway_log(SWAY_ERROR, "Unable to create backend");
|
||||||
}
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
wlr_multi_for_each_backend(server->backend, detect_proprietary, NULL);
|
wlr_multi_for_each_backend(server->backend, detect_proprietary, NULL);
|
||||||
|
|
||||||
server->renderer = wlr_renderer_autocreate(server->backend);
|
server->renderer = wlr_renderer_autocreate(server->backend);
|
||||||
if (!server->renderer) {
|
if (!server->renderer) {
|
||||||
sway_log(SWAY_ERROR, "Failed to create renderer");
|
sway_log(SWAY_ERROR, "Failed to create renderer");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
server->renderer_lost.notify = handle_renderer_lost;
|
server->renderer_lost.notify = handle_renderer_lost;
|
||||||
wl_signal_add(&server->renderer->events.lost, &server->renderer_lost);
|
wl_signal_add(&server->renderer->events.lost, &server->renderer_lost);
|
||||||
|
|
||||||
wlr_renderer_init_wl_shm(server->renderer, server->wl_display);
|
wlr_renderer_init_wl_shm(server->renderer, server->wl_display);
|
||||||
|
|
||||||
if (wlr_renderer_get_texture_formats(server->renderer, WLR_BUFFER_CAP_DMABUF) != NULL) {
|
if (wlr_renderer_get_texture_formats(server->renderer,
|
||||||
server->linux_dmabuf_v1 = wlr_linux_dmabuf_v1_create_with_renderer(
|
WLR_BUFFER_CAP_DMABUF) != NULL) {
|
||||||
server->wl_display, 4, server->renderer);
|
server->linux_dmabuf_v1 = wlr_linux_dmabuf_v1_create_with_renderer(
|
||||||
}
|
server->wl_display, 4, server->renderer);
|
||||||
if (wlr_renderer_get_drm_fd(server->renderer) >= 0 &&
|
}
|
||||||
server->renderer->features.timeline &&
|
if (wlr_renderer_get_drm_fd(server->renderer) >= 0 &&
|
||||||
server->backend->features.timeline) {
|
server->renderer->features.timeline &&
|
||||||
wlr_linux_drm_syncobj_manager_v1_create(server->wl_display, 1,
|
server->backend->features.timeline) {
|
||||||
wlr_renderer_get_drm_fd(server->renderer));
|
wlr_linux_drm_syncobj_manager_v1_create(
|
||||||
}
|
server->wl_display, 1, wlr_renderer_get_drm_fd(server->renderer));
|
||||||
|
}
|
||||||
|
|
||||||
server->allocator = wlr_allocator_autocreate(server->backend,
|
server->allocator =
|
||||||
server->renderer);
|
wlr_allocator_autocreate(server->backend, server->renderer);
|
||||||
if (!server->allocator) {
|
if (!server->allocator) {
|
||||||
sway_log(SWAY_ERROR, "Failed to create allocator");
|
sway_log(SWAY_ERROR, "Failed to create allocator");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
server->compositor = wlr_compositor_create(server->wl_display, 6,
|
server->compositor =
|
||||||
server->renderer);
|
wlr_compositor_create(server->wl_display, 6, server->renderer);
|
||||||
|
|
||||||
wlr_subcompositor_create(server->wl_display);
|
wlr_subcompositor_create(server->wl_display);
|
||||||
|
|
||||||
server->data_device_manager =
|
server->data_device_manager =
|
||||||
wlr_data_device_manager_create(server->wl_display);
|
wlr_data_device_manager_create(server->wl_display);
|
||||||
|
|
||||||
server->gamma_control_manager_v1 =
|
server->gamma_control_manager_v1 =
|
||||||
wlr_gamma_control_manager_v1_create(server->wl_display);
|
wlr_gamma_control_manager_v1_create(server->wl_display);
|
||||||
wlr_scene_set_gamma_control_manager_v1(root->root_scene,
|
wlr_scene_set_gamma_control_manager_v1(root->root_scene,
|
||||||
server->gamma_control_manager_v1);
|
server->gamma_control_manager_v1);
|
||||||
|
|
||||||
server->new_output.notify = handle_new_output;
|
server->new_output.notify = handle_new_output;
|
||||||
wl_signal_add(&server->backend->events.new_output, &server->new_output);
|
wl_signal_add(&server->backend->events.new_output, &server->new_output);
|
||||||
|
|
||||||
server->xdg_output_manager_v1 =
|
server->xdg_output_manager_v1 =
|
||||||
wlr_xdg_output_manager_v1_create(server->wl_display, root->output_layout);
|
wlr_xdg_output_manager_v1_create(server->wl_display, root->output_layout);
|
||||||
|
|
||||||
server->idle_notifier_v1 = wlr_idle_notifier_v1_create(server->wl_display);
|
server->idle_notifier_v1 = wlr_idle_notifier_v1_create(server->wl_display);
|
||||||
sway_idle_inhibit_manager_v1_init();
|
sway_idle_inhibit_manager_v1_init();
|
||||||
|
|
||||||
server->layer_shell = wlr_layer_shell_v1_create(server->wl_display,
|
server->layer_shell =
|
||||||
SWAY_LAYER_SHELL_VERSION);
|
wlr_layer_shell_v1_create(server->wl_display, SWAY_LAYER_SHELL_VERSION);
|
||||||
wl_signal_add(&server->layer_shell->events.new_surface,
|
wl_signal_add(&server->layer_shell->events.new_surface,
|
||||||
&server->layer_shell_surface);
|
&server->layer_shell_surface);
|
||||||
server->layer_shell_surface.notify = handle_layer_shell_surface;
|
server->layer_shell_surface.notify = handle_layer_shell_surface;
|
||||||
|
|
||||||
server->xdg_shell = wlr_xdg_shell_create(server->wl_display,
|
server->xdg_shell =
|
||||||
SWAY_XDG_SHELL_VERSION);
|
wlr_xdg_shell_create(server->wl_display, SWAY_XDG_SHELL_VERSION);
|
||||||
wl_signal_add(&server->xdg_shell->events.new_toplevel,
|
wl_signal_add(&server->xdg_shell->events.new_toplevel,
|
||||||
&server->xdg_shell_toplevel);
|
&server->xdg_shell_toplevel);
|
||||||
server->xdg_shell_toplevel.notify = handle_xdg_shell_toplevel;
|
server->xdg_shell_toplevel.notify = handle_xdg_shell_toplevel;
|
||||||
|
|
||||||
server->tablet_v2 = wlr_tablet_v2_create(server->wl_display);
|
server->tablet_v2 = wlr_tablet_v2_create(server->wl_display);
|
||||||
|
|
||||||
server->server_decoration_manager =
|
server->server_decoration_manager =
|
||||||
wlr_server_decoration_manager_create(server->wl_display);
|
wlr_server_decoration_manager_create(server->wl_display);
|
||||||
wlr_server_decoration_manager_set_default_mode(
|
wlr_server_decoration_manager_set_default_mode(
|
||||||
server->server_decoration_manager,
|
server->server_decoration_manager,
|
||||||
WLR_SERVER_DECORATION_MANAGER_MODE_SERVER);
|
WLR_SERVER_DECORATION_MANAGER_MODE_SERVER);
|
||||||
wl_signal_add(&server->server_decoration_manager->events.new_decoration,
|
wl_signal_add(&server->server_decoration_manager->events.new_decoration,
|
||||||
&server->server_decoration);
|
&server->server_decoration);
|
||||||
server->server_decoration.notify = handle_server_decoration;
|
server->server_decoration.notify = handle_server_decoration;
|
||||||
wl_list_init(&server->decorations);
|
wl_list_init(&server->decorations);
|
||||||
|
|
||||||
server->xdg_decoration_manager =
|
server->xdg_decoration_manager =
|
||||||
wlr_xdg_decoration_manager_v1_create(server->wl_display);
|
wlr_xdg_decoration_manager_v1_create(server->wl_display);
|
||||||
wl_signal_add(
|
wl_signal_add(&server->xdg_decoration_manager->events.new_toplevel_decoration,
|
||||||
&server->xdg_decoration_manager->events.new_toplevel_decoration,
|
&server->xdg_decoration);
|
||||||
&server->xdg_decoration);
|
server->xdg_decoration.notify = handle_xdg_decoration;
|
||||||
server->xdg_decoration.notify = handle_xdg_decoration;
|
wl_list_init(&server->xdg_decorations);
|
||||||
wl_list_init(&server->xdg_decorations);
|
|
||||||
|
|
||||||
server->relative_pointer_manager =
|
server->relative_pointer_manager =
|
||||||
wlr_relative_pointer_manager_v1_create(server->wl_display);
|
wlr_relative_pointer_manager_v1_create(server->wl_display);
|
||||||
|
|
||||||
server->pointer_constraints =
|
server->pointer_constraints =
|
||||||
wlr_pointer_constraints_v1_create(server->wl_display);
|
wlr_pointer_constraints_v1_create(server->wl_display);
|
||||||
server->pointer_constraint.notify = handle_pointer_constraint;
|
server->pointer_constraint.notify = handle_pointer_constraint;
|
||||||
wl_signal_add(&server->pointer_constraints->events.new_constraint,
|
wl_signal_add(&server->pointer_constraints->events.new_constraint,
|
||||||
&server->pointer_constraint);
|
&server->pointer_constraint);
|
||||||
|
|
||||||
wlr_presentation_create(server->wl_display, server->backend, SWAY_PRESENTATION_VERSION);
|
wlr_presentation_create(server->wl_display, server->backend,
|
||||||
wlr_alpha_modifier_v1_create(server->wl_display);
|
SWAY_PRESENTATION_VERSION);
|
||||||
|
wlr_alpha_modifier_v1_create(server->wl_display);
|
||||||
|
|
||||||
server->output_manager_v1 =
|
server->output_manager_v1 = wlr_output_manager_v1_create(server->wl_display);
|
||||||
wlr_output_manager_v1_create(server->wl_display);
|
server->output_manager_apply.notify = handle_output_manager_apply;
|
||||||
server->output_manager_apply.notify = handle_output_manager_apply;
|
wl_signal_add(&server->output_manager_v1->events.apply,
|
||||||
wl_signal_add(&server->output_manager_v1->events.apply,
|
&server->output_manager_apply);
|
||||||
&server->output_manager_apply);
|
|
||||||
server->output_manager_test.notify = handle_output_manager_test;
|
server->output_manager_test.notify = handle_output_manager_test;
|
||||||
wl_signal_add(&server->output_manager_v1->events.test,
|
wl_signal_add(&server->output_manager_v1->events.test,
|
||||||
&server->output_manager_test);
|
&server->output_manager_test);
|
||||||
|
|
@ -405,6 +427,15 @@ bool server_init(struct sway_server *server) {
|
||||||
wlr_content_type_manager_v1_create(server->wl_display, 1);
|
wlr_content_type_manager_v1_create(server->wl_display, 1);
|
||||||
wlr_fractional_scale_manager_v1_create(server->wl_display, 1);
|
wlr_fractional_scale_manager_v1_create(server->wl_display, 1);
|
||||||
|
|
||||||
|
server->ext_foreign_toplevel_image_capture_source_manager_v1 =
|
||||||
|
wlr_ext_foreign_toplevel_image_capture_source_manager_v1_create(
|
||||||
|
server->wl_display, 1);
|
||||||
|
server->new_foreign_toplevel_capture_request.notify =
|
||||||
|
handle_new_foreign_toplevel_capture_request;
|
||||||
|
wl_signal_add(&server->ext_foreign_toplevel_image_capture_source_manager_v1
|
||||||
|
->events.new_request,
|
||||||
|
&server->new_foreign_toplevel_capture_request);
|
||||||
|
|
||||||
server->tearing_control_v1 =
|
server->tearing_control_v1 =
|
||||||
wlr_tearing_control_manager_v1_create(server->wl_display, 1);
|
wlr_tearing_control_manager_v1_create(server->wl_display, 1);
|
||||||
server->tearing_control_new_object.notify = handle_new_tearing_hint;
|
server->tearing_control_new_object.notify = handle_new_tearing_hint;
|
||||||
|
|
@ -495,11 +526,19 @@ void server_fini(struct sway_server *server) {
|
||||||
wl_list_remove(&server->drm_lease_request.link);
|
wl_list_remove(&server->drm_lease_request.link);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
wl_list_remove(&server->tearing_control_new_object.link);
|
<<<<<<< HEAD wl_list_remove(&server->tearing_control_new_object.link);
|
||||||
wl_list_remove(&server->xdg_activation_v1_request_activate.link);
|
wl_list_remove(&server->xdg_activation_v1_request_activate.link);
|
||||||
wl_list_remove(&server->xdg_activation_v1_new_token.link);
|
wl_list_remove(&server->xdg_activation_v1_new_token.link);
|
||||||
wl_list_remove(&server->request_set_cursor_shape.link);
|
wl_list_remove(&server->request_set_cursor_shape.link);
|
||||||
input_manager_finish(server->input);
|
input_manager_finish(server->input);
|
||||||
|
=======
|
||||||
|
wl_list_remove(&server->tearing_control_new_object.link);
|
||||||
|
wl_list_remove(&server->xdg_activation_v1_request_activate.link);
|
||||||
|
wl_list_remove(&server->xdg_activation_v1_new_token.link);
|
||||||
|
wl_list_remove(&server->request_set_cursor_shape.link);
|
||||||
|
wl_list_remove(&server->new_foreign_toplevel_capture_request.link);
|
||||||
|
input_manager_finish(server->input);
|
||||||
|
>>>>>>> 170c9c95 (Add support for toplevel capture)
|
||||||
|
|
||||||
// TODO: free sway-specific resources
|
// TODO: free sway-specific resources
|
||||||
#if WLR_HAS_XWAYLAND
|
#if WLR_HAS_XWAYLAND
|
||||||
|
|
|
||||||
|
|
@ -49,6 +49,11 @@ bool view_init(struct sway_view *view, enum sway_view_type type,
|
||||||
failed = true;
|
failed = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
view->image_capture_scene = wlr_scene_create();
|
||||||
|
if (view->image_capture_scene == NULL) {
|
||||||
|
failed = true;
|
||||||
|
}
|
||||||
|
|
||||||
if (failed) {
|
if (failed) {
|
||||||
wlr_scene_node_destroy(&view->scene_tree->node);
|
wlr_scene_node_destroy(&view->scene_tree->node);
|
||||||
return false;
|
return false;
|
||||||
|
|
@ -81,6 +86,7 @@ void view_destroy(struct sway_view *view) {
|
||||||
list_free(view->executed_criteria);
|
list_free(view->executed_criteria);
|
||||||
|
|
||||||
view_assign_ctx(view, NULL);
|
view_assign_ctx(view, NULL);
|
||||||
|
wlr_scene_node_destroy(&view->image_capture_scene->tree.node);
|
||||||
wlr_scene_node_destroy(&view->scene_tree->node);
|
wlr_scene_node_destroy(&view->scene_tree->node);
|
||||||
if (view->impl->destroy) {
|
if (view->impl->destroy) {
|
||||||
view->impl->destroy(view);
|
view->impl->destroy(view);
|
||||||
|
|
@ -815,6 +821,7 @@ void view_map(struct sway_view *view, struct wlr_surface *wlr_surface,
|
||||||
};
|
};
|
||||||
view->ext_foreign_toplevel =
|
view->ext_foreign_toplevel =
|
||||||
wlr_ext_foreign_toplevel_handle_v1_create(server.foreign_toplevel_list, &foreign_toplevel_state);
|
wlr_ext_foreign_toplevel_handle_v1_create(server.foreign_toplevel_list, &foreign_toplevel_state);
|
||||||
|
view->ext_foreign_toplevel->data = view;
|
||||||
|
|
||||||
view->foreign_toplevel =
|
view->foreign_toplevel =
|
||||||
wlr_foreign_toplevel_handle_v1_create(server.foreign_toplevel_manager);
|
wlr_foreign_toplevel_handle_v1_create(server.foreign_toplevel_manager);
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue