You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							96 lines
						
					
					
						
							2.1 KiB
						
					
					
				
			
		
		
	
	
							96 lines
						
					
					
						
							2.1 KiB
						
					
					
				| #include "selfdrive/common/touch.h"
 | |
| 
 | |
| #include <assert.h>
 | |
| #include <dirent.h>
 | |
| #include <fcntl.h>
 | |
| #include <linux/input.h>
 | |
| #include <stdbool.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <sys/poll.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| /* this macro is used to tell if "bit" is set in "array"
 | |
|  * it selects a byte from the array, and does a boolean AND
 | |
|  * operation with a byte that only has the relevant bit set.
 | |
|  * eg. to check for the 12th bit, we do (array[1] & 1<<4)
 | |
|  */
 | |
| #define test_bit(bit, array)    (array[bit/8] & (1<<(bit%8)))
 | |
| 
 | |
| static int find_dev() {
 | |
|   int err;
 | |
| 
 | |
|   int ret = -1;
 | |
| 
 | |
|   DIR *dir = opendir("/dev/input");
 | |
|   assert(dir);
 | |
|   struct dirent* de = NULL;
 | |
|   while ((de = readdir(dir))) {
 | |
|     if (strncmp(de->d_name, "event", 5)) continue;
 | |
| 
 | |
|     int fd = openat(dirfd(dir), de->d_name, O_RDONLY);
 | |
|     assert(fd >= 0);
 | |
| 
 | |
|     unsigned char ev_bits[KEY_MAX / 8 + 1];
 | |
|     err = ioctl(fd, EVIOCGBIT(EV_ABS, sizeof(ev_bits)), ev_bits);
 | |
|     assert(err >= 0);
 | |
| 
 | |
|     if (test_bit(ABS_MT_POSITION_X, ev_bits) && test_bit(ABS_MT_POSITION_Y, ev_bits)) {
 | |
|       ret = fd;
 | |
|       break;
 | |
|     }
 | |
|     close(fd);
 | |
|   }
 | |
|   closedir(dir);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| void touch_init(TouchState *s) {
 | |
|   s->fd = find_dev();
 | |
|   assert(s->fd >= 0);
 | |
| }
 | |
| 
 | |
| int touch_poll(TouchState *s, int* out_x, int* out_y, int timeout) {
 | |
|   assert(out_x && out_y);
 | |
|   bool up = false;
 | |
|   while (true) {
 | |
|     struct pollfd polls[] = {{
 | |
|       .fd = s->fd,
 | |
|       .events = POLLIN,
 | |
|     }};
 | |
|     int err = poll(polls, 1, timeout);
 | |
|     if (err < 0) {
 | |
|       return -1;
 | |
|     }
 | |
|     if (!(polls[0].revents & POLLIN)) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     struct input_event event;
 | |
|     err = read(polls[0].fd, &event, sizeof(event));
 | |
|     if (err < sizeof(event)) {
 | |
|       return -1;
 | |
|     }
 | |
| 
 | |
|     switch (event.type) {
 | |
|     case EV_ABS:
 | |
|       if (event.code == ABS_MT_POSITION_X) {
 | |
|         s->last_x = event.value;
 | |
|       } else if (event.code == ABS_MT_POSITION_Y) {
 | |
|         s->last_y = event.value;
 | |
|       } else if (event.code == ABS_MT_TRACKING_ID && event.value != -1) {
 | |
|         up = true;
 | |
|       }
 | |
|       break;
 | |
|     default:
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   if (up) {
 | |
|     // adjust for flippening
 | |
|     *out_x = s->last_y;
 | |
|     *out_y = 1080 - s->last_x;
 | |
|   }
 | |
|   return up;
 | |
| }
 | |
| 
 |