如果搞过51的
I2C的同志,再来看看
STM32的I2C驱动,一定有相见恨晚的感觉。STM32自带I2C硬件模块,再配合ST的官方库函数,I2C在STM32这里可以玩得如火的地步。这里的这个I2C驱动算是很完整了的,可以直接拿来用到项目开发中去。好,不废话,上图:
工程结构图:
1、main.c
#include"stm32f10x.h"
#include"usart1.h"
#include"led.h"
#include"i2c_at24c02.h"
#include
struct Contact{
u8 Name[15];
//姓名
u8 Phone[11];
//电话
u8 BirthDay[10];
//生日
}WgchnlnContact[3],NewContact;
struct Contact WgchnlnContact[3]={
{
"wgchnln",
"18682101373",
"1987-02-19",
},
{
"hezepan",
"18528473728",
"1987-10-28",
},
{
"lixiaowei",
"13527492729",
"1988-12-02",
}};
#ifdef Test
void I2C_Test(void)
{
u16 i;
u8 I2c_Buf[256];
printf("写入的数据");
//填充缓冲
for(i=0;i<=255;i++)
{
I2c_Buf[i]=i;
printf("0x%x ",I2c_Buf[i]);
if(i == 9)
{
printf("");
}
}
printf("");
I2C_AT24Cx_Writes(0,I2c_Buf,256);//将I2C_Buf中顺序递增的数据写入EERPOM中
//清缓冲
for(i=0;i<=255;i++)
{
I2c_Buf[i]=0;
}
printf("读出的数据");
I2C_AT24Cx_Reads(0,I2c_Buf,256);//将EEPROM读出数据顺序保持到I2C_Buf中
//将I2C_Buf中的数据通过串口打印
for(i=0;i<256;i++)
{
if(I2c_Buf[i]!=i)
{
printf("错误:I2C EEPROM写入与读出的数据不一致");
while(1);
}
printf("0x%X ", I2c_Buf[i]);
if(i == 9)
{
printf("");
}
}
}
#endif
int main(void)
{
u8 ReadBuffer[36];
u8 i=0;
u8 j=0;
USART_Config(); //串口1初始化
printf("这是一个I2C-EEPROM-AT24C02演示实验");
//Led_Init(); //LED初始化
I2C_AT24Cx_Init(); //I2C初始化
#ifdef Test
I2C_Test();
#endif
printf("写入联系人");
for(j=0;j<3;j++)
{
I2C_AT24Cx_Writes(0+j*36,WgchnlnContact[j].Name,15);
I2C_AT24Cx_Writes(15+j*36,WgchnlnContact[j].Phone,11);
I2C_AT24Cx_Writes(26+j*36,WgchnlnContact[j].BirthDay,10);
}
j=0;
printf("读出联系人");
for(j=0;j<3;j++)
{
I2C_AT24Cx_Reads(j*36,ReadBuffer,36);
for(i=0;i<15;i++)
{
NewContact.Name[i]=ReadBuffer[i];
}
i=0;
printf("姓名:%s",NewContact.Name);
for(i=0;i<11;i++)
{
NewContact.Phone[i]=ReadBuffer[i+15];
}
i=0;
printf("电话:%s",NewContact.Phone);//为什么打印第二个联系人的号码电话会把第一个联系人的生日一同打印出来
for(i=0;i<10;i++)
{
NewContact.BirthDay[i]=ReadBuffer[i+26];
}
i=0;
printf("生日:%s",NewContact.BirthDay);
}
while(1);
}
2、今天的主角:i2c_at24c02.c
#include"stm32f10x.h"
#include"i2c_at24c02.h"
#define AT24Cx_Address 0xa0 //I2C芯片地址 EEPROM
#define AT24Cx_PageSize 8 //芯片内部一页容量
static void I2C_ATC24x_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(I2C_SCL_GPIO_RCC,ENABLE); //没有外部中断,没有重映射,所以不需要开启复用功能
GPIO_InitStructure.GPIO_Pin =I2C_SCL_Pin|I2C_SDA_Pin;
GPIO_InitStructure.GPIO_Speed =GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_AF_OD; //设置管脚为复用功能开漏输出 SDA是输入输出引脚
GPIO_Init(I2C_SCL_GPIO, &GPIO_InitStructure);
}
static void I2C_Config(void)
{
I2C_InitTypeDef
I2C_InitStructure;
RCC_APB1PeriphClockCmd(I2C_RCC,ENABLE);
I2C_DeInit(I2C);
I2C_InitStructure.I2C_ClockSpeed =100000; //100KHz I2C时钟频率
I2C_InitStructure.I2C_Mode =I2C_Mode_I2C; //I2C模式
I2C_InitStructure.I2C_DutyCycle =I2C_DutyCycle_2; //时钟占空比
I2C_InitStructure.I2C_OwnAddress1 =0x30; //主机地址 可以任意的
I2C_InitStructure.I2C_Ack =I2C_Ack_Enable; //开启ACK应答响应
I2C_InitStructure.I2C_AcknowledgedAddress =I2C_AcknowledgedAddress_7bit;//7位地址模式 非10为地址模式
I2C_Init(I2C,&I2C_InitStructure);
I2C_Cmd(I2C,ENABLE);
I2C_AcknowledgeConfig(I2C,ENABLE);
}
void I2C_AT24Cx_Init(void)
{
I2C_ATC24x_GPIO_Config();
I2C_Config();
}
void I2C_AT24Cx_Reads(u8 Address,u8 *ReadBuffer,u16 ReadNumber)
{
if(ReadNumber==0) //没有需要读取的数据
return;
while(I2C_GetFlagStatus(I2C,I2C_FLAG_BUSY));
I2C_AcknowledgeConfig(I2C, ENABLE);
I2C_GenerateSTART(I2C,ENABLE);
while(!I2C_CheckEvent(I2C,I2C_EVENT_MASTER_MODE_SELECT)); //检查是不是主模式与起始位已经发送 备注:这样做的目的就是为了清空该事件