 c06fb25d1f
			
		
	
	c06fb25d1f
	
	
		
			
	
		
	
	
		
			Some checks failed
		
		
	
	Build Kernel / Build all affected Kernels (push) Has been cancelled
				
			Build all core packages / Build all core packages for selected target (push) Has been cancelled
				
			Build and Push prebuilt tools container / Build and Push all prebuilt containers (push) Has been cancelled
				
			Build Toolchains / Build Toolchains for each target (push) Has been cancelled
				
			Build host tools / Build host tools for linux and macos based systems (push) Has been cancelled
				
			Coverity scan build / Coverity x86/64 build (push) Has been cancelled
				
			
		
			
				
	
	
		
			114 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
			
		
		
	
	
			114 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
| From: =?UTF-8?q?Rafa=C5=82=20Mi=C5=82ecki?= <rafal@milecki.pl>
 | |
| Date: Sat, 1 Oct 2016 22:54:48 +0200
 | |
| Subject: [PATCH] usb: xhci: add support for performing fake doorbell
 | |
| 
 | |
| Broadcom's Northstar XHCI controllers seem to need a special start
 | |
| procedure to work correctly. There isn't any official documentation of
 | |
| this, the problem is that controller doesn't detect any connected
 | |
| devices with default setup. Moreover connecting USB device to controller
 | |
| that doesn't run properly can cause SoC's watchdog issues.
 | |
| 
 | |
| A workaround that was successfully tested on multiple devices is to
 | |
| perform a fake doorbell. This patch adds code for doing this and enables
 | |
| it on BCM4708 family.
 | |
| ---
 | |
|  drivers/usb/host/xhci-plat.c |  6 +++++
 | |
|  drivers/usb/host/xhci.c      | 63 +++++++++++++++++++++++++++++++++++++++++---
 | |
|  drivers/usb/host/xhci.h      |  1 +
 | |
|  3 files changed, 67 insertions(+), 3 deletions(-)
 | |
| 
 | |
| --- a/drivers/usb/host/xhci-plat.c
 | |
| +++ b/drivers/usb/host/xhci-plat.c
 | |
| @@ -77,8 +77,13 @@ static int xhci_priv_resume_quirk(struct
 | |
|  static void xhci_plat_quirks(struct device *dev, struct xhci_hcd *xhci)
 | |
|  {
 | |
|  	struct xhci_plat_priv *priv = xhci_to_priv(xhci);
 | |
| +	struct platform_device*pdev = to_platform_device(dev);
 | |
| +	struct device_node *node = pdev->dev.of_node;
 | |
|  
 | |
|  	xhci->quirks |= priv->quirks;
 | |
| +
 | |
| +	if (node && of_machine_is_compatible("brcm,bcm4708"))
 | |
| +		xhci->quirks |= XHCI_FAKE_DOORBELL;
 | |
|  }
 | |
|  
 | |
|  /* called during probe() after chip reset completes */
 | |
| --- a/drivers/usb/host/xhci.c
 | |
| +++ b/drivers/usb/host/xhci.c
 | |
| @@ -161,6 +161,49 @@ int xhci_start(struct xhci_hcd *xhci)
 | |
|  	return ret;
 | |
|  }
 | |
|  
 | |
| +/**
 | |
| + * xhci_fake_doorbell - Perform a fake doorbell on a specified slot
 | |
| + *
 | |
| + * Some controllers require a fake doorbell to start correctly. Without that
 | |
| + * they simply don't detect any devices.
 | |
| + */
 | |
| +static int xhci_fake_doorbell(struct xhci_hcd *xhci, int slot_id)
 | |
| +{
 | |
| +	u32 temp;
 | |
| +
 | |
| +	/* Alloc a virt device for that slot */
 | |
| +	if (!xhci_alloc_virt_device(xhci, slot_id, NULL, GFP_NOIO)) {
 | |
| +		xhci_warn(xhci, "Could not allocate xHCI USB device data structures\n");
 | |
| +		return -ENOMEM;
 | |
| +	}
 | |
| +
 | |
| +	/* Ring fake doorbell for slot_id ep 0 */
 | |
| +	xhci_ring_ep_doorbell(xhci, slot_id, 0, 0);
 | |
| +	usleep_range(1000, 1500);
 | |
| +
 | |
| +	/* Read the status to check if HSE is set or not */
 | |
| +	temp = readl(&xhci->op_regs->status);
 | |
| +
 | |
| +	/* Clear HSE if set */
 | |
| +	if (temp & STS_FATAL) {
 | |
| +		xhci_dbg(xhci, "HSE problem detected, status: 0x%08x\n", temp);
 | |
| +		temp &= ~0x1fff;
 | |
| +		temp |= STS_FATAL;
 | |
| +		writel(temp, &xhci->op_regs->status);
 | |
| +		usleep_range(1000, 1500);
 | |
| +		readl(&xhci->op_regs->status);
 | |
| +	}
 | |
| +
 | |
| +	/* Free virt device */
 | |
| +	xhci_free_virt_device(xhci, slot_id);
 | |
| +
 | |
| +	/* We're done if controller is already running */
 | |
| +	if (readl(&xhci->op_regs->command) & CMD_RUN)
 | |
| +		return 0;
 | |
| +
 | |
| +	return xhci_start(xhci);
 | |
| +}
 | |
| +
 | |
|  /*
 | |
|   * Reset a halted HC.
 | |
|   *
 | |
| @@ -480,6 +523,15 @@ static int xhci_run_finished(struct xhci
 | |
|  		return -ENODEV;
 | |
|  	}
 | |
|  
 | |
| +	if (xhci->quirks & XHCI_FAKE_DOORBELL) {
 | |
| +		int err = xhci_fake_doorbell(xhci, 1);
 | |
| +		if (err) {
 | |
| +			xhci_halt(xhci);
 | |
| +			spin_unlock_irqrestore(&xhci->lock, flags);
 | |
| +			return err;
 | |
| +		}
 | |
| +	}
 | |
| +
 | |
|  	xhci->cmd_ring_state = CMD_RING_STATE_RUNNING;
 | |
|  
 | |
|  	if (xhci->quirks & XHCI_NEC_HOST)
 | |
| --- a/drivers/usb/host/xhci.h
 | |
| +++ b/drivers/usb/host/xhci.h
 | |
| @@ -1661,6 +1661,7 @@ struct xhci_hcd {
 | |
|  #define XHCI_WRITE_64_HI_LO	BIT_ULL(47)
 | |
|  #define XHCI_CDNS_SCTX_QUIRK	BIT_ULL(48)
 | |
|  #define XHCI_ETRON_HOST	BIT_ULL(49)
 | |
| +#define XHCI_FAKE_DOORBELL	BIT_ULL(50)
 | |
|  
 | |
|  	unsigned int		num_active_eps;
 | |
|  	unsigned int		limit_active_eps;
 |