|
模擬IIC讀寫AT24C32,只能寫前1024個(gè)Byte,讀是可以全部讀,不知道是什么問題
uint8_t ee_Test(void)
{
uint16_t i;
uint8_t write_buf[TEXT_SIZE] = {0};
uint8_t read_buf[TEXT_SIZE] = {0};
/*-----------------------------------------------------------------------------------*/
if (ee_CheckOk() == 0)
{
/* 沒有檢測到EEPROM */
printf("沒有檢測到串行EEPROM!\r\n");
return 0;
}
/*------------------------------------------------------------------------------------*/
/* 填充測試緩沖區(qū) */
for (i = 0; i < TEXT_SIZE; i++)
{
write_buf = 0xFF;
}
/*------------------------------------------------------------------------------------*/
if (ee_WriteBytes(write_buf, 2048, TEXT_SIZE) == 0)
{
printf("寫eeprom出錯(cuò)!\r\n");
return 0;
}
else
{
printf("寫eeprom成功!\r\n");
}
/*寫完之后需要適當(dāng)?shù)难訒r(shí)再去讀,不然會(huì)出錯(cuò)*/
ee_Delay(0x0FFFFF);
/*-----------------------------------------------------------------------------------*/
if (ee_ReadBytes(read_buf, 0, TEXT_SIZE) == 0)
{
printf("讀eeprom出錯(cuò)!\r\n");
return 0;
}
else
{
printf("讀eeprom成功,數(shù)據(jù)如下:\r\n");
}
/*-----------------------------------------------------------------------------------*/
for (i = 0; i < EEPROM_SIZE; i++)
{
//if(read_buf != write_buf)
//{
//printf("0x%02X ", read_buf);
//printf("錯(cuò)誤:EEPROM讀出與寫入的數(shù)據(jù)不一致");
//return 0;
//}
printf(" %02X", read_buf);
if ((i & 63) == 63)
{
printf("\r\n");
}
}
printf("eeprom讀寫測試成功\r\n");
return 1;
}
/*********************************************END OF FILE**********************/
uint8_t ee_ReadBytes(uint8_t *_pReadBuf, uint16_t _usAddress, uint16_t _usSize)
{
uint16_t i;
/* 采用串行EEPROM隨即讀取指令序列,連續(xù)讀取若干字節(jié) */
/* 第1步:發(fā)起I2C總線啟動(dòng)信號 */
i2c_Start();
/* 第2步:發(fā)起控制字節(jié),高7bit是地址,bit0是讀寫控制位,0表示寫,1表示讀 */
i2c_SendByte(EEPROM_DEV_ADDR | EEPROM_I2C_WR); /* 此處是寫指令 */
/* 第3步:等待ACK */
if (i2c_WaitAck() != 0)
{
goto cmd_fail; /* EEPROM器件無應(yīng)答 */
}
/* 第4步:發(fā)送字節(jié)地址,24C02只有256字節(jié),因此1個(gè)字節(jié)就夠了,如果是24C04以上,那么此處需要連發(fā)多個(gè)地址 */
i2c_SendByte(_usAddress >> 8);
/* 第5步:等待ACK */
if (i2c_WaitAck() != 0)
{
goto cmd_fail; /* EEPROM器件無應(yīng)答 */
}
/* 第4步:發(fā)送字節(jié)地址,24C02只有256字節(jié),因此1個(gè)字節(jié)就夠了,如果是24C04以上,那么此處需要連發(fā)多個(gè)地址 */
i2c_SendByte(_usAddress);
/* 第5步:等待ACK */
if (i2c_WaitAck() != 0)
{
goto cmd_fail; /* EEPROM器件無應(yīng)答 */
}
/* 第6步:重新啟動(dòng)I2C總線。前面的代碼的目的向EEPROM傳送地址,下面開始讀取數(shù)據(jù) */
i2c_Start();
/* 第7步:發(fā)起控制字節(jié),高7bit是地址,bit0是讀寫控制位,0表示寫,1表示讀 */
i2c_SendByte(EEPROM_DEV_ADDR | EEPROM_I2C_RD); /* 此處是讀指令 */
/* 第8步:發(fā)送ACK */
if (i2c_WaitAck() != 0)
{
goto cmd_fail; /* EEPROM器件無應(yīng)答 */
}
/* 第9步:循環(huán)讀取數(shù)據(jù) */
for (i = 0; i < _usSize; i++)
{
_pReadBuf = i2c_ReadByte(); /* 讀1個(gè)字節(jié) */
/* 每讀完1個(gè)字節(jié)后,需要發(fā)送Ack, 最后一個(gè)字節(jié)不需要Ack,發(fā)Nack */
if (i != _usSize - 1)
{
i2c_Ack(); /* 中間字節(jié)讀完后,CPU產(chǎn)生ACK信號(驅(qū)動(dòng)SDA = 0) */
}
else
{
i2c_NAck(); /* 最后1個(gè)字節(jié)讀完后,CPU產(chǎn)生NACK信號(驅(qū)動(dòng)SDA = 1) */
}
}
/* 發(fā)送I2C總線停止信號 */
i2c_Stop();
return 1; /* 執(zhí)行成功 */
cmd_fail: /* 命令執(zhí)行失敗后,切記發(fā)送停止信號,避免影響I2C總線上其他設(shè)備 */
/* 發(fā)送I2C總線停止信號 */
i2c_Stop();
return 0;
}
/*
*********************************************************************************************************
* 函 數(shù) 名: ee_WriteBytes
* 功能說明: 向串行EEPROM指定地址寫入若干數(shù)據(jù),采用頁寫操作提高寫入效率
* 形 參:_usAddress : 起始地址
* _usSize : 數(shù)據(jù)長度,單位為字節(jié)
* _pWriteBuf : 存放讀到的數(shù)據(jù)的緩沖區(qū)指針
* 返 回 值: 0 表示失敗,1表示成功
*********************************************************************************************************
*/
uint8_t ee_WriteBytes(uint8_t *_pWriteBuf, uint16_t _usAddress, uint16_t _usSize)
{
uint16_t i,m;
uint16_t usAddr;
/*
寫串行EEPROM不像讀操作可以連續(xù)讀取很多字節(jié),每次寫操作只能在同一個(gè)page。
對于24xx02,page size = 8
簡單的處理方法為:按字節(jié)寫操作模式,沒寫1個(gè)字節(jié),都發(fā)送地址
為了提高連續(xù)寫的效率: 本函數(shù)采用page wirte操作。
*/
usAddr = _usAddress;
for (i = 0; i < _usSize; i++)
{
/* 當(dāng)發(fā)送第1個(gè)字節(jié)或是頁面首地址時(shí),需要重新發(fā)起啟動(dòng)信號和地址 */
if ((i == 0) || (usAddr & (EEPROM_PAGE_SIZE - 1)) == 0)
{
/* 第0步:發(fā)停止信號,啟動(dòng)內(nèi)部寫操作 */
i2c_Stop();
/* 通過檢查器件應(yīng)答的方式,判斷內(nèi)部寫操作是否完成, 一般小于 10ms
CLK頻率為200KHz時(shí),查詢次數(shù)為30次左右
*/
for (m = 0; m < 1000; m++)
{
/* 第1步:發(fā)起I2C總線啟動(dòng)信號 */
i2c_Start();
/* 第2步:發(fā)起控制字節(jié),高7bit是地址,bit0是讀寫控制位,0表示寫,1表示讀 */
i2c_SendByte(EEPROM_DEV_ADDR | EEPROM_I2C_WR); /* 此處是寫指令 */
/* 第3步:發(fā)送一個(gè)時(shí)鐘,判斷器件是否正確應(yīng)答 */
if (i2c_WaitAck() == 0)
{
break;
}
}
if (m == 1000)
{
goto cmd_fail; /* EEPROM器件寫超時(shí) */
}
/* 第4步:發(fā)送字節(jié)地址,24C02只有256字節(jié),因此1個(gè)字節(jié)就夠了,如果是24C04以上,那么此處需要連發(fā)多個(gè)地址 */
i2c_SendByte(usAddr >> 8);
/* 第5步:等待ACK */
if (i2c_WaitAck() != 0)
{
goto cmd_fail; /* EEPROM器件無應(yīng)答 */
}
/* 第4步:發(fā)送字節(jié)地址,24C02只有256字節(jié),因此1個(gè)字節(jié)就夠了,如果是24C04以上,那么此處需要連發(fā)多個(gè)地址 */
i2c_SendByte(usAddr);
/* 第5步:等待ACK */
if (i2c_WaitAck() != 0)
{
goto cmd_fail; /* EEPROM器件無應(yīng)答 */
}
}
/* 第6步:開始寫入數(shù)據(jù) */
i2c_SendByte(_pWriteBuf);
/* 第7步:發(fā)送ACK */
if (i2c_WaitAck() != 0)
{
goto cmd_fail; /* EEPROM器件無應(yīng)答 */
}
usAddr++; /* 地址增1 */
}
/* 命令執(zhí)行成功,發(fā)送I2C總線停止信號 */
i2c_Stop();
return 1;
cmd_fail: /* 命令執(zhí)行失敗后,切記發(fā)送停止信號,避免影響I2C總線上其他設(shè)備 */
/* 發(fā)送I2C總線停止信號 */
i2c_Stop();
return 0;
}
void ee_Erase(void)
{
uint16_t i;
uint8_t buf[EEPROM_SIZE];
/* 填充緩沖區(qū) */
for (i = 0; i < EEPROM_SIZE; i++)
{
buf = 0xFF;
}
/* 寫EEPROM, 起始地址 = 0,數(shù)據(jù)長度為 256 */
if (ee_WriteBytes(buf, 0, EEPROM_SIZE) == 0)
{
printf("擦除eeprom出錯(cuò)!\r\n");
return;
}
else
{
printf("擦除eeprom成功!\r\n");
}
}
/*--------------------------------------------------------------------------------------------------*/
static void ee_Delay(__IO uint32_t nCount) //簡單的延時(shí)函數(shù)
{
for(; nCount != 0; nCount--);
}
|
|