mirror of
https://gitlab.freedesktop.org/libevdev/libevdev.git
synced 2025-12-20 16:10:07 +01:00
If the start and end of a touch are dropped, the slot, according to the kernel, may have a different state. We should inform the client of these changes even if the slot is not currently active. For most axes this doesn't matter too much as we expect them to change during an active touch anyway so we don't expect the kernel's caching to be a problem. However where the ABS_MT_TOOL_TYPE changed during a sync we need to inform the client of the new tool type so that future touchese won't be erroneously treated as e.g. palms. For a full reproducer see the test case but it comes down to: - touch down with MT_TOOL_PALM, make sure libevdev reads the state - change that slot to MT_TOOL_FINGER, trigger a sync - ensure that libevdev pushes out that tool type change even if the slot is not currently active Co-authored-by: Peter Hutterer <peter.hutterer@who-t.net> Part-of: <https://gitlab.freedesktop.org/libevdev/libevdev/-/merge_requests/124>
2285 lines
71 KiB
C
2285 lines
71 KiB
C
// SPDX-License-Identifier: MIT
|
|
/*
|
|
* Copyright © 2013 Red Hat, Inc.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include <linux/input.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <libevdev/libevdev-util.h>
|
|
|
|
#include "test-common.h"
|
|
|
|
START_TEST(test_next_event)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_KEY, BTN_LEFT, 1);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_next_event_invalid_fd)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
libevdev_set_log_function(test_logfunc_ignore_error, NULL);
|
|
|
|
dev = libevdev_new();
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EBADF);
|
|
libevdev_free(dev);
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
|
|
/* invalid (missing) flag */
|
|
rc = libevdev_next_event(dev, 0x10, &ev);
|
|
ck_assert_int_eq(rc, -EINVAL);
|
|
|
|
/* set an invalid fd */
|
|
rc = libevdev_change_fd(dev, -3);
|
|
ck_assert_int_eq(rc, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EBADF);
|
|
|
|
libevdev_set_log_function(test_logfunc_abort_on_error, NULL);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_next_event_blocking)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int fd, flags;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
|
|
fd = libevdev_get_fd(dev);
|
|
flags = fcntl(fd, F_GETFL) & ~O_NONBLOCK;
|
|
rc = fcntl(fd, F_SETFL, flags);
|
|
ck_assert_int_eq(rc, 0);
|
|
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_BLOCKING, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_KEY, BTN_LEFT, 1);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_dropped_event)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
int pipefd[2];
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
|
|
/* This is a bit complicated:
|
|
we can't get SYN_DROPPED through uinput, so we push two events down
|
|
uinput, and process those. Then write a SYN_DROPPED on a pipe,
|
|
switch the fd and read one event off the wire. Switch back, so
|
|
that when we do read off the SYN_DROPPED we have the fd back on
|
|
the device and the ioctls work.
|
|
*/
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_KEY, BTN_LEFT, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
rc = pipe2(pipefd, O_NONBLOCK);
|
|
ck_assert_int_eq(rc, 0);
|
|
|
|
libevdev_change_fd(dev, pipefd[0]);
|
|
ev.type = EV_SYN;
|
|
ev.code = SYN_DROPPED;
|
|
ev.value = 0;
|
|
rc = write(pipefd[1], &ev, sizeof(ev));
|
|
ck_assert_int_eq(rc, sizeof(ev));
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SYN, SYN_DROPPED, 0);
|
|
|
|
libevdev_change_fd(dev, uinput_device_get_fd(uidev));
|
|
/* only check for the rc, nothing actually changed on the device */
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
|
|
close(pipefd[0]);
|
|
close(pipefd[1]);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_type_filtered)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
|
|
libevdev_disable_event_type(dev, EV_REL);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
uinput_device_event(uidev, EV_REL, REL_X, 1);
|
|
uinput_device_event(uidev, EV_KEY, REL_Y, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_code_filtered)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
|
|
libevdev_disable_event_code(dev, EV_REL, REL_X);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
uinput_device_event(uidev, EV_REL, REL_X, 1);
|
|
uinput_device_event(uidev, EV_REL, REL_Y, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_REL, REL_Y, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_has_event_pending)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
|
|
ck_assert_int_eq(libevdev_has_event_pending(dev), 0);
|
|
|
|
uinput_device_event(uidev, EV_REL, REL_X, 1);
|
|
uinput_device_event(uidev, EV_REL, REL_Y, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
ck_assert_int_eq(libevdev_has_event_pending(dev), 1);
|
|
|
|
libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
|
|
ck_assert_int_eq(libevdev_has_event_pending(dev), 1);
|
|
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev)) != -EAGAIN)
|
|
;
|
|
|
|
ck_assert_int_eq(libevdev_has_event_pending(dev), 0);
|
|
|
|
libevdev_change_fd(dev, -1);
|
|
ck_assert_int_eq(libevdev_has_event_pending(dev), -EBADF);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_has_event_pending_invalid_fd)
|
|
{
|
|
struct libevdev *dev;
|
|
int rc;
|
|
|
|
libevdev_set_log_function(test_logfunc_ignore_error, NULL);
|
|
|
|
dev = libevdev_new();
|
|
rc = libevdev_has_event_pending(dev);
|
|
ck_assert_int_eq(rc, -EBADF);
|
|
|
|
libevdev_set_log_function(test_logfunc_abort_on_error, NULL);
|
|
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_button)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
EV_KEY, KEY_MAX,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_KEY, BTN_RIGHT, 1);
|
|
uinput_device_event(uidev, EV_KEY, KEY_MAX, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_KEY, BTN_LEFT, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_KEY, BTN_RIGHT, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_KEY, KEY_MAX, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
ck_assert(libevdev_get_event_value(dev, EV_KEY, BTN_LEFT));
|
|
ck_assert(libevdev_get_event_value(dev, EV_KEY, BTN_RIGHT));
|
|
ck_assert(!libevdev_get_event_value(dev, EV_KEY, BTN_MIDDLE));
|
|
ck_assert(libevdev_get_event_value(dev, EV_KEY, KEY_MAX));
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_abs)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[3] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MAX, .maximum = 1000 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_ABS, ABS_X, 100);
|
|
uinput_device_event(uidev, EV_ABS, ABS_Y, 500);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MAX, 700);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_X, 100);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_Y, 500);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MAX);
|
|
ck_assert_int_eq(ev.value, 700);
|
|
assert_event(&ev, EV_ABS, ABS_MAX, 700);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_mt)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[6] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 1 },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 2 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_X, 100,
|
|
EV_ABS, ABS_Y, 500,
|
|
EV_ABS, ABS_MT_POSITION_X, 100,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 1,
|
|
EV_ABS, ABS_MT_SLOT, 1,
|
|
EV_ABS, ABS_X, 1,
|
|
EV_ABS, ABS_Y, 5,
|
|
EV_ABS, ABS_MT_POSITION_X, 1,
|
|
EV_ABS, ABS_MT_POSITION_Y, 5,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 2,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_X, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_Y, 5);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_SLOT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_POSITION_X, 100);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_POSITION_Y, 500);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_TRACKING_ID, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_SLOT, 1);
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_POSITION_X, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_POSITION_Y, 5);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_ABS, ABS_MT_TRACKING_ID, 2);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_mt_reset_slot)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev,
|
|
last_slot_event = { .type = 0};
|
|
struct input_absinfo abs[6] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 1 },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 2 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 1,
|
|
EV_ABS, ABS_MT_POSITION_X, 100,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 1,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_MT_POSITION_X, 1,
|
|
EV_ABS, ABS_MT_POSITION_Y, 5,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 2,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_MT_SLOT))
|
|
last_slot_event = ev;
|
|
} while (rc != -EAGAIN);
|
|
|
|
ck_assert(libevdev_event_is_code(&last_slot_event, EV_ABS, ABS_MT_SLOT));
|
|
ck_assert_int_eq(last_slot_event.value, 0);
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 0);
|
|
|
|
last_slot_event.type = 0;
|
|
|
|
/* same thing again, this time swap the numbers */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_MT_POSITION_X, 100,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 1,
|
|
EV_ABS, ABS_MT_SLOT, 1,
|
|
EV_ABS, ABS_MT_POSITION_X, 1,
|
|
EV_ABS, ABS_MT_POSITION_Y, 5,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 2,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_MT_SLOT))
|
|
last_slot_event = ev;
|
|
} while (rc != -EAGAIN);
|
|
|
|
ck_assert(libevdev_event_is_code(&last_slot_event, EV_ABS, ABS_MT_SLOT));
|
|
ck_assert_int_eq(last_slot_event.value, 1);
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 1);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_led)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_LED, LED_NUML,
|
|
EV_LED, LED_CAPSL,
|
|
EV_LED, LED_MAX,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_LED, LED_NUML, 1);
|
|
uinput_device_event(uidev, EV_LED, LED_CAPSL, 1);
|
|
uinput_device_event(uidev, EV_LED, LED_MAX, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_LED, LED_NUML, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_LED, LED_CAPSL, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_LED, LED_MAX, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_LED, LED_NUML), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_LED, LED_CAPSL), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_LED, LED_MAX), 1);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_sw)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_SW, SW_HEADPHONE_INSERT,
|
|
EV_SW, SW_MICROPHONE_INSERT,
|
|
EV_SW, SW_MAX,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_SW, SW_HEADPHONE_INSERT, 1);
|
|
uinput_device_event(uidev, EV_SW, SW_MICROPHONE_INSERT, 1);
|
|
uinput_device_event(uidev, EV_SW, SW_MAX, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SW, SW_HEADPHONE_INSERT, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SW, SW_MICROPHONE_INSERT, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SW, SW_MAX, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_SW, SW_HEADPHONE_INSERT), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_SW, SW_MICROPHONE_INSERT), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_SW, SW_MAX), 1);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_tracking_ids)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
int i;
|
|
const int num_slots = 15;
|
|
int slot = -1;
|
|
unsigned long terminated[NLONGS(num_slots)];
|
|
unsigned long restarted[NLONGS(num_slots)];
|
|
struct input_absinfo abs[6] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = num_slots },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 0xff },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
/* Test the sync process to make sure we get touches terminated when
|
|
* the tracking id changes:
|
|
* 1) start a bunch of touch points
|
|
* 2) read data into libevdev, make sure state is up-to-date
|
|
* 3) change touchpoints
|
|
* 3.1) change the tracking ID on some (indicating terminated and
|
|
* re-started touchpoint)
|
|
* 3.2) change the tracking ID to -1 on some (indicating termianted
|
|
* touchpoint)
|
|
* 3.3) just update the data on others
|
|
* 4) force a sync on the device
|
|
* 5) make sure we get the right tracking ID changes in the caller
|
|
*/
|
|
|
|
/* Start a bunch of touch points */
|
|
for (i = num_slots; i >= 0; i--) {
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, i,
|
|
EV_ABS, ABS_MT_TRACKING_ID, i,
|
|
EV_ABS, ABS_X, 100 + i,
|
|
EV_ABS, ABS_Y, 500 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 100 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500 + i,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_ne(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
} while (rc >= 0);
|
|
}
|
|
|
|
/* we have a bunch of touches now, and libevdev knows it. Change all
|
|
* touches */
|
|
for (i = num_slots; i >= 0; i--) {
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, i);
|
|
if (i % 3 == 0) {
|
|
/* change some slots with a new tracking id */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_TRACKING_ID, num_slots + i,
|
|
EV_ABS, ABS_X, 200 + i,
|
|
EV_ABS, ABS_Y, 700 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 200 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 700 + i,
|
|
-1, -1);
|
|
} else if (i % 3 == 1) {
|
|
/* stop others */
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, -1);
|
|
} else {
|
|
/* just update */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_X, 200 + i,
|
|
EV_ABS, ABS_Y, 700 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 200 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 700 + i,
|
|
-1, -1);
|
|
}
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
}
|
|
|
|
/* Force sync */
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
/* now check for the right tracking IDs */
|
|
memset(terminated, 0, sizeof(terminated));
|
|
memset(restarted, 0, sizeof(restarted));
|
|
slot = -1;
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev)) != -EAGAIN) {
|
|
if (libevdev_event_is_code(&ev, EV_SYN, SYN_REPORT))
|
|
continue;
|
|
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_MT_SLOT)) {
|
|
slot = ev.value;
|
|
continue;
|
|
}
|
|
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_X) ||
|
|
libevdev_event_is_code(&ev, EV_ABS, ABS_Y))
|
|
continue;
|
|
|
|
ck_assert_int_ne(slot, -1);
|
|
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_MT_TRACKING_ID)) {
|
|
if (slot % 3 == 0) {
|
|
if (!bit_is_set(terminated, slot)) {
|
|
ck_assert_int_eq(ev.value, -1);
|
|
set_bit(terminated, slot);
|
|
} else {
|
|
ck_assert_int_eq(ev.value, num_slots + slot);
|
|
set_bit(restarted, slot);
|
|
}
|
|
} else if (slot % 3 == 1) {
|
|
ck_assert(!bit_is_set(terminated, slot));
|
|
ck_assert_int_eq(ev.value, -1);
|
|
set_bit(terminated, slot);
|
|
} else
|
|
ck_abort();
|
|
|
|
continue;
|
|
}
|
|
|
|
switch(ev.code) {
|
|
case ABS_MT_POSITION_X:
|
|
ck_assert_int_eq(ev.value, 200 + slot);
|
|
break;
|
|
case ABS_MT_POSITION_Y:
|
|
ck_assert_int_eq(ev.value, 700 + slot);
|
|
break;
|
|
default:
|
|
ck_abort();
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < num_slots; i++) {
|
|
if (i % 3 == 0) {
|
|
ck_assert(bit_is_set(terminated, i));
|
|
ck_assert(bit_is_set(restarted, i));
|
|
} else if (i % 3 == 1) {
|
|
ck_assert(bit_is_set(terminated, i));
|
|
ck_assert(!bit_is_set(restarted, i));
|
|
} else {
|
|
ck_assert(!bit_is_set(terminated, i));
|
|
ck_assert(!bit_is_set(restarted, i));
|
|
}
|
|
}
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_tracking_ids_btntool)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
const int num_slots = 5;
|
|
struct input_absinfo abs[6] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = num_slots },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 0xffff },
|
|
};
|
|
bool have_doubletap = false,
|
|
have_quadtap = false,
|
|
have_quinttap = false;
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_KEY, BTN_TOOL_FINGER,
|
|
EV_KEY, BTN_TOOL_DOUBLETAP,
|
|
EV_KEY, BTN_TOOL_TRIPLETAP,
|
|
EV_KEY, BTN_TOOL_QUADTAP,
|
|
EV_KEY, BTN_TOOL_QUINTTAP,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
/* Test the sync process to make sure we get the BTN_TOOL bits for
|
|
* touches adjusted correctly when the tracking id changes:
|
|
* 1) start a bunch of touch points
|
|
* 2) read data into libevdev, make sure state is up-to-date
|
|
* 3) change touchpoints
|
|
* 3.1) change the tracking ID on some (indicating terminated and
|
|
* re-started touchpoint)
|
|
* 3.2) change the tracking ID to -1 on some (indicating termianted
|
|
* touchpoint)
|
|
* 3.3) just update the data on others
|
|
* 4) force a sync on the device
|
|
* 5) make sure we get the right BTN_TOOL_ changes in the caller
|
|
*/
|
|
for (int i = 0; i < num_slots; i++) {
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, i,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 111,
|
|
EV_ABS, ABS_X, 100 + 10 * i,
|
|
EV_ABS, ABS_Y, 100 + 10 * i,
|
|
EV_ABS, ABS_MT_POSITION_X, 100,
|
|
EV_ABS, ABS_MT_POSITION_Y, 100,
|
|
-1, -1);
|
|
switch (i) {
|
|
case 0:
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_FINGER, 1);
|
|
break;
|
|
case 1:
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_FINGER, 0);
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_DOUBLETAP, 1);
|
|
break;
|
|
case 2:
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_DOUBLETAP, 0);
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_TRIPLETAP, 1);
|
|
break;
|
|
case 3:
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_TRIPLETAP, 0);
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_QUADTAP, 1);
|
|
break;
|
|
case 4:
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_QUADTAP, 0);
|
|
uinput_device_event(uidev, EV_KEY, BTN_TOOL_QUINTTAP, 1);
|
|
break;
|
|
default:
|
|
ck_abort();
|
|
}
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
}
|
|
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_ne(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
} while (rc >= 0);
|
|
|
|
/* we have a bunch of touches now, and libevdev knows it.
|
|
* - stop touch 0
|
|
* - stop and restart touch 1 and 4
|
|
* - leave 2, 3 unchanged
|
|
*/
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_MT_TRACKING_ID, -1,
|
|
EV_KEY, BTN_TOOL_QUINTTAP, 0,
|
|
EV_KEY, BTN_TOOL_QUADTAP, 1,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 1,
|
|
EV_ABS, ABS_MT_TRACKING_ID, -1,
|
|
EV_KEY, BTN_TOOL_QUADTAP, 0,
|
|
EV_KEY, BTN_TOOL_TRIPLETAP, 1,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 1,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 666,
|
|
EV_ABS, ABS_X, 666,
|
|
EV_ABS, ABS_Y, 666,
|
|
EV_ABS, ABS_MT_POSITION_X, 666,
|
|
EV_ABS, ABS_MT_POSITION_Y, 666,
|
|
EV_KEY, BTN_TOOL_TRIPLETAP, 0,
|
|
EV_KEY, BTN_TOOL_QUADTAP, 1,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 4,
|
|
EV_ABS, ABS_MT_TRACKING_ID, -1,
|
|
EV_KEY, BTN_TOOL_QUADTAP, 0,
|
|
EV_KEY, BTN_TOOL_TRIPLETAP, 1,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 4,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 777,
|
|
EV_ABS, ABS_X, 777,
|
|
EV_ABS, ABS_Y, 777,
|
|
EV_ABS, ABS_MT_POSITION_X, 777,
|
|
EV_ABS, ABS_MT_POSITION_Y, 777,
|
|
EV_KEY, BTN_TOOL_QUADTAP, 1,
|
|
EV_KEY, BTN_TOOL_TRIPLETAP, 0,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
/* Force sync */
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
/* In the first sync frame, we expect us to drop to 2 touches - we
|
|
* started with 5, 1 stopped, 2 stopped+restarted */
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev)) != -EAGAIN) {
|
|
if (libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_QUINTTAP)) {
|
|
ck_assert(!have_quinttap);
|
|
assert_event(&ev, EV_KEY, BTN_TOOL_QUINTTAP, 0);
|
|
have_quinttap = true;
|
|
}
|
|
|
|
if (libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_TRIPLETAP))
|
|
ck_abort();
|
|
|
|
if (libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_DOUBLETAP)) {
|
|
ck_assert(!have_doubletap);
|
|
assert_event(&ev, EV_KEY, BTN_TOOL_DOUBLETAP, 1);
|
|
have_doubletap = true;
|
|
}
|
|
|
|
ck_assert(!libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_TRIPLETAP));
|
|
ck_assert(!libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_QUADTAP));
|
|
ck_assert(!libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_FINGER));
|
|
|
|
if (libevdev_event_is_code(&ev, EV_SYN, SYN_REPORT)) {
|
|
ck_assert(have_doubletap);
|
|
ck_assert(have_quinttap);
|
|
break;
|
|
}
|
|
}
|
|
|
|
have_doubletap = false;
|
|
have_quadtap = false;
|
|
|
|
/* In the second sync frame, we expect to go back to 4 touches,
|
|
* recovering the two stopped+started touches */
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev)) != -EAGAIN) {
|
|
if (libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_QUADTAP)) {
|
|
ck_assert(!have_quadtap);
|
|
assert_event(&ev, EV_KEY, BTN_TOOL_QUADTAP, 1);
|
|
have_quadtap = true;
|
|
}
|
|
|
|
if (libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_TRIPLETAP))
|
|
ck_abort();
|
|
|
|
if (libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_DOUBLETAP)) {
|
|
ck_assert(!have_doubletap);
|
|
assert_event(&ev, EV_KEY, BTN_TOOL_DOUBLETAP, 0);
|
|
have_doubletap = true;
|
|
}
|
|
|
|
ck_assert(!libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_TRIPLETAP));
|
|
ck_assert(!libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_QUINTTAP));
|
|
ck_assert(!libevdev_event_is_code(&ev, EV_KEY, BTN_TOOL_FINGER));
|
|
|
|
if (libevdev_event_is_code(&ev, EV_SYN, SYN_REPORT)) {
|
|
ck_assert(have_doubletap);
|
|
ck_assert(have_quadtap);
|
|
break;
|
|
}
|
|
}
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_mt_tool_type)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
int i;
|
|
const int num_slots = 15;
|
|
int slot = -1;
|
|
unsigned long terminated[NLONGS(num_slots)];
|
|
struct input_absinfo abs[7] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_TOOL_TYPE, .maximum = MT_TOOL_PALM },
|
|
{ .value = ABS_MT_SLOT, .maximum = num_slots },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 0xff },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
for (i = num_slots; i >= 0; i--) {
|
|
int tool_type = MT_TOOL_FINGER;
|
|
switch (i) {
|
|
case 0:
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
tool_type = MT_TOOL_FINGER;
|
|
break;
|
|
case 4:
|
|
case 5:
|
|
case 6:
|
|
case 7:
|
|
tool_type = MT_TOOL_PALM;
|
|
break;
|
|
}
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, i,
|
|
EV_ABS, ABS_MT_TRACKING_ID, i,
|
|
EV_ABS, ABS_X, 100 + i,
|
|
EV_ABS, ABS_Y, 500 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 100 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500 + i,
|
|
EV_ABS, ABS_MT_TOOL_TYPE, tool_type,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_ne(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
} while (rc >= 0);
|
|
}
|
|
|
|
/* we have a bunch of touches now, and libevdev knows it. Change all
|
|
* touches */
|
|
for (i = num_slots; i >= 0; i--) {
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, i);
|
|
switch (i) {
|
|
/* Slot 0 is a finger and stays a finger */
|
|
case 0:
|
|
/* Slot 4 is a palm and stays a palm */
|
|
case 4:
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_X, 200 + i,
|
|
EV_ABS, ABS_Y, 700 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 200 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 700 + i,
|
|
-1, -1);
|
|
break;
|
|
/* Slot 1 is a finger and changes active touch to palm */
|
|
case 1:
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TOOL_TYPE, MT_TOOL_PALM);
|
|
break;
|
|
/* Slot 2 is a finger and terminates */
|
|
case 2:
|
|
/* Slot 6 is a palm and terminates */
|
|
case 6:
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, -1);
|
|
break;
|
|
/* Slot 3 is a finger and restarts as finger */
|
|
case 3:
|
|
/* Slot 5 is a palm and restarts as finger */
|
|
case 5:
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_TRACKING_ID, num_slots + i,
|
|
EV_ABS, ABS_X, 200 + i,
|
|
EV_ABS, ABS_Y, 700 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 200 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 700 + i,
|
|
EV_ABS, ABS_MT_TOOL_TYPE, MT_TOOL_FINGER,
|
|
-1, -1);
|
|
break;
|
|
/* Slot 7 is a palm and restarts and terminates again as finger */
|
|
case 7:
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, -1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_TRACKING_ID, num_slots + i,
|
|
EV_ABS, ABS_X, 200 + i,
|
|
EV_ABS, ABS_Y, 700 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 200 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 700 + i,
|
|
EV_ABS, ABS_MT_TOOL_TYPE, MT_TOOL_FINGER,
|
|
-1, -1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, -1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
break;
|
|
}
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
}
|
|
|
|
/* Force sync */
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
/* now check for the right tracking IDs */
|
|
memset(terminated, 0, sizeof(terminated));
|
|
slot = -1;
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev)) != -EAGAIN) {
|
|
if (libevdev_event_is_code(&ev, EV_SYN, SYN_REPORT))
|
|
continue;
|
|
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_MT_SLOT)) {
|
|
slot = ev.value;
|
|
continue;
|
|
}
|
|
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_X) ||
|
|
libevdev_event_is_code(&ev, EV_ABS, ABS_Y))
|
|
continue;
|
|
|
|
ck_assert_int_ne(slot, -1);
|
|
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_MT_TRACKING_ID)) {
|
|
switch (slot) {
|
|
case 0:
|
|
case 1:
|
|
case 4:
|
|
ck_abort_msg("No ABS_MT_TRACKING_ID expected for this slot");
|
|
break;
|
|
case 2:
|
|
case 6:
|
|
case 7:
|
|
ck_assert_int_eq(ev.value, -1);
|
|
break;
|
|
case 3:
|
|
case 5:
|
|
if (!bit_is_set(terminated, slot)) {
|
|
ck_assert_int_eq(ev.value, -1);
|
|
set_bit(terminated, slot);
|
|
} else {
|
|
ck_assert_int_eq(ev.value, num_slots + slot);
|
|
}
|
|
break;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
if (libevdev_event_is_code(&ev, EV_ABS, ABS_MT_TOOL_TYPE)) {
|
|
switch (slot) {
|
|
case 0:
|
|
case 2:
|
|
case 3:
|
|
case 4:
|
|
case 6:
|
|
ck_abort_msg("No ABS_MT_TOOL_TYPE expected for this slot");
|
|
break;
|
|
case 1:
|
|
ck_assert_int_eq(ev.value, MT_TOOL_PALM);
|
|
break;
|
|
case 5:
|
|
case 7:
|
|
ck_assert_int_eq(ev.value, MT_TOOL_FINGER);
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
switch(ev.code) {
|
|
case ABS_MT_POSITION_X:
|
|
ck_assert_int_eq(ev.value, 200 + slot);
|
|
break;
|
|
case ABS_MT_POSITION_Y:
|
|
ck_assert_int_eq(ev.value, 700 + slot);
|
|
break;
|
|
default:
|
|
ck_abort();
|
|
}
|
|
}
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_late_sync)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[6] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 1 },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 0xff},
|
|
};
|
|
int i, slot;
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
/* emulate a touch down, make sure libevdev sees it */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 1,
|
|
EV_ABS, ABS_X, 100,
|
|
EV_ABS, ABS_Y, 500,
|
|
EV_ABS, ABS_MT_POSITION_X, 100,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_ne(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
} while (rc >= 0);
|
|
|
|
/* force enough events to trigger a SYN_DROPPED */
|
|
for (i = 0; i < 100; i++) {
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_X, 100 + i,
|
|
EV_ABS, ABS_Y, 500 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 100 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500 + i,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
}
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
/* trigger the tracking ID change after getting the SYN_DROPPED */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_MT_TRACKING_ID, -1,
|
|
EV_ABS, ABS_X, 200,
|
|
EV_ABS, ABS_Y, 600,
|
|
EV_ABS, ABS_MT_POSITION_X, 200,
|
|
EV_ABS, ABS_MT_POSITION_Y, 600,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
slot = 0;
|
|
|
|
/* Now sync the device, expect the data to be equal to the last event*/
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev)) != -EAGAIN) {
|
|
if (ev.type == EV_SYN)
|
|
continue;
|
|
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
switch(ev.code) {
|
|
case ABS_MT_SLOT:
|
|
slot = ev.value;
|
|
break;
|
|
case ABS_MT_TRACKING_ID:
|
|
if (slot == 0)
|
|
ck_assert_int_eq(ev.value, -1);
|
|
break;
|
|
case ABS_X:
|
|
case ABS_MT_POSITION_X:
|
|
ck_assert_int_eq(ev.value, 200);
|
|
break;
|
|
case ABS_Y:
|
|
case ABS_MT_POSITION_Y:
|
|
ck_assert_int_eq(ev.value, 600);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* And a new tracking ID */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 2,
|
|
EV_ABS, ABS_X, 201,
|
|
EV_ABS, ABS_Y, 601,
|
|
EV_ABS, ABS_MT_POSITION_X, 201,
|
|
EV_ABS, ABS_MT_POSITION_Y, 601,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev)) != -EAGAIN) {
|
|
ck_assert_int_ne(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
if (ev.type == EV_SYN)
|
|
continue;
|
|
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
|
|
switch(ev.code) {
|
|
case ABS_MT_SLOT:
|
|
ck_assert_int_eq(ev.value, 0);
|
|
break;
|
|
case ABS_MT_TRACKING_ID:
|
|
ck_assert_int_eq(ev.value, 2);
|
|
break;
|
|
case ABS_X:
|
|
case ABS_MT_POSITION_X:
|
|
ck_assert_int_eq(ev.value, 201);
|
|
break;
|
|
case ABS_Y:
|
|
case ABS_MT_POSITION_Y:
|
|
ck_assert_int_eq(ev.value, 601);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Now we basically re-do the exact same test, just with the
|
|
tracking ID order inverted */
|
|
|
|
/* drop the tracking ID, make sure libevdev sees it */
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 0);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, -1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_ne(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
} while (rc >= 0);
|
|
|
|
/* force enough events to trigger a SYN_DROPPED */
|
|
for (i = 0; i < 100; i++) {
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_X, 100 + i,
|
|
EV_ABS, ABS_Y, 500 + i,
|
|
EV_ABS, ABS_MT_POSITION_X, 100 + i,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500 + i,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
}
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
/* trigger the new tracking ID after getting the SYN_DROPPED */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_MT_TRACKING_ID, 5,
|
|
EV_ABS, ABS_X, 200,
|
|
EV_ABS, ABS_Y, 600,
|
|
EV_ABS, ABS_MT_POSITION_X, 200,
|
|
EV_ABS, ABS_MT_POSITION_Y, 600,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
slot = 0;
|
|
|
|
/* Now sync the device, expect the data to be equal to the last event*/
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev)) != -EAGAIN) {
|
|
if (ev.type == EV_SYN)
|
|
continue;
|
|
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
switch(ev.code) {
|
|
case ABS_MT_SLOT:
|
|
slot = ev.value;
|
|
break;
|
|
case ABS_MT_TRACKING_ID:
|
|
if (slot == 0)
|
|
ck_assert_int_eq(ev.value, 5);
|
|
break;
|
|
case ABS_X:
|
|
case ABS_MT_POSITION_X:
|
|
ck_assert_int_eq(ev.value, 200);
|
|
break;
|
|
case ABS_Y:
|
|
case ABS_MT_POSITION_Y:
|
|
ck_assert_int_eq(ev.value, 600);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Drop the tracking ID */
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 0);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, -1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev)) != -EAGAIN) {
|
|
ck_assert_int_ne(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
if (ev.type == EV_SYN)
|
|
continue;
|
|
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
|
|
switch(ev.code) {
|
|
case ABS_MT_SLOT:
|
|
ck_assert_int_eq(ev.value, 0);
|
|
break;
|
|
case ABS_MT_TRACKING_ID:
|
|
ck_assert_int_eq(ev.value, -1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_fake_mt)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[] = {
|
|
{ .value = ABS_X, .minimum = 0, .maximum = 1000 },
|
|
{ .value = ABS_Y, .minimum = 0, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .minimum = 0, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .minimum = 0, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT - 1, .minimum = 0, .maximum = 2 }};
|
|
/* don't set ABS_MT_SLOT here, otherwise uinput will init
|
|
* slots and the behavior is different to real devices with
|
|
* such events */
|
|
unsigned long received[NLONGS(ABS_CNT)] = {0};
|
|
|
|
test_create_abs_device(&uidev, &dev, ARRAY_LENGTH(abs), abs, -1);
|
|
/* first set of events */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_X, 200,
|
|
EV_ABS, ABS_Y, 400,
|
|
EV_ABS, ABS_MT_POSITION_X, 100,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500,
|
|
EV_ABS, ABS_MT_SLOT - 1, 1,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
/* second set of events */
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_X, 201,
|
|
EV_ABS, ABS_Y, 401,
|
|
EV_ABS, ABS_MT_POSITION_X, 101,
|
|
EV_ABS, ABS_MT_POSITION_Y, 501,
|
|
EV_ABS, ABS_MT_SLOT - 1, 2,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
while ((rc = libevdev_next_event(dev, LIBEVDEV_READ_STATUS_SYNC, &ev)) != -EAGAIN) {
|
|
if (ev.type != EV_ABS)
|
|
continue;
|
|
|
|
ck_assert(!bit_is_set(received, ev.code));
|
|
|
|
switch(ev.code) {
|
|
/* see comment below for ABS_MT_POSITION_X
|
|
* and ABS_MT_POSITION_Y */
|
|
case ABS_MT_POSITION_X:
|
|
case ABS_MT_POSITION_Y:
|
|
ck_abort();
|
|
break;
|
|
|
|
case ABS_MT_SLOT - 1: ck_assert_int_eq(ev.value, 2); break;
|
|
case ABS_X: ck_assert_int_eq(ev.value, 201); break;
|
|
case ABS_Y: ck_assert_int_eq(ev.value, 401); break;
|
|
default:
|
|
ck_abort();
|
|
}
|
|
|
|
set_bit(received, ev.code);
|
|
}
|
|
|
|
/* Dont' expect ABS_MT values, they are ignored during the sync
|
|
* process */
|
|
ck_assert(!bit_is_set(received, ABS_MT_POSITION_X));
|
|
ck_assert(!bit_is_set(received, ABS_MT_POSITION_Y));
|
|
ck_assert(bit_is_set(received, ABS_MT_SLOT - 1));
|
|
ck_assert(bit_is_set(received, ABS_X));
|
|
ck_assert(bit_is_set(received, ABS_Y));
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_X), 201);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Y), 401);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_MT_SLOT - 1), 2);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_skipped_sync)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[2] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_X, 100);
|
|
uinput_device_event(uidev, EV_ABS, ABS_Y, 500);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_LEFT), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_X), 100);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Y), 500);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_incomplete_sync)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[2] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_X, 100);
|
|
uinput_device_event(uidev, EV_ABS, ABS_Y, 500);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
assert_event(&ev, EV_KEY, BTN_LEFT, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_LEFT), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_X), 100);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Y), 500);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_empty_sync)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
test_create_device(&uidev, &dev,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SYNC);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_SYNC, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_values)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[2] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
};
|
|
int value;
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_X, 100);
|
|
uinput_device_event(uidev, EV_ABS, ABS_Y, 500);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
/* must still be on old values */
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_LEFT), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_X), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Y), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REL, REL_X), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REL, REL_Y), 0);
|
|
|
|
ck_assert_int_eq(libevdev_fetch_event_value(dev, EV_KEY, BTN_LEFT, &value), 1);
|
|
ck_assert_int_eq(value, 0);
|
|
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
} while (rc == 0);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_LEFT), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_X), 100);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Y), 500);
|
|
|
|
/* always 0 */
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REL, REL_X), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REL, REL_Y), 0);
|
|
|
|
ck_assert_int_eq(libevdev_fetch_event_value(dev, EV_KEY, BTN_LEFT, &value), 1);
|
|
ck_assert_int_eq(value, 1);
|
|
ck_assert_int_eq(libevdev_fetch_event_value(dev, EV_ABS, ABS_X, &value), 1);
|
|
ck_assert_int_eq(value, 100);
|
|
ck_assert_int_eq(libevdev_fetch_event_value(dev, EV_ABS, ABS_Y, &value), 1);
|
|
ck_assert_int_eq(value, 500);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_values_invalid)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_absinfo abs[2] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
};
|
|
int value;
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_EXTRA), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Z), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REL, REL_Z), 0);
|
|
|
|
value = 0xab;
|
|
ck_assert_int_eq(libevdev_fetch_event_value(dev, EV_KEY, BTN_EXTRA, &value), 0);
|
|
ck_assert_int_eq(value, 0xab);
|
|
ck_assert_int_eq(libevdev_fetch_event_value(dev, EV_ABS, ABS_Z, &value), 0);
|
|
ck_assert_int_eq(value, 0xab);
|
|
ck_assert_int_eq(libevdev_fetch_event_value(dev, EV_REL, REL_Z, &value), 0);
|
|
ck_assert_int_eq(value, 0xab);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_mt_event_values)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[5] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 2 },
|
|
};
|
|
int value;
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
uinput_device_event_multiple(uidev,
|
|
EV_ABS, ABS_MT_SLOT, 0,
|
|
EV_ABS, ABS_X, 100,
|
|
EV_ABS, ABS_Y, 500,
|
|
EV_ABS, ABS_MT_POSITION_X, 100,
|
|
EV_ABS, ABS_MT_POSITION_Y, 500,
|
|
EV_ABS, ABS_MT_SLOT, 1,
|
|
EV_ABS, ABS_X, 1,
|
|
EV_ABS, ABS_Y, 5,
|
|
EV_ABS, ABS_MT_POSITION_X, 1,
|
|
EV_ABS, ABS_MT_POSITION_Y, 5,
|
|
EV_SYN, SYN_REPORT, 0,
|
|
-1, -1);
|
|
|
|
/* must still be on old values */
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 0);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_POSITION_X), 0);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_POSITION_Y), 0);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_POSITION_X), 0);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_POSITION_Y), 0);
|
|
|
|
do {
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
} while (rc == LIBEVDEV_READ_STATUS_SUCCESS);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 1);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_POSITION_X), 100);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_POSITION_Y), 500);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_POSITION_X), 1);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_POSITION_Y), 5);
|
|
|
|
ck_assert_int_eq(libevdev_fetch_slot_value(dev, 0, ABS_MT_POSITION_X, &value), 1);
|
|
ck_assert_int_eq(value, 100);
|
|
ck_assert_int_eq(libevdev_fetch_slot_value(dev, 0, ABS_MT_POSITION_Y, &value), 1);
|
|
ck_assert_int_eq(value, 500);
|
|
ck_assert_int_eq(libevdev_fetch_slot_value(dev, 1, ABS_MT_POSITION_X, &value), 1);
|
|
ck_assert_int_eq(value, 1);
|
|
ck_assert_int_eq(libevdev_fetch_slot_value(dev, 1, ABS_MT_POSITION_Y, &value), 1);
|
|
ck_assert_int_eq(value, 5);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_mt_event_values_invalid)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_absinfo abs[5] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 2 },
|
|
};
|
|
int value;
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 0);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_TOUCH_MINOR), 0);
|
|
value = 0xab;
|
|
ck_assert_int_eq(libevdev_fetch_slot_value(dev, 0, ABS_MT_TOUCH_MINOR, &value), 0);
|
|
ck_assert_int_eq(value, 0xab);
|
|
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 10, ABS_MT_POSITION_X), 0);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_X), 0);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_mt_slot_ranges_invalid)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_event ev[2];
|
|
int rc;
|
|
int num_slots = 2;
|
|
struct input_absinfo abs[5] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = num_slots - 1 },
|
|
};
|
|
int pipefd[2];
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
rc = pipe2(pipefd, O_NONBLOCK);
|
|
ck_assert_int_eq(rc, 0);
|
|
libevdev_change_fd(dev, pipefd[0]);
|
|
|
|
memset(ev, 0, sizeof(ev));
|
|
ev[0].type = EV_ABS;
|
|
ev[0].code = ABS_MT_SLOT;
|
|
ev[0].value = num_slots;
|
|
ev[1].type = EV_SYN;
|
|
ev[1].code = SYN_REPORT;
|
|
ev[1].value = 0;
|
|
rc = write(pipefd[1], ev, sizeof(ev));
|
|
ck_assert_int_eq(rc, sizeof(ev));
|
|
|
|
libevdev_set_log_function(test_logfunc_ignore_error, NULL);
|
|
|
|
libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, ev);
|
|
ck_assert(libevdev_event_is_code(ev, EV_ABS, ABS_MT_SLOT));
|
|
ck_assert_int_eq(ev[0].value, num_slots - 1);
|
|
|
|
/* drain the EV_SYN */
|
|
libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, ev);
|
|
|
|
ev[0].type = EV_ABS;
|
|
ev[0].code = ABS_MT_SLOT;
|
|
ev[0].value = -1;
|
|
ev[1].type = EV_SYN;
|
|
ev[1].code = SYN_REPORT;
|
|
ev[1].value = 0;
|
|
rc = write(pipefd[1], ev, sizeof(ev));
|
|
ck_assert_int_eq(rc, sizeof(ev));
|
|
|
|
libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, ev);
|
|
ck_assert(libevdev_event_is_code(ev, EV_ABS, ABS_MT_SLOT));
|
|
ck_assert_int_eq(ev[0].value, num_slots - 1);
|
|
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), num_slots - 1);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_MT_SLOT, num_slots), -1);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_MT_SLOT, -1), -1);
|
|
|
|
libevdev_set_log_function(test_logfunc_abort_on_error, NULL);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_mt_tracking_id_discard)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[6] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 10 },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 500 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
/* second tracking ID on same slot */
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, 2);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
libevdev_set_log_function(test_logfunc_ignore_error, NULL);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_ABS, ABS_MT_SLOT, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_ABS, ABS_MT_TRACKING_ID, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
/* expect tracking ID discarded */
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
libevdev_set_log_function(test_logfunc_abort_on_error, NULL);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_mt_tracking_id_discard_neg_1)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
struct input_absinfo abs[6] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 10 },
|
|
{ .value = ABS_MT_TRACKING_ID, .minimum = -1, .maximum = 500 },
|
|
};
|
|
int pipefd[2];
|
|
struct input_event events[] = {
|
|
{ .type = EV_ABS, .code = ABS_MT_TRACKING_ID, .value = -1 },
|
|
{ .type = EV_SYN, .code = SYN_REPORT, .value = 0 },
|
|
};
|
|
|
|
rc = pipe2(pipefd, O_NONBLOCK);
|
|
ck_assert_int_eq(rc, 0);
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
while (libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev) != -EAGAIN)
|
|
;
|
|
|
|
libevdev_set_log_function(test_logfunc_ignore_error, NULL);
|
|
|
|
/* two -1 tracking ids, need to use the pipe here, the kernel will
|
|
filter it otherwise */
|
|
libevdev_change_fd(dev, pipefd[0]);
|
|
|
|
rc = write(pipefd[1], events, sizeof(events));
|
|
ck_assert_int_eq(rc, sizeof(events));
|
|
rc = write(pipefd[1], events, sizeof(events));
|
|
ck_assert_int_eq(rc, sizeof(events));
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_ABS, ABS_MT_TRACKING_ID, -1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
/* expect second tracking ID discarded */
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
|
|
assert_event(&ev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
libevdev_set_log_function(test_logfunc_abort_on_error, NULL);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_ev_rep_values)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int delay = 500, period = 200;
|
|
test_create_device(&uidev, &dev,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
libevdev_enable_event_code(dev, EV_REP, REP_DELAY, &delay);
|
|
libevdev_enable_event_code(dev, EV_REP, REP_PERIOD, &period);
|
|
|
|
ck_assert_int_eq(libevdev_has_event_type(dev, EV_REP), 1);
|
|
ck_assert_int_eq(libevdev_has_event_code(dev, EV_REP, REP_DELAY), 1);
|
|
ck_assert_int_eq(libevdev_has_event_code(dev, EV_REP, REP_PERIOD), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REP, REP_DELAY), 500);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REP, REP_PERIOD), 200);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_value_setters)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_absinfo abs[2] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
EV_LED, LED_NUML,
|
|
EV_LED, LED_CAPSL,
|
|
EV_SW, SW_HEADPHONE_INSERT,
|
|
EV_SW, SW_TABLET_MODE,
|
|
-1);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_LEFT), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_X), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Y), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REL, REL_X), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_REL, REL_Y), 0);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_KEY, BTN_LEFT, 1), 0);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_KEY, BTN_RIGHT, 1), 0);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_LEFT), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_KEY, BTN_RIGHT), 1);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_X, 10), 0);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_Y, 20), 0);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_X), 10);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_Y), 20);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_LED, LED_NUML, 1), 0);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_LED, LED_CAPSL, 1), 0);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_LED, LED_NUML), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_LED, LED_CAPSL), 1);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_SW, SW_HEADPHONE_INSERT, 1), 0);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_SW, SW_TABLET_MODE, 1), 0);
|
|
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_SW, SW_HEADPHONE_INSERT), 1);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_SW, SW_TABLET_MODE), 1);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_value_setters_invalid)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_absinfo abs[2] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_REL, REL_X, 1), -1);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_SW, SW_DOCK, 1), -1);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_Z, 1), -1);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_MAX + 1, 0, 1), -1);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_SYN, SYN_REPORT, 0), -1);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_mt_value_setters)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_absinfo abs[5] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 2 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 1, ABS_MT_POSITION_X, 1), 0);
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 1, ABS_MT_POSITION_Y, 2), 0);
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 0, ABS_MT_POSITION_X, 3), 0);
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 0, ABS_MT_POSITION_Y, 4), 0);
|
|
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_POSITION_X), 1);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_POSITION_Y), 2);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_POSITION_X), 3);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_POSITION_Y), 4);
|
|
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 1, ABS_MT_SLOT, 1), 0);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_SLOT), 1);
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 1);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_mt_value_setters_invalid)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_absinfo abs[5] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 2 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
/* invalid axis */
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 1, ABS_Z, 1), -1);
|
|
/* valid, but non-mt axis */
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 1, ABS_X, 1), -1);
|
|
/* invalid mt axis */
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 1, ABS_MT_PRESSURE, 1), -1);
|
|
/* invalid slot no */
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 4, ABS_X, 1), -1);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_event_mt_value_setters_current_slot)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
struct input_absinfo abs[5] = {
|
|
{ .value = ABS_X, .maximum = 1000 },
|
|
{ .value = ABS_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_X, .maximum = 1000 },
|
|
{ .value = ABS_MT_POSITION_Y, .maximum = 1000 },
|
|
{ .value = ABS_MT_SLOT, .maximum = 2 },
|
|
};
|
|
|
|
test_create_abs_device(&uidev, &dev,
|
|
ARRAY_LENGTH(abs), abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
|
|
/* set_event_value/get_event_value works on the current slot */
|
|
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 0);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_MT_POSITION_X, 1), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_MT_POSITION_X), 1);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 0, ABS_MT_POSITION_X), 1);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_MT_SLOT, 1), 0);
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 1);
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_MT_POSITION_X, 2), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_MT_POSITION_X), 2);
|
|
ck_assert_int_eq(libevdev_get_slot_value(dev, 1, ABS_MT_POSITION_X), 2);
|
|
|
|
/* set slot 0, but current is still slot 1 */
|
|
ck_assert_int_eq(libevdev_set_slot_value(dev, 0, ABS_MT_POSITION_X, 3), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_MT_POSITION_X), 2);
|
|
|
|
ck_assert_int_eq(libevdev_set_event_value(dev, EV_ABS, ABS_MT_SLOT, 0), 0);
|
|
ck_assert_int_eq(libevdev_get_current_slot(dev), 0);
|
|
ck_assert_int_eq(libevdev_get_event_value(dev, EV_ABS, ABS_MT_POSITION_X), 3);
|
|
|
|
uinput_device_free(uidev);
|
|
libevdev_free(dev);
|
|
}
|
|
END_TEST
|
|
|
|
TEST_SUITE_ROOT_PRIVILEGES(libevdev_events)
|
|
{
|
|
Suite *s = suite_create("libevdev event tests");
|
|
|
|
add_test(s, test_next_event);
|
|
add_test(s, test_next_event_invalid_fd);
|
|
add_test(s, test_next_event_blocking);
|
|
add_test(s, test_syn_dropped_event);
|
|
add_test(s, test_event_type_filtered);
|
|
add_test(s, test_event_code_filtered);
|
|
add_test(s, test_has_event_pending);
|
|
add_test(s, test_has_event_pending_invalid_fd);
|
|
|
|
add_test(s, test_syn_delta_button);
|
|
add_test(s, test_syn_delta_abs);
|
|
add_test(s, test_syn_delta_mt);
|
|
add_test(s, test_syn_delta_mt_reset_slot);
|
|
add_test(s, test_syn_delta_led);
|
|
add_test(s, test_syn_delta_sw);
|
|
add_test(s, test_syn_delta_fake_mt);
|
|
add_test(s, test_syn_delta_late_sync);
|
|
add_test(s, test_syn_delta_tracking_ids);
|
|
add_test(s, test_syn_delta_tracking_ids_btntool);
|
|
add_test(s, test_syn_delta_mt_tool_type);
|
|
|
|
add_test(s, test_skipped_sync);
|
|
add_test(s, test_incomplete_sync);
|
|
add_test(s, test_empty_sync);
|
|
|
|
add_test(s, test_event_values);
|
|
add_test(s, test_event_values_invalid);
|
|
add_test(s, test_mt_event_values);
|
|
add_test(s, test_mt_event_values_invalid);
|
|
add_test(s, test_mt_slot_ranges_invalid);
|
|
add_test(s, test_mt_tracking_id_discard);
|
|
add_test(s, test_mt_tracking_id_discard_neg_1);
|
|
add_test(s, test_ev_rep_values);
|
|
|
|
add_test(s, test_event_value_setters);
|
|
add_test(s, test_event_value_setters_invalid);
|
|
add_test(s, test_event_mt_value_setters);
|
|
add_test(s, test_event_mt_value_setters_invalid);
|
|
add_test(s, test_event_mt_value_setters_current_slot);
|
|
|
|
return s;
|
|
}
|