Blob Blame History Raw
/*
 * Copyright © 2013 Red Hat, Inc.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice (including the next
 * paragraph) shall be included in all copies or substantial portions of the
 * Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 */

#include <config.h>

#include <check.h>
#include <errno.h>
#include <fcntl.h>
#include <libinput.h>
#include <libevdev/libevdev.h>
#include <unistd.h>

#include "libinput-util.h"
#include "litest.h"

START_TEST(touch_frame_events)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *event;
	int have_frame_event = 0;

	litest_drain_events(dev->libinput);

	litest_touch_down(dev, 0, 10, 10);
	libinput_dispatch(li);

	while ((event = libinput_get_event(li))) {
		if (libinput_event_get_type(event) == LIBINPUT_EVENT_TOUCH_FRAME)
			have_frame_event++;
		libinput_event_destroy(event);
	}
	ck_assert_int_eq(have_frame_event, 1);

	litest_touch_down(dev, 1, 10, 10);
	libinput_dispatch(li);

	while ((event = libinput_get_event(li))) {
		if (libinput_event_get_type(event) == LIBINPUT_EVENT_TOUCH_FRAME)
			have_frame_event++;
		libinput_event_destroy(event);
	}
	ck_assert_int_eq(have_frame_event, 2);
}
END_TEST

START_TEST(touch_downup_no_motion)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;

	litest_drain_events(li);

	litest_touch_down(dev, 0, 10, 10);
	libinput_dispatch(li);

	litest_assert_touch_down_frame(li);

	litest_touch_up(dev, 0);
	libinput_dispatch(li);

	litest_assert_touch_up_frame(li);
}
END_TEST

START_TEST(touch_abs_transform)
{
	struct litest_device *dev;
	struct libinput *libinput;
	struct libinput_event *ev;
	struct libinput_event_touch *tev;
	double fx, fy;
	bool tested = false;

	struct input_absinfo abs[] = {
		{ ABS_X, 0, 32767, 75, 0, 10 },
		{ ABS_Y, 0, 32767, 129, 0, 9 },
		{ ABS_MT_POSITION_X, 0, 32767, 0, 0, 10 },
		{ ABS_MT_POSITION_Y, 0, 32767, 0, 0, 9 },
		{ .value = -1 },
	};

	dev = litest_create_device_with_overrides(LITEST_WACOM_TOUCH,
						  "litest Highres touch device",
						  NULL, abs, NULL);

	libinput = dev->libinput;

	litest_touch_down(dev, 0, 100, 100);

	libinput_dispatch(libinput);

	while ((ev = libinput_get_event(libinput))) {
		if (libinput_event_get_type(ev) != LIBINPUT_EVENT_TOUCH_DOWN) {
			libinput_event_destroy(ev);
			continue;
		}

		tev = libinput_event_get_touch_event(ev);
		fx = libinput_event_touch_get_x_transformed(tev, 1920);
		ck_assert_int_eq(fx, 1919.0);
		fy = libinput_event_touch_get_y_transformed(tev, 720);
		ck_assert_int_eq(fy, 719.0);

		tested = true;

		libinput_event_destroy(ev);
	}

	ck_assert(tested);

	litest_delete_device(dev);
}
END_TEST

static inline void
touch_assert_seat_slot(struct libinput *li,
		       enum libinput_event_type type,
		       unsigned int slot,
		       unsigned int seat_slot)
{
	struct libinput_event *ev;
	struct libinput_event_touch *tev;

	libinput_dispatch(li);
	ev = libinput_get_event(li);
	tev = litest_is_touch_event(ev, type);
	slot = libinput_event_touch_get_slot(tev);
	ck_assert_int_eq(slot, slot);
	slot = libinput_event_touch_get_seat_slot(tev);
	ck_assert_int_eq(slot, seat_slot);
	libinput_event_destroy(ev);

	ev = libinput_get_event(li);
	litest_assert_event_type(ev, LIBINPUT_EVENT_TOUCH_FRAME);
	libinput_event_destroy(ev);
}

START_TEST(touch_seat_slot)
{
	struct litest_device *dev1 = litest_current_device();
	struct litest_device *dev2;
	struct libinput *li = dev1->libinput;

	dev2 = litest_add_device(li, LITEST_WACOM_TOUCH);

	litest_drain_events(li);

	litest_touch_down(dev1, 0, 50, 50);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_DOWN, 0, 0);

	litest_touch_down(dev2, 0, 50, 50);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_DOWN, 0, 1);

	litest_touch_down(dev2, 1, 60, 50);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_DOWN, 1, 2);

	litest_touch_down(dev1, 1, 60, 50);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_DOWN, 1, 3);

	litest_touch_move_to(dev1, 0, 50, 50, 60, 70, 10);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_MOTION, 0, 0);
	litest_drain_events(li);

	litest_touch_move_to(dev2, 1, 50, 50, 60, 70, 10);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_MOTION, 1, 2);
	litest_drain_events(li);

	litest_touch_up(dev1, 0);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_UP, 0, 0);

	litest_touch_up(dev2, 0);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_UP, 0, 1);

	litest_touch_up(dev2, 1);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_UP, 1, 2);

	litest_touch_up(dev1, 1);
	touch_assert_seat_slot(li, LIBINPUT_EVENT_TOUCH_UP, 1, 3);

	litest_delete_device(dev2);
}
END_TEST

