ESP32学习笔记(17)——I2C接口使用

发布者:MysticDreamer最新更新时间:2025-02-28 来源: jianshu关键字:ESP32  I2C接口  内部集成电路 手机看文章 扫描二维码
随时随地手机看文章

一、I2C简介

I2C(Inter-Integrated Circuit ,内部集成电路) 总线是一种由飞利浦 Philip 公司开发的串行总线。是两条串行的总线,它由一根数据线(SDA)和一根 时钟线(SCL)组成。两条线都需要上拉电阻。I2C 总线上可以接多个 I2C 设备,每个器件都有一个唯一的地址识别。同一时间只能有一个主设备,其他为从设备。通常 MCU 作为主设备控制,外设作为从设备。

ESP32有两个I2C控制器(也称为端口),负责处理I2C总线上的通信。每个I2C控制器都可以作为主机或从机运行。

ESP-IDF 编程指南——I2C

二、API说明

以下 I2C 接口位于 driver/include/driver/i2c.h。

2.1 i2c_param_config

2.2 i2c_driver_install

2.3 i2c_cmd_link_create

2.4 i2c_master_start

2.5 i2c_master_write_byte

2.6 i2c_master_write

2.7 i2c_master_read_byte

2.8 i2c_master_read

2.9 i2c_master_stop

2.10 i2c_master_cmd_begin

2.11 i2c_cmd_link_delete

三、编程流程

3.1 设置通信参数

要建立I2C通信,请先配置驱动程序。这是通过设置结构的参数来完成的i2c_config_t:


设置I2C操作模式-从机或主机i2c_mode_t


配置通讯引脚


为SDA和SCL信号分配GPIO引脚


设置是否启用ESP32的内部上拉电路


(仅限主机)设置I2C时钟速度


(仅限从机)配置以下内容


是否启用10位地址模式


定义从机地址


之后,初始化给定I2C端口的配置。为此,调用函数i2c_param_config()并将端口号和结构传递给该函数i2c_config_t。


配置示例(主机):




