Blame test/test-uinput.c

Packit 57e92c
/*
Packit 57e92c
 * Copyright © 2013 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 <linux/input.h>
Packit 57e92c
#include <errno.h>
Packit 57e92c
#include <unistd.h>
Packit 57e92c
#include <stdlib.h>
Packit 57e92c
#include <fcntl.h>
Packit 57e92c
#include <libevdev/libevdev-uinput.h>
Packit 57e92c
Packit 57e92c
#include "test-common.h"
Packit 57e92c
#define UINPUT_NODE "/dev/uinput"
Packit 57e92c
Packit 57e92c
START_TEST(test_uinput_create_device)
Packit 57e92c
{
Packit 57e92c
	struct libevdev *dev, *dev2;
Packit 57e92c
	struct libevdev_uinput *uidev;
Packit 57e92c
	int fd, uinput_fd;
Packit 57e92c
	unsigned int type, code;
Packit 57e92c
	int rc;
Packit 57e92c
	const char *devnode;
Packit 57e92c
Packit 57e92c
	dev = libevdev_new();
Packit 57e92c
	ck_assert(dev != NULL);
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_SYN);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_REL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_MAX, NULL);
Packit 57e92c
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, LIBEVDEV_UINPUT_OPEN_MANAGED, &uidev);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
	ck_assert(uidev != NULL);
Packit 57e92c
Packit 57e92c
	uinput_fd = libevdev_uinput_get_fd(uidev);
Packit 57e92c
	ck_assert_int_gt(uinput_fd, -1);
Packit 57e92c
Packit 57e92c
	devnode = libevdev_uinput_get_devnode(uidev);
Packit 57e92c
	ck_assert(devnode != NULL);
Packit 57e92c
Packit 57e92c
	fd = open(devnode, O_RDONLY);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
	rc = libevdev_new_from_fd(fd, &dev2);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
Packit 57e92c
	for (type = 0; type < EV_CNT; type++) {
Packit 57e92c
		int max = libevdev_event_type_get_max(type);
Packit 57e92c
		if (max == -1)
Packit 57e92c
			continue;
Packit 57e92c
Packit 57e92c
		for (code = 0; code < max; code++) {
Packit 57e92c
			ck_assert_int_eq(libevdev_has_event_code(dev, type, code),
Packit 57e92c
					 libevdev_has_event_code(dev2, type, code));
Packit 57e92c
		}
Packit 57e92c
	}
Packit 57e92c
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	libevdev_free(dev2);
Packit 57e92c
	libevdev_uinput_destroy(uidev);
Packit 57e92c
	close(fd);
Packit 57e92c
Packit 57e92c
	/* uinput fd is managed, so make sure it did get closed */
Packit 57e92c
	ck_assert_int_eq(close(uinput_fd), -1);
Packit 57e92c
	ck_assert_int_eq(errno, EBADF);
Packit 57e92c
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_uinput_create_device_invalid)
Packit 57e92c
{
Packit 57e92c
	struct libevdev *dev;
Packit 57e92c
	struct libevdev_uinput *uidev = NULL;
Packit 57e92c
	int rc;
Packit 57e92c
Packit 57e92c
	dev = libevdev_new();
Packit 57e92c
	ck_assert(dev != NULL);
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_SYN);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_REL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
Packit 57e92c
Packit 57e92c
	libevdev_set_log_function(test_logfunc_ignore_error, NULL);
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, -1, &uidev);
Packit 57e92c
	ck_assert_int_eq(rc, -EBADF);
Packit 57e92c
	ck_assert(uidev == NULL);
Packit 57e92c
	libevdev_set_log_function(test_logfunc_abort_on_error, NULL);