START_TEST(touch_many_slots)
{
	struct libinput *libinput;
	struct litest_device *dev;
	struct libinput_event *ev;
	int slot;
	const int num_tps = 100;
	int slot_count = 0;
	enum libinput_event_type type;

	struct input_absinfo abs[] = {
		{ ABS_MT_SLOT, 0, num_tps - 1, 0, 0, 0 },
		{ .value = -1 },
	};

	dev = litest_create_device_with_overrides(LITEST_WACOM_TOUCH,
						  "litest Multi-touch device",
						  NULL, abs, NULL);
	libinput = dev->libinput;

	for (slot = 0; slot < num_tps; ++slot)
		litest_touch_down(dev, slot, 0, 0);
	for (slot = 0; slot < num_tps; ++slot)
		litest_touch_up(dev, slot);

	libinput_dispatch(libinput);
	while ((ev = libinput_get_event(libinput))) {
		type = libinput_event_get_type(ev);

		if (type == LIBINPUT_EVENT_TOUCH_DOWN)
			slot_count++;
		else if (type == LIBINPUT_EVENT_TOUCH_UP)
			break;

		libinput_event_destroy(ev);
		libinput_dispatch(libinput);
	}

	ck_assert_notnull(ev);
	ck_assert_int_gt(slot_count, 0);

	libinput_dispatch(libinput);
	do {
		type = libinput_event_get_type(ev);
		ck_assert_int_ne(type, LIBINPUT_EVENT_TOUCH_DOWN);
		if (type == LIBINPUT_EVENT_TOUCH_UP)
			slot_count--;

		libinput_event_destroy(ev);
		libinput_dispatch(libinput);
	} while ((ev = libinput_get_event(libinput)));

	ck_assert_int_eq(slot_count, 0);

	litest_delete_device(dev);
}
END_TEST

START_TEST(touch_double_touch_down_up)
{
	struct libinput *libinput;
	struct litest_device *dev;
	struct libinput_event *ev;
	bool got_down = false;
	bool got_up = false;

	dev = litest_current_device();
	libinput = dev->libinput;

	litest_disable_log_handler(libinput);
	/* note: this test is a false negative, libevdev will filter
	 * tracking IDs re-used in the same slot. */
	litest_touch_down(dev, 0, 0, 0);
	litest_touch_down(dev, 0, 0, 0);
	litest_touch_up(dev, 0);
	litest_touch_up(dev, 0);
	libinput_dispatch(libinput);
	litest_restore_log_handler(libinput);

	while ((ev = libinput_get_event(libinput))) {
		switch (libinput_event_get_type(ev)) {
		case LIBINPUT_EVENT_TOUCH_DOWN:
			ck_assert(!got_down);
			got_down = true;
			break;
		case LIBINPUT_EVENT_TOUCH_UP:
			ck_assert(got_down);
			ck_assert(!got_up);
			got_up = true;
			break;
		default:
			break;
		}

		libinput_event_destroy(ev);
		libinput_dispatch(libinput);
	}

	ck_assert(got_down);
	ck_assert(got_up);
}
END_TEST

START_TEST(touch_calibration_scale)
{
	struct libinput *li;
	struct litest_device *dev;
	struct libinput_event *ev;
	struct libinput_event_touch *tev;
	float matrix[6] = {
		1, 0, 0,
		0, 1, 0
	};

	float calibration;
	double x, y;
	const int width = 640, height = 480;

	dev = litest_current_device();
	li = dev->libinput;

	for (calibration = 0.1; calibration < 1; calibration += 0.1) {
		libinput_device_config_calibration_set_matrix(dev->libinput_device,
							      matrix);
		litest_drain_events(li);

		litest_touch_down(dev, 0, 100, 100);
		litest_touch_up(dev, 0);
		libinput_dispatch(li);

		ev = libinput_get_event(li);
		tev = litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_DOWN);

		x = libinput_event_touch_get_x_transformed(tev, width);
		y = libinput_event_touch_get_y_transformed(tev, height);

		ck_assert_int_eq(round(x), round(width * matrix[0]));
		ck_assert_int_eq(round(y), round(height * matrix[4]));

		libinput_event_destroy(ev);
		litest_drain_events(li);

		matrix[0] = calibration;
		matrix[4] = 1 - calibration;
	}
}
END_TEST

START_TEST(touch_calibration_rotation)
{
	struct libinput *li;
	struct litest_device *dev;
	struct libinput_event *ev;
	struct libinput_event_touch *tev;
	float matrix[6];
	int i;
	double x, y;
	int width = 1024, height = 480;

	dev = litest_current_device();
	li = dev->libinput;

	for (i = 0; i < 4; i++) {
		float angle = i * M_PI/2;

		/* [ cos -sin  tx ]
		   [ sin  cos  ty ]
		   [  0    0   1  ] */
		matrix[0] = cos(angle);
		matrix[1] = -sin(angle);
		matrix[3] = sin(angle);
		matrix[4] = cos(angle);

		switch(i) {
		case 0: /* 0 deg */
			matrix[2] = 0;
			matrix[5] = 0;
			break;
		case 1: /* 90 deg cw */
			matrix[2] = 1;
			matrix[5] = 0;
			break;
		case 2: /* 180 deg cw */
			matrix[2] = 1;
			matrix[5] = 1;
			break;
		case 3: /* 270 deg cw */
			matrix[2] = 0;
			matrix[5] = 1;
			break;
		}

		libinput_device_config_calibration_set_matrix(dev->libinput_device,
							      matrix);
		litest_drain_events(li);

		litest_touch_down(dev, 0, 80, 20);
		litest_touch_up(dev, 0);
		libinput_dispatch(li);
		ev = libinput_get_event(li);
		tev = litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_DOWN);

		x = libinput_event_touch_get_x_transformed(tev, width);
		y = libinput_event_touch_get_y_transformed(tev, height);

		/* rounding errors... */
#define almost_equal(a_, b_) \
		{ ck_assert_int_ge((a_) + 0.5, (b_) - 1); \
		  ck_assert_int_le((a_) + 0.5, (b_) + 1); }
		switch(i) {
		case 0: /* 0 deg */
			almost_equal(x, width * 0.8);
			almost_equal(y, height * 0.2);
			break;
		case 1: /* 90 deg cw */
			almost_equal(x, width * 0.8);
			almost_equal(y, height * 0.8);
			break;
		case 2: /* 180 deg cw */
			almost_equal(x, width * 0.2);
			almost_equal(y, height * 0.8);
			break;
		case 3: /* 270 deg cw */
			almost_equal(x, width * 0.2);
			almost_equal(y, height * 0.2);
			break;
		}
#undef almost_equal

		libinput_event_destroy(ev);
		litest_drain_events(li);
	}
}
END_TEST

