STM32之AT24C02(EEPROM)掉电保存

简单说一下AT24C02芯片:AT24C02芯片为EEPROM,可以实现数据保存,且掉电不丢失。

AT24C02的总容量是 256个字节,该芯片通过 IIC 总线与外部连接

电路如下(原子的):

芯片的1、2、3、4脚接地;

5脚:作为数据线,也就是IIC通信的SDA;

6脚:作为时钟线,也就是IIC通信的SCL;

7脚:接GND(地)

8脚:接VCC(电源)

电源和地之间接个电容,以滤波等

                            

讲实现之前,简单介绍一下IIC:

IIC用于连接微控制器及其外围设备,由数据线 SDA 和时钟 SCL 构成的串行总线,可发送和接收数据。在 CPU 与被控 IC 之间、IC 与 IC 之间进行双向传送,高速 IIC 总线一般可达 400kbps 以上。I2C 总线在传送数据过程中共有三种类型信号, 它们分别是:开始信号、结束信号和应答信号。


开始信号:SCL 为高电平时,SDA 由高电平向低电平跳变,开始传送数据。
结束信号:SCL 为高电平时,SDA 由低电平向高电平跳变,结束传送数据。


应答信号:接收数据的 IC 在接收到 8bit 数据后,向发送数据的 IC 发出特定的低电平脉冲,表示已收到数据。CPU 向受控单元发出一个信号后,等待受控单元发出一个应答信号,CPU 接收到应答信号后,根据实际情况作出是否继续传递信号的判断。若未收到应答信号,由判断为受控单元出现故障。这些信号中,起始信号是必需的,结束信号和应答信号,都可以不要。

IIC 总线时序图

  

 代码实现:

稍微讲一下代码以及使用 :

AT24C02写数据:

//在AT24CXX指定地址写入一个数据
//WriteAddr  :写入数据的目的地址    
//DataToWrite:要写入的数据
void AT24CXX_WriteOneByte(u16 WriteAddr,u8 DataToWrite)
{				   	  	    																 
    IIC_Start();  
	if(EE_TYPE>AT24C16)
	{
		IIC_Send_Byte(0XA0);	    //发送写命令
		IIC_Wait_Ack();
		IIC_Send_Byte(WriteAddr>>8);//发送高地址	  
	}else IIC_Send_Byte(0XA0+((WriteAddr/256)<<1));   //发送器件地址0XA0,写数据 	 
	IIC_Wait_Ack();	   
    IIC_Send_Byte(WriteAddr%256);   //发送低地址
	IIC_Wait_Ack(); 	 										  		   
	IIC_Send_Byte(DataToWrite);     //发送字节							   
	IIC_Wait_Ack();  		    	   
    IIC_Stop();		//产生一个停止条件 
	delay_ms(10);	//对于EEPROM器件,每写一次要等待一段时间,否则写失败!	 
}

AT24C02 读数据

//在AT24CXX指定地址读出一个数据
//ReadAddr:开始读数的地址  
//返回值  :读到的数据
u8 AT24CXX_ReadOneByte(u16 ReadAddr)
{				  
	u8 temp=0;		  	    																 
    IIC_Start();  
	if(EE_TYPE>AT24C16)
	{
		IIC_Send_Byte(0XA0);	   //发送写命令
		IIC_Wait_Ack();
		IIC_Send_Byte(ReadAddr>>8);//发送高地址	    
	}else IIC_Send_Byte(0XA0+((ReadAddr/256)<<1));   //发送器件地址0XA0,写数据 	   
	IIC_Wait_Ack(); 
    IIC_Send_Byte(ReadAddr%256);   //发送低地址
	IIC_Wait_Ack();	    
	IIC_Start();  	 	   
	IIC_Send_Byte(0XA1);           //进入接收模式			   
	IIC_Wait_Ack();	 
    temp=IIC_Read_Byte(0);		   
    IIC_Stop();//产生一个停止条件	    
	return temp;
}

初始化代码之后,可以往AT24cXX(EEPROM),写入字符、字符串、数字等,

断电之后,数据还保存在AT24C02里面,只要使用读函数,就可以将数据读出来

示例:
const u8 TEXT_Buffer[]={6,9,12,56,88};  //要写入到24c02的数组

          //TEXT_Buffer[]={"dafeacaeceawa"}; 可以是字符串

#define SIZE sizeof(TEXT_Buffer)           //计算数组的大小                                                                                                                             

