当前位置: 代码迷 >> 综合 >> STM32之 AT24C16(EEPROM)驱动代码(程序稳定,清晰明了)
  详细解决方案

STM32之 AT24C16(EEPROM)驱动代码(程序稳定,清晰明了)

热度:69   发布时间:2023-10-18 03:33:06.0
STM32之 AT24C16(EEPROM)驱动代码(程序稳定,清晰明了)
AT24C16电路图

 

第一部分:IIC协议代码头文件(iic.h)

#ifndef IIC_H
#define IIC_H
#include "stm32f10x.h"
#include "sys.h"
#include "delay.h"#define write 0
#define read  1//IIC总线地址接口定义
#define IIC_SCL PBout(7)
#define IIC_SDA_OUT PBout(6)
#define IIC_SDA_IN PBin(6)#define IIC_INPUT_MODE_SET()  {GPIOB->CRL&=0xF0FFFFFF;GPIOB->CRL|=0x08000000;}
#define IIC_OUTPUT_MODE_SET() {GPIOB->CRL&=0xF0FFFFFF;GPIOB->CRL|=0x03000000;}//函数声明
void IIC_Init(void);
void IIC_START(void);
void IIC_STOP(void);
u8 IIC_GetACK(void);
void IIC_SendAck(u8 ack);
void IIC_WriteOneByte(u8 data);
u8 IIC_ReadOneByte(void);
#endif

第二部分:IIC协议代码(iic.c)

#include "iic.h"
#include "stdio.h"
/*
函数功能: IIC总线初始化
硬件连接:SCL-PB7SDA-PB6
*/
void IIC_Init(void)
{/*1. 开时钟*/RCC->APB2ENR|=1<<3; //PBRCC->APB2ENR|=1<<0; //开启AFIO时钟/*2. 配置模式*/
//	AFIO->MAPR |= 0x2<<24;//关闭JTAG-DP,启用SW-DPGPIOB->CRL&=0x00FFFFFF;GPIOB->CRL|=0x33000000;/*3. 上拉*/GPIOB->ODR|=0x1<<6;printf("IIC_Init OK!\n");
}/*
函数功能: 起始信号
*/
void IIC_START(void)
{IIC_OUTPUT_MODE_SET(); //配置输出模式IIC_SDA_OUT = 1;       //拉高数据线IIC_SCL = 1;           //拉高时钟线DelayUs(2);            //延时IIC_SDA_OUT = 0;       //产生下降沿DelayUs(2);            //延时IIC_SCL = 0;           //拉低时钟线//告诉从机,通信开始(主机将要给从机发送数据)。
}/*
函数功能: 停止信号
*/
void IIC_STOP(void)
{IIC_OUTPUT_MODE_SET(); 	//配置输出模式IIC_SDA_OUT = 0;        //拉低数据线IIC_SCL = 1;            //拉高时钟线DelayUs(2);             //延时IIC_SDA_OUT = 1;        //产生上升沿DelayUs(2);             //延时
}/*
函数功能:  获取从机发给主机的应答
返回值  :  0表示获取成功,1表示获取失败
目的: 读取总线上一位数据的值。这一位数据的正确值0
*/
u8 IIC_GetACK(void)
{u8 cnt=0;IIC_INPUT_MODE_SET(); //输入模式IIC_SDA_OUT=1; //上拉IIC_SCL=0; //告诉从机主机需要数据DelayUs(2);IIC_SCL=1; //告诉从机主机正在读数据while(IIC_SDA_IN) //等待应答{cnt++;DelayUs(1);if(cnt>=250)//等待时间过长,产生停止信号,返回1,表示接收应答失败{IIC_STOP();//printf("N0 ACK\n");return 1;}}IIC_SCL=0; //告诉从机,主机准备发送数据return 0;
}/*
函数功能: 主机给从机发送应答
函数参数: 1(非应答) 0(应答)
目的: 发送一位数据
*/
void IIC_SendAck(u8 ack)
{IIC_OUTPUT_MODE_SET(); 	//配置输出模式IIC_SCL = 0;              //拉低时钟线,告诉从机,主机将要发送数据if(ack) IIC_SDA_OUT = 1; else IIC_SDA_OUT = 0;     //写应答信号DelayUs(2);               //延时IIC_SCL = 1;              //拉高时钟线DelayUs(2);               //延时IIC_SCL = 0;              //拉低时钟线
}/*
函数功能:  发送一个字节数据
函数参数:  data将要发送数据
*/
void IIC_WriteOneByte(u8 data)
{u8 i;IIC_OUTPUT_MODE_SET();IIC_SCL = 0;	//拉低时钟线, 数据准备发送DelayUs(2);		//延时for(i=0;i<8;i++)//从高位开始一位一位地传送{if(data&0x80)IIC_SDA_OUT=1;//送数据口else IIC_SDA_OUT=0;data<<=1;//移出数据的最高位IIC_SCL=1;//拉高时钟线,发送数据DelayUs(2);IIC_SCL=0;//拉低时钟线,数据发送完毕DelayUs(2);}
}/*
函数功能:  读取一个字节数据
返回值  :  读取成功的数据
*/
u8 IIC_ReadOneByte(void)
{u8 data=0,i=0;IIC_INPUT_MODE_SET();//使能内部上拉,准备读取数据for(i=0;i<8;i++){IIC_SCL=0;//置时钟线为低,准备接收数据位DelayUs(2);//时钟低电平周期大于4.7μsIIC_SCL=1;//置时钟线为高使数据线上数据有效,主机开始读数据,从机不能再改变数据了,即改变SDA的电平data<<=1;if(IIC_SDA_IN)data|=0x01;//读数据 DelayUs(2);}IIC_SCL=0;return data;
}

