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