AT24CXX_Write(0,           (u8*)TEXT_Buffer,              SIZE);     //向AT24C02写入数据   

       // 0:开始写数据地址,这个地址的范围看自己用什么型号的EEPROM,这里是02,因此地址范围为0~255   

        //(u8*)TEXT_Buffer,  数组的首地址,也就是我们需要写入的数据的地址 

       //  SIZE:是写入的数据的大小,也就是我们需要存入多少数据,以字节为单位,这里计算的是数组的大小   

AT24CXX_Read(0,                datatemp,                       SIZE);                                                   

       // 0:开始读数据地址,这个地址的范围看自己用什么型号的EEPROM,这里是02,因此地址范围为0~255   

        //datatemp,  数组的首地址,读出的数据保存在数组里面 

       //  SIZE:是读出的数据的大小,也就是我们需要读多少数据,以字节为单位,这里计算的是数组的大小

开始写入的数据为TEXT_Buffer[]={6,9,12,56,88};  

因此读出时,datatemp []={6,9,12,56,88}; 先要什么数据直接对数组操作就行  

 

接下来这部分的代码主要是IIC的初始化以及一些驱动,就粘贴出来,不作详细的介绍了

//初始化IIC
void IIC_Init(void)
{					     
 	RCC->APB2ENR|=1<<4;//先使能外设IO PORTC时钟 							 
	GPIOC->CRH&=0XFFF00FFF;//PC11/12 推挽输出
	GPIOC->CRH|=0X00033000;	   
	GPIOC->ODR|=3<<11;     //PC11,12 输出高
}
//产生IIC起始信号
void IIC_Start(void)
{
	SDA_OUT();     //sda线输出
	IIC_SDA=1;	  	  
	IIC_SCL=1;
	delay_us(4);
 	IIC_SDA=0;//START:when CLK is high,DATA change form high to low 
	delay_us(4);
	IIC_SCL=0;//钳住I2C总线,准备发送或接收数据 
}	  
//产生IIC停止信号
void IIC_Stop(void)
{
	SDA_OUT();//sda线输出
	IIC_SCL=0;
	IIC_SDA=0;
 	delay_us(4); 
	IIC_SCL=1;//STOP:when CLK is high DATA change form low to high
 	delay_us(4); 
	IIC_SDA=1;//发送I2C总线结束信号 						   	
}
//等待应答信号到来
//返回值:1,接收应答失败
//        0,接收应答成功
u8 IIC_Wait_Ack(void)
{
	u8 ucErrTime=0;
	SDA_IN();      //SDA设置为输入  
	IIC_SDA=1;delay_us(1);	   
	IIC_SCL=1;delay_us(1);	 
	while(READ_SDA)
	{
		ucErrTime++;
		if(ucErrTime>250)
		{
			IIC_Stop();
			return 1;
		}
	}
	IIC_SCL=0;//时钟输出0 	   
	return 0;  
} 
//产生ACK应答
void IIC_Ack(void)
{
	IIC_SCL=0;
	SDA_OUT();
	IIC_SDA=0;
	delay_us(2);
	IIC_SCL=1;
	delay_us(2);
	IIC_SCL=0;
}
//不产生ACK应答		    
void IIC_NAck(void)
{
	IIC_SCL=0;
	SDA_OUT();
	IIC_SDA=1;
	delay_us(2);
	IIC_SCL=1;
	delay_us(2);
	IIC_SCL=0;
}					 				     
//IIC发送一个字节
//返回从机有无应答
//1,有应答
//0,无应答			  
void IIC_Send_Byte(u8 txd)
{                        
    u8 t;   
	SDA_OUT(); 	    
    IIC_SCL=0;//拉低时钟开始数据传输
    for(t=0;t<8;t++)
    {              
        IIC_SDA=(txd&0x80)>>7;
        txd<<=1; 	  
		delay_us(2);   //对TEA5767这三个延时都是必须的
		IIC_SCL=1;
		delay_us(2); 
		IIC_SCL=0;	
		delay_us(2);
    }	 
} 	    
//读1个字节,ack=1时,发送ACK,ack=0,发送nACK   
u8 IIC_Read_Byte(unsigned char ack)
{
	unsigned char i,receive=0;
	SDA_IN();//SDA设置为输入
    for(i=0;i<8;i++ )
	{
        IIC_SCL=0; 
        delay_us(2);
		IIC_SCL=1;
        receive<<=1;
        if(READ_SDA)receive++;   
		delay_us(1); 
    }					 
    if (!ack)
        IIC_NAck();//发送nACK
    else
        IIC_Ack(); //发送ACK   
    return receive;
}

总结:

1、STM32 与 AT24C02 通信方式为IIC通信;

