mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	xhci: Rework port suspend structures for limited ports.
The USB core only allows up to 31 (USB_MAXCHILDREN) ports under a roothub. The xHCI driver keeps track of which ports are suspended, which ports have a suspend change bit set, and what time the port will be done resuming. It keeps track of the first two by setting a bit in a u32 variable, suspended_ports or port_c_suspend. The xHCI driver currently assumes we can have up to 256 ports under a roothub, so it allocates an array of 8 u32 variables for both suspended_ports and port_c_suspend. It also allocates a 256-element array to keep track of when the ports will be done resuming. Since we can only have 31 roothub ports, we only need to use one u32 for each of the suspend state and change variables. We simplify the bit math that's trying to index into those arrays and set the correct bit, if we assume wIndex never exceeds 30. (wIndex is zero-based after it's decremented from the value passed in from the USB core.) Finally, we change the resume_done array to only hold 31 elements. Signed-off-by: Sarah Sharp <sarah.a.sharp@linux.intel.com> Cc: Andiry Xu <andiry.xu@amd.com>
This commit is contained in:
		
							parent
							
								
									abc4f9b099
								
							
						
					
					
						commit
						1d5810b692
					
				| @ -347,20 +347,15 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, | |||||||
| 					goto error; | 					goto error; | ||||||
| 				} | 				} | ||||||
| 				xhci_ring_device(xhci, slot_id); | 				xhci_ring_device(xhci, slot_id); | ||||||
| 				xhci->port_c_suspend[wIndex >> 5] |= | 				xhci->port_c_suspend |= 1 << wIndex; | ||||||
| 						1 << (wIndex & 31); | 				xhci->suspended_ports &= ~(1 << wIndex); | ||||||
| 				xhci->suspended_ports[wIndex >> 5] &= |  | ||||||
| 						~(1 << (wIndex & 31)); |  | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		if ((temp & PORT_PLS_MASK) == XDEV_U0 | 		if ((temp & PORT_PLS_MASK) == XDEV_U0 | ||||||
| 			&& (temp & PORT_POWER) | 			&& (temp & PORT_POWER) | ||||||
| 			&& (xhci->suspended_ports[wIndex >> 5] & | 			&& (xhci->suspended_ports & (1 << wIndex))) { | ||||||
| 			    (1 << (wIndex & 31)))) { | 			xhci->suspended_ports &= ~(1 << wIndex); | ||||||
| 			xhci->suspended_ports[wIndex >> 5] &= | 			xhci->port_c_suspend |= 1 << wIndex; | ||||||
| 					~(1 << (wIndex & 31)); |  | ||||||
| 			xhci->port_c_suspend[wIndex >> 5] |= |  | ||||||
| 					1 << (wIndex & 31); |  | ||||||
| 		} | 		} | ||||||
| 		if (temp & PORT_CONNECT) { | 		if (temp & PORT_CONNECT) { | ||||||
| 			status |= USB_PORT_STAT_CONNECTION; | 			status |= USB_PORT_STAT_CONNECTION; | ||||||
| @ -374,7 +369,7 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, | |||||||
| 			status |= USB_PORT_STAT_RESET; | 			status |= USB_PORT_STAT_RESET; | ||||||
| 		if (temp & PORT_POWER) | 		if (temp & PORT_POWER) | ||||||
| 			status |= USB_PORT_STAT_POWER; | 			status |= USB_PORT_STAT_POWER; | ||||||
| 		if (xhci->port_c_suspend[wIndex >> 5] & (1 << (wIndex & 31))) | 		if (xhci->port_c_suspend & (1 << wIndex)) | ||||||
| 			status |= 1 << USB_PORT_FEAT_C_SUSPEND; | 			status |= 1 << USB_PORT_FEAT_C_SUSPEND; | ||||||
| 		xhci_dbg(xhci, "Get port status returned 0x%x\n", status); | 		xhci_dbg(xhci, "Get port status returned 0x%x\n", status); | ||||||
| 		put_unaligned(cpu_to_le32(status), (__le32 *) buf); | 		put_unaligned(cpu_to_le32(status), (__le32 *) buf); | ||||||
| @ -421,8 +416,7 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, | |||||||
| 			spin_lock_irqsave(&xhci->lock, flags); | 			spin_lock_irqsave(&xhci->lock, flags); | ||||||
| 
 | 
 | ||||||
| 			temp = xhci_readl(xhci, addr); | 			temp = xhci_readl(xhci, addr); | ||||||
| 			xhci->suspended_ports[wIndex >> 5] |= | 			xhci->suspended_ports |= 1 << wIndex; | ||||||
| 					1 << (wIndex & (31)); |  | ||||||
| 			break; | 			break; | ||||||
| 		case USB_PORT_FEAT_POWER: | 		case USB_PORT_FEAT_POWER: | ||||||
| 			/*
 | 			/*
 | ||||||
| @ -489,8 +483,7 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, | |||||||
| 					temp |= PORT_LINK_STROBE | XDEV_U0; | 					temp |= PORT_LINK_STROBE | XDEV_U0; | ||||||
| 					xhci_writel(xhci, temp, addr); | 					xhci_writel(xhci, temp, addr); | ||||||
| 				} | 				} | ||||||
| 				xhci->port_c_suspend[wIndex >> 5] |= | 				xhci->port_c_suspend |= 1 << wIndex; | ||||||
| 						1 << (wIndex & 31); |  | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			slot_id = xhci_find_slot_id_by_port(xhci, wIndex + 1); | 			slot_id = xhci_find_slot_id_by_port(xhci, wIndex + 1); | ||||||
| @ -501,8 +494,7 @@ int xhci_hub_control(struct usb_hcd *hcd, u16 typeReq, u16 wValue, | |||||||
| 			xhci_ring_device(xhci, slot_id); | 			xhci_ring_device(xhci, slot_id); | ||||||
| 			break; | 			break; | ||||||
| 		case USB_PORT_FEAT_C_SUSPEND: | 		case USB_PORT_FEAT_C_SUSPEND: | ||||||
| 			xhci->port_c_suspend[wIndex >> 5] &= | 			xhci->port_c_suspend &= ~(1 << wIndex); | ||||||
| 					~(1 << (wIndex & 31)); |  | ||||||
| 		case USB_PORT_FEAT_C_RESET: | 		case USB_PORT_FEAT_C_RESET: | ||||||
| 		case USB_PORT_FEAT_C_CONNECTION: | 		case USB_PORT_FEAT_C_CONNECTION: | ||||||
| 		case USB_PORT_FEAT_C_OVER_CURRENT: | 		case USB_PORT_FEAT_C_OVER_CURRENT: | ||||||
| @ -560,7 +552,7 @@ int xhci_hub_status_data(struct usb_hcd *hcd, char *buf) | |||||||
| 			NUM_PORT_REGS*i; | 			NUM_PORT_REGS*i; | ||||||
| 		temp = xhci_readl(xhci, addr); | 		temp = xhci_readl(xhci, addr); | ||||||
| 		if ((temp & mask) != 0 || | 		if ((temp & mask) != 0 || | ||||||
| 			(xhci->port_c_suspend[i >> 5] &	1 << (i & 31)) || | 			(xhci->port_c_suspend & 1 << i) || | ||||||
| 			(xhci->resume_done[i] && time_after_eq( | 			(xhci->resume_done[i] && time_after_eq( | ||||||
| 			    jiffies, xhci->resume_done[i]))) { | 			    jiffies, xhci->resume_done[i]))) { | ||||||
| 			buf[(i + 1) / 8] |= 1 << (i + 1) % 8; | 			buf[(i + 1) / 8] |= 1 << (i + 1) % 8; | ||||||
|  | |||||||
| @ -1971,7 +1971,7 @@ int xhci_mem_init(struct xhci_hcd *xhci, gfp_t flags) | |||||||
| 	init_completion(&xhci->addr_dev); | 	init_completion(&xhci->addr_dev); | ||||||
| 	for (i = 0; i < MAX_HC_SLOTS; ++i) | 	for (i = 0; i < MAX_HC_SLOTS; ++i) | ||||||
| 		xhci->devs[i] = NULL; | 		xhci->devs[i] = NULL; | ||||||
| 	for (i = 0; i < MAX_HC_PORTS; ++i) | 	for (i = 0; i < USB_MAXCHILDREN; ++i) | ||||||
| 		xhci->resume_done[i] = 0; | 		xhci->resume_done[i] = 0; | ||||||
| 
 | 
 | ||||||
| 	if (scratchpad_alloc(xhci, flags)) | 	if (scratchpad_alloc(xhci, flags)) | ||||||
|  | |||||||
| @ -1248,10 +1248,11 @@ struct xhci_hcd { | |||||||
| #define	XHCI_LINK_TRB_QUIRK	(1 << 0) | #define	XHCI_LINK_TRB_QUIRK	(1 << 0) | ||||||
| #define XHCI_RESET_EP_QUIRK	(1 << 1) | #define XHCI_RESET_EP_QUIRK	(1 << 1) | ||||||
| #define XHCI_NEC_HOST		(1 << 2) | #define XHCI_NEC_HOST		(1 << 2) | ||||||
| 	u32			port_c_suspend[8];	/* port suspend change*/ | 	/* port suspend change*/ | ||||||
| 	u32			suspended_ports[8];	/* which ports are
 | 	u32			port_c_suspend; | ||||||
| 							   suspended */ | 	/* which ports are suspended */ | ||||||
| 	unsigned long		resume_done[MAX_HC_PORTS]; | 	u32			suspended_ports; | ||||||
|  | 	unsigned long		resume_done[USB_MAXCHILDREN]; | ||||||
| 	/* Is each xHCI roothub port a USB 3.0, USB 2.0, or USB 1.1 port? */ | 	/* Is each xHCI roothub port a USB 3.0, USB 2.0, or USB 1.1 port? */ | ||||||
| 	u8			*port_array; | 	u8			*port_array; | ||||||
| 	/* Array of pointers to USB 3.0 PORTSC registers */ | 	/* Array of pointers to USB 3.0 PORTSC registers */ | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Sarah Sharp
						Sarah Sharp