2017-11-03 11:28:30 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2009-04-27 19:52:28 -07:00
|
|
|
/*
|
|
|
|
* xHCI host controller driver PCI Bus Glue.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Intel Corp.
|
|
|
|
*
|
|
|
|
* Author: Sarah Sharp
|
|
|
|
* Some code borrowed from the Linux EHCI driver.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/pci.h>
|
2011-04-25 16:54:28 +01:00
|
|
|
#include <linux/slab.h>
|
2011-07-03 16:09:31 -04:00
|
|
|
#include <linux/module.h>
|
2015-07-21 17:20:25 +03:00
|
|
|
#include <linux/acpi.h>
|
2020-06-29 18:18:41 +02:00
|
|
|
#include <linux/reset.h>
|
2009-04-27 19:52:28 -07:00
|
|
|
|
|
|
|
#include "xhci.h"
|
2013-08-06 07:52:45 +03:00
|
|
|
#include "xhci-trace.h"
|
2020-05-14 17:50:38 +05:30
|
|
|
#include "xhci-pci.h"
|
2009-04-27 19:52:28 -07:00
|
|
|
|
2016-01-26 17:50:05 +02:00
|
|
|
#define SSIC_PORT_NUM 2
|
|
|
|
#define SSIC_PORT_CFG2 0x880c
|
|
|
|
#define SSIC_PORT_CFG2_OFFSET 0x30
|
2015-07-21 17:20:26 +03:00
|
|
|
#define PROG_DONE (1 << 30)
|
|
|
|
#define SSIC_PORT_UNUSED (1 << 31)
|
2020-10-28 22:31:23 +02:00
|
|
|
#define SPARSE_DISABLE_BIT 17
|
|
|
|
#define SPARSE_CNTL_ENABLE 0xC12C
|
2015-07-21 17:20:26 +03:00
|
|
|
|
2009-08-07 14:04:55 -07:00
|
|
|
/* Device for a quirk */
|
|
|
|
#define PCI_VENDOR_ID_FRESCO_LOGIC 0x1b73
|
|
|
|
#define PCI_DEVICE_ID_FRESCO_LOGIC_PDK 0x1000
|
2016-06-01 21:01:29 +02:00
|
|
|
#define PCI_DEVICE_ID_FRESCO_LOGIC_FL1009 0x1009
|
2021-10-08 12:25:47 +03:00
|
|
|
#define PCI_DEVICE_ID_FRESCO_LOGIC_FL1100 0x1100
|
xhci: Extend Fresco Logic MSI quirk.
Ali reports that plugging a device into the Fresco Logic xHCI host with
PCI device ID 1400 produces an IRQ error:
do_IRQ: 3.176 No irq handler for vector (irq -1)
Other early Fresco Logic host revisions don't support MSI, even though
their PCI config space claims they do. Extend the quirk to disabling
MSI to this chipset revision. Also enable the short transfer quirk,
since it's likely this revision also has that quirk, and it should be
harmless to enable.
04:00.0 0c03: 1b73:1400 (rev 01) (prog-if 30 [XHCI])
Subsystem: 1d5c:1000
Physical Slot: 3
Control: I/O+ Mem+ BusMaster+ SpecCycle- MemWINV- VGASnoop- ParErr- Stepping- SERR- FastB2B- DisINTx+
Status: Cap+ 66MHz- UDF- FastB2B- ParErr- DEVSEL=fast >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx-
Latency: 0, Cache Line Size: 64 bytes
Interrupt: pin A routed to IRQ 51
Region 0: Memory at d4600000 (32-bit, non-prefetchable) [size=64K]
Capabilities: [50] Power Management version 3
Flags: PMEClk- DSI- D1- D2- AuxCurrent=0mA PME(D0+,D1-,D2-,D3hot+,D3cold-)
Status: D0 NoSoftRst- PME-Enable- DSel=0 DScale=0 PME-
Capabilities: [68] MSI: Enable+ Count=1/1 Maskable- 64bit+
Address: 00000000feeff00c Data: 41b1
Capabilities: [80] Express (v1) Endpoint, MSI 00
DevCap: MaxPayload 128 bytes, PhantFunc 0, Latency L0s <2us, L1 <32us
ExtTag- AttnBtn- AttnInd- PwrInd- RBE+ FLReset-
DevCtl: Report errors: Correctable- Non-Fatal- Fatal- Unsupported-
RlxdOrd+ ExtTag- PhantFunc- AuxPwr- NoSnoop+
MaxPayload 128 bytes, MaxReadReq 512 bytes
DevSta: CorrErr- UncorrErr- FatalErr- UnsuppReq- AuxPwr- TransPend-
LnkCap: Port #0, Speed 2.5GT/s, Width x1, ASPM L0s L1, Latency L0 unlimited, L1 unlimited
ClockPM- Surprise- LLActRep- BwNot-
LnkCtl: ASPM Disabled; RCB 64 bytes Disabled- Retrain- CommClk+
ExtSynch- ClockPM- AutWidDis- BWInt- AutBWInt-
LnkSta: Speed 2.5GT/s, Width x1, TrErr- Train- SlotClk+ DLActive- BWMgmt- ABWMgmt-
Kernel driver in use: xhci_hcd
This patch should be backported to stable kernels as old as 2.6.36, that
contain the commit f5182b4155b9d686c5540a6822486400e34ddd98 "xhci:
Disable MSI for some Fresco Logic hosts."
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Reported-by: A Sh <smr.ash1991@gmail.com>
Tested-by: A Sh <smr.ash1991@gmail.com>
Cc: stable@vger.kernel.org
2012-10-17 13:44:06 -07:00
|
|
|
#define PCI_DEVICE_ID_FRESCO_LOGIC_FL1400 0x1400
|
2009-08-07 14:04:55 -07:00
|
|
|
|
2011-06-15 23:47:21 +02:00
|
|
|
#define PCI_VENDOR_ID_ETRON 0x1b6f
|
2014-07-25 22:01:19 +02:00
|
|
|
#define PCI_DEVICE_ID_EJ168 0x7023
|
2011-06-15 23:47:21 +02:00
|
|
|
|
2013-09-12 08:11:06 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_LYNXPOINT_XHCI 0x8c31
|
|
|
|
#define PCI_DEVICE_ID_INTEL_LYNXPOINT_LP_XHCI 0x9c31
|
2016-10-20 18:09:18 +03:00
|
|
|
#define PCI_DEVICE_ID_INTEL_WILDCATPOINT_LP_XHCI 0x9cb1
|
2015-03-06 17:23:19 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_CHERRYVIEW_XHCI 0x22b5
|
|
|
|
#define PCI_DEVICE_ID_INTEL_SUNRISEPOINT_H_XHCI 0xa12f
|
|
|
|
#define PCI_DEVICE_ID_INTEL_SUNRISEPOINT_LP_XHCI 0x9d2f
|
2016-01-26 17:50:08 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_BROXTON_M_XHCI 0x0aa8
|
2016-04-08 16:25:05 +03:00
|
|
|
#define PCI_DEVICE_ID_INTEL_BROXTON_B_XHCI 0x1aa8
|
2016-10-20 18:09:19 +03:00
|
|
|
#define PCI_DEVICE_ID_INTEL_APL_XHCI 0x5aa8
|
2017-05-17 18:32:00 +03:00
|
|
|
#define PCI_DEVICE_ID_INTEL_DNV_XHCI 0x19d0
|
2018-09-20 19:13:38 +03:00
|
|
|
#define PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_2C_XHCI 0x15b5
|
|
|
|
#define PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_4C_XHCI 0x15b6
|
2020-12-08 11:29:10 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_LP_XHCI 0x15c1
|
2018-09-20 19:13:38 +03:00
|
|
|
#define PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_C_2C_XHCI 0x15db
|
|
|
|
#define PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_C_4C_XHCI 0x15d4
|
|
|
|
#define PCI_DEVICE_ID_INTEL_TITAN_RIDGE_2C_XHCI 0x15e9
|
|
|
|
#define PCI_DEVICE_ID_INTEL_TITAN_RIDGE_4C_XHCI 0x15ec
|
|
|
|
#define PCI_DEVICE_ID_INTEL_TITAN_RIDGE_DD_XHCI 0x15f0
|
2019-11-15 18:50:03 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_ICE_LAKE_XHCI 0x8a13
|
2020-02-10 15:45:53 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_CML_XHCI 0xa3af
|
2020-03-12 16:45:17 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_TIGER_LAKE_XHCI 0x9a13
|
2020-12-08 11:29:11 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_MAPLE_RIDGE_XHCI 0x1138
|
2022-10-24 17:27:19 +03:00
|
|
|
#define PCI_DEVICE_ID_INTEL_ALDER_LAKE_PCH_XHCI 0x51ed
|
2022-11-30 11:19:40 +02:00
|
|
|
#define PCI_DEVICE_ID_INTEL_ALDER_LAKE_N_PCH_XHCI 0x54ed
|
2013-09-12 08:11:06 +02:00
|
|
|
|
2021-05-27 10:45:34 -05:00
|
|
|
#define PCI_DEVICE_ID_AMD_RENOIR_XHCI 0x1639
|
2018-02-12 14:24:46 +02:00
|
|
|
#define PCI_DEVICE_ID_AMD_PROMONTORYA_4 0x43b9
|
|
|
|
#define PCI_DEVICE_ID_AMD_PROMONTORYA_3 0x43ba
|
|
|
|
#define PCI_DEVICE_ID_AMD_PROMONTORYA_2 0x43bb
|
|
|
|
#define PCI_DEVICE_ID_AMD_PROMONTORYA_1 0x43bc
|
2021-10-14 15:12:00 +03:00
|
|
|
|
2020-07-27 23:24:07 -05:00
|
|
|
#define PCI_DEVICE_ID_ASMEDIA_1042_XHCI 0x1042
|
2017-07-20 14:48:27 +03:00
|
|
|
#define PCI_DEVICE_ID_ASMEDIA_1042A_XHCI 0x1142
|
2020-07-27 23:24:08 -05:00
|
|
|
#define PCI_DEVICE_ID_ASMEDIA_1142_XHCI 0x1242
|
2020-07-27 23:24:07 -05:00
|
|
|
#define PCI_DEVICE_ID_ASMEDIA_2142_XHCI 0x2142
|
2021-03-11 13:53:52 +02:00
|
|
|
#define PCI_DEVICE_ID_ASMEDIA_3242_XHCI 0x3242
|
2017-07-20 14:48:27 +03:00
|
|
|
|
2009-04-27 19:52:28 -07:00
|
|
|
static const char hcd_name[] = "xhci_hcd";
|
|
|
|
|
2014-10-03 11:35:26 +03:00
|
|
|
static struct hc_driver __read_mostly xhci_pci_hc_driver;
|
|
|
|
|
2015-05-29 17:01:46 +03:00
|
|
|
static int xhci_pci_setup(struct usb_hcd *hcd);
|
2023-03-17 17:47:10 +02:00
|
|
|
static int xhci_pci_run(struct usb_hcd *hcd);
|
2023-01-16 16:22:13 +02:00
|
|
|
static int xhci_pci_update_hub_device(struct usb_hcd *hcd, struct usb_device *hdev,
|
|
|
|
struct usb_tt *tt, gfp_t mem_flags);
|
2015-05-29 17:01:46 +03:00
|
|
|
|
|
|
|
static const struct xhci_driver_overrides xhci_pci_overrides __initconst = {
|
|
|
|
.reset = xhci_pci_setup,
|
2023-03-17 17:47:10 +02:00
|
|
|
.start = xhci_pci_run,
|
2023-01-16 16:22:13 +02:00
|
|
|
.update_hub_device = xhci_pci_update_hub_device,
|
2015-05-29 17:01:46 +03:00
|
|
|
};
|
|
|
|
|
2023-03-17 17:47:15 +02:00
|
|
|
static void xhci_msix_sync_irqs(struct xhci_hcd *xhci)
|
|
|
|
{
|
|
|
|
struct usb_hcd *hcd = xhci_to_hcd(xhci);
|
|
|
|
|
|
|
|
if (hcd->msix_enabled) {
|
|
|
|
struct pci_dev *pdev = to_pci_dev(hcd->self.controller);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < xhci->msix_count; i++)
|
|
|
|
synchronize_irq(pci_irq_vector(pdev, i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 17:47:13 +02:00
|
|
|
/* Free any IRQs and disable MSI-X */
|
|
|
|
static void xhci_cleanup_msix(struct xhci_hcd *xhci)
|
|
|
|
{
|
|
|
|
struct usb_hcd *hcd = xhci_to_hcd(xhci);
|
|
|
|
struct pci_dev *pdev = to_pci_dev(hcd->self.controller);
|
|
|
|
|
|
|
|
if (xhci->quirks & XHCI_PLAT)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* return if using legacy interrupt */
|
|
|
|
if (hcd->irq > 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (hcd->msix_enabled) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < xhci->msix_count; i++)
|
|
|
|
free_irq(pci_irq_vector(pdev, i), xhci_to_hcd(xhci));
|
|
|
|
} else {
|
|
|
|
free_irq(pci_irq_vector(pdev, 0), xhci_to_hcd(xhci));
|
|
|
|
}
|
|
|
|
|
|
|
|
pci_free_irq_vectors(pdev);
|
|
|
|
hcd->msix_enabled = 0;
|
|
|
|
}
|
|
|
|
|
2023-03-17 17:47:11 +02:00
|
|
|
/*
|
|
|
|
* Set up MSI
|
|
|
|
*/
|
|
|
|
static int xhci_setup_msi(struct xhci_hcd *xhci)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
/*
|
|
|
|
* TODO:Check with MSI Soc for sysdev
|
|
|
|
*/
|
|
|
|
struct pci_dev *pdev = to_pci_dev(xhci_to_hcd(xhci)->self.controller);
|
|
|
|
|
|
|
|
ret = pci_alloc_irq_vectors(pdev, 1, 1, PCI_IRQ_MSI);
|
|
|
|
if (ret < 0) {
|
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_init,
|
|
|
|
"failed to allocate MSI entry");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = request_irq(pdev->irq, xhci_msi_irq,
|
|
|
|
0, "xhci_hcd", xhci_to_hcd(xhci));
|
|
|
|
if (ret) {
|
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_init,
|
|
|
|
"disable MSI interrupt");
|
|
|
|
pci_free_irq_vectors(pdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up MSI-X
|
|
|
|
*/
|
|
|
|
static int xhci_setup_msix(struct xhci_hcd *xhci)
|
|
|
|
{
|
|
|
|
int i, ret;
|
|
|
|
struct usb_hcd *hcd = xhci_to_hcd(xhci);
|
|
|
|
struct pci_dev *pdev = to_pci_dev(hcd->self.controller);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* calculate number of msi-x vectors supported.
|
|
|
|
* - HCS_MAX_INTRS: the max number of interrupts the host can handle,
|
|
|
|
* with max number of interrupters based on the xhci HCSPARAMS1.
|
|
|
|
* - num_online_cpus: maximum msi-x vectors per CPUs core.
|
|
|
|
* Add additional 1 vector to ensure always available interrupt.
|
|
|
|
*/
|
|
|
|
xhci->msix_count = min(num_online_cpus() + 1,
|
|
|
|
HCS_MAX_INTRS(xhci->hcs_params1));
|
|
|
|
|
|
|
|
ret = pci_alloc_irq_vectors(pdev, xhci->msix_count, xhci->msix_count,
|
|
|
|
PCI_IRQ_MSIX);
|
|
|
|
if (ret < 0) {
|
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_init,
|
|
|
|
"Failed to enable MSI-X");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < xhci->msix_count; i++) {
|
|
|
|
ret = request_irq(pci_irq_vector(pdev, i), xhci_msi_irq, 0,
|
|
|
|
"xhci_hcd", xhci_to_hcd(xhci));
|
|
|
|
if (ret)
|
|
|
|
goto disable_msix;
|
|
|
|
}
|
|
|
|
|
|
|
|
hcd->msix_enabled = 1;
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
disable_msix:
|
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_init, "disable MSI-X interrupt");
|
|
|
|
while (--i >= 0)
|
|
|
|
free_irq(pci_irq_vector(pdev, i), xhci_to_hcd(xhci));
|
|
|
|
pci_free_irq_vectors(pdev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int xhci_try_enable_msi(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
|
|
|
struct pci_dev *pdev;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* The xhci platform device has set up IRQs through usb_add_hcd. */
|
|
|
|
if (xhci->quirks & XHCI_PLAT)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
pdev = to_pci_dev(xhci_to_hcd(xhci)->self.controller);
|
|
|
|
/*
|
|
|
|
* Some Fresco Logic host controllers advertise MSI, but fail to
|
|
|
|
* generate interrupts. Don't even try to enable MSI.
|
|
|
|
*/
|
|
|
|
if (xhci->quirks & XHCI_BROKEN_MSI)
|
|
|
|
goto legacy_irq;
|
|
|
|
|
|
|
|
/* unregister the legacy interrupt */
|
|
|
|
if (hcd->irq)
|
|
|
|
free_irq(hcd->irq, hcd);
|
|
|
|
hcd->irq = 0;
|
|
|
|
|
|
|
|
ret = xhci_setup_msix(xhci);
|
|
|
|
if (ret)
|
|
|
|
/* fall back to msi*/
|
|
|
|
ret = xhci_setup_msi(xhci);
|
|
|
|
|
|
|
|
if (!ret) {
|
|
|
|
hcd->msi_enabled = 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pdev->irq) {
|
|
|
|
xhci_err(xhci, "No msi-x/msi found and no IRQ in BIOS\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
legacy_irq:
|
|
|
|
if (!strlen(hcd->irq_descr))
|
|
|
|
snprintf(hcd->irq_descr, sizeof(hcd->irq_descr), "%s:usb%d",
|
|
|
|
hcd->driver->description, hcd->self.busnum);
|
|
|
|
|
|
|
|
/* fall back to legacy interrupt*/
|
|
|
|
ret = request_irq(pdev->irq, &usb_hcd_irq, IRQF_SHARED,
|
|
|
|
hcd->irq_descr, hcd);
|
|
|
|
if (ret) {
|
|
|
|
xhci_err(xhci, "request interrupt %d failed\n",
|
|
|
|
pdev->irq);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
hcd->irq = pdev->irq;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-03-17 17:47:10 +02:00
|
|
|
static int xhci_pci_run(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (usb_hcd_is_primary_hcd(hcd)) {
|
|
|
|
ret = xhci_try_enable_msi(hcd);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return xhci_run(hcd);
|
|
|
|
}
|
|
|
|
|
2023-03-17 17:47:12 +02:00
|
|
|
static void xhci_pci_stop(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
|
|
|
|
|
|
|
xhci_stop(hcd);
|
|
|
|
|
|
|
|
if (usb_hcd_is_primary_hcd(hcd))
|
|
|
|
xhci_cleanup_msix(xhci);
|
|
|
|
}
|
|
|
|
|
2009-04-27 19:52:28 -07:00
|
|
|
/* called after powerup, by probe or system-pm "wakeup" */
|
|
|
|
static int xhci_pci_reinit(struct xhci_hcd *xhci, struct pci_dev *pdev)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* TODO: Implement finding debug ports later.
|
|
|
|
* TODO: see if there are any quirks that need to be added to handle
|
|
|
|
* new extended capabilities.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* PCI Memory-Write-Invalidate cycle support is optional (uncommon) */
|
|
|
|
if (!pci_set_mwi(pdev))
|
|
|
|
xhci_dbg(xhci, "MWI active\n");
|
|
|
|
|
|
|
|
xhci_dbg(xhci, "Finished xhci_pci_reinit\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-09-23 14:20:00 -07:00
|
|
|
static void xhci_pci_quirks(struct device *dev, struct xhci_hcd *xhci)
|
|
|
|
{
|
2020-05-14 17:50:38 +05:30
|
|
|
struct pci_dev *pdev = to_pci_dev(dev);
|
|
|
|
struct xhci_driver_data *driver_data;
|
|
|
|
const struct pci_device_id *id;
|
|
|
|
|
2021-10-12 16:36:05 -05:00
|
|
|
id = pci_match_id(to_pci_driver(pdev->dev.driver)->id_table, pdev);
|
2020-05-14 17:50:38 +05:30
|
|
|
|
|
|
|
if (id && id->driver_data) {
|
|
|
|
driver_data = (struct xhci_driver_data *)id->driver_data;
|
|
|
|
xhci->quirks |= driver_data->quirks;
|
|
|
|
}
|
2011-09-23 14:20:00 -07:00
|
|
|
|
2009-08-07 14:04:55 -07:00
|
|
|
/* Look for vendor-specific quirks */
|
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_FRESCO_LOGIC &&
|
xhci: Extend Fresco Logic MSI quirk.
Ali reports that plugging a device into the Fresco Logic xHCI host with
PCI device ID 1400 produces an IRQ error:
do_IRQ: 3.176 No irq handler for vector (irq -1)
Other early Fresco Logic host revisions don't support MSI, even though
their PCI config space claims they do. Extend the quirk to disabling
MSI to this chipset revision. Also enable the short transfer quirk,
since it's likely this revision also has that quirk, and it should be
harmless to enable.
04:00.0 0c03: 1b73:1400 (rev 01) (prog-if 30 [XHCI])
Subsystem: 1d5c:1000
Physical Slot: 3
Control: I/O+ Mem+ BusMaster+ SpecCycle- MemWINV- VGASnoop- ParErr- Stepping- SERR- FastB2B- DisINTx+
Status: Cap+ 66MHz- UDF- FastB2B- ParErr- DEVSEL=fast >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx-
Latency: 0, Cache Line Size: 64 bytes
Interrupt: pin A routed to IRQ 51
Region 0: Memory at d4600000 (32-bit, non-prefetchable) [size=64K]
Capabilities: [50] Power Management version 3
Flags: PMEClk- DSI- D1- D2- AuxCurrent=0mA PME(D0+,D1-,D2-,D3hot+,D3cold-)
Status: D0 NoSoftRst- PME-Enable- DSel=0 DScale=0 PME-
Capabilities: [68] MSI: Enable+ Count=1/1 Maskable- 64bit+
Address: 00000000feeff00c Data: 41b1
Capabilities: [80] Express (v1) Endpoint, MSI 00
DevCap: MaxPayload 128 bytes, PhantFunc 0, Latency L0s <2us, L1 <32us
ExtTag- AttnBtn- AttnInd- PwrInd- RBE+ FLReset-
DevCtl: Report errors: Correctable- Non-Fatal- Fatal- Unsupported-
RlxdOrd+ ExtTag- PhantFunc- AuxPwr- NoSnoop+
MaxPayload 128 bytes, MaxReadReq 512 bytes
DevSta: CorrErr- UncorrErr- FatalErr- UnsuppReq- AuxPwr- TransPend-
LnkCap: Port #0, Speed 2.5GT/s, Width x1, ASPM L0s L1, Latency L0 unlimited, L1 unlimited
ClockPM- Surprise- LLActRep- BwNot-
LnkCtl: ASPM Disabled; RCB 64 bytes Disabled- Retrain- CommClk+
ExtSynch- ClockPM- AutWidDis- BWInt- AutBWInt-
LnkSta: Speed 2.5GT/s, Width x1, TrErr- Train- SlotClk+ DLActive- BWMgmt- ABWMgmt-
Kernel driver in use: xhci_hcd
This patch should be backported to stable kernels as old as 2.6.36, that
contain the commit f5182b4155b9d686c5540a6822486400e34ddd98 "xhci:
Disable MSI for some Fresco Logic hosts."
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Reported-by: A Sh <smr.ash1991@gmail.com>
Tested-by: A Sh <smr.ash1991@gmail.com>
Cc: stable@vger.kernel.org
2012-10-17 13:44:06 -07:00
|
|
|
(pdev->device == PCI_DEVICE_ID_FRESCO_LOGIC_PDK ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_FRESCO_LOGIC_FL1400)) {
|
|
|
|
if (pdev->device == PCI_DEVICE_ID_FRESCO_LOGIC_PDK &&
|
|
|
|
pdev->revision == 0x0) {
|
2009-08-07 14:04:55 -07:00
|
|
|
xhci->quirks |= XHCI_RESET_EP_QUIRK;
|
2013-08-06 07:52:45 +03:00
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
2022-05-12 01:04:49 +03:00
|
|
|
"XHCI_RESET_EP_QUIRK for this evaluation HW is deprecated");
|
2011-06-02 11:33:02 -07:00
|
|
|
}
|
2013-09-30 15:50:54 +02:00
|
|
|
if (pdev->device == PCI_DEVICE_ID_FRESCO_LOGIC_PDK &&
|
|
|
|
pdev->revision == 0x4) {
|
|
|
|
xhci->quirks |= XHCI_SLOW_SUSPEND;
|
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
|
|
|
"QUIRK: Fresco Logic xHC revision %u"
|
|
|
|
"must be suspended extra slowly",
|
|
|
|
pdev->revision);
|
|
|
|
}
|
2014-12-05 11:11:28 +01:00
|
|
|
if (pdev->device == PCI_DEVICE_ID_FRESCO_LOGIC_PDK)
|
|
|
|
xhci->quirks |= XHCI_BROKEN_STREAMS;
|
2011-06-02 11:33:02 -07:00
|
|
|
/* Fresco Logic confirms: all revisions of this chip do not
|
|
|
|
* support MSI, even though some of them claim to in their PCI
|
|
|
|
* capabilities.
|
|
|
|
*/
|
|
|
|
xhci->quirks |= XHCI_BROKEN_MSI;
|
2013-08-06 07:52:45 +03:00
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
|
|
|
"QUIRK: Fresco Logic revision %u "
|
|
|
|
"has broken MSI implementation",
|
2011-06-02 11:33:02 -07:00
|
|
|
pdev->revision);
|
xhci: Add new short TX quirk for Fresco Logic host.
Sergio reported that when he recorded audio from a USB headset mic
plugged into the USB 3.0 port on his ASUS N53SV-DH72, the audio sounded
"robotic". When plugged into the USB 2.0 port under EHCI on the same
laptop, the audio sounded fine. The device is:
Bus 002 Device 004: ID 046d:0a0c Logitech, Inc. Clear Chat Comfort USB Headset
The problem was tracked down to the Fresco Logic xHCI host controller
not correctly reporting short transfers on isochronous IN endpoints.
The driver would submit a 96 byte transfer, the device would only send
88 or 90 bytes, and the xHCI host would report the transfer had a
"successful" completion code, with an untransferred buffer length of 8
or 6 bytes.
The successful completion code and non-zero untransferred length is a
contradiction. The xHCI host is supposed to only mark a transfer as
successful if all the bytes are transferred. Otherwise, the transfer
should be marked with a short packet completion code. Without the EHCI
bus trace, we wouldn't know whether the xHCI driver should trust the
completion code or the untransferred length. With it, we know to trust
the untransferred length.
Add a new xHCI quirk for the Fresco Logic host controller. If a
transfer is reported as successful, but the untransferred length is
non-zero, print a warning. For the Fresco Logic host, change the
completion code to COMP_SHORT_TX and process the transfer like a short
transfer.
This should be backported to stable kernels that contain the commit
f5182b4155b9d686c5540a6822486400e34ddd98 "xhci: Disable MSI for some
Fresco Logic hosts." That commit was marked for stable kernels as old
as 2.6.36.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Reported-by: Sergio Correia <lists@uece.net>
Tested-by: Sergio Correia <lists@uece.net>
Cc: stable@vger.kernel.org
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2012-05-08 09:22:49 -07:00
|
|
|
xhci->quirks |= XHCI_TRUST_TX_LENGTH;
|
2009-08-07 14:04:55 -07:00
|
|
|
}
|
2011-06-02 11:33:02 -07:00
|
|
|
|
2016-06-01 21:01:29 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_FRESCO_LOGIC &&
|
|
|
|
pdev->device == PCI_DEVICE_ID_FRESCO_LOGIC_FL1009)
|
|
|
|
xhci->quirks |= XHCI_BROKEN_STREAMS;
|
|
|
|
|
2021-12-21 13:28:25 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_FRESCO_LOGIC &&
|
|
|
|
pdev->device == PCI_DEVICE_ID_FRESCO_LOGIC_FL1100)
|
|
|
|
xhci->quirks |= XHCI_TRUST_TX_LENGTH;
|
|
|
|
|
2010-05-24 13:25:28 -07:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_NEC)
|
|
|
|
xhci->quirks |= XHCI_NEC_HOST;
|
2009-08-07 14:04:55 -07:00
|
|
|
|
xHCI: AMD isoc link TRB chain bit quirk
Setting the chain (CH) bit in the link TRB of isochronous transfer rings
is required by AMD 0.96 xHCI host controller to successfully transverse
multi-TRB TD that span through different memory segments.
When a Missed Service Error event occurs, if the chain bit is not set in
the link TRB and the host skips TDs which just across a link TRB, the
host may falsely recognize the link TRB as a normal TRB. You can see
this may cause big trouble - the host does not jump to the right address
which is pointed by the link TRB, but continue fetching the memory which
is after the link TRB address, which may not even belong to the host,
and the result cannot be predicted.
This causes some big problems. Without the former patch I sent: "xHCI:
prevent infinite loop when processing MSE event", the system may hang.
With that patch applied, system does not hang, but the host still access
wrong memory address and isoc transfer will fail. With this patch,
isochronous transfer works as expected.
This patch should be applied to kernels as old as 2.6.36, which was when
the first isochronous support was added for the xHCI host controller.
Signed-off-by: Andiry Xu <andiry.xu@amd.com>
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Cc: stable@kernel.org
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-09-23 14:19:54 -07:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD && xhci->hci_version == 0x96)
|
|
|
|
xhci->quirks |= XHCI_AMD_0x96_HOST;
|
|
|
|
|
2011-03-22 17:08:14 +08:00
|
|
|
/* AMD PLL quirk */
|
2019-07-04 11:35:29 -04:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD && usb_amd_quirk_pll_check())
|
2011-03-22 17:08:14 +08:00
|
|
|
xhci->quirks |= XHCI_AMD_PLL_FIX;
|
2014-08-19 15:17:57 +03:00
|
|
|
|
2018-04-20 16:52:50 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD &&
|
2020-03-06 17:08:58 +02:00
|
|
|
(pdev->device == 0x145c ||
|
|
|
|
pdev->device == 0x15e0 ||
|
2018-04-20 16:52:50 +03:00
|
|
|
pdev->device == 0x15e1 ||
|
|
|
|
pdev->device == 0x43bb))
|
2018-03-08 17:17:17 +02:00
|
|
|
xhci->quirks |= XHCI_SUSPEND_DELAY;
|
|
|
|
|
2018-12-05 14:22:38 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD &&
|
|
|
|
(pdev->device == 0x15e0 || pdev->device == 0x15e1))
|
|
|
|
xhci->quirks |= XHCI_SNPS_BROKEN_SUSPEND;
|
|
|
|
|
2021-05-12 11:08:16 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD && pdev->device == 0x15e5) {
|
2020-10-28 22:31:23 +02:00
|
|
|
xhci->quirks |= XHCI_DISABLE_SPARSE;
|
2021-05-12 11:08:16 +03:00
|
|
|
xhci->quirks |= XHCI_RESET_ON_RESUME;
|
|
|
|
}
|
2020-10-28 22:31:23 +02:00
|
|
|
|
2014-08-19 15:17:57 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD)
|
|
|
|
xhci->quirks |= XHCI_TRUST_TX_LENGTH;
|
|
|
|
|
2018-02-12 14:24:46 +02:00
|
|
|
if ((pdev->vendor == PCI_VENDOR_ID_AMD) &&
|
|
|
|
((pdev->device == PCI_DEVICE_ID_AMD_PROMONTORYA_4) ||
|
|
|
|
(pdev->device == PCI_DEVICE_ID_AMD_PROMONTORYA_3) ||
|
|
|
|
(pdev->device == PCI_DEVICE_ID_AMD_PROMONTORYA_2) ||
|
|
|
|
(pdev->device == PCI_DEVICE_ID_AMD_PROMONTORYA_1)))
|
|
|
|
xhci->quirks |= XHCI_U2_DISABLE_WAKE;
|
|
|
|
|
2021-05-27 10:45:34 -05:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD &&
|
|
|
|
pdev->device == PCI_DEVICE_ID_AMD_RENOIR_XHCI)
|
|
|
|
xhci->quirks |= XHCI_BROKEN_D3COLD;
|
|
|
|
|
2012-05-16 13:36:24 -07:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL) {
|
|
|
|
xhci->quirks |= XHCI_LPM_SUPPORT;
|
|
|
|
xhci->quirks |= XHCI_INTEL_HOST;
|
usb: xhci: apply XHCI_AVOID_BEI quirk to all Intel xHCI controllers
When a device with an isochronous endpoint is plugged into the Intel
xHCI host controller, and the driver submits multiple frames per URB,
the xHCI driver will set the Block Event Interrupt (BEI) flag on all
but the last TD for the URB. This causes the host controller to place
an event on the event ring, but not send an interrupt. When the last
TD for the URB completes, BEI is cleared, and we get an interrupt for
the whole URB.
However, under Intel xHCI host controllers, if the event ring is full
of events from transfers with BEI set, an "Event Ring is Full" event
will be posted to the last entry of the event ring, but no interrupt
is generated. Host will cease all transfer and command executions and
wait until software completes handling the pending events in the event
ring. That means xHC stops, but event of "event ring is full" is not
notified. As the result, the xHC looks like dead to user.
This patch is to apply XHCI_AVOID_BEI quirk to Intel xHC devices. And
it should be backported to kernels as old as 3.0, that contains the
commit 69e848c2090a ("Intel xhci: Support EHCI/xHCI port switching.").
Signed-off-by: Lu Baolu <baolu.lu@linux.intel.com>
Tested-by: Alistair Grant <akgrant0710@gmail.com>
Cc: stable@vger.kernel.org
Signed-off-by: Mathias Nyman <mathias.nyman@linux.intel.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2015-03-23 18:27:42 +02:00
|
|
|
xhci->quirks |= XHCI_AVOID_BEI;
|
2012-05-16 13:36:24 -07:00
|
|
|
}
|
2011-05-25 10:43:56 -07:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_PANTHERPOINT_XHCI) {
|
Intel xhci: Limit number of active endpoints to 64.
The Panther Point chipset has an xHCI host controller that has a limit to
the number of active endpoints it can handle. Ideally, it would signal
that it can't handle anymore endpoints by returning a Resource Error for
the Configure Endpoint command, but they don't. Instead it needs software
to keep track of the number of active endpoints, across configure endpoint
commands, reset device commands, disable slot commands, and address device
commands.
Add a new endpoint context counter, xhci_hcd->num_active_eps, and use it
to track the number of endpoints the xHC has active. This gets a little
tricky, because commands to change the number of active endpoints can
fail. This patch adds a new xHCI quirk for these Intel hosts, and the new
code should not have any effect on other xHCI host controllers.
Fail a new device allocation if we don't have room for the new default
control endpoint. Use the endpoint ring pointers to determine what
endpoints were active before a Reset Device command or a Disable Slot
command, and drop those once the command completes.
Fail a configure endpoint command if it would add too many new endpoints.
We have to be a bit over zealous here, and only count the number of new
endpoints to be added, without subtracting the number of dropped
endpoints. That's because a second configure endpoint command for a
different device could sneak in before we know if the first command is
completed. If the first command dropped resources, the host controller
fails the command for some reason, and we're nearing the limit of
endpoints, we could end up oversubscribing the host.
To fix this race condition, when evaluating whether a configure endpoint
command will fix in our bandwidth budget, only add the new endpoints to
xhci->num_active_eps, and don't subtract the dropped endpoints. Ignore
changed endpoints (ones that are dropped and then re-added), as that
shouldn't effect the host's endpoint resources. When the configure
endpoint command completes, subtract off the dropped endpoints.
This may mean some configuration changes may temporarily fail, but it's
always better to under-subscribe than over-subscribe resources.
(Originally my plan had been to push the resource allocation down into the
ring allocation functions. However, that would cause us to allocate
unnecessary resources when endpoints were changed, because the xHCI driver
allocates a new ring for the changed endpoint, and only deletes the old
ring once the Configure Endpoint command succeeds. A further complication
would have been dealing with the per-device endpoint ring cache.)
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2011-05-11 16:14:58 -07:00
|
|
|
xhci->quirks |= XHCI_EP_LIMIT_QUIRK;
|
|
|
|
xhci->limit_active_eps = 64;
|
2011-09-02 11:05:54 -07:00
|
|
|
xhci->quirks |= XHCI_SW_BW_CHECKING;
|
xhci: Switch PPT ports to EHCI on shutdown.
The Intel desktop boards DH77EB and DH77DF have a hardware issue that
can be worked around by BIOS. If the USB ports are switched to xHCI on
shutdown, the xHCI host will send a spurious interrupt, which will wake
the system. Some BIOS will work around this, but not all.
The bug can be avoided if the USB ports are switched back to EHCI on
shutdown. The Intel Windows driver switches the ports back to EHCI, so
change the Linux xHCI driver to do the same.
Unfortunately, we can't tell the two effected boards apart from other
working motherboards, because the vendors will change the DMI strings
for the DH77EB and DH77DF boards to their own custom names. One example
is Compulab's mini-desktop, the Intense-PC. Instead, key off the
Panther Point xHCI host PCI vendor and device ID, and switch the ports
over for all PPT xHCI hosts.
The only impact this will have on non-effected boards is to add a couple
hundred milliseconds delay on boot when the BIOS has to switch the ports
over from EHCI to xHCI.
This patch should be backported to kernels as old as 3.0, that contain
the commit 69e848c2090aebba5698a1620604c7dccb448684 "Intel xhci: Support
EHCI/xHCI port switching."
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Reported-by: Denis Turischev <denis@compulab.co.il>
Tested-by: Denis Turischev <denis@compulab.co.il>
Cc: stable@vger.kernel.org
2012-07-23 18:59:30 +03:00
|
|
|
/*
|
|
|
|
* PPT desktop boards DH77EB and DH77DF will power back on after
|
|
|
|
* a few seconds of being shutdown. The fix for this is to
|
|
|
|
* switch the ports from xHCI to EHCI on shutdown. We can't use
|
|
|
|
* DMI information to find those particular boards (since each
|
|
|
|
* vendor will change the board name), so we have to key off all
|
|
|
|
* PPT chipsets.
|
|
|
|
*/
|
|
|
|
xhci->quirks |= XHCI_SPURIOUS_REBOOT;
|
2011-05-25 10:43:56 -07:00
|
|
|
}
|
2014-05-20 14:00:42 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
2016-10-20 18:09:18 +03:00
|
|
|
(pdev->device == PCI_DEVICE_ID_INTEL_LYNXPOINT_LP_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_WILDCATPOINT_LP_XHCI)) {
|
2014-04-25 19:20:14 +03:00
|
|
|
xhci->quirks |= XHCI_SPURIOUS_REBOOT;
|
2015-10-12 11:30:13 +03:00
|
|
|
xhci->quirks |= XHCI_SPURIOUS_WAKEUP;
|
2013-09-12 08:11:06 +02:00
|
|
|
}
|
2015-03-06 17:23:19 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
|
|
|
(pdev->device == PCI_DEVICE_ID_INTEL_SUNRISEPOINT_LP_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_SUNRISEPOINT_H_XHCI ||
|
2016-01-26 17:50:08 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_CHERRYVIEW_XHCI ||
|
2016-04-08 16:25:05 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_BROXTON_M_XHCI ||
|
2017-01-03 18:28:52 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_BROXTON_B_XHCI ||
|
2017-05-17 18:32:00 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_APL_XHCI ||
|
2020-02-10 15:45:53 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_DNV_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_CML_XHCI)) {
|
2015-03-06 17:23:19 +02:00
|
|
|
xhci->quirks |= XHCI_PME_STUCK_QUIRK;
|
|
|
|
}
|
2016-01-26 17:50:06 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
2018-10-01 18:53:05 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_CHERRYVIEW_XHCI)
|
2016-01-26 17:50:06 +02:00
|
|
|
xhci->quirks |= XHCI_SSIC_PORT_UNUSED;
|
2018-10-01 18:53:05 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
|
|
|
(pdev->device == PCI_DEVICE_ID_INTEL_CHERRYVIEW_XHCI ||
|
2019-02-20 19:50:53 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_SUNRISEPOINT_LP_XHCI ||
|
2018-10-01 18:53:05 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_APL_XHCI))
|
2018-03-20 15:57:09 +03:00
|
|
|
xhci->quirks |= XHCI_INTEL_USB_ROLE_SW;
|
2016-10-20 18:09:19 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
|
|
|
(pdev->device == PCI_DEVICE_ID_INTEL_CHERRYVIEW_XHCI ||
|
2018-10-01 18:36:07 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_SUNRISEPOINT_LP_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_SUNRISEPOINT_H_XHCI ||
|
2017-05-17 18:32:00 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_APL_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_DNV_XHCI))
|
2016-10-20 18:09:19 +03:00
|
|
|
xhci->quirks |= XHCI_MISSING_CAS;
|
|
|
|
|
2022-10-24 17:27:18 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
2022-11-30 11:19:40 +02:00
|
|
|
(pdev->device == PCI_DEVICE_ID_INTEL_ALDER_LAKE_PCH_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_ALDER_LAKE_N_PCH_XHCI))
|
2022-10-24 17:27:18 +03:00
|
|
|
xhci->quirks |= XHCI_RESET_TO_DEFAULT;
|
|
|
|
|
2018-09-20 19:13:38 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL &&
|
|
|
|
(pdev->device == PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_2C_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_4C_XHCI ||
|
2020-12-08 11:29:10 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_LP_XHCI ||
|
2018-09-20 19:13:38 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_C_2C_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_ALPINE_RIDGE_C_4C_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_TITAN_RIDGE_2C_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_TITAN_RIDGE_4C_XHCI ||
|
2019-11-15 18:50:03 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_TITAN_RIDGE_DD_XHCI ||
|
2020-03-12 16:45:17 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_ICE_LAKE_XHCI ||
|
2020-12-08 11:29:11 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_TIGER_LAKE_XHCI ||
|
2022-10-24 17:27:19 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_INTEL_MAPLE_RIDGE_XHCI))
|
2018-09-20 19:13:38 +03:00
|
|
|
xhci->quirks |= XHCI_DEFAULT_PM_RUNTIME_ALLOW;
|
|
|
|
|
2011-06-15 23:47:21 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_ETRON &&
|
2014-07-25 22:01:19 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_EJ168) {
|
2011-06-15 23:47:21 +02:00
|
|
|
xhci->quirks |= XHCI_RESET_ON_RESUME;
|
2012-07-02 13:36:23 -07:00
|
|
|
xhci->quirks |= XHCI_TRUST_TX_LENGTH;
|
2014-07-25 22:01:18 +02:00
|
|
|
xhci->quirks |= XHCI_BROKEN_STREAMS;
|
2011-06-15 23:47:21 +02:00
|
|
|
}
|
2017-12-21 15:06:15 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_RENESAS &&
|
2018-05-23 18:41:37 +01:00
|
|
|
pdev->device == 0x0014) {
|
2017-12-21 15:06:15 +02:00
|
|
|
xhci->quirks |= XHCI_TRUST_TX_LENGTH;
|
2018-05-23 18:41:37 +01:00
|
|
|
xhci->quirks |= XHCI_ZERO_64B_REGS;
|
|
|
|
}
|
2014-01-17 15:38:12 -08:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_RENESAS &&
|
2018-05-23 18:41:37 +01:00
|
|
|
pdev->device == 0x0015) {
|
2014-01-17 15:38:12 -08:00
|
|
|
xhci->quirks |= XHCI_RESET_ON_RESUME;
|
2018-05-23 18:41:37 +01:00
|
|
|
xhci->quirks |= XHCI_ZERO_64B_REGS;
|
|
|
|
}
|
2012-03-29 15:47:50 +08:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_VIA)
|
|
|
|
xhci->quirks |= XHCI_RESET_ON_RESUME;
|
2014-05-14 14:00:23 +02:00
|
|
|
|
2014-08-25 12:21:56 +02:00
|
|
|
/* See https://bugzilla.kernel.org/show_bug.cgi?id=79511 */
|
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_VIA &&
|
|
|
|
pdev->device == 0x3432)
|
|
|
|
xhci->quirks |= XHCI_BROKEN_STREAMS;
|
|
|
|
|
2021-10-08 12:25:44 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_VIA && pdev->device == 0x3483) {
|
2020-03-12 16:45:12 +02:00
|
|
|
xhci->quirks |= XHCI_LPM_SUPPORT;
|
2021-10-08 12:25:44 +03:00
|
|
|
xhci->quirks |= XHCI_EP_CTX_BROKEN_DCS;
|
|
|
|
}
|
2020-03-12 16:45:12 +02:00
|
|
|
|
2014-10-28 11:05:29 +01:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_ASMEDIA &&
|
2022-10-24 17:27:17 +03:00
|
|
|
pdev->device == PCI_DEVICE_ID_ASMEDIA_1042_XHCI) {
|
|
|
|
/*
|
|
|
|
* try to tame the ASMedia 1042 controller which reports 0.96
|
|
|
|
* but appears to behave more like 1.0
|
|
|
|
*/
|
|
|
|
xhci->quirks |= XHCI_SPURIOUS_SUCCESS;
|
2014-10-28 11:05:29 +01:00
|
|
|
xhci->quirks |= XHCI_BROKEN_STREAMS;
|
2022-10-24 17:27:17 +03:00
|
|
|
}
|
2017-06-09 14:48:41 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_ASMEDIA &&
|
2021-03-11 13:53:52 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_ASMEDIA_1042A_XHCI) {
|
2017-06-09 14:48:41 +03:00
|
|
|
xhci->quirks |= XHCI_TRUST_TX_LENGTH;
|
2021-03-11 13:53:52 +02:00
|
|
|
xhci->quirks |= XHCI_NO_64BIT_SUPPORT;
|
|
|
|
}
|
2020-07-17 06:27:34 -05:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_ASMEDIA &&
|
2020-07-27 23:24:08 -05:00
|
|
|
(pdev->device == PCI_DEVICE_ID_ASMEDIA_1142_XHCI ||
|
2021-03-11 13:53:52 +02:00
|
|
|
pdev->device == PCI_DEVICE_ID_ASMEDIA_2142_XHCI ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_ASMEDIA_3242_XHCI))
|
2020-07-17 06:27:34 -05:00
|
|
|
xhci->quirks |= XHCI_NO_64BIT_SUPPORT;
|
2014-10-28 11:05:29 +01:00
|
|
|
|
2017-07-20 14:48:27 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_ASMEDIA &&
|
|
|
|
pdev->device == PCI_DEVICE_ID_ASMEDIA_1042A_XHCI)
|
|
|
|
xhci->quirks |= XHCI_ASMEDIA_MODIFY_FLOWCONTROL;
|
|
|
|
|
2017-04-07 17:57:12 +03:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_TI && pdev->device == 0x8241)
|
|
|
|
xhci->quirks |= XHCI_LIMIT_ENDPOINT_INTERVAL_7;
|
|
|
|
|
2018-11-09 17:21:22 +02:00
|
|
|
if ((pdev->vendor == PCI_VENDOR_ID_BROADCOM ||
|
|
|
|
pdev->vendor == PCI_VENDOR_ID_CAVIUM) &&
|
|
|
|
pdev->device == 0x9026)
|
|
|
|
xhci->quirks |= XHCI_RESET_PLL_ON_DISCONNECT;
|
|
|
|
|
2021-03-11 13:53:50 +02:00
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_AMD &&
|
|
|
|
(pdev->device == PCI_DEVICE_ID_AMD_PROMONTORYA_2 ||
|
|
|
|
pdev->device == PCI_DEVICE_ID_AMD_PROMONTORYA_4))
|
|
|
|
xhci->quirks |= XHCI_NO_SOFT_RETRY;
|
|
|
|
|
2022-10-24 17:27:19 +03:00
|
|
|
/* xHC spec requires PCI devices to support D3hot and D3cold */
|
|
|
|
if (xhci->hci_version >= 0x120)
|
2021-10-14 15:12:00 +03:00
|
|
|
xhci->quirks |= XHCI_DEFAULT_PM_RUNTIME_ALLOW;
|
|
|
|
|
2014-05-14 14:00:23 +02:00
|
|
|
if (xhci->quirks & XHCI_RESET_ON_RESUME)
|
|
|
|
xhci_dbg_trace(xhci, trace_xhci_dbg_quirks,
|
|
|
|
"QUIRK: Resetting on resume");
|
2011-09-23 14:20:00 -07:00
|
|
|
}
|
2011-03-22 17:08:14 +08:00
|
|
|
|
2015-07-21 17:20:25 +03:00
|
|
|
#ifdef CONFIG_ACPI
|
|
|
|
static void xhci_pme_acpi_rtd3_enable(struct pci_dev *dev)
|
|
|
|
{
|
2017-06-05 19:40:46 +03:00
|
|
|
static const guid_t intel_dsm_guid =
|
|
|
|
GUID_INIT(0xac340cb7, 0xe901, 0x45bf,
|
|
|
|
0xb7, 0xe6, 0x2b, 0x34, 0xec, 0x93, 0x1e, 0x23);
|
2015-12-04 15:53:42 +02:00
|
|
|
union acpi_object *obj;
|
|
|
|
|
2017-06-05 19:40:46 +03:00
|
|
|
obj = acpi_evaluate_dsm(ACPI_HANDLE(&dev->dev), &intel_dsm_guid, 3, 1,
|
2015-12-04 15:53:42 +02:00
|
|
|
NULL);
|
|
|
|
ACPI_FREE(obj);
|
2015-07-21 17:20:25 +03:00
|
|
|
}
|
2023-01-16 16:22:16 +02:00
|
|
|
|
|
|
|
static void xhci_find_lpm_incapable_ports(struct usb_hcd *hcd, struct usb_device *hdev)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
|
|
|
struct xhci_hub *rhub = &xhci->usb3_rhub;
|
|
|
|
int ret;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* This is not the usb3 roothub we are looking for */
|
|
|
|
if (hcd != rhub->hcd)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (hdev->maxchild > rhub->num_ports) {
|
|
|
|
dev_err(&hdev->dev, "USB3 roothub port number mismatch\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < hdev->maxchild; i++) {
|
|
|
|
ret = usb_acpi_port_lpm_incapable(hdev, i);
|
|
|
|
|
|
|
|
dev_dbg(&hdev->dev, "port-%d disable U1/U2 _DSM: %d\n", i + 1, ret);
|
|
|
|
|
|
|
|
if (ret >= 0) {
|
|
|
|
rhub->ports[i]->lpm_incapable = ret;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-21 17:20:25 +03:00
|
|
|
#else
|
2015-12-04 15:53:42 +02:00
|
|
|
static void xhci_pme_acpi_rtd3_enable(struct pci_dev *dev) { }
|
2023-01-16 16:22:16 +02:00
|
|
|
static void xhci_find_lpm_incapable_ports(struct usb_hcd *hcd, struct usb_device *hdev) { }
|
2015-07-21 17:20:25 +03:00
|
|
|
#endif /* CONFIG_ACPI */
|
|
|
|
|
2011-09-23 14:20:00 -07:00
|
|
|
/* called during probe() after chip reset completes */
|
|
|
|
static int xhci_pci_setup(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci;
|
|
|
|
struct pci_dev *pdev = to_pci_dev(hcd->self.controller);
|
|
|
|
int retval;
|
2009-04-27 19:52:28 -07:00
|
|
|
|
2015-10-01 18:40:38 +03:00
|
|
|
xhci = hcd_to_xhci(hcd);
|
|
|
|
if (!xhci->sbrn)
|
|
|
|
pci_read_config_byte(pdev, XHCI_SBRN_OFFSET, &xhci->sbrn);
|
|
|
|
|
2017-12-08 17:59:13 +02:00
|
|
|
/* imod_interval is the interrupt moderation value in nanoseconds. */
|
|
|
|
xhci->imod_interval = 40000;
|
|
|
|
|
2011-09-23 14:20:00 -07:00
|
|
|
retval = xhci_gen_setup(hcd, xhci_pci_quirks);
|
2009-04-27 19:52:28 -07:00
|
|
|
if (retval)
|
2011-09-23 14:20:00 -07:00
|
|
|
return retval;
|
2010-07-29 22:13:22 -07:00
|
|
|
|
2011-09-23 14:20:00 -07:00
|
|
|
if (!usb_hcd_is_primary_hcd(hcd))
|
|
|
|
return 0;
|
2009-04-27 19:52:28 -07:00
|
|
|
|
2020-02-10 15:45:52 +02:00
|
|
|
if (xhci->quirks & XHCI_PME_STUCK_QUIRK)
|
|
|
|
xhci_pme_acpi_rtd3_enable(pdev);
|
|
|
|
|
2009-04-27 19:52:28 -07:00
|
|
|
xhci_dbg(xhci, "Got SBRN %u\n", (unsigned int) xhci->sbrn);
|
|
|
|
|
|
|
|
/* Find any debug ports */
|
2017-01-23 14:20:03 +02:00
|
|
|
return xhci_pci_reinit(xhci, pdev);
|
2010-10-26 11:03:44 -07:00
|
|
|
}
|
|
|
|
|
2023-01-16 16:22:13 +02:00
|
|
|
static int xhci_pci_update_hub_device(struct usb_hcd *hcd, struct usb_device *hdev,
|
|
|
|
struct usb_tt *tt, gfp_t mem_flags)
|
|
|
|
{
|
2023-01-16 16:22:16 +02:00
|
|
|
/* Check if acpi claims some USB3 roothub ports are lpm incapable */
|
|
|
|
if (!hdev->parent)
|
|
|
|
xhci_find_lpm_incapable_ports(hcd, hdev);
|
|
|
|
|
2023-01-16 16:22:13 +02:00
|
|
|
return xhci_update_hub_device(hcd, hdev, tt, mem_flags);
|
|
|
|
}
|
|
|
|
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
/*
|
|
|
|
* We need to register our own PCI probe function (instead of the USB core's
|
|
|
|
* function) in order to create a second roothub under xHCI.
|
|
|
|
*/
|
|
|
|
static int xhci_pci_probe(struct pci_dev *dev, const struct pci_device_id *id)
|
|
|
|
{
|
|
|
|
int retval;
|
|
|
|
struct xhci_hcd *xhci;
|
|
|
|
struct usb_hcd *hcd;
|
2020-05-14 17:50:38 +05:30
|
|
|
struct xhci_driver_data *driver_data;
|
2020-06-29 18:18:41 +02:00
|
|
|
struct reset_control *reset;
|
2020-05-14 17:50:38 +05:30
|
|
|
|
|
|
|
driver_data = (struct xhci_driver_data *)id->driver_data;
|
|
|
|
if (driver_data && driver_data->quirks & XHCI_RENESAS_FW_QUIRK) {
|
|
|
|
retval = renesas_xhci_check_request_fw(dev, id);
|
|
|
|
if (retval)
|
|
|
|
return retval;
|
|
|
|
}
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
|
2020-06-29 18:18:41 +02:00
|
|
|
reset = devm_reset_control_get_optional_exclusive(&dev->dev, NULL);
|
|
|
|
if (IS_ERR(reset))
|
|
|
|
return PTR_ERR(reset);
|
|
|
|
reset_control_reset(reset);
|
|
|
|
|
2014-03-03 19:30:17 +02:00
|
|
|
/* Prevent runtime suspending between USB-2 and USB-3 initialization */
|
|
|
|
pm_runtime_get_noresume(&dev->dev);
|
|
|
|
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
/* Register the USB 2.0 roothub.
|
|
|
|
* FIXME: USB core must know to register the USB 2.0 roothub first.
|
|
|
|
* This is sort of silly, because we could just set the HCD driver flags
|
|
|
|
* to say USB 2.0, but I'm not sure what the implications would be in
|
|
|
|
* the other parts of the HCD code.
|
|
|
|
*/
|
2022-08-31 15:50:52 +03:00
|
|
|
retval = usb_hcd_pci_probe(dev, &xhci_pci_hc_driver);
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
|
|
|
|
if (retval)
|
2014-03-03 19:30:17 +02:00
|
|
|
goto put_runtime_pm;
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
|
|
|
|
/* USB 2.0 roothub is stored in the PCI device now. */
|
|
|
|
hcd = dev_get_drvdata(&dev->dev);
|
|
|
|
xhci = hcd_to_xhci(hcd);
|
2020-06-29 18:18:41 +02:00
|
|
|
xhci->reset = reset;
|
2020-05-14 17:50:36 +05:30
|
|
|
xhci->shared_hcd = usb_create_shared_hcd(&xhci_pci_hc_driver, &dev->dev,
|
|
|
|
pci_name(dev), hcd);
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
if (!xhci->shared_hcd) {
|
|
|
|
retval = -ENOMEM;
|
|
|
|
goto dealloc_usb2_hcd;
|
|
|
|
}
|
|
|
|
|
2018-03-20 15:57:09 +03:00
|
|
|
retval = xhci_ext_cap_init(xhci);
|
|
|
|
if (retval)
|
|
|
|
goto put_usb3_hcd;
|
|
|
|
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
retval = usb_add_hcd(xhci->shared_hcd, dev->irq,
|
2011-09-07 16:10:52 +08:00
|
|
|
IRQF_SHARED);
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
if (retval)
|
|
|
|
goto put_usb3_hcd;
|
|
|
|
/* Roothub already marked as USB 3.0 speed */
|
xhci: Add infrastructure for host-specific LPM policies.
The choice of U1 and U2 timeouts for USB 3.0 Link Power Management (LPM)
is highly host controller specific. Here are a few examples of why it's
host specific:
1. Setting the U1/U2 timeout too short may cause the link to go into
U1/U2 in between service intervals, which some hosts may tolerate,
and some may not.
2. The host controller has to modify its bus schedule in order to take
into account the Maximum Exit Latency (MEL) to bring all the links
from the host to the device into U0. If the MEL is too big, and it
takes too long to bring the links into an active state, the host
controller may not be able to service periodic endpoints in time.
3. Host controllers may also have scheduling limitations that force
them to disable U1 or U2 if a USB device is behind too many tiers of
hubs.
We could take an educated guess at what U1/U2 timeouts may work for a
particular host controller. However, that would result in a binary
search on every new configuration or alt setting installation, with
multiple failed Evaluate Context commands. Worse, the host may blindly
accept the timeouts and just fail to update its schedule for U1/U2 exit
latencies, which could result in randomly delayed periodic transfers.
Since we don't want to cause jitter in periodic transfers, or delay
config/alt setting changes too much, lay down a framework that xHCI
vendors can extend in order to add their own U1/U2 timeout policies.
To extend the framework, they will need to:
- Modify the PCI init code to add a new xhci->quirk for their host, and
set the XHCI_LPM_SUPPORT quirk flag.
- Add their own vendor-specific hooks, like the ones that will be added
in xhci_call_host_update_timeout_for_endpoint() and
xhci_check_tier_policy()
- Make the LPM enable/disable methods call those functions based on the
xhci->quirk for their host.
An example will be provided for the Intel xHCI host controller in the
next patch.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2012-05-09 10:55:03 -07:00
|
|
|
|
2014-07-25 22:01:18 +02:00
|
|
|
if (!(xhci->quirks & XHCI_BROKEN_STREAMS) &&
|
|
|
|
HCC_MAX_PSA(xhci->hcc_params) >= 4)
|
2014-02-11 20:36:04 +01:00
|
|
|
xhci->shared_hcd->can_do_streams = 1;
|
|
|
|
|
2014-03-03 19:30:17 +02:00
|
|
|
/* USB-2 and USB-3 roothubs initialized, allow runtime pm suspend */
|
|
|
|
pm_runtime_put_noidle(&dev->dev);
|
|
|
|
|
2018-09-20 19:13:38 +03:00
|
|
|
if (xhci->quirks & XHCI_DEFAULT_PM_RUNTIME_ALLOW)
|
|
|
|
pm_runtime_allow(&dev->dev);
|
|
|
|
|
2023-01-16 16:22:10 +02:00
|
|
|
dma_set_max_seg_size(&dev->dev, UINT_MAX);
|
|
|
|
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
put_usb3_hcd:
|
|
|
|
usb_put_hcd(xhci->shared_hcd);
|
|
|
|
dealloc_usb2_hcd:
|
|
|
|
usb_hcd_pci_remove(dev);
|
2014-03-03 19:30:17 +02:00
|
|
|
put_runtime_pm:
|
|
|
|
pm_runtime_put_noidle(&dev->dev);
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2010-10-26 11:03:44 -07:00
|
|
|
static void xhci_pci_remove(struct pci_dev *dev)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci;
|
|
|
|
|
|
|
|
xhci = hcd_to_xhci(pci_get_drvdata(dev));
|
2020-05-14 17:50:38 +05:30
|
|
|
|
2016-04-08 16:25:10 +03:00
|
|
|
xhci->xhc_state |= XHCI_STATE_REMOVING;
|
2018-09-20 19:13:38 +03:00
|
|
|
|
|
|
|
if (xhci->quirks & XHCI_DEFAULT_PM_RUNTIME_ALLOW)
|
|
|
|
pm_runtime_forbid(&dev->dev);
|
|
|
|
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
if (xhci->shared_hcd) {
|
|
|
|
usb_remove_hcd(xhci->shared_hcd);
|
|
|
|
usb_put_hcd(xhci->shared_hcd);
|
2018-11-09 17:21:17 +02:00
|
|
|
xhci->shared_hcd = NULL;
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
}
|
2013-09-12 08:11:06 +02:00
|
|
|
|
|
|
|
/* Workaround for spurious wakeups at shutdown with HSW */
|
|
|
|
if (xhci->quirks & XHCI_SPURIOUS_WAKEUP)
|
|
|
|
pci_set_power_state(dev, PCI_D3hot);
|
2016-08-16 10:18:06 +03:00
|
|
|
|
|
|
|
usb_hcd_pci_remove(dev);
|
2009-04-27 19:52:28 -07:00
|
|
|
}
|
|
|
|
|
2015-09-21 17:46:11 +03:00
|
|
|
/*
|
|
|
|
* In some Intel xHCI controllers, in order to get D3 working,
|
|
|
|
* through a vendor specific SSIC CONFIG register at offset 0x883c,
|
|
|
|
* SSIC PORT need to be marked as "unused" before putting xHCI
|
|
|
|
* into D3. After D3 exit, the SSIC port need to be marked as "used".
|
|
|
|
* Without this change, xHCI might not enter D3 state.
|
|
|
|
*/
|
2016-01-26 17:50:06 +02:00
|
|
|
static void xhci_ssic_port_unused_quirk(struct usb_hcd *hcd, bool suspend)
|
2015-09-21 17:46:11 +03:00
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
|
|
|
u32 val;
|
|
|
|
void __iomem *reg;
|
2016-01-26 17:50:05 +02:00
|
|
|
int i;
|
2015-09-21 17:46:11 +03:00
|
|
|
|
2016-01-26 17:50:06 +02:00
|
|
|
for (i = 0; i < SSIC_PORT_NUM; i++) {
|
|
|
|
reg = (void __iomem *) xhci->cap_regs +
|
|
|
|
SSIC_PORT_CFG2 +
|
|
|
|
i * SSIC_PORT_CFG2_OFFSET;
|
|
|
|
|
|
|
|
/* Notify SSIC that SSIC profile programming is not done. */
|
|
|
|
val = readl(reg) & ~PROG_DONE;
|
|
|
|
writel(val, reg);
|
|
|
|
|
|
|
|
/* Mark SSIC port as unused(suspend) or used(resume) */
|
|
|
|
val = readl(reg);
|
|
|
|
if (suspend)
|
|
|
|
val |= SSIC_PORT_UNUSED;
|
|
|
|
else
|
|
|
|
val &= ~SSIC_PORT_UNUSED;
|
|
|
|
writel(val, reg);
|
|
|
|
|
|
|
|
/* Notify SSIC that SSIC profile programming is done */
|
|
|
|
val = readl(reg) | PROG_DONE;
|
|
|
|
writel(val, reg);
|
|
|
|
readl(reg);
|
2015-09-21 17:46:11 +03:00
|
|
|
}
|
2016-01-26 17:50:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure PME works on some Intel xHCI controllers by writing 1 to clear
|
|
|
|
* the Internal PME flag bit in vendor specific PMCTRL register at offset 0x80a4
|
|
|
|
*/
|
|
|
|
static void xhci_pme_quirk(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
|
|
|
void __iomem *reg;
|
|
|
|
u32 val;
|
2015-09-21 17:46:11 +03:00
|
|
|
|
|
|
|
reg = (void __iomem *) xhci->cap_regs + 0x80a4;
|
|
|
|
val = readl(reg);
|
|
|
|
writel(val | BIT(28), reg);
|
|
|
|
readl(reg);
|
|
|
|
}
|
|
|
|
|
2020-10-28 22:31:23 +02:00
|
|
|
static void xhci_sparse_control_quirk(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
u32 reg;
|
|
|
|
|
|
|
|
reg = readl(hcd->regs + SPARSE_CNTL_ENABLE);
|
|
|
|
reg &= ~BIT(SPARSE_DISABLE_BIT);
|
|
|
|
writel(reg, hcd->regs + SPARSE_CNTL_ENABLE);
|
|
|
|
}
|
|
|
|
|
2010-10-14 07:23:06 -07:00
|
|
|
static int xhci_pci_suspend(struct usb_hcd *hcd, bool do_wakeup)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
xhci: Disable D3cold for buggy TI redrivers.
Some xHCI hosts contain a "redriver" from TI that silently drops port
status connect changes if the port slips into Compliance Mode. If the
port slips into compliance mode while the host is in D0, there will not
be a port status change event. If the port slips into compliance mode
while the host is in D3, the host will not send a PME. This includes
when the system is suspended (S3) or hibernated (S4).
If this happens when the system is in S3/S4, there is nothing software
can do. Other port status change events that would normally cause the
host to wake the system from S3/S4 may also be lost. This includes
remote wakeup, disconnects and connects on other ports, and overrcurrent
events. A decision was made to _NOT_ disable system suspend/hibernate
on these systems, since users are unlikely to enable wakeup from S3/S4
for the xHCI host.
Software can deal with this issue when the system is in S0. A work
around was put in to poll the port status registers for Compliance Mode.
The xHCI driver will continue to poll the registers while the host is
runtime suspended. Unfortunately, that means we can't allow the PCI
device to go into D3cold, because power will be removed from the host,
and the config space will read as all Fs.
Disable D3cold in the xHCI PCI runtime suspend function.
This patch should be backported to kernels as old as 3.2, that
contain the commit 71c731a296f1b08a3724bd1b514b64f1bda87a23 "usb: host:
xhci: Fix Compliance Mode on SN65LVPE502CP Hardware"
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Cc: Huang Ying <ying.huang@intel.com>
Cc: stable@vger.kernel.org
2013-04-18 10:02:03 -07:00
|
|
|
struct pci_dev *pdev = to_pci_dev(hcd->self.controller);
|
2016-01-26 17:50:07 +02:00
|
|
|
int ret;
|
xhci: Disable D3cold for buggy TI redrivers.
Some xHCI hosts contain a "redriver" from TI that silently drops port
status connect changes if the port slips into Compliance Mode. If the
port slips into compliance mode while the host is in D0, there will not
be a port status change event. If the port slips into compliance mode
while the host is in D3, the host will not send a PME. This includes
when the system is suspended (S3) or hibernated (S4).
If this happens when the system is in S3/S4, there is nothing software
can do. Other port status change events that would normally cause the
host to wake the system from S3/S4 may also be lost. This includes
remote wakeup, disconnects and connects on other ports, and overrcurrent
events. A decision was made to _NOT_ disable system suspend/hibernate
on these systems, since users are unlikely to enable wakeup from S3/S4
for the xHCI host.
Software can deal with this issue when the system is in S0. A work
around was put in to poll the port status registers for Compliance Mode.
The xHCI driver will continue to poll the registers while the host is
runtime suspended. Unfortunately, that means we can't allow the PCI
device to go into D3cold, because power will be removed from the host,
and the config space will read as all Fs.
Disable D3cold in the xHCI PCI runtime suspend function.
This patch should be backported to kernels as old as 3.2, that
contain the commit 71c731a296f1b08a3724bd1b514b64f1bda87a23 "usb: host:
xhci: Fix Compliance Mode on SN65LVPE502CP Hardware"
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
Cc: Huang Ying <ying.huang@intel.com>
Cc: stable@vger.kernel.org
2013-04-18 10:02:03 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Systems with the TI redriver that loses port status change events
|
|
|
|
* need to have the registers polled during D3, so avoid D3cold.
|
|
|
|
*/
|
2021-05-27 10:45:34 -05:00
|
|
|
if (xhci->quirks & (XHCI_COMP_MODE_QUIRK | XHCI_BROKEN_D3COLD))
|
2016-06-02 11:17:12 +03:00
|
|
|
pci_d3cold_disable(pdev);
|
2010-10-14 07:23:06 -07:00
|
|
|
|
2015-03-06 17:23:19 +02:00
|
|
|
if (xhci->quirks & XHCI_PME_STUCK_QUIRK)
|
2016-01-26 17:50:06 +02:00
|
|
|
xhci_pme_quirk(hcd);
|
|
|
|
|
|
|
|
if (xhci->quirks & XHCI_SSIC_PORT_UNUSED)
|
|
|
|
xhci_ssic_port_unused_quirk(hcd, true);
|
2015-03-06 17:23:19 +02:00
|
|
|
|
2020-10-28 22:31:23 +02:00
|
|
|
if (xhci->quirks & XHCI_DISABLE_SPARSE)
|
|
|
|
xhci_sparse_control_quirk(hcd);
|
|
|
|
|
2016-01-26 17:50:07 +02:00
|
|
|
ret = xhci_suspend(xhci, do_wakeup);
|
2023-03-17 17:47:14 +02:00
|
|
|
|
|
|
|
/* synchronize irq when using MSI-X */
|
|
|
|
xhci_msix_sync_irqs(xhci);
|
|
|
|
|
2016-01-26 17:50:07 +02:00
|
|
|
if (ret && (xhci->quirks & XHCI_SSIC_PORT_UNUSED))
|
|
|
|
xhci_ssic_port_unused_quirk(hcd, false);
|
|
|
|
|
|
|
|
return ret;
|
2010-10-14 07:23:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int xhci_pci_resume(struct usb_hcd *hcd, bool hibernated)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
Intel xhci: Support EHCI/xHCI port switching.
The Intel Panther Point chipsets contain an EHCI and xHCI host controller
that shares some number of skew-dependent ports. These ports can be
switched from the EHCI to the xHCI host (and vice versa) by a hardware MUX
that is controlled by registers in the xHCI PCI configuration space. The
USB 3.0 SuperSpeed terminations on the xHCI ports can be controlled
separately from the USB 2.0 data wires.
This switchover mechanism is there to support users who do a custom
install of certain non-Linux operating systems that don't have official
USB 3.0 support. By default, the ports are under EHCI, SuperSpeed
terminations are off, and USB 3.0 devices will show up under the EHCI
controller at reduced speeds. (This was more palatable for the marketing
folks than having completely dead USB 3.0 ports if no xHCI drivers are
available.) Users should be able to turn on xHCI by default through a
BIOS option, but users are happiest when they don't have to change random
BIOS settings.
This patch introduces a driver method to switchover the ports from EHCI to
xHCI before the EHCI driver finishes PCI enumeration. We want to switch
the ports over before the USB core has the chance to enumerate devices
under EHCI, or boot from USB mass storage will fail if the boot device
connects under EHCI first, and then gets disconnected when the port
switches over to xHCI.
Add code to the xHCI PCI quirk to switch the ports from EHCI to xHCI. The
PCI quirks code will run before any other PCI probe function is called, so
this avoids the issue with boot devices.
Another issue is with BIOS behavior during system resume from hibernate.
If the BIOS doesn't support xHCI, it may switch the devices under EHCI to
allow use of the USB keyboard, mice, and mass storage devices. It's
supposed to remember the value of the port routing registers and switch
them back when the OS attempts to take control of the xHCI host controller,
but we all know not to trust BIOS writers.
Make both the xHCI driver and the EHCI driver attempt to switchover the
ports in their PCI resume functions. We can't guarantee which PCI device
will be resumed first, so this avoids any race conditions. Writing a '1'
to an already set port switchover bit or a '0' to a cleared port switchover
bit should have no effect.
The xHCI PCI configuration registers will be documented in the EDS-level
chipset spec, which is not public yet. I have permission from legal and
the Intel chipset group to release this patch early to allow good Linux
support at product launch. I've tried to document the registers as much
as possible, so please let me know if anything is unclear.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2011-02-22 09:57:15 -08:00
|
|
|
struct pci_dev *pdev = to_pci_dev(hcd->self.controller);
|
2010-10-14 07:23:06 -07:00
|
|
|
int retval = 0;
|
|
|
|
|
2020-06-29 18:18:41 +02:00
|
|
|
reset_control_reset(xhci->reset);
|
|
|
|
|
Intel xhci: Support EHCI/xHCI port switching.
The Intel Panther Point chipsets contain an EHCI and xHCI host controller
that shares some number of skew-dependent ports. These ports can be
switched from the EHCI to the xHCI host (and vice versa) by a hardware MUX
that is controlled by registers in the xHCI PCI configuration space. The
USB 3.0 SuperSpeed terminations on the xHCI ports can be controlled
separately from the USB 2.0 data wires.
This switchover mechanism is there to support users who do a custom
install of certain non-Linux operating systems that don't have official
USB 3.0 support. By default, the ports are under EHCI, SuperSpeed
terminations are off, and USB 3.0 devices will show up under the EHCI
controller at reduced speeds. (This was more palatable for the marketing
folks than having completely dead USB 3.0 ports if no xHCI drivers are
available.) Users should be able to turn on xHCI by default through a
BIOS option, but users are happiest when they don't have to change random
BIOS settings.
This patch introduces a driver method to switchover the ports from EHCI to
xHCI before the EHCI driver finishes PCI enumeration. We want to switch
the ports over before the USB core has the chance to enumerate devices
under EHCI, or boot from USB mass storage will fail if the boot device
connects under EHCI first, and then gets disconnected when the port
switches over to xHCI.
Add code to the xHCI PCI quirk to switch the ports from EHCI to xHCI. The
PCI quirks code will run before any other PCI probe function is called, so
this avoids the issue with boot devices.
Another issue is with BIOS behavior during system resume from hibernate.
If the BIOS doesn't support xHCI, it may switch the devices under EHCI to
allow use of the USB keyboard, mice, and mass storage devices. It's
supposed to remember the value of the port routing registers and switch
them back when the OS attempts to take control of the xHCI host controller,
but we all know not to trust BIOS writers.
Make both the xHCI driver and the EHCI driver attempt to switchover the
ports in their PCI resume functions. We can't guarantee which PCI device
will be resumed first, so this avoids any race conditions. Writing a '1'
to an already set port switchover bit or a '0' to a cleared port switchover
bit should have no effect.
The xHCI PCI configuration registers will be documented in the EDS-level
chipset spec, which is not public yet. I have permission from legal and
the Intel chipset group to release this patch early to allow good Linux
support at product launch. I've tried to document the registers as much
as possible, so please let me know if anything is unclear.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2011-02-22 09:57:15 -08:00
|
|
|
/* The BIOS on systems with the Intel Panther Point chipset may or may
|
|
|
|
* not support xHCI natively. That means that during system resume, it
|
|
|
|
* may switch the ports back to EHCI so that users can use their
|
|
|
|
* keyboard to select a kernel from GRUB after resume from hibernate.
|
|
|
|
*
|
|
|
|
* The BIOS is supposed to remember whether the OS had xHCI ports
|
|
|
|
* enabled before resume, and switch the ports back to xHCI when the
|
|
|
|
* BIOS/OS semaphore is written, but we all know we can't trust BIOS
|
|
|
|
* writers.
|
|
|
|
*
|
|
|
|
* Unconditionally switch the ports back to xHCI after a system resume.
|
2013-07-23 11:35:47 +03:00
|
|
|
* It should not matter whether the EHCI or xHCI controller is
|
|
|
|
* resumed first. It's enough to do the switchover in xHCI because
|
|
|
|
* USB core won't notice anything as the hub driver doesn't start
|
|
|
|
* running again until after all the devices (including both EHCI and
|
|
|
|
* xHCI host controllers) have been resumed.
|
Intel xhci: Support EHCI/xHCI port switching.
The Intel Panther Point chipsets contain an EHCI and xHCI host controller
that shares some number of skew-dependent ports. These ports can be
switched from the EHCI to the xHCI host (and vice versa) by a hardware MUX
that is controlled by registers in the xHCI PCI configuration space. The
USB 3.0 SuperSpeed terminations on the xHCI ports can be controlled
separately from the USB 2.0 data wires.
This switchover mechanism is there to support users who do a custom
install of certain non-Linux operating systems that don't have official
USB 3.0 support. By default, the ports are under EHCI, SuperSpeed
terminations are off, and USB 3.0 devices will show up under the EHCI
controller at reduced speeds. (This was more palatable for the marketing
folks than having completely dead USB 3.0 ports if no xHCI drivers are
available.) Users should be able to turn on xHCI by default through a
BIOS option, but users are happiest when they don't have to change random
BIOS settings.
This patch introduces a driver method to switchover the ports from EHCI to
xHCI before the EHCI driver finishes PCI enumeration. We want to switch
the ports over before the USB core has the chance to enumerate devices
under EHCI, or boot from USB mass storage will fail if the boot device
connects under EHCI first, and then gets disconnected when the port
switches over to xHCI.
Add code to the xHCI PCI quirk to switch the ports from EHCI to xHCI. The
PCI quirks code will run before any other PCI probe function is called, so
this avoids the issue with boot devices.
Another issue is with BIOS behavior during system resume from hibernate.
If the BIOS doesn't support xHCI, it may switch the devices under EHCI to
allow use of the USB keyboard, mice, and mass storage devices. It's
supposed to remember the value of the port routing registers and switch
them back when the OS attempts to take control of the xHCI host controller,
but we all know not to trust BIOS writers.
Make both the xHCI driver and the EHCI driver attempt to switchover the
ports in their PCI resume functions. We can't guarantee which PCI device
will be resumed first, so this avoids any race conditions. Writing a '1'
to an already set port switchover bit or a '0' to a cleared port switchover
bit should have no effect.
The xHCI PCI configuration registers will be documented in the EDS-level
chipset spec, which is not public yet. I have permission from legal and
the Intel chipset group to release this patch early to allow good Linux
support at product launch. I've tried to document the registers as much
as possible, so please let me know if anything is unclear.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2011-02-22 09:57:15 -08:00
|
|
|
*/
|
2013-07-23 11:35:47 +03:00
|
|
|
|
|
|
|
if (pdev->vendor == PCI_VENDOR_ID_INTEL)
|
|
|
|
usb_enable_intel_xhci_ports(pdev);
|
Intel xhci: Support EHCI/xHCI port switching.
The Intel Panther Point chipsets contain an EHCI and xHCI host controller
that shares some number of skew-dependent ports. These ports can be
switched from the EHCI to the xHCI host (and vice versa) by a hardware MUX
that is controlled by registers in the xHCI PCI configuration space. The
USB 3.0 SuperSpeed terminations on the xHCI ports can be controlled
separately from the USB 2.0 data wires.
This switchover mechanism is there to support users who do a custom
install of certain non-Linux operating systems that don't have official
USB 3.0 support. By default, the ports are under EHCI, SuperSpeed
terminations are off, and USB 3.0 devices will show up under the EHCI
controller at reduced speeds. (This was more palatable for the marketing
folks than having completely dead USB 3.0 ports if no xHCI drivers are
available.) Users should be able to turn on xHCI by default through a
BIOS option, but users are happiest when they don't have to change random
BIOS settings.
This patch introduces a driver method to switchover the ports from EHCI to
xHCI before the EHCI driver finishes PCI enumeration. We want to switch
the ports over before the USB core has the chance to enumerate devices
under EHCI, or boot from USB mass storage will fail if the boot device
connects under EHCI first, and then gets disconnected when the port
switches over to xHCI.
Add code to the xHCI PCI quirk to switch the ports from EHCI to xHCI. The
PCI quirks code will run before any other PCI probe function is called, so
this avoids the issue with boot devices.
Another issue is with BIOS behavior during system resume from hibernate.
If the BIOS doesn't support xHCI, it may switch the devices under EHCI to
allow use of the USB keyboard, mice, and mass storage devices. It's
supposed to remember the value of the port routing registers and switch
them back when the OS attempts to take control of the xHCI host controller,
but we all know not to trust BIOS writers.
Make both the xHCI driver and the EHCI driver attempt to switchover the
ports in their PCI resume functions. We can't guarantee which PCI device
will be resumed first, so this avoids any race conditions. Writing a '1'
to an already set port switchover bit or a '0' to a cleared port switchover
bit should have no effect.
The xHCI PCI configuration registers will be documented in the EDS-level
chipset spec, which is not public yet. I have permission from legal and
the Intel chipset group to release this patch early to allow good Linux
support at product launch. I've tried to document the registers as much
as possible, so please let me know if anything is unclear.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2011-02-22 09:57:15 -08:00
|
|
|
|
2016-01-26 17:50:06 +02:00
|
|
|
if (xhci->quirks & XHCI_SSIC_PORT_UNUSED)
|
|
|
|
xhci_ssic_port_unused_quirk(hcd, false);
|
|
|
|
|
2015-03-06 17:23:19 +02:00
|
|
|
if (xhci->quirks & XHCI_PME_STUCK_QUIRK)
|
2016-01-26 17:50:06 +02:00
|
|
|
xhci_pme_quirk(hcd);
|
2015-03-06 17:23:19 +02:00
|
|
|
|
2010-10-14 07:23:06 -07:00
|
|
|
retval = xhci_resume(xhci, hibernated);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2022-11-30 11:19:42 +02:00
|
|
|
static int xhci_pci_poweroff_late(struct usb_hcd *hcd, bool do_wakeup)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
|
|
|
struct xhci_port *port;
|
|
|
|
struct usb_device *udev;
|
|
|
|
unsigned int slot_id;
|
|
|
|
u32 portsc;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Systems with XHCI_RESET_TO_DEFAULT quirk have boot firmware that
|
|
|
|
* cause significant boot delay if usb ports are in suspended U3 state
|
|
|
|
* during boot. Some USB devices survive in U3 state over S4 hibernate
|
|
|
|
*
|
|
|
|
* Disable ports that are in U3 if remote wake is not enabled for either
|
|
|
|
* host controller or connected device
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!(xhci->quirks & XHCI_RESET_TO_DEFAULT))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (i = 0; i < HCS_MAX_PORTS(xhci->hcs_params1); i++) {
|
|
|
|
port = &xhci->hw_ports[i];
|
|
|
|
portsc = readl(port->addr);
|
|
|
|
|
|
|
|
if ((portsc & PORT_PLS_MASK) != XDEV_U3)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
slot_id = xhci_find_slot_id_by_port(port->rhub->hcd, xhci,
|
|
|
|
port->hcd_portnum + 1);
|
|
|
|
if (!slot_id || !xhci->devs[slot_id]) {
|
|
|
|
xhci_err(xhci, "No dev for slot_id %d for port %d-%d in U3\n",
|
|
|
|
slot_id, port->rhub->hcd->self.busnum, port->hcd_portnum + 1);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
udev = xhci->devs[slot_id]->udev;
|
|
|
|
|
|
|
|
/* if wakeup is enabled then don't disable the port */
|
|
|
|
if (udev->do_remote_wakeup && do_wakeup)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
xhci_dbg(xhci, "port %d-%d in U3 without wakeup, disable it\n",
|
|
|
|
port->rhub->hcd->self.busnum, port->hcd_portnum + 1);
|
|
|
|
portsc = xhci_port_state_to_neutral(portsc);
|
|
|
|
writel(portsc | PORT_PE, port->addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-12-11 16:20:04 +02:00
|
|
|
static void xhci_pci_shutdown(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
struct xhci_hcd *xhci = hcd_to_xhci(hcd);
|
|
|
|
struct pci_dev *pdev = to_pci_dev(hcd->self.controller);
|
|
|
|
|
|
|
|
xhci_shutdown(hcd);
|
2023-03-17 17:47:12 +02:00
|
|
|
xhci_cleanup_msix(xhci);
|
2019-12-11 16:20:04 +02:00
|
|
|
|
|
|
|
/* Yet another workaround for spurious wakeups at shutdown with HSW */
|
|
|
|
if (xhci->quirks & XHCI_SPURIOUS_WAKEUP)
|
|
|
|
pci_set_power_state(pdev, PCI_D3hot);
|
|
|
|
}
|
|
|
|
|
2009-04-27 19:52:28 -07:00
|
|
|
/*-------------------------------------------------------------------------*/
|
|
|
|
|
2020-05-14 17:50:38 +05:30
|
|
|
static const struct xhci_driver_data reneses_data = {
|
|
|
|
.quirks = XHCI_RENESAS_FW_QUIRK,
|
|
|
|
.firmware = "renesas_usb_fw.mem",
|
|
|
|
};
|
|
|
|
|
2009-04-27 19:52:28 -07:00
|
|
|
/* PCI driver selection metadata; PCI hotplugging uses this */
|
2020-05-14 17:50:36 +05:30
|
|
|
static const struct pci_device_id pci_ids[] = {
|
2020-05-14 17:50:38 +05:30
|
|
|
{ PCI_DEVICE(0x1912, 0x0014),
|
|
|
|
.driver_data = (unsigned long)&reneses_data,
|
|
|
|
},
|
|
|
|
{ PCI_DEVICE(0x1912, 0x0015),
|
|
|
|
.driver_data = (unsigned long)&reneses_data,
|
|
|
|
},
|
2009-04-27 19:52:28 -07:00
|
|
|
/* handle any USB 3.0 xHCI controller */
|
2020-05-14 17:50:36 +05:30
|
|
|
{ PCI_DEVICE_CLASS(PCI_CLASS_SERIAL_USB_XHCI, ~0),
|
2009-04-27 19:52:28 -07:00
|
|
|
},
|
|
|
|
{ /* end: all zeroes */ }
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(pci, pci_ids);
|
2021-07-02 00:12:24 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Without CONFIG_USB_XHCI_PCI_RENESAS renesas_xhci_check_request_fw() won't
|
|
|
|
* load firmware, so don't encumber the xhci-pci driver with it.
|
|
|
|
*/
|
|
|
|
#if IS_ENABLED(CONFIG_USB_XHCI_PCI_RENESAS)
|
2020-05-14 17:50:38 +05:30
|
|
|
MODULE_FIRMWARE("renesas_usb_fw.mem");
|
2021-07-02 00:12:24 -07:00
|
|
|
#endif
|
2009-04-27 19:52:28 -07:00
|
|
|
|
|
|
|
/* pci driver glue; this is a "new style" PCI driver module */
|
|
|
|
static struct pci_driver xhci_pci_driver = {
|
2020-02-18 19:33:02 +00:00
|
|
|
.name = hcd_name,
|
2009-04-27 19:52:28 -07:00
|
|
|
.id_table = pci_ids,
|
|
|
|
|
xhci: Register second xHCI roothub.
This patch changes the xHCI driver to allocate two roothubs. This touches
the driver initialization and shutdown paths, roothub emulation code, and
port status change event handlers. This is a rather large patch, but it
can't be broken up, or it would break git-bisect.
Make the xHCI driver register its own PCI probe function. This will call
the USB core to create the USB 2.0 roothub, and then create the USB 3.0
roothub. This gets the code for registering a shared roothub out of the
USB core, and allows other HCDs later to decide if and how many shared
roothubs they want to allocate.
Make sure the xHCI's reset method marks the xHCI host controller's primary
roothub as the USB 2.0 roothub. This ensures that the high speed bus will
be processed first when the PCI device is resumed, and any USB 3.0 devices
that have migrated over to high speed will migrate back after being reset.
This ensures that USB persist works with these odd devices.
The reset method will also mark the xHCI USB2 roothub as having an
integrated TT. Like EHCI host controllers with a "rate matching hub" the
xHCI USB 2.0 roothub doesn't have an OHCI or UHCI companion controller.
It doesn't really have a TT, but we'll lie and say it has an integrated
TT. We need to do this because the USB core will reject LS/FS devices
under a HS hub without a TT.
Other details:
-------------
The roothub emulation code is changed to return the correct number of
ports for the two roothubs. For the USB 3.0 roothub, it only reports the
USB 3.0 ports. For the USB 2.0 roothub, it reports all the LS/FS/HS
ports. The code to disable a port now checks the speed of the roothub,
and refuses to disable SuperSpeed ports under the USB 3.0 roothub.
The code for initializing a new device context must be changed to set the
proper roothub port number. Since we've split the xHCI host into two
roothubs, we can't just use the port number in the ancestor hub. Instead,
we loop through the array of hardware port status register speeds and find
the Nth port with a similar speed.
The port status change event handler is updated to figure out whether the
port that reported the change is a USB 3.0 port, or a non-SuperSpeed port.
Once it figures out the port speed, it kicks the proper roothub.
The function to find a slot ID based on the port index is updated to take
into account that the two roothubs will have over-lapping port indexes.
It checks that the virtual device with a matching port index is the same
speed as the passed in roothub.
There's also changes to the driver initialization and shutdown paths:
1. Make sure that the xhci_hcd pointer is shared across the two
usb_hcd structures. The xhci_hcd pointer is allocated and the
registers are mapped in when xhci_pci_setup() is called with the
primary HCD. When xhci_pci_setup() is called with the non-primary
HCD, the xhci_hcd pointer is stored.
2. Make sure to set the sg_tablesize for both usb_hcd structures. Set
the PCI DMA mask for the non-primary HCD to allow for 64-bit or 32-bit
DMA. (The PCI DMA mask is set from the primary HCD further down in
the xhci_pci_setup() function.)
3. Ensure that the host controller doesn't start kicking khubd in
response to port status changes before both usb_hcd structures are
registered. xhci_run() only starts the xHC running once it has been
called with the non-primary roothub. Similarly, the xhci_stop()
function only halts the host controller when it is called with the
non-primary HCD. Then on the second call, it resets and cleans up the
MSI-X irqs.
Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com>
2010-12-16 11:21:10 -08:00
|
|
|
.probe = xhci_pci_probe,
|
2010-10-26 11:03:44 -07:00
|
|
|
.remove = xhci_pci_remove,
|
2009-04-27 19:52:28 -07:00
|
|
|
/* suspend and resume implemented later */
|
|
|
|
|
|
|
|
.shutdown = usb_hcd_pci_shutdown,
|
2010-10-14 07:23:06 -07:00
|
|
|
.driver = {
|
2023-03-28 15:10:43 +02:00
|
|
|
.pm = pm_ptr(&usb_hcd_pci_pm_ops),
|
2022-10-28 18:45:35 -07:00
|
|
|
},
|
2009-04-27 19:52:28 -07:00
|
|
|
};
|
|
|
|
|
2014-10-03 11:35:29 +03:00
|
|
|
static int __init xhci_pci_init(void)
|
2009-04-27 19:52:28 -07:00
|
|
|
{
|
2015-05-29 17:01:46 +03:00
|
|
|
xhci_init_driver(&xhci_pci_hc_driver, &xhci_pci_overrides);
|
2023-03-28 15:10:43 +02:00
|
|
|
xhci_pci_hc_driver.pci_suspend = pm_ptr(xhci_pci_suspend);
|
|
|
|
xhci_pci_hc_driver.pci_resume = pm_ptr(xhci_pci_resume);
|
|
|
|
xhci_pci_hc_driver.pci_poweroff_late = pm_ptr(xhci_pci_poweroff_late);
|
|
|
|
xhci_pci_hc_driver.shutdown = pm_ptr(xhci_pci_shutdown);
|
2023-03-17 17:47:12 +02:00
|
|
|
xhci_pci_hc_driver.stop = xhci_pci_stop;
|
2009-04-27 19:52:28 -07:00
|
|
|
return pci_register_driver(&xhci_pci_driver);
|
|
|
|
}
|
2014-10-03 11:35:29 +03:00
|
|
|
module_init(xhci_pci_init);
|
2009-04-27 19:52:28 -07:00
|
|
|
|
2014-10-03 11:35:29 +03:00
|
|
|
static void __exit xhci_pci_exit(void)
|
2009-04-27 19:52:28 -07:00
|
|
|
{
|
|
|
|
pci_unregister_driver(&xhci_pci_driver);
|
|
|
|
}
|
2014-10-03 11:35:29 +03:00
|
|
|
module_exit(xhci_pci_exit);
|
|
|
|
|
|
|
|
MODULE_DESCRIPTION("xHCI PCI Host Controller Driver");
|
|
|
|
MODULE_LICENSE("GPL");
|