aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb
diff options
context:
space:
mode:
authorAndiry Xu <andiry.xu@amd.com>2010-10-14 10:23:00 -0400
committerGreg Kroah-Hartman <gregkh@suse.de>2010-10-22 13:22:13 -0400
commit561925318725a41189a69f36ebe99199b3fb84c4 (patch)
treee84c72692f8fd9ed67c298db17d26ccb9970a7a8 /drivers/usb
parentbe88fe4f4dda93e3264a887745123b1e6c4a6845 (diff)
USB: xHCI: port remote wakeup implementation
This commit implements port remote wakeup. When a port is in U3 state and resume signaling is detected from a device, the port transitions to the Resume state, and the xHC generates a Port Status Change Event. For USB3 port, software write a '0' to the PLS field to complete the resume signaling. For USB2 port, the resume should be signaling for at least 20ms, irq handler set a timer for port remote wakeup, and then finishes process in hub_control GetPortStatus. Some codes are borrowed from EHCI code. Signed-off-by: Andiry Xu <andiry.xu@amd.com> Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Diffstat (limited to 'drivers/usb')
-rw-r--r--drivers/usb/host/xhci-hub.c44
-rw-r--r--drivers/usb/host/xhci-mem.c2
-rw-r--r--drivers/usb/host/xhci-ring.c57
-rw-r--r--drivers/usb/host/xhci.h4
4 files changed, 101 insertions, 6 deletions
diff --git a/drivers/usb/host/xhci-hub.c b/drivers/usb/host/xhci-hub.c
index 14b48b261e06..8163f17e7043 100644
--- a/drivers/usb/host/xhci-hub.c
+++ b/drivers/usb/host/xhci-hub.c
@@ -123,7 +123,7 @@ static unsigned int xhci_port_speed(unsigned int port_status)
123 * writing a 0 clears the bit and writing a 1 sets the bit (RWS). 123 * writing a 0 clears the bit and writing a 1 sets the bit (RWS).
124 * For all other types (RW1S, RW1CS, RW, and RZ), writing a '0' has no effect. 124 * For all other types (RW1S, RW1CS, RW, and RZ), writing a '0' has no effect.
125 */ 125 */
126static u32 xhci_port_state_to_neutral(u32 state) 126u32 xhci_port_state_to_neutral(u32 state)
127{ 127{
128 /* Save read-only status and port state */ 128 /* Save read-only status and port state */
129 return (state & XHCI_PORT_RO) | (state & XHCI_PORT_RWS); 129 return (state & XHCI_PORT_RO) | (state & XHCI_PORT_RWS);
@@ -132,7 +132,7 @@ static u32 xhci_port_state_to_neutral(u32 state)
132/* 132/*
133 * find slot id based on port number. 133 * find slot id based on port number.
134 */ 134 */
135static int xhci_find_slot_id_by_port(struct xhci_hcd *xhci, u16 port) 135int xhci_find_slot_id_by_port(struct xhci_hcd *xhci, u16 port)
136{ 136{
137 int slot_id; 137 int slot_id;
138 int i; 138 int i;
@@ -210,7 +210,7 @@ command_cleanup:
210/* 210/*
211 * Ring device, it rings the all doorbells unconditionally. 211 * Ring device, it rings the all doorbells unconditionally.
212 */ 212 */
213static void xhci_ring_device(struct xhci_hcd *xhci, int slot_id) 213void xhci_ring_device(struct xhci_hcd *xhci, int slot_id)
214{ 214{
215 int i; 215 int i;
216 216
@@ -276,7 +276,7 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
276 struct xhci_hcd *xhci = hcd_to_xhci(hcd); 276 struct xhci_hcd *xhci = hcd_to_xhci(hcd);
277 int ports; 277 int ports;
278 unsigned long flags; 278 unsigned long flags;
279 u32 temp, status; 279 u32 temp, temp1, status;
280 int retval = 0; 280 int retval = 0;
281 u32 __iomem *addr; 281 u32 __iomem *addr;
282 int slot_id; 282 int slot_id;
@@ -315,6 +315,34 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue,
315 if ((temp & PORT_PLS_MASK) == XDEV_U3 315 if ((temp & PORT_PLS_MASK) == XDEV_U3
316 && (temp & PORT_POWER)) 316 && (temp & PORT_POWER))
317 status |= 1 << USB_PORT_FEAT_SUSPEND; 317 status |= 1 << USB_PORT_FEAT_SUSPEND;
318 if ((temp & PORT_PLS_MASK) == XDEV_RESUME) {
319 if ((temp & PORT_RESET) || !(temp & PORT_PE))
320 goto error;
321 if (!DEV_SUPERSPEED(temp) && time_after_eq(jiffies,
322 xhci->resume_done[wIndex])) {
323 xhci_dbg(xhci, "Resume USB2 port %d\n",
324 wIndex + 1);
325 xhci->resume_done[wIndex] = 0;
326 temp1 = xhci_port_state_to_neutral(temp);
327 temp1 &= ~PORT_PLS_MASK;
328 temp1 |= PORT_LINK_STROBE | XDEV_U0;
329 xhci_writel(xhci, temp1, addr);
330
331 xhci_dbg(xhci, "set port %d resume\n",
332 wIndex + 1);
333 slot_id = xhci_find_slot_id_by_port(xhci,
334 wIndex + 1);
335 if (!slot_id) {
336 xhci_dbg(xhci, "slot_id is zero\n");
337 goto error;
338 }
339 xhci_ring_device(xhci, slot_id);
340 xhci->port_c_suspend[wIndex >> 5] |=
341 1 << (wIndex & 31);
342 xhci->suspended_ports[wIndex >> 5] &=
343 ~(1 << (wIndex & 31));
344 }
345 }
318 if ((temp & PORT_PLS_MASK) == XDEV_U0 346 if ((temp & PORT_PLS_MASK) == XDEV_U0
319 && (temp & PORT_POWER) 347 && (temp & PORT_POWER)
320 && (xhci->suspended_ports[wIndex >> 5] & 348 && (xhci->suspended_ports[wIndex >> 5] &
@@ -500,6 +528,7 @@ int xhci_hub_status_data(struct usb_hcd *hcd, char *buf)
500{ 528{
501 unsigned long flags; 529 unsigned long flags;
502 u32 temp, status; 530 u32 temp, status;
531 u32 mask;
503 int i, retval; 532 int i, retval;
504 struct xhci_hcd *xhci = hcd_to_xhci(hcd); 533 struct xhci_hcd *xhci = hcd_to_xhci(hcd);
505 int ports; 534 int ports;
@@ -512,13 +541,18 @@ int xhci_hub_status_data(struct usb_hcd *hcd, char *buf)
512 memset(buf, 0, retval); 541 memset(buf, 0, retval);
513 status = 0; 542 status = 0;
514 543
544 mask = PORT_CSC | PORT_PEC | PORT_OCC;
545
515 spin_lock_irqsave(&xhci->lock, flags); 546 spin_lock_irqsave(&xhci->lock, flags);
516 /* For each port, did anything change? If so, set that bit in buf. */ 547 /* For each port, did anything change? If so, set that bit in buf. */
517 for (i = 0; i < ports; i++) { 548 for (i = 0; i < ports; i++) {
518 addr = &xhci->op_regs->port_status_base + 549 addr = &xhci->op_regs->port_status_base +
519 NUM_PORT_REGS*i; 550 NUM_PORT_REGS*i;
520 temp = xhci_readl(xhci, addr); 551 temp = xhci_readl(xhci, addr);
521 if (temp & (PORT_CSC | PORT_PEC | PORT_OCC)) { 552 if ((temp & mask) != 0 ||
553 (xhci->port_c_suspend[i >> 5] & 1 << (i & 31)) ||
554 (xhci->resume_done[i] && time_after_eq(
555 jiffies, xhci->resume_done[i]))) {
522 buf[(i + 1) / 8] |= 1 << (i + 1) % 8; 556 buf[(i + 1) / 8] |= 1 << (i + 1) % 8;
523 status = 1; 557 status = 1;
524 } 558 }
diff --git a/drivers/usb/host/xhci-mem.c b/drivers/usb/host/xhci-mem.c
index 858a82867e1d..fd888bc0422b 100644
--- a/drivers/usb/host/xhci-mem.c
+++ b/drivers/usb/host/xhci-mem.c
@@ -1803,6 +1803,8 @@ int xhci_mem_init(struct xhci_hcd *xhci, gfp_t flags)
1803 init_completion(&xhci->addr_dev); 1803 init_completion(&xhci->addr_dev);
1804 for (i = 0; i < MAX_HC_SLOTS; ++i) 1804 for (i = 0; i < MAX_HC_SLOTS; ++i)
1805 xhci->devs[i] = NULL; 1805 xhci->devs[i] = NULL;
1806 for (i = 0; i < MAX_HC_PORTS; ++i)
1807 xhci->resume_done[i] = 0;
1806 1808
1807 if (scratchpad_alloc(xhci, flags)) 1809 if (scratchpad_alloc(xhci, flags))
1808 goto fail; 1810 goto fail;
diff --git a/drivers/usb/host/xhci-ring.c b/drivers/usb/host/xhci-ring.c
index b18e00ecb468..9f3115e729b1 100644
--- a/drivers/usb/host/xhci-ring.c
+++ b/drivers/usb/host/xhci-ring.c
@@ -1165,17 +1165,72 @@ static void handle_vendor_event(struct xhci_hcd *xhci,
1165static void handle_port_status(struct xhci_hcd *xhci, 1165static void handle_port_status(struct xhci_hcd *xhci,
1166 union xhci_trb *event) 1166 union xhci_trb *event)
1167{ 1167{
1168 struct usb_hcd *hcd = xhci_to_hcd(xhci);
1168 u32 port_id; 1169 u32 port_id;
1170 u32 temp, temp1;
1171 u32 __iomem *addr;
1172 int ports;
1173 int slot_id;
1169 1174
1170 /* Port status change events always have a successful completion code */ 1175 /* Port status change events always have a successful completion code */
1171 if (GET_COMP_CODE(event->generic.field[2]) != COMP_SUCCESS) { 1176 if (GET_COMP_CODE(event->generic.field[2]) != COMP_SUCCESS) {
1172 xhci_warn(xhci, "WARN: xHC returned failed port status event\n"); 1177 xhci_warn(xhci, "WARN: xHC returned failed port status event\n");
1173 xhci->error_bitmask |= 1 << 8; 1178 xhci->error_bitmask |= 1 << 8;
1174 } 1179 }
1175 /* FIXME: core doesn't care about all port link state changes yet */
1176 port_id = GET_PORT_ID(event->generic.field[0]); 1180 port_id = GET_PORT_ID(event->generic.field[0]);
1177 xhci_dbg(xhci, "Port Status Change Event for port %d\n", port_id); 1181 xhci_dbg(xhci, "Port Status Change Event for port %d\n", port_id);
1178 1182
1183 ports = HCS_MAX_PORTS(xhci->hcs_params1);
1184 if ((port_id <= 0) || (port_id > ports)) {
1185 xhci_warn(xhci, "Invalid port id %d\n", port_id);
1186 goto cleanup;
1187 }
1188
1189 addr = &xhci->op_regs->port_status_base + NUM_PORT_REGS * (port_id - 1);
1190 temp = xhci_readl(xhci, addr);
1191 if ((temp & PORT_CONNECT) && (hcd->state == HC_STATE_SUSPENDED)) {
1192 xhci_dbg(xhci, "resume root hub\n");
1193 usb_hcd_resume_root_hub(hcd);
1194 }
1195
1196 if ((temp & PORT_PLC) && (temp & PORT_PLS_MASK) == XDEV_RESUME) {
1197 xhci_dbg(xhci, "port resume event for port %d\n", port_id);
1198
1199 temp1 = xhci_readl(xhci, &xhci->op_regs->command);
1200 if (!(temp1 & CMD_RUN)) {
1201 xhci_warn(xhci, "xHC is not running.\n");
1202 goto cleanup;
1203 }
1204
1205 if (DEV_SUPERSPEED(temp)) {
1206 xhci_dbg(xhci, "resume SS port %d\n", port_id);
1207 temp = xhci_port_state_to_neutral(temp);
1208 temp &= ~PORT_PLS_MASK;
1209 temp |= PORT_LINK_STROBE | XDEV_U0;
1210 xhci_writel(xhci, temp, addr);
1211 slot_id = xhci_find_slot_id_by_port(xhci, port_id);
1212 if (!slot_id) {
1213 xhci_dbg(xhci, "slot_id is zero\n");
1214 goto cleanup;
1215 }
1216 xhci_ring_device(xhci, slot_id);
1217 xhci_dbg(xhci, "resume SS port %d finished\n", port_id);
1218 /* Clear PORT_PLC */
1219 temp = xhci_readl(xhci, addr);
1220 temp = xhci_port_state_to_neutral(temp);
1221 temp |= PORT_PLC;
1222 xhci_writel(xhci, temp, addr);
1223 } else {
1224 xhci_dbg(xhci, "resume HS port %d\n", port_id);
1225 xhci->resume_done[port_id - 1] = jiffies +
1226 msecs_to_jiffies(20);
1227 mod_timer(&hcd->rh_timer,
1228 xhci->resume_done[port_id - 1]);
1229 /* Do the rest in GetPortStatus */
1230 }
1231 }
1232
1233cleanup:
1179 /* Update event ring dequeue pointer before dropping the lock */ 1234 /* Update event ring dequeue pointer before dropping the lock */
1180 inc_deq(xhci, xhci->event_ring, true); 1235 inc_deq(xhci, xhci->event_ring, true);
1181 1236
diff --git a/drivers/usb/host/xhci.h b/drivers/usb/host/xhci.h
index 73e5db3e89c9..ca4a923dc810 100644
--- a/drivers/usb/host/xhci.h
+++ b/drivers/usb/host/xhci.h
@@ -1215,6 +1215,7 @@ struct xhci_hcd {
1215 u32 port_c_suspend[8]; /* port suspend change*/ 1215 u32 port_c_suspend[8]; /* port suspend change*/
1216 u32 suspended_ports[8]; /* which ports are 1216 u32 suspended_ports[8]; /* which ports are
1217 suspended */ 1217 suspended */
1218 unsigned long resume_done[MAX_HC_PORTS];
1218}; 1219};
1219 1220
1220/* For testing purposes */ 1221/* For testing purposes */
@@ -1459,6 +1460,9 @@ void xhci_ring_ep_doorbell(struct xhci_hcd *xhci, unsigned int slot_id,
1459int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, u16 wIndex, 1460int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, u16 wIndex,
1460 char *buf, u16 wLength); 1461 char *buf, u16 wLength);
1461int xhci_hub_status_data(struct usb_hcd *hcd, char *buf); 1462int xhci_hub_status_data(struct usb_hcd *hcd, char *buf);
1463u32 xhci_port_state_to_neutral(u32 state);
1464int xhci_find_slot_id_by_port(struct xhci_hcd *xhci, u16 port);
1465void xhci_ring_device(struct xhci_hcd *xhci, int slot_id);
1462 1466
1463/* xHCI contexts */ 1467/* xHCI contexts */
1464struct xhci_input_control_ctx *xhci_get_input_control_ctx(struct xhci_hcd *xhci, struct xhci_container_ctx *ctx); 1468struct xhci_input_control_ctx *xhci_get_input_control_ctx(struct xhci_hcd *xhci, struct xhci_container_ctx *ctx);