STM32基础---IIC通信(以AT24C02为例)

本文介绍了STM32的I2C通信基础知识,包括I2C总线的概念、传输方向和信号过程。以AT24C02作为实例,详细阐述了起始信号、应答信号、停止信号的定义以及数据传输的步骤。此外,还提供了GPIO配置、I2C通信代码实现的概要,强调了模拟IIC的使用原因。
摘要由CSDN通过智能技术生成

一,概念

I2C(IIC, Inter-Integrated Circuit,集成电路总线)总线是由Philips公司开发的一种简单、双向二线制同步串行总线。它只需要两根线即可在连接于总线上的器件之间传送信息。他是同步半双工通信

主器件用于启动总线传送数据,并产生时钟以开放传送的器件,此时任何被寻址的器件均被认为是从器件。

(二)传输方向

在总线上主和从、发和收的关系不是恒定的,而取决于此时数据传送方向。

如果主机要发送数据给从器件,则主机首先寻址从器件,然后主动发送数据至从器件,最后由主机终止数据传送。

如果主机要接收从器件的数据,首先由主器件寻址从器件.然后主机接收从器件发送的数据,最后由主机终止接收过程。在这种情况下,主机负责产生定时时钟和终止数据传送。

二,信过程

在时钟线低电平时,数据线进行数据翻转

I2C通信,存在几种信号

1)起始信号(条件):通知从机做好通信的准备。  //时钟线为高,数据线有高到低

2)应答信号:有应答和无应答。有应答是SDA为低电平,无应答是SDA为高电平。

3)停止信号(条件):告诉从机通信已经结束。  //时钟线为高,数据线有低变高

连续通信过程如图(AT24C02写入数据时序图为例)所示:主机先发送起始信号,然后发送寻址地址(寻址地址是由生产厂家和开发者共同决定的,本例子的设备地址是0xa0),等待从机响应,接着发送数据存储地址,等待AT24C02响应,接着每写入一个字节(8bit)从机应答一下。主机发送停止信号,通信结束。

三,代码实现

GPIO口配置

static GPIO_InitTypeDef		GPIO_InitStructure;

void at24c02_Init(void)
{
	// 输出模式
	/* 1.打开GPIO时钟(根据需要开关,降低功耗) */
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);

	/* 2.配置(模拟IIC) */
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9;	// 指定引脚
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;			// 模式:输出
	GPIO_InitStructure.GPIO_OType = GPIO_OType_
  • 5
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
硬件IIC驱动: 1. 定义IIC相关的GPIO引脚; 2. 初始化IIC总线; 3. 发送IIC起始信号; 4. 发送设备地址; 5. 发送数据或读取数据; 6. 发送IIC停止信号。 以下是一个简单的STM32F103C8T6硬件IIC驱动的示例代码: ```c #include "stm32f10x.h" #include "i2c.h" #define I2C_SCL_PIN GPIO_Pin_6 #define I2C_SDA_PIN GPIO_Pin_7 #define I2C_GPIO GPIOB #define I2C_RCC_GPIO RCC_APB2Periph_GPIOB #define I2C_SPEED 100000 void i2c_init(void) { GPIO_InitTypeDef GPIO_InitStructure; I2C_InitTypeDef I2C_InitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE); RCC_APB2PeriphClockCmd(I2C_RCC_GPIO, 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; GPIO_Init(I2C_GPIO, &GPIO_InitStructure); I2C_InitStructure.I2C_Mode = I2C_Mode_I2C; I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2; I2C_InitStructure.I2C_OwnAddress1 = 0; I2C_InitStructure.I2C_Ack = I2C_Ack_Enable; I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; I2C_InitStructure.I2C_ClockSpeed = I2C_SPEED; I2C_Init(I2C1, &I2C_InitStructure); I2C_Cmd(I2C1, ENABLE); } void i2c_start(void) { while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY)); I2C_GenerateSTART(I2C1, ENABLE); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT)); } void i2c_stop(void) { I2C_GenerateSTOP(I2C1, ENABLE); while(I2C_GetFlagStatus(I2C1, I2C_FLAG_STOPF)); } void i2c_send_byte(uint8_t byte) { I2C_SendData(I2C1, byte); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED)); } uint8_t i2c_receive_byte(void) { I2C_AcknowledgeConfig(I2C1, ENABLE); while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED)); uint8_t byte = I2C_ReceiveData(I2C1); I2C_AcknowledgeConfig(I2C1, DISABLE); return byte; } void i2c_send_address(uint8_t address, uint8_t direction) { if(direction == I2C_Direction_Transmitter) { address = (address << 1); } else { address = (address << 1) | 0x01; } I2C_Send7bitAddress(I2C1, address, direction); if(direction == I2C_Direction_Transmitter) { while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); } else { while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED)); } } void i2c_write_data(uint8_t address, uint8_t* data, uint8_t length) { i2c_start(); i2c_send_address(address, I2C_Direction_Transmitter); for(uint8_t i=0; i<length; i++) { i2c_send_byte(data[i]); } i2c_stop(); } void i2c_read_data(uint8_t address, uint8_t* data, uint8_t length) { i2c_start(); i2c_send_address(address, I2C_Direction_Receiver); for(uint8_t i=0; i<length; i++) { data[i] = i2c_receive_byte(); } i2c_stop(); } ``` AT24C02读写: AT24C02是一个2Kbit的串行EEPROM芯片,使用IIC总线通信。以下是一个简单的AT24C02读写的示例代码: ```c #include "stm32f10x.h" #include "i2c.h" #define AT24C02_ADDRESS 0xA0 #define AT24C02_PAGE_SIZE 8 void at24c02_write_byte(uint16_t address, uint8_t data) { uint8_t buffer[3]; buffer[0] = address >> 8; buffer[1] = address & 0xFF; buffer[2] = data; i2c_write_data(AT24C02_ADDRESS, buffer, 3); } uint8_t at24c02_read_byte(uint16_t address) { uint8_t buffer[2]; buffer[0] = address >> 8; buffer[1] = address & 0xFF; i2c_write_data(AT24C02_ADDRESS, buffer, 2); uint8_t data; i2c_read_data(AT24C02_ADDRESS, &data, 1); return data; } void at24c02_write_page(uint16_t address, uint8_t* data, uint8_t length) { uint8_t buffer[AT24C02_PAGE_SIZE + 2]; buffer[0] = address >> 8; buffer[1] = address & 0xFF; for(uint8_t i=0; i<length; i++) { buffer[2+i] = data[i]; } i2c_write_data(AT24C02_ADDRESS, buffer, length+2); } void at24c02_read_buffer(uint16_t address, uint8_t* data, uint8_t length) { uint8_t buffer[2]; buffer[0] = address >> 8; buffer[1] = address & 0xFF; i2c_write_data(AT24C02_ADDRESS, buffer, 2); i2c_read_data(AT24C02_ADDRESS, data, length); } ``` 在使用AT24C02时,需要注意以下几点: 1. 写入单个字节时,地址只需要两个字节; 2. 读取单个字节时,需要先写入地址,然后再读取数据; 3. 写入一页数据时,一页大小为8个字节,地址需要两个字节; 4. 读取多个字节时,需要先写入地址,然后再读取指定长度的数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值