EEPROM 寫數據流程
第一步,首先是 I2C 的起始信號,接著跟上首字節,也就是我們前邊講的 I2C 的器件地址,并且在讀寫方向上選擇“寫”操作。
第二步,發送數據的存儲地址。24C02 一共 256 個字節的存儲空間,地址從 0x00~0xFF,我們想把數據存儲在哪個位置,此刻寫的就是哪個地址。
第三步,發送要存儲的數據第一個字節、第二個字節??注意在寫數據的過程中,EEPROM 每個字節都會回應一個“應答位 0”,來告訴我們寫 EEPROM 數據成功,如果沒有回應答位,說明寫入不成功。
在寫數據的過程中,每成功寫入一個字節,EEPROM 存儲空間的地址就會自動加 1,當加到 0xFF 后,再寫一個字節,地址會溢出又變成了 0x00。
EEPROM 讀數據流程
第一步,首先是 I2C 的起始信號,接著跟上首字節,也就是我們前邊講的 I2C 的器件地址,并且在讀寫方向上選擇“寫”操作。這個地方可能有同學會詫異,我們明明是讀數據為何方向也要選“寫”呢?剛才說過了,24C02 一共有 256 個地址,我們選擇寫操作,是為了把所要讀的數據的存儲地址先寫進去,告訴 EEPROM 我們要讀取哪個地址的數據。這就如同我們打電話,先撥總機號碼(EEPROM 器件地址),而后還要繼續撥分機號碼(數據地址),而撥分機號碼這個動作,主機仍然是發送方,方向依然是“寫”。
第二步,發送要讀取的數據的地址,注意是地址而非存在 EEPROM 中的數據,通知EEPROM 我要哪個分機的信息。
第三步,重新發送 I2C 起始信號和器件地址,并且在方向位選擇“讀”操作。
這三步當中,每一個字節實際上都是在“寫”,所以每一個字節 EEPROM 都會回應一個“應答位 0”。
第四步,讀取從器件發回的數據,讀一個字節,如果還想繼續讀下一個字節,就發送一個“應答位 ACK(0)”,如果不想讀了,告訴 EEPROM,我不想要數據了,別再發數據了,那就發送一個“非應答位 NAK(1)”。
和寫操作規則一樣,我們每讀一個字節,地址會自動加 1,那如果我們想繼續往下讀,給 EEPROM 一個 ACK(0)低電平,那再繼續給 SCL 完整的時序,EEPROM 會繼續往外送數據。如果我們不想讀了,要告訴 EEPROM 不要數據了,那我們直接給一個 NAK(1)高電平即可。這個地方大家要從邏輯上理解透徹,不能簡單的靠死記硬背了,一定要理解明白。梳理一下幾個要點:
A、在本例中單片機是主機,24C02 是從機;
B、無論是讀是寫,SCL 始終都是由主機控制的;
C、寫的時候應答信號由從機給出,表示從機是否正確接收了數據;
D、讀的時候應答信號則由主機給出,表示是否繼續讀下去。
那我們下面寫一個程序,讀取 EEPROM 的 0x02 這個地址上的一個數據,不管這個數據之前是多少,我們都將讀出來的數據加 1,再寫到 EEPROM 的 0x02 這個地址上。此外我們將 I2C 的程序建立一個文件,寫一個 I2C.c 程序文件,形成我們又一個程序模塊。大家也可以看出來,我們連續的這幾個程序,Lcd1602.c 文件里的程序都是一樣的,今后我們大家寫1602 顯示程序也可以直接拿過去用,大大提高了程序移植的方便性。
/******************************I2C.c 文件程序源代碼******************************/
#include
#include
#define I2CDelay() {_nop_();_nop_();_nop_();_nop_();}
sbit I2C_SCL = P3^7;
sbit I2C_SDA = P3^6;
/* 產生總線起始信號 */
void I2CStart(){
I2C_SDA = 1; //首先確保 SDA、SCL 都是高電平
I2C_SCL = 1;
I2CDelay();
I2C_SDA = 0; //先拉低 SDA
I2CDelay();
I2C_SCL = 0; //再拉低 SCL
}
/* 產生總線停止信號 */
void I2CStop(){
I2C_SCL = 0; //首先確保 SDA、SCL 都是低電平
I2C_SDA = 0;
I2CDelay();
I2C_SCL = 1; //先拉高 SCL
I2CDelay();
I2C_SDA = 1; //再拉高 SDA
I2CDelay();
}
/* I2C 總線寫操作,dat-待寫入字節,返回值-從機應答位的值 */
bit I2CWrite(unsigned char dat){
bit ack; //用于暫存應答位的值
unsigned char mask; //用于探測字節內某一位值的掩碼變量
for (mask=0x80; mask!=0; mask》》=1){ //從高位到低位依次進行
if ((mask&dat) == 0){ //該位的值輸出到 SDA 上
I2C_SDA = 0;
}else{
I2C_SDA = 1;
}
I2CDelay();
I2C_SCL = 1; //拉高 SCL
I2CDelay();
I2C_SCL = 0; //再拉低 SCL,完成一個位周期
}
I2C_SDA = 1; //8 位數據發送完后,主機釋放 SDA,以檢測從機應答
I2CDelay();
I2C_SCL = 1; //拉高 SCL
ack = I2C_SDA; //讀取此時的 SDA 值,即為從機的應答值
I2CDelay();
I2C_SCL = 0; //再拉低 SCL 完成應答位,并保持住總線
//應答值取反以符合通常的邏輯:
//0=不存在或忙或寫入失敗,1=存在且空閑或寫入成功
return (~ack);
}
/* I2C 總線讀操作,并發送非應答信號,返回值-讀到的字節 */
unsigned char I2CReadNAK(){
unsigned char mask;
unsigned char dat;
I2C_SDA = 1; //首先確保主機釋放 SDA
for (mask=0x80; mask!=0; mask》》=1){ //從高位到低位依次進行
I2CDelay();
I2C_SCL = 1; //拉高 SCL
if(I2C_SDA == 0){ //讀取 SDA 的值
dat &= ~mask; //為 0 時,dat 中對應位清零
}else{
dat |= mask; //為 1 時,dat 中對應位置 1
}
I2CDelay();
I2C_SCL = 0; //再拉低 SCL,以使從機發送出下一位
}
I2C_SDA = 1; //8 位數據發送完后,拉高 SDA,發送非應答信號
I2CDelay();
I2C_SCL = 1; //拉高 SCL
I2CDelay();
I2C_SCL = 0; //再拉低 SCL 完成非應答位,并保持住總線
return dat;
}
/* I2C 總線讀操作,并發送應答信號,返回值-讀到的字節 */
unsigned char I2CReadACK(){
unsigned char mask;
unsigned char dat;
I2C_SDA = 1; //首先確保主機釋放 SDA
for (mask=0x80; mask!=0; mask》》=1){ //從高位到低位依次進行
I2CDelay();
I2C_SCL = 1; //拉高 SCL
if(I2C_SDA == 0){ //讀取 SDA 的值
dat &= ~mask; //為 0 時,dat 中對應位清零
}else{
dat |= mask; //為 1 時,dat 中對應位置 1
}
I2CDelay();
I2C_SCL = 0; //再拉低 SCL,以使從機發送出下一位
}
I2C_SDA = 0; //8 位數據發送完后,拉低 SDA,發送應答信號
I2CDelay();
I2C_SCL = 1; //拉高 SCL
I2CDelay();
I2C_SCL = 0; //再拉低 SCL 完成應答位,并保持住總線
return dat;
}
I2C.c 文件提供了 I2C 總線所有的底層操作函數,包括起始、停止、字節寫、字節讀+應答、字節讀+非應答。
/***************************Lcd1602.c 文件程序源代碼*****************************/
#include
#define LCD1602_DB P0
sbit LCD1602_RS = P1^0;
sbit LCD1602_RW = P1^1;
sbit LCD1602_E = P1^5;
/* 等待液晶準備好 */
void LcdWaitReady(){
unsigned char sta;
LCD1602_DB = 0xFF;
LCD1602_RS = 0;
LCD1602_RW = 1;
do {
LCD1602_E = 1;
sta = LCD1602_DB; //讀取狀態字
LCD1602_E = 0;
}while (sta & 0x80); //bit7 等于 1 表示液晶正忙,重復檢測直到其等于 0 為止
}
/* 向 LCD1602 液晶寫入一字節命令,cmd-待寫入命令值 */
void LcdWriteCmd(unsigned char cmd){
LcdWaitReady();
LCD1602_RS = 0;
LCD1602_RW = 0;
LCD1602_DB = cmd;
LCD1602_E = 1;
LCD1602_E = 0;
}
/* 向 LCD1602 液晶寫入一字節數據,dat-待寫入數據值 */
void LcdWriteDat(unsigned char dat){
LcdWaitReady();
LCD1602_RS = 1;
LCD1602_RW = 0;
LCD1602_DB = dat;
LCD1602_E = 1;
LCD1602_E = 0;
}
/* 設置顯示 RAM 起始地址,亦即光標位置,(x,y)-對應屏幕上的字符坐標 */
void LcdSetCursor(unsigned char x, unsigned char y){
unsigned char addr;
if (y == 0){ //由輸入的屏幕坐標計算顯示 RAM 的地址
addr = 0x00 + x; //第一行字符地址從 0x00 起始
}else{
addr = 0x40 + x; //第二行字符地址從 0x40 起始
}
LcdWriteCmd(addr | 0x80); //設置 RAM 地址
}
/* 在液晶上顯示字符串,(x,y)-對應屏幕上的起始坐標,str-字符串指針 */
void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str){
LcdSetCursor
評論
查看更多