blob: 889a759a78b76c290d834016958774aa133a336a [file] [log] [blame]
Dennis Kempina9963ba2012-06-08 10:32:23 -07001/*
2 * Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
3 * Use of this source code is governed by a BSD-style license that can be
4 * found in the LICENSE file.
5 */
Dennis Kempin320aef12012-06-11 15:26:52 -07006#include <libevdev/libevdev.h>
Dennis Kempina9963ba2012-06-08 10:32:23 -07007
8#include <errno.h>
9#include <fcntl.h>
10#include <linux/input.h>
11#include <stdio.h>
12#include <time.h>
13#include <unistd.h>
14
Dennis Kempin320aef12012-06-11 15:26:52 -070015#include <libevdev/libevdev_event.h>
16#include <libevdev/libevdev_log.h>
17#include <libevdev/libevdev_util.h>
Dennis Kempina9963ba2012-06-08 10:32:23 -070018
19/* Number of events to attempt to read from kernel on each SIGIO */
20#define NUM_EVENTS 16
21
22#ifndef EVIOCGMTSLOTS
23#define EVIOCGMTSLOTS(len) _IOC(_IOC_READ, 'E', 0x0a, len)
24#endif
25
26/* Set clockid to be used for timestamps */
27#ifndef EVIOCSCLOCKID
28#define EVIOCSCLOCKID _IOW('E', 0xa0, int)
29#endif
30
Dennis Kempin0e265392013-02-22 12:59:03 -080031static int EvdevWriteBitmask(FILE* fp, const char* name,
32 unsigned long* bitmask, size_t num_bytes);
33static int EvdevReadBitmask(FILE* fp, const char* expected_name,
34 unsigned long* bitmask, size_t num_bytes);
35
36#ifndef EVDEV_HOLLOW
37
Dennis Kempina9963ba2012-06-08 10:32:23 -070038static void Absinfo_Print(EvdevPtr device, struct input_absinfo*);
39static const char* Event_Property_To_String(int type);
Che-Liang Chioub9fc9732012-11-26 15:49:02 -080040static EvdevClass EvdevProbeClass(EvdevInfoPtr info);
41static const char* EvdevClassToString(EvdevClass cls);
Dennis Kempina9963ba2012-06-08 10:32:23 -070042
43int EvdevOpen(EvdevPtr evdev, const char* device) {
44 evdev->fd = open(device, O_RDWR | O_NONBLOCK, 0);
45 return evdev->fd;
46}
47
48int EvdevClose(EvdevPtr evdev) {
49 close(evdev->fd);
50 evdev->fd = -1;
51 return evdev->fd;
52}
53
54int EvdevRead(EvdevPtr evdev) {
55 struct input_event ev[NUM_EVENTS];
Andrew de los Reyesec028842012-08-16 18:08:12 -070056 EventStatePtr evstate = evdev->evstate;
Dennis Kempina9963ba2012-06-08 10:32:23 -070057 int i;
58 int len;
59 bool sync_evdev_state = false;
60
61 do {
62 len = read(evdev->fd, &ev, sizeof(ev));
63 if (len <= 0)
64 return errno;
65
Andrew de los Reyesec028842012-08-16 18:08:12 -070066 /* Read as many whole struct input_event objects as we can into the
67 circular buffer */
68 if (evstate->debug_buf) {
69 for (i = 0; i < len / sizeof(*ev); i++) {
70 evstate->debug_buf[evstate->debug_buf_tail] = ev[i];
71 evstate->debug_buf_tail =
72 (evstate->debug_buf_tail + 1) % DEBUG_BUF_SIZE;
73 }
74 }
75
Dennis Kempina9963ba2012-06-08 10:32:23 -070076 /* kernel always delivers complete events, so len must be sizeof *ev */
77 if (len % sizeof(*ev))
78 return errno;
79
80 /* Process events ... */
81 for (i = 0; i < len / sizeof(ev[0]); i++) {
82 if (sync_evdev_state)
83 break;
84 if (timercmp(&ev[i].time, &evdev->before_sync_time, <)) {
85 /* Ignore events before last sync time */
86 continue;
87 } else if (timercmp(&ev[i].time, &evdev->after_sync_time, >)) {
88 /* Event_Process returns TRUE if SYN_DROPPED detected */
89 sync_evdev_state = Event_Process(evdev, &ev[i]);
90 } else {
91 /* If the event occurred during sync, then sync again */
92 sync_evdev_state = true;
93 }
94 }
95
96 } while (len == sizeof(ev));
97 /* Keep reading if kernel supplied NUM_EVENTS events. */
98
99 if (sync_evdev_state)
100 Event_Sync_State(evdev);
101
102 return Success;
103}
104
105int EvdevProbe(EvdevPtr device) {
106 int len, i;
107 int fd;
108 EvdevInfoPtr info;
109
110 fd = device->fd;
111 info = &device->info;
112 if (ioctl(fd, EVIOCGID, &info->id) < 0) {
113 LOG_ERROR(device, "ioctl EVIOCGID failed: %s\n", strerror(errno));
114 return !Success;
115 }
116
117 if (ioctl(fd, EVIOCGNAME(sizeof(info->name) - 1),
118 info->name) < 0) {
119 LOG_ERROR(device, "ioctl EVIOCGNAME failed: %s\n", strerror(errno));
120 return !Success;
121 }
122
123 len = ioctl(fd, EVIOCGPROP(sizeof(info->prop_bitmask)),
124 info->prop_bitmask);
125 if (len < 0) {
126 LOG_ERROR(device, "ioctl EVIOCGPROP failed: %s\n", strerror(errno));
127 return !Success;
128 }
129 for (i = 0; i < len*8; i++) {
130 if (TestBit(i, info->prop_bitmask))
131 LOG_DEBUG(device, "Has Property: %d (%s)\n", i,
132 Event_Property_To_String(i));
133 }
134
135 len = ioctl(fd, EVIOCGBIT(0, sizeof(info->bitmask)),
136 info->bitmask);
137 if (len < 0) {
138 LOG_ERROR(device, "ioctl EVIOCGBIT failed: %s\n",
139 strerror(errno));
140 return !Success;
141 }
142 for (i = 0; i < len*8; i++) {
143 if (TestBit(i, info->bitmask))
144 LOG_DEBUG(device, "Has Event Type %d = %s\n", i,
145 Event_Type_To_String(i));
146 }
147
148 len = ioctl(fd, EVIOCGBIT(EV_KEY, sizeof(info->key_bitmask)),
149 info->key_bitmask);
150 if (len < 0) {
151 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_KEY) failed: %s\n",
152 strerror(errno));
153 return !Success;
154 }
155 for (i = 0; i < len*8; i++) {
156 if (TestBit(i, info->key_bitmask))
157 LOG_DEBUG(device, "Has KEY[%d] = %s\n", i,
158 Event_To_String(EV_KEY, i));
159 }
160
161 len = ioctl(fd, EVIOCGBIT(EV_LED, sizeof(info->led_bitmask)),
162 info->led_bitmask);
163 if (len < 0) {
164 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_LED) failed: %s\n",
165 strerror(errno));
166 return !Success;
167 }
168 for (i = 0; i < len*8; i++) {
169 if (TestBit(i, info->led_bitmask))
170 LOG_DEBUG(device, "Has LED[%d] = %s\n", i,
171 Event_To_String(EV_LED, i));
172 }
173
174 len = ioctl(fd, EVIOCGBIT(EV_REL, sizeof(info->rel_bitmask)),
175 info->rel_bitmask);
176 if (len < 0) {
177 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_REL) failed: %s\n",
178 strerror(errno));
179 return !Success;
180 }
181 for (i = 0; i < len*8; i++) {
182 if (TestBit(i, info->rel_bitmask))
183 LOG_DEBUG(device, "Has REL[%d] = %s\n", i,
184 Event_To_String(EV_REL, i));
185 }
186
Dennis Kempina9963ba2012-06-08 10:32:23 -0700187 len = ioctl(fd, EVIOCGBIT(EV_ABS, sizeof(info->abs_bitmask)),
188 info->abs_bitmask);
189 if (len < 0) {
190 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_ABS) failed: %s\n",
191 strerror(errno));
192 return !Success;
193 }
194
195 for (i = ABS_X; i <= ABS_MAX; i++) {
196 if (TestBit(i, info->abs_bitmask)) {
197 struct input_absinfo* absinfo = &info->absinfo[i];
198 LOG_DEBUG(device, "Has ABS[%d] = %s\n", i,
199 Event_To_String(EV_ABS, i));
200 len = ioctl(fd, EVIOCGABS(i), absinfo);
201 if (len < 0) {
202 LOG_ERROR(device, "ioctl EVIOCGABS(%d) failed: %s\n", i,
203 strerror(errno));
204 return !Success;
205 }
206
207 Absinfo_Print(device, absinfo);
208 }
209 }
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800210
211 info->evdev_class = EvdevProbeClass(info);
212 LOG_DEBUG(device, "Has evdev device class = %s\n",
213 EvdevClassToString(info->evdev_class));
214 if (info->evdev_class == EvdevClassUnknown) {
215 LOG_ERROR(device, "Couldn't determine evdev class\n");
216 return !Success;
217 }
218
Dennis Kempina9963ba2012-06-08 10:32:23 -0700219 return Success;
220}
221
222int EvdevProbeAbsinfo(EvdevPtr device, size_t key) {
223 struct input_absinfo* absinfo;
224
225 absinfo = &device->info.absinfo[key];
226 if (ioctl(device->fd, EVIOCGABS(key), absinfo) < 0) {
Daniel Kurtz0d097082012-09-07 22:27:51 +0800227 LOG_ERROR(device, "ioctl EVIOCGABS(%zu) failed: %s\n", key,
Dennis Kempina9963ba2012-06-08 10:32:23 -0700228 strerror(errno));
229 return !Success;
230 } else {
231 return Success;
232 }
233}
234
235int EvdevProbeMTSlot(EvdevPtr device, MTSlotInfoPtr req) {
Dennis Kempine9cc2642012-06-25 11:52:24 -0700236 if (ioctl(device->fd, EVIOCGMTSLOTS((sizeof(*req))), req) < 0) {
Dennis Kempina9963ba2012-06-08 10:32:23 -0700237 LOG_ERROR(device, "ioctl EVIOCGMTSLOTS(req.code=%d) failed: %s\n",
Dennis Kempine9cc2642012-06-25 11:52:24 -0700238 req->code, strerror(errno));
Dennis Kempina9963ba2012-06-08 10:32:23 -0700239 return !Success;
240 } else {
241 return Success;
242 }
243}
244
245int EvdevProbeKeyState(EvdevPtr device) {
246 int len = sizeof(device->key_state_bitmask);
247
248 memset(device->key_state_bitmask, 0, len);
249 if (ioctl(device->fd, EVIOCGKEY(len), device->key_state_bitmask) < 0) {
250 LOG_ERROR(device, "ioctl EVIOCGKEY failed: %s\n", strerror(errno));
251 return !Success;
252 } else {
253 return Success;
254 }
255}
256
257int EvdevEnableMonotonic(EvdevPtr device) {
258 unsigned int clk = CLOCK_MONOTONIC;
259 return (ioctl(device->fd, EVIOCSCLOCKID, &clk) == 0) ? Success : !Success;
260}
261
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800262#endif // #ifndef EVDEV_HOLLOW
263
264/*
265 * Check if the device is a single-pressure one which reports ABS_PRESSURE only.
266 */
267int EvdevIsSinglePressureDevice(EvdevPtr device) {
268 EvdevInfoPtr info = &device->info;
269
270 return (!TestBit(ABS_MT_PRESSURE, info->abs_bitmask) &&
271 TestBit(ABS_PRESSURE, info->abs_bitmask));
272}
273
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800274int EvdevWriteInfoToFile(FILE* fp, const EvdevInfoPtr info) {
275 int ret;
276
277 ret = fprintf(fp, "# device: %s\n", info->name);
278 if (ret <= 0)
279 return ret;
280
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800281 ret = EvdevWriteBitmask(fp, "bit", info->bitmask, sizeof(info->bitmask));
282 if (ret <= 0)
283 return ret;
284
285 ret = EvdevWriteBitmask(fp, "key", info->key_bitmask,
286 sizeof(info->key_bitmask));
287 if (ret <= 0)
288 return ret;
289
290 ret = EvdevWriteBitmask(fp, "rel", info->rel_bitmask,
291 sizeof(info->rel_bitmask));
292 if (ret <= 0)
293 return ret;
294
295 ret = EvdevWriteBitmask(fp, "abs", info->abs_bitmask,
296 sizeof(info->abs_bitmask));
297 if (ret <= 0)
298 return ret;
299
300 ret = EvdevWriteBitmask(fp, "led", info->led_bitmask,
301 sizeof(info->led_bitmask));
302 if (ret <= 0)
303 return ret;
304
305 ret = EvdevWriteBitmask(fp, "prp", info->prop_bitmask,
306 sizeof(info->prop_bitmask));
307 if (ret <= 0)
308 return ret;
309
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800310 // when reading the log we need to know which absinfos
311 // exist which is stored in the abs bitmask.
312 // so we have to write absinfo after the bitmasks.
313 for (int i = ABS_X; i <= ABS_MAX; i++) {
314 if (TestBit(i, info->abs_bitmask)) {
315 struct input_absinfo* abs = &info->absinfo[i];
316 ret = fprintf(fp, "# absinfo: %d %d %d %d %d %d\n",
317 i, abs->minimum, abs->maximum,
318 abs->fuzz, abs->flat, abs->resolution);
319 if (ret <= 0)
320 return ret;
321 }
322 }
323
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800324 return 1;
325}
326
327int EvdevWriteEventToFile(FILE* fp, const struct input_event* ev) {
Mike Frysinger7abec9e2012-12-23 13:00:24 -0500328 return fprintf(fp, "E: %ld.%06u %04x %04x %d\n",
329 (long)ev->time.tv_sec, (unsigned)ev->time.tv_usec,
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800330 ev->type, ev->code, ev->value);
331}
332
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800333int EvdevReadInfoFromFile(FILE* fp, EvdevInfoPtr info) {
334 int ret;
335
Dennis Kempinbb4375b2013-03-28 12:02:54 -0700336 ret = fscanf(fp, "# device: %1024[^\n]\n", info->name);
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700337 if (ret <= 0) {
338 fprintf(stderr, "Parse device name failed\n");
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800339 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700340 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800341
342 ret = EvdevReadBitmask(fp, "bit", info->bitmask, sizeof(info->bitmask));
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700343 if (ret <= 0) {
344 fprintf(stderr, "Parse device bit failed\n");
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800345 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700346 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800347
348 ret = EvdevReadBitmask(fp, "key", info->key_bitmask,
349 sizeof(info->key_bitmask));
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700350 if (ret <= 0) {
351 fprintf(stderr, "Parse device key failed\n");
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800352 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700353 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800354
355 ret = EvdevReadBitmask(fp, "rel", info->rel_bitmask,
356 sizeof(info->rel_bitmask));
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700357 if (ret <= 0) {
358 fprintf(stderr, "Parse device rel failed\n");
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800359 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700360 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800361
362 ret = EvdevReadBitmask(fp, "abs", info->abs_bitmask,
363 sizeof(info->abs_bitmask));
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700364 if (ret <= 0) {
365 fprintf(stderr, "Parse device abs failed\n");
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800366 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700367 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800368
369 ret = EvdevReadBitmask(fp, "led", info->led_bitmask,
370 sizeof(info->led_bitmask));
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700371 if (ret <= 0) {
372 fprintf(stderr, "Parse device led failed\n");
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800373 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700374 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800375
376 ret = EvdevReadBitmask(fp, "prp", info->prop_bitmask,
377 sizeof(info->prop_bitmask));
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700378 if (ret <= 0) {
379 fprintf(stderr, "Parse device prp failed\n");
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800380 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700381 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800382
383 for (int i = ABS_X; i <= ABS_MAX; i++) {
384 if (TestBit(i, info->abs_bitmask)) {
385 struct input_absinfo abs;
386 int abs_index;
387 ret = fscanf(fp, "# absinfo: %d %d %d %d %d %d\n",
388 &abs_index, &abs.minimum, &abs.maximum,
389 &abs.fuzz, &abs.flat, &abs.resolution);
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700390 if (ret <= 0 || abs_index != i) {
391 fprintf(stderr, "Parse device absinfo failed on %d\n", i);
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800392 return -1;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700393 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800394 info->absinfo[i] = abs;
395 }
396 }
397 return 1;
398}
399
400int EvdevReadEventFromFile(FILE* fp, struct input_event* ev) {
401 unsigned long sec;
402 unsigned usec, type, code;
403 int value;
404 int ret = fscanf(fp, "E: %lu.%06u %04x %04x %d\n",
405 &sec, &usec, &type, &code, &value);
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700406 if (ret == EOF)
407 return -1;
408 if (ret <= 0) {
409 int err = errno;
410 fprintf(stderr, "EvdevReadEventFromFile failed: %d, errno: %d\n", ret, err);
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800411 return ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700412 }
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800413
414 ev->time.tv_sec = sec;
415 ev->time.tv_usec = usec;
416 ev->type = type;
417 ev->code = code;
418 ev->value = value;
419 return ret;
420}
421
422static int EvdevReadBitmask(FILE* fp, const char* expected_name,
423 unsigned long* bitmask, size_t num_bytes) {
424 unsigned char* bytes = (unsigned char*)bitmask;
425 int ret;
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700426 char name[64] = { 0 };
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800427
Chung-yih Wang2eeae932013-04-29 12:50:46 +0800428 ret = fscanf(fp, "# %63[^:]:", name);
Andrew de los Reyes8fbd3c12013-05-08 09:30:50 -0700429 if (ret <= 0 || strcmp(name, expected_name)) {
430 fprintf(stderr, "EvdevReadBitmask: %d <= 0 or [%s]!=[%s]\n",
431 ret, name, expected_name);
432 return -1;
433 }
434 memset(bitmask, 0, num_bytes);
435 for (size_t i = 0; i < num_bytes; ++i) {
436 // Make sure we don't go off the end of the line
437 int next_char = fgetc(fp);
438 if (next_char != EOF)
439 ungetc(next_char, fp);
440 if (next_char == '\n' || next_char == EOF)
441 break;
442
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800443 unsigned int tmp;
444 ret = fscanf(fp, " %02X", &tmp);
445 if (ret <= 0)
Dennis Kempin1ecc7e92013-03-29 16:58:08 -0700446 bytes[i] = 0;
447 else
448 bytes[i] = (unsigned char)tmp;
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800449 }
Chung-yih Wang2eeae932013-04-29 12:50:46 +0800450 // size(in bytes) of bitmask array may differs per platform, try to read
451 // remaining bytes if exists
452 do {
453 if (fgets(name, sizeof(name), fp) == NULL)
454 return -1;
455 } while (name[strlen(name) - 1] != '\n');
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800456 return 1;
457}
458
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800459static int EvdevWriteBitmask(FILE* fp, const char* name,
460 unsigned long* bitmask, size_t num_bytes) {
461 int ret;
462
463 unsigned char* bytes = (unsigned char*)bitmask;
464 ret = fprintf(fp, "# %s:", name);
465 if (ret <= 0)
466 return ret;
467
468 for (int i = 0; i < num_bytes; ++i) {
469 ret = fprintf(fp, " %02X", bytes[i]);
470 if (ret <= 0)
471 return ret;
472 }
473 ret = fprintf(fp, "\n");
474 return ret;
475}
Dennis Kempina9963ba2012-06-08 10:32:23 -0700476
Dennis Kempin0e265392013-02-22 12:59:03 -0800477#ifndef EVDEV_HOLLOW
478
Dennis Kempina9963ba2012-06-08 10:32:23 -0700479static const char*
480Event_Property_To_String(int type) {
481 switch (type) {
482 case INPUT_PROP_POINTER: return "POINTER"; /* needs a pointer */
483 case INPUT_PROP_DIRECT: return "DIRECT"; /* direct input devices */
484 case INPUT_PROP_BUTTONPAD: return "BUTTONPAD"; /* has button under pad */
485 case INPUT_PROP_SEMI_MT: return "SEMI_MT"; /* touch rectangle only */
486 default: return "?";
487 }
488}
489
490static void
491Absinfo_Print(EvdevPtr device, struct input_absinfo* absinfo)
492{
493 LOG_DEBUG(device, " min = %d\n", absinfo->minimum);
494 LOG_DEBUG(device, " max = %d\n", absinfo->maximum);
495 if (absinfo->fuzz)
496 LOG_DEBUG(device, " fuzz = %d\n", absinfo->fuzz);
497 if (absinfo->resolution)
498 LOG_DEBUG(device, " res = %d\n", absinfo->resolution);
499}
500
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800501/*
502 * Heuristics for determining evdev device class; similar to those of
503 * xf86-input-evdev.
504 */
505static EvdevClass EvdevProbeClass(EvdevInfoPtr info) {
506 int bit;
507 for (bit = 0; bit < BTN_MISC; bit++)
508 if (TestBit(bit, info->key_bitmask))
509 return EvdevClassKeyboard;
510
511 if (TestBit(REL_X, info->rel_bitmask) &&
Che-Liang Chiou7e45cb22013-01-15 14:57:13 -0800512 TestBit(REL_Y, info->rel_bitmask)) {
513 if (TestBit(ABS_MT_POSITION_X, info->abs_bitmask) &&
514 TestBit(ABS_MT_POSITION_Y, info->abs_bitmask))
515 return EvdevClassMultitouchMouse;
516 else
517 return EvdevClassMouse;
518 }
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800519
520 if (TestBit(ABS_X, info->abs_bitmask) &&
521 TestBit(ABS_Y, info->abs_bitmask)) {
522
523 if (TestBit(BTN_TOOL_PEN, info->key_bitmask) ||
524 TestBit(BTN_STYLUS, info->key_bitmask) ||
525 TestBit(BTN_STYLUS2, info->key_bitmask))
526 return EvdevClassTablet;
527
528 if (TestBit(ABS_PRESSURE, info->abs_bitmask) ||
529 TestBit(BTN_TOUCH, info->key_bitmask)) {
530 if (TestBit(BTN_LEFT, info->key_bitmask) ||
531 TestBit(BTN_MIDDLE, info->key_bitmask) ||
532 TestBit(BTN_RIGHT, info->key_bitmask) ||
533 TestBit(BTN_TOOL_FINGER, info->key_bitmask))
534 return EvdevClassTouchpad;
535 else
536 return EvdevClassTouchscreen;
537 }
538
539 /* Some touchscreens use BTN_LEFT rather than BTN_TOUCH */
540 if (TestBit(BTN_LEFT, info->key_bitmask))
541 return EvdevClassTouchscreen;
542 }
543
544 return EvdevClassUnknown;
545}
546
547static const char* EvdevClassToString(EvdevClass cls) {
548 switch (cls) {
549 case EvdevClassUnknown: return "EvdevClassUnknown";
550 case EvdevClassKeyboard: return "EvdevClassKeyboard";
551 case EvdevClassMouse: return "EvdevClassMouse";
Che-Liang Chiou7e45cb22013-01-15 14:57:13 -0800552 case EvdevClassMultitouchMouse: return "EvdevClassMultitouchMouse";
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800553 case EvdevClassTablet: return "EvdevClassTablet";
554 case EvdevClassTouchpad: return "EvdevClassTouchpad";
555 case EvdevClassTouchscreen: return "EvdevClassTouchscreen";
556 }
557 return "Unhandled Evdev Class";
558}
Dennis Kempin0e265392013-02-22 12:59:03 -0800559
560#endif // #ifndef EVDEV_HOLLOW