START_TEST(touch_calibration_translation)
{
	struct libinput *li;
	struct litest_device *dev;
	struct libinput_event *ev;
	struct libinput_event_touch *tev;
	float matrix[6] = {
		1, 0, 0,
		0, 1, 0
	};

	float translate;
	double x, y;
	const int width = 640, height = 480;

	dev = litest_current_device();
	li = dev->libinput;

	/* translating from 0 up to 1 device width/height */
	for (translate = 0.1; translate <= 1; translate += 0.1) {
		libinput_device_config_calibration_set_matrix(dev->libinput_device,
							      matrix);
		litest_drain_events(li);

		litest_touch_down(dev, 0, 100, 100);
		litest_touch_up(dev, 0);

		libinput_dispatch(li);
		ev = libinput_get_event(li);
		tev = litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_DOWN);

		x = libinput_event_touch_get_x_transformed(tev, width);
		y = libinput_event_touch_get_y_transformed(tev, height);

		/* sigh. rounding errors */
		ck_assert_int_ge(round(x), width + round(width * matrix[2]) - 1);
		ck_assert_int_ge(round(y), height + round(height * matrix[5]) - 1);
		ck_assert_int_le(round(x), width + round(width * matrix[2]) + 1);
		ck_assert_int_le(round(y), height + round(height * matrix[5]) + 1);

		libinput_event_destroy(ev);
		litest_drain_events(li);

		matrix[2] = translate;
		matrix[5] = 1 - translate;
	}
}
END_TEST

START_TEST(touch_calibrated_screen_path)
{
	struct litest_device *dev = litest_current_device();
	float matrix[6];
	int rc;

	rc = libinput_device_config_calibration_has_matrix(dev->libinput_device);
	ck_assert_int_eq(rc, 1);

	rc = libinput_device_config_calibration_get_matrix(dev->libinput_device,
							   matrix);
	ck_assert_int_eq(rc, 1);

	ck_assert_double_eq(matrix[0], 1.2);
	ck_assert_double_eq(matrix[1], 3.4);
	ck_assert_double_eq(matrix[2], 5.6);
	ck_assert_double_eq(matrix[3], 7.8);
	ck_assert_double_eq(matrix[4], 9.10);
	ck_assert_double_eq(matrix[5], 11.12);
}
END_TEST

START_TEST(touch_calibration_config)
{
	struct litest_device *dev = litest_current_device();
	float identity[6] = {1, 0, 0, 0, 1, 0};
	float nonidentity[6] = {1, 2, 3, 4, 5, 6};
	float matrix[6];
	enum libinput_config_status status;
	int rc;

	rc = libinput_device_config_calibration_has_matrix(dev->libinput_device);
	ck_assert_int_eq(rc, 1);

	/* Twice so we have every to-fro combination */
	for (int i = 0; i < 2; i++) {
		status = libinput_device_config_calibration_set_matrix(dev->libinput_device, identity);
		ck_assert_int_eq(status, LIBINPUT_CONFIG_STATUS_SUCCESS);
		libinput_device_config_calibration_get_matrix(dev->libinput_device, matrix);
		ck_assert_int_eq(memcmp(matrix, identity, sizeof(matrix)), 0);

		status = libinput_device_config_calibration_set_matrix(dev->libinput_device, nonidentity);
		ck_assert_int_eq(status, LIBINPUT_CONFIG_STATUS_SUCCESS);
		libinput_device_config_calibration_get_matrix(dev->libinput_device, matrix);
		ck_assert_int_eq(memcmp(matrix, nonidentity, sizeof(matrix)), 0);
	}

}
END_TEST

static int open_restricted(const char *path, int flags, void *data)
{
	int fd;
	fd = open(path, flags);
	return fd < 0 ? -errno : fd;
}
static void close_restricted(int fd, void *data)
{
	close(fd);
}

static const struct libinput_interface simple_interface = {
	.open_restricted = open_restricted,
	.close_restricted = close_restricted,
};

START_TEST(touch_calibrated_screen_udev)
{
	struct libinput *li;
	struct libinput_event *ev;
	struct libinput_device *device = NULL;
	struct udev *udev;
	float matrix[6];
	int rc;

	udev = udev_new();
	ck_assert_notnull(udev);

	li = libinput_udev_create_context(&simple_interface, NULL, udev);
	ck_assert_notnull(li);
	ck_assert_int_eq(libinput_udev_assign_seat(li, "seat0"), 0);

	libinput_dispatch(li);

	while ((ev = libinput_get_event(li))) {
		struct libinput_device *d;

		if (libinput_event_get_type(ev) !=
		    LIBINPUT_EVENT_DEVICE_ADDED) {
			libinput_event_destroy(ev);
			continue;
		}

		d = libinput_event_get_device(ev);

		if (libinput_device_get_id_vendor(d) == 0x22 &&
		    libinput_device_get_id_product(d) == 0x33) {
			device = libinput_device_ref(d);
			litest_drain_events(li);
		}
		libinput_event_destroy(ev);
	}

	litest_drain_events(li);

	ck_assert_notnull(device);
	rc = libinput_device_config_calibration_has_matrix(device);
	ck_assert_int_eq(rc, 1);

	rc = libinput_device_config_calibration_get_matrix(device, matrix);
	ck_assert_int_eq(rc, 1);

	ck_assert_double_eq(matrix[0], 1.2);
	ck_assert_double_eq(matrix[1], 3.4);
	ck_assert_double_eq(matrix[2], 5.6);
	ck_assert_double_eq(matrix[3], 7.8);
	ck_assert_double_eq(matrix[4], 9.10);
	ck_assert_double_eq(matrix[5], 11.12);

	libinput_device_unref(device);

	libinput_unref(li);
	udev_unref(udev);

}
END_TEST

