/*************************************************************************** * __________ __ ___ * Open \______ \ ____ ____ | | _\_ |__ _______ ___ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ * \/ \/ \/ \/ \/ * * Copyright (C) 2017 Marcin Bukat * Copyright (C) 2020 Solomon Peachy * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY * KIND, either express or implied. * ****************************************************************************/ #include //#include #include #include #include #include #include #include #include #include "sysfs.h" #include "button.h" #include "button-target.h" #include "panic.h" #include "kernel.h" #include "backlight.h" #include "backlight-target.h" #include "erosqlinux_codec.h" #define NR_POLL_DESC 2 static struct pollfd poll_fds[NR_POLL_DESC]; static int button_map(int keycode) { switch(keycode) { case KEY_POWER: return BUTTON_POWER; case KEY_MENU: return BUTTON_MENU; case KEY_BACK: return BUTTON_BACK; case KEY_NEXTSONG: return BUTTON_PREV; case KEY_PREVIOUSSONG: return BUTTON_NEXT; // Yes, backwards! case KEY_PLAYPAUSE: return BUTTON_PLAY; case KEY_LEFT: return BUTTON_SCROLL_BACK; case KEY_RIGHT: return BUTTON_SCROLL_FWD; case KEY_VOLUMEUP: return BUTTON_VOL_UP; case KEY_VOLUMEDOWN: return BUTTON_VOL_DOWN; default: return 0; } } void button_init_device(void) { const char * const input_devs[NR_POLL_DESC] = { "/dev/input/event0", // Rotary encoder "/dev/input/event1" // Keys }; for(int i = 0; i < NR_POLL_DESC; i++) { int fd = open(input_devs[i], O_RDONLY | O_CLOEXEC); if(fd < 0) { panicf("Cannot open input device: %s (%d)\n", input_devs[i], errno); } poll_fds[i].fd = fd; poll_fds[i].events = POLLIN; poll_fds[i].revents = 0; } } int button_read_device(void) { static int button_bitmap = 0; struct input_event event; // FIXME TODO: Make this work via HAVE_SCROLL_WHEEL instead /* Wheel gives us press+release back to back, clear them after time elapses */ static long last_tick = 0; if (button_bitmap & (BUTTON_SCROLL_BACK|BUTTON_SCROLL_FWD) && current_tick - last_tick >= 2) { button_bitmap &= ~(BUTTON_SCROLL_BACK|BUTTON_SCROLL_FWD); } /* check if there are any events pending and process them */ while(poll(poll_fds, NR_POLL_DESC, 0)) { for(int i = 0; i < NR_POLL_DESC; i++) { /* read only if non-blocking */ if(poll_fds[i].revents & POLLIN) { int size = read(poll_fds[i].fd, &event, sizeof(event)); if(size == (int)sizeof(event)) { int keycode = event.code; /* event.value == 1 means press * event.value == 0 means release */ bool press = event.value ? true : false; /* map linux event code to rockbox button bitmap */ if(press) { int bmap = button_map(keycode); if (bmap & (BUTTON_SCROLL_BACK|BUTTON_SCROLL_FWD)) last_tick = current_tick; button_bitmap |= bmap; } else { /* Wheel gives us press+release back to back; ignore the release */ int bmap = button_map(keycode) & ~(BUTTON_SCROLL_BACK|BUTTON_SCROLL_FWD); button_bitmap &= ~bmap; } } } } } return button_bitmap; } bool headphones_inserted(void) { #ifdef BOOTLOADER int ps = 0; #else int ps = erosq_get_outputs(); #endif return (ps == 2); } bool lineout_inserted(void) { #ifdef BOOTLOADER int ps = 0; #else int ps = erosq_get_outputs(); #endif return (ps == 1); } void button_close_device(void) { /* close descriptors */ for(int i = 0; i < NR_POLL_DESC; i++) { close(poll_fds[i].fd); } }