| 
									
										
										
										
											2018-08-16 10:54:45 +01:00
										 |  |  | #define _POSIX_C_SOURCE 200809L
 | 
					
						
							| 
									
										
										
										
											2018-11-11 18:14:48 +13:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-25 19:33:13 -04:00
										 |  |  | #include <assert.h>
 | 
					
						
							| 
									
										
										
										
											2017-04-25 19:19:21 -04:00
										 |  |  | #include <stdint.h>
 | 
					
						
							| 
									
										
										
										
											2018-02-12 21:29:23 +01:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							| 
									
										
										
										
											2017-04-25 21:26:29 -04:00
										 |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | #include <time.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 23:55:51 +01:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2018-11-11 18:14:48 +13:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-25 19:19:21 -04:00
										 |  |  | #include <wayland-client.h>
 | 
					
						
							| 
									
										
										
										
											2018-11-11 18:14:48 +13:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-21 10:27:45 -04:00
										 |  |  | #include <wlr/interfaces/wlr_keyboard.h>
 | 
					
						
							| 
									
										
										
										
											2018-02-12 21:29:23 +01:00
										 |  |  | #include <wlr/interfaces/wlr_output.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-02 14:17:31 -05:00
										 |  |  | #include <wlr/interfaces/wlr_switch.h>
 | 
					
						
							| 
									
										
										
										
											2017-06-22 16:53:48 +02:00
										 |  |  | #include <wlr/interfaces/wlr_touch.h>
 | 
					
						
							| 
									
										
										
										
											2022-02-09 14:08:20 -05:00
										 |  |  | #include <wlr/interfaces/wlr_tablet_tool.h>
 | 
					
						
							|  |  |  | #include <wlr/interfaces/wlr_tablet_pad.h>
 | 
					
						
							| 
									
										
										
										
											2017-06-21 12:10:07 -04:00
										 |  |  | #include <wlr/util/log.h>
 | 
					
						
							| 
									
										
										
										
											2018-11-11 18:14:48 +13:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 09:43:38 -05:00
										 |  |  | #include "interfaces/wlr_input_device.h"
 | 
					
						
							| 
									
										
										
										
											2017-04-25 19:19:21 -04:00
										 |  |  | #include "backend/wayland.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-12 21:29:23 +01:00
										 |  |  | #include "util/signal.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-05 17:44:30 -04:00
										 |  |  | #include "util/time.h"
 | 
					
						
							| 
									
										
										
										
											2017-04-25 19:19:21 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 14:08:20 -05:00
										 |  |  | static const struct wlr_touch_impl touch_impl; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | static void keyboard_handle_keymap(void *data, struct wl_keyboard *wl_keyboard, | 
					
						
							|  |  |  | 		uint32_t format, int32_t fd, uint32_t size) { | 
					
						
							| 
									
										
										
										
											2020-03-16 23:55:51 +01:00
										 |  |  | 	close(fd); | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void keyboard_handle_enter(void *data, struct wl_keyboard *wl_keyboard, | 
					
						
							|  |  |  | 		uint32_t serial, struct wl_surface *surface, struct wl_array *keys) { | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	struct wlr_keyboard *keyboard = data; | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	uint32_t *keycode_ptr; | 
					
						
							|  |  |  | 	wl_array_for_each(keycode_ptr, keys) { | 
					
						
							|  |  |  | 		struct wlr_event_keyboard_key event = { | 
					
						
							|  |  |  | 			.keycode = *keycode_ptr, | 
					
						
							| 
									
										
										
										
											2020-10-21 17:21:23 +02:00
										 |  |  | 			.state = WL_KEYBOARD_KEY_STATE_PRESSED, | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 			.time_msec = get_current_time_msec(), | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 			.update_state = false, | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 		wlr_keyboard_notify_key(keyboard, &event); | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void keyboard_handle_leave(void *data, struct wl_keyboard *wl_keyboard, | 
					
						
							|  |  |  | 		uint32_t serial, struct wl_surface *surface) { | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	struct wlr_keyboard *keyboard = data; | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	size_t num_keycodes = keyboard->num_keycodes; | 
					
						
							| 
									
										
										
										
											2021-01-05 12:08:23 +01:00
										 |  |  | 	uint32_t pressed[num_keycodes + 1]; | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	memcpy(pressed, keyboard->keycodes, | 
					
						
							| 
									
										
										
										
											2021-01-05 12:08:23 +01:00
										 |  |  | 		num_keycodes * sizeof(uint32_t)); | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-05 12:08:23 +01:00
										 |  |  | 	for (size_t i = 0; i < num_keycodes; ++i) { | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 		uint32_t keycode = pressed[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		struct wlr_event_keyboard_key event = { | 
					
						
							|  |  |  | 			.keycode = keycode, | 
					
						
							| 
									
										
										
										
											2020-10-21 17:21:23 +02:00
										 |  |  | 			.state = WL_KEYBOARD_KEY_STATE_RELEASED, | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 			.time_msec = get_current_time_msec(), | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 			.update_state = false, | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 		wlr_keyboard_notify_key(keyboard, &event); | 
					
						
							| 
									
										
										
										
											2018-06-03 13:18:57 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void keyboard_handle_key(void *data, struct wl_keyboard *wl_keyboard, | 
					
						
							|  |  |  | 		uint32_t serial, uint32_t time, uint32_t key, uint32_t state) { | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	struct wlr_keyboard *keyboard = data; | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-21 17:58:29 +01:00
										 |  |  | 	struct wlr_event_keyboard_key wlr_event = { | 
					
						
							|  |  |  | 		.keycode = key, | 
					
						
							|  |  |  | 		.state = state, | 
					
						
							|  |  |  | 		.time_msec = time, | 
					
						
							|  |  |  | 		.update_state = false, | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	wlr_keyboard_notify_key(keyboard, &wlr_event); | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void keyboard_handle_modifiers(void *data, struct wl_keyboard *wl_keyboard, | 
					
						
							|  |  |  | 		uint32_t serial, uint32_t mods_depressed, uint32_t mods_latched, | 
					
						
							|  |  |  | 		uint32_t mods_locked, uint32_t group) { | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	struct wlr_keyboard *keyboard = data; | 
					
						
							|  |  |  | 	wlr_keyboard_notify_modifiers(keyboard, mods_depressed, mods_latched, | 
					
						
							| 
									
										
										
										
											2017-10-06 12:03:34 +03:00
										 |  |  | 		mods_locked, group); | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 22:10:34 +02:00
										 |  |  | static void keyboard_handle_repeat_info(void *data, | 
					
						
							|  |  |  | 		struct wl_keyboard *wl_keyboard, int32_t rate, int32_t delay) { | 
					
						
							|  |  |  | 	// This space is intentionally left blank
 | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-04 08:57:18 -05:00
										 |  |  | static const struct wl_keyboard_listener keyboard_listener = { | 
					
						
							| 
									
										
										
										
											2017-06-22 15:41:59 +02:00
										 |  |  | 	.keymap = keyboard_handle_keymap, | 
					
						
							|  |  |  | 	.enter = keyboard_handle_enter, | 
					
						
							|  |  |  | 	.leave = keyboard_handle_leave, | 
					
						
							|  |  |  | 	.key = keyboard_handle_key, | 
					
						
							|  |  |  | 	.modifiers = keyboard_handle_modifiers, | 
					
						
							|  |  |  | 	.repeat_info = keyboard_handle_repeat_info | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | static const struct wlr_keyboard_impl keyboard_impl = { | 
					
						
							|  |  |  | 	.name = "wl-keyboard", | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void init_seat_keyboard(struct wlr_wl_seat *seat) { | 
					
						
							|  |  |  | 	assert(seat->wl_keyboard); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	char name[128] = {0}; | 
					
						
							|  |  |  | 	snprintf(name, sizeof(name), "wayland-keyboard-%s", seat->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wlr_keyboard_init(&seat->wlr_keyboard, &keyboard_impl, name); | 
					
						
							|  |  |  | 	wl_keyboard_add_listener(seat->wl_keyboard, &keyboard_listener, | 
					
						
							|  |  |  | 		&seat->wlr_keyboard); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wlr_signal_emit_safe(&seat->backend->backend.events.new_input, | 
					
						
							|  |  |  | 		&seat->wlr_keyboard.base); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | static void touch_coordinates_to_absolute(struct wlr_wl_input_device *device, | 
					
						
							|  |  |  | 		wl_fixed_t x, wl_fixed_t y, double *sx, double *sy) { | 
					
						
							|  |  |  | 	// TODO: each output needs its own touch
 | 
					
						
							|  |  |  | 	struct wlr_wl_output *output, *tmp; | 
					
						
							|  |  |  | 	wl_list_for_each_safe(output, tmp, &device->backend->outputs, link) { | 
					
						
							|  |  |  | 		*sx = wl_fixed_to_double(x) / output->wlr_output.width; | 
					
						
							|  |  |  | 		*sy = wl_fixed_to_double(y) / output->wlr_output.height; | 
					
						
							|  |  |  | 		return; // Choose the first output in the list
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	*sx = *sy = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void touch_handle_down(void *data, struct wl_touch *wl_touch, | 
					
						
							|  |  |  | 		uint32_t serial, uint32_t time, struct wl_surface *surface, | 
					
						
							|  |  |  | 		int32_t id, wl_fixed_t x, wl_fixed_t y) { | 
					
						
							|  |  |  | 	struct wlr_wl_input_device *device = data; | 
					
						
							|  |  |  | 	assert(device && device->wlr_input_device.touch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	double sx, sy; | 
					
						
							|  |  |  | 	touch_coordinates_to_absolute(device, x, y, &sx, &sy); | 
					
						
							|  |  |  | 	struct wlr_event_touch_down event = { | 
					
						
							|  |  |  | 		.device = &device->wlr_input_device, | 
					
						
							|  |  |  | 		.time_msec = time, | 
					
						
							|  |  |  | 		.touch_id = id, | 
					
						
							|  |  |  | 		.x = sx, | 
					
						
							|  |  |  | 		.y = sy | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	wlr_signal_emit_safe(&device->wlr_input_device.touch->events.down, &event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void touch_handle_up(void *data, struct wl_touch *wl_touch, | 
					
						
							|  |  |  | 		uint32_t serial, uint32_t time, int32_t id) { | 
					
						
							|  |  |  | 	struct wlr_wl_input_device *device = data; | 
					
						
							|  |  |  | 	assert(device && device->wlr_input_device.touch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct wlr_event_touch_up event = { | 
					
						
							|  |  |  | 		.device = &device->wlr_input_device, | 
					
						
							|  |  |  | 		.time_msec = time, | 
					
						
							|  |  |  | 		.touch_id = id, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	wlr_signal_emit_safe(&device->wlr_input_device.touch->events.up, &event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void touch_handle_motion(void *data, struct wl_touch *wl_touch, | 
					
						
							|  |  |  | 		uint32_t time, int32_t id, wl_fixed_t x, wl_fixed_t y) { | 
					
						
							|  |  |  | 	struct wlr_wl_input_device *device = data; | 
					
						
							|  |  |  | 	assert(device && device->wlr_input_device.touch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	double sx, sy; | 
					
						
							|  |  |  | 	touch_coordinates_to_absolute(device, x, y, &sx, &sy); | 
					
						
							|  |  |  | 	struct wlr_event_touch_motion event = { | 
					
						
							|  |  |  | 		.device = &device->wlr_input_device, | 
					
						
							|  |  |  | 		.time_msec = time, | 
					
						
							|  |  |  | 		.touch_id = id, | 
					
						
							|  |  |  | 		.x = sx, | 
					
						
							|  |  |  | 		.y = sy | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	wlr_signal_emit_safe(&device->wlr_input_device.touch->events.motion, &event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void touch_handle_frame(void *data, struct wl_touch *wl_touch) { | 
					
						
							| 
									
										
										
										
											2021-06-30 11:39:38 +02:00
										 |  |  | 	struct wlr_wl_input_device *device = data; | 
					
						
							|  |  |  | 	assert(device && device->wlr_input_device.touch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wlr_signal_emit_safe(&device->wlr_input_device.touch->events.frame, NULL); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void touch_handle_cancel(void *data, struct wl_touch *wl_touch) { | 
					
						
							|  |  |  | 	// no-op
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void touch_handle_shape(void *data, struct wl_touch *wl_touch, | 
					
						
							|  |  |  | 		int32_t id, wl_fixed_t major, wl_fixed_t minor) { | 
					
						
							|  |  |  | 	// no-op
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void touch_handle_orientation(void *data, struct wl_touch *wl_touch, | 
					
						
							|  |  |  | 		int32_t id, wl_fixed_t orientation) { | 
					
						
							|  |  |  | 	// no-op
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-04 08:57:18 -05:00
										 |  |  | static const struct wl_touch_listener touch_listener = { | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 	.down = touch_handle_down, | 
					
						
							|  |  |  | 	.up = touch_handle_up, | 
					
						
							|  |  |  | 	.motion = touch_handle_motion, | 
					
						
							|  |  |  | 	.frame = touch_handle_frame, | 
					
						
							|  |  |  | 	.cancel = touch_handle_cancel, | 
					
						
							|  |  |  | 	.shape = touch_handle_shape, | 
					
						
							|  |  |  | 	.orientation = touch_handle_orientation, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 22:10:34 +02:00
										 |  |  | static struct wlr_wl_input_device *get_wl_input_device_from_input_device( | 
					
						
							|  |  |  | 		struct wlr_input_device *wlr_dev) { | 
					
						
							|  |  |  | 	assert(wlr_input_device_is_wl(wlr_dev)); | 
					
						
							|  |  |  | 	return (struct wlr_wl_input_device *)wlr_dev; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-18 19:12:01 +02:00
										 |  |  | bool create_wl_seat(struct wl_seat *wl_seat, struct wlr_wl_backend *wl) { | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 	struct wlr_wl_seat *seat = calloc(1, sizeof(struct wlr_wl_seat)); | 
					
						
							| 
									
										
										
										
											2020-10-18 19:12:01 +02:00
										 |  |  | 	if (!seat) { | 
					
						
							|  |  |  | 		wlr_log_errno(WLR_ERROR, "Allocation failed"); | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 	seat->wl_seat = wl_seat; | 
					
						
							| 
									
										
										
										
											2020-10-03 20:48:11 +02:00
										 |  |  | 	seat->backend = wl; | 
					
						
							| 
									
										
										
										
											2020-10-03 23:06:10 +02:00
										 |  |  | 	wl_list_insert(&wl->seats, &seat->link); | 
					
						
							| 
									
										
										
										
											2020-10-03 20:48:11 +02:00
										 |  |  | 	wl_seat_add_listener(wl_seat, &seat_listener, seat); | 
					
						
							| 
									
										
										
										
											2020-10-18 19:12:01 +02:00
										 |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void destroy_wl_seats(struct wlr_wl_backend *wl) { | 
					
						
							| 
									
										
										
										
											2020-10-03 23:06:10 +02:00
										 |  |  | 	struct wlr_wl_seat *seat, *tmp_seat; | 
					
						
							|  |  |  | 	wl_list_for_each_safe(seat, tmp_seat, &wl->seats, link) { | 
					
						
							|  |  |  | 		if (seat->touch) { | 
					
						
							|  |  |  | 			wl_touch_destroy(seat->touch); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-03-04 21:59:59 -05:00
										 |  |  | 		if (seat->wl_pointer) { | 
					
						
							|  |  |  | 			finish_seat_pointer(seat); | 
					
						
							| 
									
										
										
										
											2020-10-03 23:06:10 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 		if (seat->wl_keyboard) { | 
					
						
							|  |  |  | 			wl_keyboard_release(seat->wl_keyboard); | 
					
						
							|  |  |  | 			wlr_keyboard_finish(&seat->wlr_keyboard); | 
					
						
							| 
									
										
										
										
											2020-10-03 23:06:10 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-03 23:06:10 +02:00
										 |  |  | 		free(seat->name); | 
					
						
							|  |  |  | 		assert(seat->wl_seat); | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 		wl_seat_destroy(seat->wl_seat); | 
					
						
							| 
									
										
										
										
											2020-10-03 23:06:10 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		wl_list_remove(&seat->link); | 
					
						
							|  |  |  | 		free(seat); | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct wlr_wl_seat *input_device_get_seat(struct wlr_input_device *wlr_dev) { | 
					
						
							|  |  |  | 	struct wlr_wl_input_device *dev = | 
					
						
							|  |  |  | 		get_wl_input_device_from_input_device(wlr_dev); | 
					
						
							| 
									
										
										
										
											2020-10-03 22:11:38 +02:00
										 |  |  | 	assert(dev->seat); | 
					
						
							|  |  |  | 	return dev->seat; | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-19 20:20:32 +01:00
										 |  |  | bool wlr_input_device_is_wl(struct wlr_input_device *dev) { | 
					
						
							| 
									
										
										
										
											2022-02-09 14:08:20 -05:00
										 |  |  | 	switch (dev->type) { | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_KEYBOARD: | 
					
						
							|  |  |  | 		return dev->keyboard->impl == &keyboard_impl; | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_POINTER: | 
					
						
							| 
									
										
										
										
											2022-03-04 21:59:59 -05:00
										 |  |  | 		return dev->pointer->impl == &wl_pointer_impl; | 
					
						
							| 
									
										
										
										
											2022-02-09 14:08:20 -05:00
										 |  |  | 	case WLR_INPUT_DEVICE_TOUCH: | 
					
						
							|  |  |  | 		return dev->touch->impl == &touch_impl; | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_TABLET_TOOL: | 
					
						
							|  |  |  | 		return dev->tablet->impl == &tablet_impl; | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_TABLET_PAD: | 
					
						
							|  |  |  | 		return dev->tablet_pad->impl == &tablet_pad_impl; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-12-19 20:20:32 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-10 08:36:19 +02:00
										 |  |  | struct wlr_wl_input_device *create_wl_input_device( | 
					
						
							| 
									
										
										
										
											2020-10-03 22:11:38 +02:00
										 |  |  | 		struct wlr_wl_seat *seat, enum wlr_input_device_type type) { | 
					
						
							| 
									
										
										
										
											2018-04-29 12:16:31 +01:00
										 |  |  | 	struct wlr_wl_input_device *dev = | 
					
						
							|  |  |  | 		calloc(1, sizeof(struct wlr_wl_input_device)); | 
					
						
							|  |  |  | 	if (dev == NULL) { | 
					
						
							| 
									
										
										
										
											2018-07-09 22:49:54 +01:00
										 |  |  | 		wlr_log_errno(WLR_ERROR, "Allocation failed"); | 
					
						
							| 
									
										
										
										
											2017-06-19 19:05:10 +02:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-03 22:11:38 +02:00
										 |  |  | 	dev->backend = seat->backend; | 
					
						
							|  |  |  | 	dev->seat = seat; | 
					
						
							| 
									
										
										
										
											2017-06-19 19:05:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-29 12:16:31 +01:00
										 |  |  | 	struct wlr_input_device *wlr_dev = &dev->wlr_input_device; | 
					
						
							| 
									
										
										
										
											2017-06-22 17:54:51 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-01 02:31:45 -05:00
										 |  |  | 	const char *type_name = "unknown"; | 
					
						
							| 
									
										
										
										
											2021-12-01 05:40:23 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	switch (type) { | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_KEYBOARD: | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 		wlr_log(WLR_ERROR, "can't create keyboard wlr_wl_input_device"); | 
					
						
							|  |  |  | 		free(dev); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2021-12-01 05:40:23 +09:00
										 |  |  | 	case WLR_INPUT_DEVICE_POINTER: | 
					
						
							| 
									
										
										
										
											2022-03-04 21:59:59 -05:00
										 |  |  | 		wlr_log(WLR_ERROR, "can't create pointer wlr_wl_input_device"); | 
					
						
							|  |  |  | 		free(dev); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2021-12-01 05:40:23 +09:00
										 |  |  | 	case WLR_INPUT_DEVICE_TOUCH: | 
					
						
							|  |  |  | 		type_name = "touch"; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_TABLET_TOOL: | 
					
						
							|  |  |  | 		type_name = "tablet-tool"; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_TABLET_PAD: | 
					
						
							|  |  |  | 		type_name = "tablet-pad"; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case WLR_INPUT_DEVICE_SWITCH: | 
					
						
							|  |  |  | 		type_name = "switch"; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	size_t name_size = 8 + strlen(type_name) + strlen(seat->name) + 1; | 
					
						
							| 
									
										
										
										
											2020-10-04 22:45:36 +02:00
										 |  |  | 	char name[name_size]; | 
					
						
							| 
									
										
										
										
											2021-12-01 05:40:23 +09:00
										 |  |  | 	(void) snprintf(name, name_size, "wayland-%s-%s", type_name, seat->name); | 
					
						
							| 
									
										
										
										
											2020-10-04 22:45:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 16:14:56 -05:00
										 |  |  | 	wlr_input_device_init(wlr_dev, type, name); | 
					
						
							| 
									
										
										
										
											2021-11-22 22:30:40 +01:00
										 |  |  | 	wl_list_insert(&seat->backend->devices, &dev->link); | 
					
						
							| 
									
										
										
										
											2018-04-29 12:16:31 +01:00
										 |  |  | 	return dev; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-02 13:57:28 -05:00
										 |  |  | void destroy_wl_input_device(struct wlr_wl_input_device *dev) { | 
					
						
							|  |  |  | 	/**
 | 
					
						
							|  |  |  | 	 * TODO remove the redundant wlr_input_device from wlr_wl_input_device | 
					
						
							|  |  |  | 	 * wlr_wl_input_device::wlr_input_device is not owned by its input device | 
					
						
							|  |  |  | 	 * type, which means we have 2 wlr_input_device to cleanup | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	wlr_input_device_finish(&dev->wlr_input_device); | 
					
						
							|  |  |  | 	if (dev->wlr_input_device._device) { | 
					
						
							|  |  |  | 		struct wlr_input_device *wlr_dev = &dev->wlr_input_device; | 
					
						
							|  |  |  | 		switch (wlr_dev->type) { | 
					
						
							|  |  |  | 		case WLR_INPUT_DEVICE_KEYBOARD: | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 			wlr_log(WLR_ERROR, "wlr_wl_input_device has no keyboard"); | 
					
						
							| 
									
										
										
										
											2022-03-02 13:57:28 -05:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case WLR_INPUT_DEVICE_POINTER: | 
					
						
							| 
									
										
										
										
											2022-03-04 21:59:59 -05:00
										 |  |  | 			wlr_log(WLR_ERROR, "wlr_wl_input_device has no pointer"); | 
					
						
							| 
									
										
										
										
											2022-03-02 13:57:28 -05:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2022-03-02 14:17:31 -05:00
										 |  |  | 		case WLR_INPUT_DEVICE_SWITCH: | 
					
						
							|  |  |  | 			wlr_switch_finish(wlr_dev->switch_device); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2022-03-02 15:11:25 -05:00
										 |  |  | 		case WLR_INPUT_DEVICE_TABLET_PAD: | 
					
						
							|  |  |  | 			wlr_tablet_pad_finish(wlr_dev->tablet_pad); | 
					
						
							|  |  |  | 			free(wlr_dev->tablet_pad); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2022-03-02 15:58:44 -05:00
										 |  |  | 		case WLR_INPUT_DEVICE_TABLET_TOOL: | 
					
						
							|  |  |  | 			wlr_tablet_finish(wlr_dev->tablet); | 
					
						
							|  |  |  | 			free(wlr_dev->tablet); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2022-03-02 16:07:40 -05:00
										 |  |  | 		case WLR_INPUT_DEVICE_TOUCH: | 
					
						
							|  |  |  | 			wlr_touch_finish(wlr_dev->touch); | 
					
						
							|  |  |  | 			free(wlr_dev->touch); | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2022-03-02 13:57:28 -05:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	wl_list_remove(&dev->link); | 
					
						
							|  |  |  | 	free(dev); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-04-29 12:16:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-03 22:11:38 +02:00
										 |  |  | void create_wl_touch(struct wlr_wl_seat *seat) { | 
					
						
							|  |  |  | 	assert(seat->touch); | 
					
						
							|  |  |  | 	struct wl_touch *wl_touch = seat->touch; | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 	struct wlr_wl_input_device *dev = | 
					
						
							| 
									
										
										
										
											2020-10-03 22:11:38 +02:00
										 |  |  | 		create_wl_input_device(seat, WLR_INPUT_DEVICE_TOUCH); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 	if (!dev) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct wlr_input_device *wlr_dev = &dev->wlr_input_device; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wlr_dev->touch = calloc(1, sizeof(*wlr_dev->touch)); | 
					
						
							|  |  |  | 	if (!wlr_dev->touch) { | 
					
						
							|  |  |  | 		wlr_log_errno(WLR_ERROR, "Allocation failed"); | 
					
						
							| 
									
										
										
										
											2022-02-09 14:08:20 -05:00
										 |  |  | 		destroy_wl_input_device(dev); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-02-09 14:08:20 -05:00
										 |  |  | 	wlr_touch_init(wlr_dev->touch, &touch_impl, wlr_dev->name); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	wl_touch_add_listener(wl_touch, &touch_listener, dev); | 
					
						
							| 
									
										
										
										
											2020-10-03 22:11:38 +02:00
										 |  |  | 	wlr_signal_emit_safe(&seat->backend->backend.events.new_input, wlr_dev); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-25 19:19:21 -04:00
										 |  |  | static void seat_handle_capabilities(void *data, struct wl_seat *wl_seat, | 
					
						
							|  |  |  | 		enum wl_seat_capability caps) { | 
					
						
							| 
									
										
										
										
											2020-10-03 20:48:11 +02:00
										 |  |  | 	struct wlr_wl_seat *seat = data; | 
					
						
							|  |  |  | 	struct wlr_wl_backend *backend = seat->backend; | 
					
						
							| 
									
										
										
										
											2017-04-25 19:33:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-04 21:59:59 -05:00
										 |  |  | 	if ((caps & WL_SEAT_CAPABILITY_POINTER) && seat->wl_pointer == NULL) { | 
					
						
							|  |  |  | 		wlr_log(WLR_DEBUG, "seat '%s' offering pointer", seat->name); | 
					
						
							| 
									
										
										
										
											2018-04-29 12:16:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-04 21:59:59 -05:00
										 |  |  | 		seat->wl_pointer = wl_seat_get_pointer(wl_seat); | 
					
						
							|  |  |  | 		init_seat_pointer(seat); | 
					
						
							| 
									
										
										
										
											2020-03-02 17:17:28 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-03-04 21:59:59 -05:00
										 |  |  | 	if (!(caps & WL_SEAT_CAPABILITY_POINTER) && seat->wl_pointer != NULL) { | 
					
						
							|  |  |  | 		wlr_log(WLR_DEBUG, "seat '%s' dropping pointer", seat->name); | 
					
						
							|  |  |  | 		finish_seat_pointer(seat); | 
					
						
							| 
									
										
										
										
											2017-06-19 19:05:10 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-03-02 17:17:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	if ((caps & WL_SEAT_CAPABILITY_KEYBOARD) && seat->wl_keyboard == NULL) { | 
					
						
							|  |  |  | 		wlr_log(WLR_DEBUG, "seat '%s' offering keyboard", seat->name); | 
					
						
							| 
									
										
										
										
											2017-06-19 19:05:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 16:53:48 +02:00
										 |  |  | 		struct wl_keyboard *wl_keyboard = wl_seat_get_keyboard(wl_seat); | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 		seat->wl_keyboard = wl_keyboard; | 
					
						
							| 
									
										
										
										
											2018-11-11 18:09:04 +13:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (backend->started) { | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 			init_seat_keyboard(seat); | 
					
						
							| 
									
										
										
										
											2018-11-11 18:09:04 +13:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-04-25 19:33:13 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 	if (!(caps & WL_SEAT_CAPABILITY_KEYBOARD) && seat->wl_keyboard != NULL) { | 
					
						
							|  |  |  | 		wlr_log(WLR_DEBUG, "seat '%s' dropping keyboard", seat->name); | 
					
						
							| 
									
										
										
										
											2020-03-02 17:17:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 		wl_keyboard_release(seat->wl_keyboard); | 
					
						
							|  |  |  | 		wlr_keyboard_finish(&seat->wlr_keyboard); | 
					
						
							| 
									
										
										
										
											2020-10-04 22:28:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-03 13:23:08 -05:00
										 |  |  | 		seat->wl_keyboard = NULL; | 
					
						
							| 
									
										
										
										
											2020-03-02 17:17:28 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 	if ((caps & WL_SEAT_CAPABILITY_TOUCH) && seat->touch == NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 		wlr_log(WLR_DEBUG, "seat %p offered touch", (void *)wl_seat); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 		seat->touch = wl_seat_get_touch(wl_seat); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 		if (backend->started) { | 
					
						
							| 
									
										
										
										
											2020-10-03 22:11:38 +02:00
										 |  |  | 			create_wl_touch(seat); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 	if (!(caps & WL_SEAT_CAPABILITY_TOUCH) && seat->touch != NULL) { | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 		wlr_log(WLR_DEBUG, "seat %p dropped touch", (void *)wl_seat); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-22 22:30:40 +01:00
										 |  |  | 		struct wlr_wl_input_device *device, *tmp; | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 		wl_list_for_each_safe(device, tmp, &backend->devices, link) { | 
					
						
							| 
									
										
										
										
											2021-11-22 22:30:40 +01:00
										 |  |  | 			if (device->wlr_input_device.type == WLR_INPUT_DEVICE_TOUCH) { | 
					
						
							| 
									
										
										
										
											2022-02-09 14:08:20 -05:00
										 |  |  | 				destroy_wl_input_device(device); | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 		wl_touch_release(seat->touch); | 
					
						
							|  |  |  | 		seat->touch = NULL; | 
					
						
							| 
									
										
										
										
											2020-10-18 16:33:03 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-04-25 19:19:21 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 22:10:34 +02:00
										 |  |  | static void seat_handle_name(void *data, struct wl_seat *wl_seat, | 
					
						
							|  |  |  | 		const char *name) { | 
					
						
							| 
									
										
										
										
											2020-10-03 20:48:11 +02:00
										 |  |  | 	struct wlr_wl_seat *seat = data; | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 	free(seat->name); | 
					
						
							|  |  |  | 	seat->name = strdup(name); | 
					
						
							| 
									
										
										
										
											2017-04-25 19:19:21 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const struct wl_seat_listener seat_listener = { | 
					
						
							|  |  |  | 	.capabilities = seat_handle_capabilities, | 
					
						
							|  |  |  | 	.name = seat_handle_name, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-11-12 17:56:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct wl_seat *wlr_wl_input_device_get_seat(struct wlr_input_device *wlr_dev) { | 
					
						
							| 
									
										
										
										
											2020-10-03 17:34:32 +02:00
										 |  |  | 	return input_device_get_seat(wlr_dev)->wl_seat; | 
					
						
							| 
									
										
										
										
											2019-11-12 17:56:57 +01:00
										 |  |  | } |