[ESP32]:基于esp-modbus实现serial主机

[ESP32]:基于esp-modbus实现serial主机

开发环境

  • esp idf 5.1
  • esp-modbus 1.0.13

使用如下指令添加组件,或者访问esp-modbus

idf.py add-dependency "espressif/esp-modbus^1.0.13"

Device parameters

对于master而言,需要理解的主要是device parameters,下面是一个parameters的例子

enum
{
    CID_INPUT = 0,
    CID_COIL,
    CID_HOLDING,
};

const mb_parameter_descriptor_t device_parameters[] = {
    {CID_INPUT, (const char *)("INPUT"), (const char *)(""), 1, MB_PARAM_INPUT, 0x00, 5,
     INPUT_OFFSET(input_value1), PARAM_TYPE_U16, sizeof(uint16_t), OPTS(0, 0, 0), PAR_PERMS_READ_WRITE_TRIGGER},
    {CID_COIL, (const char *)("COIL"), (const char *)(""), 1, MB_PARAM_COIL, 0x00, 16,
     COIL_OFFSET(coil_value1), PARAM_TYPE_U8, sizeof(uint8_t), OPTS(0, 0, 0), PAR_PERMS_READ_WRITE_TRIGGER},
    {CID_HOLDING, (const char *)("HOLDING"), (const char *)(""), 1, MB_PARAM_HOLDING, 0x00, 5,
     HOLDING_OFFSET(holding_value1), PARAM_TYPE_U16, sizeof(uint16_t), OPTS(0, 0, 0), PAR_PERMS_READ_WRITE_TRIGGER},
};

可以看到,我们这边分别定义了输入寄存器、线圈与保持寄存器,其中,关于mb_parameter_descriptor_t的定义如下:

typedef struct {
    uint16_t            cid;                /*!< Characteristic cid */
    const char*         param_key;          /*!< The key (name) of the parameter */
    const char*         param_units;        /*!< The physical units of the parameter */
    uint8_t             mb_slave_addr;      /*!< Slave address of device in the Modbus segment */
    mb_param_type_t     mb_param_type;      /*!< Type of modbus parameter */
    uint16_t            mb_reg_start;       /*!< This is the Modbus register address. This is the 0 based value. */
    uint16_t            mb_size;            /*!< Size of mb parameter in registers */
    uint16_t            param_offset;       /*!< Parameter name (OFFSET in the parameter structure) */
    mb_descr_type_t     param_type;         /*!< Float, U8, U16, U32, ASCII, etc. */
    mb_descr_size_t     param_size;         /*!< Number of bytes in the parameter. */
    mb_parameter_opt_t  param_opts;         /*!< Parameter options used to check limits and etc. */
    mb_param_perms_t    access;             /*!< Access permissions based on mode */
} mb_parameter_descriptor_t;

其中:

  • mb_size: 是你要读取的数量
    • CID_INPUT这个parameters里,5就是要读取5个输入寄存器
    • CID_COIL这个parameters里,16就是要读取16个线圈,注意,每一个线圈只有1个位
  • param_size:是每一个数据的字节数
    • CID_INPUT里,每一个数据的大小为2个字节,同时数据保存的类型也是uint16_t
    • CID_COIL里,每一个数据的大小为1个字节,就是sizeof(uint8_t),但是我们上面写的是16个线圈,所以读取数据的时候要写两个uint8_t来存放结果

master读取

对于读取,只需要选择好device parameters里的cid即可,下面是一个读取线圈的例子

esp_err_t modbus_master_read_coil_reg()
{
    esp_err_t err = mbc_master_get_cid_info(CID_COIL, &param_descriptor);
    if ((err != ESP_ERR_NOT_FOUND) && (param_descriptor != NULL))
    {
        void *temp_data_ptr = master_get_param_data(param_descriptor);
        assert(temp_data_ptr);
        uint8_t type = 0;
        uint8_t value[2] = {0};
        err = mbc_master_get_parameter(CID_COIL, (char *)param_descriptor->param_key, (uint8_t *)&value, &type);
        if (err == ESP_OK)
        {
            ESP_LOGI(TAG, "Coil: %d %d", (*(uint8_t *)&value[0]), *(uint8_t *)&value[1]);
        }
    }
    return err;
}

我们在刚才的device parameters里定义了要读16个线圈,返回的数据每一个是uint8_t,所以需要两个uint8_t的数据,所以我这里写一个数组。

实验效果如下:

在这里插入图片描述

在这里插入图片描述

127就是10000001

读取保持寄存与线圈的代码都差不多:

/**
 * @brief mdobus read input reg
 *
 * @return esp_err_t
 */
esp_err_t modbus_master_read_input_reg()
{
    esp_err_t err = mbc_master_get_cid_info(CID_INPUT, &param_descriptor);
    if ((err != ESP_ERR_NOT_FOUND) && (param_descriptor != NULL))
    {
        void *temp_data_ptr = master_get_param_data(param_descriptor);
        assert(temp_data_ptr);
        uint8_t type = 0;
        uint16_t value[5] = {0};
        err = mbc_master_get_parameter(CID_INPUT, (char *)param_descriptor->param_key, (uint8_t *)&value, &type);
        if (err == ESP_OK)
        {
            ESP_LOGI(TAG, "Input: %d %d %d %d %d", value[0], value[1], value[2], value[3], value[4]);
        }
    }

    return err;
}