Packit 57e92c
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_uinput_create_device_from_fd)
Packit 57e92c
{
Packit 57e92c
	struct libevdev *dev, *dev2;
Packit 57e92c
	struct libevdev_uinput *uidev;
Packit 57e92c
	int fd, fd2;
Packit 57e92c
	unsigned int type, code;
Packit 57e92c
	int rc;
Packit 57e92c
	const char *devnode;
Packit 57e92c
Packit 57e92c
	dev = libevdev_new();
Packit 57e92c
	ck_assert(dev != NULL);
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_SYN);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_REL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
Packit 57e92c
Packit 57e92c
	fd = open(UINPUT_NODE, O_RDWR);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
	ck_assert(uidev != NULL);
Packit 57e92c
Packit 57e92c
	ck_assert_int_eq(libevdev_uinput_get_fd(uidev), fd);
Packit 57e92c
Packit 57e92c
	devnode = libevdev_uinput_get_devnode(uidev);
Packit 57e92c
	ck_assert(devnode != NULL);
Packit 57e92c
Packit 57e92c
	fd2 = open(devnode, O_RDONLY);
Packit 57e92c
	ck_assert_int_gt(fd2, -1);
Packit 57e92c
	rc = libevdev_new_from_fd(fd2, &dev2);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
Packit 57e92c
	for (type = 0; type < EV_CNT; type++) {
Packit 57e92c
		int max = libevdev_event_type_get_max(type);
Packit 57e92c
		if (max == -1)
Packit 57e92c
			continue;
Packit 57e92c
Packit 57e92c
		for (code = 0; code < max; code++) {
Packit 57e92c
			ck_assert_int_eq(libevdev_has_event_code(dev, type, code),
Packit 57e92c
					 libevdev_has_event_code(dev2, type, code));
Packit 57e92c
		}
Packit 57e92c
	}
Packit 57e92c
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	libevdev_free(dev2);
Packit 57e92c
	libevdev_uinput_destroy(uidev);
Packit 57e92c
	close(fd);
Packit 57e92c
	close(fd2);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_uinput_check_syspath_time)
Packit 57e92c
{
Packit 57e92c
	struct libevdev *dev;
Packit 57e92c
	struct libevdev_uinput *uidev, *uidev2;
Packit 57e92c
	const char *syspath, *syspath2;
Packit 57e92c
	int fd, fd2;
Packit 57e92c
	int rc;
Packit 57e92c
Packit 57e92c
	dev = libevdev_new();
Packit 57e92c
	ck_assert(dev != NULL);
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_SYN);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_REL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
Packit 57e92c
Packit 57e92c
	fd = open(UINPUT_NODE, O_RDWR);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
	fd2 = open(UINPUT_NODE, O_RDWR);
Packit 57e92c
	ck_assert_int_gt(fd2, -1);
Packit 57e92c
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
Packit 57e92c
	/* sleep for 1.5 seconds. sysfs resolution is 1 second, so
Packit 57e92c
	   creating both devices without delay means
Packit 57e92c
	   libevdev_uinput_get_syspath can't actually differ between
Packit 57e92c
	   them. By waiting, we get different ctime for uidev and uidev2,
Packit 57e92c
	   and exercise that part of the code.
Packit 57e92c
	 */
Packit 57e92c
	usleep(1500000);
Packit 57e92c
Packit 57e92c
	/* create a second one to test the syspath time filtering code */
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, fd2, &uidev2);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
Packit 57e92c
	syspath = libevdev_uinput_get_syspath(uidev);
Packit 57e92c
	ck_assert(syspath != NULL);
Packit 57e92c
Packit 57e92c
	/* get syspath twice returns same pointer */
Packit 57e92c
	syspath2 = libevdev_uinput_get_syspath(uidev);
Packit 57e92c
	ck_assert(syspath == syspath2);
Packit 57e92c
Packit 57e92c
	/* second dev has different syspath */
Packit 57e92c
	syspath2 = libevdev_uinput_get_syspath(uidev2);
Packit 57e92c
	ck_assert(strcmp(syspath, syspath2) != 0);
Packit 57e92c
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	libevdev_uinput_destroy(uidev);
Packit 57e92c
	libevdev_uinput_destroy(uidev2);
Packit 57e92c
Packit 57e92c
	close(fd);
