#include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdint.h> #include <dirent.h> #include <fcntl.h> #include <sys/ioctl.h> #include <sys/inotify.h> #include <sys/limits.h> #include <sys/poll.h> #include <linux/input.h> #include <errno.h> #include <cutils/log.h> static struct pollfd *ufds; static char **device_names; static int nfds; static int open_device(const char *device) { int version; int fd; struct pollfd *new_ufds; char **new_device_names; char name[80]; char location[80]; char idstr[80]; struct input_id id; fd = open(device, O_RDWR); if(fd < 0) { return -1; } if(ioctl(fd, EVIOCGVERSION, &version)) { return -1; } if(ioctl(fd, EVIOCGID, &id)) { return -1; } name[sizeof(name) - 1] = '\0'; location[sizeof(location) - 1] = '\0'; idstr[sizeof(idstr) - 1] = '\0'; if(ioctl(fd, EVIOCGNAME(sizeof(name) - 1), &name) < 1) { //fprintf(stderr, "could not get device name for %s, %s\n", device, strerror(errno)); name[0] = '\0'; } if(ioctl(fd, EVIOCGPHYS(sizeof(location) - 1), &location) < 1) { //fprintf(stderr, "could not get location for %s, %s\n", device, strerror(errno)); location[0] = '\0'; } if(ioctl(fd, EVIOCGUNIQ(sizeof(idstr) - 1), &idstr) < 1) { //fprintf(stderr, "could not get idstring for %s, %s\n", device, strerror(errno)); idstr[0] = '\0'; } new_ufds = realloc(ufds, sizeof(ufds[0]) * (nfds + 1)); if(new_ufds == NULL) { fprintf(stderr, "out of memory\n"); return -1; } ufds = new_ufds; new_device_names = realloc(device_names, sizeof(device_names[0]) * (nfds + 1)); if(new_device_names == NULL) { fprintf(stderr, "out of memory\n"); return -1; } device_names = new_device_names; ufds[nfds].fd = fd; ufds[nfds].events = POLLIN; device_names[nfds] = strdup(device); nfds++; return 0; } int close_device(const char *device) { int i; for(i = 1; i < nfds; i++) { if(strcmp(device_names[i], device) == 0) { int count = nfds - i - 1; free(device_names[i]); memmove(device_names + i, device_names + i + 1, sizeof(device_names[0]) * count); memmove(ufds + i, ufds + i + 1, sizeof(ufds[0]) * count); nfds--; return 0; } } return -1; } static int read_notify(const char *dirname, int nfd) { int res; char devname[PATH_MAX]; char *filename; char event_buf[512]; int event_size; int event_pos = 0; struct inotify_event *event; res = read(nfd, event_buf, sizeof(event_buf)); if(res < (int)sizeof(*event)) { if(errno == EINTR) return 0; fprintf(stderr, "could not get event, %s\n", strerror(errno)); return 1; } //printf("got %d bytes of event information\n", res); strcpy(devname, dirname); filename = devname + strlen(devname); *filename++ = '/'; while(res >= (int)sizeof(*event)) { event = (struct inotify_event *)(event_buf + event_pos); //printf("%d: %08x \"%s\"\n", event->wd, event->mask, event->len ? event->name : ""); if(event->len) { strcpy(filename, event->name); if(event->mask & IN_CREATE) { open_device(devname); } else { close_device(devname); } } event_size = sizeof(*event) + event->len; res -= event_size; event_pos += event_size; } return 0; } static int scan_dir(const char *dirname) { char devname[PATH_MAX]; char *filename; DIR *dir; struct dirent *de; dir = opendir(dirname); if(dir == NULL) return -1; strcpy(devname, dirname); filename = devname + strlen(devname); *filename++ = '/'; while((de = readdir(dir))) { if(de->d_name[0] == '.' && (de->d_name[1] == '\0' || (de->d_name[1] == '.' && de->d_name[2] == '\0'))) continue; strcpy(filename, de->d_name); open_device(devname); } closedir(dir); return 0; } int init_getevent() { int res; const char *device_path = "/dev/input"; nfds = 1; ufds = calloc(1, sizeof(ufds[0])); ufds[0].fd = inotify_init(); ufds[0].events = POLLIN; res = inotify_add_watch(ufds[0].fd, device_path, IN_DELETE | IN_CREATE); if(res < 0) { return 1; } res = scan_dir(device_path); if(res < 0) { return 1; } return 0; } void uninit_getevent() { int i; for(i = 0; i < nfds; i++) { close(ufds[i].fd); } free(ufds); ufds = 0; nfds = 0; } int get_event(struct input_event* event, int timeout) { int res; int i; int pollres; const char *device_path = "/dev/input"; while(1) { pollres = poll(ufds, nfds, timeout); if (pollres == 0) { return 1; } if(ufds[0].revents & POLLIN) { read_notify(device_path, ufds[0].fd); } for(i = 1; i < nfds; i++) { if(ufds[i].revents) { if(ufds[i].revents & POLLIN) { res = read(ufds[i].fd, event, sizeof(*event)); if(res < (int)sizeof(event)) { fprintf(stderr, "could not get event\n"); return -1; } return 0; } } } } return 0; }