/** @file This driver is used for Opal Password Feature support at AHCI mode. Copyright (c) 2016 - 2018, Intel Corporation. All rights reserved.
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 "OpalPasswordPei.h" /** Start command for give slot on specific port. @param AhciBar AHCI bar address. @param Port The number of port. @param CommandSlot The number of CommandSlot. @param Timeout The timeout Value of start. @retval EFI_DEVICE_ERROR The command start unsuccessfully. @retval EFI_TIMEOUT The operation is time out. @retval EFI_SUCCESS The command start successfully. **/ EFI_STATUS EFIAPI AhciStartCommand ( IN UINT32 AhciBar, IN UINT8 Port, IN UINT8 CommandSlot, IN UINT64 Timeout ); /** Stop command running for giving port @param AhciBar AHCI bar address. @param Port The number of port. @param Timeout The timeout Value of stop. @retval EFI_DEVICE_ERROR The command stop unsuccessfully. @retval EFI_TIMEOUT The operation is time out. @retval EFI_SUCCESS The command stop successfully. **/ EFI_STATUS EFIAPI AhciStopCommand ( IN UINT32 AhciBar, IN UINT8 Port, IN UINT64 Timeout ); /** Read AHCI Operation register. @param AhciBar AHCI bar address. @param Offset The operation register offset. @return The register content read. **/ UINT32 EFIAPI AhciReadReg ( IN UINT32 AhciBar, IN UINT32 Offset ) { UINT32 Data; Data = 0; Data = MmioRead32 (AhciBar + Offset); return Data; } /** Write AHCI Operation register. @param AhciBar AHCI bar address. @param Offset The operation register offset. @param Data The Data used to write down. **/ VOID EFIAPI AhciWriteReg ( IN UINT32 AhciBar, IN UINT32 Offset, IN UINT32 Data ) { MmioWrite32 (AhciBar + Offset, Data); return ; } /** Do AND operation with the Value of AHCI Operation register. @param AhciBar AHCI bar address. @param Offset The operation register offset. @param AndData The Data used to do AND operation. **/ VOID EFIAPI AhciAndReg ( IN UINT32 AhciBar, IN UINT32 Offset, IN UINT32 AndData ) { UINT32 Data; Data = AhciReadReg (AhciBar, Offset); Data &= AndData; AhciWriteReg (AhciBar, Offset, Data); } /** Do OR operation with the Value of AHCI Operation register. @param AhciBar AHCI bar address. @param Offset The operation register offset. @param OrData The Data used to do OR operation. **/ VOID EFIAPI AhciOrReg ( IN UINT32 AhciBar, IN UINT32 Offset, IN UINT32 OrData ) { UINT32 Data; Data = AhciReadReg (AhciBar, Offset); Data |= OrData; AhciWriteReg (AhciBar, Offset, Data); } /** Wait for memory set to the test Value. @param AhciBar AHCI bar address. @param Offset The memory offset to test. @param MaskValue The mask Value of memory. @param TestValue The test Value of memory. @param Timeout The time out Value for wait memory set. @retval EFI_DEVICE_ERROR The memory is not set. @retval EFI_TIMEOUT The memory setting is time out. @retval EFI_SUCCESS The memory is correct set. **/ EFI_STATUS EFIAPI AhciWaitMmioSet ( IN UINT32 AhciBar, IN UINT32 Offset, IN UINT32 MaskValue, IN UINT32 TestValue, IN UINT64 Timeout ) { UINT32 Value; UINT32 Delay; Delay = (UINT32) (DivU64x32(Timeout, 1000) + 1); do { Value = AhciReadReg (AhciBar, Offset) & MaskValue; if (Value == TestValue) { return EFI_SUCCESS; } // // Stall for 100 microseconds. // MicroSecondDelay (100); Delay--; } while (Delay > 0); return EFI_TIMEOUT; } /** Wait for the Value of the specified system memory set to the test Value. @param Address The system memory address to test. @param MaskValue The mask Value of memory. @param TestValue The test Value of memory. @param Timeout The time out Value for wait memory set, uses 100ns as a unit. @retval EFI_TIMEOUT The system memory setting is time out. @retval EFI_SUCCESS The system memory is correct set. **/ EFI_STATUS EFIAPI AhciWaitMemSet ( IN EFI_PHYSICAL_ADDRESS Address, IN UINT32 MaskValue, IN UINT32 TestValue, IN UINT64 Timeout ) { UINT32 Value; UINT32 Delay; Delay = (UINT32) (DivU64x32 (Timeout, 1000) + 1); do { // // Access sytem memory to see if the Value is the tested one. // // The system memory pointed by Address will be updated by the // SATA Host Controller, "volatile" is introduced to prevent // compiler from optimizing the access to the memory address // to only read once. // Value = *(volatile UINT32 *) (UINTN) Address; Value &= MaskValue; if (Value == TestValue) { return EFI_SUCCESS; } // // Stall for 100 microseconds. // MicroSecondDelay (100); Delay--; } while (Delay > 0); return EFI_TIMEOUT; } /** Check the memory status to the test Value. @param[in] Address The memory address to test. @param[in] MaskValue The mask Value of memory. @param[in] TestValue The test Value of memory. @param[in, out] RetryTimes The retry times Value for waitting memory set. If 0, then just try once. @retval EFI_NOTREADY The memory is not set. @retval EFI_TIMEOUT The memory setting retry times out. @retval EFI_SUCCESS The memory is correct set. **/ EFI_STATUS EFIAPI AhciCheckMemSet ( IN UINTN Address, IN UINT32 MaskValue, IN UINT32 TestValue, IN OUT UINTN *RetryTimes OPTIONAL ) { UINT32 Value; if (RetryTimes != NULL) { (*RetryTimes)--; } Value = *(volatile UINT32 *) Address; Value &= MaskValue; if (Value == TestValue) { return EFI_SUCCESS; } if ((RetryTimes != NULL) && (*RetryTimes == 0)) { return EFI_TIMEOUT; } else { return EFI_NOT_READY; } } /** Clear the port interrupt and error status. It will also clear HBA interrupt status. @param AhciBar AHCI bar address. @param Port The number of port. **/ VOID EFIAPI AhciClearPortStatus ( IN UINT32 AhciBar, IN UINT8 Port ) { UINT32 Offset; // // Clear any error status // Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_SERR; AhciWriteReg (AhciBar, Offset, AhciReadReg (AhciBar, Offset)); // // Clear any port interrupt status // Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_IS; AhciWriteReg (AhciBar, Offset, AhciReadReg (AhciBar, Offset)); // // Clear any HBA interrupt status // AhciWriteReg (AhciBar, EFI_AHCI_IS_OFFSET, AhciReadReg (AhciBar, EFI_AHCI_IS_OFFSET)); } /** Enable the FIS running for giving port. @param AhciBar AHCI bar address. @param Port The number of port. @param Timeout The timeout Value of enabling FIS. @retval EFI_DEVICE_ERROR The FIS enable setting fails. @retval EFI_TIMEOUT The FIS enable setting is time out. @retval EFI_SUCCESS The FIS enable successfully. **/ EFI_STATUS EFIAPI AhciEnableFisReceive ( IN UINT32 AhciBar, IN UINT8 Port, IN UINT64 Timeout ) { UINT32 Offset; Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; AhciOrReg (AhciBar, Offset, EFI_AHCI_PORT_CMD_FRE); return AhciWaitMmioSet ( AhciBar, Offset, EFI_AHCI_PORT_CMD_FR, EFI_AHCI_PORT_CMD_FR, Timeout ); } /** Disable the FIS running for giving port. @param AhciBar AHCI bar address. @param Port The number of port. @param Timeout The timeout Value of disabling FIS. @retval EFI_DEVICE_ERROR The FIS disable setting fails. @retval EFI_TIMEOUT The FIS disable setting is time out. @retval EFI_UNSUPPORTED The port is in running state. @retval EFI_SUCCESS The FIS disable successfully. **/ EFI_STATUS EFIAPI AhciDisableFisReceive ( IN UINT32 AhciBar, IN UINT8 Port, IN UINT64 Timeout ) { UINT32 Offset; UINT32 Data; Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; Data = AhciReadReg (AhciBar, Offset); // // Before disabling Fis receive, the DMA engine of the port should NOT be in running status. // if ((Data & (EFI_AHCI_PORT_CMD_ST | EFI_AHCI_PORT_CMD_CR)) != 0) { return EFI_UNSUPPORTED; } // // Check if the Fis receive DMA engine for the port is running. // if ((Data & EFI_AHCI_PORT_CMD_FR) != EFI_AHCI_PORT_CMD_FR) { return EFI_SUCCESS; } AhciAndReg (AhciBar, Offset, (UINT32)~(EFI_AHCI_PORT_CMD_FRE)); return AhciWaitMmioSet ( AhciBar, Offset, EFI_AHCI_PORT_CMD_FR, 0, Timeout ); } /** Build the command list, command table and prepare the fis receiver. @param AhciContext The pointer to the AHCI_CONTEXT. @param Port The number of port. @param PortMultiplier The timeout Value of stop. @param CommandFis The control fis will be used for the transfer. @param CommandList The command list will be used for the transfer. @param AtapiCommand The atapi command will be used for the transfer. @param AtapiCommandLength The Length of the atapi command. @param CommandSlotNumber The command slot will be used for the transfer. @param DataPhysicalAddr The pointer to the Data Buffer pci bus master address. @param DataLength The Data count to be transferred. **/ VOID EFIAPI AhciBuildCommand ( IN AHCI_CONTEXT *AhciContext, IN UINT8 Port, IN UINT8 PortMultiplier, IN EFI_AHCI_COMMAND_FIS *CommandFis, IN EFI_AHCI_COMMAND_LIST *CommandList, IN EFI_AHCI_ATAPI_COMMAND *AtapiCommand OPTIONAL, IN UINT8 AtapiCommandLength, IN UINT8 CommandSlotNumber, IN OUT VOID *DataPhysicalAddr, IN UINT64 DataLength ) { EFI_AHCI_REGISTERS *AhciRegisters; UINT32 AhciBar; UINT64 BaseAddr; UINT64 PrdtNumber; UINTN RemainedData; UINTN MemAddr; DATA_64 Data64; UINT32 Offset; AhciRegisters = &AhciContext->AhciRegisters; AhciBar = AhciContext->AhciBar; // // Filling the PRDT // PrdtNumber = DivU64x32 (DataLength + EFI_AHCI_MAX_DATA_PER_PRDT - 1, EFI_AHCI_MAX_DATA_PER_PRDT); // // According to AHCI 1.3 spec, a PRDT entry can point to a maximum 4MB Data block. // It also limits that the maximum amount of the PRDT entry in the command table // is 65535. // ASSERT (PrdtNumber <= 1); Data64.Uint64 = (UINTN) (AhciRegisters->AhciRFis); BaseAddr = Data64.Uint64; ZeroMem ((VOID *)((UINTN) BaseAddr), sizeof (EFI_AHCI_RECEIVED_FIS)); ZeroMem (AhciRegisters->AhciCommandTable, sizeof (EFI_AHCI_COMMAND_TABLE)); CommandFis->AhciCFisPmNum = PortMultiplier; CopyMem (&AhciRegisters->AhciCommandTable->CommandFis, CommandFis, sizeof (EFI_AHCI_COMMAND_FIS)); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; if (AtapiCommand != NULL) { CopyMem ( &AhciRegisters->AhciCommandTable->AtapiCmd, AtapiCommand, AtapiCommandLength ); CommandList->AhciCmdA = 1; CommandList->AhciCmdP = 1; AhciOrReg (AhciBar, Offset, (EFI_AHCI_PORT_CMD_DLAE | EFI_AHCI_PORT_CMD_ATAPI)); } else { AhciAndReg (AhciBar, Offset, (UINT32)~(EFI_AHCI_PORT_CMD_DLAE | EFI_AHCI_PORT_CMD_ATAPI)); } RemainedData = (UINTN) DataLength; MemAddr = (UINTN) DataPhysicalAddr; CommandList->AhciCmdPrdtl = (UINT32)PrdtNumber; AhciRegisters->AhciCommandTable->PrdtTable.AhciPrdtDbc = (UINT32)RemainedData - 1; Data64.Uint64 = (UINT64)MemAddr; AhciRegisters->AhciCommandTable->PrdtTable.AhciPrdtDba = Data64.Uint32.Lower32; AhciRegisters->AhciCommandTable->PrdtTable.AhciPrdtDbau = Data64.Uint32.Upper32; // // Set the last PRDT to Interrupt On Complete // AhciRegisters->AhciCommandTable->PrdtTable.AhciPrdtIoc = 1; CopyMem ( (VOID *) ((UINTN) AhciRegisters->AhciCmdList + (UINTN) CommandSlotNumber * sizeof (EFI_AHCI_COMMAND_LIST)), CommandList, sizeof (EFI_AHCI_COMMAND_LIST) ); Data64.Uint64 = (UINT64)(UINTN) AhciRegisters->AhciCommandTable; AhciRegisters->AhciCmdList[CommandSlotNumber].AhciCmdCtba = Data64.Uint32.Lower32; AhciRegisters->AhciCmdList[CommandSlotNumber].AhciCmdCtbau = Data64.Uint32.Upper32; AhciRegisters->AhciCmdList[CommandSlotNumber].AhciCmdPmp = PortMultiplier; } /** Buid a command FIS. @param CmdFis A pointer to the EFI_AHCI_COMMAND_FIS Data structure. @param AtaCommandBlock A pointer to the AhciBuildCommandFis Data structure. **/ VOID EFIAPI AhciBuildCommandFis ( IN OUT EFI_AHCI_COMMAND_FIS *CmdFis, IN EFI_ATA_COMMAND_BLOCK *AtaCommandBlock ) { ZeroMem (CmdFis, sizeof (EFI_AHCI_COMMAND_FIS)); CmdFis->AhciCFisType = EFI_AHCI_FIS_REGISTER_H2D; // // Indicator it's a command // CmdFis->AhciCFisCmdInd = 0x1; CmdFis->AhciCFisCmd = AtaCommandBlock->AtaCommand; CmdFis->AhciCFisFeature = AtaCommandBlock->AtaFeatures; CmdFis->AhciCFisFeatureExp = AtaCommandBlock->AtaFeaturesExp; CmdFis->AhciCFisSecNum = AtaCommandBlock->AtaSectorNumber; CmdFis->AhciCFisSecNumExp = AtaCommandBlock->AtaSectorNumberExp; CmdFis->AhciCFisClyLow = AtaCommandBlock->AtaCylinderLow; CmdFis->AhciCFisClyLowExp = AtaCommandBlock->AtaCylinderLowExp; CmdFis->AhciCFisClyHigh = AtaCommandBlock->AtaCylinderHigh; CmdFis->AhciCFisClyHighExp = AtaCommandBlock->AtaCylinderHighExp; CmdFis->AhciCFisSecCount = AtaCommandBlock->AtaSectorCount; CmdFis->AhciCFisSecCountExp = AtaCommandBlock->AtaSectorCountExp; CmdFis->AhciCFisDevHead = (UINT8) (AtaCommandBlock->AtaDeviceHead | 0xE0); } /** Start a PIO Data transfer on specific port. @param AhciContext The pointer to the AHCI_CONTEXT. @param Port The number of port. @param PortMultiplier The timeout Value of stop. @param AtapiCommand The atapi command will be used for the transfer. @param AtapiCommandLength The Length of the atapi command. @param Read The transfer direction. @param AtaCommandBlock The EFI_ATA_COMMAND_BLOCK Data. @param AtaStatusBlock The EFI_ATA_STATUS_BLOCK Data. @param MemoryAddr The pointer to the Data Buffer. @param DataCount The Data count to be transferred. @param Timeout The timeout Value of non Data transfer. @retval EFI_DEVICE_ERROR The PIO Data transfer abort with error occurs. @retval EFI_TIMEOUT The operation is time out. @retval EFI_UNSUPPORTED The device is not ready for transfer. @retval EFI_SUCCESS The PIO Data transfer executes successfully. **/ EFI_STATUS EFIAPI AhciPioTransfer ( IN AHCI_CONTEXT *AhciContext, IN UINT8 Port, IN UINT8 PortMultiplier, IN EFI_AHCI_ATAPI_COMMAND *AtapiCommand OPTIONAL, IN UINT8 AtapiCommandLength, IN BOOLEAN Read, IN EFI_ATA_COMMAND_BLOCK *AtaCommandBlock, IN OUT EFI_ATA_STATUS_BLOCK *AtaStatusBlock, IN OUT VOID *MemoryAddr, IN UINT32 DataCount, IN UINT64 Timeout ) { EFI_STATUS Status; EFI_AHCI_REGISTERS *AhciRegisters; UINT32 AhciBar; UINT32 FisBaseAddr; UINT32 Offset; UINT32 Delay; EFI_AHCI_COMMAND_FIS CFis; EFI_AHCI_COMMAND_LIST CmdList; UINT32 PortTfd; UINT32 PrdCount; UINT32 OldRfisLo; UINT32 OldRfisHi; UINT32 OldCmdListLo; UINT32 OldCmdListHi; AhciRegisters = &AhciContext->AhciRegisters; AhciBar = AhciContext->AhciBar; Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_FB; OldRfisLo = AhciReadReg (AhciBar, Offset); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_FBU; OldRfisHi = AhciReadReg (AhciBar, Offset); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_FB; AhciWriteReg (AhciBar, Offset, (UINT32)(UINTN)AhciRegisters->AhciRFis); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_FBU; AhciWriteReg (AhciBar, Offset, 0); // // Single task envrionment, we only use one command table for all port // Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CLB; OldCmdListLo = AhciReadReg (AhciBar, Offset); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CLBU; OldCmdListHi = AhciReadReg (AhciBar, Offset); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CLB; AhciWriteReg (AhciBar, Offset, (UINT32)(UINTN)AhciRegisters->AhciCmdList); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CLBU; AhciWriteReg (AhciBar, Offset, 0); // // Package read needed // AhciBuildCommandFis (&CFis, AtaCommandBlock); ZeroMem (&CmdList, sizeof (EFI_AHCI_COMMAND_LIST)); CmdList.AhciCmdCfl = EFI_AHCI_FIS_REGISTER_H2D_LENGTH / 4; CmdList.AhciCmdW = Read ? 0 : 1; AhciBuildCommand ( AhciContext, Port, PortMultiplier, &CFis, &CmdList, AtapiCommand, AtapiCommandLength, 0, MemoryAddr, DataCount ); Status = AhciStartCommand ( AhciBar, Port, 0, Timeout ); if (EFI_ERROR (Status)) { goto Exit; } // // Checking the status and wait the driver sending Data // FisBaseAddr = (UINT32)(UINTN)AhciRegisters->AhciRFis; if (Read && (AtapiCommand == 0)) { // // Wait device sends the PIO setup fis before Data transfer // Status = EFI_TIMEOUT; Delay = (UINT32) (DivU64x32 (Timeout, 1000) + 1); do { Offset = FisBaseAddr + EFI_AHCI_PIO_FIS_OFFSET; Status = AhciCheckMemSet (Offset, EFI_AHCI_FIS_TYPE_MASK, EFI_AHCI_FIS_PIO_SETUP, 0); if (!EFI_ERROR (Status)) { Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_TFD; PortTfd = AhciReadReg (AhciBar, (UINT32) Offset); // // PxTFD will be updated if there is a D2H or SetupFIS received. // For PIO IN transfer, D2H means a device error. Therefore we only need to check the TFD after receiving a SetupFIS. // if ((PortTfd & EFI_AHCI_PORT_TFD_ERR) != 0) { Status = EFI_DEVICE_ERROR; break; } PrdCount = *(volatile UINT32 *) (&(AhciRegisters->AhciCmdList[0].AhciCmdPrdbc)); if (PrdCount == DataCount) { break; } } Offset = FisBaseAddr + EFI_AHCI_D2H_FIS_OFFSET; Status = AhciCheckMemSet (Offset, EFI_AHCI_FIS_TYPE_MASK, EFI_AHCI_FIS_REGISTER_D2H, 0); if (!EFI_ERROR (Status)) { Status = EFI_DEVICE_ERROR; break; } // // Stall for 100 microseconds. // MicroSecondDelay(100); Delay--; } while (Delay > 0); } else { // // Wait for D2H Fis is received // Offset = FisBaseAddr + EFI_AHCI_D2H_FIS_OFFSET; Status = AhciWaitMemSet ( Offset, EFI_AHCI_FIS_TYPE_MASK, EFI_AHCI_FIS_REGISTER_D2H, Timeout ); if (EFI_ERROR (Status)) { goto Exit; } Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_TFD; PortTfd = AhciReadReg (AhciBar, (UINT32) Offset); if ((PortTfd & EFI_AHCI_PORT_TFD_ERR) != 0) { Status = EFI_DEVICE_ERROR; } } Exit: AhciStopCommand ( AhciBar, Port, Timeout ); AhciDisableFisReceive ( AhciBar, Port, Timeout ); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_FB; AhciWriteReg (AhciBar, Offset, OldRfisLo); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_FBU; AhciWriteReg (AhciBar, Offset, OldRfisHi); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CLB; AhciWriteReg (AhciBar, Offset, OldCmdListLo); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CLBU; AhciWriteReg (AhciBar, Offset, OldCmdListHi); return Status; } /** Stop command running for giving port @param AhciBar AHCI bar address. @param Port The number of port. @param Timeout The timeout Value of stop. @retval EFI_DEVICE_ERROR The command stop unsuccessfully. @retval EFI_TIMEOUT The operation is time out. @retval EFI_SUCCESS The command stop successfully. **/ EFI_STATUS EFIAPI AhciStopCommand ( IN UINT32 AhciBar, IN UINT8 Port, IN UINT64 Timeout ) { UINT32 Offset; UINT32 Data; Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; Data = AhciReadReg (AhciBar, Offset); if ((Data & (EFI_AHCI_PORT_CMD_ST | EFI_AHCI_PORT_CMD_CR)) == 0) { return EFI_SUCCESS; } if ((Data & EFI_AHCI_PORT_CMD_ST) != 0) { AhciAndReg (AhciBar, Offset, (UINT32)~(EFI_AHCI_PORT_CMD_ST)); } return AhciWaitMmioSet ( AhciBar, Offset, EFI_AHCI_PORT_CMD_CR, 0, Timeout ); } /** Start command for give slot on specific port. @param AhciBar AHCI bar address. @param Port The number of port. @param CommandSlot The number of CommandSlot. @param Timeout The timeout Value of start. @retval EFI_DEVICE_ERROR The command start unsuccessfully. @retval EFI_TIMEOUT The operation is time out. @retval EFI_SUCCESS The command start successfully. **/ EFI_STATUS EFIAPI AhciStartCommand ( IN UINT32 AhciBar, IN UINT8 Port, IN UINT8 CommandSlot, IN UINT64 Timeout ) { UINT32 CmdSlotBit; EFI_STATUS Status; UINT32 PortStatus; UINT32 StartCmd; UINT32 PortTfd; UINT32 Offset; UINT32 Capability; // // Collect AHCI controller information // Capability = AhciReadReg(AhciBar, EFI_AHCI_CAPABILITY_OFFSET); CmdSlotBit = (UINT32) (1 << CommandSlot); AhciClearPortStatus ( AhciBar, Port ); Status = AhciEnableFisReceive ( AhciBar, Port, Timeout ); if (EFI_ERROR (Status)) { return Status; } Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; PortStatus = AhciReadReg (AhciBar, Offset); StartCmd = 0; if ((PortStatus & EFI_AHCI_PORT_CMD_ALPE) != 0) { StartCmd = AhciReadReg (AhciBar, Offset); StartCmd &= ~EFI_AHCI_PORT_CMD_ICC_MASK; StartCmd |= EFI_AHCI_PORT_CMD_ACTIVE; } Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_TFD; PortTfd = AhciReadReg (AhciBar, Offset); if ((PortTfd & (EFI_AHCI_PORT_TFD_BSY | EFI_AHCI_PORT_TFD_DRQ)) != 0) { if ((Capability & BIT24) != 0) { Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; AhciOrReg (AhciBar, Offset, EFI_AHCI_PORT_CMD_COL); AhciWaitMmioSet ( AhciBar, Offset, EFI_AHCI_PORT_CMD_COL, 0, Timeout ); } } Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; AhciOrReg (AhciBar, Offset, EFI_AHCI_PORT_CMD_ST | StartCmd); // // Setting the command // Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_SACT; AhciAndReg (AhciBar, Offset, 0); AhciOrReg (AhciBar, Offset, CmdSlotBit); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CI; AhciAndReg (AhciBar, Offset, 0); AhciOrReg (AhciBar, Offset, CmdSlotBit); return EFI_SUCCESS; } /** Do AHCI HBA reset. @param[in] AhciBar AHCI bar address. @param[in] Timeout The timeout Value of reset. @retval EFI_DEVICE_ERROR AHCI controller is failed to complete hardware reset. @retval EFI_TIMEOUT The reset operation is time out. @retval EFI_SUCCESS AHCI controller is reset successfully. **/ EFI_STATUS EFIAPI AhciReset ( IN UINT32 AhciBar, IN UINT64 Timeout ) { UINT32 Delay; UINT32 Value; UINT32 Capability; // // Collect AHCI controller information // Capability = AhciReadReg (AhciBar, EFI_AHCI_CAPABILITY_OFFSET); // // Enable AE before accessing any AHCI registers if Supports AHCI Mode Only is not set // if ((Capability & EFI_AHCI_CAP_SAM) == 0) { AhciOrReg (AhciBar, EFI_AHCI_GHC_OFFSET, EFI_AHCI_GHC_ENABLE); } AhciOrReg (AhciBar, EFI_AHCI_GHC_OFFSET, EFI_AHCI_GHC_RESET); Delay = (UINT32) (DivU64x32(Timeout, 1000) + 1); do { Value = AhciReadReg(AhciBar, EFI_AHCI_GHC_OFFSET); if ((Value & EFI_AHCI_GHC_RESET) == 0) { return EFI_SUCCESS; } // // Stall for 100 microseconds. // MicroSecondDelay(100); Delay--; } while (Delay > 0); return EFI_TIMEOUT; } /** Send Buffer cmd to specific device. @param[in] AhciContext The pointer to the AHCI_CONTEXT. @param[in] Port The port number of attached ATA device. @param[in] PortMultiplier The port number of port multiplier of attached ATA device. @param[in, out] Buffer The Data Buffer to store IDENTIFY PACKET Data. @retval EFI_DEVICE_ERROR The cmd abort with error occurs. @retval EFI_TIMEOUT The operation is time out. @retval EFI_UNSUPPORTED The device is not ready for executing. @retval EFI_SUCCESS The cmd executes successfully. **/ EFI_STATUS EFIAPI AhciIdentify ( IN AHCI_CONTEXT *AhciContext, IN UINT8 Port, IN UINT8 PortMultiplier, IN OUT ATA_IDENTIFY_DATA *Buffer ) { EFI_STATUS Status; EFI_ATA_COMMAND_BLOCK AtaCommandBlock; if (AhciContext == NULL || Buffer == NULL) { return EFI_INVALID_PARAMETER; } ZeroMem (&AtaCommandBlock, sizeof (EFI_ATA_COMMAND_BLOCK)); AtaCommandBlock.AtaCommand = ATA_CMD_IDENTIFY_DRIVE; AtaCommandBlock.AtaSectorCount = 1; Status = AhciPioTransfer ( AhciContext, Port, PortMultiplier, NULL, 0, TRUE, &AtaCommandBlock, NULL, Buffer, sizeof (ATA_IDENTIFY_DATA), ATA_TIMEOUT ); return Status; } /** Allocate transfer-related data struct which is used at AHCI mode. @param[in, out] AhciContext The pointer to the AHCI_CONTEXT. @retval EFI_OUT_OF_RESOURCE No enough resource. @retval EFI_SUCCESS Successful to allocate resource. **/ EFI_STATUS EFIAPI AhciAllocateResource ( IN OUT AHCI_CONTEXT *AhciContext ) { EFI_STATUS Status; EFI_AHCI_REGISTERS *AhciRegisters; EFI_PHYSICAL_ADDRESS DeviceAddress; VOID *Base; VOID *Mapping; AhciRegisters = &AhciContext->AhciRegisters; // // Allocate resources required by AHCI host controller. // Status = IoMmuAllocateBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_RECEIVED_FIS)), &Base, &DeviceAddress, &Mapping ); if (EFI_ERROR (Status)) { return EFI_OUT_OF_RESOURCES; } ASSERT (DeviceAddress == ((EFI_PHYSICAL_ADDRESS) (UINTN) Base)); AhciRegisters->AhciRFisMapping = Mapping; AhciRegisters->AhciRFis = Base; ZeroMem (AhciRegisters->AhciRFis, EFI_PAGE_SIZE * EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_RECEIVED_FIS))); Status = IoMmuAllocateBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_LIST)), &Base, &DeviceAddress, &Mapping ); if (EFI_ERROR (Status)) { IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_RECEIVED_FIS)), AhciRegisters->AhciRFis, AhciRegisters->AhciRFisMapping ); AhciRegisters->AhciRFis = NULL; return EFI_OUT_OF_RESOURCES; } ASSERT (DeviceAddress == ((EFI_PHYSICAL_ADDRESS) (UINTN) Base)); AhciRegisters->AhciCmdListMapping = Mapping; AhciRegisters->AhciCmdList = Base; ZeroMem (AhciRegisters->AhciCmdList, EFI_PAGE_SIZE * EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_LIST))); Status = IoMmuAllocateBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_TABLE)), &Base, &DeviceAddress, &Mapping ); if (EFI_ERROR (Status)) { IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_LIST)), AhciRegisters->AhciCmdList, AhciRegisters->AhciCmdListMapping ); AhciRegisters->AhciCmdList = NULL; IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_RECEIVED_FIS)), AhciRegisters->AhciRFis, AhciRegisters->AhciRFisMapping ); AhciRegisters->AhciRFis = NULL; return EFI_OUT_OF_RESOURCES; } ASSERT (DeviceAddress == ((EFI_PHYSICAL_ADDRESS) (UINTN) Base)); AhciRegisters->AhciCommandTableMapping = Mapping; AhciRegisters->AhciCommandTable = Base; ZeroMem (AhciRegisters->AhciCommandTable, EFI_PAGE_SIZE * EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_TABLE))); // // Allocate resources for data transfer. // Status = IoMmuAllocateBuffer ( EFI_SIZE_TO_PAGES (HDD_PAYLOAD), &Base, &DeviceAddress, &Mapping ); if (EFI_ERROR (Status)) { IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_RECEIVED_FIS)), AhciRegisters->AhciCommandTable, AhciRegisters->AhciCommandTableMapping ); AhciRegisters->AhciCommandTable = NULL; IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_LIST)), AhciRegisters->AhciCmdList, AhciRegisters->AhciCmdListMapping ); AhciRegisters->AhciCmdList = NULL; IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_RECEIVED_FIS)), AhciRegisters->AhciRFis, AhciRegisters->AhciRFisMapping ); AhciRegisters->AhciRFis = NULL; return EFI_OUT_OF_RESOURCES; } ASSERT (DeviceAddress == ((EFI_PHYSICAL_ADDRESS) (UINTN) Base)); AhciContext->BufferMapping = Mapping; AhciContext->Buffer = Base; ZeroMem (AhciContext->Buffer, EFI_PAGE_SIZE * EFI_SIZE_TO_PAGES (HDD_PAYLOAD)); DEBUG (( DEBUG_INFO, "%a() AhciContext 0x%x 0x%x 0x%x 0x%x\n", __FUNCTION__, AhciContext->Buffer, AhciRegisters->AhciRFis, AhciRegisters->AhciCmdList, AhciRegisters->AhciCommandTable )); return EFI_SUCCESS; } /** Free allocated transfer-related data struct which is used at AHCI mode. @param[in, out] AhciContext The pointer to the AHCI_CONTEXT. **/ VOID EFIAPI AhciFreeResource ( IN OUT AHCI_CONTEXT *AhciContext ) { EFI_AHCI_REGISTERS *AhciRegisters; AhciRegisters = &AhciContext->AhciRegisters; if (AhciContext->Buffer != NULL) { IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (HDD_PAYLOAD), AhciContext->Buffer, AhciContext->BufferMapping ); AhciContext->Buffer = NULL; } if (AhciRegisters->AhciCommandTable != NULL) { IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_TABLE)), AhciRegisters->AhciCommandTable, AhciRegisters->AhciCommandTableMapping ); AhciRegisters->AhciCommandTable = NULL; } if (AhciRegisters->AhciCmdList != NULL) { IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_COMMAND_LIST)), AhciRegisters->AhciCmdList, AhciRegisters->AhciCmdListMapping ); AhciRegisters->AhciCmdList = NULL; } if (AhciRegisters->AhciRFis != NULL) { IoMmuFreeBuffer ( EFI_SIZE_TO_PAGES (sizeof (EFI_AHCI_RECEIVED_FIS)), AhciRegisters->AhciRFis, AhciRegisters->AhciRFisMapping ); AhciRegisters->AhciRFis = NULL; } } /** Initialize ATA host controller at AHCI mode. The function is designed to initialize ATA host controller. @param[in] AhciContext The pointer to the AHCI_CONTEXT. @param[in] Port The port number to do initialization. **/ EFI_STATUS EFIAPI AhciModeInitialize ( IN AHCI_CONTEXT *AhciContext, IN UINT8 Port ) { EFI_STATUS Status; EFI_AHCI_REGISTERS *AhciRegisters; UINT32 AhciBar; UINT32 Capability; UINT32 Offset; UINT32 Data; UINT32 PhyDetectDelay; AhciRegisters = &AhciContext->AhciRegisters; AhciBar = AhciContext->AhciBar; Status = AhciReset (AhciBar, ATA_TIMEOUT); if (EFI_ERROR (Status)) { return Status; } // // Collect AHCI controller information // Capability = AhciReadReg (AhciBar, EFI_AHCI_CAPABILITY_OFFSET); // // Enable AE before accessing any AHCI registers if Supports AHCI Mode Only is not set // if ((Capability & EFI_AHCI_CAP_SAM) == 0) { AhciOrReg (AhciBar, EFI_AHCI_GHC_OFFSET, EFI_AHCI_GHC_ENABLE); } Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_FB; AhciWriteReg (AhciBar, Offset, (UINT32)(UINTN)AhciRegisters->AhciRFis); // // Single task envrionment, we only use one command table for all port // Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CLB; AhciWriteReg (AhciBar, Offset, (UINT32)(UINTN)AhciRegisters->AhciCmdList); Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_CMD; Data = AhciReadReg (AhciBar, Offset); if ((Data & EFI_AHCI_PORT_CMD_CPD) != 0) { AhciOrReg (AhciBar, Offset, EFI_AHCI_PORT_CMD_POD); } if ((Capability & BIT27) != 0) { AhciOrReg (AhciBar, Offset, EFI_AHCI_PORT_CMD_SUD); } // // Disable aggressive power management. // Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_SCTL; AhciOrReg (AhciBar, Offset, EFI_AHCI_PORT_SCTL_IPM_INIT); // // Disable the reporting of the corresponding interrupt to system software. // Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_IE; AhciAndReg (AhciBar, Offset, 0); Status = AhciEnableFisReceive ( AhciBar, Port, 5000000 ); ASSERT_EFI_ERROR (Status); if (EFI_ERROR (Status)) { return Status; } // // According to SATA1.0a spec section 5.2, we need to wait for PxTFD.BSY and PxTFD.DRQ // and PxTFD.ERR to be zero. The maximum wait time is 16s which is defined at ATA spec. // PhyDetectDelay = 16 * 1000; do { Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_SERR; if (AhciReadReg(AhciBar, Offset) != 0) { AhciWriteReg (AhciBar, Offset, AhciReadReg(AhciBar, Offset)); } Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_TFD; Data = AhciReadReg (AhciBar, Offset) & EFI_AHCI_PORT_TFD_MASK; if (Data == 0) { break; } MicroSecondDelay (1000); PhyDetectDelay--; } while (PhyDetectDelay > 0); if (PhyDetectDelay == 0) { return EFI_NOT_FOUND; } Offset = EFI_AHCI_PORT_START + Port * EFI_AHCI_PORT_REG_WIDTH + EFI_AHCI_PORT_SIG; Status = AhciWaitMmioSet ( AhciBar, Offset, 0x0000FFFF, 0x00000101, 160000000 ); if (EFI_ERROR (Status)) { return Status; } return Status; }