STM32F103学习笔记(7.1)——IIC读写AT24C02

先附上AT24C02的读写程序:提取码:spl0

目录

写字节时序

读字节时序


首先要知道,IIC的地址是七位,不是八位(有8位还有10位,只不过AT24C02用7位,以前大多用7位),在发送地址的时候,高7位是地址,最后一位是读写位

首先AT24C02的从机地址:1 0 1 0 A2 A1 A0 R/W

读写位0表写,1表读

前面四位1010是固定的,A2、A1、A0由芯片引脚决定,如果只有这一个设备,完全可以全部接地

所以AT24C02在A0 A1 A2全部接地时,地址是0x50,左移一位后是0xA0

写字节时序

1.发送起始条件

2.发送七位地址和读写位(需要接收应答)

3.发送数据地址(需要接收应答)

4.发送要写入的数据(需要接收应答)

5.继续发送要写入的数据,会按顺序写进内存,不需要再发送地址(每次发送需要接收应答)

6.直到不想继续写入,不发送应答

7.发送停止条件

代码可以是这样:

/***
addr:		从机地址
dataAdd:	数据地址
data:		数据指针
len:		写入数据长度
*/

void AT24C02_WriteByte(u8 addr, u8 dataAdd, unsigned char *data, u8 len)
{
	unsigned char i;
	IIC_Start();			//发送起始条件
	IIC_SendByte(addr<<1);	//发送七位地址,第零位0,表示写入
	IIC_ReadBck();			//读应答位
	IIC_SendByte(dataAdd);	//发送数据地址位
	IIC_ReadBck();			//读应答位
	
	for( i=0; i<len; i++)
	{
		IIC_SendByte(*(data+i));		//发送数据
		IIC_ReadBck();					//读应答位
	}
	
	IIC_Stop();				//发送停止条件
	delay_ms(10);
}

读字节时序

AT24C02的读有点不一样,它会直接从当前地址读出数据给主机,所以需要先需要用到写字节时序,让它内部的指针指向需要读字节的地址,然后发送起始条件再开始读。

代码可以是这样:

void AT24C02_ReadByte(u8 addr, u8 dataAdd, unsigned char *data, u8 len)
{
	unsigned char i;
	IIC_Start();				//发送起始条件
	IIC_SendByte(addr<<1);		//发送七位地址,第零位0,写入
	IIC_ReadBck();				//读应答位
	IIC_SendByte(dataAdd);		//发送数据地址位
	IIC_ReadBck();				//读应答位
	
	//重新发送起始条件
	IIC_Start();					//发送起始条件
	IIC_SendByte(addr<<1|1<<0);		//发送七位地址,第零位1,读
	IIC_ReadBck();					//读应答位
	for(i=0; i<len; i++)
	{
		*(data+i) = IIC_ReadByte();			//读一个字节
		IIC_SendBck();						//发送应答
	}
	IIC_Stop();						//发送停止条件
	delay_ms(10);
}
  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是 AT24C02STM32 的 I2C 驱动示例代码,仅供参考: ```c #include "stm32f10x.h" #include "stm32f10x_i2c.h" #define I2C_SPEED 100000 // I2C 通信速率为 100kHz #define I2C_TIMEOUT 10000 // I2C 超时时间为 10ms #define AT24C02_ADDRESS 0xA0 // AT24C02 设备地址 // 初始化 I2C 接口 void I2C_Init(void) { GPIO_InitTypeDef GPIO_InitStruct; I2C_InitTypeDef I2C_InitStruct; // 使能 GPIOB 和 I2C1 时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE); // 初始化 GPIOB.6 和 GPIOB.7 作为 I2C1 的 SCL 和 SDA 引脚 GPIO_InitStruct.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_OD; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStruct); // 初始化 I2C1 I2C_DeInit(I2C1); I2C_InitStruct.I2C_Mode = I2C_Mode_I2C; I2C_InitStruct.I2C_DutyCycle = I2C_DutyCycle_2; I2C_InitStruct.I2C_OwnAddress1 = 0x00; I2C_InitStruct.I2C_Ack = I2C_Ack_Enable; I2C_InitStruct.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; I2C_InitStruct.I2C_ClockSpeed = I2C_SPEED; I2C_Init(I2C1, &I2C_InitStruct); // 使能 I2C1 I2C_Cmd(I2C1, ENABLE); } // 写入一个字节到 AT24C02 的指定地址 void AT24C02_WriteByte(uint16_t addr, uint8_t data) { // 发送 START 信号 I2C_GenerateSTART(I2C1, ENABLE); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); // 发送 AT24C02 设备地址和写入控制位 I2C_Send7bitAddress(I2C1, AT24C02_ADDRESS, I2C_Direction_Transmitter); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); // 发送 AT24C02 内存地址 I2C_SendData(I2C1, (uint8_t)(addr >> 8)); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); I2C_SendData(I2C1, (uint8_t)(addr & 0xFF)); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); // 发送要写入的数据 I2C_SendData(I2C1, data); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); // 发送 STOP 信号 I2C_GenerateSTOP(I2C1, ENABLE); } // 从 AT24C02 的指定地址读取一个字节 uint8_t AT24C02_ReadByte(uint16_t addr) { uint8_t data; // 发送 START 信号 I2C_GenerateSTART(I2C1, ENABLE); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); // 发送 AT24C02 设备地址和写入控制位 I2C_Send7bitAddress(I2C1, AT24C02_ADDRESS, I2C_Direction_Transmitter); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); // 发送 AT24C02 内存地址 I2C_SendData(I2C1, (uint8_t)(addr >> 8)); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); I2C_SendData(I2C1, (uint8_t)(addr & 0xFF)); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); // 发送 RESTART 信号 I2C_GenerateSTART(I2C1, ENABLE); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); // 发送 AT24C02 设备地址读取控制位 I2C_Send7bitAddress(I2C1, AT24C02_ADDRESS, I2C_Direction_Receiver); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED)); // 禁用 ACK I2C_AcknowledgeConfig(I2C1, DISABLE); // 读取数据 I2C_GenerateSTOP(I2C1, ENABLE); // 等待 I2C 事件完成 while (!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED)); data = I2C_ReceiveData(I2C1); return data; } ``` 注意:以上代码仅供参考,实际使用时需要根据具体硬件和软件环境进行修改和调试。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

天师电通电容爆破工程师

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值