START_TEST(touch_no_left_handed)
{
	struct litest_device *dev = litest_current_device();
	struct libinput_device *d = dev->libinput_device;
	enum libinput_config_status status;
	int rc;

	rc = libinput_device_config_left_handed_is_available(d);
	ck_assert_int_eq(rc, 0);

	rc = libinput_device_config_left_handed_get(d);
	ck_assert_int_eq(rc, 0);

	rc = libinput_device_config_left_handed_get_default(d);
	ck_assert_int_eq(rc, 0);

	status = libinput_device_config_left_handed_set(d, 0);
	ck_assert_int_eq(status, LIBINPUT_CONFIG_STATUS_UNSUPPORTED);
}
END_TEST

START_TEST(fake_mt_exists)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *event;
	struct libinput_device *device;

	libinput_dispatch(li);

	event = libinput_get_event(li);
	device = libinput_event_get_device(event);

	ck_assert(!libinput_device_has_capability(device,
						  LIBINPUT_DEVICE_CAP_TOUCH));

	/* This test may need fixing if we add other fake-mt devices that
	 * have different capabilities */
	ck_assert(libinput_device_has_capability(device,
						 LIBINPUT_DEVICE_CAP_POINTER));

	libinput_event_destroy(event);
}
END_TEST

START_TEST(fake_mt_no_touch_events)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;

	litest_drain_events(li);

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 5);
	litest_touch_up(dev, 0);

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_down(dev, 1, 70, 70);
	litest_touch_move_to(dev, 0, 50, 50, 90, 40, 10);
	litest_touch_move_to(dev, 0, 70, 70, 40, 50, 10);
	litest_touch_up(dev, 0);
	litest_touch_up(dev, 1);

	litest_assert_only_typed_events(li,
					LIBINPUT_EVENT_POINTER_MOTION_ABSOLUTE);
}
END_TEST

START_TEST(touch_protocol_a_init)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_device *device = dev->libinput_device;

	ck_assert_int_ne(libinput_next_event_type(li),
			 LIBINPUT_EVENT_NONE);

	ck_assert(libinput_device_has_capability(device,
						 LIBINPUT_DEVICE_CAP_TOUCH));
}
END_TEST

START_TEST(touch_protocol_a_touch)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *ev;
	struct libinput_event_touch *tev;
	double x, y, oldx, oldy;

	litest_drain_events(li);

	litest_touch_down(dev, 0, 5, 95);
	libinput_dispatch(li);

	ev = libinput_get_event(li);
	tev = litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_DOWN);

	oldx = libinput_event_touch_get_x(tev);
	oldy = libinput_event_touch_get_y(tev);

	libinput_event_destroy(ev);

	ev = libinput_get_event(li);
	litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_FRAME);
	libinput_event_destroy(ev);

	litest_touch_move_to(dev, 0, 10, 90, 90, 10, 20);
	libinput_dispatch(li);

	while ((ev = libinput_get_event(li))) {
		if (libinput_event_get_type(ev) ==
		    LIBINPUT_EVENT_TOUCH_FRAME) {
			libinput_event_destroy(ev);
			continue;
		}
		ck_assert_int_eq(libinput_event_get_type(ev),
				 LIBINPUT_EVENT_TOUCH_MOTION);

		tev = libinput_event_get_touch_event(ev);
		x = libinput_event_touch_get_x(tev);
		y = libinput_event_touch_get_y(tev);

		ck_assert_int_gt(x, oldx);
		ck_assert_int_lt(y, oldy);

		oldx = x;
		oldy = y;

		libinput_event_destroy(ev);
	}

	litest_touch_up(dev, 0);
	libinput_dispatch(li);
	ev = libinput_get_event(li);
	litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_UP);
	libinput_event_destroy(ev);
}
END_TEST

START_TEST(touch_protocol_a_2fg_touch)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *ev;
	struct libinput_event_touch *tev;
	int pos;

	litest_drain_events(li);

	litest_touch_down(dev, 0, 5, 95);
	litest_touch_down(dev, 1, 95, 5);

	libinput_dispatch(li);
	litest_assert_touch_down_frame(li);

	ev = libinput_get_event(li);
	litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_DOWN);
	libinput_event_destroy(ev);

	ev = libinput_get_event(li);
	litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_FRAME);
	libinput_event_destroy(ev);

	for (pos = 10; pos < 100; pos += 10) {
		litest_touch_move(dev, 0, pos, 100 - pos);
		litest_touch_move(dev, 1, 100 - pos, pos);
		libinput_dispatch(li);

		ev = libinput_get_event(li);
		tev = libinput_event_get_touch_event(ev);
		ck_assert_int_eq(libinput_event_touch_get_slot(tev), 0);
		libinput_event_destroy(ev);

		ev = libinput_get_event(li);
		litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_FRAME);
		libinput_event_destroy(ev);

		ev = libinput_get_event(li);
		tev = libinput_event_get_touch_event(ev);
		ck_assert_int_eq(libinput_event_touch_get_slot(tev), 1);
		libinput_event_destroy(ev);

		ev = libinput_get_event(li);
		litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_FRAME);
		libinput_event_destroy(ev);
	}

	litest_touch_up(dev, 0);
	libinput_dispatch(li);
	litest_assert_touch_up_frame(li);

	litest_touch_up(dev, 1);
	libinput_dispatch(li);
	litest_assert_touch_up_frame(li);
}
END_TEST

