mirror of
https://gitlab.freedesktop.org/libevdev/libevdev.git
synced 2025-12-20 13:50:16 +01:00
803 lines
23 KiB
C
803 lines
23 KiB
C
/*
|
|
* Copyright © 2013 Red Hat, Inc.
|
|
*
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
* the above copyright notice appear in all copies and that both that copyright
|
|
* notice and this permission notice appear in supporting documentation, and
|
|
* that the name of the copyright holders not be used in advertising or
|
|
* publicity pertaining to distribution of the software without specific,
|
|
* written prior permission. The copyright holders make no representations
|
|
* about the suitability of this software for any purpose. It is provided "as
|
|
* is" without express or implied warranty.
|
|
*
|
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
|
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
|
|
* EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
|
|
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
|
|
* DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
|
|
* OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
#include <config.h>
|
|
#include <linux/input.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
|
|
#include "test-common.h"
|
|
|
|
START_TEST(test_next_event)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
rc = test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, 0);
|
|
ck_assert_int_eq(ev.type, EV_KEY);
|
|
ck_assert_int_eq(ev.code, BTN_LEFT);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_event)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
int pipefd[2];
|
|
|
|
rc = 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);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
/* This is a bid complicated:
|
|
we can't get SYN_DROPPED through uinput, so we push two events down
|
|
uinput, and fetch one off libevdev (reading in the other one on the
|
|
way). Then write a SYN_DROPPED on a pipe, switch the fd and read
|
|
one event off the wire (but returning the second event from
|
|
before). 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_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, 0);
|
|
ck_assert_int_eq(ev.type, EV_KEY);
|
|
ck_assert_int_eq(ev.code, BTN_LEFT);
|
|
|
|
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_NORMAL, &ev);
|
|
|
|
libevdev_change_fd(dev, uinput_device_get_fd(uidev));
|
|
|
|
ck_assert_int_eq(rc, 0);
|
|
ck_assert_int_eq(ev.type, EV_SYN);
|
|
ck_assert_int_eq(ev.code, SYN_REPORT);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
|
|
/* 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;
|
|
|
|
rc = test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
libevdev_disable_event_type(dev, EV_REL);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, 0);
|
|
ck_assert_int_eq(ev.type, EV_SYN);
|
|
ck_assert_int_eq(ev.code, SYN_REPORT);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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;
|
|
|
|
rc = test_create_device(&uidev, &dev,
|
|
EV_REL, REL_X,
|
|
EV_REL, REL_Y,
|
|
EV_KEY, BTN_LEFT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
libevdev_disable_event_code(dev, EV_REL, REL_X);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, 0);
|
|
ck_assert_int_eq(ev.type, EV_REL);
|
|
ck_assert_int_eq(ev.code, REL_Y);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, 0);
|
|
ck_assert_int_eq(ev.type, EV_SYN);
|
|
ck_assert_int_eq(ev.code, SYN_REPORT);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_NORMAL, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
libevdev_free(dev);
|
|
uinput_device_free(uidev);
|
|
|
|
}
|
|
END_TEST
|
|
|
|
START_TEST(test_syn_delta_button)
|
|
{
|
|
struct uinput_device* uidev;
|
|
struct libevdev *dev;
|
|
int rc;
|
|
struct input_event ev;
|
|
|
|
rc = 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,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
uinput_device_event(uidev, EV_KEY, BTN_LEFT, 1);
|
|
uinput_device_event(uidev, EV_KEY, BTN_RIGHT, 1);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_KEY);
|
|
ck_assert_int_eq(ev.code, BTN_LEFT);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_KEY);
|
|
ck_assert_int_eq(ev.code, BTN_RIGHT);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_SYN);
|
|
ck_assert_int_eq(ev.code, SYN_REPORT);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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));
|
|
|
|
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[2];
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
|
|
abs[1].value = ABS_Y;
|
|
abs[1].maximum = 1000;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
2, abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
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_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_X);
|
|
ck_assert_int_eq(ev.value, 100);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_Y);
|
|
ck_assert_int_eq(ev.value, 500);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_SYN);
|
|
ck_assert_int_eq(ev.code, SYN_REPORT);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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];
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
abs[1].value = ABS_MT_POSITION_X;
|
|
abs[1].maximum = 1000;
|
|
|
|
abs[2].value = ABS_Y;
|
|
abs[2].maximum = 1000;
|
|
abs[3].value = ABS_MT_POSITION_Y;
|
|
abs[3].maximum = 1000;
|
|
|
|
|
|
abs[4].value = ABS_MT_SLOT;
|
|
abs[4].maximum = 1;
|
|
abs[5].value = ABS_MT_TRACKING_ID;
|
|
abs[5].minimum = -1;
|
|
abs[5].maximum = 2;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
6, abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 0);
|
|
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_MT_POSITION_X, 100);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_POSITION_Y, 500);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_X, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_Y, 5);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_POSITION_X, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_POSITION_Y, 5);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_TRACKING_ID, 2);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_X);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_Y);
|
|
ck_assert_int_eq(ev.value, 5);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_SLOT);
|
|
ck_assert_int_eq(ev.value, 0);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_POSITION_X);
|
|
ck_assert_int_eq(ev.value, 100);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_POSITION_Y);
|
|
ck_assert_int_eq(ev.value, 500);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_TRACKING_ID);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_SLOT);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_POSITION_X);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_POSITION_Y);
|
|
ck_assert_int_eq(ev.value, 5);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_ABS);
|
|
ck_assert_int_eq(ev.code, ABS_MT_TRACKING_ID);
|
|
ck_assert_int_eq(ev.value, 2);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_SYN);
|
|
ck_assert_int_eq(ev.code, SYN_REPORT);
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, -EAGAIN);
|
|
|
|
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];
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
|
|
abs[1].value = ABS_Y;
|
|
abs[1].maximum = 1000;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
2, abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
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_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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];
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
|
|
abs[1].value = ABS_Y;
|
|
abs[1].maximum = 1000;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
2, abs,
|
|
EV_SYN, SYN_REPORT,
|
|
EV_SYN, SYN_DROPPED,
|
|
EV_KEY, BTN_LEFT,
|
|
EV_KEY, BTN_MIDDLE,
|
|
EV_KEY, BTN_RIGHT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
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_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
ck_assert_int_eq(ev.type, EV_KEY);
|
|
ck_assert_int_eq(ev.code, BTN_LEFT);
|
|
ck_assert_int_eq(ev.value, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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;
|
|
|
|
rc = 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);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_FORCE_SYNC, &ev);
|
|
ck_assert_int_eq(rc, 1);
|
|
|
|
rc = libevdev_next_event(dev, LIBEVDEV_READ_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];
|
|
int value;
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
|
|
abs[1].value = ABS_Y;
|
|
abs[1].maximum = 1000;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
2, 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_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
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_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;
|
|
int rc;
|
|
struct input_absinfo abs[2];
|
|
int value;
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
|
|
abs[1].value = ABS_Y;
|
|
abs[1].maximum = 1000;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
2, 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_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
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];
|
|
int value;
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
abs[1].value = ABS_MT_POSITION_X;
|
|
abs[1].maximum = 1000;
|
|
|
|
abs[2].value = ABS_Y;
|
|
abs[2].maximum = 1000;
|
|
abs[3].value = ABS_MT_POSITION_Y;
|
|
abs[3].maximum = 1000;
|
|
|
|
abs[4].value = ABS_MT_SLOT;
|
|
abs[4].maximum = 2;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
5, abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 0);
|
|
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_MT_POSITION_X, 100);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_POSITION_Y, 500);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_SLOT, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_X, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_Y, 5);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_POSITION_X, 1);
|
|
uinput_device_event(uidev, EV_ABS, ABS_MT_POSITION_Y, 5);
|
|
uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
|
|
|
|
/* 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_NORMAL, &ev);
|
|
} while (rc == 0);
|
|
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;
|
|
int rc;
|
|
struct input_absinfo abs[5];
|
|
int value;
|
|
|
|
memset(abs, 0, sizeof(abs));
|
|
abs[0].value = ABS_X;
|
|
abs[0].maximum = 1000;
|
|
abs[1].value = ABS_MT_POSITION_X;
|
|
abs[1].maximum = 1000;
|
|
|
|
abs[2].value = ABS_Y;
|
|
abs[2].maximum = 1000;
|
|
abs[3].value = ABS_MT_POSITION_Y;
|
|
abs[3].maximum = 1000;
|
|
|
|
abs[4].value = ABS_MT_SLOT;
|
|
abs[4].maximum = 2;
|
|
|
|
rc = test_create_abs_device(&uidev, &dev,
|
|
5, abs,
|
|
EV_SYN, SYN_REPORT,
|
|
-1);
|
|
ck_assert_msg(rc == 0, "Failed to create device: %s", strerror(-rc));
|
|
|
|
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
|
|
|
|
Suite *
|
|
libevdev_events(void)
|
|
{
|
|
Suite *s = suite_create("libevdev event tests");
|
|
|
|
TCase *tc = tcase_create("event polling");
|
|
tcase_add_test(tc, test_next_event);
|
|
tcase_add_test(tc, test_syn_event);
|
|
tcase_add_test(tc, test_event_type_filtered);
|
|
tcase_add_test(tc, test_event_code_filtered);
|
|
suite_add_tcase(s, tc);
|
|
|
|
tc = tcase_create("SYN_DROPPED deltas");
|
|
tcase_add_test(tc, test_syn_delta_button);
|
|
tcase_add_test(tc, test_syn_delta_abs);
|
|
tcase_add_test(tc, test_syn_delta_mt);
|
|
suite_add_tcase(s, tc);
|
|
|
|
tc = tcase_create("skipped syncs");
|
|
tcase_add_test(tc, test_skipped_sync);
|
|
tcase_add_test(tc, test_incomplete_sync);
|
|
tcase_add_test(tc, test_empty_sync);
|
|
suite_add_tcase(s, tc);
|
|
|
|
tc = tcase_create("event values");
|
|
tcase_add_test(tc, test_event_values);
|
|
tcase_add_test(tc, test_event_values_invalid);
|
|
tcase_add_test(tc, test_mt_event_values);
|
|
tcase_add_test(tc, test_mt_event_values_invalid);
|
|
suite_add_tcase(s, tc);
|
|
|
|
return s;
|
|
}
|
|
|