int i2c_master_port = 0;i2c_config_t conf = {

    .mode = I2C_MODE_MASTER,

    .sda_io_num = I2C_MASTER_SDA_IO,         // select GPIO specific to your project

    .sda_pullup_en = GPIO_PULLUP_ENABLE,

    .scl_io_num = I2C_MASTER_SCL_IO,         // select GPIO specific to your project

    .scl_pullup_en = GPIO_PULLUP_ENABLE,

    .master.clk_speed = I2C_MASTER_FREQ_HZ,  // select frequency specific to your project

    // .clk_flags = 0,          /*!< Optional, you can use I2C_SCLK_SRC_FLAG_* flags to choose i2c source clock here. */};

配置示例(从机):



int i2c_slave_port = I2C_SLAVE_NUM;i2c_config_t conf_slave = {

    .sda_io_num = I2C_SLAVE_SDA_IO,          // select GPIO specific to your project

    .sda_pullup_en = GPIO_PULLUP_ENABLE,

    .scl_io_num = I2C_SLAVE_SCL_IO,          // select GPIO specific to your project

    .scl_pullup_en = GPIO_PULLUP_ENABLE,

    .mode = I2C_MODE_SLAVE,

    .slave.addr_10bit_en = 0,

    .slave.slave_addr = ESP_SLAVE_ADDR,      // address of your project};

3.2 驱动程序安装

配置I2C驱动程序后,通过i2c_driver_install()使用以下参数调用该函数来安装它:


端口号,来自的两个端口号之一 i2c_port_t


主机或从机,选自 i2c_mode_t


(仅限从机)为发送和接收数据分配的缓冲区大小。由于I2C是以主机为中心的总线,因此,只有在主机请求时,数据才能从从机传输到主机。因此,从设备通常将具有一个发送缓冲区,从设备在其中写入应用程序数据。数据保留在发送缓冲区中,由主机自行决定是否由主机读取。


用于分配中断的标志(请参阅esp_hw_support / include / esp_intr_alloc.h中的ESP_INTR_FLAG_ *值)


3.3 运行I2C通信

安装I2C驱动程序后,ESP32已准备好与其他I2C设备通信。


ESP32的I2C控制器作为主设备负责与I2C从设备建立通信,并发送命令以触发从设备采取行动,例如进行测量并将读数发送回主设备。


为了更好地进行流程组织,驱动程序提供了一个称为“命令链接”的容器,该容器应填充一系列命令,然后传递给I2C控制器以执行。


3.3.1 作为主机通信

主机写

下面的示例显示了如何为I2C主设备构建命令链接,以将n个字节发送给从设备。

  1. 使用创建一个命令链接i2c_cmd_link_create()。
    然后,用要发送到从站的一系列数据填充它:

    这两个函数i2c_master_write_byte()和i2c_master_write()具有一个额外的参数指定所述主是否应当确保它已经接收到ACK位。

    1. 起始位-i2c_master_start()

    2. 从机地址- i2c_master_write_byte()。提供单字节地址作为此函数调用的参数。

    3. 数据-一个或多个字节作为参数i2c_master_write()

    4. 停止位-i2c_master_stop()

  2. 通过调用触发I2C控制器执行命令链接i2c_master_cmd_begin()。一旦触发执行,就不能修改命令链接。

  3. 传输命令后,通过调用释放命令链接使用的资源i2c_cmd_link_delete()。

  • 主机读
    下面的示例显示了如何为I2C主机构建命令链接以从从机读取n个字节。


    与写入数据相比,第4步中的命令链接不是使用i2c_master_write...函数而是使用i2c_master_read_byte()或i2c_master_read()。同样,配置步骤5中的最后一次读取,以便主机不提供ACK位。


  • 写或读指示
    发送从机地址后(请参见上图两个步骤中的步骤3),主机会和从机进行写入或读取。

有关主机实际操作的信息隐藏在从机地址的最低有效位中。

因此,主机发送的用于将数据写入从机的命令链接包含该地址,如下所示:(ESP_SLAVE_ADDR << 1) | I2C_MASTER_WRITE

i2c_master_write_byte(cmd, (ESP_SLAVE_ADDR << 1) | I2C_MASTER_WRITE, ACK_EN);

同样,从从站读取的命令链接如下所示:

i2c_master_write_byte(cmd, (ESP_SLAVE_ADDR << 1) | I2C_MASTER_READ, ACK_EN);

3.3.2 作为从机通信

安装I2C驱动程序后,ESP32已准备好与其他I2C设备通信。

该API为从机提供以下功能

  • i2c_slave_read_buffer()
    当主机将数据写入从机时,从机将自动将其存储在接收缓冲区中。这允许从机应用程序i2c_slave_read_buffer()自行决定调用该函数。如果接收缓冲区中没有数据,此函数还具有一个参数,用于指定块时间。这将允许从机应用程序在指定的超时时间内等待数据到达缓冲区。

  • i2c_slave_write_buffer()
    发送缓冲区用于存储从设备要以FIFO顺序发送给主设备的所有数据。数据将一直保留在那里,直到主设备请求为止。该函数i2c_slave_write_buffer()具有一个参数,用于指定发送缓冲区已满时的阻止时间。这将允许从机应用程序在指定的超时时间内等待发送缓冲区中足够的可用空间。

四、ESP32作为主机与BH1750光照强度传感器通信

#include

#include 'esp_log.h'

#include 'driver/i2c.h'


static const char *TAG = 'i2c-example';


#define I2C_MASTER_SCL_IO GPIO_NUM_23           /*!< gpio number for I2C master clock */

#define I2C_MASTER_SDA_IO GPIO_NUM_18           /*!< gpio number for I2C master data  */

#define I2C_MASTER_NUM 1                        /*!< I2C port number for master dev */

#define I2C_MASTER_FREQ_HZ 100000               /*!< I2C master clock frequency */

#define I2C_MASTER_TX_BUF_DISABLE 0             /*!< I2C master doesn't need buffer */

#define I2C_MASTER_RX_BUF_DISABLE 0             /*!< I2C master doesn't need buffer */


#define WRITE_BIT I2C_MASTER_WRITE              /*!< I2C master write */

#define READ_BIT I2C_MASTER_READ                /*!< I2C master read */

#define ACK_CHECK_EN 0x1                        /*!< I2C master will check ack from slave*/

#define ACK_CHECK_DIS 0x0                       /*!< I2C master will not check ack from slave */

#define ACK_VAL 0x0                             /*!< I2C ack value */

#define NACK_VAL 0x1                            /*!< I2C nack value */


#define BH1750_SLAVE_ADDR   0x23 // 从机地址

#define BH1750_PWR_DOWN     0x00 // 关闭模块

#define BH1750_PWR_ON       0x01 // 打开模块等待测量指令

#define BH1750_RST          0x07 // 重置数据寄存器值在PowerOn模式下有效

#define BH1750_CON_H        0x10 // 连续高分辨率模式,1lx,120ms

#define BH1750_CON_H2       0x11 // 连续高分辨率模式,0.5lx,120ms

#define BH1750_CON_L        0x13 // 连续低分辨率模式,4lx,16ms

#define BH1750_ONE_H        0x20 // 一次高分辨率模式,1lx,120ms,测量后模块转到PowerDown模式

#define BH1750_ONE_H2       0x21 // 一次高分辨率模式,0.5lx,120ms,测量后模块转到PowerDown模式

#define BH1750_ONE_L        0x23 // 一次低分辨率模式,4lx,16ms,测量后模块转到PowerDown模式


SemaphoreHandle_t print_mux = NULL;


/**

 @brief I2C驱动初始化

 @param 无

 @return 无

*/

int I2C_Init(void)

{

    int i2c_master_port = I2C_MASTER_NUM;

    i2c_config_t conf;

    conf.mode = I2C_MODE_MASTER;

    conf.sda_io_num = I2C_MASTER_SDA_IO;

    conf.sda_pullup_en = GPIO_PULLUP_ENABLE;

    conf.scl_io_num = I2C_MASTER_SCL_IO;

    conf.scl_pullup_en = GPIO_PULLUP_ENABLE;

    conf.master.clk_speed = I2C_MASTER_FREQ_HZ;

    i2c_param_config(i2c_master_port, &conf);

    return i2c_driver_install(i2c_master_port, conf.mode, I2C_MASTER_RX_BUF_DISABLE, I2C_MASTER_TX_BUF_DISABLE, 0);

}


/**

 @brief I2C写数据函数

 @param slaveAddr -[in] 从设备地址

 @param regAddr -[in] 寄存器地址

 @param pData -[in] 写入数据

 @param dataLen -[in] 写入数据长度

 @return 错误码

*/

int I2C_WriteData(uint8_t slaveAddr, uint8_t regAddr, uint8_t *pData, uint16_t dataLen)

{

    int ret;

    i2c_cmd_handle_t cmd = i2c_cmd_link_create();

    i2c_master_start(cmd);

    i2c_master_write_byte(cmd, (slaveAddr << 1) | WRITE_BIT, ACK_CHECK_EN);

    if(NULL != regAddr)

    {

        i2c_master_write_byte(cmd, regAddr, ACK_CHECK_EN);

    }

    i2c_master_write(cmd, pData, dataLen, ACK_CHECK_EN);

    i2c_master_stop(cmd);

    ret = i2c_master_cmd_begin(1, cmd, 1000 / portTICK_RATE_MS);

    i2c_cmd_link_delete(cmd);

    return ret;

}


/**

 @brief I2C读数据函数

 @param slaveAddr -[in] 从设备地址

 @param regAddr -[in] 寄存器地址

 @param pData -[in] 读出数据

 @param dataLen -[in] 读出数据长度

 @return 错误码

*/

int I2C_ReadData(uint8_t slaveAddr, uint8_t regAddr, uint8_t *pData, uint16_t dataLen)

{

    int ret;

    i2c_cmd_handle_t cmd = i2c_cmd_link_create();

    i2c_master_start(cmd);

    i2c_master_write_byte(cmd, (slaveAddr << 1) | READ_BIT, ACK_CHECK_EN);

    if(NULL != regAddr)

    {

        i2c_master_write_byte(cmd, regAddr, ACK_CHECK_EN);

    }

    i2c_master_read(cmd, pData, dataLen, ACK_VAL);

    i2c_master_stop(cmd);

    ret = i2c_master_cmd_begin(1, cmd, 1000 / portTICK_RATE_MS);

    i2c_cmd_link_delete(cmd);

    return ret;

}


static void i2c_test_task(void *arg)

{

    int ret;

    uint8_t sensor_data[2] = {0};

    uint8_t sensor_data_h, sensor_data_l;

    int cnt = 0;

    uint8_t data;


    while (1) {

        ESP_LOGI(TAG, 'test cnt: %d', cnt++);

        data = BH1750_PWR_ON;              // 发送启动命令

        I2C_WriteData(BH1750_SLAVE_ADDR, NULL, &data, 1);

        data = BH1750_ONE_L;               // 设置一次低分辨率模式,测量后模块转到PowerDown模式

        I2C_WriteData(BH1750_SLAVE_ADDR, NULL, &data, 1);

        vTaskDelay(30 / portTICK_RATE_MS); // 设置完成后要有一段延迟

        ret = I2C_ReadData(BH1750_SLAVE_ADDR, NULL, sensor_data, 2);

        sensor_data_h = sensor_data[0];

        sensor_data_l = sensor_data[1];

        // ret = i2c_master_sensor_test(I2C_MASTER_NUM, &sensor_data_h, &sensor_data_l);

        xSemaphoreTake(print_mux, portMAX_DELAY);

        if (ret == ESP_ERR_TIMEOUT) {

            ESP_LOGE(TAG, 'I2C Timeout');

[1] [2]
关键字:ESP32  I2C接口  内部集成电路 引用地址:ESP32学习笔记(17)——I2C接口使用

上一篇:ESP32学习笔记(18)——光强度GY-30(BH1750)使用
下一篇:ESP32学习笔记(16)——Touch Sensor(触摸按键)接口使用

推荐阅读最新更新时间:2026-03-24 16:56

内部分电源IC继续缺货,逻辑器件待观
去年底,不少制造商在年终冲刺时就受困于部分器件短缺,特别是一些采用传统工艺的IC。而进入2006年,一些功率IC缺货的现象越来越明显。本刊网站论坛上不断有“采购反映交货期延长”的声音。大型制造商受到的影响还好,而一些中小型制造商的交货期被延至3个月甚至半年以上。   “目前市场上紧缺的主要是一些采用6英寸晶圆的双极产品,这是由于全球晶圆紧缺导致,短期内供应情况不会好转。”iSuppli中国区首席代表兼高级分析师吴同伟对本刊记者表示。他解释说,主要有以下二个原因导致了目前一些功率IC的严重缺货:   第一个原因是全球半导体工厂的产能都在向300mm晶圆的90纳米和65纳米新一代工艺转型,市场对这些新一代工艺的产品需求越来越
[焦点新闻]
51单片机OLED12864 I2C接口使用教程
现在能买到的OLED12864显示屏大多为SPI和I2C接口的,I2C通信协议只需要两条总线就可以进行通信,下面介绍一下如何用51单片机使用I2C接口的OLED12864。 首先介绍一下I2C通信协议,I2C(Inter-Integrated Circuit)字面上的意思是集成电路之间,它其实是I2CBus简称,所以中文应该叫集成电路总线,它是一种串行通信总线,使用多主从架构,由飞利浦公司在1980年代为了让主板、嵌入式系统或手机用以连接低速周边设备而发展。I2C的正确读法为“I平方C”( I-squared-C )。 I2C只使用两条双向漏极开路(Open Drain)(串行数据(SDA)及串行时钟频率(
[单片机]
51单片机OLED12864 <font color='red'>I2C接口</font>使用教程
51单片机OLED12864 I2C接口使用教程
现在能买到的OLED12864显示屏大多为SPI和I2C接口的,I2C通信协议只需要两条总线就可以进行通信,下面介绍一下如何用51单片机使用I2C接口的OLED12864。 首先介绍一下I2C通信协议,I2C(Inter-Integrated Circuit)字面上的意思是集成电路之间,它其实是I2CBus简称,所以中文应该叫集成电路总线,它是一种串行通信总线,使用多主从架构,由飞利浦公司在1980年代为了让主板、嵌入式系统或手机用以连接低速周边设备而发展。I2C的正确读法为“I平方C”( I-squared-C )。 I2C只使用两条双向漏极开路(Open Drain)(串行数据(SDA)及串行时钟频率(S
[单片机]
51单片机OLED12864 <font color='red'>I2C接口</font>使用教程
STM8 AT24CXX使用I2C接口读写
软件设计 /********************************************************************* 目 的: 建立AT24CXX操作库 目标系统: 基于STM8单片机 应用软件: Cosmic CxSTM8 *********************************************************************/ #define WD_DADR 0xa0 #define RD_DADR 0xa1 #include ws_i2c.h void AT24CXX_Init(void) { I2C_Init(); } void
[单片机]
STM32:I2C接口读写EEPROM(AT24C02)试验例程
硬件平台:stm32f10xZET6 开发环境:keil MDK uVisionv4.10 开发语言:C、ST_lib_3.5固件库 EEPROM:电可擦可编程只读存储器。 【stm32f10xZET6开发板的I2C外设物理层特点】 (1)两条串行总线:一条双向数据线(SDA),一条时钟线(SCL); (2)从设备地址唯一; (3)支持总线仲裁; (4)三种速率传输模式: 标准模式100kbit/s 快速模式400kbit/s 高速模式3.4Mbit/s (目前大多I2C设备尚不支持高速模式) (5)片上的滤波器可以滤去总线数据线上的毛刺波保证数据完整; (6)连接到相同总线的IC数量受到总线的最大电容400p
[单片机]
I2C接口与SPI和UART接口的区别
一、SPI I2C UART通信速率比较: SPI I2C UART 1、同步通信 异步通信; 2、同步通信时必须有一根时钟线连接传输的两端; 3、都是串行通信方式,并行通信用于内部存储间的通信,如flash; 4、适合传输的距离和通信速率成反比关系; 3-SPI:两条合一的数据线、1时钟线、1CS(设备片选线) SPI:2数据线、1时钟线、1CS(设备片选线)/串行 同步 通信全双工 I2C:1数据线、1时钟线/串行 同步 通信半双工 传输距离比UART短 UART:2数据线、 1地线/串行 异步 通信全双工 传输距离比I2C长些 (I2C接口是“器件间”接口,是在一块板子之内传输数据) (UART是 “
[单片机]
STM32F1使用I/0模拟I2C接口
使用模拟时序的方法,对比于硬件I2C接口来说,在实时性和传输速度上会带来一些无法避免的下降,但是I2C总线本身也不是一种速度很快的总线(据相关资料可查,最高的速度为400KHZ),同时也不需要具备很高的实时性能。 所以,模拟I2C时序完全能满足绝大部分的场合要求,并且移植性得到了很大的提高。 闲话不多说,贴上代码,大家一起分享下。 首先贴出 i2c_soft.h实现: /*********************************************************************************** * 文件名 :i2c_soft.h * 描述 :使用I/0模拟I2C接口 *
[单片机]
神舟IV学习笔记(六)I2C接口EEPROM-软硬件实现
EEPROM又称为电可擦可编程只读存储器,掉电后数据不丢失。广泛用于少量数据的保存,一些增强型51单片机和AVR都有芯片内部集成EEPROM。开发板使用的是I2C接口ATMEL的24C02芯片,芯片容量是2Kbit,也就是256字节。硬件连接如图所示,与STM32芯片上的I2C1硬件接口相连。 I2C的时序我这里就不介绍了,大家可以上网查查。I2C最大的优点就是有硬件地址,可以通过不同的硬件地址,挂载不同功能的I2C芯片。缺点就是速度慢,我们常使用到400kbps。 一硬件实现 STM32官方给的范例中给出了两个封装好的文件:I2C_EE.C和I2C_EE.H 给了这样几个函数 void I2C_EE_Init(voi
[单片机]
神舟IV学习笔记(六)<font color='red'>I2C接口</font>EEPROM-软硬件实现
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2026 EEWORLD.com.cn, Inc. All rights reserved