Packit 57e92c
	close(fd2);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_uinput_check_syspath_name)
Packit 57e92c
{
Packit 57e92c
	struct libevdev *dev;
Packit 57e92c
	struct libevdev_uinput *uidev, *uidev2;
Packit 57e92c
	const char *syspath, *syspath2;
Packit 57e92c
	int fd, fd2;
Packit 57e92c
	int rc;
Packit 57e92c
Packit 57e92c
	dev = libevdev_new();
Packit 57e92c
	ck_assert(dev != NULL);
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_SYN);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_REL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
Packit 57e92c
Packit 57e92c
	fd = open(UINPUT_NODE, O_RDWR);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
	fd2 = open(UINPUT_NODE, O_RDWR);
Packit 57e92c
	ck_assert_int_gt(fd2, -1);
Packit 57e92c
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
Packit 57e92c
	/* create a second one to stress the syspath filtering code */
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME " 2");
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, fd2, &uidev2);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
Packit 57e92c
	syspath = libevdev_uinput_get_syspath(uidev);
Packit 57e92c
	ck_assert(syspath != NULL);
Packit 57e92c
Packit 57e92c
	/* get syspath twice returns same pointer */
Packit 57e92c
	syspath2 = libevdev_uinput_get_syspath(uidev);
Packit 57e92c
	ck_assert(syspath == syspath2);
Packit 57e92c
Packit 57e92c
	/* second dev has different syspath */
Packit 57e92c
	syspath2 = libevdev_uinput_get_syspath(uidev2);
Packit 57e92c
	ck_assert(strcmp(syspath, syspath2) != 0);
Packit 57e92c
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	libevdev_uinput_destroy(uidev);
Packit 57e92c
	libevdev_uinput_destroy(uidev2);
Packit 57e92c
Packit 57e92c
	close(fd);
Packit 57e92c
	close(fd2);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_uinput_events)
Packit 57e92c
{
Packit 57e92c
	struct libevdev *dev;
Packit 57e92c
	struct libevdev_uinput *uidev;
Packit 57e92c
	int fd, fd2;
Packit 57e92c
	int rc;
Packit 57e92c
	const char *devnode;
Packit 57e92c
	int i;
Packit 57e92c
	const int nevents = 5;
Packit 57e92c
	struct input_event events[] = { {{0, 0}, EV_REL, REL_X, 1},
Packit 57e92c
					{{0, 0}, EV_REL, REL_Y, -1},
Packit 57e92c
					{{0, 0}, EV_SYN, SYN_REPORT, 0},
Packit 57e92c
					{{0, 0}, EV_KEY, BTN_LEFT, 1},
Packit 57e92c
					{{0, 0}, EV_SYN, SYN_REPORT, 0}};
Packit 57e92c
	struct input_event events_read[nevents];
Packit 57e92c
Packit 57e92c
	dev = libevdev_new();
Packit 57e92c
	ck_assert(dev != NULL);
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_SYN);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_REL);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_KEY);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_KEY, BTN_LEFT, NULL);
Packit 57e92c
Packit 57e92c
	fd = open(UINPUT_NODE, O_RDWR);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
	ck_assert(uidev != NULL);
Packit 57e92c
Packit 57e92c
	devnode = libevdev_uinput_get_devnode(uidev);
Packit 57e92c
	ck_assert(devnode != NULL);
Packit 57e92c
Packit 57e92c
	fd2 = open(devnode, O_RDONLY);
Packit 57e92c
Packit 57e92c
	for (i = 0; i < nevents; i++)
Packit 57e92c
		libevdev_uinput_write_event(uidev, events[i].type, events[i].code, events[i].value);
Packit 57e92c
Packit 57e92c
	rc = read(fd2, events_read, sizeof(events_read));
Packit 57e92c
	ck_assert_int_eq(rc, sizeof(events_read));
