blob: e920a026443907740c90bd6de49e482d92987730 [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;
Michael Spang61f882f2015-04-09 11:07:40 -040059 int status = Success;
Dennis Kempina9963ba2012-06-08 10:32:23 -070060 bool sync_evdev_state = false;
61
62 do {
63 len = read(evdev->fd, &ev, sizeof(ev));
Michael Spang61f882f2015-04-09 11:07:40 -040064 if (len < 0) {
65 if (errno != EINTR && errno != EAGAIN)
66 status = errno;
Chung-yih Wang53af0892013-06-27 18:13:37 +080067 break;
Michael Spang61f882f2015-04-09 11:07:40 -040068 }
Dennis Kempina9963ba2012-06-08 10:32:23 -070069
Andrew de los Reyesec028842012-08-16 18:08:12 -070070 /* Read as many whole struct input_event objects as we can into the
71 circular buffer */
Yunlian Jiang4f5d5052015-01-20 14:27:40 -080072 for (i = 0; i < len / sizeof(*ev); i++) {
73 evstate->debug_buf[evstate->debug_buf_tail] = ev[i];
74 evstate->debug_buf_tail =
75 (evstate->debug_buf_tail + 1) % DEBUG_BUF_SIZE;
Andrew de los Reyesec028842012-08-16 18:08:12 -070076 }
77
Dennis Kempina9963ba2012-06-08 10:32:23 -070078 /* kernel always delivers complete events, so len must be sizeof *ev */
79 if (len % sizeof(*ev))
Chung-yih Wang53af0892013-06-27 18:13:37 +080080 break;
Dennis Kempina9963ba2012-06-08 10:32:23 -070081
82 /* Process events ... */
83 for (i = 0; i < len / sizeof(ev[0]); i++) {
84 if (sync_evdev_state)
85 break;
86 if (timercmp(&ev[i].time, &evdev->before_sync_time, <)) {
87 /* Ignore events before last sync time */
88 continue;
89 } else if (timercmp(&ev[i].time, &evdev->after_sync_time, >)) {
90 /* Event_Process returns TRUE if SYN_DROPPED detected */
Chung-yih Wang53af0892013-06-27 18:13:37 +080091 sync_evdev_state |= Event_Process(evdev, &ev[i]);
Dennis Kempina9963ba2012-06-08 10:32:23 -070092 } else {
93 /* If the event occurred during sync, then sync again */
94 sync_evdev_state = true;
95 }
96 }
97
98 } while (len == sizeof(ev));
99 /* Keep reading if kernel supplied NUM_EVENTS events. */
100
101 if (sync_evdev_state)
102 Event_Sync_State(evdev);
103
Michael Spang61f882f2015-04-09 11:07:40 -0400104 return status;
Dennis Kempina9963ba2012-06-08 10:32:23 -0700105}
106
107int EvdevProbe(EvdevPtr device) {
108 int len, i;
109 int fd;
110 EvdevInfoPtr info;
111
112 fd = device->fd;
113 info = &device->info;
114 if (ioctl(fd, EVIOCGID, &info->id) < 0) {
115 LOG_ERROR(device, "ioctl EVIOCGID failed: %s\n", strerror(errno));
116 return !Success;
117 }
118
119 if (ioctl(fd, EVIOCGNAME(sizeof(info->name) - 1),
120 info->name) < 0) {
121 LOG_ERROR(device, "ioctl EVIOCGNAME failed: %s\n", strerror(errno));
122 return !Success;
123 }
124
125 len = ioctl(fd, EVIOCGPROP(sizeof(info->prop_bitmask)),
126 info->prop_bitmask);
127 if (len < 0) {
128 LOG_ERROR(device, "ioctl EVIOCGPROP failed: %s\n", strerror(errno));
129 return !Success;
130 }
131 for (i = 0; i < len*8; i++) {
132 if (TestBit(i, info->prop_bitmask))
133 LOG_DEBUG(device, "Has Property: %d (%s)\n", i,
134 Event_Property_To_String(i));
135 }
136
137 len = ioctl(fd, EVIOCGBIT(0, sizeof(info->bitmask)),
138 info->bitmask);
139 if (len < 0) {
140 LOG_ERROR(device, "ioctl EVIOCGBIT failed: %s\n",
141 strerror(errno));
142 return !Success;
143 }
144 for (i = 0; i < len*8; i++) {
145 if (TestBit(i, info->bitmask))
146 LOG_DEBUG(device, "Has Event Type %d = %s\n", i,
147 Event_Type_To_String(i));
148 }
149
150 len = ioctl(fd, EVIOCGBIT(EV_KEY, sizeof(info->key_bitmask)),
151 info->key_bitmask);
152 if (len < 0) {
153 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_KEY) failed: %s\n",
154 strerror(errno));
155 return !Success;
156 }
157 for (i = 0; i < len*8; i++) {
158 if (TestBit(i, info->key_bitmask))
159 LOG_DEBUG(device, "Has KEY[%d] = %s\n", i,
160 Event_To_String(EV_KEY, i));
161 }
162
163 len = ioctl(fd, EVIOCGBIT(EV_LED, sizeof(info->led_bitmask)),
164 info->led_bitmask);
165 if (len < 0) {
166 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_LED) failed: %s\n",
167 strerror(errno));
168 return !Success;
169 }
170 for (i = 0; i < len*8; i++) {
171 if (TestBit(i, info->led_bitmask))
172 LOG_DEBUG(device, "Has LED[%d] = %s\n", i,
173 Event_To_String(EV_LED, i));
174 }
175
176 len = ioctl(fd, EVIOCGBIT(EV_REL, sizeof(info->rel_bitmask)),
177 info->rel_bitmask);
178 if (len < 0) {
179 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_REL) failed: %s\n",
180 strerror(errno));
181 return !Success;
182 }
183 for (i = 0; i < len*8; i++) {
184 if (TestBit(i, info->rel_bitmask))
185 LOG_DEBUG(device, "Has REL[%d] = %s\n", i,
186 Event_To_String(EV_REL, i));
187 }
188
Dennis Kempina9963ba2012-06-08 10:32:23 -0700189 len = ioctl(fd, EVIOCGBIT(EV_ABS, sizeof(info->abs_bitmask)),
190 info->abs_bitmask);
191 if (len < 0) {
192 LOG_ERROR(device, "ioctl EVIOCGBIT(EV_ABS) failed: %s\n",
193 strerror(errno));
194 return !Success;
195 }
196
197 for (i = ABS_X; i <= ABS_MAX; i++) {
198 if (TestBit(i, info->abs_bitmask)) {
199 struct input_absinfo* absinfo = &info->absinfo[i];
200 LOG_DEBUG(device, "Has ABS[%d] = %s\n", i,
201 Event_To_String(EV_ABS, i));
202 len = ioctl(fd, EVIOCGABS(i), absinfo);
203 if (len < 0) {
204 LOG_ERROR(device, "ioctl EVIOCGABS(%d) failed: %s\n", i,
205 strerror(errno));
206 return !Success;
207 }
208
209 Absinfo_Print(device, absinfo);
210 }
211 }
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800212
213 info->evdev_class = EvdevProbeClass(info);
214 LOG_DEBUG(device, "Has evdev device class = %s\n",
215 EvdevClassToString(info->evdev_class));
216 if (info->evdev_class == EvdevClassUnknown) {
217 LOG_ERROR(device, "Couldn't determine evdev class\n");
218 return !Success;
219 }
220
Dennis Kempina9963ba2012-06-08 10:32:23 -0700221 return Success;
222}
223
224int EvdevProbeAbsinfo(EvdevPtr device, size_t key) {
225 struct input_absinfo* absinfo;
226
227 absinfo = &device->info.absinfo[key];
228 if (ioctl(device->fd, EVIOCGABS(key), absinfo) < 0) {
Daniel Kurtz0d097082012-09-07 22:27:51 +0800229 LOG_ERROR(device, "ioctl EVIOCGABS(%zu) failed: %s\n", key,
Dennis Kempina9963ba2012-06-08 10:32:23 -0700230 strerror(errno));
231 return !Success;
232 } else {
233 return Success;
234 }
235}
236
237int EvdevProbeMTSlot(EvdevPtr device, MTSlotInfoPtr req) {
Dennis Kempine9cc2642012-06-25 11:52:24 -0700238 if (ioctl(device->fd, EVIOCGMTSLOTS((sizeof(*req))), req) < 0) {
Dennis Kempina9963ba2012-06-08 10:32:23 -0700239 LOG_ERROR(device, "ioctl EVIOCGMTSLOTS(req.code=%d) failed: %s\n",
Dennis Kempine9cc2642012-06-25 11:52:24 -0700240 req->code, strerror(errno));
Dennis Kempina9963ba2012-06-08 10:32:23 -0700241 return !Success;
242 } else {
243 return Success;
244 }
245}
246
247int EvdevProbeKeyState(EvdevPtr device) {
248 int len = sizeof(device->key_state_bitmask);
249
250 memset(device->key_state_bitmask, 0, len);
251 if (ioctl(device->fd, EVIOCGKEY(len), device->key_state_bitmask) < 0) {
252 LOG_ERROR(device, "ioctl EVIOCGKEY failed: %s\n", strerror(errno));
253 return !Success;
254 } else {
255 return Success;
256 }
257}
258
259int EvdevEnableMonotonic(EvdevPtr device) {
260 unsigned int clk = CLOCK_MONOTONIC;
261 return (ioctl(device->fd, EVIOCSCLOCKID, &clk) == 0) ? Success : !Success;
262}
263
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800264#endif // #ifndef EVDEV_HOLLOW
265
266/*
267 * Check if the device is a single-pressure one which reports ABS_PRESSURE only.
268 */
269int EvdevIsSinglePressureDevice(EvdevPtr device) {
270 EvdevInfoPtr info = &device->info;
271
272 return (!TestBit(ABS_MT_PRESSURE, info->abs_bitmask) &&
273 TestBit(ABS_PRESSURE, info->abs_bitmask));
274}
275
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800276int EvdevWriteInfoToFile(FILE* fp, const EvdevInfoPtr info) {
277 int ret;
278
279 ret = fprintf(fp, "# device: %s\n", info->name);
280 if (ret <= 0)
281 return ret;
282
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800283 ret = EvdevWriteBitmask(fp, "bit", info->bitmask, sizeof(info->bitmask));
284 if (ret <= 0)
285 return ret;
286
287 ret = EvdevWriteBitmask(fp, "key", info->key_bitmask,
288 sizeof(info->key_bitmask));
289 if (ret <= 0)
290 return ret;
291
292 ret = EvdevWriteBitmask(fp, "rel", info->rel_bitmask,
293 sizeof(info->rel_bitmask));
294 if (ret <= 0)
295 return ret;
296
297 ret = EvdevWriteBitmask(fp, "abs", info->abs_bitmask,
298 sizeof(info->abs_bitmask));
299 if (ret <= 0)
300 return ret;
301
302 ret = EvdevWriteBitmask(fp, "led", info->led_bitmask,
303 sizeof(info->led_bitmask));
304 if (ret <= 0)
305 return ret;
306
307 ret = EvdevWriteBitmask(fp, "prp", info->prop_bitmask,
308 sizeof(info->prop_bitmask));
309 if (ret <= 0)
310 return ret;
311
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800312 // when reading the log we need to know which absinfos
313 // exist which is stored in the abs bitmask.
314 // so we have to write absinfo after the bitmasks.
315 for (int i = ABS_X; i <= ABS_MAX; i++) {
316 if (TestBit(i, info->abs_bitmask)) {
317 struct input_absinfo* abs = &info->absinfo[i];
318 ret = fprintf(fp, "# absinfo: %d %d %d %d %d %d\n",
319 i, abs->minimum, abs->maximum,
320 abs->fuzz, abs->flat, abs->resolution);
321 if (ret <= 0)
322 return ret;
323 }
324 }
325
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800326 return 1;
327}
328
329int EvdevWriteEventToFile(FILE* fp, const struct input_event* ev) {
Mike Frysinger7abec9e2012-12-23 13:00:24 -0500330 return fprintf(fp, "E: %ld.%06u %04x %04x %d\n",
331 (long)ev->time.tv_sec, (unsigned)ev->time.tv_usec,
Dennis Kempin69c91ed2013-02-12 14:53:10 -0800332 ev->type, ev->code, ev->value);
333}
334
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800335int EvdevReadInfoFromFile(FILE* fp, EvdevInfoPtr info) {
336 int ret;
337
Dennis Kempinbb4375b2013-03-28 12:02:54 -0700338 ret = fscanf(fp, "# device: %1024[^\n]\n", info->name);
Dennis Kempin690b3b42013-05-29 11:45:32 -0700339 if (ret <= 0)
Dennis Kempin9ec1b7c2013-02-19 14:28:48 -0800340 return ret;
341
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;
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800507
508 if (TestBit(REL_X, info->rel_bitmask) &&
Che-Liang Chiou7e45cb22013-01-15 14:57:13 -0800509 TestBit(REL_Y, info->rel_bitmask)) {
510 if (TestBit(ABS_MT_POSITION_X, info->abs_bitmask) &&
511 TestBit(ABS_MT_POSITION_Y, info->abs_bitmask))
512 return EvdevClassMultitouchMouse;
513 else
514 return EvdevClassMouse;
515 }
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800516
517 if (TestBit(ABS_X, info->abs_bitmask) &&
518 TestBit(ABS_Y, info->abs_bitmask)) {
519
520 if (TestBit(BTN_TOOL_PEN, info->key_bitmask) ||
521 TestBit(BTN_STYLUS, info->key_bitmask) ||
522 TestBit(BTN_STYLUS2, info->key_bitmask))
523 return EvdevClassTablet;
524
525 if (TestBit(ABS_PRESSURE, info->abs_bitmask) ||
526 TestBit(BTN_TOUCH, info->key_bitmask)) {
527 if (TestBit(BTN_LEFT, info->key_bitmask) ||
528 TestBit(BTN_MIDDLE, info->key_bitmask) ||
529 TestBit(BTN_RIGHT, info->key_bitmask) ||
530 TestBit(BTN_TOOL_FINGER, info->key_bitmask))
531 return EvdevClassTouchpad;
532 else
533 return EvdevClassTouchscreen;
534 }
535
536 /* Some touchscreens use BTN_LEFT rather than BTN_TOUCH */
537 if (TestBit(BTN_LEFT, info->key_bitmask))
538 return EvdevClassTouchscreen;
539 }
540
Andrew de los Reyesa63c82a2014-04-03 16:11:26 -0700541 for (bit = 0; bit < BTN_MISC; bit++)
542 if (TestBit(bit, info->key_bitmask))
543 return EvdevClassKeyboard;
544
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800545 return EvdevClassUnknown;
546}
547
548static const char* EvdevClassToString(EvdevClass cls) {
549 switch (cls) {
550 case EvdevClassUnknown: return "EvdevClassUnknown";
551 case EvdevClassKeyboard: return "EvdevClassKeyboard";
552 case EvdevClassMouse: return "EvdevClassMouse";
Che-Liang Chiou7e45cb22013-01-15 14:57:13 -0800553 case EvdevClassMultitouchMouse: return "EvdevClassMultitouchMouse";
Che-Liang Chioub9fc9732012-11-26 15:49:02 -0800554 case EvdevClassTablet: return "EvdevClassTablet";
555 case EvdevClassTouchpad: return "EvdevClassTouchpad";
556 case EvdevClassTouchscreen: return "EvdevClassTouchscreen";
557 }
558 return "Unhandled Evdev Class";
559}
Dennis Kempin0e265392013-02-22 12:59:03 -0800560
561#endif // #ifndef EVDEV_HOLLOW