2 * Copyright © 2013 Red Hat, Inc.
4 * Permission to use, copy, modify, distribute, and sell this software and its
5 * documentation for any purpose is hereby granted without fee, provided that
6 * the above copyright notice appear in all copies and that both that copyright
7 * notice and this permission notice appear in supporting documentation, and
8 * that the name of the copyright holders not be used in advertising or
9 * publicity pertaining to distribution of the software without specific,
10 * written prior permission. The copyright holders make no representations
11 * about the suitability of this software for any purpose. It is provided "as
12 * is" without express or implied warranty.
14 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
15 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
16 * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
17 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
18 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
19 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
24 #include <linux/input.h>
29 #include <libevdev/libevdev-uinput.h>
31 #include "test-common.h"
32 #define UINPUT_NODE "/dev/uinput"
34 START_TEST(test_uinput_create_device)
36 struct libevdev *dev, *dev2;
37 struct libevdev_uinput *uidev;
39 unsigned int type, code;
44 ck_assert(dev != NULL);
45 libevdev_set_name(dev, TEST_DEVICE_NAME);
46 libevdev_enable_event_type(dev, EV_SYN);
47 libevdev_enable_event_type(dev, EV_REL);
48 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
49 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
50 libevdev_enable_event_code(dev, EV_REL, REL_MAX, NULL);
52 rc = libevdev_uinput_create_from_device(dev, LIBEVDEV_UINPUT_OPEN_MANAGED, &uidev);
53 ck_assert_int_eq(rc, 0);
54 ck_assert(uidev != NULL);
56 uinput_fd = libevdev_uinput_get_fd(uidev);
57 ck_assert_int_gt(uinput_fd, -1);
59 devnode = libevdev_uinput_get_devnode(uidev);
60 ck_assert(devnode != NULL);
62 fd = open(devnode, O_RDONLY);
63 ck_assert_int_gt(fd, -1);
64 rc = libevdev_new_from_fd(fd, &dev2);
65 ck_assert_int_eq(rc, 0);
67 for (type = 0; type < EV_CNT; type++) {
68 int max = libevdev_event_type_get_max(type);
72 for (code = 0; code < (unsigned int)max; code++) {
73 ck_assert_int_eq(libevdev_has_event_code(dev, type, code),
74 libevdev_has_event_code(dev2, type, code));
80 libevdev_uinput_destroy(uidev);
83 /* uinput fd is managed, so make sure it did get closed */
84 ck_assert_int_eq(close(uinput_fd), -1);
85 ck_assert_int_eq(errno, EBADF);
90 START_TEST(test_uinput_create_device_invalid)
93 struct libevdev_uinput *uidev = NULL;
97 ck_assert(dev != NULL);
98 libevdev_set_name(dev, TEST_DEVICE_NAME);
99 libevdev_enable_event_type(dev, EV_SYN);
100 libevdev_enable_event_type(dev, EV_REL);
101 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
102 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
104 libevdev_set_log_function(test_logfunc_ignore_error, NULL);
105 rc = libevdev_uinput_create_from_device(dev, -1, &uidev);
106 ck_assert_int_eq(rc, -EBADF);
107 ck_assert(uidev == NULL);
108 libevdev_set_log_function(test_logfunc_abort_on_error, NULL);
114 START_TEST(test_uinput_create_device_from_fd)
116 struct libevdev *dev, *dev2;
117 struct libevdev_uinput *uidev;
119 unsigned int type, code;
123 dev = libevdev_new();
124 ck_assert(dev != NULL);
125 libevdev_set_name(dev, TEST_DEVICE_NAME);
126 libevdev_enable_event_type(dev, EV_SYN);
127 libevdev_enable_event_type(dev, EV_REL);
128 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
129 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
131 fd = open(UINPUT_NODE, O_RDWR);
132 ck_assert_int_gt(fd, -1);
134 rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
135 ck_assert_int_eq(rc, 0);
136 ck_assert(uidev != NULL);
138 ck_assert_int_eq(libevdev_uinput_get_fd(uidev), fd);
140 devnode = libevdev_uinput_get_devnode(uidev);
141 ck_assert(devnode != NULL);
143 fd2 = open(devnode, O_RDONLY);
144 ck_assert_int_gt(fd2, -1);
145 rc = libevdev_new_from_fd(fd2, &dev2);
146 ck_assert_int_eq(rc, 0);
148 for (type = 0; type < EV_CNT; type++) {
149 int max = libevdev_event_type_get_max(type);
153 for (code = 0; code < (unsigned int)max; code++) {
154 ck_assert_int_eq(libevdev_has_event_code(dev, type, code),
155 libevdev_has_event_code(dev2, type, code));
161 libevdev_uinput_destroy(uidev);
168 START_TEST(test_uinput_check_devnode_bsd)
170 struct libevdev *dev;
171 struct libevdev_uinput *uidev, *uidev2;
172 const char *devnode, *devnode2;
176 dev = libevdev_new();
177 ck_assert(dev != NULL);
178 libevdev_set_name(dev, TEST_DEVICE_NAME);
179 libevdev_enable_event_type(dev, EV_SYN);
180 libevdev_enable_event_type(dev, EV_REL);
181 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
182 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
184 fd = open(UINPUT_NODE, O_RDWR);
185 ck_assert_int_gt(fd, -1);
186 fd2 = open(UINPUT_NODE, O_RDWR);
187 ck_assert_int_gt(fd2, -1);
189 rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
190 ck_assert_int_eq(rc, 0);
192 /* create a second one */
193 libevdev_set_name(dev, TEST_DEVICE_NAME " 2");
194 rc = libevdev_uinput_create_from_device(dev, fd2, &uidev2);
195 ck_assert_int_eq(rc, 0);
197 devnode = libevdev_uinput_get_devnode(uidev);
198 ck_assert(devnode != NULL);
200 /* get syspath twice returns same pointer */
201 devnode2 = libevdev_uinput_get_devnode(uidev);
202 ck_assert(devnode == devnode2);
204 /* second dev has different devnode */
205 devnode2 = libevdev_uinput_get_devnode(uidev2);
206 ck_assert(strcmp(devnode, devnode2) != 0);
208 libevdev_uinput_destroy(uidev2);
209 libevdev_uinput_destroy(uidev);
218 START_TEST(test_uinput_check_syspath_bsd)
220 struct libevdev *dev;
221 struct libevdev_uinput *uidev;
226 dev = libevdev_new();
227 ck_assert(dev != NULL);
228 libevdev_set_name(dev, TEST_DEVICE_NAME);
229 libevdev_enable_event_type(dev, EV_SYN);
230 libevdev_enable_event_type(dev, EV_REL);
231 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
232 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
234 fd = open(UINPUT_NODE, O_RDWR);
235 ck_assert_int_gt(fd, -1);
237 rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
238 ck_assert_int_eq(rc, 0);
240 syspath = libevdev_uinput_get_syspath(uidev);
241 /* FreeBSD should always return NULL for libevdev_unput_get_syspath() */
242 ck_assert(syspath == NULL);
244 libevdev_uinput_destroy(uidev);
252 #else /* !__FreeBSD__ */
254 START_TEST(test_uinput_check_syspath_time)
256 struct libevdev *dev;
257 struct libevdev_uinput *uidev, *uidev2;
258 const char *syspath, *syspath2;
262 dev = libevdev_new();
263 ck_assert(dev != NULL);
264 libevdev_set_name(dev, TEST_DEVICE_NAME);
265 libevdev_enable_event_type(dev, EV_SYN);
266 libevdev_enable_event_type(dev, EV_REL);
267 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
268 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
270 fd = open(UINPUT_NODE, O_RDWR);
271 ck_assert_int_gt(fd, -1);
272 fd2 = open(UINPUT_NODE, O_RDWR);
273 ck_assert_int_gt(fd2, -1);
275 rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
276 ck_assert_int_eq(rc, 0);
278 /* sleep for 1.5 seconds. sysfs resolution is 1 second, so
279 creating both devices without delay means
280 libevdev_uinput_get_syspath can't actually differ between
281 them. By waiting, we get different ctime for uidev and uidev2,
282 and exercise that part of the code.
286 /* create a second one to test the syspath time filtering code */
287 rc = libevdev_uinput_create_from_device(dev, fd2, &uidev2);
288 ck_assert_int_eq(rc, 0);
290 syspath = libevdev_uinput_get_syspath(uidev);
291 ck_assert(syspath != NULL);
293 /* get syspath twice returns same pointer */
294 syspath2 = libevdev_uinput_get_syspath(uidev);
295 ck_assert(syspath == syspath2);
297 /* second dev has different syspath */
298 syspath2 = libevdev_uinput_get_syspath(uidev2);
299 ck_assert(strcmp(syspath, syspath2) != 0);
302 libevdev_uinput_destroy(uidev);
303 libevdev_uinput_destroy(uidev2);
310 START_TEST(test_uinput_check_syspath_name)
312 struct libevdev *dev;
313 struct libevdev_uinput *uidev, *uidev2;
314 const char *syspath, *syspath2;
318 dev = libevdev_new();
319 ck_assert(dev != NULL);
320 libevdev_set_name(dev, TEST_DEVICE_NAME);
321 libevdev_enable_event_type(dev, EV_SYN);
322 libevdev_enable_event_type(dev, EV_REL);
323 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
324 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
326 fd = open(UINPUT_NODE, O_RDWR);
327 ck_assert_int_gt(fd, -1);
328 fd2 = open(UINPUT_NODE, O_RDWR);
329 ck_assert_int_gt(fd2, -1);
331 rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
332 ck_assert_int_eq(rc, 0);
334 /* create a second one to stress the syspath filtering code */
335 libevdev_set_name(dev, TEST_DEVICE_NAME " 2");
336 rc = libevdev_uinput_create_from_device(dev, fd2, &uidev2);
337 ck_assert_int_eq(rc, 0);
339 syspath = libevdev_uinput_get_syspath(uidev);
340 ck_assert(syspath != NULL);
342 /* get syspath twice returns same pointer */
343 syspath2 = libevdev_uinput_get_syspath(uidev);
344 ck_assert(syspath == syspath2);
346 /* second dev has different syspath */
347 syspath2 = libevdev_uinput_get_syspath(uidev2);
348 ck_assert(strcmp(syspath, syspath2) != 0);
351 libevdev_uinput_destroy(uidev);
352 libevdev_uinput_destroy(uidev2);
359 #endif /* __FreeBSD __ */
361 START_TEST(test_uinput_events)
363 struct libevdev *dev;
364 struct libevdev_uinput *uidev;
369 const int nevents = 5;
370 struct input_event events[] = { {{0, 0}, EV_REL, REL_X, 1},
371 {{0, 0}, EV_REL, REL_Y, -1},
372 {{0, 0}, EV_SYN, SYN_REPORT, 0},
373 {{0, 0}, EV_KEY, BTN_LEFT, 1},
374 {{0, 0}, EV_SYN, SYN_REPORT, 0}};
375 struct input_event events_read[nevents];
377 dev = libevdev_new();
378 ck_assert(dev != NULL);
379 libevdev_set_name(dev, TEST_DEVICE_NAME);
380 libevdev_enable_event_type(dev, EV_SYN);
381 libevdev_enable_event_type(dev, EV_REL);
382 libevdev_enable_event_type(dev, EV_KEY);
383 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
384 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
385 libevdev_enable_event_code(dev, EV_KEY, BTN_LEFT, NULL);
387 fd = open(UINPUT_NODE, O_RDWR);
388 ck_assert_int_gt(fd, -1);
390 rc = libevdev_uinput_create_from_device(dev, fd, &uidev);
391 ck_assert_int_eq(rc, 0);
392 ck_assert(uidev != NULL);
394 devnode = libevdev_uinput_get_devnode(uidev);
395 ck_assert(devnode != NULL);
397 fd2 = open(devnode, O_RDONLY);
399 for (i = 0; i < nevents; i++)
400 libevdev_uinput_write_event(uidev, events[i].type, events[i].code, events[i].value);
402 rc = read(fd2, events_read, sizeof(events_read));
403 ck_assert_int_eq(rc, sizeof(events_read));
405 for (i = 0; i < nevents; i++) {
406 ck_assert_int_eq(events[i].type, events_read[i].type);
407 ck_assert_int_eq(events[i].code, events_read[i].code);
408 ck_assert_int_eq(events[i].value, events_read[i].value);
412 libevdev_uinput_destroy(uidev);
418 START_TEST(test_uinput_properties)
420 struct libevdev *dev, *dev2;
421 struct libevdev_uinput *uidev;
426 dev = libevdev_new();
427 ck_assert(dev != NULL);
428 libevdev_set_name(dev, TEST_DEVICE_NAME);
429 libevdev_enable_event_type(dev, EV_SYN);
430 libevdev_enable_event_type(dev, EV_REL);
431 libevdev_enable_event_type(dev, EV_KEY);
432 libevdev_enable_event_code(dev, EV_REL, REL_X, NULL);
433 libevdev_enable_event_code(dev, EV_REL, REL_Y, NULL);
434 libevdev_enable_event_code(dev, EV_KEY, BTN_LEFT, NULL);
435 libevdev_enable_property(dev, INPUT_PROP_BUTTONPAD);
436 libevdev_enable_property(dev, INPUT_PROP_MAX);
438 rc = libevdev_uinput_create_from_device(dev, LIBEVDEV_UINPUT_OPEN_MANAGED, &uidev);
439 ck_assert_int_eq(rc, 0);
440 ck_assert(uidev != NULL);
442 devnode = libevdev_uinput_get_devnode(uidev);
443 ck_assert(devnode != NULL);
445 fd = open(devnode, O_RDONLY);
446 ck_assert_int_gt(fd, -1);
447 rc = libevdev_new_from_fd(fd, &dev2);
448 ck_assert_int_eq(rc, 0);
450 ck_assert(libevdev_has_property(dev2, INPUT_PROP_BUTTONPAD));
451 ck_assert(libevdev_has_property(dev2, INPUT_PROP_MAX));
455 libevdev_uinput_destroy(uidev);
460 TEST_SUITE_ROOT_PRIVILEGES(uinput_suite)
462 Suite *s = suite_create("libevdev uinput device tests");
464 add_test(s, test_uinput_create_device);
465 add_test(s, test_uinput_create_device_invalid);
466 add_test(s, test_uinput_create_device_from_fd);
468 add_test(s, test_uinput_check_devnode_bsd);
469 add_test(s, test_uinput_check_syspath_bsd);
471 add_test(s, test_uinput_check_syspath_time);
472 add_test(s, test_uinput_check_syspath_name);
475 add_test(s, test_uinput_events);
477 add_test(s, test_uinput_properties);