2011-12-23 03:20:26 +00:00

1111 lines
38 KiB
C

//*****************************************************************************
//
// eeprom.c - Driver for programming the on-chip EEPROM.
//
// Copyright (c) 2010-2011 Texas Instruments Incorporated. All rights reserved.
// Software License Agreement
//
// Texas Instruments (TI) is supplying this software for use solely and
// exclusively on TI's microcontroller products. The software is owned by
// TI and/or its suppliers, and is protected under applicable copyright
// laws. You may not combine this software with "viral" open-source
// software in order to form a larger program.
//
// THIS SOFTWARE IS PROVIDED "AS IS" AND WITH ALL FAULTS.
// NO WARRANTIES, WHETHER EXPRESS, IMPLIED OR STATUTORY, INCLUDING, BUT
// NOT LIMITED TO, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE. TI SHALL NOT, UNDER ANY
// CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL, OR CONSEQUENTIAL
// DAMAGES, FOR ANY REASON WHATSOEVER.
//
// This is part of revision 8264 of the Stellaris Peripheral Driver Library.
//
//*****************************************************************************
#include "inc/hw_eeprom.h"
#include "inc/hw_flash.h"
#include "inc/hw_ints.h"
#include "inc/hw_sysctl.h"
#include "inc/hw_types.h"
#include "driverlib/debug.h"
#include "driverlib/flash.h"
#include "driverlib/interrupt.h"
#include "driverlib/sysctl.h"
#include "driverlib/eeprom.h"
//*****************************************************************************
//
//! \addtogroup eeprom_api
//! @{
//
//*****************************************************************************
//*****************************************************************************
//
// Useful macros to extract the number of EEPROM blocks available on the target
// device and the total EEPROM storage in bytes from the EESIZE register.
//
//*****************************************************************************
#define BLOCKS_FROM_EESIZE(x) (((x) & EEPROM_EESIZE_BLKCNT_M) >> \
EEPROM_EESIZE_BLKCNT_S)
#define SIZE_FROM_EESIZE(x) ((((x) & EEPROM_EESIZE_WORDCNT_M) >> \
EEPROM_EESIZE_WORDCNT_S) * 4)
//*****************************************************************************
//
// Useful macro to extract the offset from a linear address.
//
//*****************************************************************************
#define OFFSET_FROM_ADDR(x) (((x) >> 2) & 0x0F)
//*****************************************************************************
//
// The key value required to initiate a mass erase.
//
//*****************************************************************************
#define EEPROM_MASS_ERASE_KEY ((unsigned long)0xE37B << EEPROM_EEDBGME_KEY_S)
//*****************************************************************************
//
// This function implements a workaround for a bug in Blizzard rev A silicon.
// It ensures that only the 1KB flash sector containing a given EEPROM address
// is erased if an erase/copy operation is required as a result of a following
// EEPROM write.
//
//*****************************************************************************
static void
EEPROMSetSectorMask(unsigned long ulAddress)
{
unsigned long ulMask;
//
// Determine which page contains the passed EEPROM address. The 2KB EEPROM
// is implemented in 16KB of flash with each 1KB sector of flash holding
// values for 32 consecutive EEPROM words (or 128 bytes).
//
ulMask = ~(1 << (ulAddress >> 7));
SysCtlDelay(10);
HWREG(0x400FD0FC) = 3;
SysCtlDelay(10);
HWREG(0x400AE2C0) = ulMask;
SysCtlDelay(10);
HWREG(0x400FD0FC) = 0;
SysCtlDelay(10);
}
//*****************************************************************************
//
// Clear the FSM sector erase mask to ensure that any following main array flash
// erase operations operate as expected.
//
//*****************************************************************************
static void
EEPROMClearSectorMask(void)
{
SysCtlDelay(10);
HWREG(0x400FD0FC) = 3;
SysCtlDelay(10);
HWREG(0x400AE2C0) = 0;
SysCtlDelay(10);
HWREG(0x400FD0FC) = 0;
SysCtlDelay(10);
}
//*****************************************************************************
//
// Block until the EEPROM peripheral is not busy.
//
//*****************************************************************************
static void
EEPROMWaitForDone(void)
{
//
// Is the EEPROM still busy?
//
while(HWREG(EEPROM_EEDONE) & EEPROM_EEDONE_WORKING)
{
//
// Spin while EEPROM is busy.
//
}
}
//*****************************************************************************
//
//! Performs any necessary recovery in case of power failures during write.
//!
//! This function must be called after SysCtlPeripheralEnable() and before
//! the EEPROM is accessed to check for errors resulting from power failure
//! during a previous write operation. The function detects these errors
//! and performs as much recovery as possible before returning information to
//! the caller on whether or not a previous data write was lost and must
//! be retried.
//!
//! In cases where \b EEPROM_INIT_RETRY is returned, the application is
//! responsible for determining which data write may have been lost and
//! rewriting this data. If \b EEPROM_INIT_ERROR is returned, the EEPROM was
//! unable to recover its state. This condition may or may not be resolved on
//! future resets depending upon the cause of the fault. For example, if the
//! supply voltage is unstable, retrying the operation once the voltage is
//! stabilized may clear the error.
//!
//! Failure to call this function after a reset may lead to permanent data loss
//! if the EEPROM is later written!
//!
//! \return Returns \b EEPROM_INIT_OK if no errors were detected,
//! \b EEPROM_INIT_RETRY if a previous write operation may have been
//! interrupted by a power or reset event or \b EEPROM_INIT_ERROR if the EEPROM
//! peripheral cannot currently recover from an interrupted write or erase
//! operation.
//
//*****************************************************************************
unsigned long
EEPROMInit(void)
{
unsigned long ulStatus;
//
// Insert a small delay (6 cycles + call overhead) to guard against the
// possibility that this function is called immediately after the EEPROM
// peripheral is enabled. Without this delay, there is a slight chance
// that the first EEPROM register read will fault if you are using a
// compiler with a ridiculously good optimizer!
//
SysCtlDelay(2);
//
// Make sure the EEPROM has finished its reset processing.
//
EEPROMWaitForDone();
//
// Read the EESUPP register to see if any errors have been reported.
//
ulStatus = HWREG(EEPROM_EESUPP);
//
// Did an error of some sort occur during a previous attempt to write to
// the EEPROM?
//
if(ulStatus & (EEPROM_EESUPP_PRETRY | EEPROM_EESUPP_ERETRY))
{
//
// Perform a second reset to allow the EEPROM a chance to correct
// the errors.
//
SysCtlPeripheralReset(SYSCTL_PERIPH_EEPROM0);
//
// Wait for the EEPROM to complete it's reset processing once again.
//
SysCtlDelay(2);
EEPROMWaitForDone();
//
// Read EESUPP once again to determine if the error conditions are
// cleared.
//
ulStatus = HWREG(EEPROM_EESUPP);
if(ulStatus & (EEPROM_EESUPP_PRETRY | EEPROM_EESUPP_ERETRY))
{
return(EEPROM_INIT_ERROR);
}
else
{
return(EEPROM_INIT_RETRY);
}
}
//
// The EEPROM does not indicate that any error occurred.
//
return(EEPROM_INIT_OK);
}
//*****************************************************************************
//
//! Determines the size of the EEPROM.
//!
//! This function returns the size of the EEPROM in bytes.
//!
//! \return Returns the total number of bytes in the EEPROM.
//
//*****************************************************************************
unsigned long
EEPROMSizeGet(void)
{
//
// Return the size of the EEPROM in bytes.
//
return(SIZE_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
}
//*****************************************************************************
//
//! Determines the number of blocks in the EEPROM.
//!
//! This function may be called to determine the number of blocks in the
//! EEPROM. Each block is the same size and the number of bytes of storage
//! contained in a block may be determined by dividing the size of the device,
//! obtained via a call to the EEPROMSizeGet() function, by the number of
//! blocks returned by this function.
//!
//! \return Returns the total number of bytes in the device EEPROM.
//
//*****************************************************************************
unsigned long
EEPROMBlockCountGet(void)
{
//
// Extract the number of blocks and return it to the caller.
//
return(BLOCKS_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
}
//*****************************************************************************
//
//! Reads data from the EEPROM.
//!
//! \param pulData is a pointer to storage for the data read from the EEPROM.
//! This pointer must point to at least \e ulCount bytes of available memory.
//! \param ulAddress is the byte address within the EEPROM from which data is
//! to be read. This value must be a multiple of 4.
//! \param ulCount is the number of bytes of data to read from the EEPROM.
//! This value must be a multiple of 4.
//!
//! This function may be called to read a number of words of data from a
//! word-aligned address within the EEPROM. Data read is copied into the
//! buffer pointed to by the \e pulData parameter.
//!
//! \return None.
//
//*****************************************************************************
void
EEPROMRead(unsigned long *pulData, unsigned long ulAddress,
unsigned long ulCount)
{
//
// Check parameters in a debug build.
//
ASSERT(pulData);
ASSERT(ulAddress < SIZE_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
ASSERT((ulAddress + ulCount) <= SIZE_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
ASSERT((ulAddress & 3) == 0);
ASSERT((ulCount & 3) == 0);
//
// Set the block and offset appropriately to read the first word.
//
HWREG(EEPROM_EEBLOCK) = EEPROMBlockFromAddr(ulAddress);
HWREG(EEPROM_EEOFFSET) = OFFSET_FROM_ADDR(ulAddress);
//
// Convert the byte count to a word count.
//
ulCount /= 4;
//
// Read each word in turn.
//
while(ulCount)
{
//
// Read the next word through the autoincrementing register.
//
*pulData = HWREG(EEPROM_EERDWRINC);
//
// Move on to the next word.
//
pulData++;
ulCount--;
//
// Do we need to move to the next block? This is the case if the
// offset register has just wrapped back to 0.
//
if(HWREG(EEPROM_EEOFFSET) == 0)
{
HWREG(EEPROM_EEBLOCK) += 1;
}
}
}
//*****************************************************************************
//
//! Writes data to the EEPROM.
//!
//! \param pulData points to the first word of data to write to the EEPROM.
//! \param ulAddress defines the byte address within the EEPROM that the data
//! is to be written to. This value must be a multiple of 4.
//! \param ulCount defines the number of bytes of data that is to be written.
//! This value must be a multiple of 4.
//!
//! This function may be called to write data into the EEPROM at a given
//! word-aligned address. The call is synchronous and returns only after
//! all data has been written or an error occurs.
//!
//! \return Returns 0 on success or non-zero values on failure. Failure codes
//! are logical OR combinations of \b EEPROM_RC_INVPL, \b EEPROM_RC_WRBUSY,
//! \b EEPROM_RC_NOPERM, \b EEPROM_RC_WKCOPY, \b EEPROM_RC_WKERASE, and
//! \b EEPROM_RC_WORKING.
//
//*****************************************************************************
unsigned long
EEPROMProgram(unsigned long *pulData, unsigned long ulAddress,
unsigned long ulCount)
{
unsigned long ulStatus;
//
// Check parameters in a debug build.
//
ASSERT(pulData);
ASSERT(ulAddress < SIZE_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
ASSERT((ulAddress + ulCount) <= SIZE_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
ASSERT((ulAddress & 3) == 0);
ASSERT((ulCount & 3) == 0);
//
// This is a workaround for a silicon problem on Blizzard rev A.
//
if(CLASS_IS_BLIZZARD && REVISION_IS_A0)
{
EEPROMSetSectorMask(ulAddress);
}
//
// Set the block and offset appropriately to program the first word.
//
HWREG(EEPROM_EEBLOCK) = EEPROMBlockFromAddr(ulAddress);
HWREG(EEPROM_EEOFFSET) = OFFSET_FROM_ADDR(ulAddress);
//
// Convert the byte count to a word count.
//
ulCount /= 4;
//
// Write each word in turn.
//
while(ulCount)
{
//
// Write the next word through the autoincrementing register.
//
HWREG(EEPROM_EERDWRINC) = *pulData;
//
// Wait for the write to complete.
//
do
{
//
// Read the status.
//
ulStatus = HWREG(EEPROM_EEDONE);
}
while(ulStatus & EEPROM_EEDONE_WORKING);
//
// Make sure we completed the write without errors. Note that we
// must check this per-word because write permission can be set per
// block resulting in only a section of the write not being performed.
//
if(ulStatus & (EEPROM_EEDONE_NOPERM | EEPROM_EEDONE_INVPL))
{
//
// An error was reported that would prevent the values from
// being written correctly.
//
if(CLASS_IS_BLIZZARD && REVISION_IS_A0)
{
EEPROMClearSectorMask();
}
return(ulStatus);
}
//
// Move on to the next word.
//
pulData++;
ulCount--;
//
// Do we need to move to the next block? This is the case if the
// offset register has just wrapped back to 0.
//
if(HWREG(EEPROM_EEOFFSET) == 0)
{
HWREG(EEPROM_EEBLOCK) += 1;
}
}
//
// Clear the sector protection bits to prevent possible problems when
// programming the main flash array later.
//
if(CLASS_IS_BLIZZARD && REVISION_IS_A0)
{
EEPROMClearSectorMask();
}
//
// Return the current status to the caller.
//
return(HWREG(EEPROM_EEDONE));
}
//*****************************************************************************
//
//! Writes a word to the EEPROM.
//!
//! \param ulData is the word to write to the EEPROM.
//! \param ulAddress defines the byte address within the EEPROM to which the
//! data is to be written. This value must be a multiple of 4.
//!
//! This function is intended to allow EEPROM programming under interrupt
//! control. It may be called to start the process of writing a single word of
//! data into the EEPROM at a given word-aligned address. The call is
//! asynchronous and returna immediately without waiting for the write to
//! complete. Completion of the operation is signaled by means of an
//! interrupt from the EEPROM module. The EEPROM peripheral shares a single
//! interrupt vector with the flash memory subsystem, \e INT_FLASH.
//!
//! \return Returns status and error information in the form of a logical OR
//! combinations of \b EEPROM_RC_INVPL, \b EEPROM_RC_WRBUSY,
//! \b EEPROM_RC_NOPERM, \b EEPROM_RC_WKCOPY, \b EEPROM_RC_WKERASE and
//! \b EEPROM_RC_WORKING. Flags \b EEPROM_RC_WKCOPY, \b EEPROM_RC_WKERASE, and
//! \b EEPROM_RC_WORKING are expected in normal operation and do not indicate
//! an error.
//
//*****************************************************************************
unsigned long
EEPROMProgramNonBlocking(unsigned long ulData, unsigned long ulAddress)
{
//
// Check parameters in a debug build.
//
ASSERT(ulAddress < SIZE_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
ASSERT((ulAddress & 3) == 0);
//
// This is a workaround for a silicon problem on Blizzard rev A.
//
if(CLASS_IS_BLIZZARD && REVISION_IS_A0)
{
EEPROMSetSectorMask(ulAddress);
}
//
// Set the block and offset appropriately to program the desired word.
//
HWREG(EEPROM_EEBLOCK) = EEPROMBlockFromAddr(ulAddress);
HWREG(EEPROM_EEOFFSET) = OFFSET_FROM_ADDR(ulAddress);
//
// Write the new word using the auto-incrementing register just in case
// the caller wants to write follow-on words using direct register access
//
HWREG(EEPROM_EERDWRINC) = ulData;
//
// Return the current status to the caller.
//
return(HWREG(EEPROM_EEDONE));
}
//*****************************************************************************
//
//! Erases the EEPROM and returns it to the factory default condition.
//!
//! This function completely erases the EEPROM and removes any and
//! all access protection on its blocks, leaving the device in the factory
//! default condition. After this operation, all EEPROM words contain the
//! value 0xFFFFFFFF and all blocks are accessible for both read and write
//! operations in all CPU modes. No passwords are active.
//!
//! The function is synchronous and does not return until the erase operation
//! has completed.
//!
//! \return Returns 0 on success or non-zero values on failure. Failure codes
//! are logical OR combinations of \b EEPROM_RC_INVPL, \b EEPROM_RC_WRBUSY,
//! \b EEPROM_RC_NOPERM, \b EEPROM_RC_WKCOPY, \b EEPROM_RC_WKERASE, and
//! \b EEPROM_RC_WORKING.
//
//*****************************************************************************
unsigned long
EEPROMMassErase(void)
{
//
// This is a workaround for a silicon problem on Blizzard rev A.
//
if(CLASS_IS_BLIZZARD && REVISION_IS_A0)
{
EEPROMClearSectorMask();
}
//
// Start the mass erase processing
//
HWREG(EEPROM_EEDBGME) = EEPROM_MASS_ERASE_KEY | EEPROM_EEDBGME_ME;
//
// Wait for completion.
//
EEPROMWaitForDone();
//
// Reset the peripheral. This is required so that all protection
// mechanisms and passwords are reset now that the EEPROM data has been
// scrubbed.
//
SysCtlPeripheralReset(SYSCTL_PERIPH_EEPROM0);
//
// Wait for completion again.
//
SysCtlDelay(2);
EEPROMWaitForDone();
//
// Pass any error codes back to the caller.
//
return(HWREG(EEPROM_EEDONE));
}
//*****************************************************************************
//
//! Returns the current protection level for an EEPROM block.
//!
//! \param ulBlock is the block number for which the protection level is to be
//! queried.
//!
//! This function returns the current protection settings for a given
//! EEPROM block. If block 0 is currently locked, it must be unlocked prior
//! to calling this function to query the protection setting for other blocks.
//!
//! \return Returns one of \b EEPROM_PROT_RW_LRO_URW, \b EEPROM_PROT_NA_LNA_URW
//! or \b EEPROM_PROT_RO_LNA_URO optionally OR-ed with
//! \b EEPROM_PROT_SUPERVISOR_ONLY.
//
//*****************************************************************************
unsigned long
EEPROMBlockProtectGet(unsigned long ulBlock)
{
//
// Parameter validity check.
//
ASSERT(ulBlock < BLOCKS_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
//
// Set the current block.
//
HWREG(EEPROM_EEBLOCK) = ulBlock;
//
// Return the protection flags for this block.
//
return(HWREG(EEPROM_EEPROT));
}
//*****************************************************************************
//
//! Set the current protection options for an EEPROM block.
//!
//! \param ulBlock is the block number for which the protection options are to
//! be set.
//! \param ulProtect consists of one of the values \b EEPROM_PROT_RW_LRO_URW,
//! \b EEPROM_PROT_NA_LNA_URW or \b EEPROM_PROT_RO_LNA_URO optionally ORed with
//! \b EEPROM_PROT_SUPERVISOR_ONLY.
//!
//! This function sets the protection settings for a given EEPROM block
//! assuming no protection settings have previously been written. Note that
//! protection settings applied to block 0 have special meaning and control
//! access to the EEPROM peripheral as a whole. Protection settings applied to
//! blocks numbered 1 and above are layered above any protection set on block 0
//! such that the effective protection on each block is the logical OR of the
//! protection flags set for block 0 and for the target block. This protocol
//! allows global protection options to be set for the whole device via block
//! 0 and more restrictive protection settings to be set on a block-by-block
//! basis.
//!
//! The protection flags indicate access permissions as follow:
//!
//! \b EEPROM_PROT_SUPERVISOR_ONLY restricts access to the block to threads
//! running in supervisor mode. If clear, both user and supervisor threads
//! can access the block.
//!
//! \b EEPROM_PROT_RW_LRO_URW provides read/write access to the block if no
//! password is set or if a password is set and the block is unlocked. If the
//! block is locked, only read access is permitted.
//!
//! \b EEPROM_PROT_NA_LNA_URW provides neither read nor write access unless
//! a password is set and the block is unlocked. If the block is unlocked,
//! both read and write access are permitted.
//!
//! \b EEPROM_PROT_RO_LNA_URO provides read access to the block if no password
//! is set or if a password is set and the block is unlocked. If the block is
//! password protected and locked, neither read nor write access is permitted.
//!
//! \return Returns a logical OR combination of \b EEPROM_RC_INVPL,
//! \b EEPROM_RC_WRBUSY, \b EEPROM_RC_NOPERM, \b EEPROM_RC_WKCOPY,
//! \b EEPROM_RC_WKERASE, and \b EEPROM_RC_WORKING to indicate status and error
//! conditions.
//
//*****************************************************************************
unsigned long
EEPROMBlockProtectSet(unsigned long ulBlock, unsigned long ulProtect)
{
//
// Parameter validity check.
//
ASSERT(ulBlock < BLOCKS_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
//
// Set the current block.
//
HWREG(EEPROM_EEBLOCK) = ulBlock;
//
// Set the protection options for this block.
//
HWREG(EEPROM_EEPROT) = ulProtect;
//
// Wait for the write to complete.
//
while(HWREG(EEPROM_EEDONE) & EEPROM_EEDONE_WORKING)
{
//
// Still working.
//
}
//
// Pass any error codes back to the caller.
//
return(HWREG(EEPROM_EEDONE));
}
//*****************************************************************************
//
//! Sets the password used to protect an EEPROM block.
//!
//! \param ulBlock is the EEPROM block number for which the password is to be
//! set.
//! \param pulPassword points to an array of unsigned long values comprising
//! the password to set. Each element may be any 32-bit value other than
//! 0xFFFFFFFF. This array must contain the number of elements given by the
//! \b ulCount parameter.
//! \param ulCount provides the number of unsigned longs in the \b ulPassword.
//! Valid values are 1, 2 and 3.
//!
//! This function allows the password used to unlock an EEPROM block to be
//! set. Valid passwords may be either 32, 64 or 96 bits comprising words
//! with any value other than 0xFFFFFFFF. The password may only be set once.
//! Any further attempts to set the password result in an error. Once the
//! password is set, the block remains unlocked until EEPROMBlockLock() is
//! called for that block or block 0, or a reset occurs.
//!
//! If a password is set on block 0, this affects locking of the peripheral as
//! a whole. When block 0 is locked, all other EEPROM blocks are inaccessible
//! until block 0 is unlocked. Once block 0 is unlocked, other blocks
//! become accessible according to any passwords set on those blocks and the
//! protection set for that block via a call to EEPROMBlockProtectSet().
//!
//! \return Returns a logical OR combination of \b EEPROM_RC_INVPL,
//! \b EEPROM_RC_WRBUSY, \b EEPROM_RC_NOPERM, \b EEPROM_RC_WKCOPY,
//! \b EEPROM_RC_WKERASE, and \b EEPROM_RC_WORKING to indicate status and error
//! conditions.
//
//*****************************************************************************
unsigned long
EEPROMBlockPasswordSet(unsigned long ulBlock, unsigned long *pulPassword,
unsigned long ulCount)
{
unsigned long ulReg;
//
// Check parameters in a debug build.
//
ASSERT(pulPassword);
ASSERT(ulBlock < BLOCKS_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
ASSERT(ulCount <= 3);
//
// Set the block number whose password we are about to write.
//
HWREG(EEPROM_EEBLOCK) = ulBlock;
//
// Start with the first password word.
//
ulReg = EEPROM_EEPASS0;
//
// Write the password.
//
while(ulCount)
{
//
// Start the process of writing the password.
//
HWREG(ulReg) = *pulPassword;
//
// Update values in preparation for writing the next word.
//
pulPassword++;
ulReg += 4;
ulCount--;
//
// Wait for the last word write to complete or an error to be reported.
//
while(HWREG(EEPROM_EEDONE) & EEPROM_EEDONE_WORKING)
{
//
// Still working.
//
}
}
//
// Return the final write status.
//
return(HWREG(EEPROM_EEDONE));
}
//*****************************************************************************
//
//! Locks a password-protected EEPROM block.
//!
//! \param ulBlock is the EEPROM block number which is to be locked.
//!
//! This function locks an EEPROM block that has previously been protected by
//! writing a password. Access to the block once it is locked is determined
//! by the protection settings applied via a previous call to the
//! EEPROMBlockProtectSet() function. If no password has previously been set
//! for the block, this function has no effect.
//!
//! Locking block 0 has the effect of making all other blocks in the EEPROM
//! inaccessible.
//!
//! \return Returns the lock state for the block on exit, 1 if unlocked (as
//! would be the case if no password was set) or 0 if locked.
//!
//*****************************************************************************
unsigned long
EEPROMBlockLock(unsigned long ulBlock)
{
//
// Check parameters in a debug build.
//
ASSERT(ulBlock < BLOCKS_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
//
// Select the block we are going to lock.
//
HWREG(EEPROM_EEBLOCK) = ulBlock;
//
// Lock the block.
//
HWREG(EEPROM_EEUNLOCK) = 0xFFFFFFFF;
//
// Return the current lock state.
//
return(HWREG(EEPROM_EEUNLOCK));
}
//*****************************************************************************
//
//! Unlocks a password-protected EEPROM block.
//!
//! \param ulBlock is the EEPROM block number which is to be unlocked.
//! \param pulPassword points to an array of unsigned long values containing
//! the password for the blockt. Each element must match the password
//! originally set via a call to EEPROMBlockPasswordSet().
//! \param ulCount provides the number of unsigned longs in the \b pulPassword
//! array and must match the value originally passed to
//! EEPROMBlockPasswordSet(). Valid values are 1, 2 and 3.
//!
//! This function unlocks an EEPROM block that has previously been protected by
//! writing a password. Access to the block once it is unlocked is determined
//! by the protection settings applied via a previous call to the
//! EEPROMBlockProtectSet() function.
//!
//! To successfully unlock an EEPROM block, the password provided must match
//! the password provided on the original call to EEPROMBlockPasswordSet(). If
//! an incorrect password is provided, the block remains locked.
//!
//! Unlocking block 0 has the effect of making all other blocks in the device
//! accessible according to their own access protection settings. When block
//! 0 is locked, all other EEPROM blocks are inaccessible.
//!
//! \return Returns the lock state for the block on exit, 1 if unlocked or 0 if
//! locked.
//!
//*****************************************************************************
unsigned long
EEPROMBlockUnlock(unsigned long ulBlock, unsigned long *pulPassword,
unsigned long ulCount)
{
//
// Check parameters in a debug build.
//
ASSERT(pulPassword);
ASSERT(ulBlock < BLOCKS_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
ASSERT(ulCount <= 3);
//
// Set the block that we are trying to unlock.
//
HWREG(EEPROM_EEBLOCK) = ulBlock;
//
// Write the unlock register with 0xFFFFFFFF to reset the unlock
// sequence just in case a short password was previously used to try to
// unlock the block.
//
HWREG(EEPROM_EEUNLOCK) = 0xFFFFFFFF;
//
// We need to write the password words in the opposite order when unlocking
// compared to locking so start at the end of the array.
//
pulPassword += (ulCount - 1);
//
// Write the supplied password to unlock the block.
//
while(ulCount)
{
HWREG(EEPROM_EEUNLOCK) = *pulPassword--;
ulCount--;
}
//
// Let the caller know if their password worked.
//
return(HWREG(EEPROM_EEUNLOCK));
}
//*****************************************************************************
//
//! Hides an EEPROM block until the next reset.
//!
//! \param ulBlock is the EEPROM block number which is to be hidden.
//!
//! This function hides an EEPROM block other than block 0. Once hidden, a
//! block is completely inaccessible until the next reset. This mechanism
//! allows initialization code to have access to data which is to be hidden
//! from the rest of the application. Unlike applications using passwords, an
//! application making using of block hiding need not contain any embedded
//! passwords which could be found through disassembly.
//!
//! \return None.
//!
//*****************************************************************************
void
EEPROMBlockHide(unsigned long ulBlock)
{
//
// Check parameters in a debug build.
//
ASSERT(!ulBlock);
ASSERT(ulBlock < BLOCKS_FROM_EESIZE(HWREG(EEPROM_EESIZE)));
//
// Hide the requested block.
//
HWREG(EEPROM_EEHIDE) = (1 << ulBlock);
}
//*****************************************************************************
//
//! Enables the EEPROM interrupt.
//!
//! \param ulIntFlags indicates which EEPROM interrupt source to enable. This
//! must be \b EEPROM_INT_PROGRAM currently.
//!
//! This function enables the EEPROM interrupt. When enabled, an interrupt
//! is generated when any EEPROM write or erase operation completes. The
//! EEPROM peripheral shares a single interrupt vector with the flash memory
//! subsystem, \b INT_FLASH. This function is provided as a convenience but
//! the EEPROM interrupt can also be enabled using a call to FlashIntEnable()
//! passing FLASH_INT_EEPROM in the \b ulIntFlags parameter.
//!
//! \return None.
//!
//*****************************************************************************
void
EEPROMIntEnable(unsigned long ulIntFlags)
{
//
// Look for valid interrupt sources.
//
ASSERT(ulIntFlags == EEPROM_INT_PROGRAM);
//
// Enable interrupts from the EEPROM module.
//
HWREG(EEPROM_EEINT) |= EEPROM_EEINT_INT;
//
// Enable the EEPROM interrupt in the flash controller module.
//
HWREG(FLASH_FCIM) |= FLASH_FCRIS_ERIS;
}
//*****************************************************************************
//
//! Disables the EEPROM interrupt.
//!
//! \param ulIntFlags indicates which EEPROM interrupt source to disable. This
//! must be \b EEPROM_INT_PROGRAM currently.
//!
//! This function disables the EEPROM interrupt and prevents calls to the
//! interrupt vector when any EEPROM write or erase operation completes. The
//! EEPROM peripheral shares a single interrupt vector with the flash memory
//! subsystem, \e INT_FLASH. This function is provided as a convenience but
//! the EEPROM interrupt can also be disabled using a call to FlashIntDisable()
//! passing FLASH_INT_EEPROM in the \b ulIntFlags parameter.
//!
//! \return None.
//!
//*****************************************************************************
void
EEPROMIntDisable(unsigned long ulIntFlags)
{
//
// Look for valid interrupt sources.
//
ASSERT(ulIntFlags == EEPROM_INT_PROGRAM);
//
// Disable the EEPROM interrupt in the flash controller module.
//
HWREG(FLASH_FCIM) &= ~FLASH_FCIM_EMASK;
//
// Disable interrupts from the EEPROM module.
//
HWREG(EEPROM_EEINT) &= ~EEPROM_EEINT_INT;
}
//*****************************************************************************
//
//! Reports the state of the EEPROM interrupt.
//!
//! \param bMasked determines whether the masked or unmasked state of the
//! interrupt is to be returned. If bMasked is \e true, the masked state is
//! returned, otherwise the unmasked state is returned.
//!
//! This function allows an application to query the state of the EEPROM
//! interrupt. If active, the interrupt may be cleared by calling
//! EEPROMIntClear().
//!
//! \return Returns \b EEPROM_INT_PROGRAM if an interrupt is being signaled or
//! 0 otherwise.
//
//*****************************************************************************
unsigned long
EEPROMIntStatus(tBoolean bMasked)
{
if(bMasked)
{
//
// If asked for the masked interrupt status, we check to see if the
// relevant interrupt is pending in the flash controller then return
// the appropriate EEPROM flag if it is.
//
return((HWREG(FLASH_FCMISC) & FLASH_FCMISC_EMISC) ?
EEPROM_INT_PROGRAM : 0);
}
else
{
//
// If asked for the unmasked interrupt status, infer that an interrupt
// is pending if the WORKING bit of the EEDONE register is clear. The
// actual interrupt fires on the high to low transition of this bit
// but we don't have access to an unmasked interrupt status for the
// EEPROM because it's handled via the flash controller so we have to
// make do with this instead.
//
return((HWREG(EEPROM_EEDONE) & EEPROM_EEDONE_WORKING) ?
0 : EEPROM_INT_PROGRAM);
}
}
//*****************************************************************************
//
//! Clears the EEPROM interrupt.
//!
//! \param ulIntFlags indicates which interrupt sources to clear. Currently,
//! the only valid value is \b EEPROM_INT_PROGRAM.
//!
//! This function allows an application to clear the EEPROM interrupt.
//!
//! \note Because there is a write buffer in the Cortex-M processor, it may
//! take several clock cycles before the interrupt source is actually cleared.
//! Therefore, it is recommended that the interrupt source be cleared early in
//! the interrupt handler (as opposed to the very last action) to avoid
//! returning from the interrupt handler before the interrupt source is
//! actually cleared. Failure to do so may result in the interrupt handler
//! being immediately reentered (because the interrupt controller still sees
//! the interrupt source asserted).
//!
//! \return None.
//!
//*****************************************************************************
void
EEPROMIntClear(unsigned long ulIntFlags)
{
//
// Clear the flash interrupt.
//
HWREG(FLASH_FCMISC) = FLASH_FCMISC_EMISC;
//
// Clear the sector protection bits to prevent possible problems when
// programming the main flash array later.
//
if(CLASS_IS_BLIZZARD && REVISION_IS_A0)
{
EEPROMClearSectorMask();
}
}
//*****************************************************************************
//
//! Returns status on the last EEPROM program or erase operation.
//!
//! This function returns the current status of the last program or erase
//! operation performed by the EEPROM. It is intended to provide error
//! information to applications programming or setting EEPROM protection
//! options under interrupt control.
//!
//! \return Returns 0 if the last program or erase operation completed without
//! any errors. If an operation is ongoing or an error occurred, the return
//! value is a logical OR combination of \b EEPROM_RC_INVPL,
//! \b EEPROM_RC_WRBUSY, \b EEPROM_RC_NOPERM, \b EEPROM_RC_WKCOPY,
//! \b EEPROM_RC_WKERASE, and \b EEPROM_RC_WORKING.
//!
//*****************************************************************************
unsigned long
EEPROMStatusGet(void)
{
return(HWREG(EEPROM_EEDONE));
}
//*****************************************************************************
//
// Close the Doxygen group.
//! @}
//
//*****************************************************************************