【1】24C02功能概述

2K Bit的串行EEPROM存储器,内部含有256个字节。在24C02里面有一个8字节页写缓冲器。该设备的工作电压为1.8V到6.0V,芯片的第7引脚WP为写保护引脚,将该引脚接地允许正常的读写。

怎么区分24及25形BIOS芯片_起始信号

【2】设备地址

      24C02的设备地址包括固定部分和可编程部分。可编程部分需要根据硬件引脚A0、A1和A2来设置。设备地址的最后一位用于设置数据传输的方向,即读/写位。格式如下图:

怎么区分24及25形BIOS芯片_数据_02

读操作地址为:0xA1;而写操作地址则为:0xA0

【3】读写操作中的应答信号

      在写操作中,24C02每接收一个8位字节后会产生一个应答信号。在读操作中,24C02在发送一个8位数据后会释放SDA线并监视应答信号。一旦收到应答信号,将继续发送数据。如果主机没有发送应答信号,从机则停止发送数据且等待一个停止信号。

【4】字节写操作

      24C02接收完设备地址后,产生应答信号;然后接收8位内存字节地址,产生应答信号,接着接收一个8位数据,产生应答信号;最后主机发送停止信号,字节写操作结束。

【5】页写操作

      24C02有一个8字节的页写缓冲器,也就是说可以一次连续写入8个字节到缓冲器,然后由缓冲器一次性写到EEPROM。页写操作初始化与字节写操作相同,只是主机不会在写完第一个数据后就发送停止信号,而是在24C02应答后,接着发送7个数据。
      需要注意的是,24C02接收到每个数据后,其字节地址的低3位会自动加1,高位地址不变,维持在当前页内。当内部产生的字节地址到达该页边界时,随后的数据将写入该页的页首,先前的写入的数据将会被覆盖。

【6】当前地址读操作

      24C02内部的地址寄存器会保存着上次读/写操作最后一个地址加1的值。只要芯片有电,该地址就一直保存着。如果上次读/写操作的地址为N,那么当前地址读操作就从N+1开始。当读到最后一个字节(即255处),地址会回转到0。

【7】字节读操作

      主机首先发送起始信号,接着发送设备地址和它想要读取的数据内存字节地址,执行一个伪写操作。在24C02应答主机之后,主机重新发送起始信号和从设备地址,进行读操作。24C02响应并发送应答信号,然后输出所要求的一个8位字节数据。主机接收完这个8位数据后,产生一个“非应答”信号,最后发送停止条件,字节读操作结束。

【8】连续读操作

      在24C02发送完一个8位字节数据之后,主机产生一个应答信号来响应,告知24C02要求读取更多的数据,直到读完最后一个数据,主机向24C02发送一个“非应答”信号,然后发送一个停止信号,结束此操作。

【9】基本读写操作流程

字节写操作,参考代码:

void Write_24C02(unsigned char addr, unsigned char dat)
{
	IIC_Start();		//IIC总线起始信号					
	IIC_SendByte(0xa0); 	//24C02的写设备地址
	IIC_WaitAck();		//等待从机应答	
	IIC_SendByte(addr); 	//内存字节字节
	IIC_WaitAck(); 		//等待从机应答	
	IIC_SendByte(dat); 	//写入目标数据
	IIC_WaitAck();		//等待从机应答	
	IIC_Stop();		//IIC总线停止信号		
}

字节读操作,参考代码:

unsigned char Read_24C02(unsigned char addr)
{
	unsigned char tmp;
	//进行一个伪写操作
	IIC_Start();		//IIC总线起始信号					
	IIC_SendByte(0xa0); 	//24C02写设备地址
	IIC_WaitAck();		//等待从机应答	
	IIC_SendByte(addr); 	//内存自己地址
	IIC_WaitAck(); 		//等待从机应答	
	//进行字节读操作
	IIC_Start();		//IIC总线起始信号					
	IIC_SendByte(0xa1); 	//24C02读设备地址
	IIC_WaitAck();		//等待从机应答	
	tmp = IIC_RecByte();	//读取目标数据
	IIC_Ack(0); 		//产生非应答信号		
	IIC_Stop();		//IIC总线停止信号			
	return tmp;
}

【10】进阶强化实训

怎么区分24及25形BIOS芯片_数据_03

【11】参考源码共享

#include "regx52.h"
#include "absacc.h"
#include "iic.h"

unsigned char dat1 = 0, dat2 = 0, dat3 = 0;

unsigned char code SMG_duanma[18]=
		{0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,
		 0x88,0x80,0xc6,0xc0,0x86,0x8e,0xbf,0x7f};

void DelaySMG(unsigned int t)
{
	while(t--);
}

void DisplaySMG_Bit(unsigned char pos, unsigned char value)
{
	XBYTE[0xE000] = 0xFF;
	XBYTE[0xC000] = 0x01 << pos;
	XBYTE[0xE000] = value;
}

void Write_24C02(unsigned char addr, unsigned char dat)
{
	IIC_Start();						
	IIC_SendByte(0xa0); 		
	IIC_WaitAck();				
	IIC_SendByte(addr); 		
	IIC_WaitAck(); 					
	IIC_SendByte(dat); 			
	IIC_WaitAck();					
	IIC_Stop();							
}

unsigned char Read_24C02(unsigned char addr)
{
	unsigned char tmp;
	
	IIC_Start();					
	IIC_SendByte(0xa0); 		
	IIC_WaitAck();					
	IIC_SendByte(addr); 		
	IIC_WaitAck(); 					
	
	IIC_Start();						
	IIC_SendByte(0xa1); 		
	IIC_WaitAck();				
	tmp = IIC_RecByte();		
	IIC_Ack(0); 						
	IIC_Stop();							
	return tmp;
}

void Read_Write()
{
	dat1 = Read_24C02(0x01);
	dat2 = Read_24C02(0x03);
	dat3 = Read_24C02(0x05);
	
	dat1 = dat1 + 1;
	dat2 = dat2 + 2;
	dat3 = dat3 + 3;
	
	if(dat1 > 10)
		dat1 = 0;
	if(dat2 > 20)
		dat2 = 0;
	if(dat3 > 30)
		dat3 = 0;
	
	Write_24C02(0x01, dat1);
	DelaySMG(1000);
	Write_24C02(0x03, dat2);
	DelaySMG(1000);
	Write_24C02(0x05, dat3);
	DelaySMG(1000);
}

void DisplaySMG_24C02()
{
	DisplaySMG_Bit(0, SMG_duanma[dat1/10]);
	DelaySMG(500);
	DisplaySMG_Bit(1, SMG_duanma[dat1%10]);
	DelaySMG(500);
	DisplaySMG_Bit(2, SMG_duanma[16]);
	DelaySMG(500);
	
	DisplaySMG_Bit(3, SMG_duanma[dat2/10]);
	DelaySMG(500);
  DisplaySMG_Bit(4, SMG_duanma[dat2%10]);
	DelaySMG(500);
	DisplaySMG_Bit(5, SMG_duanma[16]);
	
	DelaySMG(500);
	DisplaySMG_Bit(6, SMG_duanma[dat3/10]);
	DelaySMG(500);
	DisplaySMG_Bit(7, SMG_duanma[dat3%10]);	
	DelaySMG(500);

}

main()
{
	XBYTE[0x8000] = 0xFF;
	XBYTE[0xA000] = 0x00;
	Read_Write();
	while(1)
	{
		DisplaySMG_24C02();
	}
}