/* * Copyright (c) 2015 Cedric Hnyda <chnyda@suse.com> * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it would be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ /* * Create a virtual device (mouse), send events to /dev/uinput * and check that the events are well received in /dev/input/eventX */ #include <linux/input.h> #include "input_helper.h" #include "test.h" #include "safe_macros.h" #include "lapi/fcntl.h" #define NB_TEST 20 static void setup(void); static void send_events(void); static int verify_data(struct input_event *iev, int nb); static int check_events(void); static void cleanup(void); static int fd; static int fd2; char *TCID = "input01"; int main(int ac, char **av) { int lc; int pid; tst_parse_opts(ac, av, NULL, NULL); setup(); for (lc = 0; TEST_LOOPING(lc); ++lc) { pid = tst_fork(); switch (pid) { case 0: send_events(); exit(0); case -1: tst_brkm(TBROK | TERRNO, cleanup, "fork() failed"); default: if (check_events()) tst_resm(TFAIL, "Wrong data read from eventX"); else tst_resm(TPASS, "Data received from eventX"); break; } SAFE_WAITPID(NULL, pid, NULL, 0); } cleanup(); tst_exit(); } static void setup(void) { tst_require_root(); fd = open_uinput(); setup_mouse_events(fd); create_device(fd); fd2 = open_device(); } static void send_events(void) { int nb; for (nb = 0; nb < NB_TEST; ++nb) { send_rel_move(fd, 10, 1); usleep(1000); } } static int check_events(void) { int nb, rd; unsigned int i; struct input_event iev[64]; nb = 0; while (nb < NB_TEST * 3) { rd = read(fd2, iev, sizeof(iev)); if (rd < 0) tst_brkm(TBROK | TERRNO, cleanup, "read()"); if (rd == 0 || rd % sizeof(struct input_event)) { tst_resm(TINFO, "read() returned unexpected %i", rd); return 1; } for (i = 0; i < rd / sizeof(struct input_event); i++) { if (verify_data(&iev[i], nb++)) return 1; } } return 0; } static int verify_data(struct input_event *iev, int nb) { if (nb % 3 == 0) { if (iev->type != EV_REL) { tst_resm(TINFO, "%i: Unexpected event type %i expected %i", nb, iev->type, EV_REL); return 1; } if (iev->code != REL_X) return 1; if (iev->value != 10) return 1; return 0; } if (nb % 3 == 1) { if (iev->type != EV_REL) { tst_resm(TINFO, "%i: Unexpected event type %i expected %i", nb, iev->type, EV_REL); return 1; } if (iev->code != REL_Y) return 1; if (iev->value != 1) return 1; return 0; } if (nb % 3 == 2) { if (iev->type != EV_SYN) { tst_resm(TINFO, "%i: Unexpected event type %i expected %i", nb, iev->type, EV_SYN); return 1; } if (iev->code != 0) return 1; if (iev->value != 0) return 1; return 0; } return 1; } static void cleanup(void) { if (fd2 > 0 && close(fd2)) tst_resm(TWARN | TERRNO, "close(fd2)"); destroy_device(fd); }