2023-01-24 22:37:53 +08:00
|
|
|
|
/**
|
|
|
|
|
* @file rm_referee.C
|
|
|
|
|
* @author kidneygood (you@domain.com)
|
|
|
|
|
* @brief
|
|
|
|
|
* @version 0.1
|
|
|
|
|
* @date 2022-11-18
|
|
|
|
|
*
|
|
|
|
|
* @copyright Copyright (c) 2022
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "rm_referee.h"
|
2022-11-18 22:06:06 +08:00
|
|
|
|
#include "string.h"
|
2023-02-03 15:25:58 +08:00
|
|
|
|
#include "crc_ref.h"
|
2022-11-18 22:06:06 +08:00
|
|
|
|
#include "bsp_usart.h"
|
|
|
|
|
|
|
|
|
|
#define RE_RX_BUFFER_SIZE 200
|
|
|
|
|
|
2023-04-03 18:40:49 +08:00
|
|
|
|
static USARTInstance *referee_usart_instance;
|
|
|
|
|
static referee_tx_buffer_t referee_tx_buffer={{0},0};
|
2022-12-03 15:20:17 +08:00
|
|
|
|
static referee_info_t referee_info;
|
2023-01-09 16:17:03 +08:00
|
|
|
|
static void JudgeReadData(uint8_t *ReadFromUsart);
|
|
|
|
|
static void RefereeRxCallback();
|
|
|
|
|
|
2023-01-26 19:52:07 +08:00
|
|
|
|
uint8_t UI_Seq = 0; // 包序号,供整个referee文件使用
|
|
|
|
|
|
2023-01-09 16:17:03 +08:00
|
|
|
|
/* 裁判系统通信初始化 */
|
|
|
|
|
referee_info_t *RefereeInit(UART_HandleTypeDef *referee_usart_handle)
|
|
|
|
|
{
|
|
|
|
|
USART_Init_Config_s conf;
|
|
|
|
|
conf.module_callback = RefereeRxCallback;
|
|
|
|
|
conf.usart_handle = referee_usart_handle;
|
|
|
|
|
conf.recv_buff_size = RE_RX_BUFFER_SIZE;
|
|
|
|
|
referee_usart_instance = USARTRegister(&conf);
|
|
|
|
|
return &referee_info;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-03 18:40:49 +08:00
|
|
|
|
/**
|
|
|
|
|
* @brief 载入缓存区函数
|
|
|
|
|
* @param send 待载入的数据
|
|
|
|
|
*/
|
|
|
|
|
void RefereeLoadToBuffer(uint8_t *send, uint16_t tx_len)
|
|
|
|
|
{
|
|
|
|
|
memcpy((uint8_t *)(&referee_tx_buffer.buffer)+referee_tx_buffer.pos,send,tx_len);
|
|
|
|
|
referee_tx_buffer.pos+=tx_len;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2023-01-09 16:17:03 +08:00
|
|
|
|
/**
|
|
|
|
|
* @brief 发送函数
|
|
|
|
|
* @param send 待发送数据
|
|
|
|
|
*/
|
2023-01-26 19:52:07 +08:00
|
|
|
|
void RefereeSend(uint8_t *send, uint16_t tx_len)
|
2023-01-09 16:17:03 +08:00
|
|
|
|
{
|
2023-04-03 18:40:49 +08:00
|
|
|
|
USARTSend(referee_usart_instance, send, tx_len);//syhtodo此函数需要重写
|
2023-01-26 15:07:00 +08:00
|
|
|
|
/* syhtodo DMA请求过快会导致数据发送丢失,考虑数据尽可能打成一个整包以及队列发送,并且发送函数添加缓冲区 */
|
2023-01-09 16:17:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-03 18:40:49 +08:00
|
|
|
|
|
|
|
|
|
|
2023-01-09 16:17:03 +08:00
|
|
|
|
/*裁判系统串口接收回调函数,解析数据 */
|
|
|
|
|
static void RefereeRxCallback()
|
|
|
|
|
{
|
|
|
|
|
JudgeReadData(referee_usart_instance->recv_buff);
|
|
|
|
|
}
|
2022-11-18 22:06:06 +08:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @brief 读取裁判数据,中断中读取保证速度
|
2023-01-01 17:32:22 +08:00
|
|
|
|
* @param ReadFromUsart: 读取到的裁判系统原始数据
|
2022-11-18 22:06:06 +08:00
|
|
|
|
* @retval 是否对正误判断做处理
|
|
|
|
|
* @attention 在此判断帧头和CRC校验,无误再写入数据,不重复判断帧头
|
|
|
|
|
*/
|
2022-12-05 22:10:08 +08:00
|
|
|
|
static void JudgeReadData(uint8_t *ReadFromUsart)
|
2022-11-18 22:06:06 +08:00
|
|
|
|
{
|
2023-01-26 19:52:07 +08:00
|
|
|
|
uint16_t judge_length; // 统计一帧数据长度
|
2023-01-06 22:54:01 +08:00
|
|
|
|
if (ReadFromUsart == NULL) // 空数据包,则不作任何处理
|
2022-11-19 16:28:39 +08:00
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// 写入帧头数据(5-byte),用于判断是否开始存储裁判数据
|
2022-11-18 22:06:06 +08:00
|
|
|
|
memcpy(&referee_info.FrameHeader, ReadFromUsart, LEN_HEADER);
|
|
|
|
|
|
2022-11-19 16:28:39 +08:00
|
|
|
|
// 判断帧头数据(0)是否为0xA5
|
2023-01-13 21:45:04 +08:00
|
|
|
|
if (ReadFromUsart[SOF] == REFEREE_SOF)
|
2022-11-18 22:06:06 +08:00
|
|
|
|
{
|
2022-11-19 16:28:39 +08:00
|
|
|
|
// 帧头CRC8校验
|
2022-11-18 22:06:06 +08:00
|
|
|
|
if (Verify_CRC8_Check_Sum(ReadFromUsart, LEN_HEADER) == TRUE)
|
|
|
|
|
{
|
2022-11-19 16:28:39 +08:00
|
|
|
|
// 统计一帧数据长度(byte),用于CR16校验
|
2022-11-18 22:06:06 +08:00
|
|
|
|
judge_length = ReadFromUsart[DATA_LENGTH] + LEN_HEADER + LEN_CMDID + LEN_TAIL;
|
2022-11-19 16:28:39 +08:00
|
|
|
|
// 帧尾CRC16校验
|
2022-11-18 22:06:06 +08:00
|
|
|
|
if (Verify_CRC16_Check_Sum(ReadFromUsart, judge_length) == TRUE)
|
|
|
|
|
{
|
|
|
|
|
// 2个8位拼成16位int
|
|
|
|
|
referee_info.CmdID = (ReadFromUsart[6] << 8 | ReadFromUsart[5]);
|
2022-11-19 16:28:39 +08:00
|
|
|
|
// 解析数据命令码,将数据拷贝到相应结构体中(注意拷贝数据的长度)
|
|
|
|
|
// 第8个字节开始才是数据 data=7
|
2022-11-18 22:06:06 +08:00
|
|
|
|
switch (referee_info.CmdID)
|
|
|
|
|
{
|
|
|
|
|
case ID_game_state: // 0x0001
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.GameState, (ReadFromUsart + DATA_Offset), LEN_game_state);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_game_result: // 0x0002
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.GameResult, (ReadFromUsart + DATA_Offset), LEN_game_result);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_game_robot_survivors: // 0x0003
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.GameRobotHP, (ReadFromUsart + DATA_Offset), LEN_game_robot_HP);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_event_data: // 0x0101
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.EventData, (ReadFromUsart + DATA_Offset), LEN_event_data);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_supply_projectile_action: // 0x0102
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.SupplyProjectileAction, (ReadFromUsart + DATA_Offset), LEN_supply_projectile_action);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_game_robot_state: // 0x0201
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.GameRobotState, (ReadFromUsart + DATA_Offset), LEN_game_robot_state);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_power_heat_data: // 0x0202
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.PowerHeatData, (ReadFromUsart + DATA_Offset), LEN_power_heat_data);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_game_robot_pos: // 0x0203
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.GameRobotPos, (ReadFromUsart + DATA_Offset), LEN_game_robot_pos);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_buff_musk: // 0x0204
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.BuffMusk, (ReadFromUsart + DATA_Offset), LEN_buff_musk);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_aerial_robot_energy: // 0x0205
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.AerialRobotEnergy, (ReadFromUsart + DATA_Offset), LEN_aerial_robot_energy);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_robot_hurt: // 0x0206
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.RobotHurt, (ReadFromUsart + DATA_Offset), LEN_robot_hurt);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
case ID_shoot_data: // 0x0207
|
2023-01-12 16:11:40 +08:00
|
|
|
|
memcpy(&referee_info.ShootData, (ReadFromUsart + DATA_Offset), LEN_shoot_data);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
2023-01-26 19:52:07 +08:00
|
|
|
|
case ID_student_interactive: // 0x0301 syhtodo接收代码未测试
|
|
|
|
|
memcpy(&referee_info.ReceiveData, (ReadFromUsart + DATA_Offset), LEN_receive_data);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-01-04 21:26:11 +08:00
|
|
|
|
// 首地址加帧长度,指向CRC16下一字节,用来判断是否为0xA5,从而判断一个数据包是否有多帧数据
|
2022-11-18 22:06:06 +08:00
|
|
|
|
if (*(ReadFromUsart + sizeof(xFrameHeader) + LEN_CMDID + referee_info.FrameHeader.DataLength + LEN_TAIL) == 0xA5)
|
|
|
|
|
{
|
2023-01-04 21:26:11 +08:00
|
|
|
|
// 如果一个数据包出现了多帧数据,则再次调用解析函数,直到所有数据包解析完毕
|
2022-11-19 16:28:39 +08:00
|
|
|
|
JudgeReadData(ReadFromUsart + sizeof(xFrameHeader) + LEN_CMDID + referee_info.FrameHeader.DataLength + LEN_TAIL);
|
2022-11-18 22:06:06 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-12-05 22:10:08 +08:00
|
|
|
|
}
|