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>
2013-09-16 23:47:28 +08:00
# include <linux/usb/hcd.h>
2007-01-26 14:26:21 +01:00
# 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 } ,
2016-05-19 17:12:19 +02:00
/* USB3503 */
{ USB_DEVICE ( 0x0424 , 0x3503 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2014-11-24 11:22:38 +01:00
/* Microsoft Wireless Laser Mouse 6000 Receiver */
{ USB_DEVICE ( 0x045e , 0x00e1 ) , . 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 } ,
2014-03-04 10:52:39 -08:00
/* Logitech HD Pro Webcams C920 and C930e */
{ USB_DEVICE ( 0x046d , 0x082d ) , . driver_info = USB_QUIRK_DELAY_INIT } ,
{ USB_DEVICE ( 0x046d , 0x0843 ) , . driver_info = USB_QUIRK_DELAY_INIT } ,
2015-10-01 14:10:22 -07:00
/* Logitech ConferenceCam CC3000e */
{ USB_DEVICE ( 0x046d , 0x0847 ) , . driver_info = USB_QUIRK_DELAY_INIT } ,
{ USB_DEVICE ( 0x046d , 0x0848 ) , . driver_info = USB_QUIRK_DELAY_INIT } ,
/* Logitech PTZ Pro Camera */
{ USB_DEVICE ( 0x046d , 0x0853 ) , . driver_info = USB_QUIRK_DELAY_INIT } ,
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 } ,
2015-08-31 14:24:09 +08:00
/* Plantronic Audio 655 DSP */
{ USB_DEVICE ( 0x047f , 0xc008 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
/* Plantronic Audio 648 USB */
{ USB_DEVICE ( 0x047f , 0xc013 ) , . 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 } ,
2014-08-25 17:51:27 +02:00
/* Elan Touchscreen */
{ USB_DEVICE ( 0x04f3 , 0x0089 ) , . driver_info =
USB_QUIRK_DEVICE_QUALIFIER } ,
2014-10-09 09:29:29 +02:00
{ USB_DEVICE ( 0x04f3 , 0x009b ) , . driver_info =
USB_QUIRK_DEVICE_QUALIFIER } ,
2014-11-17 17:11:42 +01:00
{ USB_DEVICE ( 0x04f3 , 0x010c ) , . driver_info =
USB_QUIRK_DEVICE_QUALIFIER } ,
2015-05-09 11:09:11 -06:00
{ USB_DEVICE ( 0x04f3 , 0x0125 ) , . driver_info =
USB_QUIRK_DEVICE_QUALIFIER } ,
2014-10-09 09:29:30 +02:00
{ USB_DEVICE ( 0x04f3 , 0x016f ) , . driver_info =
USB_QUIRK_DEVICE_QUALIFIER } ,
2016-07-06 21:18:51 -04:00
{ USB_DEVICE ( 0x04f3 , 0x0381 ) , . driver_info =
USB_QUIRK_NO_LPM } ,
2015-11-24 16:02:04 +01:00
{ USB_DEVICE ( 0x04f3 , 0x21b8 ) , . driver_info =
USB_QUIRK_DEVICE_QUALIFIER } ,
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 } ,
2013-10-14 16:22:40 +02:00
/* MAYA44USB sound device */
{ USB_DEVICE ( 0x0a92 , 0x0091 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
2016-05-19 17:12:19 +02:00
/* ASUS Base Station(T100) */
{ USB_DEVICE ( 0x0b05 , 0x17e0 ) , . driver_info =
USB_QUIRK_IGNORE_REMOTE_WAKEUP } ,
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
2014-07-27 13:28:04 -04:00
/* Razer - Razer Blade Keyboard */
{ USB_DEVICE ( 0x1532 , 0x0116 ) , . driver_info =
USB_QUIRK_LINEAR_UFRAME_INTR_BINTERVAL } ,
2010-03-29 12:03:17 +02:00
/* BUILDWIN Photo Frame */
{ USB_DEVICE ( 0x1908 , 0x1315 ) , . driver_info =
USB_QUIRK_HONOR_BNUMINTERFACES } ,
2015-01-23 14:39:02 +08:00
/* Protocol and OTG Electrical Test Device */
{ USB_DEVICE ( 0x1a0a , 0x0200 ) , . driver_info =
USB_QUIRK_LINEAR_UFRAME_INTR_BINTERVAL } ,
2016-05-19 17:12:20 +02:00
/* Acer C120 LED Projector */
{ USB_DEVICE ( 0x1de1 , 0xc102 ) , . driver_info = USB_QUIRK_NO_LPM } ,
2015-12-10 15:27:21 -05:00
/* Blackmagic Design Intensity Shuttle */
{ USB_DEVICE ( 0x1edb , 0xbd3b ) , . driver_info = USB_QUIRK_NO_LPM } ,
/* Blackmagic Design UltraStudio SDI */
{ USB_DEVICE ( 0x1edb , 0xbd4f ) , . driver_info = USB_QUIRK_NO_LPM } ,
2016-05-19 17:12:19 +02: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 */
} ;
2013-09-16 23:47:28 +08:00
static const struct usb_device_id usb_amd_resume_quirk_list [ ] = {
/* Lenovo Mouse with Pixart controller */
{ USB_DEVICE ( 0x17ef , 0x602e ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
/* Pixart Mouse */
{ USB_DEVICE ( 0x093a , 0x2500 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
{ USB_DEVICE ( 0x093a , 0x2510 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
{ USB_DEVICE ( 0x093a , 0x2521 ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
/* Logitech Optical Mouse M90/M100 */
{ USB_DEVICE ( 0x046d , 0xc05a ) , . driver_info = USB_QUIRK_RESET_RESUME } ,
{ } /* terminating entry must be last */
} ;
2012-07-19 12:39:13 +02:00
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 ;
}
2013-09-26 11:56:50 -07:00
static int usb_amd_resume_quirk ( struct usb_device * udev )
2013-09-16 23:47:28 +08:00
{
struct usb_hcd * hcd ;
hcd = bus_to_hcd ( udev - > bus ) ;
/* The device should be attached directly to root hub */
if ( udev - > level = = 1 & & hcd - > amd_resume_bug = = 1 )
return 1 ;
return 0 ;
}
2012-07-19 12:39:13 +02:00
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 ) ;
2013-09-16 23:47:28 +08:00
/*
* Pixart - based mice would trigger remote wakeup issue on AMD
* Yangtze chipset , so set them as RESET_RESUME flag .
*/
if ( usb_amd_resume_quirk ( udev ) )
udev - > quirks | = __usb_detect_quirks ( udev ,
usb_amd_resume_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 ;
}