recoveryにおけるeventイベントの実装
ref : /bootable/recovery/events.c /bootable/recovery/ui.cpp /bootable/recovery/recovery.cpp /bootable/recovery/roots.cpp
UI関連の場合、イベント(key,touch)との関係になるのが一般的です
UI初期化:
#define MAX_DEVICES 16///dev/inputディレクトリの下のすべてのファイルの記述子を取得しますが、最大16個を超えてはいけません.
//これはfactory modeのminiuiに似ています
//UIが初期化されると、イベントをポーリングする新しいスレッドが起動します
//配布時間、cbはコールバック関数、ev_Initのときにパラメータとして渡される
//この場合は、FDディスクリプタがすべて発生しているわけではありません.少なくとも1つの時間しか発生しません.
//ファイル記述子を読み込み、時間内容をUIマスタースレッドに送る
//イベントがなければ、そのまま戻る
UI関連の場合、イベント(key,touch)との関係になるのが一般的です
UI初期化:
void RecoveryUI::Init() {
ev_init(input_callback, NULL);
pthread_create(&input_t, NULL, input_thread, NULL);
}
#define MAX_DEVICES 16///dev/inputディレクトリの下のすべてのファイルの記述子を取得しますが、最大16個を超えてはいけません.
//これはfactory modeのminiuiに似ています
int ev_init(ev_callback input_cb, void *data)
{
DIR *dir;
struct dirent *de;
int fd;
dir = opendir("/dev/input");
if(dir != 0) {
while((de = readdir(dir))) {
unsigned long ev_bits[BITS_TO_LONGS(EV_MAX)];
if(strncmp(de->d_name,"event",5)) continue;
fd = openat(dirfd(dir), de->d_name, O_RDONLY);
if(fd < 0) continue;
/* read the evbits of the input device */
if (ioctl(fd, EVIOCGBIT(0, sizeof(ev_bits)), ev_bits) < 0) {
close(fd);
continue;
}
/* TODO: add ability to specify event masks. For now, just assume
* that only EV_KEY and EV_REL event types are ever needed. */
if (!test_bit(EV_KEY, ev_bits) && !test_bit(EV_REL, ev_bits)) {
close(fd);
continue;
}
ev_fds[ev_count].fd = fd;
ev_fds[ev_count].events = POLLIN;
ev_fdinfo[ev_count].cb = input_cb;
ev_fdinfo[ev_count].data = data;
ev_count++;
ev_dev_count++;
if(ev_dev_count == MAX_DEVICES) break;
}
}
return 0;
}
//UIが初期化されると、イベントをポーリングする新しいスレッドが起動します
pthread_create(&input_t, NULL, input_thread, NULL);
// Reads input events, handles special hot keys, and adds to the key queue.
void* RecoveryUI::input_thread(void *cookie)
{
for (;;) {
if (!ev_wait(-1))
ev_dispatch();
}
return NULL;
}
//配布時間、cbはコールバック関数、ev_Initのときにパラメータとして渡される
//この場合は、FDディスクリプタがすべて発生しているわけではありません.少なくとも1つの時間しか発生しません.
void ev_dispatch(void)
{
unsigned n;
int ret;
for (n = 0; n < ev_count; n++) {
ev_callback cb = ev_fdinfo[n].cb;
if (cb && (ev_fds[n].revents & ev_fds[n].events))
cb(ev_fds[n].fd, ev_fds[n].revents, ev_fdinfo[n].data);
}
}
//ファイル記述子を読み込み、時間内容をUIマスタースレッドに送る
//イベントがなければ、そのまま戻る
int RecoveryUI::input_callback(int fd, short revents, void* data)
{
struct input_event ev;
int ret;
ret = ev_get_input(fd, revents, &ev);
if (ret)
return -1;
.......
if (ev.type == EV_KEY && ev.code <= KEY_MAX)
self->process_key(ev.code, ev.value);
return 0;
}