[RT-THREAD]agile—modbus移植记录

#include "agile_modbus_slave_app.h"

// ¼ì²éaddr
static int addr_check(agile_modbus_t *ctx, struct agile_modbus_slave_info *slave_info) {
    int slave = slave_info->sft->slave;
    if ((slave != ctx->slave) && (slave != AGILE_MODBUS_BROADCAST_ADDRESS) && (slave != 0xFF)) {
        return -AGILE_MODBUS_EXCEPTION_UNKNOW;
    }
    return 0;
}

// ÏßȦ״̬
static uint8_t _tab_bits[8] = {0};

static int get_map_buf(void *buf, int bufsz) {
    uint8_t *ptr = (uint8_t *)buf;
    for (int i = 0; i < sizeof(_tab_bits); i++) {
        ptr[i] = _tab_bits[i];
    }
    return 0;
}

static int set_map_buf(int index, int len, void *buf, int bufsz) {
    uint8_t *ptr = (uint8_t *)buf;
    for (int i = 0; i < len; i++) {
        _tab_bits[index + i] = ptr[index + i];
    }
    return 0;
}

const agile_modbus_slave_util_map_t bit_maps = {0x00, 0x00 + sizeof(_tab_bits), get_map_buf, set_map_buf};

// ÊäÈë״̬
static uint8_t _tab_input_bits[8] = {0};

static int get_input_buf(void *buf, int bufsz) {
    uint8_t *ptr = (uint8_t *)buf;
    for (int i = 0; i < sizeof(_tab_input_bits); i++) {
        ptr[i] = _tab_input_bits[i];
    }
    return 0;
}

const agile_modbus_slave_util_map_t input_bit_maps = {0x00, 0x00 + sizeof(_tab_input_bits), get_input_buf, NULL};

// ÊäÈë¼Ä´æÆ÷
static uint16_t _tab_input_registers[20] = {0};

static int get_inputReg_buf(void *buf, int bufsz) {
    uint16_t *ptr = (uint16_t *)buf;
    for (int i = 0; i < sizeof(_tab_input_registers) / sizeof(_tab_input_registers[0]); i++) {
        ptr[i] = _tab_input_registers[i];
    }
    return 0;
}

const agile_modbus_slave_util_map_t input_register_maps = {0x00, 0x00 + sizeof(_tab_input_registers), get_inputReg_buf,
                                                           NULL};

// ±£³Ö¼Ä´æÆ÷
static uint16_t _tab_registers[20] = {0};

static int get_registers_buf(void *buf, int bufsz) {
    uint16_t *ptr = (uint16_t *)buf;
    for (int i = 0; i < sizeof(_tab_registers) / sizeof(_tab_registers[0]); i++) {
        ptr[i] = _tab_registers[i];
    }
    return 0;
}

static int set_registers_buf(int index, int len, void *buf, int bufsz) {
    uint16_t *ptr = (uint16_t *)buf;
    for (int i = 0; i < len; i++) {
        _tab_registers[index + i] = ptr[index + i];
    }
    return 0;
}

const agile_modbus_slave_util_map_t register_maps = {0x00, 0x00 + sizeof(_tab_registers), get_registers_buf,
                                                     set_registers_buf};

//
const agile_modbus_slave_util_t slave_util = {&bit_maps,
                                              sizeof(bit_maps),
                                              &input_bit_maps,
                                              sizeof(input_bit_maps),
                                              &register_maps,
                                              sizeof(register_maps),
                                              &input_register_maps,
                                              sizeof(input_register_maps),
                                              addr_check,
                                              NULL,
                                              NULL};

/**
 *  @brief agile_modbus³õʼ»¯
 *  @param
 *  @retval
 */
agile_modbus_rtu_t modbus_slaver           = {0}; // modbus´Ó»ú¾ä±ú
static uint8_t     SlaverRequestBuff[256]  = {0}; // ÇëÇ󻺳åÇø
static uint8_t     SlaverResponseBuff[256] = {0}; // ÏìÓ¦»º³åÇø

void agile_modbus_init() {
    agile_modbus_rtu_init(&modbus_slaver, SlaverResponseBuff, sizeof(SlaverResponseBuff), SlaverRequestBuff,
                          sizeof(SlaverRequestBuff));
    agile_modbus_set_slave(&modbus_slaver._ctx, ThisID);
}

/**
 *  @brief agile_modbusÈÎÎñ
 *  @param
 *  @retval
 */
void agile_modbus_slave_task() {
    memcpy(SlaverRequestBuff, RS485RxBuffer, RS485RxLenth);
    if (RS485RxLenth) {
        int SlaverResponseBufLength = agile_modbus_slave_handle(&modbus_slaver._ctx, RS485RxLenth, true,
                                                                agile_modbus_slave_util_callback, &slave_util, NULL);
        if (SlaverResponseBufLength > 0) {
            _printfh(SlaverResponseBuff, SlaverResponseBufLength);
        }
    }
}

### RT-Thread Agile Modbus Slave 实现与配置 #### 添加 RT-Thread Nano 软件包 为了获取 RT-Thread Nano 的软件包,在 STM32CubeMX 中需添加 URL `https://www.rt-thread.org/download/cube/RealThread.RT-Thread.pdsc`[^1]。 #### 移植 RT-ThreadAgile Modbus 协议栈 对于基于 RT-Thread Nano 框架并使用 AHT20 温湿度传感器构建 MODBUS 接口的 Slave 设备而言,首要任务是在项目中集成 RT-Thread 并安装 Agile Modbus 协议栈。这通常涉及以下几个方面的工作: - **初始化 RT-Thread**: 配置和编译适合目标硬件平台(如 STM32)的操作系统环境。 - **引入 Agile Modbus 库**: 可以通过包管理工具下载官方支持的 Agile Modbus 组件到工程目录下,并按照文档说明完成必要的设置过程[^2]。 #### 编写主程序逻辑 一旦上述准备工作就绪,则可以着手编写应用程序代码来处理来自 Master 设备的数据请求以及向其发送响应数据。具体来说就是实现如下功能: - 定义用于存储温度和湿度测量值的全局变量; - 创建一个定时器线程周期性地调用 API 函数读取当前环境参数; - 注册回调函数监听特定寄存器地址范围内的访问事件;当检测到来自客户端的应用层指令时触发相应的动作,例如更新共享内存中的数值或将最新采集的结果打包成标准帧格式返回给对方。 ```c #include "rtthread.h" #include "agile_modbus_slave.h" // 假设已经定义好了AHT20驱动接口 extern float get_temperature(void); extern float get_humidity(void); static struct rt_semaphore sem; void modbus_task_entry(void *parameter) { while (1) { /* 等待信号量 */ rt_sem_take(&sem, RT_WAITING_FOREVER); // 处理MODBUS命令... /* 发送应答消息 */ agile_modbus_send_response(); } } int main(void) { // 初始化信号量 rt_sem_init(&sem, "modbus", 0, RT_IPC_FLAG_FIFO); // 启动MODBUS服务端线程 rt_thread_t tid = rt_thread_create("mbus", modbus_task_entry, RT_NULL, 1024, 25, 10); if (tid != RT_NULL) rt_thread_startup(tid); return 0; } ``` 此段伪代码展示了如何创建一个新的后台工作进程专门负责接收解析外部传入的消息体内容,并依据实际需求执行对应操作后再反馈回去。注意这里仅提供了一个简化版的例子供参考学习之用,真实场景下的业务流程可能会更加复杂一些。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值