2、AT24C02 为EEPROM芯片,可以实现数据的掉电保存

3、往AT24C02 写入和读出数据只需要对AT24C02的地址写入和读出就行

4、读出的数据,可以保存在数组里面,操作只需要对数组进行操作

 

 

 

 

 

 

 

  • 11
    点赞
  • 108
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
下面是一个使用STM32控制AT24C02 EEPROM的示例代码。该代码使用I2C总线进行通信,通过读写AT24C02的寄存器来实现数据存储和读取。 ```c #include "stm32f4xx.h" #define EEPROM_ADDRESS 0xA0 #define I2C_TIMEOUT 10000 void I2C1_Init(void); void I2C1_Start(void); void I2C1_Stop(void); void I2C1_Write(uint8_t data); uint8_t I2C1_Read(uint8_t ack); int main(void) { uint8_t data[] = "Hello, AT24C02!"; uint8_t readData[16]; // 初始化I2C总线 I2C1_Init(); // 启动I2C总线并发送写命令和EEPROM地址 I2C1_Start(); I2C1_Write(EEPROM_ADDRESS); I2C1_Write(0x00); // 写入数据的起始地址 // 写入数据 for (int i = 0; i < sizeof(data); i++) { I2C1_Write(data[i]); } // 停止I2C总线 I2C1_Stop(); // 延时等待写入完成 for (int i = 0; i < 1000000; i++) { __NOP(); } // 启动I2C总线并发送读命令和EEPROM地址 I2C1_Start(); I2C1_Write(EEPROM_ADDRESS); I2C1_Write(0x00); // 读取数据的起始地址 // 重启I2C总线并发送读命令 I2C1_Start(); I2C1_Write(EEPROM_ADDRESS | 0x01); // I2C总线地址后面加上读位 // 读取数据 for (int i = 0; i < sizeof(readData); i++) { readData[i] = I2C1_Read(i == sizeof(readData) - 1 ? 0 : 1); } // 停止I2C总线 I2C1_Stop(); // 输出读取的数据 for (int i = 0; i < sizeof(readData); i++) { printf("%c", readData[i]); } while (1) { } } void I2C1_Init(void) { RCC->APB1ENR |= RCC_APB1ENR_I2C1EN; RCC->AHB1ENR |= RCC_AHB1ENR_GPIOBEN; // 配置GPIOB引脚为I2C模式 GPIOB->MODER |= GPIO_MODER_MODER6_1 | GPIO_MODER_MODER7_1; GPIOB->OTYPER |= GPIO_OTYPER_OT_6 | GPIO_OTYPER_OT_7; GPIOB->OSPEEDR |= GPIO_OSPEEDER_OSPEEDR6 | GPIO_OSPEEDER_OSPEEDR7; GPIOB->PUPDR |= GPIO_PUPDR_PUPDR6_0 | GPIO_PUPDR_PUPDR7_0; GPIOB->AFR[0] |= (4 << (6 * 4)) | (4 << (7 * 4)); // 配置I2C1 I2C1->CR1 = 0; I2C1->CR2 = 42; // APB1 clock = 42 MHz I2C1->CCR = 210; // 100 kHz SCL frequency I2C1->TRISE = 43; I2C1->CR1 |= I2C_CR1_PE; } void I2C1_Start(void) { I2C1->CR1 |= I2C_CR1_START; while (!(I2C1->SR1 & I2C_SR1_SB)) ; (void)I2C1->SR1; } void I2C1_Stop(void) { I2C1->CR1 |= I2C_CR1_STOP; while (I2C1->CR1 & I2C_CR1_STOP) ; } void I2C1_Write(uint8_t data) { I2C1->DR = data; while (!(I2C1->SR1 & I2C_SR1_TXE)) ; } uint8_t I2C1_Read(uint8_t ack) { if (ack) { I2C1->CR1 |= I2C_CR1_ACK; } else { I2C1->CR1 &= ~I2C_CR1_ACK; } while (!(I2C1->SR1 & I2C_SR1_RXNE)) ; return I2C1->DR; } ``` 这个示例代码使用了STM32的I2C模块(I2C1)来与AT24C02 EEPROM进行通信。首先,代码通过I2C总线向EEPROM写入数据,然后再从EEPROM中读取数据。请注意,代码中的I2C时钟频率和EEPROM地址需要根据实际情况进行调整。 这只是一个简单的示例代码,实际应用中可能需要根据具体情况进行修改和优化。另外,请确保正确连接STM32和AT24C02 EEPROM,并正确配置I2C总线的引脚和时钟频率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值