15148fa52SDavid Herrmann /* 25148fa52SDavid Herrmann * UHID Example 35148fa52SDavid Herrmann * 4*f5e4e7fdSDavid Herrmann * Copyright (c) 2012-2013 David Herrmann <dh.herrmann@gmail.com> 55148fa52SDavid Herrmann * 65148fa52SDavid Herrmann * The code may be used by anyone for any purpose, 75148fa52SDavid Herrmann * and can serve as a starting point for developing 85148fa52SDavid Herrmann * applications using uhid. 95148fa52SDavid Herrmann */ 105148fa52SDavid Herrmann 11*f5e4e7fdSDavid Herrmann /* 12*f5e4e7fdSDavid Herrmann * UHID Example 135148fa52SDavid Herrmann * This example emulates a basic 3 buttons mouse with wheel over UHID. Run this 145148fa52SDavid Herrmann * program as root and then use the following keys to control the mouse: 155148fa52SDavid Herrmann * q: Quit the application 165148fa52SDavid Herrmann * 1: Toggle left button (down, up, ...) 175148fa52SDavid Herrmann * 2: Toggle right button 185148fa52SDavid Herrmann * 3: Toggle middle button 195148fa52SDavid Herrmann * a: Move mouse left 205148fa52SDavid Herrmann * d: Move mouse right 215148fa52SDavid Herrmann * w: Move mouse up 225148fa52SDavid Herrmann * s: Move mouse down 235148fa52SDavid Herrmann * r: Move wheel up 245148fa52SDavid Herrmann * f: Move wheel down 255148fa52SDavid Herrmann * 26*f5e4e7fdSDavid Herrmann * Additionally to 3 button mouse, 3 keyboard LEDs are also supported (LED_NUML, 27*f5e4e7fdSDavid Herrmann * LED_CAPSL and LED_SCROLLL). The device doesn't generate any related keyboard 28*f5e4e7fdSDavid Herrmann * events, though. You need to manually write the EV_LED/LED_XY/1 activation 29*f5e4e7fdSDavid Herrmann * input event to the evdev device to see it being sent to this device. 30*f5e4e7fdSDavid Herrmann * 315148fa52SDavid Herrmann * If uhid is not available as /dev/uhid, then you can pass a different path as 325148fa52SDavid Herrmann * first argument. 335148fa52SDavid Herrmann * If <linux/uhid.h> is not installed in /usr, then compile this with: 345148fa52SDavid Herrmann * gcc -o ./uhid_test -Wall -I./include ./samples/uhid/uhid-example.c 355148fa52SDavid Herrmann * And ignore the warning about kernel headers. However, it is recommended to 365148fa52SDavid Herrmann * use the installed uhid.h if available. 375148fa52SDavid Herrmann */ 385148fa52SDavid Herrmann 395148fa52SDavid Herrmann #include <errno.h> 405148fa52SDavid Herrmann #include <fcntl.h> 415148fa52SDavid Herrmann #include <poll.h> 425148fa52SDavid Herrmann #include <stdbool.h> 435148fa52SDavid Herrmann #include <stdio.h> 445148fa52SDavid Herrmann #include <stdlib.h> 455148fa52SDavid Herrmann #include <string.h> 465148fa52SDavid Herrmann #include <termios.h> 475148fa52SDavid Herrmann #include <unistd.h> 485148fa52SDavid Herrmann #include <linux/uhid.h> 495148fa52SDavid Herrmann 50*f5e4e7fdSDavid Herrmann /* 51*f5e4e7fdSDavid Herrmann * HID Report Desciptor 52*f5e4e7fdSDavid Herrmann * We emulate a basic 3 button mouse with wheel and 3 keyboard LEDs. This is 53*f5e4e7fdSDavid Herrmann * the report-descriptor as the kernel will parse it: 545148fa52SDavid Herrmann * 55*f5e4e7fdSDavid Herrmann * INPUT(1)[INPUT] 565148fa52SDavid Herrmann * Field(0) 575148fa52SDavid Herrmann * Physical(GenericDesktop.Pointer) 585148fa52SDavid Herrmann * Application(GenericDesktop.Mouse) 595148fa52SDavid Herrmann * Usage(3) 605148fa52SDavid Herrmann * Button.0001 615148fa52SDavid Herrmann * Button.0002 625148fa52SDavid Herrmann * Button.0003 635148fa52SDavid Herrmann * Logical Minimum(0) 645148fa52SDavid Herrmann * Logical Maximum(1) 655148fa52SDavid Herrmann * Report Size(1) 665148fa52SDavid Herrmann * Report Count(3) 675148fa52SDavid Herrmann * Report Offset(0) 685148fa52SDavid Herrmann * Flags( Variable Absolute ) 695148fa52SDavid Herrmann * Field(1) 705148fa52SDavid Herrmann * Physical(GenericDesktop.Pointer) 715148fa52SDavid Herrmann * Application(GenericDesktop.Mouse) 725148fa52SDavid Herrmann * Usage(3) 735148fa52SDavid Herrmann * GenericDesktop.X 745148fa52SDavid Herrmann * GenericDesktop.Y 755148fa52SDavid Herrmann * GenericDesktop.Wheel 765148fa52SDavid Herrmann * Logical Minimum(-128) 775148fa52SDavid Herrmann * Logical Maximum(127) 785148fa52SDavid Herrmann * Report Size(8) 795148fa52SDavid Herrmann * Report Count(3) 805148fa52SDavid Herrmann * Report Offset(8) 815148fa52SDavid Herrmann * Flags( Variable Relative ) 82*f5e4e7fdSDavid Herrmann * OUTPUT(2)[OUTPUT] 83*f5e4e7fdSDavid Herrmann * Field(0) 84*f5e4e7fdSDavid Herrmann * Application(GenericDesktop.Keyboard) 85*f5e4e7fdSDavid Herrmann * Usage(3) 86*f5e4e7fdSDavid Herrmann * LED.NumLock 87*f5e4e7fdSDavid Herrmann * LED.CapsLock 88*f5e4e7fdSDavid Herrmann * LED.ScrollLock 89*f5e4e7fdSDavid Herrmann * Logical Minimum(0) 90*f5e4e7fdSDavid Herrmann * Logical Maximum(1) 91*f5e4e7fdSDavid Herrmann * Report Size(1) 92*f5e4e7fdSDavid Herrmann * Report Count(3) 93*f5e4e7fdSDavid Herrmann * Report Offset(0) 94*f5e4e7fdSDavid Herrmann * Flags( Variable Absolute ) 955148fa52SDavid Herrmann * 965148fa52SDavid Herrmann * This is the mapping that we expect: 975148fa52SDavid Herrmann * Button.0001 ---> Key.LeftBtn 985148fa52SDavid Herrmann * Button.0002 ---> Key.RightBtn 995148fa52SDavid Herrmann * Button.0003 ---> Key.MiddleBtn 1005148fa52SDavid Herrmann * GenericDesktop.X ---> Relative.X 1015148fa52SDavid Herrmann * GenericDesktop.Y ---> Relative.Y 1025148fa52SDavid Herrmann * GenericDesktop.Wheel ---> Relative.Wheel 103*f5e4e7fdSDavid Herrmann * LED.NumLock ---> LED.NumLock 104*f5e4e7fdSDavid Herrmann * LED.CapsLock ---> LED.CapsLock 105*f5e4e7fdSDavid Herrmann * LED.ScrollLock ---> LED.ScrollLock 1065148fa52SDavid Herrmann * 1075148fa52SDavid Herrmann * This information can be verified by reading /sys/kernel/debug/hid/<dev>/rdesc 1085148fa52SDavid Herrmann * This file should print the same information as showed above. 1095148fa52SDavid Herrmann */ 1105148fa52SDavid Herrmann 1115148fa52SDavid Herrmann static unsigned char rdesc[] = { 112*f5e4e7fdSDavid Herrmann 0x05, 0x01, /* USAGE_PAGE (Generic Desktop) */ 113*f5e4e7fdSDavid Herrmann 0x09, 0x02, /* USAGE (Mouse) */ 114*f5e4e7fdSDavid Herrmann 0xa1, 0x01, /* COLLECTION (Application) */ 115*f5e4e7fdSDavid Herrmann 0x09, 0x01, /* USAGE (Pointer) */ 116*f5e4e7fdSDavid Herrmann 0xa1, 0x00, /* COLLECTION (Physical) */ 117*f5e4e7fdSDavid Herrmann 0x85, 0x01, /* REPORT_ID (1) */ 118*f5e4e7fdSDavid Herrmann 0x05, 0x09, /* USAGE_PAGE (Button) */ 119*f5e4e7fdSDavid Herrmann 0x19, 0x01, /* USAGE_MINIMUM (Button 1) */ 120*f5e4e7fdSDavid Herrmann 0x29, 0x03, /* USAGE_MAXIMUM (Button 3) */ 121*f5e4e7fdSDavid Herrmann 0x15, 0x00, /* LOGICAL_MINIMUM (0) */ 122*f5e4e7fdSDavid Herrmann 0x25, 0x01, /* LOGICAL_MAXIMUM (1) */ 123*f5e4e7fdSDavid Herrmann 0x95, 0x03, /* REPORT_COUNT (3) */ 124*f5e4e7fdSDavid Herrmann 0x75, 0x01, /* REPORT_SIZE (1) */ 125*f5e4e7fdSDavid Herrmann 0x81, 0x02, /* INPUT (Data,Var,Abs) */ 126*f5e4e7fdSDavid Herrmann 0x95, 0x01, /* REPORT_COUNT (1) */ 127*f5e4e7fdSDavid Herrmann 0x75, 0x05, /* REPORT_SIZE (5) */ 128*f5e4e7fdSDavid Herrmann 0x81, 0x01, /* INPUT (Cnst,Var,Abs) */ 129*f5e4e7fdSDavid Herrmann 0x05, 0x01, /* USAGE_PAGE (Generic Desktop) */ 130*f5e4e7fdSDavid Herrmann 0x09, 0x30, /* USAGE (X) */ 131*f5e4e7fdSDavid Herrmann 0x09, 0x31, /* USAGE (Y) */ 132*f5e4e7fdSDavid Herrmann 0x09, 0x38, /* USAGE (WHEEL) */ 133*f5e4e7fdSDavid Herrmann 0x15, 0x81, /* LOGICAL_MINIMUM (-127) */ 134*f5e4e7fdSDavid Herrmann 0x25, 0x7f, /* LOGICAL_MAXIMUM (127) */ 135*f5e4e7fdSDavid Herrmann 0x75, 0x08, /* REPORT_SIZE (8) */ 136*f5e4e7fdSDavid Herrmann 0x95, 0x03, /* REPORT_COUNT (3) */ 137*f5e4e7fdSDavid Herrmann 0x81, 0x06, /* INPUT (Data,Var,Rel) */ 138*f5e4e7fdSDavid Herrmann 0xc0, /* END_COLLECTION */ 139*f5e4e7fdSDavid Herrmann 0xc0, /* END_COLLECTION */ 140*f5e4e7fdSDavid Herrmann 0x05, 0x01, /* USAGE_PAGE (Generic Desktop) */ 141*f5e4e7fdSDavid Herrmann 0x09, 0x06, /* USAGE (Keyboard) */ 142*f5e4e7fdSDavid Herrmann 0xa1, 0x01, /* COLLECTION (Application) */ 143*f5e4e7fdSDavid Herrmann 0x85, 0x02, /* REPORT_ID (2) */ 144*f5e4e7fdSDavid Herrmann 0x05, 0x08, /* USAGE_PAGE (Led) */ 145*f5e4e7fdSDavid Herrmann 0x19, 0x01, /* USAGE_MINIMUM (1) */ 146*f5e4e7fdSDavid Herrmann 0x29, 0x03, /* USAGE_MAXIMUM (3) */ 147*f5e4e7fdSDavid Herrmann 0x15, 0x00, /* LOGICAL_MINIMUM (0) */ 148*f5e4e7fdSDavid Herrmann 0x25, 0x01, /* LOGICAL_MAXIMUM (1) */ 149*f5e4e7fdSDavid Herrmann 0x95, 0x03, /* REPORT_COUNT (3) */ 150*f5e4e7fdSDavid Herrmann 0x75, 0x01, /* REPORT_SIZE (1) */ 151*f5e4e7fdSDavid Herrmann 0x91, 0x02, /* Output (Data,Var,Abs) */ 152*f5e4e7fdSDavid Herrmann 0x95, 0x01, /* REPORT_COUNT (1) */ 153*f5e4e7fdSDavid Herrmann 0x75, 0x05, /* REPORT_SIZE (5) */ 154*f5e4e7fdSDavid Herrmann 0x91, 0x01, /* Output (Cnst,Var,Abs) */ 155*f5e4e7fdSDavid Herrmann 0xc0, /* END_COLLECTION */ 1565148fa52SDavid Herrmann }; 1575148fa52SDavid Herrmann 1585148fa52SDavid Herrmann static int uhid_write(int fd, const struct uhid_event *ev) 1595148fa52SDavid Herrmann { 1605148fa52SDavid Herrmann ssize_t ret; 1615148fa52SDavid Herrmann 1625148fa52SDavid Herrmann ret = write(fd, ev, sizeof(*ev)); 1635148fa52SDavid Herrmann if (ret < 0) { 1645148fa52SDavid Herrmann fprintf(stderr, "Cannot write to uhid: %m\n"); 1655148fa52SDavid Herrmann return -errno; 1665148fa52SDavid Herrmann } else if (ret != sizeof(*ev)) { 1675148fa52SDavid Herrmann fprintf(stderr, "Wrong size written to uhid: %ld != %lu\n", 1685148fa52SDavid Herrmann ret, sizeof(ev)); 1695148fa52SDavid Herrmann return -EFAULT; 1705148fa52SDavid Herrmann } else { 1715148fa52SDavid Herrmann return 0; 1725148fa52SDavid Herrmann } 1735148fa52SDavid Herrmann } 1745148fa52SDavid Herrmann 1755148fa52SDavid Herrmann static int create(int fd) 1765148fa52SDavid Herrmann { 1775148fa52SDavid Herrmann struct uhid_event ev; 1785148fa52SDavid Herrmann 1795148fa52SDavid Herrmann memset(&ev, 0, sizeof(ev)); 1805148fa52SDavid Herrmann ev.type = UHID_CREATE; 1815148fa52SDavid Herrmann strcpy((char*)ev.u.create.name, "test-uhid-device"); 1825148fa52SDavid Herrmann ev.u.create.rd_data = rdesc; 1835148fa52SDavid Herrmann ev.u.create.rd_size = sizeof(rdesc); 1845148fa52SDavid Herrmann ev.u.create.bus = BUS_USB; 1855148fa52SDavid Herrmann ev.u.create.vendor = 0x15d9; 1865148fa52SDavid Herrmann ev.u.create.product = 0x0a37; 1875148fa52SDavid Herrmann ev.u.create.version = 0; 1885148fa52SDavid Herrmann ev.u.create.country = 0; 1895148fa52SDavid Herrmann 1905148fa52SDavid Herrmann return uhid_write(fd, &ev); 1915148fa52SDavid Herrmann } 1925148fa52SDavid Herrmann 1935148fa52SDavid Herrmann static void destroy(int fd) 1945148fa52SDavid Herrmann { 1955148fa52SDavid Herrmann struct uhid_event ev; 1965148fa52SDavid Herrmann 1975148fa52SDavid Herrmann memset(&ev, 0, sizeof(ev)); 1985148fa52SDavid Herrmann ev.type = UHID_DESTROY; 1995148fa52SDavid Herrmann 2005148fa52SDavid Herrmann uhid_write(fd, &ev); 2015148fa52SDavid Herrmann } 2025148fa52SDavid Herrmann 203*f5e4e7fdSDavid Herrmann /* This parses raw output reports sent by the kernel to the device. A normal 204*f5e4e7fdSDavid Herrmann * uhid program shouldn't do this but instead just forward the raw report. 205*f5e4e7fdSDavid Herrmann * However, for ducomentational purposes, we try to detect LED events here and 206*f5e4e7fdSDavid Herrmann * print debug messages for it. */ 207*f5e4e7fdSDavid Herrmann static void handle_output(struct uhid_event *ev) 208*f5e4e7fdSDavid Herrmann { 209*f5e4e7fdSDavid Herrmann /* LED messages are adverised via OUTPUT reports; ignore the rest */ 210*f5e4e7fdSDavid Herrmann if (ev->u.output.rtype != UHID_OUTPUT_REPORT) 211*f5e4e7fdSDavid Herrmann return; 212*f5e4e7fdSDavid Herrmann /* LED reports have length 2 bytes */ 213*f5e4e7fdSDavid Herrmann if (ev->u.output.size != 2) 214*f5e4e7fdSDavid Herrmann return; 215*f5e4e7fdSDavid Herrmann /* first byte is report-id which is 0x02 for LEDs in our rdesc */ 216*f5e4e7fdSDavid Herrmann if (ev->u.output.data[0] != 0x2) 217*f5e4e7fdSDavid Herrmann return; 218*f5e4e7fdSDavid Herrmann 219*f5e4e7fdSDavid Herrmann /* print flags payload */ 220*f5e4e7fdSDavid Herrmann fprintf(stderr, "LED output report received with flags %x\n", 221*f5e4e7fdSDavid Herrmann ev->u.output.data[1]); 222*f5e4e7fdSDavid Herrmann } 223*f5e4e7fdSDavid Herrmann 2245148fa52SDavid Herrmann static int event(int fd) 2255148fa52SDavid Herrmann { 2265148fa52SDavid Herrmann struct uhid_event ev; 2275148fa52SDavid Herrmann ssize_t ret; 2285148fa52SDavid Herrmann 2295148fa52SDavid Herrmann memset(&ev, 0, sizeof(ev)); 2305148fa52SDavid Herrmann ret = read(fd, &ev, sizeof(ev)); 2315148fa52SDavid Herrmann if (ret == 0) { 2325148fa52SDavid Herrmann fprintf(stderr, "Read HUP on uhid-cdev\n"); 2335148fa52SDavid Herrmann return -EFAULT; 2345148fa52SDavid Herrmann } else if (ret < 0) { 2355148fa52SDavid Herrmann fprintf(stderr, "Cannot read uhid-cdev: %m\n"); 2365148fa52SDavid Herrmann return -errno; 2375148fa52SDavid Herrmann } else if (ret != sizeof(ev)) { 2385148fa52SDavid Herrmann fprintf(stderr, "Invalid size read from uhid-dev: %ld != %lu\n", 2395148fa52SDavid Herrmann ret, sizeof(ev)); 2405148fa52SDavid Herrmann return -EFAULT; 2415148fa52SDavid Herrmann } 2425148fa52SDavid Herrmann 2435148fa52SDavid Herrmann switch (ev.type) { 2445148fa52SDavid Herrmann case UHID_START: 2455148fa52SDavid Herrmann fprintf(stderr, "UHID_START from uhid-dev\n"); 2465148fa52SDavid Herrmann break; 2475148fa52SDavid Herrmann case UHID_STOP: 2485148fa52SDavid Herrmann fprintf(stderr, "UHID_STOP from uhid-dev\n"); 2495148fa52SDavid Herrmann break; 2505148fa52SDavid Herrmann case UHID_OPEN: 2515148fa52SDavid Herrmann fprintf(stderr, "UHID_OPEN from uhid-dev\n"); 2525148fa52SDavid Herrmann break; 2535148fa52SDavid Herrmann case UHID_CLOSE: 2545148fa52SDavid Herrmann fprintf(stderr, "UHID_CLOSE from uhid-dev\n"); 2555148fa52SDavid Herrmann break; 2565148fa52SDavid Herrmann case UHID_OUTPUT: 2575148fa52SDavid Herrmann fprintf(stderr, "UHID_OUTPUT from uhid-dev\n"); 258*f5e4e7fdSDavid Herrmann handle_output(&ev); 2595148fa52SDavid Herrmann break; 2605148fa52SDavid Herrmann case UHID_OUTPUT_EV: 2615148fa52SDavid Herrmann fprintf(stderr, "UHID_OUTPUT_EV from uhid-dev\n"); 2625148fa52SDavid Herrmann break; 2635148fa52SDavid Herrmann default: 2645148fa52SDavid Herrmann fprintf(stderr, "Invalid event from uhid-dev: %u\n", ev.type); 2655148fa52SDavid Herrmann } 2665148fa52SDavid Herrmann 2675148fa52SDavid Herrmann return 0; 2685148fa52SDavid Herrmann } 2695148fa52SDavid Herrmann 2705148fa52SDavid Herrmann static bool btn1_down; 2715148fa52SDavid Herrmann static bool btn2_down; 2725148fa52SDavid Herrmann static bool btn3_down; 2735148fa52SDavid Herrmann static signed char abs_hor; 2745148fa52SDavid Herrmann static signed char abs_ver; 2755148fa52SDavid Herrmann static signed char wheel; 2765148fa52SDavid Herrmann 2775148fa52SDavid Herrmann static int send_event(int fd) 2785148fa52SDavid Herrmann { 2795148fa52SDavid Herrmann struct uhid_event ev; 2805148fa52SDavid Herrmann 2815148fa52SDavid Herrmann memset(&ev, 0, sizeof(ev)); 2825148fa52SDavid Herrmann ev.type = UHID_INPUT; 283*f5e4e7fdSDavid Herrmann ev.u.input.size = 5; 2845148fa52SDavid Herrmann 285*f5e4e7fdSDavid Herrmann ev.u.input.data[0] = 0x1; 2865148fa52SDavid Herrmann if (btn1_down) 287*f5e4e7fdSDavid Herrmann ev.u.input.data[1] |= 0x1; 2885148fa52SDavid Herrmann if (btn2_down) 289*f5e4e7fdSDavid Herrmann ev.u.input.data[1] |= 0x2; 2905148fa52SDavid Herrmann if (btn3_down) 291*f5e4e7fdSDavid Herrmann ev.u.input.data[1] |= 0x4; 2925148fa52SDavid Herrmann 293*f5e4e7fdSDavid Herrmann ev.u.input.data[2] = abs_hor; 294*f5e4e7fdSDavid Herrmann ev.u.input.data[3] = abs_ver; 295*f5e4e7fdSDavid Herrmann ev.u.input.data[4] = wheel; 2965148fa52SDavid Herrmann 2975148fa52SDavid Herrmann return uhid_write(fd, &ev); 2985148fa52SDavid Herrmann } 2995148fa52SDavid Herrmann 3005148fa52SDavid Herrmann static int keyboard(int fd) 3015148fa52SDavid Herrmann { 3025148fa52SDavid Herrmann char buf[128]; 3035148fa52SDavid Herrmann ssize_t ret, i; 3045148fa52SDavid Herrmann 3055148fa52SDavid Herrmann ret = read(STDIN_FILENO, buf, sizeof(buf)); 3065148fa52SDavid Herrmann if (ret == 0) { 3075148fa52SDavid Herrmann fprintf(stderr, "Read HUP on stdin\n"); 3085148fa52SDavid Herrmann return -EFAULT; 3095148fa52SDavid Herrmann } else if (ret < 0) { 3105148fa52SDavid Herrmann fprintf(stderr, "Cannot read stdin: %m\n"); 3115148fa52SDavid Herrmann return -errno; 3125148fa52SDavid Herrmann } 3135148fa52SDavid Herrmann 3145148fa52SDavid Herrmann for (i = 0; i < ret; ++i) { 3155148fa52SDavid Herrmann switch (buf[i]) { 3165148fa52SDavid Herrmann case '1': 3175148fa52SDavid Herrmann btn1_down = !btn1_down; 3185148fa52SDavid Herrmann ret = send_event(fd); 3195148fa52SDavid Herrmann if (ret) 3205148fa52SDavid Herrmann return ret; 3215148fa52SDavid Herrmann break; 3225148fa52SDavid Herrmann case '2': 3235148fa52SDavid Herrmann btn2_down = !btn2_down; 3245148fa52SDavid Herrmann ret = send_event(fd); 3255148fa52SDavid Herrmann if (ret) 3265148fa52SDavid Herrmann return ret; 3275148fa52SDavid Herrmann break; 3285148fa52SDavid Herrmann case '3': 3295148fa52SDavid Herrmann btn3_down = !btn3_down; 3305148fa52SDavid Herrmann ret = send_event(fd); 3315148fa52SDavid Herrmann if (ret) 3325148fa52SDavid Herrmann return ret; 3335148fa52SDavid Herrmann break; 3345148fa52SDavid Herrmann case 'a': 3355148fa52SDavid Herrmann abs_hor = -20; 3365148fa52SDavid Herrmann ret = send_event(fd); 3375148fa52SDavid Herrmann abs_hor = 0; 3385148fa52SDavid Herrmann if (ret) 3395148fa52SDavid Herrmann return ret; 3405148fa52SDavid Herrmann break; 3415148fa52SDavid Herrmann case 'd': 3425148fa52SDavid Herrmann abs_hor = 20; 3435148fa52SDavid Herrmann ret = send_event(fd); 3445148fa52SDavid Herrmann abs_hor = 0; 3455148fa52SDavid Herrmann if (ret) 3465148fa52SDavid Herrmann return ret; 3475148fa52SDavid Herrmann break; 3485148fa52SDavid Herrmann case 'w': 3495148fa52SDavid Herrmann abs_ver = -20; 3505148fa52SDavid Herrmann ret = send_event(fd); 3515148fa52SDavid Herrmann abs_ver = 0; 3525148fa52SDavid Herrmann if (ret) 3535148fa52SDavid Herrmann return ret; 3545148fa52SDavid Herrmann break; 3555148fa52SDavid Herrmann case 's': 3565148fa52SDavid Herrmann abs_ver = 20; 3575148fa52SDavid Herrmann ret = send_event(fd); 3585148fa52SDavid Herrmann abs_ver = 0; 3595148fa52SDavid Herrmann if (ret) 3605148fa52SDavid Herrmann return ret; 3615148fa52SDavid Herrmann break; 3625148fa52SDavid Herrmann case 'r': 3635148fa52SDavid Herrmann wheel = 1; 3645148fa52SDavid Herrmann ret = send_event(fd); 3655148fa52SDavid Herrmann wheel = 0; 3665148fa52SDavid Herrmann if (ret) 3675148fa52SDavid Herrmann return ret; 3685148fa52SDavid Herrmann break; 3695148fa52SDavid Herrmann case 'f': 3705148fa52SDavid Herrmann wheel = -1; 3715148fa52SDavid Herrmann ret = send_event(fd); 3725148fa52SDavid Herrmann wheel = 0; 3735148fa52SDavid Herrmann if (ret) 3745148fa52SDavid Herrmann return ret; 3755148fa52SDavid Herrmann break; 3765148fa52SDavid Herrmann case 'q': 3775148fa52SDavid Herrmann return -ECANCELED; 3785148fa52SDavid Herrmann default: 3795148fa52SDavid Herrmann fprintf(stderr, "Invalid input: %c\n", buf[i]); 3805148fa52SDavid Herrmann } 3815148fa52SDavid Herrmann } 3825148fa52SDavid Herrmann 3835148fa52SDavid Herrmann return 0; 3845148fa52SDavid Herrmann } 3855148fa52SDavid Herrmann 3865148fa52SDavid Herrmann int main(int argc, char **argv) 3875148fa52SDavid Herrmann { 3885148fa52SDavid Herrmann int fd; 3895148fa52SDavid Herrmann const char *path = "/dev/uhid"; 3905148fa52SDavid Herrmann struct pollfd pfds[2]; 3915148fa52SDavid Herrmann int ret; 3925148fa52SDavid Herrmann struct termios state; 3935148fa52SDavid Herrmann 3945148fa52SDavid Herrmann ret = tcgetattr(STDIN_FILENO, &state); 3955148fa52SDavid Herrmann if (ret) { 3965148fa52SDavid Herrmann fprintf(stderr, "Cannot get tty state\n"); 3975148fa52SDavid Herrmann } else { 3985148fa52SDavid Herrmann state.c_lflag &= ~ICANON; 3995148fa52SDavid Herrmann state.c_cc[VMIN] = 1; 4005148fa52SDavid Herrmann ret = tcsetattr(STDIN_FILENO, TCSANOW, &state); 4015148fa52SDavid Herrmann if (ret) 4025148fa52SDavid Herrmann fprintf(stderr, "Cannot set tty state\n"); 4035148fa52SDavid Herrmann } 4045148fa52SDavid Herrmann 4055148fa52SDavid Herrmann if (argc >= 2) { 4065148fa52SDavid Herrmann if (!strcmp(argv[1], "-h") || !strcmp(argv[1], "--help")) { 4075148fa52SDavid Herrmann fprintf(stderr, "Usage: %s [%s]\n", argv[0], path); 4085148fa52SDavid Herrmann return EXIT_SUCCESS; 4095148fa52SDavid Herrmann } else { 4105148fa52SDavid Herrmann path = argv[1]; 4115148fa52SDavid Herrmann } 4125148fa52SDavid Herrmann } 4135148fa52SDavid Herrmann 4145148fa52SDavid Herrmann fprintf(stderr, "Open uhid-cdev %s\n", path); 4155148fa52SDavid Herrmann fd = open(path, O_RDWR | O_CLOEXEC); 4165148fa52SDavid Herrmann if (fd < 0) { 4175148fa52SDavid Herrmann fprintf(stderr, "Cannot open uhid-cdev %s: %m\n", path); 4185148fa52SDavid Herrmann return EXIT_FAILURE; 4195148fa52SDavid Herrmann } 4205148fa52SDavid Herrmann 4215148fa52SDavid Herrmann fprintf(stderr, "Create uhid device\n"); 4225148fa52SDavid Herrmann ret = create(fd); 4235148fa52SDavid Herrmann if (ret) { 4245148fa52SDavid Herrmann close(fd); 4255148fa52SDavid Herrmann return EXIT_FAILURE; 4265148fa52SDavid Herrmann } 4275148fa52SDavid Herrmann 4285148fa52SDavid Herrmann pfds[0].fd = STDIN_FILENO; 4295148fa52SDavid Herrmann pfds[0].events = POLLIN; 4305148fa52SDavid Herrmann pfds[1].fd = fd; 4315148fa52SDavid Herrmann pfds[1].events = POLLIN; 4325148fa52SDavid Herrmann 4335148fa52SDavid Herrmann fprintf(stderr, "Press 'q' to quit...\n"); 4345148fa52SDavid Herrmann while (1) { 4355148fa52SDavid Herrmann ret = poll(pfds, 2, -1); 4365148fa52SDavid Herrmann if (ret < 0) { 4375148fa52SDavid Herrmann fprintf(stderr, "Cannot poll for fds: %m\n"); 4385148fa52SDavid Herrmann break; 4395148fa52SDavid Herrmann } 4405148fa52SDavid Herrmann if (pfds[0].revents & POLLHUP) { 4415148fa52SDavid Herrmann fprintf(stderr, "Received HUP on stdin\n"); 4425148fa52SDavid Herrmann break; 4435148fa52SDavid Herrmann } 4445148fa52SDavid Herrmann if (pfds[1].revents & POLLHUP) { 4455148fa52SDavid Herrmann fprintf(stderr, "Received HUP on uhid-cdev\n"); 4465148fa52SDavid Herrmann break; 4475148fa52SDavid Herrmann } 4485148fa52SDavid Herrmann 4495148fa52SDavid Herrmann if (pfds[0].revents & POLLIN) { 4505148fa52SDavid Herrmann ret = keyboard(fd); 4515148fa52SDavid Herrmann if (ret) 4525148fa52SDavid Herrmann break; 4535148fa52SDavid Herrmann } 4545148fa52SDavid Herrmann if (pfds[1].revents & POLLIN) { 4555148fa52SDavid Herrmann ret = event(fd); 4565148fa52SDavid Herrmann if (ret) 4575148fa52SDavid Herrmann break; 4585148fa52SDavid Herrmann } 4595148fa52SDavid Herrmann } 4605148fa52SDavid Herrmann 4615148fa52SDavid Herrmann fprintf(stderr, "Destroy uhid device\n"); 4625148fa52SDavid Herrmann destroy(fd); 4635148fa52SDavid Herrmann return EXIT_SUCCESS; 4645148fa52SDavid Herrmann } 465