xref: /openbmc/linux/samples/uhid/uhid-example.c (revision f5e4e7fdd57691d5308cf854dd0dbcfd58799e9a)
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