]> git.proxmox.com Git - mirror_edk2.git/blame - MdeModulePkg/Bus/Usb/UsbBusDxe/UsbHub.c
Check Library usage and fix some typo.
[mirror_edk2.git] / MdeModulePkg / Bus / Usb / UsbBusDxe / UsbHub.c
CommitLineData
e237e7ae 1/** @file\r
2\r
8616fc4c 3 Unified interface for RootHub and Hub.\r
4\r
e237e7ae 5Copyright (c) 2007, Intel Corporation\r
6All rights reserved. This program and the accompanying materials\r
7are licensed and made available under the terms and conditions of the BSD License\r
8which accompanies this distribution. The full text of the license may be found at\r
9http://opensource.org/licenses/bsd-license.php\r
10\r
11THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,\r
12WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.\r
13\r
e237e7ae 14**/\r
15\r
16#include "UsbBus.h"\r
17\r
aa79b0b3 18//\r
19// Array that maps the change bit to feature value which is\r
20// used to clear these change bit. USB HUB API will clear\r
21// these change bit automatically. For non-root hub, these\r
22// bits determine whether hub will report the port in changed\r
23// bit maps.\r
24//\r
25#define USB_HUB_MAP_SIZE 5\r
26\r
27USB_CHANGE_FEATURE_MAP mHubFeatureMap[USB_HUB_MAP_SIZE] = {\r
28 {USB_PORT_STAT_C_CONNECTION, EfiUsbPortConnectChange},\r
29 {USB_PORT_STAT_C_ENABLE, EfiUsbPortEnableChange},\r
30 {USB_PORT_STAT_C_SUSPEND, EfiUsbPortSuspendChange},\r
31 {USB_PORT_STAT_C_OVERCURRENT, EfiUsbPortOverCurrentChange},\r
32 {USB_PORT_STAT_C_RESET, EfiUsbPortResetChange},\r
33};\r
34\r
35#define USB_ROOT_HUB_MAP_SIZE 5\r
36\r
37USB_CHANGE_FEATURE_MAP mRootHubFeatureMap[USB_ROOT_HUB_MAP_SIZE] = {\r
38 {USB_PORT_STAT_C_CONNECTION, EfiUsbPortConnectChange},\r
39 {USB_PORT_STAT_C_ENABLE, EfiUsbPortEnableChange},\r
40 {USB_PORT_STAT_C_SUSPEND, EfiUsbPortSuspendChange},\r
41 {USB_PORT_STAT_C_OVERCURRENT, EfiUsbPortOverCurrentChange},\r
42 {USB_PORT_STAT_C_RESET, EfiUsbPortResetChange},\r
43};\r
44\r
e237e7ae 45//\r
46// USB hub class specific requests. Although USB hub\r
47// is related to an interface, these requests are sent\r
48// to the control endpoint of the device.\r
49//\r
50\r
51\r
52/**\r
8616fc4c 53 USB hub control transfer to clear the hub feature.\r
e237e7ae 54\r
8616fc4c 55 @param HubDev The device of the hub.\r
56 @param Feature The feature to clear.\r
e237e7ae 57\r
8616fc4c 58 @retval EFI_SUCCESS Feature of the hub is cleared.\r
59 @retval Others Failed to clear the feature.\r
e237e7ae 60\r
61**/\r
e237e7ae 62EFI_STATUS\r
63UsbHubCtrlClearHubFeature (\r
64 IN USB_DEVICE *HubDev,\r
65 IN UINT16 Feature\r
66 )\r
67{\r
68 EFI_STATUS Status;\r
69\r
70 Status = UsbCtrlRequest (\r
71 HubDev,\r
72 EfiUsbNoData,\r
73 USB_REQ_TYPE_CLASS,\r
74 USB_HUB_TARGET_HUB,\r
75 USB_HUB_REQ_CLEAR_FEATURE,\r
76 Feature,\r
77 0,\r
78 NULL,\r
79 0\r
80 );\r
81\r
82 return Status;\r
83}\r
84\r
85\r
86/**\r
8616fc4c 87 Clear the feature of the device's port.\r
e237e7ae 88\r
8616fc4c 89 @param HubDev The hub device.\r
90 @param Port The port to clear feature.\r
91 @param Feature The feature to clear.\r
e237e7ae 92\r
93 @retval EFI_SUCCESS The feature of the port is cleared.\r
94 @retval Others Failed to clear the feature.\r
95\r
96**/\r
e237e7ae 97EFI_STATUS\r
98UsbHubCtrlClearPortFeature (\r
99 IN USB_DEVICE *HubDev,\r
100 IN UINT8 Port,\r
101 IN UINT16 Feature\r
102 )\r
103{\r
104 EFI_STATUS Status;\r
105\r
106 //\r
107 // In USB bus, all the port index starts from 0. But HUB\r
108 // indexes its port from 1. So, port number is added one.\r
109 //\r
110 Status = UsbCtrlRequest (\r
111 HubDev,\r
112 EfiUsbNoData,\r
113 USB_REQ_TYPE_CLASS,\r
114 USB_HUB_TARGET_PORT,\r
115 USB_HUB_REQ_CLEAR_FEATURE,\r
116 Feature,\r
c52fa98c 117 (UINT16) (Port + 1),\r
e237e7ae 118 NULL,\r
119 0\r
120 );\r
121\r
122 return Status;\r
123}\r
124\r
125\r
e237e7ae 126/**\r
127 Clear the transaction translate buffer if full/low\r
128 speed control/bulk transfer failed and the transfer\r
129 uses this hub as translator.Remember to clear the TT\r
130 buffer of transaction translator, not that of the\r
131 parent.\r
132\r
8616fc4c 133 @param HubDev The hub device.\r
134 @param Port The port of the hub.\r
135 @param DevAddr Address of the failed transaction.\r
136 @param EpNum The endpoint number of the failed transaction.\r
137 @param EpType The type of failed transaction.\r
e237e7ae 138\r
8616fc4c 139 @retval EFI_SUCCESS The TT buffer is cleared.\r
140 @retval Others Failed to clear the TT buffer.\r
e237e7ae 141\r
142**/\r
143EFI_STATUS\r
144UsbHubCtrlClearTTBuffer (\r
145 IN USB_DEVICE *HubDev,\r
146 IN UINT8 Port,\r
147 IN UINT16 DevAddr,\r
148 IN UINT16 EpNum,\r
149 IN UINT16 EpType\r
150 )\r
151{\r
152 EFI_STATUS Status;\r
153 UINT16 Value;\r
154\r
155 //\r
156 // Check USB2.0 spec page 424 for wValue's encoding\r
157 //\r
c52fa98c 158 Value = (UINT16) ((EpNum & 0x0F) | (DevAddr << 4) |\r
159 ((EpType & 0x03) << 11) | ((EpNum & 0x80) << 15));\r
e237e7ae 160\r
161 Status = UsbCtrlRequest (\r
162 HubDev,\r
163 EfiUsbNoData,\r
164 USB_REQ_TYPE_CLASS,\r
165 USB_HUB_TARGET_PORT,\r
166 USB_HUB_REQ_CLEAR_TT,\r
167 Value,\r
c52fa98c 168 (UINT16) (Port + 1),\r
e237e7ae 169 NULL,\r
170 0\r
171 );\r
172\r
173 return Status;\r
174}\r
175\r
176\r
177/**\r
8616fc4c 178 Usb hub control transfer to get the hub descriptor.\r
e237e7ae 179\r
8616fc4c 180 @param HubDev The hub device.\r
181 @param Buf The buffer to hold the descriptor.\r
182 @param Len The length to retrieve.\r
e237e7ae 183\r
8616fc4c 184 @retval EFI_SUCCESS The hub descriptor is retrieved.\r
185 @retval Others Failed to retrieve the hub descriptor.\r
e237e7ae 186\r
187**/\r
e237e7ae 188EFI_STATUS\r
189UsbHubCtrlGetHubDesc (\r
190 IN USB_DEVICE *HubDev,\r
191 OUT VOID *Buf,\r
192 IN UINTN Len\r
193 )\r
194{\r
195 EFI_STATUS Status;\r
196\r
197 Status = UsbCtrlRequest (\r
198 HubDev,\r
199 EfiUsbDataIn,\r
200 USB_REQ_TYPE_CLASS,\r
201 USB_HUB_TARGET_HUB,\r
202 USB_HUB_REQ_GET_DESC,\r
203 (UINT16) (USB_DESC_TYPE_HUB << 8),\r
204 0,\r
205 Buf,\r
206 Len\r
207 );\r
208\r
209 return Status;\r
210}\r
211\r
212\r
213/**\r
8616fc4c 214 Usb hub control transfer to get the hub status.\r
e237e7ae 215\r
8616fc4c 216 @param HubDev The hub device.\r
217 @param State The variable to return the status.\r
e237e7ae 218\r
8616fc4c 219 @retval EFI_SUCCESS The hub status is returned in State.\r
220 @retval Others Failed to get the hub status.\r
e237e7ae 221\r
222**/\r
e237e7ae 223EFI_STATUS\r
224UsbHubCtrlGetHubStatus (\r
225 IN USB_DEVICE *HubDev,\r
226 OUT UINT32 *State\r
227 )\r
228{\r
229 EFI_STATUS Status;\r
230\r
231 Status = UsbCtrlRequest (\r
232 HubDev,\r
233 EfiUsbDataIn,\r
234 USB_REQ_TYPE_CLASS,\r
235 USB_HUB_TARGET_HUB,\r
236 USB_HUB_REQ_GET_STATUS,\r
237 0,\r
238 0,\r
239 State,\r
240 4\r
241 );\r
242\r
243 return Status;\r
244}\r
245\r
246\r
247/**\r
8616fc4c 248 Usb hub control transfer to get the port status.\r
e237e7ae 249\r
8616fc4c 250 @param HubDev The hub device.\r
251 @param Port The port of the hub.\r
252 @param State Variable to return the hub port state.\r
e237e7ae 253\r
8616fc4c 254 @retval EFI_SUCCESS The port state is returned in State.\r
d17371e8 255 @retval Others Failed to retrieve the port state.\r
e237e7ae 256\r
257**/\r
e237e7ae 258EFI_STATUS\r
259UsbHubCtrlGetPortStatus (\r
260 IN USB_DEVICE *HubDev,\r
261 IN UINT8 Port,\r
262 OUT VOID *State\r
263 )\r
264{\r
265 EFI_STATUS Status;\r
266\r
267 //\r
268 // In USB bus, all the port index starts from 0. But HUB\r
269 // indexes its port from 1. So, port number is added one.\r
270 // No need to convert the hub bit to UEFI definition, they\r
271 // are the same\r
272 //\r
273 Status = UsbCtrlRequest (\r
274 HubDev,\r
275 EfiUsbDataIn,\r
276 USB_REQ_TYPE_CLASS,\r
277 USB_HUB_TARGET_PORT,\r
278 USB_HUB_REQ_GET_STATUS,\r
279 0,\r
c52fa98c 280 (UINT16) (Port + 1),\r
e237e7ae 281 State,\r
282 4\r
283 );\r
284\r
285 return Status;\r
286}\r
287\r
288\r
289/**\r
8616fc4c 290 Usb hub control transfer to reset the TT (Transaction Transaltor).\r
e237e7ae 291\r
8616fc4c 292 @param HubDev The hub device.\r
293 @param Port The port of the hub.\r
e237e7ae 294\r
8616fc4c 295 @retval EFI_SUCCESS The TT of the hub is reset.\r
296 @retval Others Failed to reset the port.\r
e237e7ae 297\r
298**/\r
e237e7ae 299EFI_STATUS\r
300UsbHubCtrlResetTT (\r
301 IN USB_DEVICE *HubDev,\r
302 IN UINT8 Port\r
303 )\r
304{\r
305 EFI_STATUS Status;\r
306\r
307 Status = UsbCtrlRequest (\r
308 HubDev,\r
309 EfiUsbNoData,\r
310 USB_REQ_TYPE_CLASS,\r
311 USB_HUB_TARGET_HUB,\r
312 USB_HUB_REQ_RESET_TT,\r
313 0,\r
c52fa98c 314 (UINT16) (Port + 1),\r
e237e7ae 315 NULL,\r
316 0\r
317 );\r
318\r
319 return Status;\r
320}\r
321\r
322\r
323/**\r
8616fc4c 324 Usb hub control transfer to set the hub feature.\r
e237e7ae 325\r
8616fc4c 326 @param HubDev The hub device.\r
327 @param Feature The feature to set.\r
e237e7ae 328\r
8616fc4c 329 @retval EFI_SUCESS The feature is set for the hub.\r
330 @retval Others Failed to set the feature.\r
e237e7ae 331\r
332**/\r
e237e7ae 333EFI_STATUS\r
334UsbHubCtrlSetHubFeature (\r
335 IN USB_DEVICE *HubDev,\r
336 IN UINT8 Feature\r
337 )\r
338{\r
339 EFI_STATUS Status;\r
340\r
341 Status = UsbCtrlRequest (\r
342 HubDev,\r
343 EfiUsbNoData,\r
344 USB_REQ_TYPE_CLASS,\r
345 USB_HUB_TARGET_HUB,\r
346 USB_HUB_REQ_SET_FEATURE,\r
347 Feature,\r
348 0,\r
349 NULL,\r
350 0\r
351 );\r
352\r
353 return Status;\r
354}\r
355\r
356\r
357/**\r
8616fc4c 358 Usb hub control transfer to set the port feature.\r
e237e7ae 359\r
8616fc4c 360 @param HubDev The Usb hub device.\r
361 @param Port The Usb port to set feature for.\r
362 @param Feature The feature to set.\r
e237e7ae 363\r
8616fc4c 364 @retval EFI_SUCCESS The feature is set for the port.\r
365 @retval Others Failed to set the feature.\r
e237e7ae 366\r
367**/\r
e237e7ae 368EFI_STATUS\r
369UsbHubCtrlSetPortFeature (\r
370 IN USB_DEVICE *HubDev,\r
371 IN UINT8 Port,\r
372 IN UINT8 Feature\r
373 )\r
374{\r
375 EFI_STATUS Status;\r
376\r
377 //\r
378 // In USB bus, all the port index starts from 0. But HUB\r
379 // indexes its port from 1. So, port number is added one.\r
380 //\r
381 Status = UsbCtrlRequest (\r
382 HubDev,\r
383 EfiUsbNoData,\r
384 USB_REQ_TYPE_CLASS,\r
385 USB_HUB_TARGET_PORT,\r
386 USB_HUB_REQ_SET_FEATURE,\r
387 Feature,\r
c52fa98c 388 (UINT16) (Port + 1),\r
e237e7ae 389 NULL,\r
390 0\r
391 );\r
392\r
393 return Status;\r
394}\r
395\r
396\r
397/**\r
398 Read the whole usb hub descriptor. It is necessary\r
399 to do it in two steps because hub descriptor is of\r
8616fc4c 400 variable length.\r
e237e7ae 401\r
8616fc4c 402 @param HubDev The hub device.\r
403 @param HubDesc The variable to return the descriptor.\r
e237e7ae 404\r
8616fc4c 405 @retval EFI_SUCCESS The hub descriptor is read.\r
406 @retval Others Failed to read the hub descriptor.\r
e237e7ae 407\r
408**/\r
e237e7ae 409EFI_STATUS\r
410UsbHubReadDesc (\r
411 IN USB_DEVICE *HubDev,\r
412 OUT EFI_USB_HUB_DESCRIPTOR *HubDesc\r
413 )\r
414{\r
415 EFI_STATUS Status;\r
416\r
417 //\r
418 // First get the hub descriptor length\r
419 //\r
420 Status = UsbHubCtrlGetHubDesc (HubDev, HubDesc, 2);\r
421\r
422 if (EFI_ERROR (Status)) {\r
423 return Status;\r
424 }\r
425\r
426 //\r
427 // Get the whole hub descriptor\r
428 //\r
429 Status = UsbHubCtrlGetHubDesc (HubDev, HubDesc, HubDesc->Length);\r
430\r
431 return Status;\r
432}\r
433\r
434\r
435\r
436/**\r
437 Ack the hub change bits. If these bits are not ACKed, Hub will\r
438 always return changed bit map from its interrupt endpoint.\r
439\r
8616fc4c 440 @param HubDev The hub device.\r
e237e7ae 441\r
8616fc4c 442 @retval EFI_SUCCESS The hub change status is ACKed.\r
443 @retval Others Failed to ACK the hub status.\r
e237e7ae 444\r
445**/\r
446EFI_STATUS\r
447UsbHubAckHubStatus (\r
448 IN USB_DEVICE *HubDev\r
449 )\r
450{\r
451 EFI_USB_PORT_STATUS HubState;\r
452 EFI_STATUS Status;\r
453\r
454 Status = UsbHubCtrlGetHubStatus (HubDev, (UINT32 *) &HubState);\r
455\r
456 if (EFI_ERROR (Status)) {\r
457 return Status;\r
458 }\r
459\r
460 if (USB_BIT_IS_SET (HubState.PortChangeStatus, USB_HUB_STAT_C_LOCAL_POWER)) {\r
461 UsbHubCtrlClearHubFeature (HubDev, USB_HUB_C_HUB_LOCAL_POWER);\r
462 }\r
463\r
464 if (USB_BIT_IS_SET (HubState.PortChangeStatus, USB_HUB_STAT_C_OVER_CURRENT)) {\r
465 UsbHubCtrlClearHubFeature (HubDev, USB_HUB_C_HUB_OVER_CURRENT);\r
466 }\r
467\r
468 return EFI_SUCCESS;\r
469}\r
470\r
471\r
472/**\r
473 Test whether the interface is a hub interface.\r
474\r
8616fc4c 475 @param UsbIf The interface to test.\r
e237e7ae 476\r
8616fc4c 477 @retval TRUE The interface is a hub interface.\r
478 @retval FALSE The interface isn't a hub interface.\r
e237e7ae 479\r
480**/\r
481BOOLEAN\r
482UsbIsHubInterface (\r
483 IN USB_INTERFACE *UsbIf\r
484 )\r
485{\r
486 EFI_USB_INTERFACE_DESCRIPTOR *Setting;\r
487\r
488 //\r
489 // If the hub is a high-speed hub with multiple TT,\r
490 // the hub will has a default setting of single TT.\r
491 //\r
492 Setting = &UsbIf->IfSetting->Desc;\r
493\r
494 if ((Setting->InterfaceClass == USB_HUB_CLASS_CODE) &&\r
495 (Setting->InterfaceSubClass == USB_HUB_SUBCLASS_CODE)) {\r
496\r
497 return TRUE;\r
498 }\r
499\r
500 return FALSE;\r
501}\r
502\r
503\r
504/**\r
505 The callback function to the USB hub status change\r
506 interrupt endpoint. It is called periodically by\r
507 the underlying host controller.\r
508\r
8616fc4c 509 @param Data The data read.\r
510 @param DataLength The length of the data read.\r
511 @param Context The context.\r
512 @param Result The result of the last interrupt transfer.\r
e237e7ae 513\r
8616fc4c 514 @retval EFI_SUCCESS The process is OK.\r
515 @retval EFI_OUT_OF_RESOURCES Failed to allocate resource.\r
e237e7ae 516\r
517**/\r
e237e7ae 518EFI_STATUS\r
519UsbOnHubInterrupt (\r
520 IN VOID *Data,\r
521 IN UINTN DataLength,\r
522 IN VOID *Context,\r
523 IN UINT32 Result\r
524 )\r
525{\r
526 USB_INTERFACE *HubIf;\r
527 EFI_USB_IO_PROTOCOL *UsbIo;\r
528 EFI_USB_ENDPOINT_DESCRIPTOR *EpDesc;\r
529 EFI_STATUS Status;\r
530\r
531 HubIf = (USB_INTERFACE *) Context;\r
532 UsbIo = &(HubIf->UsbIo);\r
533 EpDesc = &(HubIf->HubEp->Desc);\r
534\r
535 if (Result != EFI_USB_NOERROR) {\r
536 //\r
537 // If endpoint is stalled, clear the stall. Use UsbIo to access\r
538 // the control transfer so internal status are maintained.\r
539 //\r
540 if (USB_BIT_IS_SET (Result, EFI_USB_ERR_STALL)) {\r
541 UsbIoClearFeature (\r
542 UsbIo,\r
543 USB_TARGET_ENDPOINT,\r
544 USB_FEATURE_ENDPOINT_HALT,\r
545 EpDesc->EndpointAddress\r
546 );\r
547 }\r
548\r
549 //\r
550 // Delete and submit a new async interrupt\r
551 //\r
552 Status = UsbIo->UsbAsyncInterruptTransfer (\r
553 UsbIo,\r
554 EpDesc->EndpointAddress,\r
555 FALSE,\r
556 0,\r
557 0,\r
558 NULL,\r
559 NULL\r
560 );\r
561\r
562 if (EFI_ERROR (Status)) {\r
d2577026 563 DEBUG (( EFI_D_ERROR, "UsbOnHubInterrupt: failed to remove async transfer - %r\n", Status));\r
e237e7ae 564 return Status;\r
565 }\r
566\r
567 Status = UsbIo->UsbAsyncInterruptTransfer (\r
568 UsbIo,\r
569 EpDesc->EndpointAddress,\r
570 TRUE,\r
571 USB_HUB_POLL_INTERVAL,\r
572 HubIf->NumOfPort / 8 + 1,\r
573 UsbOnHubInterrupt,\r
574 HubIf\r
575 );\r
576\r
577 if (EFI_ERROR (Status)) {\r
d2577026 578 DEBUG (( EFI_D_ERROR, "UsbOnHubInterrupt: failed to submit new async transfer - %r\n", Status));\r
e237e7ae 579 }\r
580\r
581 return Status;\r
582 }\r
583\r
584 if ((DataLength == 0) || (Data == NULL)) {\r
585 return EFI_SUCCESS;\r
586 }\r
587\r
588 //\r
589 // OK, actually something is changed, save the change map\r
590 // then signal the HUB to do enumeration. This is a good\r
591 // practise since UsbOnHubInterrupt is called in the context\r
592 // of host contrller's AsyncInterrupt monitor.\r
593 //\r
594 HubIf->ChangeMap = AllocateZeroPool (DataLength);\r
595\r
596 if (HubIf->ChangeMap == NULL) {\r
597 return EFI_OUT_OF_RESOURCES;\r
598 }\r
599\r
600 CopyMem (HubIf->ChangeMap, Data, DataLength);\r
601 gBS->SignalEvent (HubIf->HubNotify);\r
602\r
603 return EFI_SUCCESS;\r
604}\r
605\r
e237e7ae 606\r
607\r
608\r
609/**\r
8616fc4c 610 Initialize the device for a non-root hub.\r
e237e7ae 611\r
8616fc4c 612 @param HubIf The USB hub interface.\r
e237e7ae 613\r
8616fc4c 614 @retval EFI_SUCCESS The hub is initialized.\r
615 @retval EFI_DEVICE_ERROR Failed to initialize the hub.\r
e237e7ae 616\r
617**/\r
e237e7ae 618EFI_STATUS\r
619UsbHubInit (\r
620 IN USB_INTERFACE *HubIf\r
621 )\r
622{\r
623 EFI_USB_HUB_DESCRIPTOR HubDesc;\r
624 USB_ENDPOINT_DESC *EpDesc;\r
625 USB_INTERFACE_SETTING *Setting;\r
626 EFI_USB_IO_PROTOCOL *UsbIo;\r
627 USB_DEVICE *HubDev;\r
628 EFI_STATUS Status;\r
629 UINT8 Index;\r
d17371e8 630 UINT8 NumEndpoints;\r
e237e7ae 631\r
632 //\r
633 // Locate the interrupt endpoint for port change map\r
634 //\r
635 HubIf->IsHub = FALSE;\r
636 Setting = HubIf->IfSetting;\r
637 HubDev = HubIf->Device;\r
638 EpDesc = NULL;\r
d17371e8 639 NumEndpoints = Setting->Desc.NumEndpoints;\r
e237e7ae 640\r
d17371e8 641 for (Index = 0; Index < NumEndpoints; Index++) {\r
e237e7ae 642 ASSERT ((Setting->Endpoints != NULL) && (Setting->Endpoints[Index] != NULL));\r
643\r
644 EpDesc = Setting->Endpoints[Index];\r
645\r
646 if (USB_BIT_IS_SET (EpDesc->Desc.EndpointAddress, USB_ENDPOINT_DIR_IN) &&\r
647 (USB_ENDPOINT_TYPE (&EpDesc->Desc) == USB_ENDPOINT_INTERRUPT)) {\r
648 break;\r
649 }\r
650 }\r
651\r
d17371e8 652 if (Index == NumEndpoints) {\r
d2577026 653 DEBUG (( EFI_D_ERROR, "UsbHubInit: no interrupt endpoint found for hub %d\n", HubDev->Address));\r
e237e7ae 654 return EFI_DEVICE_ERROR;\r
655 }\r
656\r
657 Status = UsbHubReadDesc (HubDev, &HubDesc);\r
658\r
659 if (EFI_ERROR (Status)) {\r
d2577026 660 DEBUG (( EFI_D_ERROR, "UsbHubInit: failed to read HUB descriptor %r\n", Status));\r
e237e7ae 661 return Status;\r
662 }\r
663\r
664 HubIf->NumOfPort = HubDesc.NumPorts;\r
665\r
d2577026 666 DEBUG (( EFI_D_INFO, "UsbHubInit: hub %d has %d ports\n", HubDev->Address,HubIf->NumOfPort));\r
e237e7ae 667\r
668 //\r
669 // Create an event to enumerate the hub's port. On\r
670 //\r
671 Status = gBS->CreateEvent (\r
672 EVT_NOTIFY_SIGNAL,\r
673 TPL_CALLBACK,\r
674 UsbHubEnumeration,\r
675 HubIf,\r
676 &HubIf->HubNotify\r
677 );\r
678\r
679 if (EFI_ERROR (Status)) {\r
d2577026 680 DEBUG (( EFI_D_ERROR, "UsbHubInit: failed to create signal for hub %d - %r\n",\r
e237e7ae 681 HubDev->Address, Status));\r
682\r
683 return Status;\r
684 }\r
685\r
686 //\r
687 // Create AsyncInterrupt to query hub port change endpoint\r
688 // periodically. If the hub ports are changed, hub will return\r
689 // changed port map from the interrupt endpoint. The port map\r
690 // must be able to hold (HubIf->NumOfPort + 1) bits (one bit for\r
691 // host change status).\r
692 //\r
693 UsbIo = &HubIf->UsbIo;\r
694 Status = UsbIo->UsbAsyncInterruptTransfer (\r
695 UsbIo,\r
696 EpDesc->Desc.EndpointAddress,\r
697 TRUE,\r
698 USB_HUB_POLL_INTERVAL,\r
699 HubIf->NumOfPort / 8 + 1,\r
700 UsbOnHubInterrupt,\r
701 HubIf\r
702 );\r
703\r
704 if (EFI_ERROR (Status)) {\r
d2577026 705 DEBUG (( EFI_D_ERROR, "UsbHubInit: failed to queue interrupt transfer for hub %d - %r\n",\r
e237e7ae 706 HubDev->Address, Status));\r
707\r
708 gBS->CloseEvent (HubIf->HubNotify);\r
709 HubIf->HubNotify = NULL;\r
710\r
711 return Status;\r
712 }\r
713\r
714 //\r
715 // OK, set IsHub to TRUE. Now usb bus can handle this device\r
716 // as a working HUB. If failed eariler, bus driver will not\r
717 // recognize it as a hub. Other parts of the bus should be able\r
718 // to work.\r
719 //\r
720 HubIf->IsHub = TRUE;\r
721 HubIf->HubApi = &mUsbHubApi;\r
722 HubIf->HubEp = EpDesc;\r
723\r
724 //\r
725 // Feed power to all the hub ports. It should be ok\r
726 // for both gang/individual powered hubs.\r
727 //\r
728 for (Index = 0; Index < HubDesc.NumPorts; Index++) {\r
c52fa98c 729 UsbHubCtrlSetPortFeature (HubIf->Device, Index, (EFI_USB_PORT_FEATURE) USB_HUB_PORT_POWER);\r
e237e7ae 730 }\r
731\r
41e8ff27 732 gBS->Stall (HubDesc.PwrOn2PwrGood * USB_SET_PORT_POWER_STALL);\r
e237e7ae 733 UsbHubAckHubStatus (HubIf->Device);\r
734\r
d2577026 735 DEBUG (( EFI_D_INFO, "UsbHubInit: hub %d initialized\n", HubDev->Address));\r
e237e7ae 736 return Status;\r
737}\r
738\r
739\r
740\r
741/**\r
742 Get the port status. This function is required to\r
743 ACK the port change bits although it will return\r
744 the port changes in PortState. Bus enumeration code\r
745 doesn't need to ACK the port change bits.\r
746\r
8616fc4c 747 @param HubIf The hub interface.\r
748 @param Port The port of the hub to get state.\r
749 @param PortState Variable to return the port state.\r
e237e7ae 750\r
8616fc4c 751 @retval EFI_SUCCESS The port status is successfully returned.\r
752 @retval Others Failed to return the status.\r
e237e7ae 753\r
754**/\r
e237e7ae 755EFI_STATUS\r
756UsbHubGetPortStatus (\r
757 IN USB_INTERFACE *HubIf,\r
758 IN UINT8 Port,\r
759 OUT EFI_USB_PORT_STATUS *PortState\r
760 )\r
761{\r
762 EFI_STATUS Status;\r
763\r
764 Status = UsbHubCtrlGetPortStatus (HubIf->Device, Port, PortState);\r
765\r
766 return Status;\r
767}\r
768\r
769\r
770\r
771/**\r
772 Clear the port change status.\r
773\r
8616fc4c 774 @param HubIf The hub interface.\r
775 @param Port The hub port.\r
e237e7ae 776\r
8616fc4c 777 @return None.\r
e237e7ae 778\r
779**/\r
e237e7ae 780VOID\r
781UsbHubClearPortChange (\r
782 IN USB_INTERFACE *HubIf,\r
783 IN UINT8 Port\r
784 )\r
785{\r
786 EFI_USB_PORT_STATUS PortState;\r
787 USB_CHANGE_FEATURE_MAP *Map;\r
788 UINTN Index;\r
789 EFI_STATUS Status;\r
790\r
791 Status = UsbHubGetPortStatus (HubIf, Port, &PortState);\r
792\r
793 if (EFI_ERROR (Status)) {\r
794 return;\r
795 }\r
796\r
797 //\r
798 // OK, get the usb port status, now ACK the change bits.\r
799 // Don't return error when failed to clear the change bits.\r
800 // It may lead to extra port state report. USB bus should\r
801 // be able to handle this.\r
802 //\r
803 for (Index = 0; Index < USB_HUB_MAP_SIZE; Index++) {\r
804 Map = &mHubFeatureMap[Index];\r
805\r
806 if (USB_BIT_IS_SET (PortState.PortChangeStatus, Map->ChangedBit)) {\r
23c326c2 807 UsbHubCtrlClearPortFeature (HubIf->Device, Port, (UINT16) Map->Feature);\r
e237e7ae 808 }\r
809 }\r
810}\r
811\r
812\r
813\r
814/**\r
8616fc4c 815 Function to set the port feature for non-root hub.\r
e237e7ae 816\r
8616fc4c 817 @param HubIf The hub interface.\r
818 @param Port The port of the hub.\r
819 @param Feature The feature of the port to set.\r
e237e7ae 820\r
8616fc4c 821 @retval EFI_SUCCESS The hub port feature is set.\r
822 @retval Others Failed to set the port feature.\r
e237e7ae 823\r
824**/\r
e237e7ae 825EFI_STATUS\r
826UsbHubSetPortFeature (\r
827 IN USB_INTERFACE *HubIf,\r
828 IN UINT8 Port,\r
829 IN EFI_USB_PORT_FEATURE Feature\r
830 )\r
831{\r
832 EFI_STATUS Status;\r
833\r
c52fa98c 834 Status = UsbHubCtrlSetPortFeature (HubIf->Device, Port, (UINT8) Feature);\r
e237e7ae 835\r
836 return Status;\r
837}\r
838\r
839\r
840/**\r
8616fc4c 841 Interface function to clear the port feature for non-root hub.\r
e237e7ae 842\r
8616fc4c 843 @param HubIf The hub interface.\r
844 @param Port The port of the hub to clear feature for.\r
845 @param Feature The feature to clear.\r
e237e7ae 846\r
8616fc4c 847 @retval EFI_SUCCESS The port feature is cleared.\r
848 @retval Others Failed to clear the port feature.\r
e237e7ae 849\r
850**/\r
e237e7ae 851EFI_STATUS\r
852UsbHubClearPortFeature (\r
853 IN USB_INTERFACE *HubIf,\r
854 IN UINT8 Port,\r
855 IN EFI_USB_PORT_FEATURE Feature\r
856 )\r
857{\r
858 EFI_STATUS Status;\r
859\r
c52fa98c 860 Status = UsbHubCtrlClearPortFeature (HubIf->Device, Port, (UINT8) Feature);\r
e237e7ae 861\r
862 return Status;\r
863}\r
864\r
865\r
866/**\r
d17371e8 867 Interface function to reset the port.\r
e237e7ae 868\r
8616fc4c 869 @param HubIf The hub interface.\r
870 @param Port The port to reset.\r
e237e7ae 871\r
8616fc4c 872 @retval EFI_SUCCESS The hub port is reset.\r
873 @retval EFI_TIMEOUT Failed to reset the port in time.\r
874 @retval Others Failed to reset the port.\r
e237e7ae 875\r
876**/\r
e237e7ae 877EFI_STATUS\r
878UsbHubResetPort (\r
879 IN USB_INTERFACE *HubIf,\r
880 IN UINT8 Port\r
881 )\r
882{\r
883 EFI_USB_PORT_STATUS PortState;\r
884 UINTN Index;\r
885 EFI_STATUS Status;\r
886\r
c52fa98c 887 Status = UsbHubSetPortFeature (HubIf, Port, (EFI_USB_PORT_FEATURE) USB_HUB_PORT_RESET);\r
e237e7ae 888\r
889 if (EFI_ERROR (Status)) {\r
890 return Status;\r
891 }\r
892\r
893 //\r
894 // Drive the reset signal for at least 10ms. Check USB 2.0 Spec\r
895 // section 7.1.7.5 for timing requirements.\r
896 //\r
41e8ff27 897 gBS->Stall (USB_SET_PORT_RESET_STALL);\r
e237e7ae 898\r
899 //\r
900 // USB hub will clear RESET bit if reset is actually finished.\r
901 //\r
902 ZeroMem (&PortState, sizeof (EFI_USB_PORT_STATUS));\r
903\r
41e8ff27 904 for (Index = 0; Index < USB_WAIT_PORT_STS_CHANGE_LOOP; Index++) {\r
e237e7ae 905 Status = UsbHubGetPortStatus (HubIf, Port, &PortState);\r
906\r
907 if (!EFI_ERROR (Status) &&\r
908 !USB_BIT_IS_SET (PortState.PortStatus, USB_PORT_STAT_RESET)) {\r
909\r
910 return EFI_SUCCESS;\r
911 }\r
912\r
41e8ff27 913 gBS->Stall (USB_WAIT_PORT_STS_CHANGE_STALL);\r
e237e7ae 914 }\r
915\r
916 return EFI_TIMEOUT;\r
917}\r
918\r
919\r
920/**\r
8616fc4c 921 Release the hub's control of the interface.\r
e237e7ae 922\r
8616fc4c 923 @param HubIf The hub interface.\r
e237e7ae 924\r
8616fc4c 925 @retval EFI_SUCCESS The interface is release of hub control.\r
e237e7ae 926\r
927**/\r
e237e7ae 928EFI_STATUS\r
929UsbHubRelease (\r
930 IN USB_INTERFACE *HubIf\r
931 )\r
932{\r
933 EFI_USB_IO_PROTOCOL *UsbIo;\r
934 EFI_STATUS Status;\r
935\r
936 UsbIo = &HubIf->UsbIo;\r
937 Status = UsbIo->UsbAsyncInterruptTransfer (\r
938 UsbIo,\r
939 HubIf->HubEp->Desc.EndpointAddress,\r
940 FALSE,\r
941 USB_HUB_POLL_INTERVAL,\r
942 0,\r
943 NULL,\r
944 0\r
945 );\r
946\r
947 if (EFI_ERROR (Status)) {\r
948 return Status;\r
949 }\r
950\r
951 gBS->CloseEvent (HubIf->HubNotify);\r
952\r
953 HubIf->IsHub = FALSE;\r
954 HubIf->HubApi = NULL;\r
955 HubIf->HubEp = NULL;\r
956 HubIf->HubNotify = NULL;\r
957\r
d2577026 958 DEBUG (( EFI_D_INFO, "UsbHubRelease: hub device %d released\n", HubIf->Device->Address));\r
e237e7ae 959 return EFI_SUCCESS;\r
960}\r
961\r
962\r
963\r
964/**\r
8616fc4c 965 Initialize the interface for root hub.\r
e237e7ae 966\r
8616fc4c 967 @param HubIf The root hub interface.\r
e237e7ae 968\r
d17371e8 969 @retval EFI_SUCCESS The interface is initialized for root hub.\r
8616fc4c 970 @retval Others Failed to initialize the hub.\r
e237e7ae 971\r
972**/\r
e237e7ae 973EFI_STATUS\r
974UsbRootHubInit (\r
975 IN USB_INTERFACE *HubIf\r
976 )\r
977{\r
978 EFI_STATUS Status;\r
979 UINT8 MaxSpeed;\r
980 UINT8 NumOfPort;\r
981 UINT8 Support64;\r
982\r
983 Status = UsbHcGetCapability (HubIf->Device->Bus, &MaxSpeed, &NumOfPort, &Support64);\r
984\r
985 if (EFI_ERROR (Status)) {\r
986 return Status;\r
987 }\r
988\r
0e549d5b 989 DEBUG (( EFI_D_INFO, "UsbRootHubInit: root hub %p - max speed %d, %d ports\n",\r
e237e7ae 990 HubIf, MaxSpeed, NumOfPort));\r
991\r
992 HubIf->IsHub = TRUE;\r
993 HubIf->HubApi = &mUsbRootHubApi;\r
994 HubIf->HubEp = NULL;\r
995 HubIf->MaxSpeed = MaxSpeed;\r
996 HubIf->NumOfPort = NumOfPort;\r
997 HubIf->HubNotify = NULL;\r
998\r
999 //\r
1000 // Create a timer to poll root hub ports periodically\r
1001 //\r
1002 Status = gBS->CreateEvent (\r
1003 EVT_TIMER | EVT_NOTIFY_SIGNAL,\r
1004 TPL_CALLBACK,\r
1005 UsbRootHubEnumeration,\r
1006 HubIf,\r
1007 &HubIf->HubNotify\r
1008 );\r
1009\r
1010 if (EFI_ERROR (Status)) {\r
1011 return Status;\r
1012 }\r
1013\r
d2577026 1014 //\r
1015 // It should signal the event immediately here, or device detection\r
1016 // by bus enumeration might be delayed by the timer interval.\r
1017 //\r
1018 gBS->SignalEvent (HubIf->HubNotify);\r
1019\r
e237e7ae 1020 Status = gBS->SetTimer (\r
1021 HubIf->HubNotify,\r
1022 TimerPeriodic,\r
1023 USB_ROOTHUB_POLL_INTERVAL\r
1024 );\r
1025\r
1026 if (EFI_ERROR (Status)) {\r
1027 gBS->CloseEvent (HubIf->HubNotify);\r
1028 }\r
1029\r
1030 return Status;\r
1031}\r
1032\r
1033\r
e237e7ae 1034/**\r
1035 Get the port status. This function is required to\r
1036 ACK the port change bits although it will return\r
1037 the port changes in PortState. Bus enumeration code\r
1038 doesn't need to ACK the port change bits.\r
1039\r
8616fc4c 1040 @param HubIf The root hub interface.\r
1041 @param Port The root hub port to get the state.\r
1042 @param PortState Variable to return the port state.\r
e237e7ae 1043\r
8616fc4c 1044 @retval EFI_SUCCESS The port state is returned.\r
1045 @retval Others Failed to retrieve the port state.\r
e237e7ae 1046\r
1047**/\r
e237e7ae 1048EFI_STATUS\r
1049UsbRootHubGetPortStatus (\r
1050 IN USB_INTERFACE *HubIf,\r
1051 IN UINT8 Port,\r
1052 OUT EFI_USB_PORT_STATUS *PortState\r
1053 )\r
1054{\r
1055 USB_BUS *Bus;\r
1056 EFI_STATUS Status;\r
1057\r
1058 Bus = HubIf->Device->Bus;\r
1059 Status = UsbHcGetRootHubPortStatus (Bus, Port, PortState);\r
1060\r
1061 return Status;\r
1062}\r
1063\r
1064\r
1065/**\r
1066 Clear the port change status.\r
1067\r
8616fc4c 1068 @param HubIf The root hub interface.\r
1069 @param Port The root hub port.\r
e237e7ae 1070\r
eb6d5108 1071 @return None.\r
e237e7ae 1072\r
1073**/\r
e237e7ae 1074VOID\r
1075UsbRootHubClearPortChange (\r
1076 IN USB_INTERFACE *HubIf,\r
1077 IN UINT8 Port\r
1078 )\r
1079{\r
1080 EFI_USB_PORT_STATUS PortState;\r
1081 USB_CHANGE_FEATURE_MAP *Map;\r
1082 UINTN Index;\r
1083 EFI_STATUS Status;\r
1084\r
1085 Status = UsbRootHubGetPortStatus (HubIf, Port, &PortState);\r
1086\r
1087 if (EFI_ERROR (Status)) {\r
1088 return;\r
1089 }\r
1090\r
1091 //\r
1092 // OK, get the usb port status, now ACK the change bits.\r
1093 // Don't return error when failed to clear the change bits.\r
1094 // It may lead to extra port state report. USB bus should\r
1095 // be able to handle this.\r
1096 //\r
1097 for (Index = 0; Index < USB_ROOT_HUB_MAP_SIZE; Index++) {\r
1098 Map = &mRootHubFeatureMap[Index];\r
1099\r
1100 if (USB_BIT_IS_SET (PortState.PortChangeStatus, Map->ChangedBit)) {\r
c52fa98c 1101 UsbHcClearRootHubPortFeature (HubIf->Device->Bus, Port, (EFI_USB_PORT_FEATURE) Map->Feature);\r
e237e7ae 1102 }\r
1103 }\r
1104}\r
1105\r
1106\r
e237e7ae 1107/**\r
8616fc4c 1108 Set the root hub port feature.\r
e237e7ae 1109\r
8616fc4c 1110 @param HubIf The Usb hub interface.\r
1111 @param Port The hub port.\r
1112 @param Feature The feature to set.\r
e237e7ae 1113\r
8616fc4c 1114 @retval EFI_SUCCESS The root hub port is set with the feature.\r
1115 @retval Others Failed to set the feature.\r
e237e7ae 1116\r
1117**/\r
e237e7ae 1118EFI_STATUS\r
1119UsbRootHubSetPortFeature (\r
1120 IN USB_INTERFACE *HubIf,\r
1121 IN UINT8 Port,\r
1122 IN EFI_USB_PORT_FEATURE Feature\r
1123 )\r
1124{\r
1125 EFI_STATUS Status;\r
1126\r
1127 Status = UsbHcSetRootHubPortFeature (HubIf->Device->Bus, Port, Feature);\r
1128\r
1129 return Status;\r
1130}\r
1131\r
1132\r
1133/**\r
8616fc4c 1134 Clear the root hub port feature.\r
e237e7ae 1135\r
8616fc4c 1136 @param HubIf The root hub interface.\r
1137 @param Port The root hub port.\r
1138 @param Feature The feature to clear.\r
e237e7ae 1139\r
8616fc4c 1140 @retval EFI_SUCCESS The root hub port is cleared of the feature.\r
1141 @retval Others Failed to clear the feature.\r
e237e7ae 1142\r
1143**/\r
e237e7ae 1144EFI_STATUS\r
1145UsbRootHubClearPortFeature (\r
1146 IN USB_INTERFACE *HubIf,\r
1147 IN UINT8 Port,\r
1148 IN EFI_USB_PORT_FEATURE Feature\r
1149 )\r
1150{\r
1151 EFI_STATUS Status;\r
1152\r
1153 Status = UsbHcClearRootHubPortFeature (HubIf->Device->Bus, Port, Feature);\r
1154\r
1155 return Status;\r
1156}\r
1157\r
1158\r
1159/**\r
d17371e8 1160 Interface function to reset the root hub port.\r
e237e7ae 1161\r
8616fc4c 1162 @param RootIf The root hub interface.\r
1163 @param Port The port to reset.\r
e237e7ae 1164\r
8616fc4c 1165 @retval EFI_SUCCESS The hub port is reset.\r
1166 @retval EFI_TIMEOUT Failed to reset the port in time.\r
1167 @retval EFI_NOT_FOUND The low/full speed device connected to high speed.\r
1168 root hub is released to the companion UHCI.\r
1169 @retval Others Failed to reset the port.\r
e237e7ae 1170\r
1171**/\r
e237e7ae 1172EFI_STATUS\r
1173UsbRootHubResetPort (\r
1174 IN USB_INTERFACE *RootIf,\r
1175 IN UINT8 Port\r
1176 )\r
1177{\r
1178 USB_BUS *Bus;\r
1179 EFI_STATUS Status;\r
1180 EFI_USB_PORT_STATUS PortState;\r
1181 UINTN Index;\r
1182\r
1183 //\r
1184 // Notice: although EHCI requires that ENABLED bit be cleared\r
1185 // when reset the port, we don't need to care that here. It\r
1186 // should be handled in the EHCI driver.\r
1187 //\r
1188 Bus = RootIf->Device->Bus;\r
1189 Status = UsbHcSetRootHubPortFeature (Bus, Port, EfiUsbPortReset);\r
1190\r
1191 if (EFI_ERROR (Status)) {\r
d2577026 1192 DEBUG (( EFI_D_ERROR, "UsbRootHubResetPort: failed to start reset on port %d\n", Port));\r
e237e7ae 1193 return Status;\r
1194 }\r
1195\r
1196 //\r
1197 // Drive the reset signal for at least 50ms. Check USB 2.0 Spec\r
1198 // section 7.1.7.5 for timing requirements.\r
1199 //\r
41e8ff27 1200 gBS->Stall (USB_SET_ROOT_PORT_RESET_STALL);\r
e237e7ae 1201\r
1202 Status = UsbHcClearRootHubPortFeature (Bus, Port, EfiUsbPortReset);\r
1203\r
1204 if (EFI_ERROR (Status)) {\r
d2577026 1205 DEBUG (( EFI_D_ERROR, "UsbRootHubResetPort: failed to clear reset on port %d\n", Port));\r
e237e7ae 1206 return Status;\r
1207 }\r
1208\r
41e8ff27 1209 gBS->Stall (USB_CLR_ROOT_PORT_RESET_STALL);\r
e237e7ae 1210\r
1211 //\r
1212 // USB host controller won't clear the RESET bit until\r
1213 // reset is actually finished.\r
1214 //\r
1215 ZeroMem (&PortState, sizeof (EFI_USB_PORT_STATUS));\r
1216\r
41e8ff27 1217 for (Index = 0; Index < USB_WAIT_PORT_STS_CHANGE_LOOP; Index++) {\r
e237e7ae 1218 Status = UsbHcGetRootHubPortStatus (Bus, Port, &PortState);\r
1219\r
1220 if (EFI_ERROR (Status)) {\r
1221 return Status;\r
1222 }\r
1223\r
1224 if (!USB_BIT_IS_SET (PortState.PortStatus, USB_PORT_STAT_RESET)) {\r
1225 break;\r
1226 }\r
1227\r
41e8ff27 1228 gBS->Stall (USB_WAIT_PORT_STS_CHANGE_STALL);\r
e237e7ae 1229 }\r
1230\r
41e8ff27 1231 if (Index == USB_WAIT_PORT_STS_CHANGE_LOOP) {\r
1232 DEBUG ((EFI_D_ERROR, "UsbRootHubResetPort: reset not finished in time on port %d\n", Port));\r
e237e7ae 1233 return EFI_TIMEOUT;\r
1234 }\r
1235\r
1236 if (!USB_BIT_IS_SET (PortState.PortStatus, USB_PORT_STAT_ENABLE)) {\r
1237 //\r
1238 // OK, the port is reset. If root hub is of high speed and\r
1239 // the device is of low/full speed, release the ownership to\r
1240 // companion UHCI. If root hub is of full speed, it won't\r
1241 // automatically enable the port, we need to enable it manually.\r
1242 //\r
1243 if (RootIf->MaxSpeed == EFI_USB_SPEED_HIGH) {\r
d2577026 1244 DEBUG (( EFI_D_ERROR, "UsbRootHubResetPort: release low/full speed device (%d) to UHCI\n", Port));\r
e237e7ae 1245\r
1246 UsbRootHubSetPortFeature (RootIf, Port, EfiUsbPortOwner);\r
1247 return EFI_NOT_FOUND;\r
1248\r
1249 } else {\r
1250\r
1251 Status = UsbRootHubSetPortFeature (RootIf, Port, EfiUsbPortEnable);\r
1252\r
1253 if (EFI_ERROR (Status)) {\r
d2577026 1254 DEBUG (( EFI_D_ERROR, "UsbRootHubResetPort: failed to enable port %d for UHCI\n", Port));\r
e237e7ae 1255 return Status;\r
1256 }\r
1257\r
41e8ff27 1258 gBS->Stall (USB_SET_ROOT_PORT_ENABLE_STALL);\r
e237e7ae 1259 }\r
1260 }\r
1261\r
1262 return EFI_SUCCESS;\r
1263}\r
1264\r
1265\r
1266/**\r
8616fc4c 1267 Release the root hub's control of the interface.\r
e237e7ae 1268\r
8616fc4c 1269 @param HubIf The root hub interface.\r
e237e7ae 1270\r
1271 @retval EFI_SUCCESS The root hub's control of the interface is\r
1272 released.\r
1273\r
1274**/\r
e237e7ae 1275EFI_STATUS\r
1276UsbRootHubRelease (\r
1277 IN USB_INTERFACE *HubIf\r
1278 )\r
1279{\r
0e549d5b 1280 DEBUG (( EFI_D_INFO, "UsbRootHubRelease: root hub released for hub %p\n", HubIf));\r
e237e7ae 1281\r
1282 gBS->SetTimer (HubIf->HubNotify, TimerCancel, USB_ROOTHUB_POLL_INTERVAL);\r
1283 gBS->CloseEvent (HubIf->HubNotify);\r
1284\r
1285 return EFI_SUCCESS;\r
1286}\r
1287\r
1288USB_HUB_API mUsbHubApi = {\r
1289 UsbHubInit,\r
1290 UsbHubGetPortStatus,\r
1291 UsbHubClearPortChange,\r
1292 UsbHubSetPortFeature,\r
1293 UsbHubClearPortFeature,\r
1294 UsbHubResetPort,\r
1295 UsbHubRelease\r
1296};\r
1297\r
1298USB_HUB_API mUsbRootHubApi = {\r
1299 UsbRootHubInit,\r
1300 UsbRootHubGetPortStatus,\r
1301 UsbRootHubClearPortChange,\r
1302 UsbRootHubSetPortFeature,\r
1303 UsbRootHubClearPortFeature,\r
1304 UsbRootHubResetPort,\r
1305 UsbRootHubRelease\r
1306};\r