mirror of
https://github.com/oliverschmidt/contiki.git
synced 2025-01-11 19:29:50 +00:00
366 lines
12 KiB
C
366 lines
12 KiB
C
/** @file hal/micro/cortexm3/flash.c
|
|
* @brief Implements the generic flash manipulation routines.
|
|
*
|
|
* The file 'flash-sw-spec.txt' should provide *all* the information needed
|
|
* to understand and work with the FLITF and flash.
|
|
*
|
|
*
|
|
* <!--(C) COPYRIGHT 2010 STMicroelectronics. All rights reserved. -->
|
|
*/
|
|
|
|
#include PLATFORM_HEADER
|
|
#include "error.h"
|
|
#include "hal/micro/cortexm3/bootloader/fib-bootloader.h"
|
|
#include "hal/micro/cortexm3/mpu.h"
|
|
#include "memmap.h"
|
|
#include "flash.h"
|
|
|
|
#ifdef FLASH_PROGRAMMING_WITH_EMPTY_FIB
|
|
#define ST_EMU_TEST
|
|
#endif
|
|
// A translation table used to convert FibStatus codes to corresponding
|
|
// StStatus values
|
|
static const StStatus fibToStStatus[] = {
|
|
ST_SUCCESS, // FIB_SUCCESS 0
|
|
ST_BAD_ARGUMENT, // FIB_ERR_UNALIGNED 1
|
|
ST_BAD_ARGUMENT, // FIB_ERR_INVALID_ADDRESS 2
|
|
ST_BAD_ARGUMENT, // FIB_ERR_INVALID_TYPE 3
|
|
ST_ERR_FLASH_PROG_FAIL, // FIB_ERR_WRITE_PROTECTED 4
|
|
ST_ERR_FLASH_PROG_FAIL, // FIB_ERR_WRITE_FAILED 5
|
|
ST_ERR_FLASH_WRITE_INHIBITED, // FIB_ERR_ERASE_REQUIRED 6
|
|
ST_ERR_FLASH_VERIFY_FAILED // FIB_ERR_VERIFY_FAILED 7
|
|
};
|
|
|
|
//The purpose of flashEraseIsActive and halFlashEraseIsActive() is so that
|
|
//interrupts can query the flash library to find out of Flash Erase is
|
|
//active when their ISR gets invoked. This is useful because Flash Erase
|
|
//causes the chip to go ATOMIC for 21ms and this delay will disrupt interrupt
|
|
//latency. By having a sinple API that an ISR can query for this state,
|
|
//the ISR can appriopriately adjust for a 21ms latency time.
|
|
boolean flashEraseIsActive = FALSE;
|
|
boolean halFlashEraseIsActive(void)
|
|
{
|
|
return flashEraseIsActive;
|
|
}
|
|
|
|
|
|
// Emulators do not have FIB bootloaders, so need to include a copy of
|
|
// these core flash routines.
|
|
|
|
#if defined(ST_EMU_TEST)
|
|
|
|
static void enableFlitf(void)
|
|
{
|
|
//First, unlock the FLITF by writing the two key values to the Flash
|
|
//Protection Unlock register
|
|
FPEC_KEY = FPEC_KEY1;
|
|
FPEC_KEY = FPEC_KEY2;
|
|
|
|
//Second, unlock the CIB by writing the two key values to the CIB
|
|
//Protection Unlock register
|
|
OPT_KEY = FPEC_KEY1;
|
|
OPT_KEY = FPEC_KEY2;
|
|
|
|
//Turn on the FPEC clock for flash manipulation operations
|
|
FPEC_CLKREQ = FPEC_CLKREQ_FIELD;
|
|
|
|
//make sure the FPEC clock is running before we proceed
|
|
while( (FPEC_CLKSTAT&FPEC_CLKACK) != FPEC_CLKACK) {}
|
|
|
|
//just in case, wait until the flash is no longer busy
|
|
while( (FLASH_STATUS&FLASH_STATUS_FLA_BSY) == FLASH_STATUS_FLA_BSY ) {}
|
|
}
|
|
|
|
static void disableFlitf(void)
|
|
{
|
|
//make sure the FPEC is completely idle before turning off the clock
|
|
while( (FPEC_CLKSTAT&FPEC_CLKBSY) == FPEC_CLKBSY) {}
|
|
|
|
//Turn off the FPEC clock now that we're done
|
|
FPEC_CLKREQ = FPEC_CLKREQ_RESET;
|
|
|
|
//Set LOCK and clear OPTWREN to lock both the FLITF and the CIB.
|
|
//NOTE: The PROG bit must also be cleared otherwise Flash can still
|
|
// be programmed even with the LOCK bit set. BugzID: 6267
|
|
FLASH_CTRL = FLASH_CTRL_LOCK; //lock the flash from further accesses
|
|
}
|
|
|
|
static FibStatus fibFlashWrite(int32u address, int8u *data, int32u length, int32u dummy)
|
|
{
|
|
int32u i;
|
|
int16u *ptr;
|
|
FibStatus status = FIB_SUCCESS;
|
|
// Address and length must be half-word aligned.
|
|
if ((address & 1) || (length & 1)) {
|
|
return FIB_ERR_UNALIGNED;
|
|
}
|
|
// Start and end address must be in MFB or CIB.
|
|
if (!((address >= MFB_BOTTOM && address + length <= MFB_TOP + 1)
|
|
|| (address >= CIB_BOTTOM && address + length <= CIB_TOP + 1))) {
|
|
return FIB_ERR_INVALID_ADDRESS;
|
|
}
|
|
enableFlitf();
|
|
ptr = (int16u *)address;
|
|
for (i = 0; i < length; i += 2) {
|
|
int16u currentData = *ptr;
|
|
int16u newData = HIGH_LOW_TO_INT(data[i + 1], data[i]);
|
|
// Only program the data if it makes sense to do so.
|
|
if (currentData == newData) {
|
|
// If the new data matches the flash, don't bother doing anything.
|
|
} else if (currentData == 0xFFFF || newData == 0x0000) {
|
|
// If the flash is 0xFFFF we're allowed to write anything.
|
|
// If the new data is 0x0000 it doesn't matter what the flash is.
|
|
// OPTWREN must stay set to keep CIB unlocked.
|
|
if ((CIB_OB_BOTTOM <= (int32u)ptr) && ((int32u)ptr <= CIB_OB_TOP)) {
|
|
FLASH_CTRL = (FLASH_CTRL_OPTWREN | FLASH_CTRL_OPTPROG);
|
|
} else {
|
|
FLASH_CTRL = (FLASH_CTRL_OPTWREN | FLASH_CTRL_PROG);
|
|
}
|
|
// Assigning data to the address performs the actual write.
|
|
(*ptr) = newData;
|
|
// Wait for the busy bit to clear, indicating operation is done.
|
|
while ((FLASH_STATUS & FLASH_STATUS_FLA_BSY) != 0) {}
|
|
// Reset the operation complete flag.
|
|
FLASH_STATUS = FLASH_STATUS_EOP;
|
|
// Check if any error bits have been tripped, and if so, exit.
|
|
// The bit PAGE_PROG_ERR is not relevant in this programming mode.
|
|
if (FLASH_STATUS & (FLASH_STATUS_WRP_ERR | FLASH_STATUS_PROG_ERR)) {
|
|
if (FLASH_STATUS & FLASH_STATUS_WRP_ERR) {
|
|
status = FIB_ERR_WRITE_PROTECTED;
|
|
} else {
|
|
status = FIB_ERR_WRITE_FAILED;
|
|
}
|
|
FLASH_STATUS = FLASH_STATUS_WRP_ERR;
|
|
FLASH_STATUS = FLASH_STATUS_PROG_ERR;
|
|
break;
|
|
}
|
|
} else {
|
|
status = FIB_ERR_ERASE_REQUIRED;
|
|
break;
|
|
}
|
|
ptr++;
|
|
}
|
|
disableFlitf();
|
|
return status;
|
|
}
|
|
|
|
static FibStatus fibFlashWriteVerify(int32u address, int8u *data, int32u length)
|
|
{
|
|
int32u i;
|
|
int8u *ptr = (int8u *)address;
|
|
for (i = 0; i < length; i++) {
|
|
if (*ptr != data[i]) {
|
|
return FIB_ERR_VERIFY_FAILED;
|
|
}
|
|
ptr++;
|
|
}
|
|
return FIB_SUCCESS;
|
|
}
|
|
|
|
static FibStatus fibFlashErase(FibEraseType eraseType, int32u address)
|
|
{
|
|
int32u eraseOp;
|
|
int32u *ptr;
|
|
int32u length;
|
|
FibStatus status = FIB_SUCCESS;
|
|
if (BYTE_0(eraseType) == MFB_MASS_ERASE) {
|
|
eraseOp = FLASH_CTRL_MASSERASE;
|
|
ptr = (int32u *)MFB_BOTTOM;
|
|
length = MFB_SIZE_W;
|
|
} else if (BYTE_0(eraseType) == MFB_PAGE_ERASE) {
|
|
if (address < MFB_BOTTOM || address > MFB_TOP) {
|
|
return FIB_ERR_INVALID_ADDRESS;
|
|
}
|
|
eraseOp = FLASH_CTRL_PAGEERASE;
|
|
ptr = (int32u *)(address & MFB_PAGE_MASK_B);
|
|
length = MFB_PAGE_SIZE_W;
|
|
} else if (BYTE_0(eraseType) == CIB_ERASE) {
|
|
eraseOp = FLASH_CTRL_OPTWREN | FLASH_CTRL_OPTERASE;
|
|
ptr = (int32u *)CIB_BOTTOM;
|
|
length = CIB_SIZE_W;
|
|
} else {
|
|
return FIB_ERR_INVALID_TYPE;
|
|
}
|
|
if ((eraseType & DO_ERASE) != 0) {
|
|
enableFlitf();
|
|
FLASH_CTRL = eraseOp;
|
|
if (BYTE_0(eraseType) == MFB_PAGE_ERASE) {
|
|
FLASH_ADDR = (address & MFB_PAGE_MASK_B);
|
|
}
|
|
eraseOp |= FLASH_CTRL_FLA_START;
|
|
// Perform the actual erase.
|
|
FLASH_CTRL = eraseOp;
|
|
// Wait for the busy bit to clear, indicating operation is done.
|
|
while ((FLASH_STATUS & FLASH_STATUS_FLA_BSY) != 0) {}
|
|
// Reset the operation complete flag.
|
|
FLASH_STATUS = FLASH_STATUS_EOP;
|
|
// Check for errors; the only relevant one for erasing is write protection.
|
|
if (FLASH_STATUS & FLASH_STATUS_WRP_ERR) {
|
|
FLASH_STATUS = FLASH_STATUS_WRP_ERR;
|
|
status = FIB_ERR_WRITE_PROTECTED;
|
|
}
|
|
disableFlitf();
|
|
}
|
|
if (status == FIB_SUCCESS
|
|
&& (eraseType & DO_VERIFY) != 0) {
|
|
int32u i;
|
|
for (i = 0; i < length; i++) {
|
|
if (*ptr != 0xFFFFFFFF) {
|
|
return FIB_ERR_VERIFY_FAILED;
|
|
}
|
|
ptr++;
|
|
}
|
|
}
|
|
return status;
|
|
}
|
|
#endif // ST_EMU_TEST
|
|
|
|
static boolean verifyFib(void)
|
|
{
|
|
// Ensure that a programmed FIB of a proper version is present
|
|
return ( (halFixedAddressTable.baseTable.type == FIXED_ADDRESS_TABLE_TYPE) &&
|
|
( ( (halFixedAddressTable.baseTable.version & FAT_MAJOR_VERSION_MASK)
|
|
== 0x0000 ) &&
|
|
(halFixedAddressTable.baseTable.version >= 0x0002)
|
|
)
|
|
);
|
|
}
|
|
|
|
//The parameter 'eraseType' chooses which erasure will be performed while
|
|
//the 'address' parameter chooses the page to be erased during MFB page erase.
|
|
StStatus halInternalFlashErase(int8u eraseType, int32u address)
|
|
{
|
|
FibStatus status;
|
|
|
|
ATOMIC(
|
|
BYPASS_MPU(
|
|
flashEraseIsActive = TRUE;
|
|
#if defined(ST_EMU_TEST)
|
|
// Always try to use the FIB bootloader if its present
|
|
if(verifyFib()) {
|
|
status = halFixedAddressTable.fibFlashErase(
|
|
(((int32u)eraseType) | DO_ERASE),
|
|
address);
|
|
} else {
|
|
status = fibFlashErase((((int32u)eraseType) | DO_ERASE), address);
|
|
}
|
|
#else
|
|
|
|
|
|
|
|
assert(verifyFib());
|
|
status = halFixedAddressTable.fibFlashErase(
|
|
(((int32u)eraseType) | DO_ERASE),
|
|
address);
|
|
#endif
|
|
)
|
|
)
|
|
//If there are any interrupts pending that could have been delayed for 21ms,
|
|
//they will be serviced here since we exit the ATOMIC block. These ISRs
|
|
//can query the flash library and find out that erasing is active. After
|
|
//this point, we're no longer ATOMIC/disrupting latency so our erase
|
|
//active flag should be cleared.
|
|
flashEraseIsActive = FALSE;
|
|
|
|
if(status!=FIB_SUCCESS) {
|
|
return fibToStStatus[status];
|
|
}
|
|
|
|
#if defined(ST_EMU_TEST)
|
|
// Always try to use the FIB bootloader if its present
|
|
if(verifyFib()) {
|
|
status = halFixedAddressTable.fibFlashErase(
|
|
(((int32u)eraseType) | DO_VERIFY),
|
|
address);
|
|
} else {
|
|
status = fibFlashErase((((int32u)eraseType) | DO_VERIFY), address);
|
|
}
|
|
#else
|
|
status = halFixedAddressTable.fibFlashErase(
|
|
(((int32u)eraseType) | DO_VERIFY),
|
|
address);
|
|
#endif
|
|
return fibToStStatus[status];
|
|
}
|
|
|
|
|
|
//The parameter 'address' defines the starting address of where the
|
|
//programming will occur - this parameter MUST be half-word aligned since all
|
|
//programming operations are HW. The parameter 'data' is a pointer to a buffer
|
|
//containin the 16bit half-words to be written. Length is the number of 16bit
|
|
//half-words contained in 'data' to be written to flash.
|
|
//NOTE: This function can NOT write the option bytes and will throw an error
|
|
//if that is attempted.
|
|
StStatus halInternalFlashWrite(int32u address, int16u * data, int32u length)
|
|
{
|
|
FibStatus status;
|
|
|
|
length = length * 2; // fib routines specify length in bytes
|
|
|
|
ATOMIC(
|
|
BYPASS_MPU(
|
|
#if defined(ST_EMU_TEST)
|
|
// Always try to use the FIB bootloader if its present
|
|
if(verifyFib()) {
|
|
status = halFixedAddressTable.fibFlashWrite(address,
|
|
(int8u *)data,
|
|
length,
|
|
0);
|
|
} else {
|
|
status = fibFlashWrite(address, (int8u *)data, length, 0);
|
|
}
|
|
#else
|
|
|
|
|
|
|
|
// Ensure that a programmed FIB of a proper version is present
|
|
assert(verifyFib());
|
|
status = halFixedAddressTable.fibFlashWrite(address,
|
|
(int8u *)data,
|
|
length,
|
|
0);
|
|
#endif
|
|
)
|
|
)
|
|
|
|
if(status!=FIB_SUCCESS) {
|
|
return fibToStStatus[status];
|
|
}
|
|
|
|
#if defined(ST_EMU_TEST)
|
|
// Always try to use the FIB bootloader if its present
|
|
if(verifyFib()) {
|
|
status = halFixedAddressTable.fibFlashWrite(address,
|
|
(int8u *)data,
|
|
0,
|
|
length);
|
|
} else {
|
|
status = fibFlashWriteVerify(address, (int8u *)data, length);
|
|
}
|
|
#else
|
|
status = halFixedAddressTable.fibFlashWrite(address,
|
|
(int8u *)data,
|
|
0,
|
|
length);
|
|
#endif
|
|
|
|
return fibToStStatus[status];
|
|
}
|
|
|
|
|
|
//The parameter 'byte' is the option byte number to be programmed. This
|
|
//parameter can have a value of 0 through 7. 'data' is the 8bit value to be
|
|
//programmed into the option byte since the hardware will calculate the
|
|
//compliment and program the full 16bit option byte.
|
|
StStatus halInternalCibOptionByteWrite(int8u byte, int8u data)
|
|
{
|
|
int16u dataAndInverse = HIGH_LOW_TO_INT(~data, data);
|
|
// There are only 8 option bytes, don't try to program more than that.
|
|
if (byte > 7) {
|
|
return ST_ERR_FLASH_PROG_FAIL;
|
|
}
|
|
return halInternalFlashWrite(CIB_OB_BOTTOM + (byte << 1), &dataAndInverse, 1);
|
|
}
|
|
|
|
|