Blame test/test-kernel.c

Packit 57e92c
/*
Packit 57e92c
 * Copyright © 2014 Red Hat, Inc.
Packit 57e92c
 *
Packit 57e92c
 * Permission to use, copy, modify, distribute, and sell this software and its
Packit 57e92c
 * documentation for any purpose is hereby granted without fee, provided that
Packit 57e92c
 * the above copyright notice appear in all copies and that both that copyright
Packit 57e92c
 * notice and this permission notice appear in supporting documentation, and
Packit 57e92c
 * that the name of the copyright holders not be used in advertising or
Packit 57e92c
 * publicity pertaining to distribution of the software without specific,
Packit 57e92c
 * written prior permission.  The copyright holders make no representations
Packit 57e92c
 * about the suitability of this software for any purpose.  It is provided "as
Packit 57e92c
 * is" without express or implied warranty.
Packit 57e92c
 *
Packit 57e92c
 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
Packit 57e92c
 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
Packit 57e92c
 * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
Packit 57e92c
 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
Packit 57e92c
 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
Packit 57e92c
 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
Packit 57e92c
 * OF THIS SOFTWARE.
Packit 57e92c
 */
Packit 57e92c
Packit 57e92c
#include <config.h>
Packit 57e92c
#include <errno.h>
Packit 57e92c
#include <inttypes.h>
Packit 57e92c
#include <unistd.h>
Packit 57e92c
#include <time.h>
Packit 57e92c
#include <sys/types.h>
Packit 57e92c
#include <sys/stat.h>
Packit 57e92c
#include <fcntl.h>
Packit 57e92c
#include <stdio.h>
Packit 57e92c
#include <linux/input.h>
Packit 57e92c
Packit 57e92c
#include <libevdev/libevdev.h>
Packit 57e92c
#include <libevdev/libevdev-uinput.h>
Packit 57e92c
#include "test-common.h"
Packit 57e92c
Packit 57e92c
START_TEST(test_revoke)
Packit 57e92c
{
Packit 57e92c
	struct uinput_device* uidev;
Packit 57e92c
	struct libevdev *dev, *dev2;
Packit 57e92c
	int rc, fd;
Packit 57e92c
	struct input_event ev1, ev2;
Packit 57e92c
	int dev_fd;
Packit 57e92c
Packit 57e92c
	test_create_device(&uidev, &dev,
Packit 57e92c
			   EV_SYN, SYN_REPORT,
Packit 57e92c
			   EV_REL, REL_X,
Packit 57e92c
			   EV_REL, REL_Y,
Packit 57e92c
			   EV_REL, REL_WHEEL,
Packit 57e92c
			   EV_KEY, BTN_LEFT,
Packit 57e92c
			   EV_KEY, BTN_MIDDLE,
Packit 57e92c
			   EV_KEY, BTN_RIGHT,
Packit 57e92c
			   -1);
Packit 57e92c
Packit 57e92c
	fd = open(uinput_device_get_devnode(uidev), O_RDONLY|O_NONBLOCK);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
	rc = libevdev_new_from_fd(fd, &dev2);
Packit 57e92c
	ck_assert_msg(rc == 0, "Failed to create second device: %s", strerror(-rc));
Packit 57e92c
Packit 57e92c
	uinput_device_event(uidev, EV_REL, REL_X, 1);
Packit 57e92c
	uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
Packit 57e92c
Packit 57e92c
	for (int i = 0; i < 2; i++) {
Packit 57e92c
		rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev1);
Packit 57e92c
		ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
Packit 57e92c
Packit 57e92c
		rc = libevdev_next_event(dev2, LIBEVDEV_READ_FLAG_NORMAL, &ev2);
Packit 57e92c
		ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
Packit 57e92c
Packit 57e92c
		ck_assert_int_eq(ev1.type, ev2.type);
Packit 57e92c
		ck_assert_int_eq(ev1.code, ev2.code);
Packit 57e92c
		ck_assert_int_eq(ev1.value, ev2.value);
Packit 57e92c
	}
Packit 57e92c
Packit 57e92c
	/* revoke first device, expect it closed, second device still open */
Packit 57e92c
	dev_fd = libevdev_get_fd(dev);
Packit 57e92c
	ck_assert_int_ge(dev_fd, 0);
Packit 57e92c
	rc = ioctl(dev_fd, EVIOCREVOKE, NULL);
Packit 57e92c
	if (rc == -1 && errno == EINVAL) {
Packit 57e92c
		fprintf(stderr, "WARNING: skipping EVIOCREVOKE test, not suported by current kernel\n");
Packit 57e92c
		goto out;
Packit 57e92c
	}
Packit 57e92c
	ck_assert_msg(rc == 0, "Failed to revoke device: %s", strerror(errno));