/**
 * @brief modbus read holding reg
 *
 * @return esp_err_t
 */
esp_err_t modbus_master_read_holding_reg()
{
    esp_err_t err = mbc_master_get_cid_info(CID_HOLDING, &param_descriptor);
    if ((err != ESP_ERR_NOT_FOUND) && (param_descriptor != NULL))
    {
        void *temp_data_ptr = master_get_param_data(param_descriptor);
        assert(temp_data_ptr);
        uint8_t type = 0;
        uint16_t value[5] = {0};
        err = mbc_master_get_parameter(CID_HOLDING, (char *)param_descriptor->param_key, (uint8_t *)&value, &type);
        if (err == ESP_OK)
        {
            ESP_LOGI(TAG, "Holding: %d %d %d %d %d", value[0], value[1], value[2], value[3], value[4]);
        }
    }

    return err;
}

完整代码参考:项目代码

Ref

modbus协议

  • 7
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 可以使用 Arduino 的第三方库来实现 ESP32 上的 Modbus RTU slave 功能。其中,一个常用的库是 "Modbus-Master-Slave-for-Arduino"。 使用这个库之前,需要先在 Arduino IDE 中安装 ESP32 的开发板支持。之后,可以在 Arduino IDE 的库管理器中搜索 "Modbus-Master-Slave-for-Arduino" 并安装该库。 示例代码如下: ``` #include <ModbusRtu.h> Modbus slave(1, 9600); void setup() { Serial.begin(9600); slave.begin(Serial); slave.addHreg(0, 100); } void loop() { slave.task(); } ``` 这段代码定义了一个 slave 对象,并将其与 ESP32 的串口连接。在 setup 函数中,启动了串口通信,并将 slave 对象与串口连接。在 loop 函数中,调用 slave.task() 以执行 Modbus 协议的相关任务。 具体使用方式和完整例程可以参考库的文档。 ### 回答2: 要在ESP32 Arduino中实现Modbus RTU从机功能,可以按照以下步骤进行: 1. 软件安装:首先,确保已经在Arduino IDE中安装了ESP32的开发环境,并且已经具备ESP32开发板的基本使用经验。 2. 下载库文件:在Arduino库管理器中搜索ModbusSlave库并安装。这个库是用来实现Modbus RTU从机功能的。 3. 引入库文件:在Arduino IDE的代码编辑界面中,点击“工具”菜单,然后选择“包含库”并点击“ModbusSlave”来引入库文件。 4. 编写代码:编写ESP32 Arduino的代码来实现Modbus RTU从机功能。可以参考ModbusSlave库提供的示例代码,并根据具体需求对代码进行修改。 5. 配置串口:使用ESP32Serial库来配置串口连接设置。根据Modbus RTU协议的要求,配置正确的波特率、起始位、停止位和奇偶校验位等参数。 6. 实现Modbus函数:在代码中实现所需的Modbus函数,例如读取保持寄存器的数据、写入保持寄存器的数据等。这些函数将响应Modbus主机的请求。 7. 运行代码:将代码烧录到ESP32开发板中,并用Modbus主机设备进行测试。确保能够正常读写数据,且与主机设备通信稳定。 总结:通过安装库文件、编写代码和配置串口等步骤,可以在ESP32 Arduino中实现Modbus RTU从机功能。这样,ESP32开发板就能够与其他Modbus主机设备进行通信,并实现数据读写功能。 ### 回答3: ESP32是一款功能强大的微控制器,可以通过Arduino编程环境使用。modbusRTU是一种通信协议,常用于工业控制系统中的设备之间的通信。ESP32可以通过串口(Serial)功能实现modbusRTU从站(slave)的功能。 首先,需要在Arduino IDE中安装ESP32开发板的支持库。然后,创建一个新的Arduino项目,并在项目中引入modbusRTU库。 在代码中,首先需要初始化串口,并设置波特率和其他相关参数。设置ESP32作为modbusRTU从站时,需要设置从站地址。可以使用`modbus_configure`函数来完成这些操作。 接下来,可以在代码中添加相关的modbus功能。通过`modbus_update`函数,可以将ESP32从站中需要响应的数据放入缓冲区。当主站发起读写操作时,ESP32会根据缓冲区中的数据进行响应。 为了实现modbusRTU从站功能,还需要处理从主站接收到的数据。可以通过`modbus_read_registers`函数读取主站发送的寄存器数据,或者使用`modbus_write_registers`函数将ESP32从站中的寄存器数据发送到主站。 最后,可以在Arduino的`loop`函数中不断监听和处理modbus通信。根据项目需求,可以添加其他功能,如数据处理、传感器读取和控制等。 总结来说,ESP32通过Arduino编程环境提供了方便易用的串口功能,可实现modbusRTU从站的功能。通过合理调用modbusRTU库中的函数,可以实现从站地址设置、数据响应和寄存器操作等功能。这样,ESP32就可以与其他设备进行modbus通信,实现工业控制系统中的数据交互。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值