/* in_evdev.c - linux evdev interface support Copyright (C) 2015-2018 mittorn 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 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. */ #include "platform/platform.h" #ifdef XASH_USE_EVDEV #include "common.h" #include "input.h" #include "client.h" #include <fcntl.h> #include <errno.h> #include <linux/input.h> #include <dirent.h> #define MAX_EVDEV_DEVICES 5 struct evdev_s { int initialized, devices; int fds[MAX_EVDEV_DEVICES]; string paths[MAX_EVDEV_DEVICES]; qboolean grab; double grabtime; float x, y; qboolean chars; qboolean shift; } evdev; static convar_t *evdev_keydebug; static int KeycodeFromEvdev(int keycode, int value) { switch (keycode) { case KEY_0: return '0'; case KEY_1: return '1'; case KEY_2: return '2'; case KEY_3: return '3'; case KEY_4: return '4'; case KEY_5: return '5'; case KEY_6: return '6'; case KEY_7: return '7'; case KEY_8: return '8'; case KEY_9: return '9'; case KEY_BACKSPACE: return K_BACKSPACE; case KEY_ENTER: return K_ENTER; case KEY_ESC: return K_ESCAPE; case KEY_KP0: return K_KP_INS; case KEY_KP1: return K_KP_END; case KEY_KP2: return K_KP_DOWNARROW; case KEY_KP3: return K_KP_PGDN; case KEY_KP4: return K_KP_LEFTARROW; case KEY_KP5: return K_KP_5; case KEY_KP6: return K_KP_RIGHTARROW; case KEY_KP7: return K_KP_HOME; case KEY_KP8: return K_KP_UPARROW; case KEY_KP9: return K_KP_PGUP; case KEY_KPDOT: return K_KP_DEL; case KEY_KPENTER: return K_KP_ENTER; case KEY_Q: return 'q'; case KEY_W: return 'w'; case KEY_E: return 'e'; case KEY_R: return 'r'; case KEY_T: return 't'; case KEY_Y: return 'y'; case KEY_U: return 'u'; case KEY_I: return 'i'; case KEY_O: return 'o'; case KEY_P: return 'p'; case KEY_A: return 'a'; case KEY_S: return 's'; case KEY_D: return 'd'; case KEY_F: return 'f'; case KEY_G: return 'g'; case KEY_H: return 'h'; case KEY_J: return 'j'; case KEY_K: return 'k'; case KEY_L: return 'l'; case KEY_Z: return 'z'; case KEY_X: return 'x'; case KEY_C: return 'c'; case KEY_V: return 'v'; case KEY_B: return 'b'; case KEY_N: return 'n'; case KEY_M: return 'm'; case KEY_LEFTBRACE: return '['; case KEY_RIGHTBRACE: return ']'; case KEY_MINUS: return '-'; case KEY_EQUAL: return '='; case KEY_TAB: return K_TAB; case KEY_SEMICOLON: return ';'; case KEY_APOSTROPHE: return '\''; case KEY_GRAVE: return '`'; case KEY_BACKSLASH: return '\\'; case KEY_COMMA: return ','; case KEY_DOT: return '.'; case KEY_SLASH: return '/'; case KEY_SPACE: return K_SPACE; case KEY_KPASTERISK: return '*'; case KEY_RIGHTCTRL: case KEY_LEFTCTRL: return K_CTRL; case KEY_RIGHTSHIFT: case KEY_LEFTSHIFT: return K_SHIFT; case KEY_LEFT: return K_LEFTARROW; case KEY_RIGHT: return K_RIGHTARROW; case KEY_UP: return K_UPARROW; case KEY_DOWN: return K_DOWNARROW; case BTN_LEFT: return K_MOUSE1; case BTN_RIGHT: return K_MOUSE2; case BTN_MIDDLE: return K_MOUSE3; case KEY_POWER: return K_ESCAPE; case KEY_VOLUMEDOWN: return K_PGDN; case KEY_VOLUMEUP: return K_PGUP; case KEY_PLAYPAUSE: return K_ENTER; default: break; } return 0; } static void Evdev_CheckPermissions( void ) { #ifdef __ANDROID__ system( "su 0 chmod 664 /dev/input/event*" ); #endif } void Evdev_Setup( void ) { if( evdev.initialized ) return; #ifdef __ANDROID__ system( "su 0 supolicy --live \"allow appdomain input_device dir { ioctl read getattr search open }\" \"allow appdomain input_device chr_file { ioctl read write getattr lock append open }\"" ); system( "su 0 setenforce permissive" ); #endif evdev.initialized = true; } #define EV_HASBIT( x, y ) ( x[y / 32] & (1 << y % 32) ) void Evdev_Autodetect_f( void ) { int i; DIR *dir; struct dirent *entry; Evdev_Setup(); Evdev_CheckPermissions(); if( !( dir = opendir( "/dev/input" ) ) ) return; while( ( entry = readdir( dir ) ) ) { int fd; string path; uint types[EV_MAX] = {0}; uint codes[( KEY_MAX - 1 ) / 32 + 1] = {0}; qboolean hasbtnmouse; if( evdev.devices >= MAX_EVDEV_DEVICES ) continue; Q_snprintf( path, MAX_STRING, "/dev/input/%s", entry->d_name ); for( i = 0; i < evdev.devices; i++ ) if( !Q_strncmp( evdev.paths[i], path, MAX_STRING ) ) goto next; if( Q_strncmp( entry->d_name, "event", 5 ) ) continue; fd = open( path, O_RDONLY | O_NONBLOCK ); if( fd == -1 ) continue; ioctl( fd, EVIOCGBIT( 0, EV_MAX ), types ); if( !EV_HASBIT( types, EV_KEY ) ) goto close; ioctl( fd, EVIOCGBIT( EV_KEY, KEY_MAX ), codes ); if( EV_HASBIT( codes, KEY_LEFTCTRL ) && EV_HASBIT( codes, KEY_SPACE ) ) goto open; if( !EV_HASBIT( codes, BTN_MOUSE ) ) goto close; if( EV_HASBIT( types, EV_REL ) ) { memset( codes, 0, sizeof( codes ) ); ioctl( fd, EVIOCGBIT( EV_REL, KEY_MAX ), codes ); if( !EV_HASBIT( codes, REL_X ) ) goto close; if( !EV_HASBIT( codes, REL_Y ) ) goto close; if( !EV_HASBIT( codes, REL_WHEEL ) ) goto close; goto open; } goto close; open: Q_strncpy( evdev.paths[evdev.devices], path, MAX_STRING ); evdev.fds[evdev.devices++] = fd; Con_Printf( "Opened device %s\n", path ); #if XASH_INPUT == INPUT_EVDEV if( Sys_CheckParm( "-grab" ) ) ioctl( evdev.fds[i], EVIOCGRAB, (void*) 1 ); #endif goto next; close: close( fd ); next: continue; } closedir( dir ); } /* =========== Evdev_OpenDevice For shitty systems that cannot provide relative mouse axes =========== */ void Evdev_OpenDevice ( const char *path ) { int ret, i; if ( evdev.devices >= MAX_EVDEV_DEVICES ) { Con_Printf( "Only %d devices supported!\n", MAX_EVDEV_DEVICES ); return; } Evdev_Setup(); Evdev_CheckPermissions(); // use root to grant access to evdev for( i = 0; i < evdev.devices; i++ ) { if( !Q_strncmp( evdev.paths[i], path, MAX_STRING ) ) { Con_Printf( "device %s already open!\n", path ); return; } } ret = open( path, O_RDONLY | O_NONBLOCK ); if( ret < 0 ) { Con_Reportf( S_ERROR "Could not open input device %s: %s\n", path, strerror( errno ) ); return; } Con_Printf( "Input device #%d: %s opened sucessfully\n", evdev.devices, path ); evdev.fds[evdev.devices] = ret; Q_strncpy( evdev.paths[evdev.devices++], path, MAX_STRING ); #if XASH_INPUT == INPUT_EVDEV if( Sys_CheckParm( "-grab" ) ) ioctl( evdev.fds[i], EVIOCGRAB, (void*) 1 ); #endif } void Evdev_OpenDevice_f( void ) { if( Cmd_Argc() < 2 ) Con_Printf( S_USAGE "evdev_opendevice <path>\n" ); Evdev_OpenDevice( Cmd_Argv( 1 ) ); } /* =========== Evdev_CloseDevice_f =========== */ void Evdev_CloseDevice_f ( void ) { uint i; const char *arg; if( Cmd_Argc() < 2 ) return; arg = Cmd_Argv( 1 ); if( Q_isdigit( arg ) ) i = Q_atoi( arg ); else for( i = 0; i < evdev.devices; i++ ) if( !Q_strncmp( evdev.paths[i], arg, MAX_STRING ) ) break; if( i >= evdev.devices ) { Con_Printf( "Device %s is not open\n", arg ); return; } close( evdev.fds[i] ); evdev.devices--; Con_Printf( "Device %s closed successfully\n", evdev.paths[i] ); for( ; i < evdev.devices; i++ ) { Q_strncpy( evdev.paths[i], evdev.paths[i+1], MAX_STRING ); evdev.fds[i] = evdev.fds[i+1]; } } void IN_EvdevFrame ( void ) { int dx = 0, dy = 0, i; for( i = 0; i < evdev.devices; i++ ) { struct input_event ev; while ( read( evdev.fds[i], &ev, 16) == 16 ) { if ( ev.type == EV_REL ) { switch ( ev.code ) { case REL_X: dx += ev.value; break; case REL_Y: dy += ev.value; break; case REL_WHEEL: if( ev.value > 0) { Key_Event( K_MWHEELDOWN, 1 ); Key_Event( K_MWHEELDOWN, 0 ); } else { Key_Event( K_MWHEELUP, 1 ); Key_Event( K_MWHEELUP, 0 ); } break; } } else if ( ( ev.type == EV_KEY ) && (cls.key_dest == key_game || XASH_INPUT == INPUT_EVDEV ) ) { int key = KeycodeFromEvdev( ev.code, ev.value ); if( CVAR_TO_BOOL(evdev_keydebug) ) Con_Printf( "key %d %d %d\n", ev.code, key, ev.value ); Key_Event( key , ev.value ); if( evdev.chars && ev.value ) { if( key >= 32 && key < 127 ) { if( evdev.shift ) { key = Key_ToUpper( key ); } CL_CharEvent( key ); } } if( key == K_SHIFT ) evdev.shift = ev.value; } } if( evdev.grab && evdev.grabtime <= host.realtime ) { ioctl( evdev.fds[i], EVIOCGRAB, (void*) 1 ); Key_ClearStates(); } if( CVAR_TO_BOOL(m_ignore) ) continue; evdev.x += -dx * m_yaw->value; evdev.y += dy * m_pitch->value; } if( evdev.grabtime <= host.realtime ) evdev.grab = false; } void Evdev_SetGrab( qboolean grab ) { // grab only if evdev is secondary input source #if XASH_INPUT != INPUT_EVDEV int i; if( grab ) { Key_Event( K_ESCAPE, 0 ); //Do not leave ESC down evdev.grabtime = host.realtime + 0.5; Key_ClearStates(); } else { for( i = 0; i < evdev.devices; i++ ) ioctl( evdev.fds[i], EVIOCGRAB, (void*) 0 ); } evdev.grab = grab; #endif } void IN_EvdevMove( float *yaw, float *pitch ) { *yaw += evdev.x; *pitch += evdev.y; evdev.x = evdev.y = 0.0f; } #if XASH_INPUT == INPUT_EVDEV void Platform_EnableTextInput( qboolean enable ) { evdev.chars = enable; evdev.shift = false; } #endif void Evdev_Init( void ) { Cmd_AddCommand ("evdev_open", Evdev_OpenDevice_f, "Open event device"); Cmd_AddCommand ("evdev_close", Evdev_CloseDevice_f, "Close event device"); Cmd_AddCommand ("evdev_autodetect", Evdev_Autodetect_f, "Automaticly open mouses and keyboards"); #if XASH_INPUT == INPUT_EVDEV Evdev_Autodetect_f(); #endif } void Evdev_Shutdown( void ) { int i; Cmd_RemoveCommand( "evdev_open" ); Cmd_RemoveCommand( "evdev_close" ); Cmd_RemoveCommand( "evdev_autodetect" ); evdev_keydebug = Cvar_Get( "evdev_keydebug", "0", 0, "print key events to console" ); for( i = 0; i < evdev.devices; i++ ) { ioctl( evdev.fds[i], EVIOCGRAB, (void*) 0 ); close( evdev.fds[i] ); evdev.fds[i] = -1; } evdev.devices = 0; } #endif // XASH_USE_EVDEV