2011-01-07 23:45:50 +01:00
/*
* HID driver for multitouch panels
*
* Copyright ( c ) 2010 - 2011 Stephane Chatty < chatty @ enac . fr >
* Copyright ( c ) 2010 - 2011 Benjamin Tissoires < benjamin . tissoires @ gmail . com >
* Copyright ( c ) 2010 - 2011 Ecole Nationale de l ' Aviation Civile , France
*
2011-03-09 06:20:57 +01:00
* This code is partly based on hid - egalax . c :
*
* Copyright ( c ) 2010 Stephane Chatty < chatty @ enac . fr >
* Copyright ( c ) 2010 Henrik Rydberg < rydberg @ euromail . se >
* Copyright ( c ) 2010 Canonical , Ltd .
*
2011-03-22 17:34:01 +01:00
* This code is partly based on hid - 3 m - pct . c :
*
* Copyright ( c ) 2009 - 2010 Stephane Chatty < chatty @ enac . fr >
* Copyright ( c ) 2010 Henrik Rydberg < rydberg @ euromail . se >
* Copyright ( c ) 2010 Canonical , Ltd .
*
2011-01-07 23:45:50 +01: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 .
*/
# include <linux/device.h>
# include <linux/hid.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/usb.h>
# include <linux/input/mt.h>
# include "usbhid/usbhid.h"
MODULE_AUTHOR ( " Stephane Chatty <chatty@enac.fr> " ) ;
2011-01-31 11:28:21 +01:00
MODULE_AUTHOR ( " Benjamin Tissoires <benjamin.tissoires@gmail.com> " ) ;
2011-01-07 23:45:50 +01:00
MODULE_DESCRIPTION ( " HID multitouch panels " ) ;
MODULE_LICENSE ( " GPL " ) ;
# include "hid-ids.h"
/* quirks to control the device */
# define MT_QUIRK_NOT_SEEN_MEANS_UP (1 << 0)
# define MT_QUIRK_SLOT_IS_CONTACTID (1 << 1)
2011-01-11 16:45:54 +01:00
# define MT_QUIRK_CYPRESS (1 << 2)
2011-01-07 23:47:27 +01:00
# define MT_QUIRK_SLOT_IS_CONTACTNUMBER (1 << 3)
2011-09-17 22:27:30 +02:00
# define MT_QUIRK_ALWAYS_VALID (1 << 4)
# define MT_QUIRK_VALID_IS_INRANGE (1 << 5)
# define MT_QUIRK_VALID_IS_CONFIDENCE (1 << 6)
# define MT_QUIRK_SLOT_IS_CONTACTID_MINUS_ONE (1 << 8)
2011-01-07 23:45:50 +01:00
struct mt_slot {
__s32 x , y , p , w , h ;
__s32 contactid ; /* the device ContactID assigned to this slot */
bool touch_state ; /* is the touch valid? */
bool seen_in_this_frame ; /* has this slot been updated */
} ;
2011-11-23 10:54:28 +01:00
struct mt_class {
__s32 name ; /* MT_CLS */
__s32 quirks ;
__s32 sn_move ; /* Signal/noise ratio for move events */
__s32 sn_width ; /* Signal/noise ratio for width events */
__s32 sn_height ; /* Signal/noise ratio for height events */
__s32 sn_pressure ; /* Signal/noise ratio for pressure events */
__u8 maxcontacts ;
} ;
2011-01-07 23:45:50 +01:00
struct mt_device {
struct mt_slot curdata ; /* placeholder of incoming data */
2011-11-23 10:54:28 +01:00
struct mt_class mtclass ; /* our mt device class */
2011-01-07 23:45:50 +01:00
unsigned last_field_index ; /* last field index of the report */
unsigned last_slot_field ; /* the last field of a slot */
2011-06-12 08:22:08 +02:00
int last_mt_collection ; /* last known mt-related collection */
2011-01-07 23:45:50 +01:00
__s8 inputmode ; /* InputMode HID feature, -1 if non-existent */
__u8 num_received ; /* how many contacts we received */
__u8 num_expected ; /* expected last contact index */
2011-03-18 14:27:52 +01:00
__u8 maxcontacts ;
2011-01-07 23:45:50 +01:00
bool curvalid ; /* is the current contact valid? */
2011-03-18 14:27:52 +01:00
struct mt_slot * slots ;
2011-01-07 23:45:50 +01:00
} ;
/* classes of device behavior */
2011-05-20 15:59:34 +02:00
# define MT_CLS_DEFAULT 0x0001
2011-09-17 22:27:30 +02:00
# define MT_CLS_SERIAL 0x0002
# define MT_CLS_CONFIDENCE 0x0003
2011-11-29 13:13:10 +01:00
# define MT_CLS_CONFIDENCE_CONTACT_ID 0x0004
# define MT_CLS_CONFIDENCE_MINUS_ONE 0x0005
# define MT_CLS_DUAL_INRANGE_CONTACTID 0x0006
# define MT_CLS_DUAL_INRANGE_CONTACTNUMBER 0x0007
# define MT_CLS_DUAL_NSMU_CONTACTID 0x0008
2011-12-15 11:09:06 +08:00
# define MT_CLS_INRANGE_CONTACTNUMBER 0x0009
2011-05-20 15:59:34 +02:00
/* vendor specific classes */
# define MT_CLS_3M 0x0101
# define MT_CLS_CYPRESS 0x0102
# define MT_CLS_EGALAX 0x0103
2011-11-23 10:54:34 +01:00
# define MT_CLS_EGALAX_SERIAL 0x0104
2011-01-07 23:45:50 +01:00
2011-03-18 14:27:52 +01:00
# define MT_DEFAULT_MAXCONTACT 10
2011-01-07 23:45:50 +01:00
/*
* these device - dependent functions determine what slot corresponds
* to a valid contact that was just read .
*/
2011-01-07 23:46:30 +01:00
static int cypress_compute_slot ( struct mt_device * td )
{
if ( td - > curdata . contactid ! = 0 | | td - > num_received = = 0 )
return td - > curdata . contactid ;
else
return - 1 ;
}
2011-01-07 23:45:50 +01:00
static int find_slot_from_contactid ( struct mt_device * td )
{
int i ;
2011-03-18 14:27:52 +01:00
for ( i = 0 ; i < td - > maxcontacts ; + + i ) {
2011-01-07 23:45:50 +01:00
if ( td - > slots [ i ] . contactid = = td - > curdata . contactid & &
td - > slots [ i ] . touch_state )
return i ;
}
2011-03-18 14:27:52 +01:00
for ( i = 0 ; i < td - > maxcontacts ; + + i ) {
2011-01-07 23:45:50 +01:00
if ( ! td - > slots [ i ] . seen_in_this_frame & &
! td - > slots [ i ] . touch_state )
return i ;
}
/* should not occurs. If this happens that means
* that the device sent more touches that it says
* in the report descriptor . It is ignored then . */
2011-01-11 16:45:54 +01:00
return - 1 ;
2011-01-07 23:45:50 +01:00
}
2011-11-22 23:23:37 +01:00
static struct mt_class mt_classes [ ] = {
2011-01-11 16:45:54 +01:00
{ . name = MT_CLS_DEFAULT ,
2011-03-18 14:27:52 +01:00
. quirks = MT_QUIRK_NOT_SEEN_MEANS_UP } ,
2011-09-17 22:27:30 +02:00
{ . name = MT_CLS_SERIAL ,
. quirks = MT_QUIRK_ALWAYS_VALID } ,
2011-05-20 15:59:34 +02:00
{ . name = MT_CLS_CONFIDENCE ,
. quirks = MT_QUIRK_VALID_IS_CONFIDENCE } ,
2011-11-29 13:13:10 +01:00
{ . name = MT_CLS_CONFIDENCE_CONTACT_ID ,
. quirks = MT_QUIRK_VALID_IS_CONFIDENCE |
MT_QUIRK_SLOT_IS_CONTACTID } ,
2011-05-20 15:59:34 +02:00
{ . name = MT_CLS_CONFIDENCE_MINUS_ONE ,
. quirks = MT_QUIRK_VALID_IS_CONFIDENCE |
MT_QUIRK_SLOT_IS_CONTACTID_MINUS_ONE } ,
2011-01-31 11:28:20 +01:00
{ . name = MT_CLS_DUAL_INRANGE_CONTACTID ,
2011-01-11 16:45:54 +01:00
. quirks = MT_QUIRK_VALID_IS_INRANGE |
MT_QUIRK_SLOT_IS_CONTACTID ,
. maxcontacts = 2 } ,
2011-01-31 11:28:20 +01:00
{ . name = MT_CLS_DUAL_INRANGE_CONTACTNUMBER ,
2011-01-11 16:45:54 +01:00
. quirks = MT_QUIRK_VALID_IS_INRANGE |
MT_QUIRK_SLOT_IS_CONTACTNUMBER ,
. maxcontacts = 2 } ,
2011-05-20 15:59:34 +02:00
{ . name = MT_CLS_DUAL_NSMU_CONTACTID ,
. quirks = MT_QUIRK_NOT_SEEN_MEANS_UP |
MT_QUIRK_SLOT_IS_CONTACTID ,
. maxcontacts = 2 } ,
2011-12-15 11:09:06 +08:00
{ . name = MT_CLS_INRANGE_CONTACTNUMBER ,
. quirks = MT_QUIRK_VALID_IS_INRANGE |
MT_QUIRK_SLOT_IS_CONTACTNUMBER } ,
2011-05-20 15:59:34 +02:00
/*
* vendor specific classes
*/
{ . name = MT_CLS_3M ,
. quirks = MT_QUIRK_VALID_IS_CONFIDENCE |
MT_QUIRK_SLOT_IS_CONTACTID ,
. sn_move = 2048 ,
. sn_width = 128 ,
. sn_height = 128 } ,
2011-01-11 16:45:54 +01:00
{ . name = MT_CLS_CYPRESS ,
. quirks = MT_QUIRK_NOT_SEEN_MEANS_UP |
MT_QUIRK_CYPRESS ,
. maxcontacts = 10 } ,
2011-03-09 06:20:57 +01:00
{ . name = MT_CLS_EGALAX ,
. quirks = MT_QUIRK_SLOT_IS_CONTACTID |
2011-11-23 10:54:29 +01:00
MT_QUIRK_VALID_IS_INRANGE ,
2011-03-09 06:20:57 +01:00
. sn_move = 4096 ,
. sn_pressure = 32 ,
} ,
2011-11-23 10:54:34 +01:00
{ . name = MT_CLS_EGALAX_SERIAL ,
. quirks = MT_QUIRK_SLOT_IS_CONTACTID |
MT_QUIRK_ALWAYS_VALID ,
2011-03-09 06:20:57 +01:00
. sn_move = 4096 ,
. sn_pressure = 32 ,
} ,
2011-03-18 14:27:53 +01:00
2011-01-11 16:45:54 +01:00
{ }
2011-01-07 23:45:50 +01:00
} ;
2011-11-23 10:54:28 +01:00
static ssize_t mt_show_quirks ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
{
struct hid_device * hdev = container_of ( dev , struct hid_device , dev ) ;
struct mt_device * td = hid_get_drvdata ( hdev ) ;
return sprintf ( buf , " %u \n " , td - > mtclass . quirks ) ;
}
static ssize_t mt_set_quirks ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
{
struct hid_device * hdev = container_of ( dev , struct hid_device , dev ) ;
struct mt_device * td = hid_get_drvdata ( hdev ) ;
unsigned long val ;
if ( kstrtoul ( buf , 0 , & val ) )
return - EINVAL ;
td - > mtclass . quirks = val ;
return count ;
}
static DEVICE_ATTR ( quirks , S_IWUSR | S_IRUGO , mt_show_quirks , mt_set_quirks ) ;
static struct attribute * sysfs_attrs [ ] = {
& dev_attr_quirks . attr ,
NULL
} ;
static struct attribute_group mt_attribute_group = {
. attrs = sysfs_attrs
} ;
2011-02-24 19:30:59 +01:00
static void mt_feature_mapping ( struct hid_device * hdev ,
2011-01-07 23:45:50 +01:00
struct hid_field * field , struct hid_usage * usage )
{
2011-03-18 14:27:52 +01:00
struct mt_device * td = hid_get_drvdata ( hdev ) ;
switch ( usage - > hid ) {
case HID_DG_INPUTMODE :
2011-01-07 23:45:50 +01:00
td - > inputmode = field - > report - > id ;
2011-03-18 14:27:52 +01:00
break ;
case HID_DG_CONTACTMAX :
td - > maxcontacts = field - > value [ 0 ] ;
2011-11-23 10:54:28 +01:00
if ( td - > mtclass . maxcontacts )
2011-03-18 14:27:52 +01:00
/* check if the maxcontacts is given by the class */
2011-11-23 10:54:28 +01:00
td - > maxcontacts = td - > mtclass . maxcontacts ;
2011-03-18 14:27:52 +01:00
break ;
2011-01-07 23:45:50 +01:00
}
}
static void set_abs ( struct input_dev * input , unsigned int code ,
struct hid_field * field , int snratio )
{
int fmin = field - > logical_minimum ;
int fmax = field - > logical_maximum ;
int fuzz = snratio ? ( fmax - fmin ) / snratio : 0 ;
input_set_abs_params ( input , code , fmin , fmax , fuzz , 0 ) ;
}
static int mt_input_mapping ( struct hid_device * hdev , struct hid_input * hi ,
struct hid_field * field , struct hid_usage * usage ,
unsigned long * * bit , int * max )
{
struct mt_device * td = hid_get_drvdata ( hdev ) ;
2011-11-23 10:54:28 +01:00
struct mt_class * cls = & td - > mtclass ;
2011-03-09 06:20:57 +01:00
2011-08-15 16:44:28 -07:00
/* Only map fields from TouchScreen or TouchPad collections.
* We need to ignore fields that belong to other collections
* such as Mouse that might have the same GenericDesktop usages . */
if ( field - > application = = HID_DG_TOUCHSCREEN )
set_bit ( INPUT_PROP_DIRECT , hi - > input - > propbit ) ;
else if ( field - > application = = HID_DG_TOUCHPAD )
set_bit ( INPUT_PROP_POINTER , hi - > input - > propbit ) ;
else
return 0 ;
2011-11-23 10:54:29 +01:00
/* eGalax devices provide a Digitizer.Stylus input which overrides
* the correct Digitizers . Finger X / Y ranges .
* Let ' s just ignore this input . */
if ( field - > physical = = HID_DG_STYLUS )
return - 1 ;
2011-01-07 23:45:50 +01:00
switch ( usage - > hid & HID_USAGE_PAGE ) {
case HID_UP_GENDESK :
switch ( usage - > hid ) {
case HID_GD_X :
hid_map_usage ( hi , usage , bit , max ,
EV_ABS , ABS_MT_POSITION_X ) ;
set_abs ( hi - > input , ABS_MT_POSITION_X , field ,
cls - > sn_move ) ;
/* touchscreen emulation */
set_abs ( hi - > input , ABS_X , field , cls - > sn_move ) ;
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_GD_Y :
hid_map_usage ( hi , usage , bit , max ,
EV_ABS , ABS_MT_POSITION_Y ) ;
set_abs ( hi - > input , ABS_MT_POSITION_Y , field ,
cls - > sn_move ) ;
/* touchscreen emulation */
set_abs ( hi - > input , ABS_Y , field , cls - > sn_move ) ;
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
}
return 0 ;
case HID_UP_DIGITIZER :
switch ( usage - > hid ) {
case HID_DG_INRANGE :
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_CONFIDENCE :
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_TIPSWITCH :
hid_map_usage ( hi , usage , bit , max , EV_KEY , BTN_TOUCH ) ;
input_set_capability ( hi - > input , EV_KEY , BTN_TOUCH ) ;
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_CONTACTID :
2011-06-21 15:01:53 +02:00
if ( ! td - > maxcontacts )
td - > maxcontacts = MT_DEFAULT_MAXCONTACT ;
2011-03-18 14:27:52 +01:00
input_mt_init_slots ( hi - > input , td - > maxcontacts ) ;
2011-01-07 23:45:50 +01:00
td - > last_slot_field = usage - > hid ;
2011-04-21 14:15:59 +02:00
td - > last_field_index = field - > index ;
2011-06-12 08:22:08 +02:00
td - > last_mt_collection = usage - > collection_index ;
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_WIDTH :
hid_map_usage ( hi , usage , bit , max ,
EV_ABS , ABS_MT_TOUCH_MAJOR ) ;
2011-03-22 17:34:01 +01:00
set_abs ( hi - > input , ABS_MT_TOUCH_MAJOR , field ,
cls - > sn_width ) ;
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_HEIGHT :
hid_map_usage ( hi , usage , bit , max ,
EV_ABS , ABS_MT_TOUCH_MINOR ) ;
2011-03-22 17:34:01 +01:00
set_abs ( hi - > input , ABS_MT_TOUCH_MINOR , field ,
cls - > sn_height ) ;
2011-03-18 14:27:55 +01:00
input_set_abs_params ( hi - > input ,
ABS_MT_ORIENTATION , 0 , 1 , 0 , 0 ) ;
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_TIPPRESSURE :
hid_map_usage ( hi , usage , bit , max ,
EV_ABS , ABS_MT_PRESSURE ) ;
set_abs ( hi - > input , ABS_MT_PRESSURE , field ,
cls - > sn_pressure ) ;
/* touchscreen emulation */
set_abs ( hi - > input , ABS_PRESSURE , field ,
cls - > sn_pressure ) ;
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index ) {
td - > last_slot_field = usage - > hid ;
td - > last_field_index = field - > index ;
}
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_CONTACTCOUNT :
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index )
td - > last_field_index = field - > index ;
2011-01-07 23:45:50 +01:00
return 1 ;
case HID_DG_CONTACTMAX :
/* we don't set td->last_slot_field as contactcount and
* contact max are global to the report */
2011-06-12 08:22:08 +02:00
if ( td - > last_mt_collection = = usage - > collection_index )
td - > last_field_index = field - > index ;
2011-01-07 23:45:50 +01:00
return - 1 ;
}
/* let hid-input decide for the others */
return 0 ;
case 0xff000000 :
/* we do not want to map these: no input-oriented meaning */
return - 1 ;
}
return 0 ;
}
static int mt_input_mapped ( struct hid_device * hdev , struct hid_input * hi ,
struct hid_field * field , struct hid_usage * usage ,
unsigned long * * bit , int * max )
{
if ( usage - > type = = EV_KEY | | usage - > type = = EV_ABS )
set_bit ( usage - > type , hi - > input - > evbit ) ;
return - 1 ;
}
static int mt_compute_slot ( struct mt_device * td )
{
2011-11-23 10:54:28 +01:00
__s32 quirks = td - > mtclass . quirks ;
2011-01-07 23:45:50 +01:00
2011-01-11 16:45:54 +01:00
if ( quirks & MT_QUIRK_SLOT_IS_CONTACTID )
return td - > curdata . contactid ;
2011-01-07 23:45:50 +01:00
2011-01-11 16:45:54 +01:00
if ( quirks & MT_QUIRK_CYPRESS )
2011-01-07 23:46:30 +01:00
return cypress_compute_slot ( td ) ;
2011-01-11 16:45:54 +01:00
if ( quirks & MT_QUIRK_SLOT_IS_CONTACTNUMBER )
return td - > num_received ;
2011-01-07 23:47:27 +01:00
2011-04-22 11:51:48 +02:00
if ( quirks & MT_QUIRK_SLOT_IS_CONTACTID_MINUS_ONE )
return td - > curdata . contactid - 1 ;
2011-01-07 23:45:50 +01:00
return find_slot_from_contactid ( td ) ;
}
/*
* this function is called when a whole contact has been processed ,
* so that it can assign it to a slot and store the data there
*/
static void mt_complete_slot ( struct mt_device * td )
{
2011-01-11 16:45:54 +01:00
td - > curdata . seen_in_this_frame = true ;
2011-01-07 23:45:50 +01:00
if ( td - > curvalid ) {
int slotnum = mt_compute_slot ( td ) ;
2011-03-18 14:27:52 +01:00
if ( slotnum > = 0 & & slotnum < td - > maxcontacts )
2011-01-11 16:45:54 +01:00
td - > slots [ slotnum ] = td - > curdata ;
2011-01-07 23:45:50 +01:00
}
td - > num_received + + ;
}
/*
* this function is called when a whole packet has been received and processed ,
* so that it can decide what to send to the input layer .
*/
static void mt_emit_event ( struct mt_device * td , struct input_dev * input )
{
int i ;
2011-03-18 14:27:52 +01:00
for ( i = 0 ; i < td - > maxcontacts ; + + i ) {
2011-01-07 23:45:50 +01:00
struct mt_slot * s = & ( td - > slots [ i ] ) ;
2011-11-23 10:54:28 +01:00
if ( ( td - > mtclass . quirks & MT_QUIRK_NOT_SEEN_MEANS_UP ) & &
2011-01-07 23:45:50 +01:00
! s - > seen_in_this_frame ) {
s - > touch_state = false ;
}
input_mt_slot ( input , i ) ;
input_mt_report_slot_state ( input , MT_TOOL_FINGER ,
s - > touch_state ) ;
2011-01-11 16:45:54 +01:00
if ( s - > touch_state ) {
2011-03-22 17:34:01 +01:00
/* this finger is on the screen */
int wide = ( s - > w > s - > h ) ;
/* divided by two to match visual scale of touch */
int major = max ( s - > w , s - > h ) > > 1 ;
int minor = min ( s - > w , s - > h ) > > 1 ;
2011-01-11 16:45:54 +01:00
input_event ( input , EV_ABS , ABS_MT_POSITION_X , s - > x ) ;
input_event ( input , EV_ABS , ABS_MT_POSITION_Y , s - > y ) ;
2011-03-22 17:34:01 +01:00
input_event ( input , EV_ABS , ABS_MT_ORIENTATION , wide ) ;
2011-01-11 16:45:54 +01:00
input_event ( input , EV_ABS , ABS_MT_PRESSURE , s - > p ) ;
2011-03-22 17:34:01 +01:00
input_event ( input , EV_ABS , ABS_MT_TOUCH_MAJOR , major ) ;
input_event ( input , EV_ABS , ABS_MT_TOUCH_MINOR , minor ) ;
2011-01-11 16:45:54 +01:00
}
2011-01-07 23:45:50 +01:00
s - > seen_in_this_frame = false ;
}
input_mt_report_pointer_emulation ( input , true ) ;
input_sync ( input ) ;
td - > num_received = 0 ;
}
static int mt_event ( struct hid_device * hid , struct hid_field * field ,
struct hid_usage * usage , __s32 value )
{
struct mt_device * td = hid_get_drvdata ( hid ) ;
2011-11-23 10:54:28 +01:00
__s32 quirks = td - > mtclass . quirks ;
2011-01-07 23:45:50 +01:00
2011-03-18 14:27:52 +01:00
if ( hid - > claimed & HID_CLAIMED_INPUT & & td - > slots ) {
2011-01-07 23:45:50 +01:00
switch ( usage - > hid ) {
case HID_DG_INRANGE :
2011-09-17 22:27:30 +02:00
if ( quirks & MT_QUIRK_ALWAYS_VALID )
td - > curvalid = true ;
else if ( quirks & MT_QUIRK_VALID_IS_INRANGE )
2011-01-11 16:45:54 +01:00
td - > curvalid = value ;
2011-01-07 23:45:50 +01:00
break ;
case HID_DG_TIPSWITCH :
2011-01-11 16:45:54 +01:00
if ( quirks & MT_QUIRK_NOT_SEEN_MEANS_UP )
td - > curvalid = value ;
2011-01-07 23:45:50 +01:00
td - > curdata . touch_state = value ;
break ;
case HID_DG_CONFIDENCE :
2011-01-11 16:45:54 +01:00
if ( quirks & MT_QUIRK_VALID_IS_CONFIDENCE )
td - > curvalid = value ;
2011-01-07 23:45:50 +01:00
break ;
case HID_DG_CONTACTID :
td - > curdata . contactid = value ;
break ;
case HID_DG_TIPPRESSURE :
td - > curdata . p = value ;
break ;
case HID_GD_X :
td - > curdata . x = value ;
break ;
case HID_GD_Y :
td - > curdata . y = value ;
break ;
case HID_DG_WIDTH :
td - > curdata . w = value ;
break ;
case HID_DG_HEIGHT :
td - > curdata . h = value ;
break ;
case HID_DG_CONTACTCOUNT :
/*
2011-01-11 16:45:54 +01:00
* Includes multi - packet support where subsequent
* packets are sent with zero contactcount .
2011-01-07 23:45:50 +01:00
*/
if ( value )
2011-01-11 16:45:54 +01:00
td - > num_expected = value ;
2011-01-07 23:45:50 +01:00
break ;
default :
/* fallback to the generic hidinput handling */
return 0 ;
}
2011-03-09 06:35:25 +01:00
if ( usage - > hid = = td - > last_slot_field ) {
2011-01-11 16:45:54 +01:00
mt_complete_slot ( td ) ;
2011-03-09 06:35:25 +01:00
}
2011-01-11 16:45:54 +01:00
if ( field - > index = = td - > last_field_index
& & td - > num_received > = td - > num_expected )
mt_emit_event ( td , field - > hidinput - > input ) ;
2011-01-07 23:45:50 +01:00
2011-01-11 16:45:54 +01:00
}
2011-01-07 23:45:50 +01:00
/* we have handled the hidinput part, now remains hiddev */
if ( hid - > claimed & HID_CLAIMED_HIDDEV & & hid - > hiddev_hid_event )
hid - > hiddev_hid_event ( hid , field , usage , value ) ;
return 1 ;
}
static void mt_set_input_mode ( struct hid_device * hdev )
{
struct mt_device * td = hid_get_drvdata ( hdev ) ;
struct hid_report * r ;
struct hid_report_enum * re ;
if ( td - > inputmode < 0 )
return ;
re = & ( hdev - > report_enum [ HID_FEATURE_REPORT ] ) ;
r = re - > report_id_hash [ td - > inputmode ] ;
if ( r ) {
r - > field [ 0 ] - > value [ 0 ] = 0x02 ;
usbhid_submit_report ( hdev , r , USB_DIR_OUT ) ;
}
}
static int mt_probe ( struct hid_device * hdev , const struct hid_device_id * id )
{
2011-01-11 16:45:54 +01:00
int ret , i ;
2011-01-07 23:45:50 +01:00
struct mt_device * td ;
2011-01-11 16:45:54 +01:00
struct mt_class * mtclass = mt_classes ; /* MT_CLS_DEFAULT */
for ( i = 0 ; mt_classes [ i ] . name ; i + + ) {
if ( id - > driver_data = = mt_classes [ i ] . name ) {
mtclass = & ( mt_classes [ i ] ) ;
break ;
}
}
2011-01-07 23:45:50 +01:00
2011-11-01 15:26:31 +01:00
/* This allows the driver to correctly support devices
* that emit events over several HID messages .
*/
hdev - > quirks | = HID_QUIRK_NO_INPUT_SYNC ;
2011-01-07 23:45:50 +01:00
2011-03-18 14:27:52 +01:00
td = kzalloc ( sizeof ( struct mt_device ) , GFP_KERNEL ) ;
2011-01-07 23:45:50 +01:00
if ( ! td ) {
dev_err ( & hdev - > dev , " cannot allocate multitouch data \n " ) ;
return - ENOMEM ;
}
2011-11-23 10:54:28 +01:00
td - > mtclass = * mtclass ;
2011-01-07 23:45:50 +01:00
td - > inputmode = - 1 ;
2011-06-12 08:22:08 +02:00
td - > last_mt_collection = - 1 ;
2011-01-07 23:45:50 +01:00
hid_set_drvdata ( hdev , td ) ;
ret = hid_parse ( hdev ) ;
if ( ret ! = 0 )
goto fail ;
ret = hid_hw_start ( hdev , HID_CONNECT_DEFAULT ) ;
2011-01-11 16:45:54 +01:00
if ( ret )
2011-01-07 23:45:50 +01:00
goto fail ;
2011-03-18 14:27:52 +01:00
td - > slots = kzalloc ( td - > maxcontacts * sizeof ( struct mt_slot ) ,
GFP_KERNEL ) ;
if ( ! td - > slots ) {
dev_err ( & hdev - > dev , " cannot allocate multitouch slots \n " ) ;
hid_hw_stop ( hdev ) ;
ret = - ENOMEM ;
goto fail ;
}
2011-11-23 10:54:28 +01:00
ret = sysfs_create_group ( & hdev - > dev . kobj , & mt_attribute_group ) ;
2011-01-07 23:45:50 +01:00
mt_set_input_mode ( hdev ) ;
return 0 ;
fail :
kfree ( td ) ;
return ret ;
}
# ifdef CONFIG_PM
static int mt_reset_resume ( struct hid_device * hdev )
{
mt_set_input_mode ( hdev ) ;
return 0 ;
}
# endif
static void mt_remove ( struct hid_device * hdev )
{
struct mt_device * td = hid_get_drvdata ( hdev ) ;
2011-11-23 10:54:28 +01:00
sysfs_remove_group ( & hdev - > dev . kobj , & mt_attribute_group ) ;
2011-01-07 23:45:50 +01:00
hid_hw_stop ( hdev ) ;
2011-03-18 14:27:52 +01:00
kfree ( td - > slots ) ;
2011-01-07 23:45:50 +01:00
kfree ( td ) ;
hid_set_drvdata ( hdev , NULL ) ;
}
static const struct hid_device_id mt_devices [ ] = {
2011-03-22 17:34:01 +01:00
/* 3M panels */
{ . driver_data = MT_CLS_3M ,
HID_USB_DEVICE ( USB_VENDOR_ID_3M ,
USB_DEVICE_ID_3M1968 ) } ,
{ . driver_data = MT_CLS_3M ,
HID_USB_DEVICE ( USB_VENDOR_ID_3M ,
USB_DEVICE_ID_3M2256 ) } ,
2011-12-23 15:41:00 +01:00
{ . driver_data = MT_CLS_3M ,
HID_USB_DEVICE ( USB_VENDOR_ID_3M ,
USB_DEVICE_ID_3M3266 ) } ,
2011-03-22 17:34:01 +01:00
2011-05-19 14:18:13 +02:00
/* ActionStar panels */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_ACTIONSTAR ,
USB_DEVICE_ID_ACTIONSTAR_1011 ) } ,
2011-12-23 15:40:59 +01:00
/* Atmel panels */
{ . driver_data = MT_CLS_SERIAL ,
HID_USB_DEVICE ( USB_VENDOR_ID_ATMEL ,
USB_DEVICE_ID_ATMEL_MULTITOUCH ) } ,
2011-03-18 14:27:54 +01:00
/* Cando panels */
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTNUMBER ,
HID_USB_DEVICE ( USB_VENDOR_ID_CANDO ,
USB_DEVICE_ID_CANDO_MULTI_TOUCH ) } ,
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTNUMBER ,
HID_USB_DEVICE ( USB_VENDOR_ID_CANDO ,
USB_DEVICE_ID_CANDO_MULTI_TOUCH_10_1 ) } ,
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTNUMBER ,
HID_USB_DEVICE ( USB_VENDOR_ID_CANDO ,
USB_DEVICE_ID_CANDO_MULTI_TOUCH_11_6 ) } ,
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTNUMBER ,
HID_USB_DEVICE ( USB_VENDOR_ID_CANDO ,
USB_DEVICE_ID_CANDO_MULTI_TOUCH_15_6 ) } ,
2011-05-28 02:03:47 +08:00
/* Chunghwa Telecom touch panels */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_CHUNGHWAT ,
USB_DEVICE_ID_CHUNGHWAT_MULTITOUCH ) } ,
2011-05-19 14:18:14 +02:00
/* CVTouch panels */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_CVTOUCH ,
USB_DEVICE_ID_CVTOUCH_SCREEN ) } ,
2011-01-07 23:46:30 +01:00
/* Cypress panel */
{ . driver_data = MT_CLS_CYPRESS ,
HID_USB_DEVICE ( USB_VENDOR_ID_CYPRESS ,
USB_DEVICE_ID_CYPRESS_TRUETOUCH ) } ,
2011-05-20 15:59:34 +02:00
/* eGalax devices (resistive) */
2011-11-23 10:54:31 +01:00
{ . driver_data = MT_CLS_EGALAX ,
2011-05-20 15:59:34 +02:00
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
2011-11-23 10:54:31 +01:00
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_480D ) } ,
{ . driver_data = MT_CLS_EGALAX ,
2011-05-20 15:59:34 +02:00
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
2011-11-23 10:54:31 +01:00
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_480E ) } ,
2011-05-20 15:59:34 +02:00
/* eGalax devices (capacitive) */
2011-11-23 10:54:31 +01:00
{ . driver_data = MT_CLS_EGALAX ,
2011-05-20 15:59:34 +02:00
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
2011-11-23 10:54:31 +01:00
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_720C ) } ,
{ . driver_data = MT_CLS_EGALAX ,
2011-05-20 15:59:34 +02:00
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
2011-11-23 10:54:31 +01:00
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_726B ) } ,
{ . driver_data = MT_CLS_EGALAX ,
2011-05-20 15:59:34 +02:00
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
2011-11-23 10:54:31 +01:00
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_72A1 ) } ,
2011-11-23 10:54:33 +01:00
{ . driver_data = MT_CLS_EGALAX ,
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_72FA ) } ,
2011-11-23 10:54:32 +01:00
{ . driver_data = MT_CLS_EGALAX ,
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_7302 ) } ,
2011-11-23 10:54:34 +01:00
{ . driver_data = MT_CLS_EGALAX_SERIAL ,
2011-11-23 10:54:27 +01:00
HID_USB_DEVICE ( USB_VENDOR_ID_DWAV ,
2011-11-23 10:54:31 +01:00
USB_DEVICE_ID_DWAV_EGALAX_MULTITOUCH_A001 ) } ,
2011-05-20 15:59:34 +02:00
2011-05-19 11:37:29 +02:00
/* Elo TouchSystems IntelliTouch Plus panel */
{ . driver_data = MT_CLS_DUAL_NSMU_CONTACTID ,
HID_USB_DEVICE ( USB_VENDOR_ID_ELO ,
USB_DEVICE_ID_ELO_TS2515 ) } ,
2011-01-07 23:47:27 +01:00
/* GeneralTouch panel */
2011-01-31 11:28:20 +01:00
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTNUMBER ,
2011-01-07 23:47:27 +01:00
HID_USB_DEVICE ( USB_VENDOR_ID_GENERAL_TOUCH ,
USB_DEVICE_ID_GENERAL_TOUCH_WIN7_TWOFINGERS ) } ,
2011-05-19 14:18:15 +02:00
/* GoodTouch panels */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_GOODTOUCH ,
USB_DEVICE_ID_GOODTOUCH_000f ) } ,
2011-11-29 13:13:12 +01:00
/* Hanvon panels */
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTID ,
HID_USB_DEVICE ( USB_VENDOR_ID_HANVON_ALT ,
USB_DEVICE_ID_HANVON_ALT_MULTITOUCH ) } ,
2011-09-17 22:27:30 +02:00
/* Ideacom panel */
{ . driver_data = MT_CLS_SERIAL ,
HID_USB_DEVICE ( USB_VENDOR_ID_IDEACOM ,
USB_DEVICE_ID_IDEACOM_IDC6650 ) } ,
2011-05-09 23:54:14 +08:00
/* Ilitek dual touch panel */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_ILITEK ,
USB_DEVICE_ID_ILITEK_MULTITOUCH ) } ,
2011-01-31 11:28:22 +01:00
/* IRTOUCH panels */
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTID ,
HID_USB_DEVICE ( USB_VENDOR_ID_IRTOUCHSYSTEMS ,
USB_DEVICE_ID_IRTOUCH_INFRARED_USB ) } ,
2011-08-15 21:12:09 -07:00
/* LG Display panels */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_LG ,
USB_DEVICE_ID_LG_MULTITOUCH ) } ,
2011-05-18 15:27:24 +02:00
/* Lumio panels */
{ . driver_data = MT_CLS_CONFIDENCE_MINUS_ONE ,
HID_USB_DEVICE ( USB_VENDOR_ID_LUMIO ,
USB_DEVICE_ID_CRYSTALTOUCH ) } ,
2011-06-21 15:01:55 +02:00
{ . driver_data = MT_CLS_CONFIDENCE_MINUS_ONE ,
HID_USB_DEVICE ( USB_VENDOR_ID_LUMIO ,
USB_DEVICE_ID_CRYSTALTOUCH_DUAL ) } ,
2011-05-18 15:27:24 +02:00
2011-04-22 11:51:48 +02:00
/* MosArt panels */
{ . driver_data = MT_CLS_CONFIDENCE_MINUS_ONE ,
HID_USB_DEVICE ( USB_VENDOR_ID_ASUS ,
USB_DEVICE_ID_ASUS_T91MT ) } ,
{ . driver_data = MT_CLS_CONFIDENCE_MINUS_ONE ,
HID_USB_DEVICE ( USB_VENDOR_ID_ASUS ,
USB_DEVICE_ID_ASUSTEK_MULTITOUCH_YFO ) } ,
{ . driver_data = MT_CLS_CONFIDENCE_MINUS_ONE ,
HID_USB_DEVICE ( USB_VENDOR_ID_TURBOX ,
USB_DEVICE_ID_TURBOX_TOUCHSCREEN_MOSART ) } ,
2011-04-21 16:21:52 +02:00
/* PenMount panels */
{ . driver_data = MT_CLS_CONFIDENCE ,
HID_USB_DEVICE ( USB_VENDOR_ID_PENMOUNT ,
USB_DEVICE_ID_PENMOUNT_PCI ) } ,
2011-12-15 11:09:06 +08:00
/* PixArt optical touch screen */
{ . driver_data = MT_CLS_INRANGE_CONTACTNUMBER ,
HID_USB_DEVICE ( USB_VENDOR_ID_PIXART ,
USB_DEVICE_ID_PIXART_OPTICAL_TOUCH_SCREEN ) } ,
{ . driver_data = MT_CLS_INRANGE_CONTACTNUMBER ,
HID_USB_DEVICE ( USB_VENDOR_ID_PIXART ,
USB_DEVICE_ID_PIXART_OPTICAL_TOUCH_SCREEN1 ) } ,
{ . driver_data = MT_CLS_INRANGE_CONTACTNUMBER ,
HID_USB_DEVICE ( USB_VENDOR_ID_PIXART ,
USB_DEVICE_ID_PIXART_OPTICAL_TOUCH_SCREEN2 ) } ,
2011-01-07 23:45:50 +01:00
/* PixCir-based panels */
2011-01-31 11:28:20 +01:00
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTID ,
2011-01-07 23:45:50 +01:00
HID_USB_DEVICE ( USB_VENDOR_ID_HANVON ,
USB_DEVICE_ID_HANVON_MULTITOUCH ) } ,
2011-01-31 11:28:20 +01:00
{ . driver_data = MT_CLS_DUAL_INRANGE_CONTACTID ,
2011-01-07 23:45:50 +01:00
HID_USB_DEVICE ( USB_VENDOR_ID_CANDO ,
USB_DEVICE_ID_CANDO_PIXCIR_MULTI_TOUCH ) } ,
2011-11-29 13:13:10 +01:00
/* Quanta-based panels */
{ . driver_data = MT_CLS_CONFIDENCE_CONTACT_ID ,
HID_USB_DEVICE ( USB_VENDOR_ID_QUANTA ,
USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH ) } ,
{ . driver_data = MT_CLS_CONFIDENCE_CONTACT_ID ,
HID_USB_DEVICE ( USB_VENDOR_ID_QUANTA ,
USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3001 ) } ,
{ . driver_data = MT_CLS_CONFIDENCE_CONTACT_ID ,
HID_USB_DEVICE ( USB_VENDOR_ID_QUANTA ,
USB_DEVICE_ID_QUANTA_OPTICAL_TOUCH_3008 ) } ,
2011-03-18 14:27:53 +01:00
/* Stantum panels */
2011-05-19 14:18:18 +02:00
{ . driver_data = MT_CLS_CONFIDENCE ,
2011-03-18 14:27:53 +01:00
HID_USB_DEVICE ( USB_VENDOR_ID_STANTUM ,
USB_DEVICE_ID_MTP ) } ,
2011-05-19 14:18:18 +02:00
{ . driver_data = MT_CLS_CONFIDENCE ,
2011-06-21 15:01:54 +02:00
HID_USB_DEVICE ( USB_VENDOR_ID_STANTUM_STM ,
2011-03-18 14:27:53 +01:00
USB_DEVICE_ID_MTP_STM ) } ,
2011-05-19 14:18:18 +02:00
{ . driver_data = MT_CLS_CONFIDENCE ,
2011-06-21 15:01:54 +02:00
HID_USB_DEVICE ( USB_VENDOR_ID_STANTUM_SITRONIX ,
2011-03-18 14:27:53 +01:00
USB_DEVICE_ID_MTP_SITRONIX ) } ,
2011-05-19 14:18:16 +02:00
/* Touch International panels */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_TOUCH_INTL ,
USB_DEVICE_ID_TOUCH_INTL_MULTI_TOUCH ) } ,
2011-05-19 14:18:17 +02:00
/* Unitec panels */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_UNITEC ,
USB_DEVICE_ID_UNITEC_USB_TOUCH_0709 ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_UNITEC ,
USB_DEVICE_ID_UNITEC_USB_TOUCH_0A19 ) } ,
2011-07-15 16:58:06 +08:00
/* XAT */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XAT ,
USB_DEVICE_ID_XAT_CSR ) } ,
2011-05-19 14:18:17 +02:00
2012-01-05 11:53:46 +09:00
/* Xiroku */
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_SPX ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_MPX ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_CSR ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_SPX1 ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_MPX1 ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_CSR1 ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_SPX2 ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_MPX2 ) } ,
{ . driver_data = MT_CLS_DEFAULT ,
HID_USB_DEVICE ( USB_VENDOR_ID_XIROKU ,
USB_DEVICE_ID_XIROKU_CSR2 ) } ,
2011-01-07 23:45:50 +01:00
{ }
} ;
MODULE_DEVICE_TABLE ( hid , mt_devices ) ;
static const struct hid_usage_id mt_grabbed_usages [ ] = {
{ HID_ANY_ID , HID_ANY_ID , HID_ANY_ID } ,
{ HID_ANY_ID - 1 , HID_ANY_ID - 1 , HID_ANY_ID - 1 }
} ;
static struct hid_driver mt_driver = {
. name = " hid-multitouch " ,
. id_table = mt_devices ,
. probe = mt_probe ,
. remove = mt_remove ,
. input_mapping = mt_input_mapping ,
. input_mapped = mt_input_mapped ,
. feature_mapping = mt_feature_mapping ,
. usage_table = mt_grabbed_usages ,
. event = mt_event ,
# ifdef CONFIG_PM
. reset_resume = mt_reset_resume ,
# endif
} ;
static int __init mt_init ( void )
{
return hid_register_driver ( & mt_driver ) ;
}
static void __exit mt_exit ( void )
{
hid_unregister_driver ( & mt_driver ) ;
}
module_init ( mt_init ) ;
module_exit ( mt_exit ) ;