第三部分:AT24C16代码头文件(AT24C16.h)

#ifndef AT24C16_H
#define AT24C16_H
#include <stdio.h>
#include "stm32f10x.h"
#include "iic.h"#define AT24C16_WRITE_ADDR 0xA0
#define AT24C16_READ_ADDR  0xA1void AT24C16_WriteOneByte(u8 data,u8 addr);
void AT24C16_ReadData(u16 addr,u16 len,char *p);
void AT24C16_WriteData(u16 addr,u16 len,u8 *p);
void AT24C16_PageWrite(u16 addr,u16 len,u8 *p);#endif

第四部分:AT24C16代码(AT24C16.c)

#include "at24c16.h"/*
函数功能:  AT24C16写一个字节
函数参数:data:写入的字节数据addr:存放位置(0~2048)
*/void AT24C16_WriteOneByte(u8 data,u8 addr)
{IIC_START();IIC_WriteOneByte(AT24C16_WRITE_ADDR); //发送设备地址IIC_GetACK(); //获取应答IIC_WriteOneByte(addr);  //发送存放数据的地址IIC_GetACK(); //获取应答IIC_WriteOneByte(data);  //发送实际要存放的数据IIC_GetACK(); //获取应答IIC_STOP();   //发送停止信号DelayMs(10);  //等待写完成
}/*
函数功能: 指定位置读取指定数量的数据
函数参数:addr: 从哪里开始读取数据len : 读取多长的数据*p  : 存放数据的缓冲区编写程序对AT24C16第100页的第3个字节进行读数据的时候,步骤如下:
1)发送起始信号;
2)发送器件地址0XA6(1010 0110,1010是固定地址,011是页地址的高三位,0表示写操作);
3)发送操作地址0X43(0100 0011,0100是页地址的低四位,0011是页地址偏移量,即第100页内的第三个字节,
4)发送要写的数据,
5)发送终止信号。
(对于AT24C02,直接写设备地址和数据地址)
*/
void AT24C16_ReadData(u16 addr,u16 len,char *p)
{u16 i;u8 page_addr,page_addr_H,Devicce_Write_Addr,Devicce_Read_Addr;page_addr=addr>>4;page_addr_H=page_addr>>5;Devicce_Write_Addr=AT24C16_WRITE_ADDR + (page_addr_H<<1);Devicce_Read_Addr=AT24C16_READ_ADDR+(page_addr_H<<1);
//	printf("Devicce_Write_Addr = 0x%X \n Devicce_Read_Addr = 0x%X \n addr =0x%X \n",Devicce_Write_Addr,Devicce_Read_Addr,addr&0xFF);IIC_START();IIC_WriteOneByte(Devicce_Write_Addr); //发送设备地址(写)IIC_GetACK(); //获取应答IIC_WriteOneByte(addr&0xFF); //发送存放数据的地址(即将读取数据的地址)IIC_GetACK(); //获取应答IIC_START();IIC_WriteOneByte(Devicce_Read_Addr); //发送设备地址(读)IIC_GetACK(); //获取应答for(i=0;i<len;i++){p[i]=IIC_ReadOneByte(); //接收数据IIC_SendAck(0);  //给从机发送应答}IIC_SendAck(1);  //给从机发送非应答IIC_STOP();      //停止信号
} /*
函数功能: 指定位置写入指定数量的数据
函数参数:addr: 从哪里开始写数据len : 写入多长的数据*p  : 存放数据的缓冲区
*/
void AT24C16_WriteData(u16 addr,u16 len,u8 *p)
{u8 page_remain=16-addr%16; //得到当前页剩余的字节数量if(page_remain>=len){page_remain=len;}while(1){AT24C16_PageWrite(addr,page_remain,p);if(page_remain==len)break;addr+=page_remain;p+=page_remain;len-=page_remain;if(len>=16)page_remain=16;else page_remain=len;}
}/*
函数功能:  AT24C16页写函数
函数参数:addr: 从哪里开始写数据len : 写入多长的数据*p  : 存放数据的缓冲区说明: AT24C16内部有一个页写缓冲器,页地址超出之后会复位到当前页的起始地址。
AT24C16页缓冲器大小:  16个字节
编写程序对AT24C16第100页的第3个字节进行写数据的时候,步骤如下:
1)发送起始信号;
2)发送器件地址0XA6(1010 0110,1010是固定地址,011是页地址的高三位,0表示写操作);
3)发送操作地址0X43(0100 0011,0100是页地址的低四位,0011是页地址偏移量,即第100页内的第三个字节,
4)发送要写的数据,
5)发送终止信号。
(对于AT24C02,直接写设备地址和数据地址)
*/
void AT24C16_PageWrite(u16 addr,u16 len,u8 *p)
{u8 i;u8 page_addr,page_addr_H,Devicce_Write_Addr,Devicce_Read_Addr;page_addr=addr>>4;page_addr_H=page_addr>>5;Devicce_Write_Addr=AT24C16_WRITE_ADDR + (page_addr_H<<1);Devicce_Read_Addr=AT24C16_READ_ADDR+(page_addr_H<<1);printf("Devicce_Write_Addr = 0x%X \n Devicce_Read_Addr = 0x%X \n addr =0x%X \n",Devicce_Write_Addr,Devicce_Read_Addr,addr&0xFF);IIC_START();IIC_WriteOneByte(Devicce_Write_Addr); //发送设备地址IIC_GetACK(); //获取应答IIC_WriteOneByte(addr&0xFF);  //发送存放数据的地址IIC_GetACK(); //获取应答for(i=0;i<len;i++){IIC_WriteOneByte(p[i]);  //发送实际要存放的数据IIC_GetACK(); //获取应答}IIC_STOP();   //发送停止信号DelayMs(10);  //等待写完成
}