START_TEST(touch_initial_state)
{
	struct litest_device *dev;
	struct libinput *libinput1, *libinput2;
	struct libinput_event *ev1 = NULL;
	struct libinput_event *ev2 = NULL;
	struct libinput_event_touch *t1, *t2;
	struct libinput_device *device1, *device2;
	int axis = _i; /* looped test */

	dev = litest_current_device();
	device1 = dev->libinput_device;
	libinput_device_config_tap_set_enabled(device1,
					       LIBINPUT_CONFIG_TAP_DISABLED);

	libinput1 = dev->libinput;
	litest_touch_down(dev, 0, 40, 60);
	litest_touch_up(dev, 0);

	/* device is now on some x/y value */
	litest_drain_events(libinput1);

	libinput2 = litest_create_context();
	device2 = libinput_path_add_device(libinput2,
					   libevdev_uinput_get_devnode(
							       dev->uinput));
	libinput_device_config_tap_set_enabled(device2,
					       LIBINPUT_CONFIG_TAP_DISABLED);
	litest_drain_events(libinput2);

	if (axis == ABS_X)
		litest_touch_down(dev, 0, 40, 70);
	else
		litest_touch_down(dev, 0, 70, 60);
	litest_touch_up(dev, 0);

	libinput_dispatch(libinput1);
	libinput_dispatch(libinput2);

	while (libinput_next_event_type(libinput1)) {
		ev1 = libinput_get_event(libinput1);
		ev2 = libinput_get_event(libinput2);

		t1 = litest_is_touch_event(ev1, 0);
		t2 = litest_is_touch_event(ev2, 0);

		ck_assert_int_eq(libinput_event_get_type(ev1),
				 libinput_event_get_type(ev2));

		if (libinput_event_get_type(ev1) == LIBINPUT_EVENT_TOUCH_UP ||
		    libinput_event_get_type(ev1) == LIBINPUT_EVENT_TOUCH_FRAME)
			break;

		ck_assert_int_eq(libinput_event_touch_get_x(t1),
				 libinput_event_touch_get_x(t2));
		ck_assert_int_eq(libinput_event_touch_get_y(t1),
				 libinput_event_touch_get_y(t2));

		libinput_event_destroy(ev1);
		libinput_event_destroy(ev2);
	}

	libinput_event_destroy(ev1);
	libinput_event_destroy(ev2);

	libinput_unref(libinput2);
}
END_TEST

START_TEST(touch_time_usec)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *event;
	struct libinput_event_touch *tev;
	uint64_t time_usec;

	litest_drain_events(dev->libinput);

	litest_touch_down(dev, 0, 10, 10);
	libinput_dispatch(li);

	event = libinput_get_event(li);
	tev = litest_is_touch_event(event, LIBINPUT_EVENT_TOUCH_DOWN);
	time_usec = libinput_event_touch_get_time_usec(tev);
	ck_assert_int_eq(libinput_event_touch_get_time(tev),
			 (uint32_t) (time_usec / 1000));
	libinput_event_destroy(event);
}
END_TEST

START_TEST(touch_fuzz)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *event;
	int i;
	int x = 700, y = 300;

	litest_drain_events(dev->libinput);

	litest_event(dev, EV_ABS, ABS_MT_TRACKING_ID, 30);
	litest_event(dev, EV_ABS, ABS_MT_SLOT, 0);
	litest_event(dev, EV_ABS, ABS_MT_POSITION_X, x);
	litest_event(dev, EV_ABS, ABS_MT_POSITION_Y, y);
	litest_event(dev, EV_KEY, BTN_TOUCH, 1);
	litest_event(dev, EV_SYN, SYN_REPORT, 0);
	libinput_dispatch(li);

	event = libinput_get_event(li);
	litest_is_touch_event(event, LIBINPUT_EVENT_TOUCH_DOWN);
	libinput_event_destroy(event);
	event = libinput_get_event(li);
	litest_is_touch_event(event, LIBINPUT_EVENT_TOUCH_FRAME);
	libinput_event_destroy(event);

	litest_drain_events(li);

	for (i = 0; i < 50; i++) {
		if (i % 2) {
			x++;
			y--;
		} else {
			x--;
			y++;
		}
		litest_event(dev, EV_ABS, ABS_MT_POSITION_X, x);
		litest_event(dev, EV_ABS, ABS_MT_POSITION_Y, y);
		litest_event(dev, EV_SYN, SYN_REPORT, 0);
		libinput_dispatch(li);
		litest_assert_empty_queue(li);
	}
}
END_TEST

START_TEST(touch_fuzz_property)
{
	struct litest_device *dev = litest_current_device();
	struct udev_device *d;
	const char *prop;
	int fuzz;

	ck_assert_int_eq(libevdev_get_abs_fuzz(dev->evdev, ABS_X), 0);
	ck_assert_int_eq(libevdev_get_abs_fuzz(dev->evdev, ABS_Y), 0);

	d = libinput_device_get_udev_device(dev->libinput_device);
	prop = udev_device_get_property_value(d, "LIBINPUT_FUZZ_00");
	ck_assert_notnull(prop);
	ck_assert(safe_atoi(prop, &fuzz));
	ck_assert_int_eq(fuzz, 10); /* device-specific */

	prop = udev_device_get_property_value(d, "LIBINPUT_FUZZ_01");
	ck_assert_notnull(prop);
	ck_assert(safe_atoi(prop, &fuzz));
	ck_assert_int_eq(fuzz, 12); /* device-specific */

	udev_device_unref(d);
}
END_TEST

