📄 at25f2048.c
字号:
////////////////////////////////////////////////////////////////////////////
//
// Program to check the functionality of core accesses to Flash
// device
//
// - PRD
//
#ifdef __ADSP21375__
#include <Cdef21375.h>
#include <def21375.h>
#elif __ADSP21369__
#include <Cdef21369.h>
#include <def21369.h>
#endif
#include <sru.h>
#include <sysreg.h>
#include <stdlib.h> // malloc includes
#include "adi_at25f2048.h" // flash-AT25F2048 includes
#include "Services_Sharc.h" // system services buffers
#include "util.h" // library struct includes
#include "Errors.h" // error type includes
#include "post_common.h"
//////////////////////////////////////////////////////////////////////////////
//
// COMMON DEFINES
//
//////////////////////////////////////////////////////////////////////////////
#define BAUD_RATE_DIVISOR 100
#define BUFFER_SIZE 2
#ifdef _AT25F512_
#define NUM_SECTORS 2
#define SECTOR_SIZE 0x8000
#else
#define NUM_SECTORS 4
#define SECTOR_SIZE 0x10000
#endif
#define FLASH_START_ADDR 0x0000000 // start address
#define FLASH_SIZE (SECTOR_SIZE * NUM_SECTORS) // size of FLASH in 8-bit words. (i.e. 512K x 8)
#ifdef _FULL_FLASH_TEST_
#define FLASH_TEST_SIZE FLASH_SIZE
#define FLASH_TEST_START 0
#else
#define FLASH_TEST_SIZE SECTOR_SIZE
#define FLASH_TEST_START (SECTOR_SIZE * (NUM_SECTORS-1) ) // test last sector
#endif //_FULL_FLASH_TEST_
static volatile int AFP_Error = 0; // contains last error encountered
static volatile int AFP_ManCode = -1; // 0x1F = Atmel
static volatile int AFP_DevCode = -1; // 0x63 = AT25F2048, 0x60 = AT25F512
static volatile bool AFP_Verify = FALSE; // verify writes or not
//////////////////////////////////////////////////////////////////////////////
//
// Function prototypes
//
//////////////////////////////////////////////////////////////////////////////
static ERROR_CODE SetupForFlash(void);
static ERROR_CODE GetFlashInfo(void);
static ERROR_CODE EraseFlash(void);
static ERROR_CODE WriteData( unsigned long ulStart, long lCount, long lStride, int *pnData );
static ERROR_CODE ReadData( unsigned long ulStart, long lCount, long lStride, int *pnData );
//////////////////////////////////////////////////////////////////////////////
//
// Stand alone test jig
//
//////////////////////////////////////////////////////////////////////////////
#ifdef _STANDALONE_ // use this to run standalone tests
int main(void)
{
int bPassed = 0;
bPassed = TEST_AT25F2048_FLASH();
return 0;
}
#endif //#ifdef _STANDALONE_
//////////////////////////////////////////////////////////////////////////////
// int TEST_AT25F2048_FLASH(void)
//
// PURPOSE: Test the AT25F2048 flash
//
//////////////////////////////////////////////////////////////////////////////
int TEST_AT25F2048_FLASH(void)
{
int nManCode = 0;
int nDevCode = 0;
int nValue;
int nTestValue;
int nIndex = 0;
int bError = 1; // returning 1 indicates a pass, anything else is a fail
sysreg_bit_clr( sysreg_IRPTL, SPIHI); // clear any pending SPI interrupt latch
sysreg_bit_clr( sysreg_IMASK, SPIHI); // enable SPI receive interrupt
sysreg_bit_clr( sysreg_MODE1, IRPTEN); // enable global interrupts
*pSPIDMAC = 0;
// open the device
AFP_Error = ADIAT25F2048EntryPoint.adi_pdd_Open( NULL, // DevMgr handle
0, // pdd entry point
NULL, // device instance
NULL, // client handle callback identifier
ADI_DEV_DIRECTION_BIDIRECTIONAL,// data direction for this device
NULL, // DevMgr handle for this device
NULL, // handle to DmaMgr for this device
NULL, // handle to deferred callback service
NULL ); // client's callback function
// setup the device so the DSP can access it
if (SetupForFlash() != NO_ERR)
return 0;
AFP_Error = GetFlashInfo();
if( (0x1F != AFP_ManCode) || (0x63 != AFP_DevCode) )
return 0;
// All A's
EraseFlash();
// write incrementing values to each FLASH location
nTestValue = 0xAA;
for(nIndex = 0; nIndex < FLASH_TEST_SIZE; nIndex++ )
{
if(NO_ERR != WriteData( (FLASH_TEST_START + nIndex), 1, 1, &nTestValue ) )
{
return 0;
}
}
// verify incrementing values
nTestValue = 0xAA;
for(nIndex = 0; nIndex < FLASH_TEST_SIZE; nIndex++ )
{
if( NO_ERR != ReadData( (FLASH_TEST_START + nIndex), 1, 1, &nValue ) )
{
return 0;
}
if( (0xFF & nTestValue) != nValue )
{
return 0;
}
}
// All 5's
/* AT25F2048_EraseFlash();
// write incrementing values to each FLASH location
nTestValue = 0x55;
for(nIndex = 0; nIndex < FLASH_TEST_SIZE; nIndex++ )
{
if( NO_ERR != AT25F2048_WriteFlash( FLASH_START + FLASH_TEST_START + nIndex, nTestValue ) )
{
return 0;
}
}
// verify incrementing values
nTestValue = 0x55;
for(nIndex = 0; nIndex < FLASH_TEST_SIZE; nIndex++ )
{
if( NO_ERR != AT25F2048_ReadFlash( FLASH_START + FLASH_TEST_START + nIndex, &nValue ) )
{
return 0;
}
if( (0xFF & nTestValue) != nValue )
{
return 0;
}
}
*/
// Close the Device
AFP_Error = ADIAT25F2048EntryPoint.adi_pdd_Close(NULL);
return bError;
}
//----------- S e t u p F o r F l a s h ( ) ----------//
//
// PURPOSE
// Perform necessary setup for the processor to talk to the
// flash such as external memory interface registers, etc.
//
// RETURN VALUE
// ERROR_CODE - value if any error occurs during Opcode scan
// NO_ERR - otherwise
static ERROR_CODE SetupForFlash(void)
{
ERROR_CODE ErrorCode = NO_ERR;
int nStatus;
SRU(SPI_CLK_O,DPI_PB03_I);
SRU(HIGH,DPI_PBEN03_I);
// for the flag pins to act as chip select
SRU(FLAG4_O, DPI_PB05_I);
SRU(HIGH, DPI_PBEN05_I);
//First set flag 4 as an output
sysreg_bit_set( sysreg_FLAGS, FLG4O ); //asm("bit set flags FLG4O;");
sysreg_bit_set( sysreg_FLAGS, FLG4 ); //asm("bit set flags FLG4;"); //Logic high
*pSPIBAUD = BAUD_RATE_DIVISOR;
// reset the flash to a known state
COMMAND_STRUCT SCmdStruct;
SCmdStruct.SReset.ulFlashStartAddr = FLASH_START_ADDR;
AFP_Error = ADIAT25F2048EntryPoint.adi_pdd_Control(NULL, CNTRL_RESET, &SCmdStruct );
return ErrorCode;
}
//----------- G e t F l a s h I n f o ( ) ----------//
//
// PURPOSE
// Get the manufacturer code and device code
//
// RETURN VALUE
// ERROR_CODE - value if any error occurs
// NO_ERR - otherwise
//
// CHANGES
// 9-28-2005 Created
static ERROR_CODE GetFlashInfo(void)
{
ERROR_CODE ErrorCode = NO_ERR; //return error code
static GET_CODES_STRUCT SGetCodes; //structure for GetCodes
COMMAND_STRUCT CmdStruct;
//setup code so that flash programmer can just read memory instead of call GetCodes().
CmdStruct.SGetCodes.pManCode = (unsigned long *)&AFP_ManCode;
CmdStruct.SGetCodes.pDevCode = (unsigned long *)&AFP_DevCode;
CmdStruct.SGetCodes.ulFlashStartAddr = FLASH_START_ADDR;
AFP_Error = ADIAT25F2048EntryPoint.adi_pdd_Control(NULL, CNTRL_GET_CODES, &CmdStruct );
return(ErrorCode);
}
//////////////////////////////////////////////////////////////
// ERROR_CODE EraseFlash()
//
// Sends an "erase all" command to the flash.
//
//////////////////////////////////////////////////////////////
static ERROR_CODE EraseFlash(void)
{
COMMAND_STRUCT CmdStruct;
ERROR_CODE ErrorCode = NO_ERR; //return error code
#ifdef _FULL_FLASH_TEST_ // erase the entire flash
CmdStruct.SEraseAll.ulFlashStartAddr = FLASH_START_ADDR; //FlashStartAddress
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Control(NULL, CNTRL_ERASE_ALL, &CmdStruct );
#else // only erase last sector
CmdStruct.SEraseSect.nSectorNum = (NUM_SECTORS - 1); // Sector Number to erase
CmdStruct.SEraseSect.ulFlashStartAddr = FLASH_START_ADDR; // FlashStartAddress
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Control(NULL, CNTRL_ERASE_SECT, &CmdStruct);
#endif
return ErrorCode;
}
//----------- W r i t e D a t a ( ) ----------//
//
// PURPOSE
// Write a buffer to flash device.
//
// INPUTS
// unsigned long ulStart - address in flash to start the writes at
// long lCount - number of elements to write, in this case bytes
// long lStride - number of locations to skip between writes
// int *pnData - pointer to data buffer
//
// RETURN VALUE
// ERROR_CODE - value if any error occurs during writing
// NO_ERR - otherwise
//
// CHANGES
// 9-28-2005 Created
ERROR_CODE WriteData( unsigned long ulStart, long lCount, long lStride, int *pnData )
{
long i = 0; // loop counter
ERROR_CODE ErrorCode = NO_ERR; // tells whether there was an error trying to write
int nCompare = 0; // value that we use to verify flash
bool bVerifyError = FALSE; // lets us know if there was a verify error
unsigned long ulAbsoluteAddr; // current address to write
unsigned long ulSector = 0; // sector number to verify address
ADI_DEV_1D_BUFFER WriteBuff; // buffer pointer
ADI_DEV_1D_BUFFER ReadBuff; // buffer pointer
ulAbsoluteAddr = FLASH_START_ADDR + ulStart;
COMMAND_STRUCT CmdStruct; //structure for GetSectStartEnd
// if the user wants to verify then do it
if( AFP_Verify == TRUE )
{
// write the buffer up to BUFFER_SIZE items
for (i = 0; ( i < lCount ) && ( ErrorCode == NO_ERR ); i++, ulAbsoluteAddr += lStride)
{
// check to see that the address is within a valid sector
CmdStruct.SGetSectNum.ulOffset = ulAbsoluteAddr;
CmdStruct.SGetSectNum.pSectorNum = &ulSector;
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Control(NULL, CNTRL_GET_SECTNUM, &CmdStruct );
if( NO_ERR == ErrorCode )
{
// unlock the flash, do the write, increase shift, and wait for completion
WriteBuff.Data = (void *)&pnData[i];
WriteBuff.pAdditionalInfo = (void *)&ulAbsoluteAddr ;
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Write(NULL, ADI_DEV_1D, (ADI_DEV_BUFFER *)&WriteBuff);
ReadBuff.Data = (void *)&nCompare;
ReadBuff.pAdditionalInfo = (void *)&ulAbsoluteAddr ;
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Read(NULL, ADI_DEV_1D, (ADI_DEV_BUFFER *)&ReadBuff);
if( ( nCompare ) != (pnData[i] & 0xFF) )
{
bVerifyError = TRUE;
break;
}
}
else
{
return ErrorCode;
}
}
// return appropriate error code if there was a verification error
if( bVerifyError == TRUE )
return VERIFY_WRITE;
}
// the user does not want to verify
else
{
// write the buffer up to BUFFER_SIZE items
for (i = 0; ( i < lCount ) && ( ErrorCode == NO_ERR ); i++, ulAbsoluteAddr += lStride)
{
// check to see that the address is within a valid sector
CmdStruct.SGetSectNum.ulOffset = ulAbsoluteAddr;
CmdStruct.SGetSectNum.pSectorNum = &ulSector;
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Control(NULL, CNTRL_GET_SECTNUM, &CmdStruct );
if( NO_ERR == ErrorCode )
{
// unlock the flash, do the write, increase shift, and wait for completion
WriteBuff.Data = (void *)&pnData[i] ;
WriteBuff.pAdditionalInfo = (void *)&ulAbsoluteAddr ;
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Write(NULL, ADI_DEV_1D, (ADI_DEV_BUFFER *)&WriteBuff);
}
else
{
return ErrorCode;
}
}
}
// return the appropriate error code
return ErrorCode;
}
//----------- R e a d D a t a ( ) ----------//
//
// PURPOSE
// Read a buffer from flash device.
//
// INPUTS
// unsigned long ulStart - address in flash to start the reads at
// long lCount - number of elements to read, in this case bytes
// long lStride - number of locations to skip between reads
// int *pnData - pointer to data buffer to fill
//
// RETURN VALUE
// ERROR_CODE - value if any error occurs during reading
// NO_ERR - otherwise
//
// CHANGES
// 9-28-2005 Created
static ERROR_CODE ReadData( unsigned long ulStart, long lCount, long lStride, int *pnData )
{
long i = 0; // loop counter
ERROR_CODE ErrorCode = NO_ERR; // tells whether there was an error trying to read
unsigned long ulAbsoluteAddr; // current address to read
unsigned long ulSector = 0; // sector number to verify address
unsigned long ulMask =0xffff;
ADI_DEV_1D_BUFFER ReadBuff; // buffer pointer
COMMAND_STRUCT CmdStruct; //structure for GetSectStartEnd
ulAbsoluteAddr = FLASH_START_ADDR + ulStart;
// read the buffer up to BUFFER_SIZE items
for (i = 0; (i < lCount) && (i < BUFFER_SIZE); i++, ulAbsoluteAddr += lStride)
{
// check to see that the address is within a valid sector
CmdStruct.SGetSectNum.ulOffset = ulAbsoluteAddr;
CmdStruct.SGetSectNum.pSectorNum = &ulSector;
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Control(NULL, CNTRL_GET_SECTNUM, &CmdStruct );
if( NO_ERR == ErrorCode )
{
ReadBuff.Data = (void *)&pnData[i];
ReadBuff.pAdditionalInfo = (void *)&ulAbsoluteAddr ;
ErrorCode = (ERROR_CODE) ADIAT25F2048EntryPoint.adi_pdd_Read(NULL, ADI_DEV_1D, (ADI_DEV_BUFFER *)&ReadBuff);
}
else
{
return ErrorCode;
}
}
// return the appropriate error code
return ErrorCode;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -