mirror of
				https://gitlab.freedesktop.org/wlroots/wlroots.git
				synced 2025-11-03 09:01:40 -05:00 
			
		
		
		
	Instead of calling xcb_flush() directly, wait until the FD is writable. Ideally we'd have a non-blocking variant instead of xcb_flush(), but libxcb doesn't have this. Also libxcb blocks when its internal buffer is full, but not much we can do here.
		
			
				
	
	
		
			345 lines
		
	
	
	
		
			9.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
	
		
			9.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
#include <assert.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <wlr/types/wlr_data_device.h>
 | 
						|
#include <wlr/util/log.h>
 | 
						|
#include <xcb/xfixes.h>
 | 
						|
#include "xwayland/selection.h"
 | 
						|
#include "xwayland/xwm.h"
 | 
						|
 | 
						|
void xwm_selection_transfer_remove_event_source(
 | 
						|
		struct wlr_xwm_selection_transfer *transfer) {
 | 
						|
	if (transfer->event_source != NULL) {
 | 
						|
		wl_event_source_remove(transfer->event_source);
 | 
						|
		transfer->event_source = NULL;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void xwm_selection_transfer_close_wl_client_fd(
 | 
						|
		struct wlr_xwm_selection_transfer *transfer) {
 | 
						|
	if (transfer->wl_client_fd >= 0) {
 | 
						|
		close(transfer->wl_client_fd);
 | 
						|
		transfer->wl_client_fd = -1;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void xwm_selection_transfer_destroy_property_reply(
 | 
						|
		struct wlr_xwm_selection_transfer *transfer) {
 | 
						|
	free(transfer->property_reply);
 | 
						|
	transfer->property_reply = NULL;
 | 
						|
}
 | 
						|
 | 
						|
void xwm_selection_transfer_init(struct wlr_xwm_selection_transfer *transfer,
 | 
						|
		struct wlr_xwm_selection *selection) {
 | 
						|
	*transfer = (struct wlr_xwm_selection_transfer){
 | 
						|
		.selection = selection,
 | 
						|
		.wl_client_fd = -1,
 | 
						|
	};
 | 
						|
}
 | 
						|
 | 
						|
void xwm_selection_transfer_destroy(
 | 
						|
		struct wlr_xwm_selection_transfer *transfer) {
 | 
						|
	if (!transfer) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	xwm_selection_transfer_destroy_property_reply(transfer);
 | 
						|
	xwm_selection_transfer_remove_event_source(transfer);
 | 
						|
	xwm_selection_transfer_close_wl_client_fd(transfer);
 | 
						|
 | 
						|
	if (transfer->incoming_window) {
 | 
						|
		struct wlr_xwm *xwm = transfer->selection->xwm;
 | 
						|
		xcb_destroy_window(xwm->xcb_conn, transfer->incoming_window);
 | 
						|
		xwm_schedule_flush(xwm);
 | 
						|
	}
 | 
						|
 | 
						|
	wl_list_remove(&transfer->link);
 | 
						|
	free(transfer);
 | 
						|
}
 | 
						|
 | 
						|
xcb_atom_t xwm_mime_type_to_atom(struct wlr_xwm *xwm, char *mime_type) {
 | 
						|
	if (strcmp(mime_type, "text/plain;charset=utf-8") == 0) {
 | 
						|
		return xwm->atoms[UTF8_STRING];
 | 
						|
	} else if (strcmp(mime_type, "text/plain") == 0) {
 | 
						|
		return xwm->atoms[TEXT];
 | 
						|
	}
 | 
						|
 | 
						|
	xcb_intern_atom_cookie_t cookie =
 | 
						|
		xcb_intern_atom(xwm->xcb_conn, 0, strlen(mime_type), mime_type);
 | 
						|
	xcb_intern_atom_reply_t *reply =
 | 
						|
		xcb_intern_atom_reply(xwm->xcb_conn, cookie, NULL);
 | 
						|
	if (reply == NULL) {
 | 
						|
		return XCB_ATOM_NONE;
 | 
						|
	}
 | 
						|
	xcb_atom_t atom = reply->atom;
 | 
						|
	free(reply);
 | 
						|
	return atom;
 | 
						|
}
 | 
						|
 | 
						|
char *xwm_mime_type_from_atom(struct wlr_xwm *xwm, xcb_atom_t atom) {
 | 
						|
	if (atom == xwm->atoms[UTF8_STRING]) {
 | 
						|
		return strdup("text/plain;charset=utf-8");
 | 
						|
	} else if (atom == xwm->atoms[TEXT]) {
 | 
						|
		return strdup("text/plain");
 | 
						|
	} else {
 | 
						|
		return xwm_get_atom_name(xwm, atom);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
struct wlr_xwm_selection *xwm_get_selection(struct wlr_xwm *xwm,
 | 
						|
		xcb_atom_t selection_atom) {
 | 
						|
	if (selection_atom == xwm->atoms[CLIPBOARD]) {
 | 
						|
		return &xwm->clipboard_selection;
 | 
						|
	} else if (selection_atom == xwm->atoms[PRIMARY]) {
 | 
						|
		return &xwm->primary_selection;
 | 
						|
	} else if (selection_atom == xwm->atoms[DND_SELECTION]) {
 | 
						|
		return &xwm->dnd_selection;
 | 
						|
	} else {
 | 
						|
		return NULL;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int xwm_handle_selection_property_notify(struct wlr_xwm *xwm,
 | 
						|
		xcb_property_notify_event_t *event) {
 | 
						|
	struct wlr_xwm_selection *selections[] = {
 | 
						|
		&xwm->clipboard_selection,
 | 
						|
		&xwm->primary_selection,
 | 
						|
		&xwm->dnd_selection,
 | 
						|
	};
 | 
						|
 | 
						|
	for (size_t i = 0; i < sizeof(selections)/sizeof(selections[0]); ++i) {
 | 
						|
		struct wlr_xwm_selection *selection = selections[i];
 | 
						|
 | 
						|
		if (event->state == XCB_PROPERTY_NEW_VALUE &&
 | 
						|
				event->atom == xwm->atoms[WL_SELECTION]) {
 | 
						|
			struct wlr_xwm_selection_transfer *transfer =
 | 
						|
				xwm_selection_find_incoming_transfer_by_window(selection,
 | 
						|
						event->window);
 | 
						|
			if (transfer) {
 | 
						|
				if (transfer->incr) {
 | 
						|
					xwm_get_incr_chunk(transfer);
 | 
						|
				}
 | 
						|
 | 
						|
				return 1;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		struct wlr_xwm_selection_transfer *outgoing;
 | 
						|
		wl_list_for_each(outgoing, &selection->outgoing, link) {
 | 
						|
			if (event->window == outgoing->request.requestor) {
 | 
						|
				if (event->state == XCB_PROPERTY_DELETE &&
 | 
						|
						event->atom == outgoing->request.property &&
 | 
						|
						outgoing->incr) {
 | 
						|
					xwm_send_incr_chunk(outgoing);
 | 
						|
				}
 | 
						|
				return 1;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int xwm_handle_selection_event(struct wlr_xwm *xwm,
 | 
						|
		xcb_generic_event_t *event) {
 | 
						|
	if (xwm->seat == NULL) {
 | 
						|
		wlr_log(WLR_DEBUG, "not handling selection events: "
 | 
						|
			"no seat assigned to xwayland");
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
 | 
						|
	switch (event->response_type & XCB_EVENT_RESPONSE_TYPE_MASK) {
 | 
						|
	case XCB_SELECTION_NOTIFY:
 | 
						|
		xwm_handle_selection_notify(xwm, (xcb_selection_notify_event_t *)event);
 | 
						|
		return 1;
 | 
						|
	case XCB_PROPERTY_NOTIFY:
 | 
						|
		return xwm_handle_selection_property_notify(xwm,
 | 
						|
			(xcb_property_notify_event_t *)event);
 | 
						|
	case XCB_SELECTION_REQUEST:
 | 
						|
		xwm_handle_selection_request(xwm,
 | 
						|
			(xcb_selection_request_event_t *)event);
 | 
						|
		return 1;
 | 
						|
	}
 | 
						|
 | 
						|
	switch (event->response_type - xwm->xfixes->first_event) {
 | 
						|
	case XCB_XFIXES_SELECTION_NOTIFY:
 | 
						|
		// an X11 window has copied something to the clipboard
 | 
						|
		return xwm_handle_xfixes_selection_notify(xwm,
 | 
						|
			(xcb_xfixes_selection_notify_event_t *)event);
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
void xwm_selection_init(struct wlr_xwm_selection *selection,
 | 
						|
		struct wlr_xwm *xwm, xcb_atom_t atom) {
 | 
						|
	*selection = (struct wlr_xwm_selection){
 | 
						|
		.xwm = xwm,
 | 
						|
		.atom = atom,
 | 
						|
		.window = xcb_generate_id(xwm->xcb_conn),
 | 
						|
	};
 | 
						|
	wl_list_init(&selection->incoming);
 | 
						|
	wl_list_init(&selection->outgoing);
 | 
						|
 | 
						|
	if (atom == xwm->atoms[DND_SELECTION]) {
 | 
						|
		xcb_create_window(
 | 
						|
			xwm->xcb_conn,
 | 
						|
			XCB_COPY_FROM_PARENT,
 | 
						|
			selection->window,
 | 
						|
			xwm->screen->root,
 | 
						|
			0, 0,
 | 
						|
			8192, 8192,
 | 
						|
			0,
 | 
						|
			XCB_WINDOW_CLASS_INPUT_ONLY,
 | 
						|
			xwm->screen->root_visual,
 | 
						|
			XCB_CW_EVENT_MASK,
 | 
						|
			(uint32_t[]){
 | 
						|
				XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY | XCB_EVENT_MASK_PROPERTY_CHANGE
 | 
						|
			}
 | 
						|
		);
 | 
						|
 | 
						|
		xcb_change_property(
 | 
						|
			xwm->xcb_conn,
 | 
						|
			XCB_PROP_MODE_REPLACE,
 | 
						|
			selection->window,
 | 
						|
			xwm->atoms[DND_AWARE],
 | 
						|
			XCB_ATOM_ATOM,
 | 
						|
			32, // format
 | 
						|
			1,
 | 
						|
			&(uint32_t){XDND_VERSION}
 | 
						|
		);
 | 
						|
	} else {
 | 
						|
		xcb_create_window(
 | 
						|
			xwm->xcb_conn,
 | 
						|
			XCB_COPY_FROM_PARENT,
 | 
						|
			selection->window,
 | 
						|
			xwm->screen->root,
 | 
						|
			0, 0,
 | 
						|
			10, 10,
 | 
						|
			0,
 | 
						|
			XCB_WINDOW_CLASS_INPUT_OUTPUT,
 | 
						|
			xwm->screen->root_visual,
 | 
						|
			XCB_CW_EVENT_MASK,
 | 
						|
			(uint32_t[]){
 | 
						|
				XCB_EVENT_MASK_SUBSTRUCTURE_NOTIFY | XCB_EVENT_MASK_PROPERTY_CHANGE
 | 
						|
			}
 | 
						|
		);
 | 
						|
 | 
						|
		if (atom == xwm->atoms[CLIPBOARD]) {
 | 
						|
			xcb_set_selection_owner(xwm->xcb_conn, selection->window,
 | 
						|
				xwm->atoms[CLIPBOARD_MANAGER], XCB_TIME_CURRENT_TIME);
 | 
						|
		} else {
 | 
						|
			assert(atom == xwm->atoms[PRIMARY]);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	uint32_t mask =
 | 
						|
		XCB_XFIXES_SELECTION_EVENT_MASK_SET_SELECTION_OWNER |
 | 
						|
		XCB_XFIXES_SELECTION_EVENT_MASK_SELECTION_WINDOW_DESTROY |
 | 
						|
		XCB_XFIXES_SELECTION_EVENT_MASK_SELECTION_CLIENT_CLOSE;
 | 
						|
	xcb_xfixes_select_selection_input(xwm->xcb_conn, selection->window,
 | 
						|
		selection->atom, mask);
 | 
						|
}
 | 
						|
 | 
						|
void xwm_selection_finish(struct wlr_xwm_selection *selection) {
 | 
						|
	if (!selection) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	struct wlr_xwm_selection_transfer *outgoing, *tmp;
 | 
						|
	wl_list_for_each_safe(outgoing, tmp, &selection->outgoing, link) {
 | 
						|
		wlr_log(WLR_INFO, "destroyed pending transfer %p", outgoing);
 | 
						|
		xwm_selection_transfer_destroy_outgoing(outgoing);
 | 
						|
	}
 | 
						|
 | 
						|
	struct wlr_xwm_selection_transfer *incoming;
 | 
						|
	wl_list_for_each_safe(incoming, tmp, &selection->incoming, link) {
 | 
						|
		xwm_selection_transfer_destroy(incoming);
 | 
						|
	}
 | 
						|
 | 
						|
	xcb_destroy_window(selection->xwm->xcb_conn, selection->window);
 | 
						|
}
 | 
						|
 | 
						|
static void xwm_selection_set_owner(struct wlr_xwm_selection *selection,
 | 
						|
		bool set) {
 | 
						|
	if (set) {
 | 
						|
		xcb_set_selection_owner(selection->xwm->xcb_conn,
 | 
						|
			selection->window,
 | 
						|
			selection->atom,
 | 
						|
			XCB_TIME_CURRENT_TIME);
 | 
						|
		xwm_schedule_flush(selection->xwm);
 | 
						|
	} else {
 | 
						|
		if (selection->owner == selection->window) {
 | 
						|
			xcb_set_selection_owner(selection->xwm->xcb_conn,
 | 
						|
				XCB_WINDOW_NONE,
 | 
						|
				selection->atom,
 | 
						|
				selection->timestamp);
 | 
						|
			xwm_schedule_flush(selection->xwm);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void handle_seat_set_selection(struct wl_listener *listener,
 | 
						|
		void *data) {
 | 
						|
	struct wlr_seat *seat = data;
 | 
						|
	struct wlr_xwm *xwm =
 | 
						|
		wl_container_of(listener, xwm, seat_set_selection);
 | 
						|
	struct wlr_data_source *source = seat->selection_source;
 | 
						|
 | 
						|
	if (source != NULL && data_source_is_xwayland(source)) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	xwm_selection_set_owner(&xwm->clipboard_selection, source != NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void handle_seat_set_primary_selection(struct wl_listener *listener,
 | 
						|
		void *data) {
 | 
						|
	struct wlr_seat *seat = data;
 | 
						|
	struct wlr_xwm *xwm =
 | 
						|
		wl_container_of(listener, xwm, seat_set_primary_selection);
 | 
						|
	struct wlr_primary_selection_source *source =
 | 
						|
		seat->primary_selection_source;
 | 
						|
 | 
						|
	if (source != NULL && primary_selection_source_is_xwayland(source)) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	xwm_selection_set_owner(&xwm->primary_selection, source != NULL);
 | 
						|
}
 | 
						|
 | 
						|
static void seat_handle_start_drag(struct wl_listener *listener, void *data) {
 | 
						|
	struct wlr_xwm *xwm = wl_container_of(listener, xwm, seat_start_drag);
 | 
						|
	struct wlr_drag *drag = data;
 | 
						|
 | 
						|
	xwm_selection_set_owner(&xwm->dnd_selection, drag != NULL);
 | 
						|
	xwm_seat_handle_start_drag(xwm, drag);
 | 
						|
}
 | 
						|
 | 
						|
void xwm_set_seat(struct wlr_xwm *xwm, struct wlr_seat *seat) {
 | 
						|
	if (xwm->seat != NULL) {
 | 
						|
		wl_list_remove(&xwm->seat_set_selection.link);
 | 
						|
		wl_list_remove(&xwm->seat_set_primary_selection.link);
 | 
						|
		wl_list_remove(&xwm->seat_start_drag.link);
 | 
						|
		xwm->seat = NULL;
 | 
						|
	}
 | 
						|
 | 
						|
	if (seat == NULL) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	xwm->seat = seat;
 | 
						|
 | 
						|
	wl_signal_add(&seat->events.set_selection, &xwm->seat_set_selection);
 | 
						|
	xwm->seat_set_selection.notify = handle_seat_set_selection;
 | 
						|
	wl_signal_add(&seat->events.set_primary_selection,
 | 
						|
		&xwm->seat_set_primary_selection);
 | 
						|
	xwm->seat_set_primary_selection.notify = handle_seat_set_primary_selection;
 | 
						|
	wl_signal_add(&seat->events.start_drag, &xwm->seat_start_drag);
 | 
						|
	xwm->seat_start_drag.notify = seat_handle_start_drag;
 | 
						|
 | 
						|
	handle_seat_set_selection(&xwm->seat_set_selection, seat);
 | 
						|
	handle_seat_set_primary_selection(&xwm->seat_set_primary_selection, seat);
 | 
						|
}
 |