2007-01-26 14:26:21 +01:00
/*
* USB device quirk handling logic and table
*
* Copyright ( c ) 2007 Oliver Neukum
* Copyright ( c ) 2007 Greg Kroah - Hartman < gregkh @ suse . de >
*
* 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 , version 2.
*
*
*/
# include <linux/usb.h>
# include <linux/usb/quirks.h>
# include "usb.h"
2012-07-19 12:39:13 +02:00
/* Lists of quirky USB devices, split in device quirks and interface quirks.
* Device quirks are applied at the very beginning of the enumeration process ,
* right after reading the device descriptor . They can thus only match on device
* information .
*
* Interface quirks are applied after reading all the configuration descriptors .
* They can match on both device and interface information .
*
* Note that the DELAY_INIT and HONOR_BNUMINTERFACES quirks do not make sense as
* interface quirks , as they only influence the enumeration process which is run
* before processing the interface quirks .
*
* Please keep the lists ordered by :
2007-01-26 14:26:21 +01:00
* 1 ) Vendor ID
* 2 ) Product ID
* 3 ) Class ID
*/
static const struct usb_device_id usb_quirk_list [ ] = {
2007-09-04 16:11:41 +02:00
/* CBM - Flash disk */
{ USB_DEVICE ( 0x0204 , 0x6025 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2008-02-27 15:43:47 -05:00
2007-01-26 14:26:21 +01:00
/* HP 5300/5370C scanner */
2008-02-27 15:43:47 -05:00
{ USB_DEVICE ( 0x03f0 , 0x0701 ) , . driver_info =
USB_QUIRK_STRING_FETCH_255 } ,
2007-08-28 10:34:22 +02:00
2008-02-04 16:34:11 +01:00
/* Creative SB Audigy 2 NX */
{ USB_DEVICE ( 0x041e , 0x3020 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2012-12-05 16:17:49 +01:00
/* Microsoft LifeCam-VX700 v2.0 */
{ USB_DEVICE ( 0x045e , 0x0770 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2012-07-19 12:39:14 +02:00
/* Logitech Quickcam Fusion */
{ USB_DEVICE ( 0x046d , 0x08c1 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-09-13 08:42:21 +02:00
2012-07-19 12:39:14 +02:00
/* Logitech Quickcam Orbit MP */
{ USB_DEVICE ( 0x046d , 0x08c2 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-09-13 08:42:21 +02:00
2012-07-19 12:39:14 +02:00
/* Logitech Quickcam Pro for Notebook */
{ USB_DEVICE ( 0x046d , 0x08c3 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-09-29 20:42:52 +01:00
2012-07-19 12:39:14 +02:00
/* Logitech Quickcam Pro 5000 */
{ USB_DEVICE ( 0x046d , 0x08c5 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-09-13 08:42:21 +02:00
2012-07-19 12:39:14 +02:00
/* Logitech Quickcam OEM Dell Notebook */
{ USB_DEVICE ( 0x046d , 0x08c6 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-10-26 13:53:17 -04:00
2012-07-19 12:39:14 +02:00
/* Logitech Quickcam OEM Cisco VT Camera II */
{ USB_DEVICE ( 0x046d , 0x08c7 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-10-27 21:06:26 -07:00
2010-07-22 00:05:01 +02:00
/* Logitech Harmony 700-series */
{ USB_DEVICE ( 0x046d , 0xc122 ) , . driver_info = USB_QUIRK_DELAY_INIT } ,
2008-02-27 15:43:47 -05:00
/* Philips PSC805 audio device */
{ USB_DEVICE ( 0x0471 , 0x0155 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2010-07-09 13:18:50 +01:00
/* Artisman Watchdog Dongle */
{ USB_DEVICE ( 0x04b4 , 0x0526 ) , . driver_info =
USB_QUIRK_CONFIG_INTF_STRINGS } ,
2012-09-04 10:41:02 -04:00
/* Microchip Joss Optical infrared touchboard device */
{ USB_DEVICE ( 0x04d8 , 0x000c ) , . driver_info =
USB_QUIRK_CONFIG_INTF_STRINGS } ,
2013-08-14 11:01:46 +02:00
/* CarrolTouch 4000U */
{ USB_DEVICE ( 0x04e7 , 0x0009 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
/* CarrolTouch 4500U */
{ USB_DEVICE ( 0x04e7 , 0x0030 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-02-05 21:52:00 +01:00
/* Samsung Android phone modem - ID conflict with SPH-I500 */
{ USB_DEVICE ( 0x04e8 , 0x6601 ) , . driver_info =
USB_QUIRK_CONFIG_INTF_STRINGS } ,
2008-02-04 16:34:11 +01:00
/* Roland SC-8820 */
{ USB_DEVICE ( 0x0582 , 0x0007 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
/* Edirol SD-20 */
{ USB_DEVICE ( 0x0582 , 0x0027 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2013-04-30 10:18:04 +02:00
/* Alcor Micro Corp. Hub */
{ USB_DEVICE ( 0x058f , 0x9254 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2008-06-17 11:56:55 -04:00
/* appletouch */
{ USB_DEVICE ( 0x05ac , 0x021a ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2008-05-27 09:05:46 +02:00
/* Avision AV600U */
{ USB_DEVICE ( 0x0638 , 0x0a13 ) , . driver_info =
USB_QUIRK_STRING_FETCH_255 } ,
2009-03-18 14:28:53 -04:00
/* Saitek Cyborg Gold Joystick */
{ USB_DEVICE ( 0x06a3 , 0x0006 ) , . driver_info =
USB_QUIRK_CONFIG_INTF_STRINGS } ,
2012-01-03 09:58:54 +01:00
/* Guillemot Webcam Hercules Dualpix Exchange (2nd ID) */
2011-09-13 08:42:21 +02:00
{ USB_DEVICE ( 0x06f8 , 0x0804 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2012-01-03 09:58:54 +01:00
/* Guillemot Webcam Hercules Dualpix Exchange*/
{ USB_DEVICE ( 0x06f8 , 0x3005 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2012-04-30 13:05:34 +02:00
/* Midiman M-Audio Keystation 88es */
{ USB_DEVICE ( 0x0763 , 0x0192 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2007-09-04 12:15:08 -03:00
/* M-Systems Flash Disk Pioneers */
{ USB_DEVICE ( 0x08ec , 0x1000 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2011-02-17 10:26:38 -05:00
/* Keytouch QWERTY Panel keyboard */
{ USB_DEVICE ( 0x0926 , 0x3333 ) , . driver_info =
USB_QUIRK_CONFIG_INTF_STRINGS } ,
2008-03-11 10:20:12 -04:00
/* X-Rite/Gretag-Macbeth Eye-One Pro display colorimeter */
{ USB_DEVICE ( 0x0971 , 0x2000 ) , . driver_info = USB_QUIRK_NO_SET_INTF } ,
2010-07-14 18:26:22 +02:00
/* Broadcom BCM92035DGROM BT dongle */
{ USB_DEVICE ( 0x0a5c , 0x2021 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2008-02-27 15:43:47 -05:00
/* Action Semiconductor flash disk */
{ USB_DEVICE ( 0x10d6 , 0x2200 ) , . driver_info =
USB_QUIRK_STRING_FETCH_255 } ,
2007-06-19 17:16:10 -04:00
2007-09-04 12:15:08 -03:00
/* SKYMEDI USB_DRIVE */
{ USB_DEVICE ( 0x1516 , 0x8628 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2007-01-26 14:26:21 +01:00
2010-03-29 12:03:17 +02:00
/* BUILDWIN Photo Frame */
{ USB_DEVICE ( 0x1908 , 0x1315 ) , . driver_info =
USB_QUIRK_HONOR_BNUMINTERFACES } ,
2008-02-27 15:43:47 -05:00
/* INTEL VALUE SSD */
{ USB_DEVICE ( 0x8086 , 0xf1a5 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2007-01-26 14:26:21 +01:00
{ } /* terminating entry must be last */
} ;
2012-07-19 12:39:13 +02:00
static const struct usb_device_id usb_interface_quirk_list [ ] = {
2012-07-19 12:39:14 +02:00
/* Logitech UVC Cameras */
{ USB_VENDOR_AND_INTERFACE_INFO ( 0x046d , USB_CLASS_VIDEO , 1 , 0 ) ,
. driver_info = USB_QUIRK_RESET_RESUME } ,
2012-07-19 12:39:13 +02:00
{ } /* terminating entry must be last */
} ;
static bool usb_match_any_interface ( struct usb_device * udev ,
const struct usb_device_id * id )
{
unsigned int i ;
for ( i = 0 ; i < udev - > descriptor . bNumConfigurations ; + + i ) {
struct usb_host_config * cfg = & udev - > config [ i ] ;
unsigned int j ;
for ( j = 0 ; j < cfg - > desc . bNumInterfaces ; + + j ) {
struct usb_interface_cache * cache ;
struct usb_host_interface * intf ;
cache = cfg - > intf_cache [ j ] ;
if ( cache - > num_altsetting = = 0 )
continue ;
intf = & cache - > altsetting [ 0 ] ;
if ( usb_match_one_id_intf ( udev , intf , id ) )
return true ;
}
}
return false ;
}
static u32 __usb_detect_quirks ( struct usb_device * udev ,
const struct usb_device_id * id )
2007-01-26 14:26:21 +01:00
{
2012-07-19 12:39:13 +02:00
u32 quirks = 0 ;
2007-01-26 14:26:21 +01:00
2012-07-19 12:39:13 +02:00
for ( ; id - > match_flags ; id + + ) {
if ( ! usb_match_device ( udev , id ) )
continue ;
if ( ( id - > match_flags & USB_DEVICE_ID_MATCH_INT_INFO ) & &
! usb_match_any_interface ( udev , id ) )
continue ;
quirks | = ( u32 ) ( id - > driver_info ) ;
2007-01-26 14:26:21 +01:00
}
2012-07-19 12:39:13 +02:00
return quirks ;
2007-01-26 14:26:21 +01:00
}
/*
* Detect any quirks the device has , and do any housekeeping for it if needed .
*/
void usb_detect_quirks ( struct usb_device * udev )
{
2012-07-19 12:39:13 +02:00
udev - > quirks = __usb_detect_quirks ( udev , usb_quirk_list ) ;
2007-01-26 14:26:21 +01:00
if ( udev - > quirks )
dev_dbg ( & udev - > dev , " USB quirks for this device: %x \n " ,
2012-07-19 12:39:13 +02:00
udev - > quirks ) ;
2007-01-26 14:26:21 +01:00
2013-03-13 15:57:31 -07:00
# ifdef CONFIG_USB_DEFAULT_PERSIST
if ( ! ( udev - > quirks & USB_QUIRK_RESET ) )
2008-03-03 15:15:59 -05:00
udev - > persist_enabled = 1 ;
# else
2013-03-13 15:57:31 -07:00
/* Hubs are automatically enabled for USB-PERSIST */
if ( udev - > descriptor . bDeviceClass = = USB_CLASS_HUB )
2009-12-18 12:14:21 +01:00
udev - > persist_enabled = 1 ;
2013-03-13 15:57:31 -07:00
# endif /* CONFIG_USB_DEFAULT_PERSIST */
2007-01-26 14:26:21 +01:00
}
2012-07-19 12:39:13 +02:00
void usb_detect_interface_quirks ( struct usb_device * udev )
{
u32 quirks ;
quirks = __usb_detect_quirks ( udev , usb_interface_quirk_list ) ;
if ( quirks = = 0 )
return ;
dev_dbg ( & udev - > dev , " USB interface quirks for this device: %x \n " ,
quirks ) ;
udev - > quirks | = quirks ;
}