# 介绍 HART 协议模块是 C 语言编写的,使用者无需对 HART 协议了解,通过接口文档可以快速完成应用层协议开发。 模块可以分别被主机和从机使用,调用少量接口完成开发,模块接口测试覆盖率 100%。 # 架构 ## 架构说明 HART 协议模块是设备与主机之间的通信协议,主机和从机之间通过串口通信,主机通过串口发送命令,从机接收命令并返回响应数据。因此,主机和从机都需要使用 HART 协议模块,主机和从机的接口不同,主机接口主要是发送命令,从机接口主要是接收命令并返回响应数据。 考虑到移植性在设计阶段模块因避免使用操作系统相关的接口,模块接口使用回调函数的方式实现,主机和从机需要实现不同的回调函数,主机需要实现发送数据的回调函数,从机需要实现接收数据的回调函数。 # 内容说明 1. 7.9 版本认证支持命令 78、79、534 2. safeHART 数字化安全连锁,一种通讯协议,主要是对数据部分增加了 CRC 校验和序列号,打开 Burst-Mode 后发送的命令发送的是 safeHART 命令。 https://library.fieldcommgroup.org/20085/TS20085/4.0/#page=36 3. 2024 年 3 月 8 日后提交测试必须是 3.8 版本测试 4. 新增 548-553 指令 # 软件设计 ## 结构图 │ hart.c │ hart_cache.c │ hart_frame.c │ readme.md ├─inc │ hart.h │ hart_common_tables_specification.h │ hart_frame.h │ hart_frame_user.h │ ├─lib │ ├─flow │ │ example.c │ │ flow.h │ │ flow_core.c │ │ flow_core.h │ │ flow_def.h │ │ flow_sem.h │ │ README.md │ │ │ ├─inc │ │ data_type_def.h │ │ debug.h │ │ lib.h │ │ log.h │ │ malloc.h │ │ osel_arch.h │ │ sqqueue.h │ │ │ └─src │ debug.c │ lib.c │ malloc.c │ sqqueue.c │ ├─master │ ├─inc │ │ hart_master.h │ │ hart_master_frame.h │ │ hart_master_req.h │ │ hart_master_rsp.h │ │ │ └─src │ hart_master.c │ hart_master_frame.c │ hart_master_req.c │ hart_master_req_user.c │ hart_master_rsp.c │ ├─public │ └─slave ├─inc │ hart_slave.h │ hart_slave_frame.h │ hart_slave_req.h │ └─src hart_slave.c hart_slave_frame.c hart_slave_req.c hart_slave_req_user.c ## 组织结构 ## 模块说明 | 文件 | 路径 |
说明
| | :-------------------- | :------ | :-------------------------------------------------------------------------------------------------------------------------------------- | | hart.c | ./ | 用于处理 HART(高阶实时通信技术)模块的接口。这个文件包含了模块的初始化、数据处理、Flash 读写等函数,以及一个用于处理用户事件的回调函数 | | hart_cache.c | ./ | 一个缓存消息模块,用于处理设备操作需要时间暂时不回复消息的情况 | | hart_frame.c | ./ | 包含了用于处理帧数据结构、响应码、RTC 时间转换、日期检测等功能 | | hart_slave.c | ./slave | Hart 从机处理函数,用于处理主机的请求并返回相应的响应 | | hart_slave_req.c | ./slave | 从机命令响应的处理函数 | | hart_slave_req_user.c | ./slave | 从机命令响应用户自定义指令的处理函数 | > 主机模块说明暂时省略 ### hart.c 1. hart_init 函数:用于初始化 Hart 模块,根据传入的参数配置模块的工作模式(主模块或从模块)和接口。 2. hart_handle 函数:用于处理接收到的数据,根据模块的工作模式将数据分发给对应的主机或从机处理。 3. hart_storage_write 和 hart_storage_read 函数:用于实现 Flash 的读写操作。 4. perform_self_test 和 perform_self_test_finish 函数:用于自检设备。 5. device_reset 函数:用于设备复位。 6. squawk_control 函数:用于呼叫,根据传入的参数打开或关闭呼叫功能。 7. armed 函数:用于判断是否有技术人员按下一个特殊的按钮或按钮组合。 8. common_event 函数:用于处理用户自定义的事件。 ### hart_cache.c 1. 全局变量:sqqueue_ctrl_t cache,这是一个队列控制结构体,用于控制缓存队列的属性。 2. 缓存消息结构体:hart_cache_t,这个结构体包含了一个 uuid 字段,用于存储消息的唯一标识。还有其他一些与具体设备相关的字段,例如设备状态、操作结果等。 3. hart_cache_init(),用于初始化缓存队列。 4. hart_cache_add(),用于向缓存中添加一条消息。首先检查缓存是否已满,如果满则返回。然后分配内存,拷贝数据,并设置 uuid。最后将数据入队。 5. hart_cache_free(),用于释放缓存中一条消息的内存。 6. hart_cache_get(),用于根据 uuid 获取缓存中一条消息。首先检查缓存是否为空,如果为空则返回 NULL。然后遍历缓存中的消息,如果找到匹配的 uuid,则返回该消息。否则将当前消息出队,然后重新入队。 ### hart_frame.c 1. hart_frame_data_length_start 和 hart_frame_data_length_end 函数用于在处理请求开始和结束时记录帧数据长度位置。 2. hart_frame_response_code_start 和 hart_frame_response_code_end 函数用于在处理请求开始和结束时记录帧响应码起始位置。 3. hart_frame_response_communication_code 和 hart_frame_master_response_operate_code 函数用于设置和获取响应码第一个字节和第二个字节。 4. hart_is_write_command 函数用于判断指令是否是写。 5. hart_is_support_command 函数用于判断版本是否支持当前指令。 6. rtc_to_timestamp 函数用于将 RTC 时间转时间戳。 7. get_rtc_date 函数用于获取 RTC 的日期。 8. check_date 函数用于检测日期是否有效。 9. is_broadcast_address 函数用于判断地址是否为广播地址。 10. timestamp_to_hmsms 函数用于将时间戳转换为小时:分钟:秒:毫秒的格式。 ### hart_slave.c 1. hart_slave_state_check 来检查命令是否合法。如果命令合法,它将调用相应的处理函数 hart_command_ptr_arr[command]来处理命令。如果处理成功,它将返回 True,否则返回 False。 在处理完命令后,函数将响应数据的长度存储在 handle->response->cache_data.payload_length 中。如果响应数据长度不为 0,它将分配内存来存储响应数据,并将数据复制到分配的内存中。最后,它将响应数据结构 handle->response->cache_data 添加到缓存中,并发送响应数据。 2. hart_slave_handle 函数,该函数用于处理从机的请求。 - 首先,检查输入参数是否有效。 - 获取从机的地址。 - 比较主机请求的地址和从机地址是否一致。如果不一致,返回错误。 - 获取命令码。 - 根据命令码查找对应的处理函数。如果找不到对应的处理函数,返回错误。 - 执行处理函数,并将结果存储在 handle->response 结构体中。 - 计算响应数据的长度。 - 如果响应数据长度不为 0,为响应数据分配内存。 - 将响应数据结构体添加到缓存中。 - 返回处理成功。 3. hart_slave_init 函数,用于初始化从机。 - 检查输入参数是否有效。 - 初始化从机的请求处理结构体。 - 返回处理成功 4. hart_slave_device_send 函数,用于发送响应帧给一个 HART 从设备 - 定义一些变量,如响应数据长度、异或校验、数据长度指针、响应码指针等。 - 填充响应帧的数据,包括前导码、地址、命令、负载等。 - 计算响应帧的数据长度,并将结果存储在 data_len_p 中。 - 计算响应帧的校验码,并将结果存储在 response_code_p 中。 - 更新响应帧的长度字段和响应码字段。 - 如果响应帧的数据长度大于 0,调用 handle->response_call 函数将响应帧发送给指定的 UART 索引。同时,增加设备属性的消息计数器。 5. hart_slave_device_attribute_init 函数,用于从机设备属性初始化 ### hart_slave_req.c 1. hart_slave_req_init 函数,用于初始化哈希表,用于注册从端发送给主端的命令的处理函数。 - hart_command_ptr_arr 数组,用于存储命令码和处理函数的映射关系。因为命令码的使用不是连续的,因此占用了大量的空间,但是提高了查找效率。 ### hart_slave_req_user.c 实现方式上同 hart_slave_req.c,和通用命令做区分 # 移植步骤 这里以 STM32 为例,介绍如何移植 HART 协议模块。 1. 将 HART 协议模块的源码添加到工程中。 HART 协议模块中用到的 lib 也一并加入到工程中。 2. 添加宏定义和头文件 宏定义:STM32,SLAVE 头文件路径 3. HART 协议和应用层的中间层 该文件主要实现 HART 模块需要的和 MCU 相关的接口,例如串口发送、Flash 读写等。 hart_user_data_refush(): 刷新用户实时数据,包括模拟量和数字量。 response(uint8_t uart_index, uint8_t *data, uint16_t len): 发送数据(hart 协议数据帧)。 flash_read(uint32_t addr, uint8_t *data, uint16_t len): flash 读取接口。 flash_write(uint32_t addr, uint8_t *data, uint16_t len): flash 写入接口。 perform_self_test(void): 执行自检。 device_reset(void): 设备复位。 squawk_control(BOOL open, uint8_t second): 设备呼叫,0 一直呼叫,1-255 秒。 armed(void): 技术人员按下一个特殊的按钮或按钮组合,指示从机应响应 command74。 set_dynamics(device_variable_dynamics_t *const dynamics): 设置动态变量。 hart_rx_cb(uint8_t uart_index, uint8_t \*data, uint16_t len): 串口 1、5 接收中断回调函数。 hart_tx_complete_cb(void): 串口 1 发送完成回调函数。 hart_uart_init(void): 串口 1 初始化。 hart_uart_dinit(void): 串口 1 去初始化。 hart_ble_init(void): BLE 初始化。 hart_ble_dinit(void): BLE 去初始化。 hart_init(void): HART 初始化。 hart_dinit(void): HART 去初始化。 hart_task(void): HART 任务。 源码实现: ``` /* * @Author: xxx * @Date: 2023-08-02 08:28:36 * @LastEditors: xxx * @LastEditTime: 2023-08-23 14:15:44 * @Description: 此文件主要实现板卡的HART功能 * Copyright (c) 2023 by xxx, All Rights Reserved. */ #include "app.h" #include "hart.h" #include "hart_frame.h" #include "uarts.h" #include "test_bsp.h" #define HART_UART1 USART1 #define HART_UART2 USART5 #define UART_RXSIZE (240u) #define UART_TXSIZE (240u) uart_t *uarts[APP_UART_MAX]; app_dynamics_t app_dynamics; static __IO BOOL hart_idle = TRUE; static void hart_user_data_refush(void) { app_analog_quantity_t *analog_quantity = &app_dynamics.analog_quantity; app_digital_quantity_t *digital_quantity = &app_dynamics.digital_quantity; // 模拟量 analog_quantity->input_current = adc_raw[ADC_LOOP_CHANNEL]; // 输入电流 analog_quantity->valve_feedback = adc_raw[ADC_PSB_CHANNEL]; // 阀门反馈 analog_quantity->atmospheric_pressure_source = adc_raw[ADC_BP_CHANNEL]; // 气压源压力 analog_quantity->pressure_at_port_a = 0; // A口压力 analog_quantity->pressure_at_port_b = 0; // B口压力 analog_quantity->amplifier_circuit = adc_raw[ADC_IPSB_CHANNEL]; // 放大器回路 analog_quantity->built_in_temperature = 0; // 内置温度 analog_quantity->ip_output_detection = rt_data.ip_output; // IP输出检测 analog_quantity->valve_percentage = actual_travel; // 阀位百分比 // 数字量 digital_quantity->input_current = loop_current; // 输入电流 digital_quantity->valve_feedback = adc_raw[ADC_PSB_CHANNEL]; // 阀门反馈 digital_quantity->atmospheric_pressure_source = adc_raw[ADC_BP_CHANNEL]; // 气压源压力 digital_quantity->pressure_at_port_a = 0; // A口压力 digital_quantity->pressure_at_port_b = 0; // B口压力 digital_quantity->amplifier_circuit = adc_raw[ADC_IPSB_CHANNEL]; // 放大器回路 digital_quantity->built_in_temperature = 0; // 内置温度 digital_quantity->ip_output_detection = rt_data.ip_output; // IP输出检测 digital_quantity->target_row = target_travel; // 目标行程 digital_quantity->current_row = actual_travel; // 实际行程 digital_quantity->friction = 0; // 摩擦力 digital_quantity->spring_force = 0; // 弹簧力 } /** * @brief 发送数据(hart协议数据帧) * @param {uint8_t} *txBuf * @param {uint16_t} len * @return {*} */ static void response(uint8_t uart_index, uint8_t *data, uint16_t len) { #ifdef STM32 if (uart_index == APP_UART_1) { HART_CD_OFF(); // 因为DMA发送调用uart_send_data接口后会立即退出,不需要等待发送是否成功,因此HART_CD_ON需要在DMA发送完成中断中调用 } uart_send_data(uarts[uart_index], data, len); #else LOG_HEX(data, len); #endif } /** * @brief flash读取接口 * @param {uint32_t} addr * @param {uint8_t} *data * @param {uint16_t} len * @return {*} */ static BOOL flash_read(uint32_t addr, uint8_t *data, uint16_t len) { m95_1_normal_read(addr, data, len); return TRUE; } /** * @brief flash写入接口 * @param {uint32_t} addr * @param {uint8_t} *data * @param {uint16_t} len * @return {*} */ static BOOL flash_write(uint32_t addr, uint8_t *data, uint16_t len) { m95_1_normal_write(addr, data, len); return TRUE; } /** * @brief 执行自检 * @param {*} * @return {*} */ static void perform_self_test(void) { } /** * @brief 设备复位 * @param {*} * @return {*} */ static void device_reset(void) { } /** * @brief 设备呼叫,0一直呼叫,1-255秒 * @param {*} * @return {*} */ static void squawk_control(BOOL open, uint8_t second) { } /** * @brief 技术人员按下一个特殊的按钮或按钮组合,指示从机应响应command74 * @param {*} * @return {*} */ static BOOL armed(void) { return TRUE; } /** * @brief 设置动态变量 * @return {*} */ static BOOL set_dynamics(device_variable_dynamics_t *const dynamics) { app_analog_quantity_t *analog_quantity = &app_dynamics.analog_quantity; app_digital_quantity_t *digital_quantity = &app_dynamics.digital_quantity; // 模拟量 dynamics->dynamics_user.analog_quantity.input_current = &analog_quantity->input_current; // 输入电流 dynamics->dynamics_user.analog_quantity.valve_feedback = &analog_quantity->valve_feedback; // 阀门反馈 dynamics->dynamics_user.analog_quantity.atmospheric_pressure_source = &analog_quantity->atmospheric_pressure_source; // 大气压力源 dynamics->dynamics_user.analog_quantity.pressure_at_port_a = &analog_quantity->pressure_at_port_a; // A口压力 dynamics->dynamics_user.analog_quantity.pressure_at_port_b = &analog_quantity->pressure_at_port_b; // B口压力 dynamics->dynamics_user.analog_quantity.amplifier_circuit = &analog_quantity->amplifier_circuit; // 放大器电路 dynamics->dynamics_user.analog_quantity.built_in_temperature = &analog_quantity->built_in_temperature; // 内置温度 dynamics->dynamics_user.analog_quantity.ip_output_detection = &analog_quantity->ip_output_detection; // IP输出检测 dynamics->dynamics_user.analog_quantity.valve_percentage = &analog_quantity->valve_percentage; // 阀位百分比 // 数字量 dynamics->dynamics_user.digital_quantity.input_current = &digital_quantity->input_current; // 输入电流 dynamics->dynamics_user.digital_quantity.valve_feedback = &digital_quantity->valve_feedback; // 阀门反馈 dynamics->dynamics_user.digital_quantity.atmospheric_pressure_source = &digital_quantity->atmospheric_pressure_source; // 大气压力源 dynamics->dynamics_user.digital_quantity.pressure_at_port_a = &digital_quantity->pressure_at_port_a; // A口压力 dynamics->dynamics_user.digital_quantity.pressure_at_port_b = &digital_quantity->pressure_at_port_b; // B口压力 dynamics->dynamics_user.digital_quantity.amplifier_circuit = &digital_quantity->amplifier_circuit; // 放大器电路 dynamics->dynamics_user.digital_quantity.built_in_temperature = &digital_quantity->built_in_temperature; // 内置温度 dynamics->dynamics_user.digital_quantity.ip_output_detection = &digital_quantity->ip_output_detection; // IP输出检测 dynamics->dynamics_user.digital_quantity.target_row = &digital_quantity->target_row; // 目标行程 dynamics->dynamics_user.digital_quantity.current_row = &digital_quantity->current_row; // 当前行程 dynamics->dynamics_user.digital_quantity.friction = &digital_quantity->friction; // 摩擦力 dynamics->dynamics_user.digital_quantity.spring_force = &digital_quantity->spring_force; // 弹簧力 return TRUE; } static BOOL common_event(hart_interface_user_event_e event, const void *const data) { switch (event) { case HART_COMMAND_257_EVENT: // 寻找、计算P / I / D break; case HART_COMMAND_258_EVENT: // 计算 摩擦力 break; case HART_COMMAND_259_EVENT: // 计算 弹簧力 break; case HART_COMMAND_UPDATE_EVENT: // 刷新用户实时数据 hart_user_data_refush(); break; case HART_COMMAND_500_EVENT: // 测试命令 return hart_user_test((hart_user_req_t *)data); default: return FALSE; } return TRUE; } // 串口1、5接收中断回调函数 static void hart_rx_cb(uint8_t uart_index, uint8_t *data, uint16_t len) { DBG_ASSERT(uart_index < APP_UART_MAX __DBG_LINE); hart_idle = FALSE; // 串口1接收HART数据,因为是空闲中断处理这里只过滤掉前导码 hart_handle(uart_index, data, len); hart_idle = TRUE; } static void hart_tx_complete_cb(void) { // 串口1发送完成回调函数 HART_CD_ON(); } void hart_uart_init(void) { LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_USART1); GPIO_SET_OUTPUT(HART_EN_GPIO_Port, HART_EN_Pin); GPIO_SET_OUTPUT(HART_RST_GPIO_Port, HART_RST_Pin); GPIO_SET_OUTPUT(HART_CD_GPIO_Port, HART_CD_Pin); GPIO_SET_ALTERNATE(HART_TX_GPIO_Port, HART_TX_Pin); GPIO_SET_ALTERNATE(HART_RX_GPIO_Port, HART_RX_Pin); delay_ms(10); // 串口1初始化开始 HART_CD_ON(); HART_RST_OFF(); delay_ms(20); HART_RST_ON(); HART_EN_DISABLE(); HART_EN_ENABLE(); HART_CD_ON(); if (uarts[APP_UART_1] == NULL) { uarts[APP_UART_1] = uart_create(HART_UART1, TRUE, UART_RXSIZE, hart_rx_cb, TRUE, UART_TXSIZE, hart_tx_complete_cb); uarts[APP_UART_1]->uart_index = APP_UART_1; uarts[APP_UART_1]->dma = DMA1; uarts[APP_UART_1]->dma_rx_channel = LL_DMA_CHANNEL_3; uarts[APP_UART_1]->dma_tx_channel = LL_DMA_CHANNEL_2; uart_recv_en(uarts[APP_UART_1]); } // 串口1初始化结束 } void hart_uart_dinit(void) { HART_EN_DISABLE(); LL_APB2_GRP1_DisableClock(LL_APB2_GRP1_PERIPH_USART1); LL_USART_Disable(USART1); GPIO_SET_ANALOG(HART_EN_GPIO_Port, HART_EN_Pin); GPIO_SET_ANALOG(HART_RST_GPIO_Port, HART_RST_Pin); GPIO_SET_ANALOG(HART_CD_GPIO_Port, HART_CD_Pin); GPIO_SET_ANALOG(HART_TX_GPIO_Port, HART_TX_Pin); GPIO_SET_ANALOG(HART_RX_GPIO_Port, HART_RX_Pin); } void hart_ble_init(void) { LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_USART5); GPIO_SET_OUTPUT(BLE_PWR_GPIO_Port, BLE_PWR_Pin); GPIO_SET_INPUT(BLE_STATE_GPIO_Port, BLE_STATE_Pin); GPIO_SET_ALTERNATE(BLE_TX_GPIO_Port, BLE_TX_Pin); GPIO_SET_ALTERNATE(BLE_RX_GPIO_Port, BLE_RX_Pin); BLE_EN_ENABLE(); delay_ms(100); if (uarts[APP_UART_2] == NULL) { uarts[APP_UART_2] = uart_create(HART_UART2, TRUE, UART_RXSIZE, hart_rx_cb, TRUE, UART_TXSIZE, NULL); uarts[APP_UART_2]->uart_index = APP_UART_2; uarts[APP_UART_2]->dma = DMA1; uarts[APP_UART_2]->dma_rx_channel = LL_DMA_CHANNEL_6; uarts[APP_UART_2]->dma_tx_channel = LL_DMA_CHANNEL_7; uart_recv_en(uarts[APP_UART_2]); } } void hart_ble_dinit(void) { BLE_EN_DISABLE(); LL_APB1_GRP1_DisableClock(LL_APB1_GRP1_PERIPH_USART5); LL_USART_Disable(USART5); GPIO_SET_ANALOG(BLE_PWR_GPIO_Port, BLE_PWR_Pin); GPIO_SET_ANALOG(BLE_STATE_GPIO_Port, BLE_STATE_Pin); GPIO_SET_ANALOG(BLE_TX_GPIO_Port, BLE_TX_Pin); GPIO_SET_ANALOG(BLE_RX_GPIO_Port, BLE_RX_Pin); } BOOL app_hart_is_idle(void) { return hart_idle; } BOOL app_hart_init(void) { hart_init_t init; init.hart_protocol_version = HART_PROTOCOL_VERSION_7; init.dir = MODULE_SLAVE; init.interface.response = response; init.interface.flash_read = flash_read; init.interface.flash_write = flash_write; init.interface.perform_self_test = perform_self_test; init.interface.device_reset = device_reset; init.interface.squawk_control = squawk_control; init.interface.armed = armed; init.interface.set_dynamics = set_dynamics; init.interface.common_event = common_event; uarts[APP_UART_1] = NULL; uarts[APP_UART_2] = NULL; // 判断当前电流 >=8mA启动蓝牙 >=3.8mA启动uart // 注:串口初始化移动到all_flow // hart_uart_init(); // hart_ble_init(); return hart_init(&init); } ```