I2C(Inter-Intergrated Circuit)总线详解
一、I2C总线基本概念
1.1 I2C简介
I2C(Inter-Integrated Circuit)是由Philips公司开发的一种串行、同步、半双工的通信总线。主要特点:
-
两根线:SDA(数据线)和SCL(时钟线)
-
多主多从结构
-
7位/10位地址寻址
-
标准模式 (100kbps)、快速模式 (400kbps)、高速模式(3.4Mbps)
1.2 物理层特性
SDA ─────┬─────────┐
│ │
上拉 上拉
│ │
SCL ─────┴─────────┘
主机 从机
二、I2C通信时序
2.1 基本时序单元
// 开始信号:SCL高电平时,SDA从高变低
void i2c_start(void) {
SDA_HIGH();
SCL_HIGH();
DELAY();
SDA_LOW();
DELAY();
SCL_LOW();
}
// 停止信号:SCL高电平时,SDA从低变高
void i2c_stop(void) {
SDA_LOW();
SCL_HIGH();
DELAY();
SDA_HIGH();
DELAY();
}
2.2 数据传输格式
开始信号 + 7位地址 + 1位R/W + ACK + 数据 + ACK + ... + 停止信号
-
地址位:7位从机地址(0x00-0x7F)
-
R/W位:0-写,1-读
-
ACK位:每个字节后接收方应答
三、i.MX6UL的I2C控制器
3.1 寄存器概览
// I2C相关寄存器定义(从原代码提取)
#define I2CR_IEN (1 << 7) // I2C使能位
#define I2CR_MSTA (1 << 5) // 主从模式选择位
#define I2CR_MTX (1 << 4) // 收发模式位
#define I2CR_TXAK (1 << 3) // 应答类型位
#define I2CR_RSTA (1 << 2) // 重复起始信号
#define I2SR_ICF (1 << 7) // 数据传输完成标志
#define I2SR_IBB (1 << 5) // 总线忙标志
#define I2SR_IAL (1 << 4) // 仲裁丢失标志
#define I2SR_IIF (1 << 1) // 中断标志
#define I2SR_RXAK (1 << 0) // 接收应答标志
3.2 初始化流程(从原代码分析)
void i2c_init(I2C_Type *base)
{
// 1. 引脚复用配置
IOMUXC_SetPinMux(IOMUXC_UART4_RX_DATA_I2C1_SDA, 1); // SDA
IOMUXC_SetPinMux(IOMUXC_UART4_TX_DATA_I2C1_SCL, 1); // SCL
// 2. 电气特性配置(上拉、驱动能力等)
IOMUXC_SetPinConfig(IOMUXC_UART4_RX_DATA_I2C1_SDA, 0x10B0);
IOMUXC_SetPinConfig(IOMUXC_UART4_TX_DATA_I2C1_SCL, 0x10B0);
// 3. I2C控制器初始化
base->I2CR &= ~I2CR_IEN; // 先失能I2C
base->IFDR = 0x15; // 设置时钟分频(384KHz)
base->I2CR |= I2CR_IEN; // 使能I2C
}
四、I2C写操作实现分析
4.1 完整的写数据流程
// 原代码中的i2c_write函数分析
void i2c_write(I2C_Type *base, unsigned char dev_addr,
unsigned char reg_addr, unsigned char *data,
unsigned int len)
{
int stat = 0;
// 0. 清除状态标志
base->I2SR &= ~(I2SR_IAL | I2SR_IIF);
// 1. 设置为发送模式
base->I2CR |= I2CR_MTX;
// 2. 产生START信号(设置为主机模式)
base->I2CR |= I2CR_MSTA;
// 3. 发送从机地址(写模式)
base->I2DR = ((dev_addr << 1) | 0); // 地址左移1位,最低位0表示写
stat = wait_i2c_iif(base); // 等待传输完成
// 4. 发送寄存器地址
base->I2DR = reg_addr;
stat = wait_i2c_iif(base);
// 5. 循环发送数据
for (int i = 0; i < len; i++)
{
base->I2DR = data[i];
stat = wait_i2c_iif(base);
}
// 6. 产生STOP信号
stop:
base->I2CR &= ~I2CR_MSTA; // 清除主机模式,产生STOP
while ((base->I2SR & I2SR_IBB) != 0) // 等待总线空闲
{
// 超时处理(原代码中待完善)
}
delay_ms(5); // 延时确保STOP完成
}
4.2 等待中断标志函数
// 等待传输完成的中断标志
int wait_i2c_iif(I2C_Type *base)
{
// 等待IIF(中断标志)置位
while ((base->I2SR & I2SR_IIF) == 0){
// 建议添加超时机制
// if(timeout) return -2;
}
// 清除中断标志
base->I2SR &= ~I2SR_IIF;
// 检查ACK(应答)
if ((base->I2SR & I2SR_RXAK) != 0)
{
return -1; // NACK,从机未应答
}
return 0; // ACK,传输成功
}
五、常见问题与调试技巧
5.1 常见问题
-
无应答(NACK)
-
检查从机地址是否正确
-
确认从机设备是否上电
-
检查上拉电阻(通常4.7KΩ)
-
-
仲裁丢失
-
多个主机同时尝试控制总线
-
总线被意外干扰
-
-
时钟速率问题
-
确保IFDR寄存器配置正确
-
考虑总线电容导致的信号完整性
-
5.2 调试建议
// 添加调试信息的改进版本
void i2c_write_debug(I2C_Type *base, ...)
{
printf("I2C Write Start - Device: 0x%02X, Reg: 0x%02X\n",
dev_addr, reg_addr);
// ... 原有代码 ...
if (stat == -1) {
printf("ERROR: NACK received at byte %d\n", i);
// 可以检查具体状态寄存器
printf("I2SR: 0x%02X\n", base->I2SR);
printf("I2CR: 0x%02X\n", base->I2CR);
}
printf("I2C Write Complete\n");
}
六、实际应用示例
6.1 EEPROM(AT24C02)操作
// 向EEPROM写入数据
void eeprom_write_page(uint8_t dev_addr, uint8_t mem_addr,
uint8_t *data, uint8_t len)
{
// AT24C02的器件地址:0xA0(7位地址为0x50)
uint8_t eeprom_addr = 0x50; // 1010000b
// 页写入(最多8字节)
if (len > 8) len = 8;
// 使用原代码的i2c_write函数
i2c_write(I2C1, eeprom_addr, mem_addr, data, len);
// EEPROM需要写入周期时间(5ms)
delay_ms(5);
}
// 从EEPROM读取数据
void eeprom_read_byte(uint8_t dev_addr, uint8_t mem_addr,
uint8_t *data, uint8_t len)
{
// 需要先发送要读取的地址,然后重新开始读操作
// 这里需要扩展原代码支持读操作
}
总结
I2C总线是一种简单高效的串行通信协议,特别适合板内低速外设通信。原代码实现了基本的写操作,但还有以下可以完善的地方:
-
缺少读操作:需要实现完整的读功能
-
错误处理不足:需要添加超时、重试机制
-
调试支持:添加状态输出便于调试
-
中断支持:可以考虑使用中断方式提高效率
理解I2C协议的关键是掌握其时序:START→地址+R/W→ACK→数据→ACK→...→STOP。在实际应用中,要特别注意时序的严格性和错误处理的完备性。