一个通用的24C01-24C256共9种EEPROM的字节读写操作程序,此程序有五个入口条件,分别为读写数据缓冲区指针,进行读写的字节数,EEPROM首址,EEPROM控制字节,以及EEPROM类型。此程序结构性良好,具有极好的容错性,程序机器码也不多:DataBuff为读写数据输入/输出缓冲区的首址Length 为要读写数据的字节数量Addr 为EEPROM的片内地址 AT24256为0~32767Control 为EEPROM的控制字节,具体形式为(1)(0)(1)(0)(A2)(A1)(A0)(R/W),其中R/W=1,表示读操作,R/W=0为写操作,A2,A1,A0为EEPROM的页选或片选地址;enumer为枚举变量,需为AT2401至AT24256中的一种,分别对应AT24C01至AT24C256;函数返回值为一个位变量,若返回1表示此次操作失效,0表示操作成功;ERROR为允许最大次数,若出现ERRORCOUNT次操作失效后,则函数中止操作,并返回1。SDA和SCL由用户自定义,对于1K位,2K位,4K位,8K位,16K位芯片采用一个8位长的字节地址码,对于32K位以上的采用2个8位长的字节地址码直接寻址,而4K位,8K位,16K位配合页面地址来寻址。
#include <intrins.h> #define ERROR 10 //允许ERROR的最大次数 sbit SDA=P4^2; sbit SCL=P4^4; enum eepromtype {AT2401,AT2402,AT2404,AT2408,AT2416,AT2432,AT2464,AT24128,AT24256};/*器件的型号*/ enum eepromtype enumer; //定义一个枚举变量 unsigned char buf1[]="aaaaaaaaaa"; /* 发送缓冲区 */ unsigned char buf2 [12]; /* 接收缓冲区 */ void senddd(unsigned d1); /* * * * * * * * 一个简单延时程序 * * * * * * * * * * * * */ void Delay(unsigned char DelayCount) { while(DelayCount--); } /* * * * * 以下是对IIC总线的操作子程序 * * * * */ /* * * * * * 启动总线 * * * * */ void Start(void) { SCL=0; /* SCL处于高电平时,SDA从高电平转向低电平表示 */ SDA=1; /* 一个"开始"状态,该状态必须在其他命令之前执行 */ Delay(1); SCL=1; Delay(1); SDA=0; Delay(1); SCL=0; SDA=1; Delay(1); } /* * * * * 停止IIC总线 * * * * */ void Stop(void) { SCL=0; /*SCL处于高电平时,SDA从低电平转向高电平 */ SDA=0; /*表示一个"停止"状态,该状态终止所有通讯 */ Delay(1); SCL=1; Delay(1); /* 空操作 */ SDA=1; Delay(1); SCL=0; Delay(1000); } /* * * * * 检查应答位 * * * * */ bit RecAck(void) { SCL=0; SDA=1; Delay(20); SCL=1; Delay(1); Delay(1); CY=SDA; /* 因为返回值总是放在CY中的 */ SCL=0; Delay(20); return(CY); } /* * * * *对IIC总线产生应答 * * * * */ void Ack(void) { SDA=0; /* EEPROM通过在收到每个地址或数据之后, */ SCL=1; /* 置SDA低电平的方式确认表示收到读SDA口状态 */ Delay(1); SCL=0; Delay(1); SDA=1; } /* * * * * * * * * 不对IIC总线产生应答 * * * * */ void NoAck(void) { SDA=1; SCL=1; Delay(1); SCL=0; } /* * * * * * * * * 向IIC总线写数据 * * * * */ void Send(unsigned char sendbyte) { unsigned char data j=8; for(;j>0;j--) { SCL=0; sendbyte <<= 1; /* 使CY=sendbyte^7; */ SDA=CY; /* CY 进位标志位 */ Delay(1); SCL=1; Delay(20); } SCL=0; Delay(1); } /* * * * * * * * * 从IIC总线上读数据子程序 * * * * */ unsigned char Receive(void) { register receivebyte,i=8; SCL=0; while(i--) { SCL=1; receivebyte = (receivebyte <<1 ) | SDA; Delay(1); SCL=0; Delay(1); } return(receivebyte); } /* ----- AT24C01~AT24C256 的读写程序 ------ */ bit RW24xx(unsigned char *DataBuff,unsigned char Length,unsigned int Addr, unsigned char Control,enum eepromtype enumer) { unsigned char data j,i=ERROR; bit errorflag=1; /* 出错标志 */ while(i--) { Start(); /* 启动总线 */ Send(Control & 0xfe); /* 向IIC总线写数据,器件地址 */ if(RecAck()) continue; /* 如写不正确结束本次循环 */ if(enumer > AT2416) { Send((unsigned char)(Addr >> 8));//把整型数据转换为字符型数据:弃高取低,只取低8位.如果容量大于32K位,使用16位地址寻址,写入高八位地址 if(RecAck()) continue; } Send((unsigned char)Addr); /* 向IIC总线写数据 */ if(RecAck()) continue; /* 如写正确结束本次循环 */ if(!(Control & 0x01)) //判断是读器件还是写器件 { j=Length; errorflag=0; /* 清错误特征位 */ while(j--) { Send(*DataBuff++); /* 向IIC总线写数据 */ if(!RecAck()) continue; /* 如写正确结束本次循环 */ errorflag=1; break; } if(errorflag==1) continue; break; } else { Start(); /* 启动总线 */ Send(Control); /* 向IIC总线写数据 */ if(RecAck()) continue;//器件没应答结束本次本层循环 while(--Length) /* 字节长为0结束 */ { *DataBuff ++= Receive(); Ack(); /* 对IIC总线产生应答 */ } *DataBuff=Receive(); /* 读最后一个字节 */ NoAck(); /* 不对IIC总线产生应答 */ errorflag=0; break; } } Stop(); /* 停止IIC总线 */ if(!(Control & 0x01)) { Delay(255); Delay(255); Delay(255); } return(errorflag); }
下面是主函数:
#include <REG52.H> #include "24c256_1.h" #include<string.h> unsigned char temp[64]; unsigned char rxnum; unsigned char flag2; void senddd(unsigned d1); void uart(void) interrupt 4// 串口中断 { unsigned char k = 0; if(RI) //接收 { } else //发送 { TI=0; } } //查询方式发送单个字符 void senddd(unsigned d1) { ES=0; //TE=1; TI=0;SBUF=d1;while(!TI); //TE=0; TI=0; ES=1; } //Send String Tor Uart void printu(char * str) { char *ct = str; while (*ct != '\0') { if (*ct == '\n') { SBUF = 13; while (!TI); TI = 0; } SBUF=*ct; while (!TI); TI = 0; ct++; } } //UART初始化 void initUart(void) { TMOD &=0x0f; TMOD |=0x20; SCON=0x50; IP = 0x10; //PCON|=0x80; TH1=0xf7; TL1=0xf7; //115200,normal //TH1=0xfa; TL1=0xfa; //倍速 TR1=1; ET1=0; RI=0; //AUXR = 0xA0; //TE=0; //允许接收 仅485时用。 } void main(void) { unsigned char Control,*p1,*p2; unsigned char Length; unsigned int addr ,i; /* 24Cxx片内地址 */ P4SW|= 0x10; initUart(); ES =1; ET1=0; EA =1; senddd('O'); senddd('N'); printu(" starting...\r\n"); p1=buf1;p2=buf2; printu(p1); printu("\r\n"); addr=0; ////片内地址 AT24C256为0~32767///// Length=10; ////// 读写长度 ////// enumer=AT24256; /// 读写AT24C256//// Control=0xa0; /// 写操作/// RW24xx(p1,Length,addr,Control,enumer); // 写 // Control=0xa1; ///读操作/// for(i=0;i<10000;i++); //要加入延时···才可正确的读取数据! // Delay(1000);//1:6.18us; 2:8.36us 3:10.55us △=2.18us RW24xx(p2,Length,addr,Control,enumer);// 读 printu(p2); printu("\r\n"); }
下面是运行结果:
相关推荐
基于stm32f103,IO模拟IIC总线实现对AT24C256读写操作,对不同系列的AT24CXX适用,只需稍微修改既可以使用。
NULL 博文链接:https://liuzongming.iteye.com/blog/1886903
1、基于IIC协议,采用verilog编写AT24C16驱动程序。 2、实现功能:在AT24C16的地址0~99之间顺序写入数据0~99,然后在读取出来,读取的数据通过串口调试助手显示出来。 3、测试平台Quartus 17.1
一个通用的24C01-24C256共9种EEPROM的字节读写操作程序,此程序有五个入口条件,分别为读写数据缓冲区指针, 进行读写的字节数,EEPROM首址,EEPROM控制字节, 以及EEPROM类型。此程序结构性良好,具有极好的容错性...
实现EEPROM芯片的读取和写入数据 模拟IIC实现 MSP430F149单片机 可以移植到STM32 STC51等其他系列单片机
模拟IIC读写at24c512,用at24c02程序改写,其中包括奇数位读写乱码、读写0xff问题的解决办法。欢迎登门。
IIC读写AT24C02 目的:记录开机的次数,并用LCD1602显示记录次数 要求:AT89C52 + AT24C02 + LCD1602 (1) 画出51单片机操作IIC读写AT24C02的PROTEUS原理图 (2) 编写代码实现IIC读写的操作 (3) IIC存贮芯片AT...
AT24C04读写程序。用AT2051单片机实现。晶振为12MHz
IIC 24C02读写数据,并在1602上显示读出来的数据
LPC17XX IIC AT24C16读写测试程序,非常适合初学者使用
主MCU是STM32F103C8T6芯片,做IIC总线主控。 EEPROM是AT24C256,做IIC总线从控。
stm32f103读写AT24C256的C语言程序
该文档提供了用stm32单片机进行AT24C02芯片的数据的读写代码。
清晰明确的操作思路 详细的注释 通过IIC总线读写AT24C02B(EEPROM),同时用LED反映读出数据状态
本实验通过KEY1按键来控制24C02的写入,通过另外一个按键KEY0来控制24C02的读取。并在LCD模块上面显示 相关信息。同时,我们可以通过USMART控制在24C02的任意地址写入和读取数据。
最新单片机仿真 对I2C总线上挂接多个AT24C02的读写操作最新单片机仿真 对I2C总线上挂接多个AT24C02的读写操作最新单片机仿真 对I2C总线上挂接多个AT24C02的读写操作最新单片机仿真 对I2C总线上挂接多个AT24C02的读写...
一个简单的IIC操作程序 用于AT24C08芯片的读写操作,不同的片子,具体请修改片子地址以及页大小
使用CubeMx搭建的硬件IIC读写AT24C02工程模版
STC12C5A IIC接口读写 AT24C02 89C5A IIC 读写AT24C02 51的的protuse仿真
官网给的模板直接套上就行,史上最权威的IIC读写程序,走过路过不要错过