?? flash.c
字號:
flashstate = AT91F_DataFlashWaitReady(DataFlash.pDataFlashDesc, AT91C_DATAFLASH_TIMEOUT);
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
if(start_addr > 0 || len < 512)
{
flashstate = AT91F_DataFlashSendCommand(&DataFlash,0x53, 4, block_number * 528);//追寫
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
flashstate = AT91F_DataFlashWaitReady(DataFlash.pDataFlashDesc, AT91C_DATAFLASH_TIMEOUT);
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
flashstate = AT91F_DataFlashPagePgmBuf(&DataFlash, buf, block_number * 528 + start_addr, len);
}
else
{
flashstate = AT91F_DataFlashPagePgmBuf(&DataFlash, buf, block_number * DataFlash.pDevice->pages_size, DataFlash.pDevice->pages_size);
}
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
flashstate = AT91F_DataFlashWaitReady(DataFlash.pDataFlashDesc, AT91C_DATAFLASH_TIMEOUT);
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
//發(fā)一個命令,校驗(yàn)一下寫進(jìn)去的東西是否正確
flashstate = AT91F_DataFlashSendCommand (&DataFlash, 0x60, 4, block_number * DataFlash.pDevice->pages_size);
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
flashstate = AT91F_DataFlashWaitReady(DataFlash.pDataFlashDesc, AT91C_DATAFLASH_TIMEOUT);
if(DataFlash.pDataFlashDesc->DataFlash_state & 0x40)
{
return 0;
}
return flashstate;
}
/*****************************************************/
//建立文件系統(tǒng)
//剛開始時認(rèn)為每個扁區(qū)都是好的.
//將文件索引表中的每一項(xiàng)都指向每一個扁區(qū)(第i項(xiàng)指向第i個扁區(qū))
//參數(shù): file_index_startblock 表示文件系統(tǒng)的所在位置的起始扁區(qū)
//返回值如果為DATAFLASH_OK(0x1)就認(rèn)為是成功
//否則返回DATAFLASH_BUSY(0x0) 或 DATAFLASH_ERROR(0x2)
/****************************************************/
unsigned int mk_fs_flash( )
{
unsigned int i,j;
unsigned short block_cs = 0;
unsigned short file_index[264];
unsigned short file_index2[264];
unsigned short file_index_startblock;
AT91S_DataFlashStatus flashstate;
file_index_startblock = 8128;
for(i = 0; i < 32; i++)
{
block_cs = 0;
for(j = 0; j < 256; j++)
{
file_index[j] = i* 256 + j;
file_index2[j] = file_index[j];
block_cs +=file_index[j];
}
file_index[FLASH_CS_POS] = block_cs;//cs校驗(yàn)
file_index2[FLASH_CS_POS] = block_cs;//cs校驗(yàn)
//寫主文件索引表
flashstate = write_flash_block((unsigned char *)file_index, (file_index_startblock + i));
if(DATAFLASH_OK != flashstate)//RETURN VALUE EITHER DATAFLASH_ERROR OR DATAFLASH_OK
{
return flashstate;
}
//寫從文件索引表
flashstate = write_flash_block((unsigned char *)file_index2, (file_index_startblock + i + 32));
if(DATAFLASH_OK != flashstate)//RETURN VALUE EITHER DATAFLASH_ERROR OR DATAFLASH_OK
{
return flashstate;
}
}
return (DATAFLASH_OK);
}
/*****************************************************/
//檢驗(yàn)建立的文件系統(tǒng)是否是正確的
//返回值如果不為DATAFLASH_OK就認(rèn)為是錯的
/****************************************************/
unsigned int check_mk_fs( )
{
unsigned int i,j;
unsigned short block_cs = 0;
unsigned short file_index[264];
unsigned char flash_readbuf[528];
unsigned short *pread;
unsigned short file_index_startblock;
AT91S_DataFlashStatus flashstate;
file_index_startblock = 8128;
for(i = 0; i < 32; i++)
{
block_cs = 0;
for(j = 0; j < 256; j++)
{
file_index[j] = i* 256 + j;
block_cs +=file_index[j];
}
file_index[FLASH_CS_POS] = block_cs;//cs校驗(yàn)
//檢驗(yàn)主文件索引表
flashstate = read_flash_block(flash_readbuf, (file_index_startblock + i));//Added by ChengDong Lu at 04/19/2006
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
pread = (unsigned short *)flash_readbuf;
for(j = 0; j < 257; j++)
{
if(pread[j] != file_index[j])
{
return (0);
}
}
//檢驗(yàn)從文件索引表
flashstate = read_flash_block(flash_readbuf, (file_index_startblock + i + 32));//Added by ChengDong Lu at 04/19/2006
if(DATAFLASH_OK != flashstate)
{
return flashstate;
}
pread = (unsigned short *)flash_readbuf;
for(j = 0; j < 257; j++)
{
if(pread[j] != file_index[j])
{
return (0);
}
}
}
return (DATAFLASH_OK);
}
/*******************************************************/
//功能:
// 讀文件索引表中指定的扃區(qū)
// block_number是從文件索引表的第一個扁區(qū)(8128)開始計(jì)數(shù)
// 先從主文件索引表讀出當(dāng)前的扇區(qū)
// 如果讀出成功(能正常的讀出來,并且算出的cs和它原來的cs相等),則返回 1
// 否則續(xù)繼讀從文件索引表相應(yīng)的扃區(qū)
// 如果讀出成功(能正常的讀出來,并且算出的cs和它原來的cs相等),則返回 1
// 否則返回0
/*******************************************************/
unsigned int read_fs_index(unsigned char *fs_buf,unsigned int block_number)
{
unsigned short i, j;
unsigned short block_cs = 0;
unsigned short *pread;
unsigned short file_index_startblock;
AT91S_DataFlashStatus flashstate;
file_index_startblock = 8128;
for(i = 0; i < 2; i++)
{
flashstate = read_flash_block(fs_buf, (file_index_startblock + block_number + i*32));
if(DATAFLASH_OK != flashstate)
{
continue;
}
pread = (unsigned short *)fs_buf;
block_cs = 0;
for(j = 0; j < 256; j++)
{
block_cs += pread[j];
}
if( pread[FLASH_CS_POS] == block_cs) //成功
{
return (1);
}
}
return (0);
}
/***************************************************************************/
//功能:
// 寫文件索引表中的扁區(qū)
// block_number是從文件索引表的第一個扁區(qū)(8128)開始計(jì)數(shù)
// 先寫主文件索引表相應(yīng)的扃區(qū)
// 再從文件索引表相應(yīng)的扃區(qū)
// 如果有一個以上能寫成功則返回 1
// 否則返回0
/***************************************************************************/
unsigned int write_fs_index(unsigned char *fs_buf,unsigned int block_number)
{
unsigned short i,j;
unsigned short file_index_startblock;
unsigned short write_fs_flag = 0;
unsigned short *pread;
unsigned short block_cs = 0;
unsigned short slater_buf[264];
AT91S_DataFlashStatus flashstate;
file_index_startblock = 8128;
pread = (unsigned short *)fs_buf;
for(j = 0; j < 256; j++)
{
block_cs += pread[j];
slater_buf[j] = fs_buf[j];
}
slater_buf[FLASH_CS_POS] = block_cs;
fs_buf[FLASH_CS_POS] = block_cs;
for(i = 0; i < 2; i++)
{
if(i == 0)
{
flashstate = write_flash_block(fs_buf, (file_index_startblock + block_number + i*32));
if(DATAFLASH_OK != flashstate)
{
write_fs_flag++;
continue;
}
}
else
{
flashstate = write_flash_block((unsigned char *)slater_buf, (file_index_startblock + block_number + i*32));
if(DATAFLASH_OK != flashstate)
{
write_fs_flag++;
continue;
}
}
}
if(write_fs_flag > 0)
{
return (0);
}
else
{
return (1);
}
}
/*************************************************************************************************/
// flash的分配
//
//編程日志 第 0 - 9 扇區(qū)(10)
//報(bào)警日志 第 10 - 19 扇區(qū)(10)
//操作日志 第 20 - 29 扇區(qū)(10)
//任務(wù) 第 30 - 7069 扇區(qū)(64 *110 = 7040)
//其它數(shù)據(jù) 第 7070 - 7679 扇區(qū)
//備份區(qū) 第 7680 - 8126 扇區(qū)(要是以上的扇區(qū)壞了就從這個備份區(qū)中找一個好的扇區(qū)映射過去)
//文件索引區(qū) 第 8128 - 8191 扇區(qū)(只要文件索引區(qū)中有一個扇區(qū)是壞的就認(rèn)為這個flash不可用)
//功能:
// 從備份區(qū)(7680 - 7679)中找出一塊好的未用的扁區(qū)
// 它對應(yīng)于文件索引表中的第30塊扁區(qū)所有的數(shù)據(jù)項(xiàng)(0 - 255)
// 和第31塊扁區(qū)中的第0 - 191項(xiàng)(192 - 255 這64項(xiàng)對應(yīng)于文件表所占用的扁區(qū))
//返回值:
// 如果能找到則將這個扁區(qū)置為已用,返回這個塊扁區(qū)的編號
// 否則返回0
//
// 文件索引項(xiàng)描述符
// --------------------------------------------------
// | 15 | 14 | 13 - 0 |
// |----------|-------------|-----------------------|
// | 0 | 0 | 對應(yīng)扁區(qū)的編號 |
// |----------|-------------|-----------------------|
//
//
//備份區(qū)對應(yīng)的文件索引項(xiàng)描述符
// --------------------------------------------------
// | 15 | 14 | 13 - 0 |
// |----------|-------------|-----------------------|
// |使用標(biāo)志 | 好壞標(biāo)志 | 對應(yīng)扁區(qū)的編號 |
// |----------|-------------|-----------------------|
// | 0:未用 |0 : 好 | |
// | 1:已用 |1 : 壞 | |
// --------------------------------------------------
/*************************************************************************************************/
unsigned int find_empty_block(void)
{
unsigned short fs_buf[264];
unsigned short find_number;
unsigned short i,j;
unsigned int find_result;
for(i = 0; i < 2; i++)
{
find_result = read_fs_index((unsigned char*)fs_buf, 30 + i);
if(i == 0)
{
find_number = 256;
}
else
{
find_number = 192;
}
if(find_result == 1)
{
for(j = 0; j < find_number; j++)
{
if((fs_buf[j] & 0xc000) == 0)
{
fs_buf[j] = fs_buf[j] | 0x1000;
find_result = write_fs_index((unsigned char *)fs_buf,30 + i);
if(find_result == 1)
{
return (fs_buf[j]);
}
else
{
return (0);
}
}
}
}
}
return (0);
}
/*******************************************************************************/
//功能: 讀flash
//成功返回 1 ,否則返回0
/*******************************************************************************/
int flash_read(void *buf, int len, unsigned int BaseAddr)
{
unsigned int fs_block;
unsigned int read_block; //當(dāng)前要讀的扁區(qū)
unsigned int start_addr; //要讀的數(shù)據(jù)所在扇區(qū)的起始地址
unsigned int state;
unsigned int fs_index;
unsigned char * write_pos;
unsigned char read_temp[528];
unsigned short fs_temp[264];
if((BaseAddr + len) >= (7680 * 512))
{
return (0);
}
fs_block = (BaseAddr / 512) /256; //要讀的扁區(qū)的索引項(xiàng)在文件索引表中的扇區(qū)號
fs_index = (BaseAddr / 512) % 256; //要讀的扇區(qū)的索引項(xiàng)在文件索引表中的扇區(qū)號的對應(yīng)的位置
start_addr = BaseAddr % 512; //要讀的數(shù)據(jù)所在扇區(qū)的位置
state = read_fs_index((unsigned char *)fs_temp, fs_block);//Closed by ChengDong Lu at 04/20/2006
// state = read_fs_index((unsigned char *)fs_temp, fs_block - 8128);
if(state != 1)
{
return (0);
}
write_pos = (unsigned char*)buf;//Added by ChengDong Lu at 04/19/2006
while(len > 0)
{
if(fs_index >=256)
{
fs_block++;
state = read_fs_index((unsigned char *)fs_temp, fs_block);//Closed by ChengDong Lu at 04/20/2006
// state = read_fs_index((unsigned char *)fs_temp, fs_block - 8128);
if(state != 1)
{
return (0);
}
?? 快捷鍵說明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -