storage: accept some UAS devices if streams are unavailable
authorOliver Neukum <oneukum@suse.de>
Tue, 11 Feb 2014 19:36:04 +0000 (20:36 +0100)
committerSarah Sharp <sarah.a.sharp@linux.intel.com>
Tue, 4 Mar 2014 23:41:09 +0000 (15:41 -0800)
On some older XHCIs streams are not supported and the UAS driver
will fail at probe time. For those devices storage should try
to bind to UAS devices.
This patch adds a flag for stream support to HCDs and evaluates
it.

[Note: Sarah fixed a bug where the USB 2.0 root hub, not USB 3.0 root
hub would get marked as being able to support streams.]

Signed-off-by: Oliver Neukum <oliver@neukum.org>
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Acked-by: Hans de Goede <hdegoede@redhat.com>
drivers/usb/host/xhci-pci.c
drivers/usb/host/xhci-plat.c
drivers/usb/storage/uas-detect.h
include/linux/usb/hcd.h

index 6c03584ac15f8671c4f09474da1461753326841e..cbf0c5cffc925cf3bae40e110616fe26b527b46d 100644 (file)
@@ -222,6 +222,9 @@ static int xhci_pci_probe(struct pci_dev *dev, const struct pci_device_id *id)
                goto put_usb3_hcd;
        /* Roothub already marked as USB 3.0 speed */
 
+       if (HCC_MAX_PSA(xhci->hcc_params) >= 4)
+               xhci->shared_hcd->can_do_streams = 1;
+
        return 0;
 
 put_usb3_hcd:
index 8affef910782ed6f406b7d362e38ac0c0c9341df..151901ce1ba933598b09cd8d01a077a7b66c1cc9 100644 (file)
@@ -158,6 +158,9 @@ static int xhci_plat_probe(struct platform_device *pdev)
         */
        *((struct xhci_hcd **) xhci->shared_hcd->hcd_priv) = xhci;
 
+       if (HCC_MAX_PSA(xhci->hcc_params) >= 4)
+               xhci->shared_hcd->can_do_streams = 1;
+
        ret = usb_add_hcd(xhci->shared_hcd, irq, IRQF_SHARED);
        if (ret)
                goto put_usb3_hcd;
index b8a02e12a8a2ae5af68a10443ba048486b66e679..bb05b984d5f67e2098ab035649bc18870bbf721f 100644 (file)
@@ -72,6 +72,7 @@ static int uas_use_uas_driver(struct usb_interface *intf,
 {
        struct usb_host_endpoint *eps[4] = { };
        struct usb_device *udev = interface_to_usbdev(intf);
+       struct usb_hcd *hcd = bus_to_hcd(udev->bus);
        unsigned long flags = id->driver_info;
        int r, alt;
 
@@ -80,6 +81,9 @@ static int uas_use_uas_driver(struct usb_interface *intf,
        if (flags & US_FL_IGNORE_UAS)
                return 0;
 
+       if (udev->speed >= USB_SPEED_SUPER && !hcd->can_do_streams)
+               return 0;
+
        alt = uas_find_uas_alt_setting(intf);
        if (alt < 0)
                return 0;
index efe8d8a7c7addebb55cc152afffa80c060d4725d..485cd5e2100c426ed29969c68bc50890c72e9aed 100644 (file)
@@ -143,6 +143,7 @@ struct usb_hcd {
        unsigned                authorized_default:1;
        unsigned                has_tt:1;       /* Integrated TT in root hub */
        unsigned                amd_resume_bug:1; /* AMD remote wakeup quirk */
+       unsigned                can_do_streams:1; /* HC supports streams */
 
        unsigned int            irq;            /* irq allocated */
        void __iomem            *regs;          /* device memory/io */