/** * @file command.c * @brief 串口命令解析与处理模块实现 * @details 实现基于状态机的协议解析器,支持 D5 03 LEN [cmd] CRC 格式的命令处理, * 包含命令帧解析、响应生成和传感器状态管理功能。 * @author Hulk * @date 2025-08-13 * @version 1.0.0 * @ingroup Command */ #include "command.h" #include "uart_ring_buffer.h" #include "uart.h" #include "led.h" #include #include #include #include #include "board_config.h" #include "gd32e23x_usart.h" #include "i2c.h" #include "core_cm23.h" /* ============================================================================ * 协议格式说明 * ============================================================================ */ /** * @name 协议帧格式 * @{ * @details * Host -> Device 命令帧格式: * [0] HEADER = 0xD5 // 包头标识 * [1] BOARD_TYPE = 0x01 // 板卡类型标识 * [2] LEN = 数据区字节数 // 有效载荷长度 * [3..(3+LEN-1)] 数据 // 命令数据,如 "M1", "M2S123" * [last] CRC = 校验码 // 从索引1到(last-1)的累加和低8位 * * 最小协议包长度为 6 字节 * 数据示例(两字节命令):"M1" / "M2" / "M3" * * Device -> Host 响应帧格式: * [0] 0xB5 // 响应包头 * [1] TYPE // 响应类型(0xF0=成功,0xF1..=错误类型) * [2] LEN // 响应数据长度 * [3..(3+LEN-1)] 数据 // 响应数据,如 "ok", "err" * [last] CRC // 校验码(同命令帧规则) * @} */ /* ============================================================================ * 协议常量定义 * ============================================================================ */ /** @name 协议帧标识符 * @{ */ #define PROTOCOL_PACKAGE_HEADER 0xD5 /**< 命令帧包头标识 */ /** @} */ /** @name 命令长度限制 * @{ */ #define COMMAND_MIN_LEN 2 /**< 最小命令长度,如"M1" */ #define PROTOCOL_MIN_FRAME_LEN (3 + COMMAND_MIN_LEN + 1) /**< 最小完整帧长度:header+type+len+payload+crc = 6 */ #define PROTOCOL_MAX_FRAME_LEN 32 /**< 最大完整帧长度 */ /** @} */ /** @name 响应帧标识符 * @{ */ #define RESP_HEADER 0xB5 /**< 响应帧包头标识 >**/ #define RESP_TYPE_OK 0xF0 /**< 成功响应类型 >**/ #define RESP_TYPE_CRC_ERR 0xF1 /**< CRC校验错误 >**/ #define RESP_TYPE_HEADER_ERR 0xF2 /**< 包头错误 >**/ #define RESP_TYPE_TYPE_ERR 0xF3 /**< 板类型错误 >**/ #define RESP_TYPE_LEN_ERR 0xF4 /**< 长度错误 >**/ #define RESP_TYPE_PARAM_ERR 0xFD /**< 参数错误 >**/ #define RESP_TYPE_CMD_ERR 0xFE /**< 命令错误 >**/ #define RESP_TYPE_ERR 0xFF /**< 通用错误 >**/ /** @} */ /* ============================================================================ * 模块内部变量 * ============================================================================ */ /** @name 预设响应数据 * @{ */ static const uint8_t s_report_status_ok[] __attribute__((unused)) = { 'o', 'k' }; /**< 成功响应数据 */ static const uint8_t s_report_status_err[] = { 'e','r','r' }; /**< 错误响应数据 */ /** @} */ void system_software_reset(void) { // 确保所有待处理的内存访问完成 __DSB(); // 执行系统复位 NVIC_SystemReset(); // 以下代码不会执行 while(1); } /* Debug output control */ #ifdef COM_DEBUG #include #define COMMAND_DEBUG(fmt, ...) printf("[COMMAND] " fmt "\n", ##__VA_ARGS__) #else #define COMMAND_DEBUG(fmt, ...) #endif /* ============================================================================ * 公共接口函数 * ============================================================================ */ /** * @brief 计算协议包的 8 位累加校验值(Checksum)。 * @details 对输入缓冲区逐字节累加并取低 8 位,累加范围为 data[1] 至 data[len-2], * 即不包含包头 HEADER(索引 0)与尾部 CRC 字节(索引 len-1)。 * 当 len 小于最小协议帧长度(PACKAGE_MIN_LENGTH)时返回 0。 * @param data 指向待校验的完整协议包缓冲区。 * @param len 缓冲区总长度(字节),应满足 header + type + len + payload + crc 的最小格式。 * @return uint8_t 计算得到的 8 位校验值。 * @note 本函数实现为简单求和校验(Checksum),非多项式 CRC;与本协议“从索引 1 累加到 len-2”的规则一致。 * @ingroup Command */ static uint8_t command_sum_crc_calc(const uint8_t *data, uint8_t len) { uint16_t crc = 0; // 仅在满足协议最小帧长时计算(header + type + len + payload + crc) if (len < PROTOCOL_MIN_FRAME_LEN) return 0; // 累加从索引 1 到 len-2 的字节(不含 header 和 crc 字节) for (uint8_t i = 1; i < (len - 1); i++) { crc += data[i]; } return (uint8_t)(crc & 0xFF); } /** * @brief 发送协议响应帧(使用GD32E230标准库)。 * @details 构造并发送格式为 B5 TYPE LEN [payload] CRC 的响应帧, * 自动计算CRC校验值并通过串口输出。 * @param type 响应类型码(如 RESP_TYPE_OK, RESP_TYPE_CRC_ERR 等)。 * @param payload 指向响应数据的缓冲区,当len为0时可为NULL。 * @param len 响应数据长度(字节),为0时不复制payload数据。 * @note 内部使用固定大小缓冲区,超长响应将被丢弃。 * @warning 使用GD32E230标准库函数发送,确保串口已正确初始化。 * @ingroup Command */ static void send_response(uint8_t type, const uint8_t *payload, uint8_t len) { uint8_t buf_len = (uint8_t)(3 + len + 1); uint8_t buf[16]; // 简单场景足够,必要时可增大 if (buf_len > sizeof(buf)) return; // 防御 buf[0] = RESP_HEADER; buf[1] = type; buf[2] = len; // 简化逻辑:只有当len > 0且payload非空时才复制数据 if (len > 0 && payload != NULL) { for (uint8_t i = 0; i < len; i++) { buf[3 + i] = payload[i]; } } buf[buf_len - 1] = command_sum_crc_calc(buf, buf_len); // 使用GD32E230标准库函数逐字节发送(标准库实现) for (uint8_t i = 0; i < buf_len; i++) { // 等待发送缓冲区空 while (usart_flag_get(UART_PHY, USART_FLAG_TBE) == RESET) {} usart_data_transmit(UART_PHY, buf[i]); } // 等待发送完成 while (usart_flag_get(UART_PHY, USART_FLAG_TC) == RESET) {} // // 使用printf发送(通过重定向到串口) // for (uint8_t i = 0; i < buf_len; i++) { // printf("%c", buf[i]); // } // // 刷新缓冲区 // fflush(stdout); } /** * @brief 发送协议响应帧(调试用,发送到DEBUG_UART)。 * @details 构造并发送格式为 B5 TYPE LEN [payload] CRC 的响应帧, * 自动计算CRC校验值并通过DEBUG_UART输出。 * @param type 响应类型码(如 RESP_TYPE_OK, RESP_TYPE_CRC_ERR 等)。 * @param payload 指向响应数据的缓冲区,当len为0时可为NULL。 * @param len 响应数据长度(字节),为0时不复制payload数据。 * @note 内部使用固定大小缓冲区,超长响应将被丢弃。 * @warning 使用GD32E230标准库函数发送,确保DEBUG_UART已正确初始化。 * @ingroup Command */ static void send_response_debug(uint8_t type, const uint8_t *payload, uint8_t len) __attribute__((unused)); static void send_response_debug(uint8_t type, const uint8_t *payload, uint8_t len) { uint8_t buf_len = (uint8_t)(3 + len + 1); uint8_t buf[16]; // 简单场景足够,必要时可增大 if (buf_len > sizeof(buf)) return; // 防御 buf[0] = RESP_HEADER; buf[1] = type; buf[2] = len; // 简化逻辑:只有当len > 0且payload非空时才复制数据 if (len > 0 && payload != NULL) { for (uint8_t i = 0; i < len; i++) { buf[3 + i] = payload[i]; } } buf[buf_len - 1] = command_sum_crc_calc(buf, buf_len); // 使用GD32E230标准库函数逐字节发送(标准库实现) for (uint8_t i = 0; i < buf_len; i++) { // 等待发送缓冲区空 while (usart_flag_get(DEBUG_UART, USART_FLAG_TBE) == RESET) {} usart_data_transmit(DEBUG_UART, buf[i]); } // 发送换行符 \r\n // while (usart_flag_get(DEBUG_UART, USART_FLAG_TBE) == RESET) {} // usart_data_transmit(DEBUG_UART, '\r'); // while (usart_flag_get(DEBUG_UART, USART_FLAG_TBE) == RESET) {} // usart_data_transmit(DEBUG_UART, '\n'); // 等待发送完成 while (usart_flag_get(DEBUG_UART, USART_FLAG_TC) == RESET) {} } /** * @brief 判断字符是否为十进制数字字符。 * @param c 待检查的字符(ASCII码值)。 * @return bool 判断结果。 * @retval true 字符为 '0' 到 '9' 之间的数字字符。 * @retval false 字符不是十进制数字字符。 * @ingroup Command */ static inline bool is_dec_digit(uint8_t c) { return (c >= '0' && c <= '9'); } /** * @brief 将一个无符号整数转换为字符串并追加到缓冲区。 * @param value 要转换的数字。 * @param buffer 指向目标缓冲区的指针,转换后的字符串将写入此处。 * @return uint8_t 写入的字符数。 */ static uint8_t uint_to_str(uint32_t value, char *buffer) { char temp[10]; // 32位无符号整数最多10位 int i = 0; if (value == 0) { buffer[0] = '0'; return 1; } // 将数字逆序转换为字符存入临时数组 while (value > 0) { temp[i++] = (char)((value % 10) + '0'); value /= 10; } // 将逆序的字符串反转并存入目标缓冲区 uint8_t len = (uint8_t)i; for (int j = 0; j < len; j++) { buffer[j] = temp[--i]; } return len; } /** * @brief 将有符号整数转换为字符串 * @param value 要转换的数字 * @param buffer 目标缓冲区 * @return uint8_t 写入的字符数 */ static uint8_t __attribute__((unused)) int_to_str(int32_t value, char *buffer) { uint8_t len = 0; if (value < 0) { buffer[0] = '-'; len++; // 处理最小负数溢出问题 (虽然int16不会溢出int32,但为了健壮性) // 这里直接取反转为正数处理 value = -value; } len += uint_to_str((uint32_t)value, &buffer[len]); return len; } /** * @brief 从缓冲区解析十进制无符号整数。 * @details 从指定位置开始连续读取十进制数字字符,累加构成32位无符号整数。 * 遇到非数字字符或到达长度限制时停止解析。 * @param s 指向待解析字符缓冲区的起始位置。 * @param n 允许解析的最大字符数。 * @param out 输出参数,存储解析结果,可为NULL。 * @return uint8_t 实际消耗的字符数。 * @retval 0 首字符不是数字,解析失败。 * @retval >0 成功解析的数字字符个数。 * @note 不处理符号、空白字符或进制前缀。 * @warning 不进行溢出检查,超出uint32_t范围时按无符号算术溢出处理。 * @ingroup Command */ static uint8_t parse_uint_dec(const uint8_t *s, uint8_t n, uint32_t *out) { uint8_t i = 0; uint32_t v = 0; while (i < n && is_dec_digit(s[i])) { v = v * 10u + (uint32_t)(s[i] - '0'); i++; } if (i == 0) return 0; // 未读到数字 if (out) *out = v; // return i; } /** * @brief 通用T参数解析函数 * @details 解析命令中的T参数(定时参数),格式为T<数字> * @param cmd 指向命令缓冲区 * @param cmd_index 当前解析位置的指针(会被更新) * @param cmd_len 命令总长度 * @param timer_value 输出参数,存储解析到的定时值 * @return bool 解析结果 * @retval true 成功解析到T参数 * @retval false 没有T参数或解析失败 * @note 如果解析成功,cmd_index会被更新到T参数后的位置 * @ingroup Command */ static bool __attribute__((unused)) parse_timer_parameter(const uint8_t *cmd, uint8_t *cmd_index, uint8_t cmd_len, uint32_t *timer_value) { if (*cmd_index >= cmd_len || cmd[*cmd_index] != 'T') { return false; // 没有T参数 } uint8_t temp_index = *cmd_index + 1; // T后的位置 const uint8_t used_timer_cmd = parse_uint_dec(&cmd[temp_index], (uint8_t)(cmd_len - temp_index), timer_value); if (used_timer_cmd == 0) { return false; // T后面没有有效数字 } *cmd_index = (uint8_t)(temp_index + used_timer_cmd); // 更新索引 return true; } /** * @brief 检查命令是否完全解析完毕 * @details 验证命令中是否还有未解析的字符,用于格式验证 * @param cmd_index 当前解析位置 * @param cmd_len 命令总长度 * @return bool 检查结果 * @retval true 命令完全解析完毕 * @retval false 还有未解析的字符 * @ingroup Command */ static bool __attribute__((unused)) is_command_fully_parsed(uint8_t cmd_index, uint8_t cmd_len) { return (cmd_index == cmd_len); } /** * @brief 在命令字符串中查找指定参数的值 * @param cmd 指向命令起始位置(Mxxx 之后) * @param cmd_len 命令剩余长度 * @param key 要查找的参数字符(如 'P', 'S') * @param value 输出参数,存储解析到的值 * @return bool 是否找到该参数 */ static bool __attribute__((unused)) find_parameter_value(const uint8_t *cmd, uint8_t cmd_len, char key, uint32_t *value) { for (uint8_t i = 0; i < cmd_len; i++) { if (cmd[i] == (uint8_t)key) { uint8_t param_idx = i + 1; if (param_idx >= cmd_len) return false; if (parse_uint_dec(&cmd[param_idx], cmd_len - param_idx, value) > 0) { return true; } return false; } } return false; } /* ============================================================================ * 命令处理函数 * ============================================================================ */ /** * @brief 解析并处理完整的命令帧。 * @details 处理经过协议校验的完整命令帧,支持以下命令格式: * - 无参数命令:M<数字>(如 M1、M2、M10、M201) * - 带参数命令:M<数字>S<参数>(如 M100S123,参数为十进制) * * 支持的命令: * - M1: 开启LED,启用传感器上报 * - M2: 关闭LED,禁用传感器上报 * - M100S: 设置PWM值(示例) * * @param frame 指向完整命令帧的缓冲区(从包头0xD5开始)。 * @param len 命令帧总长度(字节)。 * @note 函数内部进行帧格式校验,格式错误时自动发送错误响应。 * @warning 假设输入帧已通过基本协议校验(包头、类型、CRC等)。 * @ingroup Command */ void handle_command(const uint8_t *frame, uint8_t len) { // 帧格式:D5 03 LEN [cmd] CRC; cmd 支持变长,如 "M1"、"M10"、"M201"、"M123S400",有最小长度限制和命令长度校验 uint8_t cmd_len = frame[2]; if (len < PROTOCOL_MIN_FRAME_LEN || (uint8_t)(3 + cmd_len + 1) != len) return; // 长度不匹配或者小于最小限制 const uint8_t *cmd = &frame[3]; // 提取命令部分 // 命令必须以 'M' 开头 if (cmd[0] != 'M'){ send_response(RESP_TYPE_TYPE_ERR, s_report_status_err, sizeof(s_report_status_err)); return; } // 从 'M' 后开始解析 uint8_t cmd_index = 1; // 解析M后的十进制数,即命令本体 uint32_t base_cmd = 0; uint8_t used_base_cmd = parse_uint_dec(&cmd[cmd_index], (cmd_len - cmd_index), &base_cmd); if (used_base_cmd == 0) { // 'M' 后没有数字,格式错误 send_response(RESP_TYPE_LEN_ERR, s_report_status_err, sizeof(s_report_status_err)); return; } cmd_index = (uint8_t)(cmd_index + used_base_cmd); // 更新索引到命令后 switch (base_cmd) { case 1u: // M1 send_response(RESP_TYPE_OK, s_report_status_ok, sizeof(s_report_status_ok)); return; /* ========================================== * M888 软件重启命令 * ========================================== */ case 888u: // 先发送确认响应 send_response(RESP_TYPE_OK, s_report_status_ok, sizeof(s_report_status_ok)); // 短暂延时确保响应发送完成 delay_ms(100); // 执行软件重启 system_software_reset(); return; /* ========================================== * M999 输出固件版本号命令 * ========================================== */ case 999u: //M999: 输出固件版本号 { char version_str[16]; char *p = version_str; *p++ = 'v'; p += uint_to_str(BOARD_TYPE_CODE, p); *p++ = '.'; p += uint_to_str(FW_VERSION_MAJOR, p); *p++ = '.'; p += uint_to_str(FW_VERSION_MINOR, p); *p++ = '.'; p += uint_to_str(FW_VERSION_PATCH, p); *p = '\0'; // null-terminate for printf safety uint8_t n = (uint8_t)(p - version_str); send_response(RESP_TYPE_OK, (uint8_t *)version_str, n); COMMAND_DEBUG("Firmware Version: %s", version_str); } return; /* ========================================== * M9999 进入OTA模式 * ========================================== */ case 9999u: //M9999: 进入OTA模式 __disable_irq(); // 关中断,防止竞态条件 NVIC_SystemReset(); // 触发系统复位,进入Bootloader return; default: send_response(RESP_TYPE_CMD_ERR, s_report_status_err, sizeof(s_report_status_err)); break; } } /** * @brief 处理串口环形缓冲区中的命令数据,解析完整的协议帧。 * @details 本函数实现一个基于状态机的协议解析器,用于处理格式为 D5 03 LEN [cmd] CRC 的命令帧: * - 状态1:等待包头字节 PROTOCOL_PACKAGE_HEADER (0xD5) * - 状态2:接收板卡类型字节 PROTOCOL_BOARD_TYPE * - 状态3:接收长度字段并计算期望的完整帧长度 * - 状态4:继续接收剩余数据直到完整帧 * - 状态5:对完整帧进行校验(包头、板卡类型、CRC)并处理 * * 函数采用非阻塞方式处理,每次调用处理缓冲区中所有可用数据。 * 遇到格式错误、长度异常或校验失败时自动重置状态机。 * * @note 本函数使用静态变量维护解析状态,因此不可重入。在中断环境中使用需注意并发安全。 * 协议帧最大长度受 PROTOCOL_MAX_FRAME_LEN 限制,超出范围的帧将被丢弃。 * * @warning 函数依赖 uart_ring_buffer_available() 和 uart_ring_buffer_get() * 正确实现,若这些函数有缺陷可能导致死循环或数据丢失。 * * @see handle_command() 用于处理校验通过的完整命令帧 * @see command_sum_crc_calc() 用于计算和校验 CRC 值 * @see send_response() 用于发送错误响应 * * @ingroup Command */ void command_process(void) { static uint8_t cmd_buf[PROTOCOL_MAX_FRAME_LEN]; static uint8_t cmd_len = 0; static uint8_t expected_cmd_len = 0; // 0 表示尚未确定总长度 while (uart_ring_buffer_available() > 0) { int byte = uart_ring_buffer_get(); if (byte < 0) break; if (cmd_len == 0) { if ((uint8_t)byte == PROTOCOL_PACKAGE_HEADER) { cmd_buf[cmd_len++] = (uint8_t)byte; expected_cmd_len = 0; // 等待进一步字段以确定长度 } else { // 丢弃非起始字节 } continue; } if (cmd_len >= PROTOCOL_MAX_FRAME_LEN) { // 防御:缓冲溢出,复位状态机 cmd_len = 0; expected_cmd_len = 0; continue; } // 缓存后续字节 cmd_buf[cmd_len++] = (uint8_t)byte; // 当到达长度字段(索引 2)后,确定总长度:3 + LEN + 1 if (cmd_len == 3) { uint8_t payload_len = cmd_buf[2]; expected_cmd_len = (uint8_t)(3 + payload_len + 1); if (expected_cmd_len > PROTOCOL_MAX_FRAME_LEN) { // 异常:长度超界,复位状态机 cmd_len = 0; expected_cmd_len = 0; } continue; } if (expected_cmd_len > 0 && cmd_len == expected_cmd_len) { // 到帧尾,进行各项校验 bool verification_status = true; #ifdef DEBUG_VERBOSE if (cmd_buf[0] != PROTOCOL_PACKAGE_HEADER) { send_response(RESP_TYPE_HEADER_ERR, s_report_status_err, sizeof(s_report_status_err)); verification_status = false; } #endif if (verification_status && cmd_buf[1] != PROTOCOL_BOARD_TYPE) { send_response(RESP_TYPE_TYPE_ERR, s_report_status_err, sizeof(s_report_status_err)); verification_status = false; } if (verification_status) { uint8_t crc_calc = command_sum_crc_calc(cmd_buf, expected_cmd_len); uint8_t crc_recv = cmd_buf[expected_cmd_len - 1]; if (crc_calc != crc_recv) { send_response(RESP_TYPE_CRC_ERR, s_report_status_err, sizeof(s_report_status_err)); verification_status = false; } } if (verification_status) { handle_command(cmd_buf, expected_cmd_len); } else { // 验证失败时清空缓冲区,避免后续帧受影响 uart_rx_irq_pause(); uart_ring_buffer_clear(); uart_rx_irq_resume(); } // 复位,等待下一帧 cmd_len = 0; expected_cmd_len = 0; } } } /** * @brief 执行命令(简化版) * @details 根据命令字符串直接构造命令帧并执行,无需手动构造协议帧 * @param cmd_str 命令字符串(如"M730S0T1000"、"M731S100"等) * @note 简化的测试函数,自动处理协议帧构造、CRC计算和命令执行 * @ingroup Command */ void command_execute(const char *cmd_str) { if (cmd_str == NULL) return; uint8_t cmd_len = (uint8_t)strlen(cmd_str); uint8_t frame_len = 3 + cmd_len + 1; // header + type + len + cmd + crc uint8_t frame_buf[32]; // 简单固定缓冲区 if (frame_len > sizeof(frame_buf)) return; // 构造命令帧 frame_buf[0] = PROTOCOL_PACKAGE_HEADER; // 0xD5 frame_buf[1] = PROTOCOL_BOARD_TYPE; // Board Type frame_buf[2] = cmd_len; // 命令长度 // 复制命令数据 for (uint8_t i = 0; i < cmd_len; i++) { frame_buf[3 + i] = (uint8_t)cmd_str[i]; } // 计算CRC uint16_t crc = 0; for (uint8_t i = 1; i < (frame_len - 1); i++) { crc += frame_buf[i]; } frame_buf[frame_len - 1] = (uint8_t)(crc & 0xFF); // 清空缓冲区并执行命令 uart_rx_irq_pause(); uart_ring_buffer_clear(); uart_rx_irq_resume(); for (uint8_t i = 0; i < frame_len; i++) { uart_ring_buffer_put(frame_buf[i]); } command_process(); }