zoukankan      html  css  js  c++  java
  • AT24C02/04/08/16 操作说明

    我们这里介绍一下常见的EEPROM,ATMEL的AT24x系列中的AT24C02,学会了这个芯片,其他系列的芯片也是类似的。

    AT24C02的存储容量为2K bit,内容分成32页,每页8Byte (共256Byte)。

    基础概念:

    • 存储容量: 存储的总数据量

    • 页:储存单元的组织方式

    • 寻址方式:访问元器件的方式,通常计算地址与这个概念有关。

    寻址方式

    操作时有两种寻址方式:芯片寻址和片内子地址寻址。

    芯片寻址

    AT24C02的芯片地址为1010,其地址控制字格式为1 0 1 0 A2 A1 A0 R/W

    其中A2,A1,A0代表"可编程地址选择位"。A2,A1,A0引脚接高/低电平后得到确定的三位编码,与1010形成7位编码,即为该器件的地址码。

    R/W为芯片读写控制位;该位为0,表示芯片进行写操作。取决于芯片手册中的定义,通常,W是低电平有效(0)。

    有些大容量的芯片,地址组成位包括了P0P1等选择位;而这些选择位实际上是"页选择位",用来表示内部地址(如下)。但是实际上原理是类似的。

    片内子地址寻址

    芯片寻址可对内部 256 Bytes中的任一个进行读/写操作,其寻址范围为00~FF,共256个寻址单位。

    该类器件要通过I2C总线操作,读写过程中都要先寻址,这类器件地址由两个字节组成。

    如下表,1010是固定的,A表示器件地址,可以拉高和拉低,I2C总线上可以并接2的n次方个器件。P表示具体的内部地址数,比如at24c02共有256个字节,第二个地址字节完全可以满足,不用P。但是at24c04一个有512个字节,需要9位地址线,第一个字节中的p就表示地址线了,p=0表示低256字节,1表示高256字节。

    注意:一般页写可以连续写8个数据。主机每发送一个字节都要接受从机的应答信号。

    该类器件是采用I2C总线进行操作的,器件地址根据容量的不同稍有不同,如下:

    对于芯片的A0,A1,A2脚:(外部引脚用于芯片寻址,当接入多个IIC设备时,通过引脚电平不同,可以对多个设备进行读写操作)

    • 24C01/02:A0,A1,A2都是从设备地址。
    • 24C04,A1,A2是从设备地址,A0变为P0
    • 24C08,A2是从设备地址,A0,A1变为P0、P1
    • 24C16及以上,只有P0、P1、P2

    其中,A表示器件地址,P表示页地址

    在读写的时候首先是:起始条件+器件地址

    器件地址如上所示在16k中,需要两个字节表示内部地址,正是P 2/1/0 和8位具体的地址

    AT24C02/04/08/16 操作说明

    另外有关于容量的说明:

    • AT24C01:一共128页,每页1字节,共需7位地址数据

    • AT24C02:一共256页,每页1字节,共需8位地址数据

    • AT24C04:一共256页,每页2字节,共需9位地址数据

    • AT24C08:分4个块,一共256页,每页4字节,共需10位地址数据

    • AT24C16:分8块,一共256页,每页8字节,共需11位地址数据

    在读写的时候要注意,一般在写的时候有页写,在读的时候是连续读,可以设置读取的个数,根据应答和非应答来结束读取。

    具体的控制时序省略(关键在于:理解好地址和储存单元的计算关系,结合I2C的时序),只是需要注意:

    在读取EEPROM的时候接收到多个数据时,如果不是最后一个数据,单片机要主动发出应答信号,但是接收到最后一个数据后,就不再主动应答,而是等待应答,然后结束条件。

    大B与小b之间的换算:

    我们常说的MB、KB中的B都是Bytes;但有一些储存芯片,它们是以Bit作为计量单位的;而 有 8Bit = 1 Byte

    因此,小b除以8即可得到大B方式的表示方法。

    读写操作时序

    写数据

    第一步:首先是I2C的起始信号,接着跟上首字节,即EEPROM的地址和读写位的组合,读写方向上选择“写”操作。

    第二步:发送要写入数据的EEPROM内部存储地址。

    第三步:发送要存储的数据第一个字节、第二个字节… …。

    1、写数据(单片机发送)过程中,每个字节结束后EEPROM都会回应一个“应答位0”,告诉我们写EEPROM成功,如果没有应答表示未成功。

    2、写数据过程中,每成功写入一个字节,EEPROM地址自动加1,当加到最大值,会溢出。

    //函数见I2C部分
    //向E2PROM中写入一个字节,addr为字节地址
    void E2WriteByte(unsigned char addr,unsigned char dat)
    {
    	I2CStart();
    	I2CWrite(0x50<<1);//寻址后选择为写操作
    	I2CWrite(addr);	  //写入存储地址
    	I2CWrite(dat);
    	I2CStop();
    }
    

    读数据

    第一步:首先是I2C的起始信号,接着跟上首字节,即EEPROM的地址和读写位的组合,读写方向上选择“写”操作。

    第二步:发送要读取的EEPROM内部存储地址。

    第三步:重新发送I2C的起始信号和器件地址,并且在方向位选择“读”操作。
    (在这三步中,每一个字节实际上都是在“写”,因此EEPROM都会回应一个“应答位0”。)

    第四步:读取从器件发回的数据,每读一个字节,如果还想继续读下一个字节,就发送一个“应答位0”,如果不想继续读了,就发送一个“非应答位1”。

    应答位: ACK

    非应答位:NACK

    //读取E2PROM中一个字节,addr为字节地址
    unsigned cahr E2ReadByte(unsigned char addr)
    {
    	unsigned char dat;
    	I2CStart();
    	I2CWrite(0x50<<1);
    	I2CWrite(addr);
    	I2CStart();
    	I2CWrite((0x50<<1)|0x01);
    	dat=I2CReadNAK();
    	I2CStop();
    	return dat;
    }
    

    多字节进行读写操作

    //读取函数,buf为数据指针,addr为E2中的起始地址,len为读取长度
    void E2Read(unsigned char *buf,unsigned char addr,unsigned char len)
    {
    		do{
    			I2CStart();
    			if(I2CWrite(0x50<<1))
    			{
    				break;
    			}
    			I2CStop();
    		}while(1);//查询当前是否可以进行读写操作
    		I2CWrite(addr);
    		I2CStart();
    		I2CWrite((0x50<<1)|0x01);
    		while(len>1)//连续读取len-1个字节
    		{
    		   *buf++=I2CReadACK();
    		   len--;
    		}
    		*buf=I2CReadNAK();
    		I2CStop();
    }
    //写入函数,buf为源数据指针,addr为起始地址,len为写入长度
    void E2Write(unsigned char *buf,unsigned char addr,unsigned char len)
    {
    	while(len--)
    	{
    		do{
    			I2CStart();
    			if(I2CWrite(0x50<<1))
    				break;
    			I2CStop();
    		} while(1);
    		I2CWrite(addr++); 
    		I2CWrite(*buf++);
    		I2CStop();
    	}
    }
    

    页写入功能

    一次性写入一页,写完再发送停止位就不用写完一个字节就检测一次

    连续向页内写入几个字节后,最后发送停止位
    E2检测到停止位,将该页写入
    如果数据跨页,则发送一个停止位等待E2空闲后,即上一页完全写入非易失区域后,进行下一页的写入

    void E2Write(unsigned char *buf,unsigned char addr,unsigned char len)
    {
    	while(len>0)
    	{
    		do{
    			I2CStart();
    			if(I2CWrite(0x50<<1))
    				break;
    			I2CStop();
    		} while(1);
    		I2CWrite(addr); 
    		while(len>0)
    		{
    			I2CWrite(*buf++);
    			len--;
    			addr++;
    			if((addr&0x07)==0)//检查地址是否到达页边界,每页8字节,固检测第三位即可
    				break;
    		}
    		I2CStop();
    	}	
    }
    

    效率对比

    多字节写入和页写入程序都编写出来了,而且页写入的程序我们还特地跨页写的数据,它们的写入时间到底差别多大呢。

    我们用一些工具可以测量一下,比如示波器,逻辑分析仪等工具。

    我现在把两次写入时间用逻辑分析仪给抓了出来,并且用时间标签 T1 和 T2 标注了开始位置和结束位置,如图所示,右侧显示的|T1-T2|就是最终写入 5 个字节所耗费的时间。

    多字节一个一个写入,每次写入后都需要再次通信检测 EEPROM 是否在“忙”,因此耗费了大量的时间。

    同样的写入 5 个字节的数据:

    • 一个一个写入用了 8.4ms 左右的时间
    • 使用页写入,只用了 3.5ms 左右的时间

    多字节写入时间

    跨页写入时间

    附录:关于IIC-EEPROM页写问题

    结论:无论从什么地址开始连续写,不能超过(跨过)一整页。

    背景:

    在向 EEPROM 连续写入多个字节的数据时,如果每写一个字节都要等待的话,整体上的写入效率就太低了。因此 EEPROM 的厂商就想了一个办法,把 EEPROM 分页管理。24C01、24C02 这两个型号是 8 个字节一个页,而 24C04、24C08、24C16 是 16 个字节一页。例如AT24C02,一共是 256 个字节,8 个字节一页,那么就一共有 32 页。

    分配好页之后,如果我们在同一个页内连续写入几个字节后,最后再发送停止位的时序。EEPROM 检测到这个停止位后,就会一次性把这一页的数据写到非易失区域,就不需要像上节课那样写一个字节检测一次了,并且页写入的时间也不会超过 5ms。如果我们写入的数据跨页了,那么写完了一页之后,我们要发送一个停止位,然后等待并且检测 EEPROM 的空闲模式,一直等到把上一页数据完全写到非易失区域后,再进行下一页的写入,这样就可以在很大程度上提高数据的写入效率。

    本质:页写的原理是通过内部写缓冲RAM实现的(读操作不需要,可以连续读)。

    详解:

    AT24CXX系列的EEPROM为了提高写效率,提供了页写功能。

    内部有个一页大小的写缓冲RAM(地址范围也就是从00到页大小)。

    发生写操作时,开始送入的地址对应的页被选中,并将其内容映像到缓冲RAM,数据从低端地址对应的缓冲RAM地址开始修改,超过这个地址范围就回到00。

    写完后,就会把开始确定的EEPROM页擦除,再把一整页RAM数据写入。所有写数据都发生在开始写地址时确定的页上。

    如,页容量为128,一页都是从00开始按128字节分成一个个的页;

    此时,0页就是07F,1页就是80FF,类推,边界就是128字节的整数倍地址。

    页RAM的地址范围为7位00~7F,写入时高端地址就是页号。

    发生写操作,开始送入的地址对应的页被锁存,后续不论写多少,都在这个页中,只是一个页内的地址进行加一,超过就归零开始。从F0开始写32个字节,那么开始送入的地址为F0,就会锁定在1号页(第2个页)上,底端7位页内部地址开始从70H开始写,到达7F时回到00再到10H,也就是写在了F0FF,808F。

    也就是,从01开始写也只能到7F,再往80写就跑到00上去了,这就是写操作的翻卷,datasheet上都有说明。

    就是从边界前写两个字节也要分两次写。页是绝对的,按整页大小排列,不是从开始写入的地址开始算。

    读没有页的问题,可以从任意地址开始读取任意大小数据,只是超过整个存储器容量时地址才回卷。但一次性访问的数据长度也不要太大。

    所以,使用分页的存储器进行储存时,要做好存储管理,尽量同时读写的数据放在一个页上。

    ref :

  • 相关阅读:
    单例设计模式
    MySQL数据类型
    MySQL创建、修改、删除数据库
    HTTP请求与响应
    netcore在CentOS7 下使用处理图片的问题
    SQL删除重复数据
    不同浏览器对cookie大小与个数的限制
    asp.net实现SQL2005的通知数据缓存
    MS SQL 设置自增长字段默认值
    MS SQL 批量操作
  • 原文地址:https://www.cnblogs.com/schips/p/at24cxx.html
Copyright © 2011-2022 走看看