| /** @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/BaseMemoryLib.h> | |
| #include <Guid/EventGroup.h> | |
| #include <Library/SpiFlash.H> | |
| #define FLASH_SIZE 0x400000 | |
| #define FLASH_DEVICE_BASE_ADDRESS (0xFFFFFFFF-FLASH_SIZE+1) | |
| 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_64KB; | |
| while ( (NumBytes > 0) && (NumBytes <= MAX_FWH_SIZE) ) { | |
| Status = mSpiProtocol->Execute ( | |
| mSpiProtocol, | |
| SPI_BERASE, | |
| 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; | |
| } | |