2013-01-25 12:28:06 +01:00
|
|
|
/** @file NorFlashDxe.c
|
|
|
|
|
2014-04-08 20:02:32 +02:00
|
|
|
Copyright (c) 2011 - 2014, ARM Ltd. All rights reserved.<BR>
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
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 <Library/UefiLib.h>
|
|
|
|
#include <Library/BaseMemoryLib.h>
|
|
|
|
#include <Library/MemoryAllocationLib.h>
|
|
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
|
|
#include <Library/PcdLib.h>
|
|
|
|
|
|
|
|
#include "NorFlashDxe.h"
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Global variable declarations
|
|
|
|
//
|
|
|
|
NOR_FLASH_INSTANCE **mNorFlashInstances;
|
|
|
|
|
|
|
|
NOR_FLASH_INSTANCE mNorFlashInstanceTemplate = {
|
|
|
|
NOR_FLASH_SIGNATURE, // Signature
|
|
|
|
NULL, // Handle ... NEED TO BE FILLED
|
|
|
|
|
|
|
|
FALSE, // Initialized
|
|
|
|
NULL, // Initialize
|
|
|
|
|
|
|
|
0, // DeviceBaseAddress ... NEED TO BE FILLED
|
|
|
|
0, // RegionBaseAddress ... NEED TO BE FILLED
|
|
|
|
0, // Size ... NEED TO BE FILLED
|
|
|
|
0, // StartLba
|
|
|
|
|
2011-02-01 06:41:42 +01:00
|
|
|
{
|
|
|
|
EFI_BLOCK_IO_PROTOCOL_REVISION2, // Revision
|
|
|
|
NULL, // Media ... NEED TO BE FILLED
|
|
|
|
NorFlashBlockIoReset, // Reset;
|
2013-01-25 12:28:06 +01:00
|
|
|
NorFlashBlockIoReadBlocks, // ReadBlocks
|
|
|
|
NorFlashBlockIoWriteBlocks, // WriteBlocks
|
2011-02-01 06:41:42 +01:00
|
|
|
NorFlashBlockIoFlushBlocks // FlushBlocks
|
2013-01-25 12:28:06 +01:00
|
|
|
}, // BlockIoProtocol
|
|
|
|
|
2011-02-01 06:41:42 +01:00
|
|
|
{
|
|
|
|
0, // MediaId ... NEED TO BE FILLED
|
|
|
|
FALSE, // RemovableMedia
|
|
|
|
TRUE, // MediaPresent
|
|
|
|
FALSE, // LogicalPartition
|
|
|
|
FALSE, // ReadOnly
|
|
|
|
FALSE, // WriteCaching;
|
|
|
|
0, // BlockSize ... NEED TO BE FILLED
|
|
|
|
4, // IoAlign
|
|
|
|
0, // LastBlock ... NEED TO BE FILLED
|
|
|
|
0, // LowestAlignedLba
|
|
|
|
1, // LogicalBlocksPerPhysicalBlock
|
2013-01-25 12:28:06 +01:00
|
|
|
}, //Media;
|
|
|
|
|
|
|
|
FALSE, // SupportFvb ... NEED TO BE FILLED
|
2011-02-01 06:41:42 +01:00
|
|
|
{
|
2013-01-25 12:28:06 +01:00
|
|
|
FvbGetAttributes, // GetAttributes
|
|
|
|
FvbSetAttributes, // SetAttributes
|
|
|
|
FvbGetPhysicalAddress, // GetPhysicalAddress
|
|
|
|
FvbGetBlockSize, // GetBlockSize
|
|
|
|
FvbRead, // Read
|
|
|
|
FvbWrite, // Write
|
2011-02-01 06:41:42 +01:00
|
|
|
FvbEraseBlocks, // EraseBlocks
|
|
|
|
NULL, //ParentHandle
|
2013-01-25 12:28:06 +01:00
|
|
|
}, // FvbProtoccol;
|
2014-04-08 20:02:32 +02:00
|
|
|
NULL, // FvbBuffer
|
2013-01-25 12:28:06 +01:00
|
|
|
{
|
|
|
|
{
|
|
|
|
{
|
|
|
|
HARDWARE_DEVICE_PATH,
|
|
|
|
HW_VENDOR_DP,
|
|
|
|
(UINT8)( sizeof(VENDOR_DEVICE_PATH) ),
|
|
|
|
(UINT8)((sizeof(VENDOR_DEVICE_PATH)) >> 8),
|
|
|
|
},
|
|
|
|
{ 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0 }, // GUID ... NEED TO BE FILLED
|
|
|
|
},
|
|
|
|
{
|
|
|
|
END_DEVICE_PATH_TYPE,
|
|
|
|
END_ENTIRE_DEVICE_PATH_SUBTYPE,
|
|
|
|
sizeof (EFI_DEVICE_PATH_PROTOCOL),
|
|
|
|
0
|
|
|
|
}
|
|
|
|
} // DevicePath
|
|
|
|
};
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashCreateInstance (
|
|
|
|
IN UINTN NorFlashDeviceBase,
|
|
|
|
IN UINTN NorFlashRegionBase,
|
|
|
|
IN UINTN NorFlashSize,
|
|
|
|
IN UINT32 MediaId,
|
|
|
|
IN UINT32 BlockSize,
|
|
|
|
IN BOOLEAN SupportFvb,
|
|
|
|
IN CONST GUID *NorFlashGuid,
|
|
|
|
OUT NOR_FLASH_INSTANCE** NorFlashInstance
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
NOR_FLASH_INSTANCE* Instance;
|
|
|
|
|
|
|
|
ASSERT(NorFlashInstance != NULL);
|
|
|
|
|
2014-04-08 20:02:32 +02:00
|
|
|
Instance = AllocateRuntimeCopyPool (sizeof(NOR_FLASH_INSTANCE),&mNorFlashInstanceTemplate);
|
2013-01-25 12:28:06 +01:00
|
|
|
if (Instance == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
Instance->DeviceBaseAddress = NorFlashDeviceBase;
|
|
|
|
Instance->RegionBaseAddress = NorFlashRegionBase;
|
|
|
|
Instance->Size = NorFlashSize;
|
|
|
|
|
|
|
|
Instance->BlockIoProtocol.Media = &Instance->Media;
|
|
|
|
Instance->Media.MediaId = MediaId;
|
|
|
|
Instance->Media.BlockSize = BlockSize;
|
|
|
|
Instance->Media.LastBlock = (NorFlashSize / BlockSize)-1;
|
2011-02-01 06:41:42 +01:00
|
|
|
|
2014-04-08 20:02:32 +02:00
|
|
|
CopyGuid (&Instance->DevicePath.Vendor.Guid, NorFlashGuid);
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
if (SupportFvb) {
|
|
|
|
Instance->SupportFvb = TRUE;
|
|
|
|
Instance->Initialize = NorFlashFvbInitialize;
|
2014-04-08 20:02:32 +02:00
|
|
|
Instance->FvbBuffer = AllocateRuntimePool (BlockSize);;
|
|
|
|
if (Instance->FvbBuffer == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
Status = gBS->InstallMultipleProtocolInterfaces (
|
|
|
|
&Instance->Handle,
|
|
|
|
&gEfiDevicePathProtocolGuid, &Instance->DevicePath,
|
|
|
|
&gEfiBlockIoProtocolGuid, &Instance->BlockIoProtocol,
|
|
|
|
&gEfiFirmwareVolumeBlockProtocolGuid, &Instance->FvbProtocol,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
if (EFI_ERROR(Status)) {
|
2014-04-08 20:02:32 +02:00
|
|
|
FreePool (Instance);
|
2013-01-25 12:28:06 +01:00
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Instance->Initialized = TRUE;
|
|
|
|
|
|
|
|
Status = gBS->InstallMultipleProtocolInterfaces (
|
|
|
|
&Instance->Handle,
|
|
|
|
&gEfiDevicePathProtocolGuid, &Instance->DevicePath,
|
|
|
|
&gEfiBlockIoProtocolGuid, &Instance->BlockIoProtocol,
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
if (EFI_ERROR(Status)) {
|
2014-04-08 20:02:32 +02:00
|
|
|
FreePool (Instance);
|
2013-01-25 12:28:06 +01:00
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
}
|
2011-02-01 06:41:42 +01:00
|
|
|
|
|
|
|
*NorFlashInstance = Instance;
|
2013-01-25 12:28:06 +01:00
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
UINT32
|
|
|
|
NorFlashReadStatusRegister (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN SR_Address
|
|
|
|
)
|
|
|
|
{
|
|
|
|
// Prepare to read the status register
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_READ_STATUS_REGISTER);
|
|
|
|
return MmioRead32 (Instance->DeviceBaseAddress);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOLEAN
|
|
|
|
NorFlashBlockIsLocked (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN BlockAddress
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 LockStatus;
|
|
|
|
BOOLEAN BlockIsLocked;
|
|
|
|
|
|
|
|
BlockIsLocked = TRUE;
|
|
|
|
|
|
|
|
// Send command for reading device id
|
|
|
|
SEND_NOR_COMMAND (BlockAddress, 2, P30_CMD_READ_DEVICE_ID);
|
|
|
|
|
|
|
|
// Read block lock status
|
|
|
|
LockStatus = MmioRead32 (CREATE_NOR_ADDRESS(BlockAddress, 2));
|
|
|
|
|
|
|
|
// Decode block lock status
|
|
|
|
LockStatus = FOLD_32BIT_INTO_16BIT(LockStatus);
|
|
|
|
|
|
|
|
if ((LockStatus & 0x2) != 0) {
|
|
|
|
DEBUG((EFI_D_ERROR, "NorFlashBlockIsLocked: WARNING: Block LOCKED DOWN\n"));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((LockStatus & 0x1) == 0) {
|
|
|
|
// This means the block is unlocked
|
|
|
|
DEBUG((DEBUG_BLKIO, "UnlockSingleBlock: Block 0x%08x unlocked\n", BlockAddress));
|
|
|
|
BlockIsLocked = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return BlockIsLocked;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashUnlockSingleBlock (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN BlockAddress
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status = EFI_SUCCESS;
|
|
|
|
UINT32 LockStatus;
|
|
|
|
|
|
|
|
// Raise the Task Priority Level to TPL_NOTIFY to serialise all its operations
|
|
|
|
// and to protect shared data structures.
|
|
|
|
|
|
|
|
if (FeaturePcdGet (PcdNorFlashCheckBlockLocked) == TRUE) {
|
|
|
|
do {
|
|
|
|
// Request a lock setup
|
|
|
|
SEND_NOR_COMMAND (BlockAddress, 0, P30_CMD_LOCK_BLOCK_SETUP);
|
|
|
|
|
|
|
|
// Request an unlock
|
|
|
|
SEND_NOR_COMMAND (BlockAddress, 0, P30_CMD_UNLOCK_BLOCK);
|
|
|
|
|
|
|
|
// Send command for reading device id
|
|
|
|
SEND_NOR_COMMAND (BlockAddress, 2, P30_CMD_READ_DEVICE_ID);
|
|
|
|
|
|
|
|
// Read block lock status
|
|
|
|
LockStatus = MmioRead32 (CREATE_NOR_ADDRESS(BlockAddress, 2));
|
|
|
|
|
|
|
|
// Decode block lock status
|
|
|
|
LockStatus = FOLD_32BIT_INTO_16BIT(LockStatus);
|
|
|
|
} while ((LockStatus & 0x1) == 1);
|
|
|
|
} else {
|
|
|
|
// Request a lock setup
|
|
|
|
SEND_NOR_COMMAND (BlockAddress, 0, P30_CMD_LOCK_BLOCK_SETUP);
|
|
|
|
|
|
|
|
// Request an unlock
|
|
|
|
SEND_NOR_COMMAND (BlockAddress, 0, P30_CMD_UNLOCK_BLOCK);
|
|
|
|
|
|
|
|
// Wait until the status register gives us the all clear
|
|
|
|
do {
|
|
|
|
LockStatus = NorFlashReadStatusRegister (Instance, BlockAddress);
|
|
|
|
} while ((LockStatus & P30_SR_BIT_WRITE) != P30_SR_BIT_WRITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put device back into Read Array mode
|
|
|
|
SEND_NOR_COMMAND (BlockAddress, 0, P30_CMD_READ_ARRAY);
|
|
|
|
|
|
|
|
DEBUG((DEBUG_BLKIO, "UnlockSingleBlock: BlockAddress=0x%08x, Exit Status = \"%r\".\n", BlockAddress, Status));
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashUnlockSingleBlockIfNecessary (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN BlockAddress
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
if (NorFlashBlockIsLocked (Instance, BlockAddress) == TRUE) {
|
|
|
|
Status = NorFlashUnlockSingleBlock (Instance, BlockAddress);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The following function presumes that the block has already been unlocked.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashEraseSingleBlock (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN BlockAddress
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 StatusRegister;
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
// Request a block erase and then confirm it
|
|
|
|
SEND_NOR_COMMAND(BlockAddress, 0, P30_CMD_BLOCK_ERASE_SETUP);
|
|
|
|
SEND_NOR_COMMAND(BlockAddress, 0, P30_CMD_BLOCK_ERASE_CONFIRM);
|
|
|
|
|
|
|
|
// Wait until the status register gives us the all clear
|
|
|
|
do {
|
|
|
|
StatusRegister = NorFlashReadStatusRegister (Instance, BlockAddress);
|
|
|
|
} while ((StatusRegister & P30_SR_BIT_WRITE) != P30_SR_BIT_WRITE);
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_VPP) {
|
|
|
|
DEBUG((EFI_D_ERROR,"EraseSingleBlock(BlockAddress=0x%08x: VPP Range Error\n", BlockAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((StatusRegister & (P30_SR_BIT_ERASE | P30_SR_BIT_PROGRAM)) == (P30_SR_BIT_ERASE | P30_SR_BIT_PROGRAM)) {
|
|
|
|
DEBUG((EFI_D_ERROR,"EraseSingleBlock(BlockAddress=0x%08x: Command Sequence Error\n", BlockAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_ERASE) {
|
|
|
|
DEBUG((EFI_D_ERROR,"EraseSingleBlock(BlockAddress=0x%08x: Block Erase Error StatusRegister:0x%X\n", BlockAddress, StatusRegister));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_BLOCK_LOCKED) {
|
|
|
|
// The debug level message has been reduced because a device lock might happen. In this case we just retry it ...
|
|
|
|
DEBUG((EFI_D_INFO,"EraseSingleBlock(BlockAddress=0x%08x: Block Locked Error\n", BlockAddress));
|
|
|
|
Status = EFI_WRITE_PROTECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
// Clear the Status Register
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_CLEAR_STATUS_REGISTER);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put device back into Read Array mode
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_READ_ARRAY);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The following function presumes that the block has already been unlocked.
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashUnlockAndEraseSingleBlock (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN BlockAddress
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN Index;
|
|
|
|
EFI_TPL OriginalTPL;
|
|
|
|
|
2014-04-08 20:02:32 +02:00
|
|
|
if (!EfiAtRuntime ()) {
|
|
|
|
// Raise TPL to TPL_HIGH to stop anyone from interrupting us.
|
|
|
|
OriginalTPL = gBS->RaiseTPL (TPL_HIGH_LEVEL);
|
|
|
|
} else {
|
|
|
|
// This initialization is only to prevent the compiler to complain about the
|
|
|
|
// use of uninitialized variables
|
|
|
|
OriginalTPL = TPL_HIGH_LEVEL;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
Index = 0;
|
|
|
|
// The block erase might fail a first time (SW bug ?). Retry it ...
|
|
|
|
do {
|
|
|
|
// Unlock the block if we have to
|
|
|
|
Status = NorFlashUnlockSingleBlockIfNecessary (Instance, BlockAddress);
|
|
|
|
if (!EFI_ERROR(Status)) {
|
|
|
|
Status = NorFlashEraseSingleBlock (Instance, BlockAddress);
|
|
|
|
}
|
|
|
|
Index++;
|
|
|
|
} while ((Index < NOR_FLASH_ERASE_RETRY) && (Status == EFI_WRITE_PROTECTED));
|
|
|
|
|
|
|
|
if (Index == NOR_FLASH_ERASE_RETRY) {
|
|
|
|
DEBUG((EFI_D_ERROR,"EraseSingleBlock(BlockAddress=0x%08x: Block Locked Error (try to erase %d times)\n", BlockAddress,Index));
|
|
|
|
}
|
|
|
|
|
2014-04-08 20:02:32 +02:00
|
|
|
if (!EfiAtRuntime ()) {
|
|
|
|
// Interruptions can resume.
|
|
|
|
gBS->RestoreTPL (OriginalTPL);
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashWriteSingleWord (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN WordAddress,
|
|
|
|
IN UINT32 WriteData
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 StatusRegister;
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
// Request a write single word command
|
|
|
|
SEND_NOR_COMMAND(WordAddress, 0, P30_CMD_WORD_PROGRAM_SETUP);
|
|
|
|
|
|
|
|
// Store the word into NOR Flash;
|
|
|
|
MmioWrite32 (WordAddress, WriteData);
|
|
|
|
|
|
|
|
// Wait for the write to complete and then check for any errors; i.e. check the Status Register
|
|
|
|
do {
|
|
|
|
// Prepare to read the status register
|
|
|
|
StatusRegister = NorFlashReadStatusRegister (Instance, WordAddress);
|
|
|
|
// The chip is busy while the WRITE bit is not asserted
|
|
|
|
} while ((StatusRegister & P30_SR_BIT_WRITE) != P30_SR_BIT_WRITE);
|
|
|
|
|
|
|
|
|
|
|
|
// Perform a full status check:
|
|
|
|
// Mask the relevant bits of Status Register.
|
|
|
|
// Everything should be zero, if not, we have a problem
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_VPP) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashWriteSingleWord(WordAddress:0x%X): VPP Range Error\n",WordAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_PROGRAM) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashWriteSingleWord(WordAddress:0x%X): Program Error\n",WordAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_BLOCK_LOCKED) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashWriteSingleWord(WordAddress:0x%X): Device Protect Error\n",WordAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!EFI_ERROR(Status)) {
|
|
|
|
// Clear the Status Register
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_CLEAR_STATUS_REGISTER);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put device back into Read Array mode
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_READ_ARRAY);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Writes data to the NOR Flash using the Buffered Programming method.
|
|
|
|
*
|
|
|
|
* The maximum size of the on-chip buffer is 32-words, because of hardware restrictions.
|
|
|
|
* Therefore this function will only handle buffers up to 32 words or 128 bytes.
|
|
|
|
* To deal with larger buffers, call this function again.
|
|
|
|
*
|
|
|
|
* This function presumes that both the TargetAddress and the TargetAddress+BufferSize
|
|
|
|
* exist entirely within the NOR Flash. Therefore these conditions will not be checked here.
|
|
|
|
*
|
|
|
|
* In buffered programming, if the target address not at the beginning of a 32-bit word boundary,
|
|
|
|
* then programming time is doubled and power consumption is increased.
|
|
|
|
* Therefore, it is a requirement to align buffer writes to 32-bit word boundaries.
|
|
|
|
* i.e. the last 4 bits of the target start address must be zero: 0x......00
|
|
|
|
*/
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashWriteBuffer (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN UINTN TargetAddress,
|
|
|
|
IN UINTN BufferSizeInBytes,
|
|
|
|
IN UINT32 *Buffer
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN BufferSizeInWords;
|
|
|
|
UINTN Count;
|
|
|
|
volatile UINT32 *Data;
|
|
|
|
UINTN WaitForBuffer;
|
|
|
|
BOOLEAN BufferAvailable;
|
|
|
|
UINT32 StatusRegister;
|
|
|
|
|
|
|
|
WaitForBuffer = MAX_BUFFERED_PROG_ITERATIONS;
|
|
|
|
BufferAvailable = FALSE;
|
|
|
|
|
|
|
|
// Check that the target address does not cross a 32-word boundary.
|
|
|
|
if ((TargetAddress & BOUNDARY_OF_32_WORDS) != 0) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check there are some data to program
|
|
|
|
if (BufferSizeInBytes == 0) {
|
|
|
|
return EFI_BUFFER_TOO_SMALL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the buffer size does not exceed the maximum hardware buffer size on chip.
|
|
|
|
if (BufferSizeInBytes > P30_MAX_BUFFER_SIZE_IN_BYTES) {
|
|
|
|
return EFI_BAD_BUFFER_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the buffer size is a multiple of 32-bit words
|
|
|
|
if ((BufferSizeInBytes % 4) != 0) {
|
|
|
|
return EFI_BAD_BUFFER_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pre-programming conditions checked, now start the algorithm.
|
|
|
|
|
|
|
|
// Prepare the data destination address
|
|
|
|
Data = (UINT32 *)TargetAddress;
|
|
|
|
|
|
|
|
// Check the availability of the buffer
|
|
|
|
do {
|
|
|
|
// Issue the Buffered Program Setup command
|
|
|
|
SEND_NOR_COMMAND(TargetAddress, 0, P30_CMD_BUFFERED_PROGRAM_SETUP);
|
|
|
|
|
|
|
|
// Read back the status register bit#7 from the same address
|
|
|
|
if (((*Data) & P30_SR_BIT_WRITE) == P30_SR_BIT_WRITE) {
|
|
|
|
BufferAvailable = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the loop counter
|
|
|
|
WaitForBuffer--;
|
|
|
|
|
|
|
|
} while ((WaitForBuffer > 0) && (BufferAvailable == FALSE));
|
|
|
|
|
|
|
|
// The buffer was not available for writing
|
|
|
|
if (WaitForBuffer == 0) {
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
goto EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
// From now on we work in 32-bit words
|
|
|
|
BufferSizeInWords = BufferSizeInBytes / (UINTN)4;
|
|
|
|
|
|
|
|
// Write the word count, which is (buffer_size_in_words - 1),
|
|
|
|
// because word count 0 means one word.
|
|
|
|
SEND_NOR_COMMAND(TargetAddress, 0, (BufferSizeInWords - 1));
|
|
|
|
|
|
|
|
// Write the data to the NOR Flash, advancing each address by 4 bytes
|
|
|
|
for(Count=0; Count < BufferSizeInWords; Count++, Data++, Buffer++) {
|
|
|
|
*Data = *Buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Issue the Buffered Program Confirm command, to start the programming operation
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_BUFFERED_PROGRAM_CONFIRM);
|
|
|
|
|
|
|
|
// Wait for the write to complete and then check for any errors; i.e. check the Status Register
|
|
|
|
do {
|
|
|
|
StatusRegister = NorFlashReadStatusRegister (Instance, TargetAddress);
|
|
|
|
// The chip is busy while the WRITE bit is not asserted
|
|
|
|
} while ((StatusRegister & P30_SR_BIT_WRITE) != P30_SR_BIT_WRITE);
|
|
|
|
|
|
|
|
|
|
|
|
// Perform a full status check:
|
|
|
|
// Mask the relevant bits of Status Register.
|
|
|
|
// Everything should be zero, if not, we have a problem
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_VPP) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashWriteBuffer(TargetAddress:0x%X): VPP Range Error\n", TargetAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_PROGRAM) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashWriteBuffer(TargetAddress:0x%X): Program Error\n", TargetAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (StatusRegister & P30_SR_BIT_BLOCK_LOCKED) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashWriteBuffer(TargetAddress:0x%X): Device Protect Error\n",TargetAddress));
|
|
|
|
Status = EFI_DEVICE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!EFI_ERROR(Status)) {
|
|
|
|
// Clear the Status Register
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_CLEAR_STATUS_REGISTER);
|
|
|
|
}
|
|
|
|
|
|
|
|
EXIT:
|
|
|
|
// Put device back into Read Array mode
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_READ_ARRAY);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashWriteSingleBlock (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN EFI_LBA Lba,
|
|
|
|
IN UINT32 *DataBuffer,
|
|
|
|
IN UINT32 BlockSizeInWords
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN WordAddress;
|
|
|
|
UINT32 WordIndex;
|
|
|
|
UINTN BufferIndex;
|
|
|
|
UINTN BlockAddress;
|
|
|
|
UINTN BuffersInBlock;
|
|
|
|
UINTN RemainingWords;
|
|
|
|
EFI_TPL OriginalTPL;
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
// Get the physical address of the block
|
|
|
|
BlockAddress = GET_NOR_BLOCK_ADDRESS (Instance->RegionBaseAddress, Lba, BlockSizeInWords * 4);
|
|
|
|
|
|
|
|
// Start writing from the first address at the start of the block
|
|
|
|
WordAddress = BlockAddress;
|
|
|
|
|
2014-04-08 20:02:32 +02:00
|
|
|
if (!EfiAtRuntime ()) {
|
|
|
|
// Raise TPL to TPL_HIGH to stop anyone from interrupting us.
|
|
|
|
OriginalTPL = gBS->RaiseTPL (TPL_HIGH_LEVEL);
|
|
|
|
} else {
|
|
|
|
// This initialization is only to prevent the compiler to complain about the
|
|
|
|
// use of uninitialized variables
|
|
|
|
OriginalTPL = TPL_HIGH_LEVEL;
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
Status = NorFlashUnlockAndEraseSingleBlock (Instance, BlockAddress);
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
DEBUG((EFI_D_ERROR, "WriteSingleBlock: ERROR - Failed to Unlock and Erase the single block at 0x%X\n", BlockAddress));
|
|
|
|
goto EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
// To speed up the programming operation, NOR Flash is programmed using the Buffered Programming method.
|
|
|
|
|
|
|
|
// Check that the address starts at a 32-word boundary, i.e. last 7 bits must be zero
|
|
|
|
if ((WordAddress & BOUNDARY_OF_32_WORDS) == 0x00) {
|
|
|
|
|
|
|
|
// First, break the entire block into buffer-sized chunks.
|
2013-03-12 01:44:30 +01:00
|
|
|
BuffersInBlock = (UINTN)(BlockSizeInWords * 4) / P30_MAX_BUFFER_SIZE_IN_BYTES;
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
// Then feed each buffer chunk to the NOR Flash
|
|
|
|
for(BufferIndex=0;
|
|
|
|
BufferIndex < BuffersInBlock;
|
|
|
|
BufferIndex++, WordAddress += P30_MAX_BUFFER_SIZE_IN_BYTES, DataBuffer += P30_MAX_BUFFER_SIZE_IN_WORDS
|
|
|
|
) {
|
|
|
|
Status = NorFlashWriteBuffer (Instance, WordAddress, P30_MAX_BUFFER_SIZE_IN_BYTES, DataBuffer);
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
goto EXIT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finally, finish off any remaining words that are less than the maximum size of the buffer
|
|
|
|
RemainingWords = BlockSizeInWords % P30_MAX_BUFFER_SIZE_IN_WORDS;
|
|
|
|
|
|
|
|
if(RemainingWords != 0) {
|
|
|
|
Status = NorFlashWriteBuffer (Instance, WordAddress, (RemainingWords * 4), DataBuffer);
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
goto EXIT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// For now, use the single word programming algorithm
|
|
|
|
// It is unlikely that the NOR Flash will exist in an address which falls within a 32 word boundary range,
|
|
|
|
// i.e. which ends in the range 0x......01 - 0x......7F.
|
|
|
|
for(WordIndex=0; WordIndex<BlockSizeInWords; WordIndex++, DataBuffer++, WordAddress = WordAddress + 4) {
|
|
|
|
Status = NorFlashWriteSingleWord (Instance, WordAddress, *DataBuffer);
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
goto EXIT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EXIT:
|
2014-04-08 20:02:32 +02:00
|
|
|
if (!EfiAtRuntime ()) {
|
|
|
|
// Interruptions can resume.
|
|
|
|
gBS->RestoreTPL (OriginalTPL);
|
|
|
|
}
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
DEBUG((EFI_D_ERROR, "NOR FLASH Programming [WriteSingleBlock] failed at address 0x%08x. Exit Status = \"%r\".\n", WordAddress, Status));
|
|
|
|
}
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashWriteBlocks (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN EFI_LBA Lba,
|
|
|
|
IN UINTN BufferSizeInBytes,
|
|
|
|
IN VOID *Buffer
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 *pWriteBuffer;
|
|
|
|
EFI_STATUS Status = EFI_SUCCESS;
|
|
|
|
EFI_LBA CurrentBlock;
|
|
|
|
UINT32 BlockSizeInWords;
|
|
|
|
UINT32 NumBlocks;
|
|
|
|
UINT32 BlockCount;
|
|
|
|
|
|
|
|
// The buffer must be valid
|
|
|
|
if (Buffer == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(Instance->Media.ReadOnly == TRUE) {
|
|
|
|
return EFI_WRITE_PROTECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We must have some bytes to read
|
|
|
|
DEBUG((DEBUG_BLKIO, "NorFlashWriteBlocks: BufferSizeInBytes=0x%x\n", BufferSizeInBytes));
|
|
|
|
if(BufferSizeInBytes == 0) {
|
|
|
|
return EFI_BAD_BUFFER_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The size of the buffer must be a multiple of the block size
|
|
|
|
DEBUG((DEBUG_BLKIO, "NorFlashWriteBlocks: BlockSize in bytes =0x%x\n", Instance->Media.BlockSize));
|
|
|
|
if ((BufferSizeInBytes % Instance->Media.BlockSize) != 0) {
|
|
|
|
return EFI_BAD_BUFFER_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// All blocks must be within the device
|
|
|
|
NumBlocks = ((UINT32)BufferSizeInBytes) / Instance->Media.BlockSize ;
|
|
|
|
|
|
|
|
DEBUG((DEBUG_BLKIO, "NorFlashWriteBlocks: NumBlocks=%d, LastBlock=%ld, Lba=%ld.\n", NumBlocks, Instance->Media.LastBlock, Lba));
|
|
|
|
|
|
|
|
if ((Lba + NumBlocks) > (Instance->Media.LastBlock + 1)) {
|
|
|
|
DEBUG((EFI_D_ERROR, "NorFlashWriteBlocks: ERROR - Write will exceed last block.\n"));
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
BlockSizeInWords = Instance->Media.BlockSize / 4;
|
|
|
|
|
|
|
|
// Because the target *Buffer is a pointer to VOID, we must put all the data into a pointer
|
|
|
|
// to a proper data type, so use *ReadBuffer
|
|
|
|
pWriteBuffer = (UINT32 *)Buffer;
|
|
|
|
|
|
|
|
CurrentBlock = Lba;
|
|
|
|
for (BlockCount=0; BlockCount < NumBlocks; BlockCount++, CurrentBlock++, pWriteBuffer = pWriteBuffer + BlockSizeInWords) {
|
|
|
|
|
|
|
|
DEBUG((DEBUG_BLKIO, "NorFlashWriteBlocks: Writing block #%d\n", (UINTN)CurrentBlock));
|
|
|
|
|
|
|
|
Status = NorFlashWriteSingleBlock (Instance, CurrentBlock, pWriteBuffer, BlockSizeInWords);
|
|
|
|
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DEBUG((DEBUG_BLKIO, "NorFlashWriteBlocks: Exit Status = \"%r\".\n", Status));
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashReadBlocks (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance,
|
|
|
|
IN EFI_LBA Lba,
|
|
|
|
IN UINTN BufferSizeInBytes,
|
|
|
|
OUT VOID *Buffer
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINT32 NumBlocks;
|
|
|
|
UINTN StartAddress;
|
|
|
|
|
2013-04-14 11:25:34 +02:00
|
|
|
DEBUG((DEBUG_BLKIO, "NorFlashReadBlocks: BufferSize=0x%xB BlockSize=0x%xB LastBlock=%ld, Lba=%ld.\n",
|
|
|
|
BufferSizeInBytes, Instance->Media.BlockSize, Instance->Media.LastBlock, Lba));
|
|
|
|
|
2013-01-25 12:28:06 +01:00
|
|
|
// The buffer must be valid
|
|
|
|
if (Buffer == NULL) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
2013-04-14 11:25:34 +02:00
|
|
|
// Return if we have not any byte to read
|
|
|
|
if (BufferSizeInBytes == 0) {
|
|
|
|
return EFI_SUCCESS;
|
2013-01-25 12:28:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// The size of the buffer must be a multiple of the block size
|
|
|
|
if ((BufferSizeInBytes % Instance->Media.BlockSize) != 0) {
|
|
|
|
return EFI_BAD_BUFFER_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// All blocks must be within the device
|
|
|
|
NumBlocks = ((UINT32)BufferSizeInBytes) / Instance->Media.BlockSize ;
|
|
|
|
|
|
|
|
if ((Lba + NumBlocks) > (Instance->Media.LastBlock + 1)) {
|
|
|
|
DEBUG((EFI_D_ERROR, "NorFlashReadBlocks: ERROR - Read will exceed last block\n"));
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the address to start reading from
|
|
|
|
StartAddress = GET_NOR_BLOCK_ADDRESS (Instance->RegionBaseAddress,
|
|
|
|
Lba,
|
|
|
|
Instance->Media.BlockSize
|
|
|
|
);
|
|
|
|
|
|
|
|
// Put the device into Read Array mode
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_READ_ARRAY);
|
|
|
|
|
|
|
|
// Readout the data
|
|
|
|
CopyMem(Buffer, (UINTN *)StartAddress, BufferSizeInBytes);
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
NorFlashReset (
|
|
|
|
IN NOR_FLASH_INSTANCE *Instance
|
|
|
|
)
|
|
|
|
{
|
|
|
|
// As there is no specific RESET to perform, ensure that the devices is in the default Read Array mode
|
|
|
|
SEND_NOR_COMMAND (Instance->DeviceBaseAddress, 0, P30_CMD_READ_ARRAY);
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
EFI_STATUS
|
|
|
|
EFIAPI
|
|
|
|
NorFlashInitialise (
|
|
|
|
IN EFI_HANDLE ImageHandle,
|
|
|
|
IN EFI_SYSTEM_TABLE *SystemTable
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 Index;
|
|
|
|
NOR_FLASH_DESCRIPTION* NorFlashDevices;
|
|
|
|
UINT32 NorFlashDeviceCount;
|
|
|
|
BOOLEAN ContainVariableStorage;
|
|
|
|
|
|
|
|
Status = NorFlashPlatformInitialization ();
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashInitialise: Fail to initialize Nor Flash devices\n"));
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = NorFlashPlatformGetDevices (&NorFlashDevices,&NorFlashDeviceCount);
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashInitialise: Fail to get Nor Flash devices\n"));
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2014-04-08 20:02:32 +02:00
|
|
|
mNorFlashInstances = AllocateRuntimePool (sizeof(NOR_FLASH_INSTANCE*) * NorFlashDeviceCount);
|
2013-01-25 12:28:06 +01:00
|
|
|
|
|
|
|
for (Index = 0; Index < NorFlashDeviceCount; Index++) {
|
|
|
|
// Check if this NOR Flash device contain the variable storage region
|
|
|
|
ContainVariableStorage =
|
|
|
|
(NorFlashDevices[Index].RegionBaseAddress <= PcdGet32 (PcdFlashNvStorageVariableBase)) &&
|
|
|
|
(PcdGet32 (PcdFlashNvStorageVariableBase) + PcdGet32 (PcdFlashNvStorageVariableSize) <= NorFlashDevices[Index].RegionBaseAddress + NorFlashDevices[Index].Size);
|
|
|
|
|
|
|
|
Status = NorFlashCreateInstance (
|
|
|
|
NorFlashDevices[Index].DeviceBaseAddress,
|
|
|
|
NorFlashDevices[Index].RegionBaseAddress,
|
|
|
|
NorFlashDevices[Index].Size,
|
|
|
|
Index,
|
|
|
|
NorFlashDevices[Index].BlockSize,
|
|
|
|
ContainVariableStorage,
|
|
|
|
&NorFlashDevices[Index].Guid,
|
|
|
|
&mNorFlashInstances[Index]
|
|
|
|
);
|
|
|
|
if (EFI_ERROR(Status)) {
|
|
|
|
DEBUG((EFI_D_ERROR,"NorFlashInitialise: Fail to create instance for NorFlash[%d]\n",Index));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|