START_TEST(touch_release_on_unplug)
{
	struct litest_device *dev;
	struct libinput *li;
	struct libinput_event *ev;

	li = litest_create_context();
	dev = litest_add_device(li, LITEST_GENERIC_MULTITOUCH_SCREEN);
	litest_drain_events(li);

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 10);
	litest_drain_events(li);

	/* Touch is still down when device is removed, espect a release */
	litest_delete_device(dev);
	libinput_dispatch(li);

	ev = libinput_get_event(li);
	litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_CANCEL);
	libinput_event_destroy(ev);

	ev = libinput_get_event(li);
	litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_FRAME);
	libinput_event_destroy(ev);

	ev = libinput_get_event(li);
	litest_assert_event_type(ev, LIBINPUT_EVENT_DEVICE_REMOVED);
	libinput_event_destroy(ev);

	libinput_unref(li);
}
END_TEST

START_TEST(touch_invalid_range_over)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *ev;
	struct libinput_event_touch *t;
	double x, y;

	litest_drain_events(li);

	/* Touch outside the valid area */
	litest_event(dev, EV_ABS, ABS_MT_SLOT, 0);
	litest_event(dev, EV_ABS, ABS_MT_TRACKING_ID, 1);
	litest_event(dev, EV_ABS, ABS_X, 4000);
	litest_event(dev, EV_ABS, ABS_Y, 5000);
	litest_event(dev, EV_ABS, ABS_MT_POSITION_X, 4000);
	litest_event(dev, EV_ABS, ABS_MT_POSITION_Y, 5000);
	litest_event(dev, EV_KEY, BTN_TOUCH, 1);
	litest_event(dev, EV_SYN, SYN_REPORT, 0);
	libinput_dispatch(li);

	/* Expect the mm to be correct regardless */
	ev = libinput_get_event(li);
	t = litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_DOWN);
	x = libinput_event_touch_get_x(t);
	y = libinput_event_touch_get_y(t);
	ck_assert_double_eq(x, 300); /* device has resolution 10 */
	ck_assert_double_eq(y, 300); /* device has resolution 10 */

	/* Expect the percentage to be correct too, even if > 100% */
	x = libinput_event_touch_get_x_transformed(t, 100);
	y = libinput_event_touch_get_y_transformed(t, 100);
	ck_assert_double_eq(round(x), 200);
	ck_assert_double_eq(round(y), 120);

	libinput_event_destroy(ev);
}
END_TEST

START_TEST(touch_invalid_range_under)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct libinput_event *ev;
	struct libinput_event_touch *t;
	double x, y;

	litest_drain_events(li);

	/* Touch outside the valid area */
	litest_event(dev, EV_ABS, ABS_MT_SLOT, 0);
	litest_event(dev, EV_ABS, ABS_MT_TRACKING_ID, 1);
	litest_event(dev, EV_ABS, ABS_X, -500);
	litest_event(dev, EV_ABS, ABS_Y, 1000);
	litest_event(dev, EV_ABS, ABS_MT_POSITION_X, -500);
	litest_event(dev, EV_ABS, ABS_MT_POSITION_Y, 1000);
	litest_event(dev, EV_KEY, BTN_TOUCH, 1);
	litest_event(dev, EV_SYN, SYN_REPORT, 0);
	libinput_dispatch(li);

	/* Expect the mm to be correct regardless */
	ev = libinput_get_event(li);
	t = litest_is_touch_event(ev, LIBINPUT_EVENT_TOUCH_DOWN);
	x = libinput_event_touch_get_x(t);
	y = libinput_event_touch_get_y(t);
	ck_assert_double_eq(x, -150); /* device has resolution 10 */
	ck_assert_double_eq(y, -100); /* device has resolution 10 */

	/* Expect the percentage to be correct too, even if > 100% */
	x = libinput_event_touch_get_x_transformed(t, 100);
	y = libinput_event_touch_get_y_transformed(t, 100);
	ck_assert_double_eq(round(x), -100);
	ck_assert_double_eq(round(y), -40);

	libinput_event_destroy(ev);
}
END_TEST

START_TEST(touch_count_st)
{
	struct litest_device *dev = litest_current_device();
	struct libinput_device *device = dev->libinput_device;

	ck_assert_int_eq(libinput_device_touch_get_touch_count(device), 1);
}
END_TEST

START_TEST(touch_count_mt)
{
	struct litest_device *dev = litest_current_device();
	struct libinput_device *device = dev->libinput_device;
	struct libevdev *evdev = dev->evdev;

	ck_assert_int_eq(libinput_device_touch_get_touch_count(device),
			 libevdev_get_num_slots(evdev));
}
END_TEST

START_TEST(touch_count_unknown)
{
	struct litest_device *dev = litest_current_device();
	struct libinput_device *device = dev->libinput_device;

	ck_assert_int_eq(libinput_device_touch_get_touch_count(device), 0);
}
END_TEST

START_TEST(touch_count_invalid)
{
	struct litest_device *dev = litest_current_device();
	struct libinput_device *device = dev->libinput_device;

	ck_assert_int_eq(libinput_device_touch_get_touch_count(device), -1);
}
END_TEST

static inline bool
touch_has_tool_palm(struct litest_device *dev)
{
	return libevdev_has_event_code(dev->evdev, EV_ABS, ABS_MT_TOOL_TYPE);
}

