2011-08-05 04:00:30 +04:00
/*
2015-01-15 14:40:38 +03:00
* expose input properties via udev
2011-08-05 04:00:30 +04:00
*
* Copyright ( C ) 2009 Martin Pitt < martin . pitt @ ubuntu . com >
* Portions Copyright ( C ) 2004 David Zeuthen , < david @ fubar . dk >
2012-11-12 22:36:23 +04:00
* Copyright ( C ) 2011 Kay Sievers < kay @ vrfy . org >
2015-01-15 14:40:38 +03:00
* Copyright ( C ) 2014 Carlos Garnacho < carlosg @ gnome . org >
2015-01-15 14:38:57 +03:00
* Copyright ( C ) 2014 David Herrmann < dh . herrmann @ gmail . com >
2011-08-05 04:00:30 +04:00
*
* 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 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 .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include <stdio.h>
# include <stdlib.h>
# include <stdarg.h>
# include <unistd.h>
# include <string.h>
# include <errno.h>
# include <linux/limits.h>
# include <linux/input.h>
# include "udev.h"
2015-01-15 14:40:38 +03:00
# include "util.h"
2011-08-05 04:00:30 +04:00
/* we must use this kernel-compatible implementation */
# define BITS_PER_LONG (sizeof(unsigned long) * 8)
# define NBITS(x) ((((x)-1) / BITS_PER_LONG)+1)
# define OFF(x) ((x)%BITS_PER_LONG)
# define BIT(x) (1UL<<OFF(x))
# define LONG(x) ((x) / BITS_PER_LONG)
# define test_bit(bit, array) ((array[LONG(bit)] >> OFF(bit)) & 1)
2015-01-15 14:40:38 +03:00
static inline int abs_size_mm ( const struct input_absinfo * absinfo ) {
/* Resolution is defined to be in units/mm for ABS_X/Y */
return ( absinfo - > maximum - absinfo - > minimum ) / absinfo - > resolution ;
}
static void extract_info ( struct udev_device * dev , const char * devpath , bool test ) {
char width [ DECIMAL_STR_MAX ( int ) ] , height [ DECIMAL_STR_MAX ( int ) ] ;
struct input_absinfo xabsinfo = { } , yabsinfo = { } ;
_cleanup_close_ int fd = - 1 ;
fd = open ( devpath , O_RDONLY | O_CLOEXEC ) ;
if ( fd < 0 )
return ;
if ( ioctl ( fd , EVIOCGABS ( ABS_X ) , & xabsinfo ) < 0 | |
ioctl ( fd , EVIOCGABS ( ABS_Y ) , & yabsinfo ) < 0 )
return ;
if ( xabsinfo . resolution < = 0 | | yabsinfo . resolution < = 0 )
return ;
snprintf ( width , sizeof ( width ) , " %d " , abs_size_mm ( & xabsinfo ) ) ;
snprintf ( height , sizeof ( height ) , " %d " , abs_size_mm ( & yabsinfo ) ) ;
udev_builtin_add_property ( dev , test , " ID_INPUT_WIDTH_MM " , width ) ;
udev_builtin_add_property ( dev , test , " ID_INPUT_HEIGHT_MM " , height ) ;
}
2011-08-05 04:00:30 +04:00
/*
* Read a capability attribute and return bitmask .
* @ param dev udev_device
* @ param attr sysfs attribute name ( e . g . " capabilities/key " )
* @ param bitmask : Output array which has a sizeof of bitmask_size
*/
static void get_cap_mask ( struct udev_device * dev ,
2012-01-10 04:34:15 +04:00
struct udev_device * pdev , const char * attr ,
unsigned long * bitmask , size_t bitmask_size ,
2014-07-29 17:47:41 +04:00
bool test ) {
2015-01-15 14:38:57 +03:00
const char * v ;
2012-01-10 04:34:15 +04:00
char text [ 4096 ] ;
unsigned i ;
char * word ;
unsigned long val ;
2015-01-15 14:38:57 +03:00
v = udev_device_get_sysattr_value ( pdev , attr ) ;
if ( ! v )
v = " " ;
snprintf ( text , sizeof ( text ) , " %s " , v ) ;
2013-12-24 19:39:37 +04:00
log_debug ( " %s raw kernel attribute: %s " , attr , text ) ;
2012-01-10 04:34:15 +04:00
2014-01-31 09:51:32 +04:00
memzero ( bitmask , bitmask_size ) ;
2012-01-10 04:34:15 +04:00
i = 0 ;
while ( ( word = strrchr ( text , ' ' ) ) ! = NULL ) {
val = strtoul ( word + 1 , NULL , 16 ) ;
if ( i < bitmask_size / sizeof ( unsigned long ) )
bitmask [ i ] = val ;
else
2013-12-24 19:39:37 +04:00
log_debug ( " ignoring %s block %lX which is larger than maximum size " , attr , val ) ;
2012-01-10 04:34:15 +04:00
* word = ' \0 ' ;
+ + i ;
}
val = strtoul ( text , NULL , 16 ) ;
if ( i < bitmask_size / sizeof ( unsigned long ) )
bitmask [ i ] = val ;
else
2013-12-24 19:39:37 +04:00
log_debug ( " ignoring %s block %lX which is larger than maximum size " , attr , val ) ;
2012-01-10 04:34:15 +04:00
if ( test ) {
/* printf pattern with the right unsigned long number of hex chars */
2015-01-21 06:22:15 +03:00
snprintf ( text , sizeof ( text ) , " bit %%4u: %%0%zulX \n " , 2 * sizeof ( unsigned long ) ) ;
2013-12-24 19:39:37 +04:00
log_debug ( " %s decoded bit map: " , attr ) ;
2012-01-10 04:34:15 +04:00
val = bitmask_size / sizeof ( unsigned long ) ;
/* skip over leading zeros */
while ( bitmask [ val - 1 ] = = 0 & & val > 0 )
- - val ;
2014-02-20 21:18:32 +04:00
for ( i = 0 ; i < val ; + + i ) {
DISABLE_WARNING_FORMAT_NONLITERAL ;
2012-04-08 18:06:20 +04:00
log_debug ( text , i * BITS_PER_LONG , bitmask [ i ] ) ;
2014-02-20 21:18:32 +04:00
REENABLE_WARNING ;
}
2012-01-10 04:34:15 +04:00
}
2011-08-05 04:00:30 +04:00
}
/* pointer devices */
2015-04-13 12:15:00 +03:00
static bool test_pointers ( struct udev_device * dev ,
2015-04-14 02:22:50 +03:00
const unsigned long * bitmask_ev ,
const unsigned long * bitmask_abs ,
const unsigned long * bitmask_key ,
const unsigned long * bitmask_rel ,
const unsigned long * bitmask_props ,
bool test ) {
2015-05-29 15:54:52 +03:00
bool has_abs_coordinates = false ;
bool has_rel_coordinates = false ;
bool has_joystick_axes_or_buttons = false ;
bool has_touch = false ;
bool has_3d_coordinates = false ;
bool has_keys = false ;
bool stylus_or_pen = false ;
bool finger_but_no_pen = false ;
bool has_mouse_button = false ;
bool is_mouse = false ;
bool is_touchpad = false ;
bool is_touchscreen = false ;
bool is_tablet = false ;
bool is_joystick = false ;
bool is_accelerometer = false ;
bool is_pointing_stick = false ;
has_keys = test_bit ( EV_KEY , bitmask_ev ) ;
has_abs_coordinates = test_bit ( ABS_X , bitmask_abs ) & & test_bit ( ABS_Y , bitmask_abs ) ;
has_3d_coordinates = has_abs_coordinates & & test_bit ( ABS_Z , bitmask_abs ) ;
is_accelerometer = test_bit ( INPUT_PROP_ACCELEROMETER , bitmask_props ) ;
if ( ! has_keys & & has_3d_coordinates )
is_accelerometer = true ;
if ( is_accelerometer ) {
2015-04-03 13:07:32 +03:00
udev_builtin_add_property ( dev , test , " ID_INPUT_ACCELEROMETER " , " 1 " ) ;
2015-04-13 12:15:00 +03:00
return true ;
2015-04-03 13:07:32 +03:00
}
2015-05-29 15:54:52 +03:00
is_pointing_stick = test_bit ( INPUT_PROP_POINTING_STICK , bitmask_props ) ;
stylus_or_pen = test_bit ( BTN_STYLUS , bitmask_key ) | | test_bit ( BTN_TOOL_PEN , bitmask_key ) ;
finger_but_no_pen = test_bit ( BTN_TOOL_FINGER , bitmask_key ) & & ! test_bit ( BTN_TOOL_PEN , bitmask_key ) ;
has_mouse_button = test_bit ( BTN_LEFT , bitmask_key ) ;
has_rel_coordinates = test_bit ( EV_REL , bitmask_ev ) & & test_bit ( REL_X , bitmask_rel ) & & test_bit ( REL_Y , bitmask_rel ) ;
has_touch = test_bit ( BTN_TOUCH , bitmask_key ) ;
/* joysticks don't necessarily have buttons; e. g.
* rudders / pedals are joystick - like , but buttonless ; they have
* other fancy axes */
has_joystick_axes_or_buttons = test_bit ( BTN_TRIGGER , bitmask_key ) | |
test_bit ( BTN_A , bitmask_key ) | |
test_bit ( BTN_1 , bitmask_key ) | |
test_bit ( ABS_RX , bitmask_abs ) | |
test_bit ( ABS_RY , bitmask_abs ) | |
test_bit ( ABS_RZ , bitmask_abs ) | |
test_bit ( ABS_THROTTLE , bitmask_abs ) | |
test_bit ( ABS_RUDDER , bitmask_abs ) | |
test_bit ( ABS_WHEEL , bitmask_abs ) | |
test_bit ( ABS_GAS , bitmask_abs ) | |
test_bit ( ABS_BRAKE , bitmask_abs ) ;
if ( has_abs_coordinates ) {
if ( stylus_or_pen )
is_tablet = true ;
else if ( finger_but_no_pen )
is_touchpad = true ;
else if ( has_mouse_button )
2012-01-10 04:34:15 +04:00
/* This path is taken by VMware's USB mouse, which has
* absolute axes , but no touch / pressure button . */
2015-05-29 15:54:52 +03:00
is_mouse = true ;
else if ( has_touch )
is_touchscreen = true ;
else if ( has_joystick_axes_or_buttons )
is_joystick = true ;
2012-01-10 04:34:15 +04:00
}
2015-05-29 15:54:52 +03:00
if ( has_rel_coordinates & & has_mouse_button )
is_mouse = true ;
2012-01-10 04:34:15 +04:00
2015-05-29 15:54:52 +03:00
if ( is_pointing_stick )
udev_builtin_add_property ( dev , test , " ID_INPUT_POINTINGSTICK " , " 1 " ) ;
if ( is_mouse )
2012-01-10 04:34:15 +04:00
udev_builtin_add_property ( dev , test , " ID_INPUT_MOUSE " , " 1 " ) ;
2015-05-29 15:54:52 +03:00
if ( is_touchpad )
2012-01-10 04:34:15 +04:00
udev_builtin_add_property ( dev , test , " ID_INPUT_TOUCHPAD " , " 1 " ) ;
2015-05-29 15:54:52 +03:00
if ( is_touchscreen )
udev_builtin_add_property ( dev , test , " ID_INPUT_TOUCHSCREEN " , " 1 " ) ;
if ( is_joystick )
udev_builtin_add_property ( dev , test , " ID_INPUT_JOYSTICK " , " 1 " ) ;
if ( is_tablet )
udev_builtin_add_property ( dev , test , " ID_INPUT_TABLET " , " 1 " ) ;
return is_tablet | | is_mouse | | is_touchpad | | is_touchscreen | | is_joystick | | is_pointing_stick ;
2011-08-05 04:00:30 +04:00
}
/* key like devices */
2015-04-13 12:15:00 +03:00
static bool test_key ( struct udev_device * dev ,
2015-04-14 02:22:50 +03:00
const unsigned long * bitmask_ev ,
const unsigned long * bitmask_key ,
bool test ) {
2012-01-10 04:34:15 +04:00
unsigned i ;
unsigned long found ;
unsigned long mask ;
2015-04-13 12:15:00 +03:00
bool ret = false ;
2012-01-10 04:34:15 +04:00
/* do we have any KEY_* capability? */
2015-04-14 02:22:50 +03:00
if ( ! test_bit ( EV_KEY , bitmask_ev ) ) {
2013-12-24 19:39:37 +04:00
log_debug ( " test_key: no EV_KEY capability " ) ;
2015-04-13 12:15:00 +03:00
return false ;
2012-01-10 04:34:15 +04:00
}
/* only consider KEY_* here, not BTN_* */
found = 0 ;
for ( i = 0 ; i < BTN_MISC / BITS_PER_LONG ; + + i ) {
found | = bitmask_key [ i ] ;
2013-12-24 19:39:37 +04:00
log_debug ( " test_key: checking bit block %lu for any keys; found=%i " , ( unsigned long ) i * BITS_PER_LONG , found > 0 ) ;
2012-01-10 04:34:15 +04:00
}
/* If there are no keys in the lower block, check the higher block */
if ( ! found ) {
for ( i = KEY_OK ; i < BTN_TRIGGER_HAPPY ; + + i ) {
2015-04-14 02:22:50 +03:00
if ( test_bit ( i , bitmask_key ) ) {
2013-12-24 19:39:37 +04:00
log_debug ( " test_key: Found key %x in high block " , i ) ;
2012-01-10 04:34:15 +04:00
found = 1 ;
break ;
}
}
}
2015-04-13 12:15:00 +03:00
if ( found > 0 ) {
2012-01-10 04:34:15 +04:00
udev_builtin_add_property ( dev , test , " ID_INPUT_KEY " , " 1 " ) ;
2015-04-13 12:15:00 +03:00
ret = true ;
}
2012-01-10 04:34:15 +04:00
/* the first 32 bits are ESC, numbers, and Q to D; if we have all of
* those , consider it a full keyboard ; do not test KEY_RESERVED , though */
mask = 0xFFFFFFFE ;
2015-04-13 12:15:00 +03:00
if ( ( bitmask_key [ 0 ] & mask ) = = mask ) {
2012-01-10 04:34:15 +04:00
udev_builtin_add_property ( dev , test , " ID_INPUT_KEYBOARD " , " 1 " ) ;
2015-04-13 12:15:00 +03:00
ret = true ;
}
return ret ;
2011-08-05 04:00:30 +04:00
}
2014-07-29 17:47:41 +04:00
static int builtin_input_id ( struct udev_device * dev , int argc , char * argv [ ] , bool test ) {
2012-01-10 04:34:15 +04:00
struct udev_device * pdev ;
unsigned long bitmask_ev [ NBITS ( EV_MAX ) ] ;
unsigned long bitmask_abs [ NBITS ( ABS_MAX ) ] ;
unsigned long bitmask_key [ NBITS ( KEY_MAX ) ] ;
unsigned long bitmask_rel [ NBITS ( REL_MAX ) ] ;
2015-03-26 07:08:35 +03:00
unsigned long bitmask_props [ NBITS ( INPUT_PROP_MAX ) ] ;
2015-01-15 14:40:38 +03:00
const char * sysname , * devnode ;
2015-04-13 12:15:01 +03:00
bool is_pointer ;
bool is_key ;
2012-01-10 04:34:15 +04:00
2015-06-02 17:52:07 +03:00
assert ( dev ) ;
2012-01-10 04:34:15 +04:00
/* walk up the parental chain until we find the real input device; the
* argument is very likely a subdevice of this , like eventN */
pdev = dev ;
while ( pdev ! = NULL & & udev_device_get_sysattr_value ( pdev , " capabilities/ev " ) = = NULL )
pdev = udev_device_get_parent_with_subsystem_devtype ( pdev , " input " , NULL ) ;
2015-01-15 14:40:38 +03:00
if ( pdev ) {
/* Use this as a flag that input devices were detected, so that this
* program doesn ' t need to be called more than once per device */
udev_builtin_add_property ( dev , test , " ID_INPUT " , " 1 " ) ;
get_cap_mask ( dev , pdev , " capabilities/ev " , bitmask_ev , sizeof ( bitmask_ev ) , test ) ;
get_cap_mask ( dev , pdev , " capabilities/abs " , bitmask_abs , sizeof ( bitmask_abs ) , test ) ;
get_cap_mask ( dev , pdev , " capabilities/rel " , bitmask_rel , sizeof ( bitmask_rel ) , test ) ;
get_cap_mask ( dev , pdev , " capabilities/key " , bitmask_key , sizeof ( bitmask_key ) , test ) ;
2015-03-26 07:08:35 +03:00
get_cap_mask ( dev , pdev , " properties " , bitmask_props , sizeof ( bitmask_props ) , test ) ;
2015-04-13 12:15:01 +03:00
is_pointer = test_pointers ( dev , bitmask_ev , bitmask_abs ,
bitmask_key , bitmask_rel ,
bitmask_props , test ) ;
is_key = test_key ( dev , bitmask_ev , bitmask_key , test ) ;
/* Some evdev nodes have only a scrollwheel */
if ( ! is_pointer & & ! is_key & & test_bit ( EV_REL , bitmask_ev ) & &
( test_bit ( REL_WHEEL , bitmask_rel ) | | test_bit ( REL_HWHEEL , bitmask_rel ) ) )
udev_builtin_add_property ( dev , test , " ID_INPUT_KEY " , " 1 " ) ;
2015-01-15 14:40:38 +03:00
}
devnode = udev_device_get_devnode ( dev ) ;
sysname = udev_device_get_sysname ( dev ) ;
if ( devnode & & sysname & & startswith ( sysname , " event " ) )
extract_info ( dev , devnode , test ) ;
2012-01-10 04:34:15 +04:00
return EXIT_SUCCESS ;
2011-08-05 04:00:30 +04:00
}
const struct udev_builtin udev_builtin_input_id = {
2012-01-10 04:34:15 +04:00
. name = " input_id " ,
. cmd = builtin_input_id ,
2015-01-05 15:19:55 +03:00
. help = " Input device properties " ,
2011-08-05 04:00:30 +04:00
} ;