Packit 57e92c
Packit 57e92c
	for (i = 0; i < nevents; i++) {
Packit 57e92c
		ck_assert_int_eq(events[i].type, events_read[i].type);
Packit 57e92c
		ck_assert_int_eq(events[i].code, events_read[i].code);
Packit 57e92c
		ck_assert_int_eq(events[i].value, events_read[i].value);
Packit 57e92c
	}
Packit 57e92c
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	libevdev_uinput_destroy(uidev);
Packit 57e92c
	close(fd);
Packit 57e92c
	close(fd2);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
START_TEST(test_uinput_properties)
Packit 57e92c
{
Packit 57e92c
	struct libevdev *dev, *dev2;
Packit 57e92c
	struct libevdev_uinput *uidev;
Packit 57e92c
	int fd;
Packit 57e92c
	int rc;
Packit 57e92c
	const char *devnode;
Packit 57e92c
Packit 57e92c
	dev = libevdev_new();
Packit 57e92c
	ck_assert(dev != NULL);
Packit 57e92c
	libevdev_set_name(dev, TEST_DEVICE_NAME);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_SYN);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_REL);
Packit 57e92c
	libevdev_enable_event_type(dev, EV_KEY);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
Packit 57e92c
	libevdev_enable_event_code(dev, EV_KEY, BTN_LEFT, NULL);
Packit 57e92c
	libevdev_enable_property(dev, INPUT_PROP_BUTTONPAD);
Packit 57e92c
	libevdev_enable_property(dev, INPUT_PROP_MAX);
Packit 57e92c
Packit 57e92c
	rc = libevdev_uinput_create_from_device(dev, LIBEVDEV_UINPUT_OPEN_MANAGED, &uidev);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
	ck_assert(uidev != NULL);
Packit 57e92c
Packit 57e92c
	devnode = libevdev_uinput_get_devnode(uidev);
Packit 57e92c
	ck_assert(devnode != NULL);
Packit 57e92c
Packit 57e92c
	fd = open(devnode, O_RDONLY);
Packit 57e92c
	ck_assert_int_gt(fd, -1);
Packit 57e92c
	rc = libevdev_new_from_fd(fd, &dev2);
Packit 57e92c
	ck_assert_int_eq(rc, 0);
Packit 57e92c
Packit 57e92c
	ck_assert(libevdev_has_property(dev2, INPUT_PROP_BUTTONPAD));
Packit 57e92c
	ck_assert(libevdev_has_property(dev2, INPUT_PROP_MAX));
Packit 57e92c
Packit 57e92c
	libevdev_free(dev);
Packit 57e92c
	libevdev_free(dev2);
Packit 57e92c
	libevdev_uinput_destroy(uidev);
Packit 57e92c
	close(fd);
Packit 57e92c
}
Packit 57e92c
END_TEST
Packit 57e92c
Packit 57e92c
TEST_SUITE_ROOT_PRIVILEGES(uinput_suite)
Packit 57e92c
{
Packit 57e92c
	Suite *s = suite_create("libevdev uinput device tests");
Packit 57e92c
Packit 57e92c
	TCase *tc = tcase_create("device creation");
Packit 57e92c
	tcase_add_test(tc, test_uinput_create_device);
Packit 57e92c
	tcase_add_test(tc, test_uinput_create_device_invalid);
Packit 57e92c
	tcase_add_test(tc, test_uinput_create_device_from_fd);
Packit 57e92c
	tcase_add_test(tc, test_uinput_check_syspath_time);
Packit 57e92c
	tcase_add_test(tc, test_uinput_check_syspath_name);
Packit 57e92c
	suite_add_tcase(s, tc);
Packit 57e92c
Packit 57e92c
	tc = tcase_create("device events");
Packit 57e92c
	tcase_add_test(tc, test_uinput_events);
Packit 57e92c
	suite_add_tcase(s, tc);
Packit 57e92c
Packit 57e92c
	tc = tcase_create("device properties");
Packit 57e92c
	tcase_add_test(tc, test_uinput_properties);
Packit 57e92c
	suite_add_tcase(s, tc);
Packit 57e92c
Packit 57e92c
	return s;
Packit 57e92c
}