START_TEST(touch_palm_detect_tool_palm)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct axis_replacement axes[] = {
		{ ABS_MT_TOOL_TYPE, MT_TOOL_PALM },
		{ -1, 0 }
	};

	if (!touch_has_tool_palm(dev))
		return;

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 10);
	litest_drain_events(li);

	litest_touch_move_to_extended(dev, 0, 50, 50, 70, 70, axes, 10);
	libinput_dispatch(li);
	litest_assert_touch_cancel(li);

	litest_touch_move_to(dev, 0, 70, 70, 50, 40, 10);
	litest_touch_up(dev, 0);

	litest_assert_empty_queue(li);
}
END_TEST

START_TEST(touch_palm_detect_tool_palm_on_off)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct axis_replacement axes[] = {
		{ ABS_MT_TOOL_TYPE, MT_TOOL_PALM },
		{ -1, 0 }
	};

	if (!touch_has_tool_palm(dev))
		return;

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 10);
	litest_drain_events(li);

	litest_touch_move_to_extended(dev, 0, 50, 50, 70, 70, axes, 10);
	libinput_dispatch(li);
	litest_assert_touch_cancel(li);

	litest_touch_move_to(dev, 0, 70, 70, 50, 40, 10);
	litest_assert_empty_queue(li);

	litest_axis_set_value(axes, ABS_MT_TOOL_TYPE, MT_TOOL_FINGER);
	litest_touch_move_to_extended(dev, 0, 50, 40, 70, 70, axes, 10);
	litest_touch_up(dev, 0);

	litest_assert_empty_queue(li);
}
END_TEST

START_TEST(touch_palm_detect_tool_palm_keep_type)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct axis_replacement axes[] = {
		{ ABS_MT_TOOL_TYPE, MT_TOOL_PALM },
		{ -1, 0 }
	};

	if (!touch_has_tool_palm(dev))
		return;

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 10);
	litest_touch_move_to_extended(dev, 0, 50, 50, 70, 70, axes, 10);
	litest_touch_up(dev, 0);
	litest_drain_events(li);

	/* ABS_MT_TOOL_TYPE never reset to finger, so a new touch
	   should be ignored outright */
	litest_touch_down_extended(dev, 0, 50, 50, axes);

	/* Test the revert to finger case too */
	litest_axis_set_value(axes, ABS_MT_TOOL_TYPE, MT_TOOL_FINGER);
	litest_touch_move_to(dev, 0, 70, 70, 50, 40, 10);
	litest_touch_up(dev, 0);

	litest_assert_empty_queue(li);
}
END_TEST

START_TEST(touch_palm_detect_tool_palm_2fg)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct axis_replacement axes[] = {
		{ ABS_MT_TOOL_TYPE, MT_TOOL_PALM },
		{ -1, 0 }
	};

	if (!touch_has_tool_palm(dev))
		return;

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_down(dev, 1, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 10);
	litest_drain_events(li);

	litest_touch_move_to_extended(dev, 0, 50, 50, 70, 70, axes, 10);
	libinput_dispatch(li);
	litest_assert_touch_cancel(li);

	litest_touch_move_to(dev, 1, 50, 50, 70, 70, 10);
	libinput_dispatch(li);
	litest_assert_touch_motion_frame(li);

	litest_touch_up(dev, 1);
	libinput_dispatch(li);
	litest_assert_touch_up_frame(li);

	litest_touch_move_to(dev, 0, 70, 70, 50, 40, 10);
	litest_touch_up(dev, 0);

	litest_assert_empty_queue(li);
}
END_TEST

START_TEST(touch_palm_detect_tool_palm_on_off_2fg)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct axis_replacement axes[] = {
		{ ABS_MT_TOOL_TYPE, MT_TOOL_PALM },
		{ -1, 0 }
	};

	if (!touch_has_tool_palm(dev))
		return;

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_down(dev, 1, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 10);
	litest_drain_events(li);

	litest_touch_move_to_extended(dev, 0, 50, 50, 70, 70, axes, 10);
	libinput_dispatch(li);
	litest_assert_touch_cancel(li);

	litest_touch_move_to(dev, 1, 50, 50, 70, 70, 10);
	libinput_dispatch(li);
	litest_assert_touch_motion_frame(li);

	litest_axis_set_value(axes, ABS_MT_TOOL_TYPE, MT_TOOL_FINGER);
	litest_touch_move_to_extended(dev, 0, 50, 40, 70, 70, axes, 10);
	litest_assert_empty_queue(li);

	litest_touch_move_to(dev, 1, 70, 70, 50, 40, 10);
	libinput_dispatch(li);
	litest_assert_touch_motion_frame(li);

	litest_touch_up(dev, 1);
	libinput_dispatch(li);
	litest_assert_touch_up_frame(li);

	litest_touch_move_to(dev, 0, 70, 70, 50, 40, 10);
	litest_touch_up(dev, 0);

	litest_assert_empty_queue(li);
}
END_TEST

START_TEST(touch_palm_detect_tool_palm_keep_type_2fg)
{
	struct litest_device *dev = litest_current_device();
	struct libinput *li = dev->libinput;
	struct axis_replacement axes[] = {
		{ ABS_MT_TOOL_TYPE, MT_TOOL_PALM },
		{ -1, 0 }
	};

	if (!touch_has_tool_palm(dev))
		return;

	litest_touch_down(dev, 0, 50, 50);
	litest_touch_down(dev, 1, 50, 50);
	litest_touch_move_to(dev, 0, 50, 50, 70, 70, 10);
	litest_touch_move_to_extended(dev, 0, 50, 50, 70, 70, axes, 10);
	litest_touch_up(dev, 0);
	litest_drain_events(li);

	litest_touch_move_to(dev, 1, 50, 50, 70, 70, 10);
	libinput_dispatch(li);
	litest_assert_touch_motion_frame(li);

	/* ABS_MT_TOOL_TYPE never reset to finger, so a new touch
	   should be ignored outright */
	litest_touch_down_extended(dev, 0, 50, 50, axes);

	/* Test the revert to finger case too */
	litest_axis_set_value(axes, ABS_MT_TOOL_TYPE, MT_TOOL_FINGER);
	litest_touch_move_to(dev, 0, 70, 70, 50, 40, 10);
	litest_touch_up(dev, 0);

	litest_assert_empty_queue(li);

	litest_touch_up(dev, 1);
	libinput_dispatch(li);
	litest_assert_touch_up_frame(li);
}
END_TEST

