2005-04-17 02:20:36 +04:00
/*
* USB HID support for Linux
*
* Copyright ( c ) 1999 Andreas Gal
2005-09-05 09:12:01 +04:00
* Copyright ( c ) 2000 - 2005 Vojtech Pavlik < vojtech @ suse . cz >
* Copyright ( c ) 2005 Michael Haboustak < mike - @ cinci . rr . com > for Concept2 , Inc
2007-02-21 21:27:49 +03:00
* Copyright ( c ) 2006 - 2007 Jiri Kosina
2005-04-17 02:20:36 +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 .
*/
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/list.h>
# include <linux/mm.h>
# include <linux/smp_lock.h>
# include <linux/spinlock.h>
# include <asm/unaligned.h>
# include <asm/byteorder.h>
# include <linux/input.h>
# include <linux/wait.h>
# include <linux/usb.h>
2006-12-08 20:40:44 +03:00
# include <linux/hid.h>
2005-04-17 02:20:36 +04:00
# include <linux/hiddev.h>
2007-01-25 13:43:31 +03:00
# include <linux/hid-debug.h>
2007-05-14 11:57:40 +04:00
# include <linux/hidraw.h>
2006-12-08 20:41:03 +03:00
# include "usbhid.h"
2005-04-17 02:20:36 +04:00
/*
* Version Information
*/
2005-09-05 09:12:01 +04:00
# define DRIVER_VERSION "v2.6"
2007-04-16 13:29:28 +04:00
# define DRIVER_AUTHOR "Andreas Gal, Vojtech Pavlik, Jiri Kosina"
2005-04-17 02:20:36 +04:00
# define DRIVER_DESC "USB HID core driver"
# define DRIVER_LICENSE "GPL"
static char * hid_types [ ] = { " Device " , " Pointer " , " Mouse " , " Device " , " Joystick " ,
" Gamepad " , " Keyboard " , " Keypad " , " Multi-Axis Controller " } ;
/*
* Module parameters .
*/
static unsigned int hid_mousepoll_interval ;
module_param_named ( mousepoll , hid_mousepoll_interval , uint , 0644 ) ;
MODULE_PARM_DESC ( mousepoll , " Polling interval of mice " ) ;
2007-04-19 16:56:12 +04:00
/* Quirks specified at module load time */
static char * quirks_param [ MAX_USBHID_BOOT_QUIRKS ] = { [ 0 . . . ( MAX_USBHID_BOOT_QUIRKS - 1 ) ] = NULL } ;
module_param_array_named ( quirks , quirks_param , charp , NULL , 0444 ) ;
MODULE_PARM_DESC ( quirks , " Add/modify USB HID quirks by specifying "
" quirks=vendorID:productID:quirks "
" where vendorID, productID, and quirks are all in "
" 0x-prefixed hex " ) ;
2007-06-19 16:09:14 +04:00
static char * rdesc_quirks_param [ MAX_USBHID_BOOT_QUIRKS ] = { [ 0 . . . ( MAX_USBHID_BOOT_QUIRKS - 1 ) ] = NULL } ;
module_param_array_named ( rdesc_quirks , rdesc_quirks_param , charp , NULL , 0444 ) ;
MODULE_PARM_DESC ( rdesc_quirks , " Add/modify report descriptor quirks by specifying "
" rdesc_quirks=vendorID:productID:rdesc_quirks "
" where vendorID, productID, and rdesc_quirks are all in "
" 0x-prefixed hex " ) ;
2005-04-17 02:20:36 +04:00
/*
2007-04-04 01:39:37 +04:00
* Input submission and I / O error handler .
2005-04-17 02:20:36 +04:00
*/
2007-04-04 01:39:37 +04:00
static void hid_io_error ( struct hid_device * hid ) ;
/* Start up the input URB */
static int hid_start_in ( struct hid_device * hid )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
2007-04-04 01:39:37 +04:00
int rc = 0 ;
struct usbhid_device * usbhid = hid - > driver_data ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
spin_lock_irqsave ( & usbhid - > inlock , flags ) ;
if ( hid - > open > 0 & & ! test_bit ( HID_SUSPENDED , & usbhid - > iofl ) & &
! test_and_set_bit ( HID_IN_RUNNING , & usbhid - > iofl ) ) {
rc = usb_submit_urb ( usbhid - > urbin , GFP_ATOMIC ) ;
if ( rc ! = 0 )
clear_bit ( HID_IN_RUNNING , & usbhid - > iofl ) ;
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
spin_unlock_irqrestore ( & usbhid - > inlock , flags ) ;
return rc ;
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
/* I/O retry timer routine */
static void hid_retry_timeout ( unsigned long _hid )
2005-04-17 02:20:36 +04:00
{
2007-04-04 01:39:37 +04:00
struct hid_device * hid = ( struct hid_device * ) _hid ;
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid = hid - > driver_data ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
dev_dbg ( & usbhid - > intf - > dev , " retrying intr urb \n " ) ;
if ( hid_start_in ( hid ) )
hid_io_error ( hid ) ;
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
/* Workqueue routine to reset the device or clear a halt */
static void hid_reset ( struct work_struct * work )
2005-04-17 02:20:36 +04:00
{
2007-04-04 01:39:37 +04:00
struct usbhid_device * usbhid =
container_of ( work , struct usbhid_device , reset_work ) ;
struct hid_device * hid = usbhid - > hid ;
int rc_lock , rc = 0 ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( test_bit ( HID_CLEAR_HALT , & usbhid - > iofl ) ) {
dev_dbg ( & usbhid - > intf - > dev , " clear halt \n " ) ;
rc = usb_clear_halt ( hid_to_usb_dev ( hid ) , usbhid - > urbin - > pipe ) ;
clear_bit ( HID_CLEAR_HALT , & usbhid - > iofl ) ;
hid_start_in ( hid ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
else if ( test_bit ( HID_RESET_PENDING , & usbhid - > iofl ) ) {
dev_dbg ( & usbhid - > intf - > dev , " resetting device \n " ) ;
rc = rc_lock = usb_lock_device_for_reset ( hid_to_usb_dev ( hid ) , usbhid - > intf ) ;
if ( rc_lock > = 0 ) {
rc = usb_reset_composite_device ( hid_to_usb_dev ( hid ) , usbhid - > intf ) ;
if ( rc_lock )
usb_unlock_device ( hid_to_usb_dev ( hid ) ) ;
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
clear_bit ( HID_RESET_PENDING , & usbhid - > iofl ) ;
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
switch ( rc ) {
case 0 :
if ( ! test_bit ( HID_IN_RUNNING , & usbhid - > iofl ) )
hid_io_error ( hid ) ;
break ;
default :
2007-05-30 17:07:13 +04:00
err_hid ( " can't reset device, %s-%s/input%d, status %d " ,
2007-04-04 01:39:37 +04:00
hid_to_usb_dev ( hid ) - > bus - > bus_name ,
hid_to_usb_dev ( hid ) - > devpath ,
usbhid - > ifnum , rc ) ;
/* FALLTHROUGH */
case - EHOSTUNREACH :
case - ENODEV :
case - EINTR :
break ;
2005-04-17 02:20:36 +04:00
}
}
2007-04-04 01:39:37 +04:00
/* Main I/O error handler */
static void hid_io_error ( struct hid_device * hid )
2006-12-08 20:40:44 +03:00
{
2007-04-04 01:39:37 +04:00
unsigned long flags ;
struct usbhid_device * usbhid = hid - > driver_data ;
2006-12-08 20:40:44 +03:00
2007-04-04 01:39:37 +04:00
spin_lock_irqsave ( & usbhid - > inlock , flags ) ;
2006-12-08 20:40:44 +03:00
2007-04-04 01:39:37 +04:00
/* Stop when disconnected */
if ( usb_get_intfdata ( usbhid - > intf ) = = NULL )
goto done ;
2006-12-08 20:40:44 +03:00
2007-03-20 21:03:31 +03:00
/* If it has been a while since the last error, we'll assume
* this a brand new error and reset the retry timeout . */
if ( time_after ( jiffies , usbhid - > stop_retry + HZ / 2 ) )
usbhid - > retry_delay = 0 ;
2007-04-04 01:39:37 +04:00
/* When an error occurs, retry at increasing intervals */
if ( usbhid - > retry_delay = = 0 ) {
usbhid - > retry_delay = 13 ; /* Then 26, 52, 104, 104, ... */
usbhid - > stop_retry = jiffies + msecs_to_jiffies ( 1000 ) ;
} else if ( usbhid - > retry_delay < 100 )
usbhid - > retry_delay * = 2 ;
2006-12-08 20:40:44 +03:00
2007-04-04 01:39:37 +04:00
if ( time_after ( jiffies , usbhid - > stop_retry ) ) {
2006-12-08 20:41:03 +03:00
2007-04-04 01:39:37 +04:00
/* Retries failed, so do a port reset */
if ( ! test_and_set_bit ( HID_RESET_PENDING , & usbhid - > iofl ) ) {
schedule_work ( & usbhid - > reset_work ) ;
goto done ;
}
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
mod_timer ( & usbhid - > io_retry ,
jiffies + msecs_to_jiffies ( usbhid - > retry_delay ) ) ;
done :
spin_unlock_irqrestore ( & usbhid - > inlock , flags ) ;
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
/*
* Input interrupt completion handler .
*/
2005-05-29 11:28:00 +04:00
2007-04-04 01:39:37 +04:00
static void hid_irq_in ( struct urb * urb )
2005-04-17 02:20:36 +04:00
{
2007-04-04 01:39:37 +04:00
struct hid_device * hid = urb - > context ;
struct usbhid_device * usbhid = hid - > driver_data ;
int status ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
switch ( urb - > status ) {
case 0 : /* success */
usbhid - > retry_delay = 0 ;
hid_input_report ( urb - > context , HID_INPUT_REPORT ,
urb - > transfer_buffer ,
urb - > actual_length , 1 ) ;
break ;
case - EPIPE : /* stall */
clear_bit ( HID_IN_RUNNING , & usbhid - > iofl ) ;
set_bit ( HID_CLEAR_HALT , & usbhid - > iofl ) ;
schedule_work ( & usbhid - > reset_work ) ;
return ;
case - ECONNRESET : /* unlink */
case - ENOENT :
case - ESHUTDOWN : /* unplug */
clear_bit ( HID_IN_RUNNING , & usbhid - > iofl ) ;
return ;
case - EILSEQ : /* protocol error or unplug */
case - EPROTO : /* protocol error or unplug */
case - ETIME : /* protocol error or unplug */
case - ETIMEDOUT : /* Should never happen, but... */
clear_bit ( HID_IN_RUNNING , & usbhid - > iofl ) ;
hid_io_error ( hid ) ;
return ;
default : /* error */
warn ( " input irq status %d received " , urb - > status ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
status = usb_submit_urb ( urb , GFP_ATOMIC ) ;
if ( status ) {
clear_bit ( HID_IN_RUNNING , & usbhid - > iofl ) ;
if ( status ! = - EPERM ) {
2007-05-30 17:07:13 +04:00
err_hid ( " can't resubmit intr, %s-%s/input%d, status %d " ,
2007-04-04 01:39:37 +04:00
hid_to_usb_dev ( hid ) - > bus - > bus_name ,
hid_to_usb_dev ( hid ) - > devpath ,
usbhid - > ifnum , status ) ;
hid_io_error ( hid ) ;
}
}
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
static int hid_submit_out ( struct hid_device * hid )
2005-04-17 02:20:36 +04:00
{
2007-04-04 01:39:37 +04:00
struct hid_report * report ;
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid = hid - > driver_data ;
2007-04-04 01:39:37 +04:00
report = usbhid - > out [ usbhid - > outtail ] ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
hid_output_report ( report , usbhid - > outbuf ) ;
usbhid - > urbout - > transfer_buffer_length = ( ( report - > size - 1 ) > > 3 ) + 1 + ( report - > id > 0 ) ;
usbhid - > urbout - > dev = hid_to_usb_dev ( hid ) ;
2006-03-30 00:41:07 +04:00
2007-05-30 17:07:13 +04:00
dbg_hid ( " submitting out urb \n " ) ;
2006-10-18 16:47:37 +04:00
2007-04-04 01:39:37 +04:00
if ( usb_submit_urb ( usbhid - > urbout , GFP_ATOMIC ) ) {
2007-05-30 17:07:13 +04:00
err_hid ( " usb_submit_urb(out) failed " ) ;
2007-04-04 01:39:37 +04:00
return - 1 ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
return 0 ;
}
static int hid_submit_ctrl ( struct hid_device * hid )
2005-04-17 02:20:36 +04:00
{
struct hid_report * report ;
2007-04-04 01:39:37 +04:00
unsigned char dir ;
int len ;
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid = hid - > driver_data ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
report = usbhid - > ctrl [ usbhid - > ctrltail ] . report ;
dir = usbhid - > ctrl [ usbhid - > ctrltail ] . dir ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
len = ( ( report - > size - 1 ) > > 3 ) + 1 + ( report - > id > 0 ) ;
if ( dir = = USB_DIR_OUT ) {
hid_output_report ( report , usbhid - > ctrlbuf ) ;
usbhid - > urbctrl - > pipe = usb_sndctrlpipe ( hid_to_usb_dev ( hid ) , 0 ) ;
usbhid - > urbctrl - > transfer_buffer_length = len ;
} else {
int maxpacket , padlen ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
usbhid - > urbctrl - > pipe = usb_rcvctrlpipe ( hid_to_usb_dev ( hid ) , 0 ) ;
maxpacket = usb_maxpacket ( hid_to_usb_dev ( hid ) , usbhid - > urbctrl - > pipe , 0 ) ;
if ( maxpacket > 0 ) {
2008-02-15 15:53:11 +03:00
padlen = DIV_ROUND_UP ( len , maxpacket ) ;
2007-04-04 01:39:37 +04:00
padlen * = maxpacket ;
if ( padlen > usbhid - > bufsize )
padlen = usbhid - > bufsize ;
} else
padlen = 0 ;
usbhid - > urbctrl - > transfer_buffer_length = padlen ;
2005-04-17 02:20:36 +04:00
}
2007-04-04 01:39:37 +04:00
usbhid - > urbctrl - > dev = hid_to_usb_dev ( hid ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
usbhid - > cr - > bRequestType = USB_TYPE_CLASS | USB_RECIP_INTERFACE | dir ;
usbhid - > cr - > bRequest = ( dir = = USB_DIR_OUT ) ? HID_REQ_SET_REPORT : HID_REQ_GET_REPORT ;
usbhid - > cr - > wValue = cpu_to_le16 ( ( ( report - > type + 1 ) < < 8 ) | report - > id ) ;
usbhid - > cr - > wIndex = cpu_to_le16 ( usbhid - > ifnum ) ;
usbhid - > cr - > wLength = cpu_to_le16 ( len ) ;
2005-04-17 02:20:36 +04:00
2007-05-30 17:07:13 +04:00
dbg_hid ( " submitting ctrl urb: %s wValue=0x%04x wIndex=0x%04x wLength=%u \n " ,
2007-04-04 01:39:37 +04:00
usbhid - > cr - > bRequest = = HID_REQ_SET_REPORT ? " Set_Report " : " Get_Report " ,
usbhid - > cr - > wValue , usbhid - > cr - > wIndex , usbhid - > cr - > wLength ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( usb_submit_urb ( usbhid - > urbctrl , GFP_ATOMIC ) ) {
2007-05-30 17:07:13 +04:00
err_hid ( " usb_submit_urb(ctrl) failed " ) ;
2007-04-04 01:39:37 +04:00
return - 1 ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
return 0 ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
/*
* Output interrupt completion handler .
*/
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
static void hid_irq_out ( struct urb * urb )
{
struct hid_device * hid = urb - > context ;
struct usbhid_device * usbhid = hid - > driver_data ;
unsigned long flags ;
int unplug = 0 ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
switch ( urb - > status ) {
case 0 : /* success */
break ;
case - ESHUTDOWN : /* unplug */
unplug = 1 ;
case - EILSEQ : /* protocol error or unplug */
case - EPROTO : /* protocol error or unplug */
case - ECONNRESET : /* unlink */
case - ENOENT :
break ;
default : /* error */
warn ( " output irq status %d received " , urb - > status ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
spin_lock_irqsave ( & usbhid - > outlock , flags ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( unplug )
usbhid - > outtail = usbhid - > outhead ;
else
usbhid - > outtail = ( usbhid - > outtail + 1 ) & ( HID_OUTPUT_FIFO_SIZE - 1 ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( usbhid - > outhead ! = usbhid - > outtail ) {
if ( hid_submit_out ( hid ) ) {
clear_bit ( HID_OUT_RUNNING , & usbhid - > iofl ) ;
wake_up ( & hid - > wait ) ;
}
spin_unlock_irqrestore ( & usbhid - > outlock , flags ) ;
return ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
clear_bit ( HID_OUT_RUNNING , & usbhid - > iofl ) ;
spin_unlock_irqrestore ( & usbhid - > outlock , flags ) ;
wake_up ( & hid - > wait ) ;
}
2005-07-11 10:08:40 +04:00
2007-04-04 01:39:37 +04:00
/*
* Control pipe completion handler .
*/
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
static void hid_ctrl ( struct urb * urb )
{
struct hid_device * hid = urb - > context ;
struct usbhid_device * usbhid = hid - > driver_data ;
unsigned long flags ;
int unplug = 0 ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
spin_lock_irqsave ( & usbhid - > ctrllock , flags ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
switch ( urb - > status ) {
case 0 : /* success */
if ( usbhid - > ctrl [ usbhid - > ctrltail ] . dir = = USB_DIR_IN )
hid_input_report ( urb - > context , usbhid - > ctrl [ usbhid - > ctrltail ] . report - > type ,
urb - > transfer_buffer , urb - > actual_length , 0 ) ;
break ;
case - ESHUTDOWN : /* unplug */
unplug = 1 ;
case - EILSEQ : /* protocol error or unplug */
case - EPROTO : /* protocol error or unplug */
case - ECONNRESET : /* unlink */
case - ENOENT :
case - EPIPE : /* report not available */
break ;
default : /* error */
warn ( " ctrl urb status %d received " , urb - > status ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( unplug )
usbhid - > ctrltail = usbhid - > ctrlhead ;
else
usbhid - > ctrltail = ( usbhid - > ctrltail + 1 ) & ( HID_CONTROL_FIFO_SIZE - 1 ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( usbhid - > ctrlhead ! = usbhid - > ctrltail ) {
if ( hid_submit_ctrl ( hid ) ) {
clear_bit ( HID_CTRL_RUNNING , & usbhid - > iofl ) ;
wake_up ( & hid - > wait ) ;
}
spin_unlock_irqrestore ( & usbhid - > ctrllock , flags ) ;
return ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
clear_bit ( HID_CTRL_RUNNING , & usbhid - > iofl ) ;
spin_unlock_irqrestore ( & usbhid - > ctrllock , flags ) ;
wake_up ( & hid - > wait ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
void usbhid_submit_report ( struct hid_device * hid , struct hid_report * report , unsigned char dir )
{
int head ;
unsigned long flags ;
struct usbhid_device * usbhid = hid - > driver_data ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( ( hid - > quirks & HID_QUIRK_NOGET ) & & dir = = USB_DIR_IN )
return ;
2006-06-30 11:44:03 +04:00
2007-04-04 01:39:37 +04:00
if ( usbhid - > urbout & & dir = = USB_DIR_OUT & & report - > type = = HID_OUTPUT_REPORT ) {
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
spin_lock_irqsave ( & usbhid - > outlock , flags ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
if ( ( head = ( usbhid - > outhead + 1 ) & ( HID_OUTPUT_FIFO_SIZE - 1 ) ) = = usbhid - > outtail ) {
spin_unlock_irqrestore ( & usbhid - > outlock , flags ) ;
warn ( " output queue full " ) ;
return ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
usbhid - > out [ usbhid - > outhead ] = report ;
usbhid - > outhead = head ;
2005-06-03 09:18:12 +04:00
2007-04-04 01:39:37 +04:00
if ( ! test_and_set_bit ( HID_OUT_RUNNING , & usbhid - > iofl ) )
if ( hid_submit_out ( hid ) )
clear_bit ( HID_OUT_RUNNING , & usbhid - > iofl ) ;
2005-06-28 00:44:22 +04:00
2007-04-04 01:39:37 +04:00
spin_unlock_irqrestore ( & usbhid - > outlock , flags ) ;
return ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
spin_lock_irqsave ( & usbhid - > ctrllock , flags ) ;
2006-01-14 08:25:39 +03:00
2007-04-04 01:39:37 +04:00
if ( ( head = ( usbhid - > ctrlhead + 1 ) & ( HID_CONTROL_FIFO_SIZE - 1 ) ) = = usbhid - > ctrltail ) {
spin_unlock_irqrestore ( & usbhid - > ctrllock , flags ) ;
warn ( " control queue full " ) ;
return ;
}
2006-09-16 18:17:34 +04:00
2007-04-04 01:39:37 +04:00
usbhid - > ctrl [ usbhid - > ctrlhead ] . report = report ;
usbhid - > ctrl [ usbhid - > ctrlhead ] . dir = dir ;
usbhid - > ctrlhead = head ;
2006-09-16 18:17:34 +04:00
2007-04-04 01:39:37 +04:00
if ( ! test_and_set_bit ( HID_CTRL_RUNNING , & usbhid - > iofl ) )
if ( hid_submit_ctrl ( hid ) )
clear_bit ( HID_CTRL_RUNNING , & usbhid - > iofl ) ;
2006-09-15 18:23:35 +04:00
2007-04-04 01:39:37 +04:00
spin_unlock_irqrestore ( & usbhid - > ctrllock , flags ) ;
}
2006-10-27 23:08:54 +04:00
2007-04-04 01:39:37 +04:00
static int usb_hidinput_input_event ( struct input_dev * dev , unsigned int type , unsigned int code , int value )
{
2007-05-09 12:17:31 +04:00
struct hid_device * hid = input_get_drvdata ( dev ) ;
2007-04-04 01:39:37 +04:00
struct hid_field * field ;
int offset ;
2006-11-05 06:49:53 +03:00
2007-04-04 01:39:37 +04:00
if ( type = = EV_FF )
return input_ff_event ( dev , type , code , value ) ;
2007-01-11 12:14:33 +03:00
2007-04-04 01:39:37 +04:00
if ( type ! = EV_LED )
return - 1 ;
2007-01-11 17:51:17 +03:00
2007-04-04 01:39:37 +04:00
if ( ( offset = hidinput_find_field ( hid , type , code , & field ) ) = = - 1 ) {
warn ( " event field not found " ) ;
return - 1 ;
}
2007-01-16 07:11:52 +03:00
2007-04-04 01:39:37 +04:00
hid_set_field ( field , offset , value ) ;
usbhid_submit_report ( hid , field - > report , USB_DIR_OUT ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
return 0 ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
int usbhid_wait_io ( struct hid_device * hid )
{
struct usbhid_device * usbhid = hid - > driver_data ;
2007-03-01 11:54:44 +03:00
2007-04-04 01:39:37 +04:00
if ( ! wait_event_timeout ( hid - > wait , ( ! test_bit ( HID_CTRL_RUNNING , & usbhid - > iofl ) & &
! test_bit ( HID_OUT_RUNNING , & usbhid - > iofl ) ) ,
10 * HZ ) ) {
2007-05-30 17:07:13 +04:00
dbg_hid ( " timeout waiting for ctrl or out queue to clear \n " ) ;
2007-04-04 01:39:37 +04:00
return - 1 ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
return 0 ;
}
2005-06-06 11:22:37 +04:00
2007-04-04 01:39:37 +04:00
static int hid_set_idle ( struct usb_device * dev , int ifnum , int report , int idle )
{
return usb_control_msg ( dev , usb_sndctrlpipe ( dev , 0 ) ,
HID_REQ_SET_IDLE , USB_TYPE_CLASS | USB_RECIP_INTERFACE , ( idle < < 8 ) | report ,
ifnum , NULL , 0 , USB_CTRL_SET_TIMEOUT ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
static int hid_get_class_descriptor ( struct usb_device * dev , int ifnum ,
unsigned char type , void * buf , int size )
{
int result , retries = 4 ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
memset ( buf , 0 , size ) ;
2005-04-17 02:20:36 +04:00
2007-04-04 01:39:37 +04:00
do {
result = usb_control_msg ( dev , usb_rcvctrlpipe ( dev , 0 ) ,
USB_REQ_GET_DESCRIPTOR , USB_RECIP_INTERFACE | USB_DIR_IN ,
( type < < 8 ) , ifnum , buf , size , USB_CTRL_GET_TIMEOUT ) ;
retries - - ;
} while ( result < size & & retries ) ;
return result ;
}
2006-01-14 08:25:39 +03:00
2007-04-04 01:39:37 +04:00
int usbhid_open ( struct hid_device * hid )
{
2007-07-11 16:48:58 +04:00
struct usbhid_device * usbhid = hid - > driver_data ;
int res ;
if ( ! hid - > open + + ) {
res = usb_autopm_get_interface ( usbhid - > intf ) ;
if ( res < 0 ) {
hid - > open - - ;
return - EIO ;
}
}
2007-04-04 01:39:37 +04:00
if ( hid_start_in ( hid ) )
hid_io_error ( hid ) ;
return 0 ;
}
2007-02-05 12:06:01 +03:00
2007-04-04 01:39:37 +04:00
void usbhid_close ( struct hid_device * hid )
{
struct usbhid_device * usbhid = hid - > driver_data ;
2006-01-14 18:08:06 +03:00
2007-07-11 16:48:58 +04:00
if ( ! - - hid - > open ) {
2007-04-04 01:39:37 +04:00
usb_kill_urb ( usbhid - > urbin ) ;
2007-07-11 16:48:58 +04:00
usb_autopm_put_interface ( usbhid - > intf ) ;
}
2007-04-04 01:39:37 +04:00
}
2006-03-30 00:41:07 +04:00
2007-04-04 01:39:37 +04:00
/*
* Initialize all reports
*/
2006-11-05 06:49:53 +03:00
2007-04-04 01:39:37 +04:00
void usbhid_init_reports ( struct hid_device * hid )
{
struct hid_report * report ;
struct usbhid_device * usbhid = hid - > driver_data ;
int err , ret ;
2006-11-05 06:49:53 +03:00
2007-04-04 01:39:37 +04:00
list_for_each_entry ( report , & hid - > report_enum [ HID_INPUT_REPORT ] . report_list , list )
usbhid_submit_report ( hid , report , USB_DIR_IN ) ;
2007-01-11 17:51:17 +03:00
2007-04-04 01:39:37 +04:00
list_for_each_entry ( report , & hid - > report_enum [ HID_FEATURE_REPORT ] . report_list , list )
usbhid_submit_report ( hid , report , USB_DIR_IN ) ;
2007-01-16 07:11:52 +03:00
2007-04-04 01:39:37 +04:00
err = 0 ;
ret = usbhid_wait_io ( hid ) ;
while ( ret ) {
err | = ret ;
if ( test_bit ( HID_CTRL_RUNNING , & usbhid - > iofl ) )
usb_kill_urb ( usbhid - > urbctrl ) ;
if ( test_bit ( HID_OUT_RUNNING , & usbhid - > iofl ) )
usb_kill_urb ( usbhid - > urbout ) ;
ret = usbhid_wait_io ( hid ) ;
}
2007-02-06 03:40:57 +03:00
2007-04-04 01:39:37 +04:00
if ( err )
warn ( " timeout initializing reports " ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-06 16:33:18 +04:00
/*
* Reset LEDs which BIOS might have left on . For now , just NumLock ( 0x01 ) .
*/
static int hid_find_field_early ( struct hid_device * hid , unsigned int page ,
unsigned int hid_code , struct hid_field * * pfield )
{
struct hid_report * report ;
struct hid_field * field ;
struct hid_usage * usage ;
int i , j ;
list_for_each_entry ( report , & hid - > report_enum [ HID_OUTPUT_REPORT ] . report_list , list ) {
for ( i = 0 ; i < report - > maxfield ; i + + ) {
field = report - > field [ i ] ;
for ( j = 0 ; j < field - > maxusage ; j + + ) {
usage = & field - > usage [ j ] ;
if ( ( usage - > hid & HID_USAGE_PAGE ) = = page & &
( usage - > hid & 0xFFFF ) = = hid_code ) {
* pfield = field ;
return j ;
}
}
}
}
return - 1 ;
}
static void usbhid_set_leds ( struct hid_device * hid )
{
struct hid_field * field ;
int offset ;
if ( ( offset = hid_find_field_early ( hid , HID_UP_LED , 0x01 , & field ) ) ! = - 1 ) {
hid_set_field ( field , offset , 0 ) ;
usbhid_submit_report ( hid , field - > report , USB_DIR_OUT ) ;
}
}
2005-09-05 09:12:01 +04:00
/*
* Traverse the supplied list of reports and find the longest
*/
static void hid_find_max_report ( struct hid_device * hid , unsigned int type , int * max )
{
struct hid_report * report ;
int size ;
list_for_each_entry ( report , & hid - > report_enum [ type ] . report_list , list ) {
size = ( ( report - > size - 1 ) > > 3 ) + 1 ;
if ( type = = HID_INPUT_REPORT & & hid - > report_enum [ type ] . numbered )
size + + ;
if ( * max < size )
* max = size ;
}
}
2005-04-17 02:20:36 +04:00
static int hid_alloc_buffers ( struct usb_device * dev , struct hid_device * hid )
{
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid = hid - > driver_data ;
if ( ! ( usbhid - > inbuf = usb_buffer_alloc ( dev , usbhid - > bufsize , GFP_ATOMIC , & usbhid - > inbuf_dma ) ) )
2005-04-17 02:20:36 +04:00
return - 1 ;
2006-12-08 20:41:03 +03:00
if ( ! ( usbhid - > outbuf = usb_buffer_alloc ( dev , usbhid - > bufsize , GFP_ATOMIC , & usbhid - > outbuf_dma ) ) )
2005-04-17 02:20:36 +04:00
return - 1 ;
2006-12-08 20:41:03 +03:00
if ( ! ( usbhid - > cr = usb_buffer_alloc ( dev , sizeof ( * ( usbhid - > cr ) ) , GFP_ATOMIC , & usbhid - > cr_dma ) ) )
2005-04-17 02:20:36 +04:00
return - 1 ;
2006-12-08 20:41:03 +03:00
if ( ! ( usbhid - > ctrlbuf = usb_buffer_alloc ( dev , usbhid - > bufsize , GFP_ATOMIC , & usbhid - > ctrlbuf_dma ) ) )
2005-04-17 02:20:36 +04:00
return - 1 ;
return 0 ;
}
2007-05-14 11:54:30 +04:00
static int usbhid_output_raw_report ( struct hid_device * hid , __u8 * buf , size_t count )
{
struct usbhid_device * usbhid = hid - > driver_data ;
struct usb_device * dev = hid_to_usb_dev ( hid ) ;
struct usb_interface * intf = usbhid - > intf ;
struct usb_host_interface * interface = intf - > cur_altsetting ;
int ret ;
ret = usb_control_msg ( dev , usb_sndctrlpipe ( dev , 0 ) ,
HID_REQ_SET_REPORT ,
USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE ,
cpu_to_le16 ( ( ( HID_OUTPUT_REPORT + 1 ) < < 8 ) | * buf ) ,
interface - > desc . bInterfaceNumber , buf + 1 , count - 1 ,
USB_CTRL_SET_TIMEOUT ) ;
/* count also the report id */
if ( ret > 0 )
ret + + ;
return ret ;
}
2005-04-17 02:20:36 +04:00
static void hid_free_buffers ( struct usb_device * dev , struct hid_device * hid )
{
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid = hid - > driver_data ;
2007-05-03 09:04:52 +04:00
usb_buffer_free ( dev , usbhid - > bufsize , usbhid - > inbuf , usbhid - > inbuf_dma ) ;
usb_buffer_free ( dev , usbhid - > bufsize , usbhid - > outbuf , usbhid - > outbuf_dma ) ;
usb_buffer_free ( dev , sizeof ( * ( usbhid - > cr ) ) , usbhid - > cr , usbhid - > cr_dma ) ;
usb_buffer_free ( dev , usbhid - > bufsize , usbhid - > ctrlbuf , usbhid - > ctrlbuf_dma ) ;
2005-04-17 02:20:36 +04:00
}
2007-01-16 07:11:52 +03:00
/*
* Sending HID_REQ_GET_REPORT changes the operation mode of the ps3 controller
* to " operational " . Without this , the ps3 controller will not report any
* events .
*/
static void hid_fixup_sony_ps3_controller ( struct usb_device * dev , int ifnum )
{
int result ;
char * buf = kmalloc ( 18 , GFP_KERNEL ) ;
if ( ! buf )
return ;
result = usb_control_msg ( dev , usb_rcvctrlpipe ( dev , 0 ) ,
HID_REQ_GET_REPORT ,
USB_DIR_IN | USB_TYPE_CLASS |
USB_RECIP_INTERFACE ,
( 3 < < 8 ) | 0xf2 , ifnum , buf , 17 ,
USB_CTRL_GET_TIMEOUT ) ;
if ( result < 0 )
2007-05-30 17:07:13 +04:00
err_hid ( " %s failed: %d \n " , __func__ , result ) ;
2007-01-16 07:11:52 +03:00
kfree ( buf ) ;
}
2005-04-17 02:20:36 +04:00
static struct hid_device * usb_hid_configure ( struct usb_interface * intf )
{
struct usb_host_interface * interface = intf - > cur_altsetting ;
struct usb_device * dev = interface_to_usbdev ( intf ) ;
struct hid_descriptor * hdesc ;
struct hid_device * hid ;
2007-04-19 15:27:04 +04:00
u32 quirks = 0 ;
2007-03-20 21:23:16 +03:00
unsigned rsize = 0 ;
2005-09-15 11:01:47 +04:00
char * rdesc ;
int n , len , insize = 0 ;
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid ;
2005-04-17 02:20:36 +04:00
2007-04-19 15:27:04 +04:00
quirks = usbhid_lookup_quirk ( le16_to_cpu ( dev - > descriptor . idVendor ) ,
le16_to_cpu ( dev - > descriptor . idProduct ) ) ;
2005-04-17 02:20:36 +04:00
2006-05-15 22:49:04 +04:00
/* Many keyboards and mice don't like to be polled for reports,
* so we will always set the HID_QUIRK_NOGET flag for them . */
if ( interface - > desc . bInterfaceSubClass = = USB_INTERFACE_SUBCLASS_BOOT ) {
if ( interface - > desc . bInterfaceProtocol = = USB_INTERFACE_PROTOCOL_KEYBOARD | |
interface - > desc . bInterfaceProtocol = = USB_INTERFACE_PROTOCOL_MOUSE )
quirks | = HID_QUIRK_NOGET ;
}
2005-04-17 02:20:36 +04:00
if ( quirks & HID_QUIRK_IGNORE )
return NULL ;
2007-02-05 12:06:01 +03:00
if ( ( quirks & HID_QUIRK_IGNORE_MOUSE ) & &
( interface - > desc . bInterfaceProtocol = = USB_INTERFACE_PROTOCOL_MOUSE ) )
return NULL ;
2005-09-15 11:01:47 +04:00
if ( usb_get_extra_descriptor ( interface , HID_DT_HID , & hdesc ) & &
( ! interface - > desc . bNumEndpoints | |
usb_get_extra_descriptor ( & interface - > endpoint [ 0 ] , HID_DT_HID , & hdesc ) ) ) {
2007-05-30 17:07:13 +04:00
dbg_hid ( " class descriptor not present \n " ) ;
2005-09-15 11:01:47 +04:00
return NULL ;
2005-04-17 02:20:36 +04:00
}
for ( n = 0 ; n < hdesc - > bNumDescriptors ; n + + )
if ( hdesc - > desc [ n ] . bDescriptorType = = HID_DT_REPORT )
rsize = le16_to_cpu ( hdesc - > desc [ n ] . wDescriptorLength ) ;
if ( ! rsize | | rsize > HID_MAX_DESCRIPTOR_SIZE ) {
2007-05-30 17:07:13 +04:00
dbg_hid ( " weird size of report descriptor (%u) \n " , rsize ) ;
2005-04-17 02:20:36 +04:00
return NULL ;
}
if ( ! ( rdesc = kmalloc ( rsize , GFP_KERNEL ) ) ) {
2007-05-30 17:07:13 +04:00
dbg_hid ( " couldn't allocate rdesc memory \n " ) ;
2005-04-17 02:20:36 +04:00
return NULL ;
}
2005-05-29 11:28:00 +04:00
hid_set_idle ( dev , interface - > desc . bInterfaceNumber , 0 , 0 ) ;
2005-04-17 02:20:36 +04:00
if ( ( n = hid_get_class_descriptor ( dev , interface - > desc . bInterfaceNumber , HID_DT_REPORT , rdesc , rsize ) ) < 0 ) {
2007-05-30 17:07:13 +04:00
dbg_hid ( " reading report descriptor failed \n " ) ;
2005-04-17 02:20:36 +04:00
kfree ( rdesc ) ;
return NULL ;
}
2007-06-19 16:09:14 +04:00
usbhid_fixup_report_descriptor ( le16_to_cpu ( dev - > descriptor . idVendor ) ,
le16_to_cpu ( dev - > descriptor . idProduct ) , rdesc ,
rsize , rdesc_quirks_param ) ;
2007-05-02 13:55:42 +04:00
2007-05-30 17:07:13 +04:00
dbg_hid ( " report descriptor (size %u, read %d) = " , rsize , n ) ;
2005-04-17 02:20:36 +04:00
for ( n = 0 ; n < rsize ; n + + )
2007-05-30 17:07:13 +04:00
dbg_hid_line ( " %02x " , ( unsigned char ) rdesc [ n ] ) ;
dbg_hid_line ( " \n " ) ;
2005-04-17 02:20:36 +04:00
if ( ! ( hid = hid_parse_report ( rdesc , n ) ) ) {
2007-05-30 17:07:13 +04:00
dbg_hid ( " parsing report descriptor failed \n " ) ;
2005-04-17 02:20:36 +04:00
kfree ( rdesc ) ;
return NULL ;
}
kfree ( rdesc ) ;
hid - > quirks = quirks ;
2006-12-08 20:41:03 +03:00
if ( ! ( usbhid = kzalloc ( sizeof ( struct usbhid_device ) , GFP_KERNEL ) ) )
2007-07-30 17:15:26 +04:00
goto fail_no_usbhid ;
2006-12-08 20:41:03 +03:00
hid - > driver_data = usbhid ;
usbhid - > hid = hid ;
usbhid - > bufsize = HID_MIN_BUFFER_SIZE ;
hid_find_max_report ( hid , HID_INPUT_REPORT , & usbhid - > bufsize ) ;
hid_find_max_report ( hid , HID_OUTPUT_REPORT , & usbhid - > bufsize ) ;
hid_find_max_report ( hid , HID_FEATURE_REPORT , & usbhid - > bufsize ) ;
2005-09-05 09:12:01 +04:00
2006-12-08 20:41:03 +03:00
if ( usbhid - > bufsize > HID_MAX_BUFFER_SIZE )
usbhid - > bufsize = HID_MAX_BUFFER_SIZE ;
2005-09-05 09:12:01 +04:00
hid_find_max_report ( hid , HID_INPUT_REPORT , & insize ) ;
if ( insize > HID_MAX_BUFFER_SIZE )
insize = HID_MAX_BUFFER_SIZE ;
2005-04-17 02:20:36 +04:00
if ( hid_alloc_buffers ( dev , hid ) ) {
hid_free_buffers ( dev , hid ) ;
goto fail ;
}
for ( n = 0 ; n < interface - > desc . bNumEndpoints ; n + + ) {
struct usb_endpoint_descriptor * endpoint ;
int pipe ;
int interval ;
endpoint = & interface - > endpoint [ n ] . desc ;
if ( ( endpoint - > bmAttributes & 3 ) ! = 3 ) /* Not an interrupt endpoint */
continue ;
interval = endpoint - > bInterval ;
/* Change the polling interval of mice. */
if ( hid - > collection - > usage = = HID_GD_MOUSE & & hid_mousepoll_interval > 0 )
interval = hid_mousepoll_interval ;
2005-05-29 11:29:01 +04:00
2006-10-26 20:02:51 +04:00
if ( usb_endpoint_dir_in ( endpoint ) ) {
2006-12-08 20:41:03 +03:00
if ( usbhid - > urbin )
2005-04-17 02:20:36 +04:00
continue ;
2006-12-08 20:41:03 +03:00
if ( ! ( usbhid - > urbin = usb_alloc_urb ( 0 , GFP_KERNEL ) ) )
2005-04-17 02:20:36 +04:00
goto fail ;
pipe = usb_rcvintpipe ( dev , endpoint - > bEndpointAddress ) ;
2006-12-08 20:41:03 +03:00
usb_fill_int_urb ( usbhid - > urbin , dev , pipe , usbhid - > inbuf , insize ,
2005-04-17 02:20:36 +04:00
hid_irq_in , hid , interval ) ;
2006-12-08 20:41:03 +03:00
usbhid - > urbin - > transfer_dma = usbhid - > inbuf_dma ;
usbhid - > urbin - > transfer_flags | = URB_NO_TRANSFER_DMA_MAP ;
2005-04-17 02:20:36 +04:00
} else {
2006-12-08 20:41:03 +03:00
if ( usbhid - > urbout )
2005-04-17 02:20:36 +04:00
continue ;
2006-12-08 20:41:03 +03:00
if ( ! ( usbhid - > urbout = usb_alloc_urb ( 0 , GFP_KERNEL ) ) )
2005-04-17 02:20:36 +04:00
goto fail ;
pipe = usb_sndintpipe ( dev , endpoint - > bEndpointAddress ) ;
2006-12-08 20:41:03 +03:00
usb_fill_int_urb ( usbhid - > urbout , dev , pipe , usbhid - > outbuf , 0 ,
2005-04-17 02:20:36 +04:00
hid_irq_out , hid , interval ) ;
2006-12-08 20:41:03 +03:00
usbhid - > urbout - > transfer_dma = usbhid - > outbuf_dma ;
usbhid - > urbout - > transfer_flags | = URB_NO_TRANSFER_DMA_MAP ;
2005-04-17 02:20:36 +04:00
}
}
2006-12-08 20:41:03 +03:00
if ( ! usbhid - > urbin ) {
2007-05-30 17:07:13 +04:00
err_hid ( " couldn't find an input interrupt endpoint " ) ;
2005-04-17 02:20:36 +04:00
goto fail ;
}
init_waitqueue_head ( & hid - > wait ) ;
2006-12-08 20:41:03 +03:00
INIT_WORK ( & usbhid - > reset_work , hid_reset ) ;
setup_timer ( & usbhid - > io_retry , hid_retry_timeout , ( unsigned long ) hid ) ;
2006-01-31 20:58:38 +03:00
2006-12-08 20:41:03 +03:00
spin_lock_init ( & usbhid - > inlock ) ;
spin_lock_init ( & usbhid - > outlock ) ;
spin_lock_init ( & usbhid - > ctrllock ) ;
2005-04-17 02:20:36 +04:00
hid - > version = le16_to_cpu ( hdesc - > bcdHID ) ;
hid - > country = hdesc - > bCountryCode ;
2007-01-19 20:28:17 +03:00
hid - > dev = & intf - > dev ;
2006-12-08 20:41:03 +03:00
usbhid - > intf = intf ;
usbhid - > ifnum = interface - > desc . bInterfaceNumber ;
2005-04-17 02:20:36 +04:00
hid - > name [ 0 ] = 0 ;
2005-09-15 11:01:47 +04:00
if ( dev - > manufacturer )
strlcpy ( hid - > name , dev - > manufacturer , sizeof ( hid - > name ) ) ;
if ( dev - > product ) {
if ( dev - > manufacturer )
strlcat ( hid - > name , " " , sizeof ( hid - > name ) ) ;
strlcat ( hid - > name , dev - > product , sizeof ( hid - > name ) ) ;
}
if ( ! strlen ( hid - > name ) )
snprintf ( hid - > name , sizeof ( hid - > name ) , " HID %04x:%04x " ,
le16_to_cpu ( dev - > descriptor . idVendor ) ,
le16_to_cpu ( dev - > descriptor . idProduct ) ) ;
2005-04-17 02:20:36 +04:00
2006-12-08 20:40:53 +03:00
hid - > bus = BUS_USB ;
2007-02-11 20:20:25 +03:00
hid - > vendor = le16_to_cpu ( dev - > descriptor . idVendor ) ;
hid - > product = le16_to_cpu ( dev - > descriptor . idProduct ) ;
2006-12-08 20:40:53 +03:00
2005-09-15 11:01:47 +04:00
usb_make_path ( dev , hid - > phys , sizeof ( hid - > phys ) ) ;
strlcat ( hid - > phys , " /input " , sizeof ( hid - > phys ) ) ;
len = strlen ( hid - > phys ) ;
if ( len < sizeof ( hid - > phys ) - 1 )
snprintf ( hid - > phys + len , sizeof ( hid - > phys ) - len ,
" %d " , intf - > altsetting [ 0 ] . desc . bInterfaceNumber ) ;
2005-04-17 02:20:36 +04:00
if ( usb_string ( dev , dev - > descriptor . iSerialNumber , hid - > uniq , 64 ) < = 0 )
hid - > uniq [ 0 ] = 0 ;
2006-12-08 20:41:03 +03:00
usbhid - > urbctrl = usb_alloc_urb ( 0 , GFP_KERNEL ) ;
if ( ! usbhid - > urbctrl )
2005-04-17 02:20:36 +04:00
goto fail ;
2005-09-15 11:01:47 +04:00
2006-12-08 20:41:03 +03:00
usb_fill_control_urb ( usbhid - > urbctrl , dev , 0 , ( void * ) usbhid - > cr ,
usbhid - > ctrlbuf , 1 , hid_ctrl , hid ) ;
usbhid - > urbctrl - > setup_dma = usbhid - > cr_dma ;
usbhid - > urbctrl - > transfer_dma = usbhid - > ctrlbuf_dma ;
usbhid - > urbctrl - > transfer_flags | = ( URB_NO_TRANSFER_DMA_MAP | URB_NO_SETUP_DMA_MAP ) ;
2006-12-08 20:40:53 +03:00
hid - > hidinput_input_event = usb_hidinput_input_event ;
2007-01-24 13:54:19 +03:00
hid - > hid_open = usbhid_open ;
hid - > hid_close = usbhid_close ;
2006-12-08 20:41:10 +03:00
# ifdef CONFIG_USB_HIDDEV
hid - > hiddev_hid_event = hiddev_hid_event ;
2006-12-08 20:41:17 +03:00
hid - > hiddev_report_event = hiddev_report_event ;
2006-12-08 20:41:10 +03:00
# endif
2007-05-14 11:54:30 +04:00
hid - > hid_output_raw_report = usbhid_output_raw_report ;
2005-04-17 02:20:36 +04:00
return hid ;
fail :
2006-12-08 20:41:03 +03:00
usb_free_urb ( usbhid - > urbin ) ;
usb_free_urb ( usbhid - > urbout ) ;
usb_free_urb ( usbhid - > urbctrl ) ;
2007-08-01 14:32:27 +04:00
hid_free_buffers ( dev , hid ) ;
2007-07-30 17:40:06 +04:00
kfree ( usbhid ) ;
2007-07-30 17:15:26 +04:00
fail_no_usbhid :
2005-04-17 02:20:36 +04:00
hid_free_device ( hid ) ;
return NULL ;
}
static void hid_disconnect ( struct usb_interface * intf )
{
struct hid_device * hid = usb_get_intfdata ( intf ) ;
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid ;
2005-04-17 02:20:36 +04:00
if ( ! hid )
return ;
2006-12-08 20:41:03 +03:00
usbhid = hid - > driver_data ;
spin_lock_irq ( & usbhid - > inlock ) ; /* Sync with error handler */
2005-04-17 02:20:36 +04:00
usb_set_intfdata ( intf , NULL ) ;
2006-12-08 20:41:03 +03:00
spin_unlock_irq ( & usbhid - > inlock ) ;
usb_kill_urb ( usbhid - > urbin ) ;
usb_kill_urb ( usbhid - > urbout ) ;
usb_kill_urb ( usbhid - > urbctrl ) ;
2005-04-17 02:20:36 +04:00
2006-12-08 20:41:03 +03:00
del_timer_sync ( & usbhid - > io_retry ) ;
2007-05-30 19:11:12 +04:00
cancel_work_sync ( & usbhid - > reset_work ) ;
2006-01-31 20:58:38 +03:00
2005-04-17 02:20:36 +04:00
if ( hid - > claimed & HID_CLAIMED_INPUT )
hidinput_disconnect ( hid ) ;
if ( hid - > claimed & HID_CLAIMED_HIDDEV )
hiddev_disconnect ( hid ) ;
2007-05-14 11:57:40 +04:00
if ( hid - > claimed & HID_CLAIMED_HIDRAW )
hidraw_disconnect ( hid ) ;
2005-04-17 02:20:36 +04:00
2006-12-08 20:41:03 +03:00
usb_free_urb ( usbhid - > urbin ) ;
usb_free_urb ( usbhid - > urbctrl ) ;
usb_free_urb ( usbhid - > urbout ) ;
2005-04-17 02:20:36 +04:00
2007-01-19 20:28:17 +03:00
hid_free_buffers ( hid_to_usb_dev ( hid ) , hid ) ;
2007-08-01 14:32:27 +04:00
kfree ( usbhid ) ;
2005-04-17 02:20:36 +04:00
hid_free_device ( hid ) ;
}
static int hid_probe ( struct usb_interface * intf , const struct usb_device_id * id )
{
struct hid_device * hid ;
char path [ 64 ] ;
int i ;
char * c ;
2007-05-30 17:07:13 +04:00
dbg_hid ( " HID probe called for ifnum %d \n " ,
2005-04-17 02:20:36 +04:00
intf - > altsetting - > desc . bInterfaceNumber ) ;
if ( ! ( hid = usb_hid_configure ( intf ) ) )
2005-06-22 19:53:28 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
2006-12-08 20:41:03 +03:00
usbhid_init_reports ( hid ) ;
2005-04-17 02:20:36 +04:00
hid_dump_device ( hid ) ;
2007-04-06 16:33:18 +04:00
if ( hid - > quirks & HID_QUIRK_RESET_LEDS )
usbhid_set_leds ( hid ) ;
2005-04-17 02:20:36 +04:00
if ( ! hidinput_connect ( hid ) )
hid - > claimed | = HID_CLAIMED_INPUT ;
if ( ! hiddev_connect ( hid ) )
hid - > claimed | = HID_CLAIMED_HIDDEV ;
2007-05-14 11:57:40 +04:00
if ( ! hidraw_connect ( hid ) )
hid - > claimed | = HID_CLAIMED_HIDRAW ;
2005-04-17 02:20:36 +04:00
usb_set_intfdata ( intf , hid ) ;
if ( ! hid - > claimed ) {
2007-05-14 11:57:40 +04:00
printk ( " HID device claimed by neither input, hiddev nor hidraw \n " ) ;
2005-04-17 02:20:36 +04:00
hid_disconnect ( intf ) ;
2005-06-22 19:53:28 +04:00
return - ENODEV ;
2005-04-17 02:20:36 +04:00
}
2007-01-11 17:51:16 +03:00
if ( ( hid - > claimed & HID_CLAIMED_INPUT ) )
2006-12-08 20:41:03 +03:00
hid_ff_init ( hid ) ;
2007-01-16 07:11:52 +03:00
if ( hid - > quirks & HID_QUIRK_SONY_PS3_CONTROLLER )
hid_fixup_sony_ps3_controller ( interface_to_usbdev ( intf ) ,
intf - > cur_altsetting - > desc . bInterfaceNumber ) ;
2005-04-17 02:20:36 +04:00
printk ( KERN_INFO ) ;
if ( hid - > claimed & HID_CLAIMED_INPUT )
printk ( " input " ) ;
2007-05-14 11:57:40 +04:00
if ( ( hid - > claimed & HID_CLAIMED_INPUT ) & & ( ( hid - > claimed & HID_CLAIMED_HIDDEV ) | |
hid - > claimed & HID_CLAIMED_HIDRAW ) )
2005-04-17 02:20:36 +04:00
printk ( " , " ) ;
if ( hid - > claimed & HID_CLAIMED_HIDDEV )
printk ( " hiddev%d " , hid - > minor ) ;
2007-05-14 11:57:40 +04:00
if ( ( hid - > claimed & HID_CLAIMED_INPUT ) & & ( hid - > claimed & HID_CLAIMED_HIDDEV ) & &
( hid - > claimed & HID_CLAIMED_HIDRAW ) )
printk ( " , " ) ;
if ( hid - > claimed & HID_CLAIMED_HIDRAW )
printk ( " hidraw%d " , ( ( struct hidraw * ) hid - > hidraw ) - > minor ) ;
2005-04-17 02:20:36 +04:00
c = " Device " ;
for ( i = 0 ; i < hid - > maxcollection ; i + + ) {
if ( hid - > collection [ i ] . type = = HID_COLLECTION_APPLICATION & &
( hid - > collection [ i ] . usage & HID_USAGE_PAGE ) = = HID_UP_GENDESK & &
( hid - > collection [ i ] . usage & 0xffff ) < ARRAY_SIZE ( hid_types ) ) {
c = hid_types [ hid - > collection [ i ] . usage & 0xffff ] ;
break ;
}
}
usb_make_path ( interface_to_usbdev ( intf ) , path , 63 ) ;
printk ( " : USB HID v%x.%02x %s [%s] on %s \n " ,
hid - > version > > 8 , hid - > version & 0xff , c , hid - > name , path ) ;
return 0 ;
}
2005-04-19 04:39:22 +04:00
static int hid_suspend ( struct usb_interface * intf , pm_message_t message )
2005-04-17 02:20:36 +04:00
{
struct hid_device * hid = usb_get_intfdata ( intf ) ;
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid = hid - > driver_data ;
2005-04-17 02:20:36 +04:00
2006-12-08 20:41:03 +03:00
spin_lock_irq ( & usbhid - > inlock ) ; /* Sync with error handler */
set_bit ( HID_SUSPENDED , & usbhid - > iofl ) ;
spin_unlock_irq ( & usbhid - > inlock ) ;
del_timer ( & usbhid - > io_retry ) ;
usb_kill_urb ( usbhid - > urbin ) ;
2005-04-17 02:20:36 +04:00
dev_dbg ( & intf - > dev , " suspend \n " ) ;
return 0 ;
}
static int hid_resume ( struct usb_interface * intf )
{
struct hid_device * hid = usb_get_intfdata ( intf ) ;
2006-12-08 20:41:03 +03:00
struct usbhid_device * usbhid = hid - > driver_data ;
2005-04-17 02:20:36 +04:00
int status ;
2006-12-08 20:41:03 +03:00
clear_bit ( HID_SUSPENDED , & usbhid - > iofl ) ;
usbhid - > retry_delay = 0 ;
2006-01-31 20:58:38 +03:00
status = hid_start_in ( hid ) ;
2005-04-17 02:20:36 +04:00
dev_dbg ( & intf - > dev , " resume status %d \n " , status ) ;
return status ;
}
2006-06-01 21:55:28 +04:00
/* Treat USB reset pretty much the same as suspend/resume */
2007-05-30 23:38:16 +04:00
static int hid_pre_reset ( struct usb_interface * intf )
2006-06-01 21:55:28 +04:00
{
/* FIXME: What if the interface is already suspended? */
hid_suspend ( intf , PMSG_ON ) ;
2007-05-30 23:38:16 +04:00
return 0 ;
2006-06-01 21:55:28 +04:00
}
2007-05-30 23:38:16 +04:00
/* Same routine used for post_reset and reset_resume */
static int hid_post_reset ( struct usb_interface * intf )
2006-06-01 21:55:28 +04:00
{
struct usb_device * dev = interface_to_usbdev ( intf ) ;
hid_set_idle ( dev , intf - > cur_altsetting - > desc . bInterfaceNumber , 0 , 0 ) ;
/* FIXME: Any more reinitialization needed? */
2007-05-30 23:38:16 +04:00
return hid_resume ( intf ) ;
2006-06-01 21:55:28 +04:00
}
2005-04-17 02:20:36 +04:00
static struct usb_device_id hid_usb_ids [ ] = {
{ . match_flags = USB_DEVICE_ID_MATCH_INT_CLASS ,
. bInterfaceClass = USB_INTERFACE_CLASS_HID } ,
{ } /* Terminating entry */
} ;
MODULE_DEVICE_TABLE ( usb , hid_usb_ids ) ;
static struct usb_driver hid_driver = {
. name = " usbhid " ,
. probe = hid_probe ,
. disconnect = hid_disconnect ,
. suspend = hid_suspend ,
. resume = hid_resume ,
2007-05-30 23:38:16 +04:00
. reset_resume = hid_post_reset ,
2006-06-01 21:55:28 +04:00
. pre_reset = hid_pre_reset ,
. post_reset = hid_post_reset ,
2005-04-17 02:20:36 +04:00
. id_table = hid_usb_ids ,
2007-07-11 16:48:58 +04:00
. supports_autosuspend = 1 ,
2005-04-17 02:20:36 +04:00
} ;
static int __init hid_init ( void )
{
int retval ;
2007-04-19 16:56:12 +04:00
retval = usbhid_quirks_init ( quirks_param ) ;
if ( retval )
goto usbhid_quirks_init_fail ;
2005-04-17 02:20:36 +04:00
retval = hiddev_init ( ) ;
if ( retval )
goto hiddev_init_fail ;
retval = usb_register ( & hid_driver ) ;
if ( retval )
goto usb_register_fail ;
info ( DRIVER_VERSION " : " DRIVER_DESC ) ;
return 0 ;
usb_register_fail :
hiddev_exit ( ) ;
hiddev_init_fail :
2007-04-19 16:56:12 +04:00
usbhid_quirks_exit ( ) ;
usbhid_quirks_init_fail :
2005-04-17 02:20:36 +04:00
return retval ;
}
static void __exit hid_exit ( void )
{
usb_deregister ( & hid_driver ) ;
hiddev_exit ( ) ;
2007-04-19 16:56:12 +04:00
usbhid_quirks_exit ( ) ;
2005-04-17 02:20:36 +04:00
}
module_init ( hid_init ) ;
module_exit ( hid_exit ) ;
MODULE_AUTHOR ( DRIVER_AUTHOR ) ;
MODULE_DESCRIPTION ( DRIVER_DESC ) ;
MODULE_LICENSE ( DRIVER_LICENSE ) ;