/** @file | |
Copyright (c) 2004 - 2016, 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 that 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 <PiDxe.h> | |
#include <Library/FlashDeviceLib.h> | |
#include <Library/DebugLib.h> | |
#include <Library/BaseLib.h> | |
#include <Library/UefiBootServicesTableLib.h> | |
#include <Library/UefiRuntimeServicesTableLib.h> | |
#include <Library/BaseMemoryLib.h> | |
#include <Library/UefiRuntimeLib.h> | |
#include <Protocol/SmmBase2.h> | |
#include <Guid/EventGroup.h> | |
#include "SpiChipDefinitions.h" | |
UINTN FlashDeviceBase = FLASH_DEVICE_BASE_ADDRESS; | |
EFI_SPI_PROTOCOL *mSpiProtocol = NULL; | |
EFI_STATUS | |
SpiFlashErase ( | |
UINT8 *BaseAddress, | |
UINTN NumBytes | |
) | |
{ | |
EFI_STATUS Status = EFI_SUCCESS; | |
UINT32 SectorSize; | |
UINT32 SpiAddress; | |
SpiAddress = (UINT32)(UINTN)(BaseAddress) - (UINT32)FlashDeviceBase; | |
SectorSize = SECTOR_SIZE_4KB; | |
while ( (NumBytes > 0) && (NumBytes <= MAX_FWH_SIZE) ) { | |
Status = mSpiProtocol->Execute ( | |
mSpiProtocol, | |
SPI_SERASE, | |
SPI_WREN, | |
FALSE, | |
TRUE, | |
FALSE, | |
(UINT32) SpiAddress, | |
0, | |
NULL, | |
EnumSpiRegionBios | |
); | |
if (EFI_ERROR (Status)) { | |
break; | |
} | |
SpiAddress += SectorSize; | |
NumBytes -= SectorSize; | |
} | |
return Status; | |
} | |
EFI_STATUS | |
SpiFlashBlockErase ( | |
UINT8 *BaseAddress, | |
UINTN NumBytes | |
) | |
{ | |
EFI_STATUS Status = EFI_SUCCESS; | |
UINT32 SectorSize; | |
UINT32 SpiAddress; | |
SpiAddress = (UINT32)(UINTN)(BaseAddress) - (UINT32)FlashDeviceBase; | |
SectorSize = SECTOR_SIZE_4KB; | |
while ( (NumBytes > 0) && (NumBytes <= MAX_FWH_SIZE) ) { | |
Status = mSpiProtocol->Execute ( | |
mSpiProtocol, | |
SPI_SERASE, | |
SPI_WREN, | |
FALSE, | |
TRUE, | |
FALSE, | |
(UINT32) SpiAddress, | |
0, | |
NULL, | |
EnumSpiRegionBios | |
); | |
if (EFI_ERROR (Status)) { | |
break; | |
} | |
SpiAddress += SectorSize; | |
NumBytes -= SectorSize; | |
} | |
return Status; | |
} | |
static | |
EFI_STATUS | |
SpiFlashWrite ( | |
UINT8 *DstBufferPtr, | |
UINT8 *Byte, | |
IN UINTN Length | |
) | |
{ | |
EFI_STATUS Status; | |
UINT32 NumBytes = (UINT32)Length; | |
UINT8* pBuf8 = Byte; | |
UINT32 SpiAddress; | |
SpiAddress = (UINT32)(UINTN)(DstBufferPtr) - (UINT32)FlashDeviceBase; | |
Status = mSpiProtocol->Execute ( | |
mSpiProtocol, | |
SPI_PROG, | |
SPI_WREN, | |
TRUE, | |
TRUE, | |
TRUE, | |
(UINT32)SpiAddress, | |
NumBytes, | |
pBuf8, | |
EnumSpiRegionBios | |
); | |
return Status; | |
} | |
/** | |
Read the Serial Flash Status Registers. | |
@param SpiStatus Pointer to a caller-allocated UINT8. On successful return, it contains the | |
status data read from the Serial Flash Status Register. | |
@retval EFI_SUCCESS Operation success, status is returned in SpiStatus. | |
@retval EFI_DEVICE_ERROR The block device is not functioning correctly and the operation failed. | |
**/ | |
EFI_STATUS | |
ReadStatusRegister ( | |
UINT8 *SpiStatus | |
) | |
{ | |
EFI_STATUS Status; | |
Status = mSpiProtocol->Execute ( | |
mSpiProtocol, | |
SPI_RDSR, | |
SPI_WREN, | |
TRUE, | |
FALSE, | |
FALSE, | |
0, | |
1, | |
SpiStatus, | |
EnumSpiRegionBios | |
); | |
return Status; | |
} | |
EFI_STATUS | |
SpiFlashLock ( | |
IN UINT8 *BaseAddress, | |
IN UINTN NumBytes, | |
IN BOOLEAN Lock | |
) | |
{ | |
EFI_STATUS Status; | |
UINT8 SpiData; | |
UINT8 SpiStatus; | |
if (Lock) { | |
SpiData = SF_SR_WPE; | |
} else { | |
SpiData = 0; | |
} | |
// | |
// Always disable block protection to workaround tool issue. | |
// Feature may be re-enabled in a future bios. | |
// | |
SpiData = 0; | |
Status = mSpiProtocol->Execute ( | |
mSpiProtocol, | |
SPI_WRSR, | |
SPI_EWSR, | |
TRUE, | |
TRUE, | |
TRUE, | |
0, | |
1, | |
&SpiData, | |
EnumSpiRegionBios | |
); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
Status = ReadStatusRegister (&SpiStatus); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if ((SpiStatus & SpiData) != SpiData) { | |
Status = EFI_DEVICE_ERROR; | |
} | |
return Status; | |
} | |
/** | |
Read NumBytes bytes of data from the address specified by | |
PAddress into Buffer. | |
@param[in] PAddress The starting physical address of the read. | |
@param[in,out] NumBytes On input, the number of bytes to read. On output, the number | |
of bytes actually read. | |
@param[out] Buffer The destination data buffer for the read. | |
@retval EFI_SUCCESS. Opertion is successful. | |
@retval EFI_DEVICE_ERROR If there is any device errors. | |
**/ | |
EFI_STATUS | |
EFIAPI | |
LibFvbFlashDeviceRead ( | |
IN UINTN PAddress, | |
IN OUT UINTN *NumBytes, | |
OUT UINT8 *Buffer | |
) | |
{ | |
CopyMem(Buffer, (VOID*)PAddress, *NumBytes); | |
return EFI_SUCCESS; | |
} | |
/** | |
Write NumBytes bytes of data from Buffer to the address specified by | |
PAddresss. | |
@param[in] PAddress The starting physical address of the write. | |
@param[in,out] NumBytes On input, the number of bytes to write. On output, | |
the actual number of bytes written. | |
@param[in] Buffer The source data buffer for the write. | |
@retval EFI_SUCCESS. Opertion is successful. | |
@retval EFI_DEVICE_ERROR If there is any device errors. | |
**/ | |
EFI_STATUS | |
EFIAPI | |
LibFvbFlashDeviceWrite ( | |
IN UINTN PAddress, | |
IN OUT UINTN *NumBytes, | |
IN UINT8 *Buffer | |
) | |
{ | |
EFI_STATUS Status; | |
Status = SpiFlashWrite((UINT8 *)PAddress, Buffer, *NumBytes); | |
return Status; | |
} | |
/** | |
Erase the block staring at PAddress. | |
@param[in] PAddress The starting physical address of the block to be erased. | |
This library assume that caller garantee that the PAddress | |
is at the starting address of this block. | |
@param[in] LbaLength The length of the logical block to be erased. | |
@retval EFI_SUCCESS. Opertion is successful. | |
@retval EFI_DEVICE_ERROR If there is any device errors. | |
**/ | |
EFI_STATUS | |
EFIAPI | |
LibFvbFlashDeviceBlockErase ( | |
IN UINTN PAddress, | |
IN UINTN LbaLength | |
) | |
{ | |
EFI_STATUS Status; | |
Status = SpiFlashBlockErase((UINT8 *)PAddress, LbaLength); | |
return Status; | |
} | |
/** | |
Lock or unlock the block staring at PAddress. | |
@param[in] PAddress The starting physical address of region to be (un)locked. | |
@param[in] LbaLength The length of the logical block to be erased. | |
@param[in] Lock TRUE to lock. FALSE to unlock. | |
@retval EFI_SUCCESS. Opertion is successful. | |
@retval EFI_DEVICE_ERROR If there is any device errors. | |
**/ | |
EFI_STATUS | |
EFIAPI | |
LibFvbFlashDeviceBlockLock ( | |
IN UINTN PAddress, | |
IN UINTN LbaLength, | |
IN BOOLEAN Lock | |
) | |
{ | |
EFI_STATUS Status; | |
Status = SpiFlashLock((UINT8*)PAddress, LbaLength, Lock); | |
return Status; | |
} | |
VOID | |
EFIAPI | |
LibFvbFlashDeviceVirtualAddressChangeNotifyEvent ( | |
IN EFI_EVENT Event, | |
IN VOID *Context | |
) | |
{ | |
gRT->ConvertPointer (0, (VOID **) &mSpiProtocol); | |
gRT->ConvertPointer (0, (VOID **) &FlashDeviceBase); | |
} | |
/** | |
The library constructuor. | |
The function does the necessary initialization work for this library | |
instance. Please put all initialization works in it. | |
@param[in] ImageHandle The firmware allocated handle for the UEFI image. | |
@param[in] SystemTable A pointer to the EFI system table. | |
@retval EFI_SUCCESS The function always return EFI_SUCCESS for now. | |
It will ASSERT on error for debug version. | |
@retval EFI_ERROR Please reference LocateProtocol for error code details. | |
**/ | |
EFI_STATUS | |
EFIAPI | |
LibFvbFlashDeviceSupportInit ( | |
IN EFI_HANDLE ImageHandle, | |
IN EFI_SYSTEM_TABLE *SystemTable | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_EVENT Event; | |
UINT8 SfId[3]; | |
UINT8 FlashIndex; | |
UINT8 SpiReadError; | |
UINT8 SpiNotMatchError; | |
EFI_SMM_BASE2_PROTOCOL *SmmBase; | |
BOOLEAN InSmm; | |
SpiReadError = 0x00; | |
SpiNotMatchError = 0x00; | |
InSmm = FALSE; | |
Status = gBS->LocateProtocol ( | |
&gEfiSmmBase2ProtocolGuid, | |
NULL, | |
(void **)&SmmBase | |
); | |
if (!EFI_ERROR(Status)) { | |
Status = SmmBase->InSmm(SmmBase, &InSmm); | |
if (EFI_ERROR(Status)) { | |
InSmm = FALSE; | |
} | |
} | |
if (!InSmm) { | |
Status = gBS->LocateProtocol ( | |
&gEfiSpiProtocolGuid, | |
NULL, | |
(VOID **)&mSpiProtocol | |
); | |
ASSERT_EFI_ERROR (Status); | |
Status = gBS->CreateEventEx ( | |
EVT_NOTIFY_SIGNAL, | |
TPL_NOTIFY, | |
LibFvbFlashDeviceVirtualAddressChangeNotifyEvent, | |
NULL, | |
&gEfiEventVirtualAddressChangeGuid, | |
&Event | |
); | |
ASSERT_EFI_ERROR (Status); | |
} else { | |
Status = gBS->LocateProtocol ( | |
&gEfiSmmSpiProtocolGuid, | |
NULL, | |
(VOID **)&mSpiProtocol | |
); | |
ASSERT_EFI_ERROR (Status); | |
} | |
for (FlashIndex = EnumSpiFlashW25Q64; FlashIndex < EnumSpiFlashMax; FlashIndex++) { | |
Status = mSpiProtocol->Init (mSpiProtocol, &(mInitTable[FlashIndex])); | |
if (!EFI_ERROR (Status)) { | |
// | |
// Read Vendor/Device IDs to check if the driver supports the Serial Flash device. | |
// | |
Status = mSpiProtocol->Execute ( | |
mSpiProtocol, | |
SPI_READ_ID, | |
SPI_WREN, | |
TRUE, | |
FALSE, | |
FALSE, | |
0, | |
3, | |
SfId, | |
EnumSpiRegionAll | |
); | |
if (!EFI_ERROR (Status)) { | |
if ((SfId[0] == mInitTable[FlashIndex].VendorId) && | |
(SfId[1] == mInitTable[FlashIndex].DeviceId0) && | |
(SfId[2] == mInitTable[FlashIndex].DeviceId1)) { | |
// | |
// Found a matching SPI device, FlashIndex now contains flash device. | |
// | |
DEBUG ((DEBUG_ERROR, "OK - Found SPI Flash Type in SPI Flash Driver, Device Type ID 0 = 0x%02x!\n", mInitTable[FlashIndex].DeviceId0)); | |
DEBUG ((DEBUG_ERROR, "Device Type ID 1 = 0x%02x!\n", mInitTable[FlashIndex].DeviceId1)); | |
if (mInitTable[FlashIndex].BiosStartOffset == (UINTN) (-1)) { | |
DEBUG ((DEBUG_ERROR, "ERROR - The size of BIOS image is bigger than SPI Flash device!\n")); | |
CpuDeadLoop (); | |
} | |
break; | |
} else { | |
SpiNotMatchError++; | |
} | |
} else { | |
SpiReadError++; | |
} | |
} | |
} | |
DEBUG ((DEBUG_ERROR, "SPI flash chip VID = 0x%X, DID0 = 0x%X, DID1 = 0x%X\n", SfId[0], SfId[1], SfId[2])); | |
if (FlashIndex < EnumSpiFlashMax) { | |
return EFI_SUCCESS; | |
} else { | |
if (SpiReadError != 0) { | |
DEBUG ((DEBUG_ERROR, "ERROR - SPI Read ID execution failed! Error Count = %d\n", SpiReadError)); | |
} | |
else { | |
if (SpiNotMatchError != 0) { | |
DEBUG ((DEBUG_ERROR, "ERROR - No supported SPI flash chip found! Error Count = %d\n", SpiNotMatchError)); | |
DEBUG ((DEBUG_ERROR, "SPI flash chip VID = 0x%X, DID0 = 0x%X, DID1 = 0x%X\n", SfId[0], SfId[1], SfId[2])); | |
} | |
} | |
return EFI_UNSUPPORTED; | |
} | |
} | |