TEST_COLLECTION(touch)
{
	struct range axes = { ABS_X, ABS_Y + 1};

	litest_add("touch:frame", touch_frame_events, LITEST_TOUCH, LITEST_ANY);
	litest_add("touch:down", touch_downup_no_motion, LITEST_TOUCH, LITEST_ANY);
	litest_add("touch:down", touch_downup_no_motion, LITEST_SINGLE_TOUCH, LITEST_TOUCHPAD);
	litest_add_no_device("touch:abs-transform", touch_abs_transform);
	litest_add("touch:slots", touch_seat_slot, LITEST_TOUCH, LITEST_TOUCHPAD);
	litest_add_no_device("touch:slots", touch_many_slots);
	litest_add("touch:double-touch-down-up", touch_double_touch_down_up, LITEST_TOUCH, LITEST_PROTOCOL_A);
	litest_add("touch:calibration", touch_calibration_scale, LITEST_TOUCH, LITEST_TOUCHPAD);
	litest_add("touch:calibration", touch_calibration_scale, LITEST_SINGLE_TOUCH, LITEST_TOUCHPAD);
	litest_add("touch:calibration", touch_calibration_rotation, LITEST_TOUCH, LITEST_TOUCHPAD);
	litest_add("touch:calibration", touch_calibration_rotation, LITEST_SINGLE_TOUCH, LITEST_TOUCHPAD);
	litest_add("touch:calibration", touch_calibration_translation, LITEST_TOUCH, LITEST_TOUCHPAD);
	litest_add("touch:calibration", touch_calibration_translation, LITEST_SINGLE_TOUCH, LITEST_TOUCHPAD);
	litest_add_for_device("touch:calibration", touch_calibrated_screen_path, LITEST_CALIBRATED_TOUCHSCREEN);
	litest_add_for_device("touch:calibration", touch_calibrated_screen_udev, LITEST_CALIBRATED_TOUCHSCREEN);
	litest_add("touch:calibration", touch_calibration_config, LITEST_TOUCH, LITEST_ANY);

	litest_add("touch:left-handed", touch_no_left_handed, LITEST_TOUCH, LITEST_ANY);

	litest_add("touch:fake-mt", fake_mt_exists, LITEST_FAKE_MT, LITEST_ANY);
	litest_add("touch:fake-mt", fake_mt_no_touch_events, LITEST_FAKE_MT, LITEST_ANY);

	litest_add("touch:protocol a", touch_protocol_a_init, LITEST_PROTOCOL_A, LITEST_ANY);
	litest_add("touch:protocol a", touch_protocol_a_touch, LITEST_PROTOCOL_A, LITEST_ANY);
	litest_add("touch:protocol a", touch_protocol_a_2fg_touch, LITEST_PROTOCOL_A, LITEST_ANY);

	litest_add_ranged("touch:state", touch_initial_state, LITEST_TOUCH, LITEST_PROTOCOL_A, &axes);

	litest_add("touch:time", touch_time_usec, LITEST_TOUCH, LITEST_TOUCHPAD);

	litest_add_for_device("touch:fuzz", touch_fuzz, LITEST_MULTITOUCH_FUZZ_SCREEN);
	litest_add_for_device("touch:fuzz", touch_fuzz_property, LITEST_MULTITOUCH_FUZZ_SCREEN);

	litest_add_no_device("touch:release", touch_release_on_unplug);

	litest_add_for_device("touch:range", touch_invalid_range_over, LITEST_TOUCHSCREEN_INVALID_RANGE);
	litest_add_for_device("touch:range", touch_invalid_range_under, LITEST_TOUCHSCREEN_INVALID_RANGE);

	litest_add("touch:count", touch_count_st, LITEST_SINGLE_TOUCH, LITEST_TOUCHPAD);
	litest_add("touch:count", touch_count_mt, LITEST_TOUCH, LITEST_SINGLE_TOUCH|LITEST_PROTOCOL_A);
	litest_add("touch:count", touch_count_unknown, LITEST_PROTOCOL_A, LITEST_ANY);
	litest_add("touch:count", touch_count_invalid, LITEST_ANY, LITEST_TOUCH|LITEST_SINGLE_TOUCH|LITEST_PROTOCOL_A);

	litest_add("touch:tool", touch_palm_detect_tool_palm, LITEST_TOUCH, LITEST_ANY);
	litest_add("touch:tool", touch_palm_detect_tool_palm_on_off, LITEST_TOUCH, LITEST_ANY);
	litest_add("touch:tool", touch_palm_detect_tool_palm_keep_type, LITEST_TOUCH, LITEST_ANY);
	litest_add("touch:tool", touch_palm_detect_tool_palm_2fg, LITEST_TOUCH, LITEST_SINGLE_TOUCH);
	litest_add("touch:tool", touch_palm_detect_tool_palm_on_off_2fg, LITEST_TOUCH, LITEST_SINGLE_TOUCH);
	litest_add("touch:tool", touch_palm_detect_tool_palm_keep_type_2fg, LITEST_TOUCH, LITEST_ANY);
}