|
||
---|---|---|
.. | ||
inc | ||
master | ||
public/img | ||
slave | ||
test | ||
Makefile | ||
hart.c | ||
hart_cache.c | ||
hart_frame.c | ||
readme.md |
readme.md
介绍
HART 协议模块是 C 语言编写的,使用者无需对 HART 协议了解,通过接口文档可以快速完成应用层协议开发。 模块可以分别被主机和从机使用,调用少量接口完成开发,模块接口测试覆盖率 100%。
架构

架构说明
HART 协议模块是设备与主机之间的通信协议,主机和从机之间通过串口通信,主机通过串口发送命令,从机接收命令并返回响应数据。因此,主机和从机都需要使用 HART 协议模块,主机和从机的接口不同,主机接口主要是发送命令,从机接口主要是接收命令并返回响应数据。
考虑到移植性在设计阶段模块因避免使用操作系统相关的接口,模块接口使用回调函数的方式实现,主机和从机需要实现不同的回调函数,主机需要实现发送数据的回调函数,从机需要实现接收数据的回调函数。
内容说明
- 7.9 版本认证支持命令 78、79、534
- safeHART 数字化安全连锁,一种通讯协议,主要是对数据部分增加了 CRC 校验和序列号,打开 Burst-Mode 后发送的命令发送的是 safeHART 命令。 https://library.fieldcommgroup.org/20085/TS20085/4.0/#page=36
- 2024 年 3 月 8 日后提交测试必须是 3.8 版本测试
- 新增 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
- hart_init 函数:用于初始化 Hart 模块,根据传入的参数配置模块的工作模式(主模块或从模块)和接口。
- hart_handle 函数:用于处理接收到的数据,根据模块的工作模式将数据分发给对应的主机或从机处理。
- hart_storage_write 和 hart_storage_read 函数:用于实现 Flash 的读写操作。
- perform_self_test 和 perform_self_test_finish 函数:用于自检设备。
- device_reset 函数:用于设备复位。
- squawk_control 函数:用于呼叫,根据传入的参数打开或关闭呼叫功能。
- armed 函数:用于判断是否有技术人员按下一个特殊的按钮或按钮组合。
- common_event 函数:用于处理用户自定义的事件。
hart_cache.c
- 全局变量:sqqueue_ctrl_t cache,这是一个队列控制结构体,用于控制缓存队列的属性。
- 缓存消息结构体:hart_cache_t,这个结构体包含了一个 uuid 字段,用于存储消息的唯一标识。还有其他一些与具体设备相关的字段,例如设备状态、操作结果等。
- hart_cache_init(),用于初始化缓存队列。
- hart_cache_add(),用于向缓存中添加一条消息。首先检查缓存是否已满,如果满则返回。然后分配内存,拷贝数据,并设置 uuid。最后将数据入队。
- hart_cache_free(),用于释放缓存中一条消息的内存。
- hart_cache_get(),用于根据 uuid 获取缓存中一条消息。首先检查缓存是否为空,如果为空则返回 NULL。然后遍历缓存中的消息,如果找到匹配的 uuid,则返回该消息。否则将当前消息出队,然后重新入队。
hart_frame.c
- hart_frame_data_length_start 和 hart_frame_data_length_end 函数用于在处理请求开始和结束时记录帧数据长度位置。
- hart_frame_response_code_start 和 hart_frame_response_code_end 函数用于在处理请求开始和结束时记录帧响应码起始位置。
- hart_frame_response_communication_code 和 hart_frame_master_response_operate_code 函数用于设置和获取响应码第一个字节和第二个字节。
- hart_is_write_command 函数用于判断指令是否是写。
- hart_is_support_command 函数用于判断版本是否支持当前指令。
- rtc_to_timestamp 函数用于将 RTC 时间转时间戳。
- get_rtc_date 函数用于获取 RTC 的日期。
- check_date 函数用于检测日期是否有效。
- is_broadcast_address 函数用于判断地址是否为广播地址。
- timestamp_to_hmsms 函数用于将时间戳转换为小时:分钟:秒:毫秒的格式。
hart_slave.c
-
hart_slave_state_check 来检查命令是否合法。如果命令合法,它将调用相应的处理函数 hart_command_ptr_arr[command]来处理命令。如果处理成功,它将返回 True,否则返回 False。 在处理完命令后,函数将响应数据的长度存储在 handle->response->cache_data.payload_length 中。如果响应数据长度不为 0,它将分配内存来存储响应数据,并将数据复制到分配的内存中。最后,它将响应数据结构 handle->response->cache_data 添加到缓存中,并发送响应数据。
-
hart_slave_handle 函数,该函数用于处理从机的请求。
- 首先,检查输入参数是否有效。
- 获取从机的地址。
- 比较主机请求的地址和从机地址是否一致。如果不一致,返回错误。
- 获取命令码。
- 根据命令码查找对应的处理函数。如果找不到对应的处理函数,返回错误。
- 执行处理函数,并将结果存储在 handle->response 结构体中。
- 计算响应数据的长度。
- 如果响应数据长度不为 0,为响应数据分配内存。
- 将响应数据结构体添加到缓存中。
- 返回处理成功。
- hart_slave_init 函数,用于初始化从机。
- 检查输入参数是否有效。
- 初始化从机的请求处理结构体。
- 返回处理成功
- hart_slave_device_send 函数,用于发送响应帧给一个 HART 从设备
- 定义一些变量,如响应数据长度、异或校验、数据长度指针、响应码指针等。
- 填充响应帧的数据,包括前导码、地址、命令、负载等。
- 计算响应帧的数据长度,并将结果存储在 data_len_p 中。
- 计算响应帧的校验码,并将结果存储在 response_code_p 中。
- 更新响应帧的长度字段和响应码字段。
- 如果响应帧的数据长度大于 0,调用 handle->response_call 函数将响应帧发送给指定的 UART 索引。同时,增加设备属性的消息计数器。
- hart_slave_device_attribute_init 函数,用于从机设备属性初始化
hart_slave_req.c
- hart_slave_req_init 函数,用于初始化哈希表,用于注册从端发送给主端的命令的处理函数。
- hart_command_ptr_arr 数组,用于存储命令码和处理函数的映射关系。因为命令码的使用不是连续的,因此占用了大量的空间,但是提高了查找效率。
hart_slave_req_user.c
实现方式上同 hart_slave_req.c,和通用命令做区分
移植步骤
这里以 STM32 为例,介绍如何移植 HART 协议模块。
-
将 HART 协议模块的源码添加到工程中。 HART 协议模块中用到的 lib 也一并加入到工程中。
-
添加宏定义和头文件 宏定义:STM32,SLAVE
头文件路径
-
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);
}