Packit 57e92c
Packit 57e92c
	uinput_device_event(uidev, EV_REL, REL_X, 1);
Packit 57e92c
	uinput_device_event(uidev, EV_SYN, SYN_REPORT, 0);
Packit 57e92c
Packit 57e92c
	rc = libevdev_next_event(dev, LIBEVDEV_READ_FLAG_NORMAL, &ev1);
Packit 57e92c
	ck_assert_int_eq(rc, -ENODEV);
Packit 57e92c
Packit 57e92c
	rc = libevdev_next_event(dev2, LIBEVDEV_READ_FLAG_NORMAL, &ev2);
Packit 57e92c
	ck_assert_int_eq(rc, LIBEVDEV_READ_STATUS_SUCCESS);
Packit 57e92c
Packit 57e92c
out:
Packit 57e92c
	uinput_device_free(uidev);
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	libevdev_free(dev2);
Packit 57e92c
	close(fd);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_revoke_invalid)
Packit 57e92c
{
Packit 57e92c
	struct uinput_device* uidev;
Packit 57e92c
	struct libevdev *dev;
Packit 57e92c
	int rc;
Packit 57e92c
	int dev_fd;
Packit 57e92c
Packit 57e92c
	test_create_device(&uidev, &dev,
Packit 57e92c
			   EV_SYN, SYN_REPORT,
Packit 57e92c
			   EV_REL, REL_X,
Packit 57e92c
			   EV_REL, REL_Y,
Packit 57e92c
			   EV_REL, REL_WHEEL,
Packit 57e92c
			   EV_KEY, BTN_LEFT,
Packit 57e92c
			   EV_KEY, BTN_MIDDLE,
Packit 57e92c
			   EV_KEY, BTN_RIGHT,
Packit 57e92c
			   -1);
Packit 57e92c
Packit 57e92c
	dev_fd = libevdev_get_fd(dev);
Packit 57e92c
	ck_assert_int_ge(dev_fd, 0);
Packit 57e92c
	/* ioctl requires 0 as value */
Packit 57e92c
	rc = ioctl(dev_fd, EVIOCREVOKE, 1);
Packit 57e92c
	ck_assert_int_eq(rc, -1);
Packit 57e92c
	ck_assert_int_eq(errno, EINVAL);
Packit 57e92c
Packit 57e92c
	uinput_device_free(uidev);
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_revoke_fail_after)
Packit 57e92c
{
Packit 57e92c
	struct uinput_device* uidev;
Packit 57e92c
	struct libevdev *dev, *dev2 = NULL;
Packit 57e92c
	int rc, fd;
Packit 57e92c
Packit 57e92c
	test_create_device(&uidev, &dev,
Packit 57e92c
			   EV_SYN, SYN_REPORT,
Packit 57e92c
			   EV_REL, REL_X,
Packit 57e92c
			   EV_REL, REL_Y,
Packit 57e92c
			   EV_REL, REL_WHEEL,
Packit 57e92c
			   EV_KEY, BTN_LEFT,
Packit 57e92c
			   EV_KEY, BTN_MIDDLE,
Packit 57e92c
			   EV_KEY, BTN_RIGHT,
Packit 57e92c
			   -1);
Packit 57e92c
Packit 57e92c
	fd = open(uinput_device_get_devnode(uidev), O_RDONLY|O_NONBLOCK);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
Packit 57e92c
	rc = ioctl(fd, EVIOCREVOKE, NULL);
Packit 57e92c
	if (rc == -1 && errno == EINVAL) {
Packit 57e92c
		fprintf(stderr, "WARNING: skipping EVIOCREVOKE test, not suported by current kernel\n");
Packit 57e92c
		goto out;
Packit 57e92c
	}
Packit 57e92c
	ck_assert_msg(rc == 0, "Failed to revoke device: %s", strerror(errno));
Packit 57e92c
Packit 57e92c
	rc = libevdev_new_from_fd(fd, &dev2);
Packit 57e92c
	ck_assert_int_eq(rc, -ENODEV);
Packit 57e92c
Packit 57e92c
out:
Packit 57e92c
	uinput_device_free(uidev);
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	close(fd);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
TEST_SUITE_ROOT_PRIVILEGES(kernel)
Packit 57e92c
{
Packit 57e92c
	Suite *s = suite_create("kernel");
Packit 57e92c
Packit 57e92c
	TCase *tc = tcase_create("EVIOCREVOKE");
Packit 57e92c
	tcase_add_test(tc, test_revoke);
Packit 57e92c
	tcase_add_test(tc, test_revoke_invalid);
Packit 57e92c
	tcase_add_test(tc, test_revoke_fail_after);
Packit 57e92c
	suite_add_tcase(s, tc);
Packit 57e92c
Packit 57e92c
	return s;
Packit 57e92c
}