There are cases that the operands of an expression are all with rank less
than UINT64/INT64 and the result of the expression is explicitly cast to
UINT64/INT64 to fit the target size.
An example will be:
UINT32 a,b;
// a and b can be any unsigned int type with rank less than UINT64, like
// UINT8, UINT16, etc.
UINT64 c;
c = (UINT64) (a + b);
Some static code checkers may warn that the expression result might
overflow within the rank of "int" (integer promotions) and the result is
then cast to a bigger size.
The commit refines codes by the following rules:
1). When the expression is possible to overflow the range of unsigned int/
int:
c = (UINT64)a + b;
2). When the expression will not overflow within the rank of "int", remove
the explicit type casts:
c = a + b;
3). When the expression will be cast to pointer of possible greater size:
UINT32 a,b;
VOID *c;
c = (VOID *)(UINTN)(a + b); --> c = (VOID *)((UINTN)a + b);
4). When one side of a comparison expression contains only operands with
rank less than UINT32:
UINT8 a;
UINT16 b;
UINTN c;
if ((UINTN)(a + b) > c) {...} --> if (((UINT32)a + b) > c) {...}
For rule 4), if we remove the 'UINTN' type cast like:
if (a + b > c) {...}
The VS compiler will complain with warning C4018 (signed/unsigned
mismatch, level 3 warning) due to promoting 'a + b' to type 'int'.
Contributed-under: TianoCore Contribution Agreement 1.0
Signed-off-by: Hao Wu <hao.a.wu@intel.com>
Reviewed-by: Feng Tian <feng.tian@intel.com>
		
	
		
			
				
	
	
		
			750 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			750 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
| 
 | |
|   The XHCI register operation routines.
 | |
| 
 | |
| Copyright (c) 2011 - 2017, Intel Corporation. All rights reserved.<BR>
 | |
| This program and the accompanying materials
 | |
| are licensed and made available under the terms and conditions of the BSD License
 | |
| which accompanies this distribution.  The full text of the license may be found at
 | |
| http://opensource.org/licenses/bsd-license.php
 | |
| 
 | |
| THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
 | |
| WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
 | |
| 
 | |
| **/
 | |
| 
 | |
| #include "Xhci.h"
 | |
| 
 | |
| /**
 | |
|   Read 1-byte width XHCI capability register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the 1-byte width capability register.
 | |
| 
 | |
|   @return The register content read.
 | |
|   @retval If err, return 0xFF.
 | |
| 
 | |
| **/
 | |
| UINT8
 | |
| XhcReadCapReg8 (
 | |
|   IN  USB_XHCI_INSTANCE   *Xhc,
 | |
|   IN  UINT32              Offset
 | |
|   )
 | |
