// Code for handling XHCI "Super speed" USB controllers. // // Copyright (C) 2013 Gerd Hoffmann <kraxel@redhat.com> // Copyright (C) 2014 Kevin O'Connor <kevin@koconnor.net> // // This file may be distributed under the terms of the GNU LGPLv3 license. #include "config.h" // CONFIG_* #include "malloc.h" // memalign_low #include "memmap.h" // PAGE_SIZE #include "output.h" // dprintf #include "pcidevice.h" // foreachpci #include "pci_ids.h" // PCI_CLASS_SERIAL_USB_XHCI #include "pci_regs.h" // PCI_BASE_ADDRESS_0 #include "string.h" // memcpy #include "usb.h" // struct usb_s #include "usb-xhci.h" // struct ehci_qh #include "util.h" // timer_calc #include "x86.h" // readl // -------------------------------------------------------------- // configuration #define XHCI_RING_ITEMS 16 #define XHCI_RING_SIZE (XHCI_RING_ITEMS*sizeof(struct xhci_trb)) /* * xhci_ring structs are allocated with XHCI_RING_SIZE alignment, * then we can get it from a trb pointer (provided by evt ring). */ #define XHCI_RING(_trb) \ ((struct xhci_ring*)((u32)(_trb) & ~(XHCI_RING_SIZE-1))) // -------------------------------------------------------------- // bit definitions #define XHCI_CMD_RS (1<<0) #define XHCI_CMD_HCRST (1<<1) #define XHCI_CMD_INTE (1<<2) #define XHCI_CMD_HSEE (1<<3) #define XHCI_CMD_LHCRST (1<<7) #define XHCI_CMD_CSS (1<<8) #define XHCI_CMD_CRS (1<<9) #define XHCI_CMD_EWE (1<<10) #define XHCI_CMD_EU3S (1<<11) #define XHCI_STS_HCH (1<<0) #define XHCI_STS_HSE (1<<2) #define XHCI_STS_EINT (1<<3) #define XHCI_STS_PCD (1<<4) #define XHCI_STS_SSS (1<<8) #define XHCI_STS_RSS (1<<9) #define XHCI_STS_SRE (1<<10) #define XHCI_STS_CNR (1<<11) #define XHCI_STS_HCE (1<<12) #define XHCI_PORTSC_CCS (1<<0) #define XHCI_PORTSC_PED (1<<1) #define XHCI_PORTSC_OCA (1<<3) #define XHCI_PORTSC_PR (1<<4) #define XHCI_PORTSC_PLS_SHIFT 5 #define XHCI_PORTSC_PLS_MASK 0xf #define XHCI_PORTSC_PP (1<<9) #define XHCI_PORTSC_SPEED_SHIFT 10 #define XHCI_PORTSC_SPEED_MASK 0xf #define XHCI_PORTSC_SPEED_FULL (1<<10) #define XHCI_PORTSC_SPEED_LOW (2<<10) #define XHCI_PORTSC_SPEED_HIGH (3<<10) #define XHCI_PORTSC_SPEED_SUPER (4<<10) #define XHCI_PORTSC_PIC_SHIFT 14 #define XHCI_PORTSC_PIC_MASK 0x3 #define XHCI_PORTSC_LWS (1<<16) #define XHCI_PORTSC_CSC (1<<17) #define XHCI_PORTSC_PEC (1<<18) #define XHCI_PORTSC_WRC (1<<19) #define XHCI_PORTSC_OCC (1<<20) #define XHCI_PORTSC_PRC (1<<21) #define XHCI_PORTSC_PLC (1<<22) #define XHCI_PORTSC_CEC (1<<23) #define XHCI_PORTSC_CAS (1<<24) #define XHCI_PORTSC_WCE (1<<25) #define XHCI_PORTSC_WDE (1<<26) #define XHCI_PORTSC_WOE (1<<27) #define XHCI_PORTSC_DR (1<<30) #define XHCI_PORTSC_WPR (1<<31) #define TRB_C (1<<0) #define TRB_TYPE_SHIFT 10 #define TRB_TYPE_MASK 0x3f #define TRB_TYPE(t) (((t) >> TRB_TYPE_SHIFT) & TRB_TYPE_MASK) #define TRB_EV_ED (1<<2) #define TRB_TR_ENT (1<<1) #define TRB_TR_ISP (1<<2) #define TRB_TR_NS (1<<3) #define TRB_TR_CH (1<<4) #define TRB_TR_IOC (1<<5) #define TRB_TR_IDT (1<<6) #define TRB_TR_TBC_SHIFT 7 #define TRB_TR_TBC_MASK 0x3 #define TRB_TR_BEI (1<<9) #define TRB_TR_TLBPC_SHIFT 16 #define TRB_TR_TLBPC_MASK 0xf #define TRB_TR_FRAMEID_SHIFT 20 #define TRB_TR_FRAMEID_MASK 0x7ff #define TRB_TR_SIA (1<<31) #define TRB_TR_DIR (1<<16) #define TRB_CR_SLOTID_SHIFT 24 #define TRB_CR_SLOTID_MASK 0xff #define TRB_CR_EPID_SHIFT 16 #define TRB_CR_EPID_MASK 0x1f #define TRB_CR_BSR (1<<9) #define TRB_CR_DC (1<<9) #define TRB_LK_TC (1<<1) #define TRB_INTR_SHIFT 22 #define TRB_INTR_MASK 0x3ff #define TRB_INTR(t) (((t).status >> TRB_INTR_SHIFT) & TRB_INTR_MASK) typedef enum TRBType { TRB_RESERVED = 0, TR_NORMAL, TR_SETUP, TR_DATA, TR_STATUS, TR_ISOCH, TR_LINK, TR_EVDATA, TR_NOOP, CR_ENABLE_SLOT, CR_DISABLE_SLOT, CR_ADDRESS_DEVICE, CR_CONFIGURE_ENDPOINT, CR_EVALUATE_CONTEXT, CR_RESET_ENDPOINT, CR_STOP_ENDPOINT, CR_SET_TR_DEQUEUE, CR_RESET_DEVICE, CR_FORCE_EVENT, CR_NEGOTIATE_BW, CR_SET_LATENCY_TOLERANCE, CR_GET_PORT_BANDWIDTH, CR_FORCE_HEADER, CR_NOOP, ER_TRANSFER = 32, ER_COMMAND_COMPLETE, ER_PORT_STATUS_CHANGE, ER_BANDWIDTH_REQUEST, ER_DOORBELL, ER_HOST_CONTROLLER, ER_DEVICE_NOTIFICATION, ER_MFINDEX_WRAP, } TRBType; typedef enum TRBCCode { CC_INVALID = 0, CC_SUCCESS, CC_DATA_BUFFER_ERROR, CC_BABBLE_DETECTED, CC_USB_TRANSACTION_ERROR, CC_TRB_ERROR, CC_STALL_ERROR, CC_RESOURCE_ERROR, CC_BANDWIDTH_ERROR, CC_NO_SLOTS_ERROR, CC_INVALID_STREAM_TYPE_ERROR, CC_SLOT_NOT_ENABLED_ERROR, CC_EP_NOT_ENABLED_ERROR, CC_SHORT_PACKET, CC_RING_UNDERRUN, CC_RING_OVERRUN, CC_VF_ER_FULL, CC_PARAMETER_ERROR, CC_BANDWIDTH_OVERRUN, CC_CONTEXT_STATE_ERROR, CC_NO_PING_RESPONSE_ERROR, CC_EVENT_RING_FULL_ERROR, CC_INCOMPATIBLE_DEVICE_ERROR, CC_MISSED_SERVICE_ERROR, CC_COMMAND_RING_STOPPED, CC_COMMAND_ABORTED, CC_STOPPED, CC_STOPPED_LENGTH_INVALID, CC_MAX_EXIT_LATENCY_TOO_LARGE_ERROR = 29, CC_ISOCH_BUFFER_OVERRUN = 31, CC_EVENT_LOST_ERROR, CC_UNDEFINED_ERROR, CC_INVALID_STREAM_ID_ERROR, CC_SECONDARY_BANDWIDTH_ERROR, CC_SPLIT_TRANSACTION_ERROR } TRBCCode; enum { PLS_U0 = 0, PLS_U1 = 1, PLS_U2 = 2, PLS_U3 = 3, PLS_DISABLED = 4, PLS_RX_DETECT = 5, PLS_INACTIVE = 6, PLS_POLLING = 7, PLS_RECOVERY = 8, PLS_HOT_RESET = 9, PLS_COMPILANCE_MODE = 10, PLS_TEST_MODE = 11, PLS_RESUME = 15, }; #define xhci_get_field(data, field) \ (((data) >> field##_SHIFT) & field##_MASK) // -------------------------------------------------------------- // state structs struct xhci_ring { struct xhci_trb ring[XHCI_RING_ITEMS]; struct xhci_trb evt; u32 eidx; u32 nidx; u32 cs; struct mutex_s lock; }; struct xhci_portmap { u8 start; u8 count; }; struct usb_xhci_s { struct usb_s usb; /* devinfo */ u32 xcap; u32 ports; u32 slots; u8 context64; struct xhci_portmap usb2; struct xhci_portmap usb3; /* xhci registers */ struct xhci_caps *caps; struct xhci_op *op; struct xhci_pr *pr; struct xhci_ir *ir; struct xhci_db *db; /* xhci data structures */ struct xhci_devlist *devs; struct xhci_ring *cmds; struct xhci_ring *evts; struct xhci_er_seg *eseg; }; struct xhci_pipe { struct xhci_ring reqs; struct usb_pipe pipe; u32 slotid; u32 epid; void *buf; int bufused; }; // -------------------------------------------------------------- // tables static const char *speed_name[16] = { [ 0 ] = " - ", [ 1 ] = "Full", [ 2 ] = "Low", [ 3 ] = "High", [ 4 ] = "Super", }; static const int speed_from_xhci[16] = { [ 0 ] = -1, [ 1 ] = USB_FULLSPEED, [ 2 ] = USB_LOWSPEED, [ 3 ] = USB_HIGHSPEED, [ 4 ] = USB_SUPERSPEED, [ 5 ... 15 ] = -1, }; static const int speed_to_xhci[] = { [ USB_FULLSPEED ] = 1, [ USB_LOWSPEED ] = 2, [ USB_HIGHSPEED ] = 3, [ USB_SUPERSPEED ] = 4, }; static int wait_bit(u32 *reg, u32 mask, int value, u32 timeout) { u32 end = timer_calc(timeout); while ((readl(reg) & mask) != value) { if (timer_check(end)) { warn_timeout(); return -1; } yield(); } return 0; } /**************************************************************** * Root hub ****************************************************************/ #define XHCI_TIME_POSTPOWER 20 // Check if device attached to port static void xhci_print_port_state(int loglevel, const char *prefix, u32 port, u32 portsc) { u32 pls = xhci_get_field(portsc, XHCI_PORTSC_PLS); u32 speed = xhci_get_field(portsc, XHCI_PORTSC_SPEED); dprintf(loglevel, "%s port #%d: 0x%08x,%s%s pls %d, speed %d [%s]\n", prefix, port + 1, portsc, (portsc & XHCI_PORTSC_PP) ? " powered," : "", (portsc & XHCI_PORTSC_PED) ? " enabled," : "", pls, speed, speed_name[speed]); } static int xhci_hub_detect(struct usbhub_s *hub, u32 port) { struct usb_xhci_s *xhci = container_of(hub->cntl, struct usb_xhci_s, usb); u32 portsc = readl(&xhci->pr[port].portsc); return (portsc & XHCI_PORTSC_CCS) ? 1 : 0; } // Reset device on port static int xhci_hub_reset(struct usbhub_s *hub, u32 port) { struct usb_xhci_s *xhci = container_of(hub->cntl, struct usb_xhci_s, usb); u32 portsc = readl(&xhci->pr[port].portsc); if (!(portsc & XHCI_PORTSC_CCS)) // Device no longer connected?! return -1; switch (xhci_get_field(portsc, XHCI_PORTSC_PLS)) { case PLS_U0: // A USB3 port - controller automatically performs reset break; case PLS_POLLING: // A USB2 port - perform device reset xhci_print_port_state(3, __func__, port, portsc); writel(&xhci->pr[port].portsc, portsc | XHCI_PORTSC_PR); break; default: return -1; } // Wait for device to complete reset and be enabled u32 end = timer_calc(100); for (;;) { portsc = readl(&xhci->pr[port].portsc); if (!(portsc & XHCI_PORTSC_CCS)) // Device disconnected during reset return -1; if (portsc & XHCI_PORTSC_PED) // Reset complete break; if (timer_check(end)) { warn_timeout(); return -1; } yield(); } int rc = speed_from_xhci[xhci_get_field(portsc, XHCI_PORTSC_SPEED)]; xhci_print_port_state(1, "XHCI", port, portsc); return rc; } static int xhci_hub_portmap(struct usbhub_s *hub, u32 vport) { struct usb_xhci_s *xhci = container_of(hub->cntl, struct usb_xhci_s, usb); u32 pport = vport + 1; if (vport + 1 >= xhci->usb3.start && vport + 1 < xhci->usb3.start + xhci->usb3.count) pport = vport + 2 - xhci->usb3.start; if (vport + 1 >= xhci->usb2.start && vport + 1 < xhci->usb2.start + xhci->usb2.count) pport = vport + 2 - xhci->usb2.start; return pport; } static void xhci_hub_disconnect(struct usbhub_s *hub, u32 port) { // XXX - should turn the port power off. } static struct usbhub_op_s xhci_hub_ops = { .detect = xhci_hub_detect, .reset = xhci_hub_reset, .portmap = xhci_hub_portmap, .disconnect = xhci_hub_disconnect, }; // Find any devices connected to the root hub. static int xhci_check_ports(struct usb_xhci_s *xhci) { // Wait for port power to stabilize. msleep(XHCI_TIME_POSTPOWER); struct usbhub_s hub; memset(&hub, 0, sizeof(hub)); hub.cntl = &xhci->usb; hub.portcount = xhci->ports; hub.op = &xhci_hub_ops; usb_enumerate(&hub); return hub.devcount; } /**************************************************************** * Setup ****************************************************************/ static void xhci_free_pipes(struct usb_xhci_s *xhci) { // XXX - should walk list of pipes and free unused pipes. } static void configure_xhci(void *data) { struct usb_xhci_s *xhci = data; u32 reg; xhci->devs = memalign_high(64, sizeof(*xhci->devs) * (xhci->slots + 1)); xhci->eseg = memalign_high(64, sizeof(*xhci->eseg)); xhci->cmds = memalign_high(XHCI_RING_SIZE, sizeof(*xhci->cmds)); xhci->evts = memalign_high(XHCI_RING_SIZE, sizeof(*xhci->evts)); if (!xhci->devs || !xhci->cmds || !xhci->evts || !xhci->eseg) { warn_noalloc(); goto fail; } memset(xhci->devs, 0, sizeof(*xhci->devs) * (xhci->slots + 1)); memset(xhci->cmds, 0, sizeof(*xhci->cmds)); memset(xhci->evts, 0, sizeof(*xhci->evts)); memset(xhci->eseg, 0, sizeof(*xhci->eseg)); reg = readl(&xhci->op->usbcmd); if (reg & XHCI_CMD_RS) { reg &= ~XHCI_CMD_RS; writel(&xhci->op->usbcmd, reg); if (wait_bit(&xhci->op->usbsts, XHCI_STS_HCH, XHCI_STS_HCH, 32) != 0) goto fail; } dprintf(3, "%s: resetting\n", __func__); writel(&xhci->op->usbcmd, XHCI_CMD_HCRST); if (wait_bit(&xhci->op->usbcmd, XHCI_CMD_HCRST, 0, 1000) != 0) goto fail; if (wait_bit(&xhci->op->usbsts, XHCI_STS_CNR, 0, 1000) != 0) goto fail; writel(&xhci->op->config, xhci->slots); writel(&xhci->op->dcbaap_low, (u32)xhci->devs); writel(&xhci->op->dcbaap_high, 0); writel(&xhci->op->crcr_low, (u32)xhci->cmds | 1); writel(&xhci->op->crcr_high, 0); xhci->cmds->cs = 1; xhci->eseg->ptr_low = (u32)xhci->evts; xhci->eseg->ptr_high = 0; xhci->eseg->size = XHCI_RING_ITEMS; writel(&xhci->ir->erstsz, 1); writel(&xhci->ir->erdp_low, (u32)xhci->evts); writel(&xhci->ir->erdp_high, 0); writel(&xhci->ir->erstba_low, (u32)xhci->eseg); writel(&xhci->ir->erstba_high, 0); xhci->evts->cs = 1; reg = readl(&xhci->caps->hcsparams2); u32 spb = (reg >> 21 & 0x1f) << 5 | reg >> 27; if (spb) { dprintf(3, "%s: setup %d scratch pad buffers\n", __func__, spb); u64 *spba = memalign_high(64, sizeof(*spba) * spb); void *pad = memalign_high(PAGE_SIZE, PAGE_SIZE * spb); if (!spba || !pad) { warn_noalloc(); free(spba); free(pad); goto fail; } int i; for (i = 0; i < spb; i++) spba[i] = (u32)pad + (i * PAGE_SIZE); xhci->devs[0].ptr_low = (u32)spba; xhci->devs[0].ptr_high = 0; } reg = readl(&xhci->op->usbcmd); reg |= XHCI_CMD_RS; writel(&xhci->op->usbcmd, reg); // Find devices int count = xhci_check_ports(xhci); xhci_free_pipes(xhci); if (count) // Success return; // No devices found - shutdown and free controller. dprintf(1, "XHCI no devices found\n"); reg = readl(&xhci->op->usbcmd); reg &= ~XHCI_CMD_RS; writel(&xhci->op->usbcmd, reg); wait_bit(&xhci->op->usbsts, XHCI_STS_HCH, XHCI_STS_HCH, 32); fail: free(xhci->eseg); free(xhci->evts); free(xhci->cmds); free(xhci->devs); free(xhci); } static struct usb_xhci_s* xhci_controller_setup(void *baseaddr) { struct usb_xhci_s *xhci = malloc_high(sizeof(*xhci)); if (!xhci) { warn_noalloc(); return NULL; } memset(xhci, 0, sizeof(*xhci)); xhci->caps = baseaddr; xhci->op = baseaddr + readb(&xhci->caps->caplength); xhci->pr = baseaddr + readb(&xhci->caps->caplength) + 0x400; xhci->db = baseaddr + readl(&xhci->caps->dboff); xhci->ir = baseaddr + readl(&xhci->caps->rtsoff) + 0x20; u32 hcs1 = readl(&xhci->caps->hcsparams1); u32 hcc = readl(&xhci->caps->hccparams); xhci->ports = (hcs1 >> 24) & 0xff; xhci->slots = hcs1 & 0xff; xhci->xcap = ((hcc >> 16) & 0xffff) << 2; xhci->context64 = (hcc & 0x04) ? 1 : 0; xhci->usb.type = USB_TYPE_XHCI; dprintf(1, "XHCI init: regs @ %p, %d ports, %d slots" ", %d byte contexts\n" , xhci->caps, xhci->ports, xhci->slots , xhci->context64 ? 64 : 32); if (xhci->xcap) { u32 off; void *addr = baseaddr + xhci->xcap; do { struct xhci_xcap *xcap = addr; u32 ports, name, cap = readl(&xcap->cap); switch (cap & 0xff) { case 0x02: name = readl(&xcap->data[0]); ports = readl(&xcap->data[1]); u8 major = (cap >> 24) & 0xff; u8 minor = (cap >> 16) & 0xff; u8 count = (ports >> 8) & 0xff; u8 start = (ports >> 0) & 0xff; dprintf(1, "XHCI protocol %c%c%c%c %x.%02x" ", %d ports (offset %d), def %x\n" , (name >> 0) & 0xff , (name >> 8) & 0xff , (name >> 16) & 0xff , (name >> 24) & 0xff , major, minor , count, start , ports >> 16); if (name == 0x20425355 /* "USB " */) { if (major == 2) { xhci->usb2.start = start; xhci->usb2.count = count; } if (major == 3) { xhci->usb3.start = start; xhci->usb3.count = count; } } break; default: dprintf(1, "XHCI extcap 0x%x @ %p\n", cap & 0xff, addr); break; } off = (cap >> 8) & 0xff; addr += off << 2; } while (off > 0); } u32 pagesize = readl(&xhci->op->pagesize); if (PAGE_SIZE != (pagesize<<12)) { dprintf(1, "XHCI driver does not support page size code %d\n" , pagesize<<12); free(xhci); return NULL; } return xhci; } static void xhci_controller_setup_pci(struct pci_device *pci) { struct usb_xhci_s *xhci; void *baseaddr; baseaddr = pci_enable_membar(pci, PCI_BASE_ADDRESS_0); if (!baseaddr) return; dprintf(1, "PCI: XHCI at %pP (mmio %p)\n", pci, baseaddr); pci_enable_busmaster(pci); xhci = xhci_controller_setup(baseaddr); if (!xhci) return; xhci->usb.pci = pci; run_thread(configure_xhci, xhci); } static void xhci_controller_setup_acpi(struct acpi_device *dev) { struct usb_xhci_s *xhci; u64 mem, unused; void *baseaddr; if (acpi_dsdt_find_mem(dev, &mem, &unused) < 0) return; if (mem >= 0x100000000ll) return; baseaddr = (void*)(u32)mem; dprintf(1, "ACPI: XHCI at mmio %p\n", baseaddr); xhci = xhci_controller_setup(baseaddr); if (!xhci) return; xhci->usb.mmio = baseaddr; run_thread(configure_xhci, xhci); } void xhci_setup(void) { if (! CONFIG_USB_XHCI) return; struct pci_device *pci; foreachpci(pci) { if (pci_classprog(pci) == PCI_CLASS_SERIAL_USB_XHCI) xhci_controller_setup_pci(pci); } u16 xhci_eisaid = 0x0d10; struct acpi_device *dev; for (dev = acpi_dsdt_find_eisaid(NULL, xhci_eisaid); dev != NULL; dev = acpi_dsdt_find_eisaid(dev, xhci_eisaid)) { xhci_controller_setup_acpi(dev); } } /**************************************************************** * End point communication ****************************************************************/ // Signal the hardware to process events on a TRB ring static void xhci_doorbell(struct usb_xhci_s *xhci, u32 slotid, u32 value) { dprintf(5, "%s: slotid %d, epid %d\n", __func__, slotid, value); struct xhci_db *db = xhci->db; void *addr = &db[slotid].doorbell; writel(addr, value); } // Dequeue events on the XHCI command ring generated by the hardware static void xhci_process_events(struct usb_xhci_s *xhci) { struct xhci_ring *evts = xhci->evts; for (;;) { /* check for event */ u32 nidx = evts->nidx; u32 cs = evts->cs; struct xhci_trb *etrb = evts->ring + nidx; u32 control = etrb->control; if ((control & TRB_C) != (cs ? 1 : 0)) return; /* process event */ u32 evt_type = TRB_TYPE(control); u32 evt_cc = (etrb->status >> 24) & 0xff; switch (evt_type) { case ER_TRANSFER: case ER_COMMAND_COMPLETE: { struct xhci_trb *rtrb = (void*)etrb->ptr_low; struct xhci_ring *ring = XHCI_RING(rtrb); struct xhci_trb *evt = &ring->evt; u32 eidx = rtrb - ring->ring + 1; dprintf(5, "%s: ring %p [trb %p, evt %p, type %d, eidx %d, cc %d]\n", __func__, ring, rtrb, evt, evt_type, eidx, evt_cc); memcpy(evt, etrb, sizeof(*etrb)); ring->eidx = eidx; break; } case ER_PORT_STATUS_CHANGE: { u32 port = ((etrb->ptr_low >> 24) & 0xff) - 1; // Read status, and clear port status change bits u32 portsc = readl(&xhci->pr[port].portsc); u32 pclear = (((portsc & ~(XHCI_PORTSC_PED|XHCI_PORTSC_PR)) & ~(XHCI_PORTSC_PLS_MASK<<XHCI_PORTSC_PLS_SHIFT)) | (1<<XHCI_PORTSC_PLS_SHIFT)); writel(&xhci->pr[port].portsc, pclear); xhci_print_port_state(3, __func__, port, portsc); break; } default: dprintf(1, "%s: unknown event, type %d, cc %d\n", __func__, evt_type, evt_cc); break; } /* move ring index, notify xhci */ nidx++; if (nidx == XHCI_RING_ITEMS) { nidx = 0; cs = cs ? 0 : 1; evts->cs = cs; } evts->nidx = nidx; struct xhci_ir *ir = xhci->ir; u32 erdp = (u32)(evts->ring + nidx); writel(&ir->erdp_low, erdp); writel(&ir->erdp_high, 0); } } // Check if a ring has any pending TRBs static int xhci_ring_busy(struct xhci_ring *ring) { u32 eidx = ring->eidx; u32 nidx = ring->nidx; return (eidx != nidx); } // Wait for a ring to empty (all TRBs processed by hardware) static int xhci_event_wait(struct usb_xhci_s *xhci, struct xhci_ring *ring, u32 timeout) { u32 end = timer_calc(timeout); for (;;) { xhci_process_events(xhci); if (!xhci_ring_busy(ring)) { u32 status = ring->evt.status; return (status >> 24) & 0xff; } if (timer_check(end)) { warn_timeout(); return -1; } yield(); } } // Add a TRB to the given ring static void xhci_trb_fill(struct xhci_ring *ring , void *data, u32 xferlen, u32 flags) { struct xhci_trb *dst = &ring->ring[ring->nidx]; if (flags & TRB_TR_IDT) { memcpy(&dst->ptr_low, data, xferlen); } else { dst->ptr_low = (u32)data; dst->ptr_high = 0; } dst->status = xferlen; dst->control = flags | (ring->cs ? TRB_C : 0); } // Queue a TRB onto a ring, wrapping ring as needed static void xhci_trb_queue(struct xhci_ring *ring, void *data, u32 xferlen, u32 flags) { if (ring->nidx >= ARRAY_SIZE(ring->ring) - 1) { xhci_trb_fill(ring, ring->ring, 0, (TR_LINK << 10) | TRB_LK_TC); ring->nidx = 0; ring->cs ^= 1; dprintf(5, "%s: ring %p [linked]\n", __func__, ring); } xhci_trb_fill(ring, data, xferlen, flags); ring->nidx++; dprintf(5, "%s: ring %p [nidx %d, len %d]\n", __func__, ring, ring->nidx, xferlen); } // Submit a command to the xhci controller ring static int xhci_cmd_submit(struct usb_xhci_s *xhci, struct xhci_inctx *inctx , u32 flags) { if (inctx) { struct xhci_slotctx *slot = (void*)&inctx[1 << xhci->context64]; u32 port = ((slot->ctx[1] >> 16) & 0xff) - 1; u32 portsc = readl(&xhci->pr[port].portsc); if (!(portsc & XHCI_PORTSC_CCS)) { // Device no longer connected?! xhci_print_port_state(1, __func__, port, portsc); return -1; } } mutex_lock(&xhci->cmds->lock); xhci_trb_queue(xhci->cmds, inctx, 0, flags); xhci_doorbell(xhci, 0, 0); int rc = xhci_event_wait(xhci, xhci->cmds, 1000); mutex_unlock(&xhci->cmds->lock); return rc; } static int xhci_cmd_enable_slot(struct usb_xhci_s *xhci) { dprintf(3, "%s:\n", __func__); int cc = xhci_cmd_submit(xhci, NULL, CR_ENABLE_SLOT << 10); if (cc != CC_SUCCESS) return -1; return (xhci->cmds->evt.control >> 24) & 0xff; } static int xhci_cmd_disable_slot(struct usb_xhci_s *xhci, u32 slotid) { dprintf(3, "%s: slotid %d\n", __func__, slotid); return xhci_cmd_submit(xhci, NULL, (CR_DISABLE_SLOT << 10) | (slotid << 24)); } static int xhci_cmd_address_device(struct usb_xhci_s *xhci, u32 slotid , struct xhci_inctx *inctx) { dprintf(3, "%s: slotid %d\n", __func__, slotid); return xhci_cmd_submit(xhci, inctx , (CR_ADDRESS_DEVICE << 10) | (slotid << 24)); } static int xhci_cmd_configure_endpoint(struct usb_xhci_s *xhci, u32 slotid , struct xhci_inctx *inctx) { dprintf(3, "%s: slotid %d, add 0x%x, del 0x%x\n", __func__, slotid, inctx->add, inctx->del); return xhci_cmd_submit(xhci, inctx , (CR_CONFIGURE_ENDPOINT << 10) | (slotid << 24)); } static int xhci_cmd_evaluate_context(struct usb_xhci_s *xhci, u32 slotid , struct xhci_inctx *inctx) { dprintf(3, "%s: slotid %d, add 0x%x, del 0x%x\n", __func__, slotid, inctx->add, inctx->del); return xhci_cmd_submit(xhci, inctx , (CR_EVALUATE_CONTEXT << 10) | (slotid << 24)); } static struct xhci_inctx * xhci_alloc_inctx(struct usbdevice_s *usbdev, int maxepid) { struct usb_xhci_s *xhci = container_of( usbdev->hub->cntl, struct usb_xhci_s, usb); int size = (sizeof(struct xhci_inctx) * 33) << xhci->context64; struct xhci_inctx *in = memalign_tmphigh(2048 << xhci->context64, size); if (!in) { warn_noalloc(); return NULL; } memset(in, 0, size); struct xhci_slotctx *slot = (void*)&in[1 << xhci->context64]; slot->ctx[0] |= maxepid << 27; // context entries slot->ctx[0] |= speed_to_xhci[usbdev->speed] << 20; // Set high-speed hub flags. struct usbdevice_s *hubdev = usbdev->hub->usbdev; if (hubdev) { if (usbdev->speed == USB_LOWSPEED || usbdev->speed == USB_FULLSPEED) { struct xhci_pipe *hpipe = container_of( hubdev->defpipe, struct xhci_pipe, pipe); if (hubdev->speed == USB_HIGHSPEED) { slot->ctx[2] |= hpipe->slotid; slot->ctx[2] |= (usbdev->port+1) << 8; } else { struct xhci_slotctx *hslot = (void*)xhci->devs[hpipe->slotid].ptr_low; slot->ctx[2] = hslot->ctx[2]; } } u32 route = 0; while (usbdev->hub->usbdev) { route <<= 4; route |= (usbdev->port+1) & 0xf; usbdev = usbdev->hub->usbdev; } slot->ctx[0] |= route; } slot->ctx[1] |= (usbdev->port+1) << 16; return in; } static int xhci_config_hub(struct usbhub_s *hub) { struct usb_xhci_s *xhci = container_of( hub->cntl, struct usb_xhci_s, usb); struct xhci_pipe *pipe = container_of( hub->usbdev->defpipe, struct xhci_pipe, pipe); struct xhci_slotctx *hdslot = (void*)xhci->devs[pipe->slotid].ptr_low; if ((hdslot->ctx[3] >> 27) == 3) // Already configured return 0; struct xhci_inctx *in = xhci_alloc_inctx(hub->usbdev, 1); if (!in) return -1; in->add = 0x01; struct xhci_slotctx *slot = (void*)&in[1 << xhci->context64]; slot->ctx[0] |= 1 << 26; slot->ctx[1] |= hub->portcount << 24; int cc = xhci_cmd_configure_endpoint(xhci, pipe->slotid, in); free(in); if (cc != CC_SUCCESS) { dprintf(1, "%s: configure hub: failed (cc %d)\n", __func__, cc); return -1; } return 0; } static struct usb_pipe * xhci_alloc_pipe(struct usbdevice_s *usbdev , struct usb_endpoint_descriptor *epdesc) { u8 eptype = epdesc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK; struct usb_xhci_s *xhci = container_of( usbdev->hub->cntl, struct usb_xhci_s, usb); struct xhci_pipe *pipe; u32 epid; if (epdesc->bEndpointAddress == 0) { epid = 1; } else { epid = (epdesc->bEndpointAddress & 0x0f) * 2; epid += (epdesc->bEndpointAddress & USB_DIR_IN) ? 1 : 0; } if (eptype == USB_ENDPOINT_XFER_CONTROL) pipe = memalign_high(XHCI_RING_SIZE, sizeof(*pipe)); else pipe = memalign_low(XHCI_RING_SIZE, sizeof(*pipe)); if (!pipe) { warn_noalloc(); return NULL; } memset(pipe, 0, sizeof(*pipe)); usb_desc2pipe(&pipe->pipe, usbdev, epdesc); pipe->epid = epid; pipe->reqs.cs = 1; if (eptype == USB_ENDPOINT_XFER_INT) { pipe->buf = malloc_high(pipe->pipe.maxpacket); if (!pipe->buf) { warn_noalloc(); free(pipe); return NULL; } } // Allocate input context and initialize endpoint info. struct xhci_inctx *in = xhci_alloc_inctx(usbdev, epid); if (!in) goto fail; in->add = 0x01 | (1 << epid); struct xhci_epctx *ep = (void*)&in[(pipe->epid+1) << xhci->context64]; if (eptype == USB_ENDPOINT_XFER_INT) ep->ctx[0] = (usb_get_period(usbdev, epdesc) + 3) << 16; ep->ctx[1] |= eptype << 3; if (epdesc->bEndpointAddress & USB_DIR_IN || eptype == USB_ENDPOINT_XFER_CONTROL) ep->ctx[1] |= 1 << 5; ep->ctx[1] |= pipe->pipe.maxpacket << 16; ep->deq_low = (u32)&pipe->reqs.ring[0]; ep->deq_low |= 1; // dcs ep->length = pipe->pipe.maxpacket; dprintf(3, "%s: usbdev %p, ring %p, slotid %d, epid %d\n", __func__, usbdev, &pipe->reqs, pipe->slotid, pipe->epid); if (pipe->epid == 1) { if (usbdev->hub->usbdev) { // Make sure parent hub is configured. int ret = xhci_config_hub(usbdev->hub); if (ret) goto fail; } // Enable slot. u32 size = (sizeof(struct xhci_slotctx) * 32) << xhci->context64; struct xhci_slotctx *dev = memalign_high(1024 << xhci->context64, size); if (!dev) { warn_noalloc(); goto fail; } int slotid = xhci_cmd_enable_slot(xhci); if (slotid < 0) { dprintf(1, "%s: enable slot: failed\n", __func__); free(dev); goto fail; } dprintf(3, "%s: enable slot: got slotid %d\n", __func__, slotid); memset(dev, 0, size); xhci->devs[slotid].ptr_low = (u32)dev; xhci->devs[slotid].ptr_high = 0; // Send set_address command. int cc = xhci_cmd_address_device(xhci, slotid, in); if (cc != CC_SUCCESS) { dprintf(1, "%s: address device: failed (cc %d)\n", __func__, cc); cc = xhci_cmd_disable_slot(xhci, slotid); if (cc != CC_SUCCESS) { dprintf(1, "%s: disable failed (cc %d)\n", __func__, cc); goto fail; } xhci->devs[slotid].ptr_low = 0; free(dev); goto fail; } pipe->slotid = slotid; } else { struct xhci_pipe *defpipe = container_of( usbdev->defpipe, struct xhci_pipe, pipe); pipe->slotid = defpipe->slotid; // Send configure command. int cc = xhci_cmd_configure_endpoint(xhci, pipe->slotid, in); if (cc != CC_SUCCESS) { dprintf(1, "%s: configure endpoint: failed (cc %d)\n", __func__, cc); goto fail; } } free(in); return &pipe->pipe; fail: free(pipe->buf); free(pipe); free(in); return NULL; } struct usb_pipe * xhci_realloc_pipe(struct usbdevice_s *usbdev, struct usb_pipe *upipe , struct usb_endpoint_descriptor *epdesc) { if (!CONFIG_USB_XHCI) return NULL; if (!epdesc) { usb_add_freelist(upipe); return NULL; } if (!upipe) return xhci_alloc_pipe(usbdev, epdesc); u8 eptype = epdesc->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK; int oldmaxpacket = upipe->maxpacket; usb_desc2pipe(upipe, usbdev, epdesc); struct xhci_pipe *pipe = container_of(upipe, struct xhci_pipe, pipe); struct usb_xhci_s *xhci = container_of( pipe->pipe.cntl, struct usb_xhci_s, usb); dprintf(3, "%s: usbdev %p, ring %p, slotid %d, epid %d\n", __func__, usbdev, &pipe->reqs, pipe->slotid, pipe->epid); if (eptype != USB_ENDPOINT_XFER_CONTROL || upipe->maxpacket == oldmaxpacket) return upipe; // maxpacket has changed on control endpoint - update controller. dprintf(1, "%s: reconf ctl endpoint pkt size: %d -> %d\n", __func__, oldmaxpacket, pipe->pipe.maxpacket); struct xhci_inctx *in = xhci_alloc_inctx(usbdev, 1); if (!in) return upipe; in->add = (1 << 1); struct xhci_epctx *ep = (void*)&in[2 << xhci->context64]; ep->ctx[1] |= (pipe->pipe.maxpacket << 16); int cc = xhci_cmd_evaluate_context(xhci, pipe->slotid, in); if (cc != CC_SUCCESS) { dprintf(1, "%s: reconf ctl endpoint: failed (cc %d)\n", __func__, cc); } free(in); return upipe; } // Submit a USB "setup" message request to the pipe's ring static void xhci_xfer_setup(struct xhci_pipe *pipe, int dir, void *cmd , void *data, int datalen) { struct usb_xhci_s *xhci = container_of( pipe->pipe.cntl, struct usb_xhci_s, usb); xhci_trb_queue(&pipe->reqs, cmd, USB_CONTROL_SETUP_SIZE , (TR_SETUP << 10) | TRB_TR_IDT | ((datalen ? (dir ? 3 : 2) : 0) << 16)); if (datalen) xhci_trb_queue(&pipe->reqs, data, datalen, (TR_DATA << 10) | ((dir ? 1 : 0) << 16)); xhci_trb_queue(&pipe->reqs, NULL, 0, (TR_STATUS << 10) | TRB_TR_IOC | ((dir ? 0 : 1) << 16)); xhci_doorbell(xhci, pipe->slotid, pipe->epid); } // Submit a USB transfer request to the pipe's ring static void xhci_xfer_normal(struct xhci_pipe *pipe, void *data, int datalen) { struct usb_xhci_s *xhci = container_of( pipe->pipe.cntl, struct usb_xhci_s, usb); xhci_trb_queue(&pipe->reqs, data, datalen, (TR_NORMAL << 10) | TRB_TR_IOC); xhci_doorbell(xhci, pipe->slotid, pipe->epid); } int xhci_send_pipe(struct usb_pipe *p, int dir, const void *cmd , void *data, int datalen) { if (!CONFIG_USB_XHCI) return -1; struct xhci_pipe *pipe = container_of(p, struct xhci_pipe, pipe); struct usb_xhci_s *xhci = container_of( pipe->pipe.cntl, struct usb_xhci_s, usb); if (cmd) { const struct usb_ctrlrequest *req = cmd; if (req->bRequest == USB_REQ_SET_ADDRESS) // Set address command sent during xhci_alloc_pipe. return 0; xhci_xfer_setup(pipe, dir, (void*)req, data, datalen); } else { xhci_xfer_normal(pipe, data, datalen); } int cc = xhci_event_wait(xhci, &pipe->reqs, usb_xfer_time(p, datalen)); if (cc != CC_SUCCESS) { dprintf(1, "%s: xfer failed (cc %d)\n", __func__, cc); return -1; } return 0; } int VISIBLE32FLAT xhci_poll_intr(struct usb_pipe *p, void *data) { if (!CONFIG_USB_XHCI) return -1; struct xhci_pipe *pipe = container_of(p, struct xhci_pipe, pipe); struct usb_xhci_s *xhci = container_of( pipe->pipe.cntl, struct usb_xhci_s, usb); u32 len = pipe->pipe.maxpacket; void *buf = pipe->buf; int bufused = pipe->bufused; if (!bufused) { xhci_xfer_normal(pipe, buf, len); bufused = 1; pipe->bufused = bufused; return -1; } xhci_process_events(xhci); if (xhci_ring_busy(&pipe->reqs)) return -1; dprintf(5, "%s: st %x ct %x [ %p <= %p / %d ]\n", __func__, pipe->reqs.evt.status, pipe->reqs.evt.control, data, buf, len); memcpy(data, buf, len); xhci_xfer_normal(pipe, buf, len); return 0; }