refactor dependencies
[ginge.git] / loader / host_pnd.c
CommitLineData
3d295a9f 1// vim:shiftwidth=2:expandtab
2#define _GNU_SOURCE
3#include <stdio.h>
4#include <string.h>
5#include <sys/types.h>
6#include <sys/stat.h>
7#include <fcntl.h>
8#include <unistd.h>
9#include <sys/ioctl.h>
10#include <errno.h>
11#include <linux/input.h>
12
13#include "header.h"
7fd42181 14#include "realfuncs.h"
3d295a9f 15
6720e4e4 16static int ifds[2] = { -1, -1 };
3d295a9f 17static int keystate;
18
4d045184 19int host_init(void)
3d295a9f 20{
21 char buff[64];
6720e4e4 22 int i, ifd, ret;
3d295a9f 23
6720e4e4 24 for (ifd = -1, i = 0; ifds[0] == -1 || ifds[1] == -1; i++) {
3d295a9f 25 snprintf(buff, sizeof(buff), "/dev/input/event%i", i);
4d045184 26 ifd = open(buff, O_RDONLY | O_NONBLOCK);
3d295a9f 27 if (ifd == -1)
28 break;
29
6720e4e4 30 ret = ioctl(ifd, EVIOCGNAME(sizeof(buff)), buff);
31 if (ret < 0)
3d295a9f 32 break;
6720e4e4 33
34 if (strcasestr(buff, "gpio") != NULL) {
35 ifds[0] = ifd;
36 continue;
37 }
38 if (strcasestr(buff, "keypad") != NULL) {
39 ifds[1] = ifd;
40 continue;
41 }
3d295a9f 42 close(ifd);
43 }
44
6720e4e4 45 if (ifds[0] < 0)
46 fprintf(stderr, PFX "missing buttons\n");
47 if (ifds[1] < 0)
48 fprintf(stderr, PFX "missing keypad\n");
4d045184 49
50 return 0;
3d295a9f 51}
52
53static const struct {
54 unsigned short key;
55 unsigned short btn;
56} key_map[] = {
57 { KEY_LEFT, GP2X_LEFT },
58 { KEY_RIGHT, GP2X_RIGHT },
59 { KEY_UP, GP2X_UP },
60 { KEY_DOWN, GP2X_DOWN },
61 { KEY_PAGEUP, GP2X_Y },
62 { BTN_BASE, GP2X_Y },
63 { KEY_END, GP2X_B },
64 { BTN_BASE2, GP2X_B },
65 { KEY_PAGEDOWN, GP2X_X },
66 { BTN_BASE3, GP2X_X },
67 { KEY_HOME, GP2X_A },
68 { BTN_BASE4, GP2X_A },
69 { KEY_RIGHTSHIFT, GP2X_L },
70 { BTN_TL, GP2X_L },
71 { KEY_RIGHTCTRL, GP2X_R },
72 { BTN_TR, GP2X_R },
73 { KEY_LEFTALT, GP2X_START },
74 { BTN_START, GP2X_START },
75 { KEY_LEFTCTRL, GP2X_SELECT },
76 { BTN_SELECT, GP2X_SELECT },
6720e4e4 77 { KEY_COMMA, GP2X_VOL_DOWN },
78 { KEY_DOT, GP2X_VOL_UP },
7000b522 79 { KEY_1, GP2X_PUSH },
3d295a9f 80};
81
82int host_read_btns(void)
83{
84 struct input_event ev;
85 int i, ret;
86
3d295a9f 87 while (1)
88 {
6720e4e4 89 ret = read(ifds[0], &ev, sizeof(ev));
3d295a9f 90 if (ret < (int) sizeof(ev)) {
91 if (errno != EAGAIN && errno != EWOULDBLOCK)
6720e4e4 92 perror(PFX "read error");
3d295a9f 93
6720e4e4 94 ret = read(ifds[1], &ev, sizeof(ev));
95 if (ret < (int) sizeof(ev))
96 if (errno != EAGAIN && errno != EWOULDBLOCK)
97 perror(PFX "read error");
3d295a9f 98 }
99
6720e4e4 100 if (ret < (int) sizeof(ev))
101 return keystate;
102
3d295a9f 103 if (ev.type != EV_KEY)
104 continue;
105
7000b522 106 if (ev.code == KEY_Q && ev.value) {
107 // exit() might not be enough because loader and app data is out of sync,
108 // and other threads (which are really processes) might not exit properly.
109 system("killall ginge_sloader");
110 usleep(300000);
111 system("killall -9 ginge_sloader");
112 exit(1);
113 }
114
3d295a9f 115 for (i = 0; i < sizeof(key_map) / sizeof(key_map[0]); i++) {
116 if (key_map[i].key != ev.code)
117 continue;
118 if (ev.value)
119 keystate |= (1 << key_map[i].btn);
120 else
121 keystate &= ~(1 << key_map[i].btn);
122 break;
123 }
124 }
125
126 return keystate;
127}
128