别再只盯着FFT了!用Matlab玩转Chirp Z变换(CZT),轻松搞定非均匀频率分析
2026/6/6 3:24:40
I²C(Inter-Integrated Circuit),中文名为“集成电路总线”,是一种由 Philips 公司(现 NXP)在 1980 年代推出的串行通信总线协议。它广泛应用于嵌入式系统中,用于连接低速外设,比如 EEPROM、实时时钟(RTC)、传感器、OLED 显示屏等。
I²C 是一种同步、半双工的串行通信方式,其核心由以下两个信号组成:
| 信号 | 功能说明 |
|---|---|
| SCL(Serial Clock Line) | 时钟信号,由主机产生,控制数据传输节奏 |
| SDA(Serial Data Line) | 数据信号,双向传输,用于发送和接收数据 |
包含两个互补的 MOSFET(或三极管):
当输出高电平时,上管导通,下管关闭,引脚被“推”到高电平;
当输出低电平时,下管导通,上管关闭,引脚被“拉”到低电平。
开漏输出只包含一个下拉的 N-MOS 管,没有上拉部分:
当一个芯片输出高电平,真实的引脚电平。由外界芯片,上下拉电路决定
I²C 每次传输 8 位数据,以字节为单位。每个字节后紧跟一个应答位(ACK/NACK)。
每次通信开始时,主机先发送一个地址帧,包含:
| 寄存器 | 功能 |
|---|---|
I2CR(I2C Control Register) | 控制 I²C 工作模式、使能、速度等 |
I2SR(I2C Status Register) | 当前状态(如是否忙、是否收到 ACK) |
I2DR(I2C Data Register) | 数据收发缓冲区 |
I2ADR(I2C Address Register) | 设置从机地址(可选) |
i2c_init)引脚复用配置:将 UART4 的 RX/TX 引脚复用为 I²C1 的 SDA 和 SCL。
引脚电气配置:设置引脚为开漏输出模式,并配置上拉电阻、驱动强度等。0x10B0这个值通常包含了开漏使能(ODE)和上拉使能(PUS)等位。
void i2c_init(I2C_Type *base) { IOMUXC_SetPinMux(IOMUXC_UART4_RX_DATA_I2C1_SDA,1); IOMUXC_SetPinMux(IOMUXC_UART4_TX_DATA_I2C1_SCL,1); IOMUXC_SetPinConfig(IOMUXC_UART4_RX_DATA_I2C1_SDA, 0x10B0); IOMUXC_SetPinConfig(IOMUXC_UART4_TX_DATA_I2C1_SCL, 0x10B0); base->I2CR &= ~I2CR_IEN; base->IFDR=0x15; base->I2CR |= I2CR_IEN; }i2c_write)流程步骤:
1.初始化:配置引脚为开漏模式,设置上拉电阻,使能I²C控制器
2. 发送起始信号:主机发起START
3. 发送设备地址:发送7位从机地址+写标志(0)
4. 发送寄存器地址:发送要写入的寄存器地址
5. 发送数据:依次发送要写入的数据字节
6. 发送停止信号:主机发送STOP结束通信
void i2c_write(I2C_Type *base, unsigned char dev_addr, unsigned short reg_addr, int reg_len, unsigned char *data, unsigned int len ) { int stat = 0; //0. 清除IAL、IIF, base->I2SR &= ~(I2SR_IAL | I2SR_IIF); //1. 设置为发送模式 base->I2CR |= I2CR_MTX; //2.start信号 base->I2CR |= I2CR_MSTA; //3.发送从机地址及数据流向位 base->I2DR = ((dev_addr << 1) | 0); stat = wait_i2c_iif(base); if (stat != 0) goto stop; //4.发送从机寄存器地址 int i=reg_len-1; for (; i >=0; i--) { base->I2DR = (reg_addr >> (i * 8))& 0xFF; stat = wait_i2c_iif(base); if (stat != 0) goto stop; } for (i=0; i < len; i++) { base->I2DR = data[i]; stat = wait_i2c_iif(base); if (stat != 0) goto stop; } stop: //发送stop信号 base->I2CR &= ~I2CR_MSTA; while ((base->I2SR & I2SR_IBB) != 0) { //设计超时机制 } }i2c_read)void i2c_read(I2C_Type *base, unsigned char dev_addr, unsigned short reg_addr, int reg_len, unsigned char *data, unsigned int len) { int stat=0; base->I2SR &= ~(I2SR_IAL | I2SR_IIF); base->I2CR |= I2CR_MTX; //2.start信号 base->I2CR |= I2CR_MSTA; //3.发送从机地址及数据流向位 base->I2DR = ((dev_addr << 1) | 0); stat = wait_i2c_iif(base); if (stat != 0) goto stop; //4.发送从机寄存器地址 int i = reg_len - 1; for (; i >= 0; i--) { base->I2DR = (reg_addr >> (i * 8)) & 0xFF; stat = wait_i2c_iif(base); if (stat != 0) goto stop; } //5.重发start base->I2CR |= I2CR_RSTA; //6.发送从机地址及数据流向位 base->I2DR = ((dev_addr << 1)|1); stat = wait_i2c_iif(base); if (stat != 0) goto stop; //7.设置为接受模式 base->I2CR &= ~I2CR_MTX; //8.设置应答模式 if (len >1) { base->I2CR &= ~I2CR_TXAK; }else { base->I2CR |= I2CR_TXAK; } //9.触发一次伪读read_buff data[0]=base->I2DR; for (i=0; i < len; i++) { stat = wait_i2c_iif(base); if (i== len-2) { base->I2CR |= I2CR_TXAK; } if (i==len-1) { base->I2CR |= I2CR_MTX; } data[i]=base->I2DR; if (stat != 0) goto stop; } stop: //发出stop信号 base->I2CR &= ~I2CR_MSTA; while ((base->I2SR & I2SR_IBB) != 0) { //设计超时机制 } }