| {
 | |
|   UINT8                   Data;
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Read (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint8,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              (UINT64) Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcReadCapReg: Pci Io read error - %r at %d\n", Status, Offset));
 | |
|     Data = 0xFF;
 | |
|   }
 | |
| 
 | |
|   return Data;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Read 4-bytes width XHCI capability register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the 4-bytes width capability register.
 | |
| 
 | |
|   @return The register content read.
 | |
|   @retval If err, return 0xFFFFFFFF.
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| XhcReadCapReg (
 | |
|   IN  USB_XHCI_INSTANCE   *Xhc,
 | |
|   IN  UINT32              Offset
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Read (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              (UINT64) Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcReadCapReg: Pci Io read error - %r at %d\n", Status, Offset));
 | |
|     Data = 0xFFFFFFFF;
 | |
|   }
 | |
| 
 | |
|   return Data;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Read 4-bytes width XHCI Operational register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the 4-bytes width operational register.
 | |
| 
 | |
|   @return The register content read.
 | |
|   @retval If err, return 0xFFFFFFFF.
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| XhcReadOpReg (
 | |
|   IN  USB_XHCI_INSTANCE   *Xhc,
 | |
|   IN  UINT32              Offset
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->CapLength != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Read (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->CapLength + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcReadOpReg: Pci Io Read error - %r at %d\n", Status, Offset));
 | |
|     Data = 0xFFFFFFFF;
 | |
|   }
 | |
| 
 | |
|   return Data;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Write the data to the 4-bytes width XHCI operational register.
 | |
| 
 | |
|   @param  Xhc      The XHCI Instance.
 | |
|   @param  Offset   The offset of the 4-bytes width operational register.
 | |
|   @param  Data     The data to write.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcWriteOpReg (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Data
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->CapLength != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Write (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->CapLength + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcWriteOpReg: Pci Io Write error: %r at %d\n", Status, Offset));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Write the data to the 2-bytes width XHCI operational register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the 2-bytes width operational register.
 | |
|   @param  Data         The data to write.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcWriteOpReg16 (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT16               Data
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->CapLength != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Write (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint16,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->CapLength + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcWriteOpReg16: Pci Io Write error: %r at %d\n", Status, Offset));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Read XHCI door bell register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the door bell register.
 | |
| 
 | |
|   @return The register content read
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| XhcReadDoorBellReg (
 | |
|   IN  USB_XHCI_INSTANCE   *Xhc,
 | |
|   IN  UINT32              Offset
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->DBOff != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Read (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->DBOff + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcReadDoorBellReg: Pci Io Read error - %r at %d\n", Status, Offset));
 | |
|     Data = 0xFFFFFFFF;
 | |
|   }
 | |
| 
 | |
|   return Data;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Write the data to the XHCI door bell register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the door bell register.
 | |
|   @param  Data         The data to write.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcWriteDoorBellReg (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Data
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->DBOff != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Write (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->DBOff + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcWriteOpReg: Pci Io Write error: %r at %d\n", Status, Offset));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Read XHCI runtime register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the runtime register.
 | |
| 
 | |
|   @return The register content read
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| XhcReadRuntimeReg (
 | |
|   IN  USB_XHCI_INSTANCE   *Xhc,
 | |
|   IN  UINT32              Offset
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->RTSOff != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Read (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->RTSOff + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcReadRuntimeReg: Pci Io Read error - %r at %d\n", Status, Offset));
 | |
|     Data = 0xFFFFFFFF;
 | |
|   }
 | |
| 
 | |
|   return Data;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Write the data to the XHCI runtime register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the runtime register.
 | |
|   @param  Data         The data to write.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcWriteRuntimeReg (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Data
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->RTSOff != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Write (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->RTSOff + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcWriteRuntimeReg: Pci Io Write error: %r at %d\n", Status, Offset));
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Read XHCI extended capability register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the extended capability register.
 | |
| 
 | |
|   @return The register content read
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| XhcReadExtCapReg (
 | |
|   IN  USB_XHCI_INSTANCE   *Xhc,
 | |
|   IN  UINT32              Offset
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->ExtCapRegBase != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Read (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->ExtCapRegBase + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcReadExtCapReg: Pci Io Read error - %r at %d\n", Status, Offset));
 | |
|     Data = 0xFFFFFFFF;
 | |
|   }
 | |
| 
 | |
|   return Data;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Write the data to the XHCI extended capability register.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the extended capability register.
 | |
|   @param  Data         The data to write.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcWriteExtCapReg (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Data
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   ASSERT (Xhc->ExtCapRegBase != 0);
 | |
| 
 | |
|   Status = Xhc->PciIo->Mem.Write (
 | |
|                              Xhc->PciIo,
 | |
|                              EfiPciIoWidthUint32,
 | |
|                              XHC_BAR_INDEX,
 | |
|                              Xhc->ExtCapRegBase + Offset,
 | |
|                              1,
 | |
|                              &Data
 | |
|                              );
 | |
| 
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     DEBUG ((EFI_D_ERROR, "XhcWriteExtCapReg: Pci Io Write error: %r at %d\n", Status, Offset));
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Set one bit of the runtime register while keeping other bits.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the runtime register.
 | |
|   @param  Bit          The bit mask of the register to set.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcSetRuntimeRegBit (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Bit
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
| 
 | |
|   Data  = XhcReadRuntimeReg (Xhc, Offset);
 | |
|   Data |= Bit;
 | |
|   XhcWriteRuntimeReg (Xhc, Offset, Data);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Clear one bit of the runtime register while keeping other bits.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the runtime register.
 | |
|   @param  Bit          The bit mask of the register to set.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcClearRuntimeRegBit (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Bit
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
| 
 | |
|   Data  = XhcReadRuntimeReg (Xhc, Offset);
 | |
|   Data &= ~Bit;
 | |
|   XhcWriteRuntimeReg (Xhc, Offset, Data);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Set one bit of the operational register while keeping other bits.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the operational register.
 | |
|   @param  Bit          The bit mask of the register to set.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcSetOpRegBit (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Bit
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
| 
 | |
|   Data  = XhcReadOpReg (Xhc, Offset);
 | |
|   Data |= Bit;
 | |
|   XhcWriteOpReg (Xhc, Offset, Data);
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Clear one bit of the operational register while keeping other bits.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the operational register.
 | |
|   @param  Bit          The bit mask of the register to clear.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcClearOpRegBit (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Bit
 | |
|   )
 | |
| {
 | |
|   UINT32                  Data;
 | |
| 
 | |
|   Data  = XhcReadOpReg (Xhc, Offset);
 | |
|   Data &= ~Bit;
 | |
|   XhcWriteOpReg (Xhc, Offset, Data);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Wait the operation register's bit as specified by Bit
 | |
|   to become set (or clear).
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Offset       The offset of the operation register.
 | |
|   @param  Bit          The bit of the register to wait for.
 | |
|   @param  WaitToSet    Wait the bit to set or clear.
 | |
|   @param  Timeout      The time to wait before abort (in millisecond, ms).
 | |
| 
 | |
|   @retval EFI_SUCCESS  The bit successfully changed by host controller.
 | |
|   @retval EFI_TIMEOUT  The time out occurred.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| XhcWaitOpRegBit (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Offset,
 | |
|   IN UINT32               Bit,
 | |
|   IN BOOLEAN              WaitToSet,
 | |
|   IN UINT32               Timeout
 | |
|   )
 | |
| {
 | |
|   UINT32                  Index;
 | |
|   UINT64                  Loop;
 | |
| 
 | |
|   Loop   = Timeout * XHC_1_MILLISECOND;
 | |
| 
 | |
|   for (Index = 0; Index < Loop; Index++) {
 | |
|     if (XHC_REG_BIT_IS_SET (Xhc, Offset, Bit) == WaitToSet) {
 | |
|       return EFI_SUCCESS;
 | |
|     }
 | |
| 
 | |
|     gBS->Stall (XHC_1_MICROSECOND);
 | |
|   }
 | |
| 
 | |
|   return EFI_TIMEOUT;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Set Bios Ownership
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcSetBiosOwnership (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc
 | |
|   )
 | |
| {
 | |
|   UINT32                    Buffer;
 | |
| 
 | |
|   if (Xhc->UsbLegSupOffset == 0xFFFFFFFF) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "XhcSetBiosOwnership: called to set BIOS ownership\n"));
 | |
| 
 | |
|   Buffer = XhcReadExtCapReg (Xhc, Xhc->UsbLegSupOffset);
 | |
|   Buffer = ((Buffer & (~USBLEGSP_OS_SEMAPHORE)) | USBLEGSP_BIOS_SEMAPHORE);
 | |
|   XhcWriteExtCapReg (Xhc, Xhc->UsbLegSupOffset, Buffer);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Clear Bios Ownership
 | |
| 
 | |
|   @param  Xhc       The XHCI Instance.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| XhcClearBiosOwnership (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc
 | |
|   )
 | |
| {
 | |
|   UINT32                    Buffer;
 | |
| 
 | |
|   if (Xhc->UsbLegSupOffset == 0xFFFFFFFF) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "XhcClearBiosOwnership: called to clear BIOS ownership\n"));
 | |
| 
 | |
|   Buffer = XhcReadExtCapReg (Xhc, Xhc->UsbLegSupOffset);
 | |
|   Buffer = ((Buffer & (~USBLEGSP_BIOS_SEMAPHORE)) | USBLEGSP_OS_SEMAPHORE);
 | |
|   XhcWriteExtCapReg (Xhc, Xhc->UsbLegSupOffset, Buffer);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Calculate the offset of the XHCI capability.
 | |
| 
 | |
|   @param  Xhc     The XHCI Instance.
 | |
|   @param  CapId   The XHCI Capability ID.
 | |
| 
 | |
|   @return The offset of XHCI legacy support capability register.
 | |
| 
 | |
| **/
 | |
| UINT32
 | |
| XhcGetCapabilityAddr (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT8                CapId
 | |
|   )
 | |
| {
 | |
|   UINT32 ExtCapOffset;
 | |
|   UINT8  NextExtCapReg;
 | |
|   UINT32 Data;
 | |
| 
 | |
|   ExtCapOffset = 0;
 | |
| 
 | |
|   do {
 | |
|     //
 | |
|     // Check if the extended capability register's capability id is USB Legacy Support.
 | |
|     //
 | |
|     Data = XhcReadExtCapReg (Xhc, ExtCapOffset);
 | |
|     if ((Data & 0xFF) == CapId) {
 | |
|       return ExtCapOffset;
 | |
|     }
 | |
|     //
 | |
|     // If not, then traverse all of the ext capability registers till finding out it.
 | |
|     //
 | |
|     NextExtCapReg = (UINT8)((Data >> 8) & 0xFF);
 | |
|     ExtCapOffset += (NextExtCapReg << 2);
 | |
|   } while (NextExtCapReg != 0);
 | |
| 
 | |
|   return 0xFFFFFFFF;
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Whether the XHCI host controller is halted.
 | |
| 
 | |
|   @param  Xhc     The XHCI Instance.
 | |
| 
 | |
|   @retval TRUE    The controller is halted.
 | |
|   @retval FALSE   It isn't halted.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| XhcIsHalt (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc
 | |
|   )
 | |
| {
 | |
|   return XHC_REG_BIT_IS_SET (Xhc, XHC_USBSTS_OFFSET, XHC_USBSTS_HALT);
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Whether system error occurred.
 | |
| 
 | |
|   @param  Xhc      The XHCI Instance.
 | |
| 
 | |
|   @retval TRUE     System error happened.
 | |
|   @retval FALSE    No system error.
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| XhcIsSysError (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc
 | |
|   )
 | |
| {
 | |
|   return XHC_REG_BIT_IS_SET (Xhc, XHC_USBSTS_OFFSET, XHC_USBSTS_HSE);
 | |
| }
 | |
| 
 | |
| /**
 | |
|   Reset the XHCI host controller.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Timeout      Time to wait before abort (in millisecond, ms).
 | |
| 
 | |
|   @retval EFI_SUCCESS  The XHCI host controller is reset.
 | |
|   @return Others       Failed to reset the XHCI before Timeout.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| XhcResetHC (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Timeout
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   Status = EFI_SUCCESS;
 | |
| 
 | |
|   DEBUG ((EFI_D_INFO, "XhcResetHC!\n"));
 | |
|   //
 | |
|   // Host can only be reset when it is halt. If not so, halt it
 | |
|   //
 | |
|   if (!XHC_REG_BIT_IS_SET (Xhc, XHC_USBSTS_OFFSET, XHC_USBSTS_HALT)) {
 | |
|     Status = XhcHaltHC (Xhc, Timeout);
 | |
| 
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if ((Xhc->DebugCapSupOffset == 0xFFFFFFFF) || ((XhcReadExtCapReg (Xhc, Xhc->DebugCapSupOffset) & 0xFF) != XHC_CAP_USB_DEBUG) ||
 | |
|       ((XhcReadExtCapReg (Xhc, Xhc->DebugCapSupOffset + XHC_DC_DCCTRL) & BIT0) == 0)) {
 | |
|     XhcSetOpRegBit (Xhc, XHC_USBCMD_OFFSET, XHC_USBCMD_RESET);
 | |
|     //
 | |
|     // Some XHCI host controllers require to have extra 1ms delay before accessing any MMIO register during reset.
 | |
|     // Otherwise there may have the timeout case happened.
 | |
|     // The below is a workaround to solve such problem.
 | |
|     //
 | |
|     gBS->Stall (XHC_1_MILLISECOND);
 | |
|     Status = XhcWaitOpRegBit (Xhc, XHC_USBCMD_OFFSET, XHC_USBCMD_RESET, FALSE, Timeout);
 | |
|   }
 | |
| 
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Halt the XHCI host controller.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Timeout      Time to wait before abort (in millisecond, ms).
 | |
| 
 | |
|   @return EFI_SUCCESS  The XHCI host controller is halt.
 | |
|   @return EFI_TIMEOUT  Failed to halt the XHCI before Timeout.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| XhcHaltHC (
 | |
|   IN USB_XHCI_INSTANCE   *Xhc,
 | |
|   IN UINT32              Timeout
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   XhcClearOpRegBit (Xhc, XHC_USBCMD_OFFSET, XHC_USBCMD_RUN);
 | |
|   Status = XhcWaitOpRegBit (Xhc, XHC_USBSTS_OFFSET, XHC_USBSTS_HALT, TRUE, Timeout);
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Set the XHCI host controller to run.
 | |
| 
 | |
|   @param  Xhc          The XHCI Instance.
 | |
|   @param  Timeout      Time to wait before abort (in millisecond, ms).
 | |
| 
 | |
|   @return EFI_SUCCESS  The XHCI host controller is running.
 | |
|   @return EFI_TIMEOUT  Failed to set the XHCI to run before Timeout.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| XhcRunHC (
 | |
|   IN USB_XHCI_INSTANCE    *Xhc,
 | |
|   IN UINT32               Timeout
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS              Status;
 | |
| 
 | |
|   XhcSetOpRegBit (Xhc, XHC_USBCMD_OFFSET, XHC_USBCMD_RUN);
 | |
|   Status = XhcWaitOpRegBit (Xhc, XHC_USBSTS_OFFSET, XHC_USBSTS_HALT, FALSE, Timeout);
 | |
|   return Status;
 | |
| }
 | |
| 
 |