sentry_gimbal_hzz/modules/referee/rm_referee.c

169 lines
5.8 KiB
C
Raw Normal View History

/**
* @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"
#include "crc_ref.h"
2022-11-18 22:06:06 +08:00
#include "bsp_usart.h"
#include "task.h"
2022-11-18 22:06:06 +08:00
#define RE_RX_BUFFER_SIZE 200
static USARTInstance *referee_usart_instance; // 裁判系统串口实例
static referee_info_t referee_info; // 裁判系统数据
static Referee_Interactive_info_t *UI_tmp; // UI绘制需要的机器人状态数据
2023-01-09 16:17:03 +08:00
static void RefereeRxCallback();
static void JudgeReadData(uint8_t *buff);
2023-01-09 16:17:03 +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, Referee_Interactive_info_t *UI_data)
2023-01-09 16:17:03 +08:00
{
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);
UI_tmp = UI_data;
2023-01-09 16:17:03 +08:00
return &referee_info;
}
void RefereeGetUIData(referee_info_t **recv_info_pp, Referee_Interactive_info_t **UI_data_pp)
2023-04-03 18:40:49 +08:00
{
*recv_info_pp = &referee_info;
*UI_data_pp = UI_tmp;
2023-04-03 18:40:49 +08:00
}
void CommBetweenRobotSend(referee_id_t *_id, robot_interactive_data_t *_data)
{
Communicate_SendData_t SendData;
uint8_t temp_datalength = Interactive_Data_LEN_Head + Communicate_Data_LEN; // 计算交互数据长度 6+n,n为交互数据长度
SendData.FrameHeader.SOF = REFEREE_SOF;
SendData.FrameHeader.DataLength = temp_datalength;
SendData.FrameHeader.Seq = UI_Seq;
SendData.FrameHeader.CRC8 = Get_CRC8_Check_Sum((uint8_t *)&SendData, LEN_CRC8, 0xFF);
SendData.CmdID = ID_student_interactive;
SendData.datahead.data_cmd_id = Communicate_Data_ID;
SendData.datahead.sender_ID = _id->Robot_ID;
SendData.datahead.receiver_ID = _id->Receiver_Robot_ID;
SendData.Data = *_data;
SendData.frametail = Get_CRC16_Check_Sum((uint8_t *)&SendData, LEN_HEADER + LEN_CMDID + temp_datalength, 0xFFFF);
RefereeSend((uint8_t *)&SendData, LEN_HEADER + LEN_CMDID + temp_datalength + LEN_TAIL); // 发送
UI_Seq++; // 包序号+1
}
2023-04-03 18:40:49 +08:00
2023-01-09 16:17:03 +08:00
/**
* @brief ,使
* @param
2023-01-09 16:17:03 +08:00
*/
void RefereeSend(uint8_t *send, uint16_t tx_len)
2023-01-09 16:17:03 +08:00
{
static TickType_t xLastWakeTime;
USARTSend(referee_usart_instance, send, tx_len, USART_TRANSFER_DMA);
vTaskDelayUntil(&xLastWakeTime, 120);
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 ,
* @param buff:
2022-11-18 22:06:06 +08:00
* @retval
* @attention CRC校验,
*/
static void JudgeReadData(uint8_t *buff)
2022-11-18 22:06:06 +08:00
{
uint16_t judge_length; // 统计一帧数据长度
if (buff == NULL) // 空数据包,则不作任何处理
return;
// 写入帧头数据(5-byte),用于判断是否开始存储裁判数据
memcpy(&referee_info.FrameHeader, buff, LEN_HEADER);
2022-11-18 22:06:06 +08:00
// 判断帧头数据(0)是否为0xA5
if (buff[SOF] == REFEREE_SOF)
2022-11-18 22:06:06 +08:00
{
// 帧头CRC8校验
if (Verify_CRC8_Check_Sum(buff, LEN_HEADER) == TRUE)
2022-11-18 22:06:06 +08:00
{
// 统计一帧数据长度(byte),用于CR16校验
judge_length = buff[DATA_LENGTH] + LEN_HEADER + LEN_CMDID + LEN_TAIL;
// 帧尾CRC16校验
if (Verify_CRC16_Check_Sum(buff, judge_length) == TRUE)
2022-11-18 22:06:06 +08:00
{
// 2个8位拼成16位int
referee_info.CmdID = (buff[6] << 8 | buff[5]);
// 解析数据命令码,将数据拷贝到相应结构体中(注意拷贝数据的长度)
// 第8个字节开始才是数据 data=7
2022-11-18 22:06:06 +08:00
switch (referee_info.CmdID)
{
case ID_game_state: // 0x0001
memcpy(&referee_info.GameState, (buff + DATA_Offset), LEN_game_state);
2022-11-18 22:06:06 +08:00
break;
case ID_game_result: // 0x0002
memcpy(&referee_info.GameResult, (buff + DATA_Offset), LEN_game_result);
2022-11-18 22:06:06 +08:00
break;
case ID_game_robot_survivors: // 0x0003
memcpy(&referee_info.GameRobotHP, (buff + DATA_Offset), LEN_game_robot_HP);
2022-11-18 22:06:06 +08:00
break;
case ID_event_data: // 0x0101
memcpy(&referee_info.EventData, (buff + DATA_Offset), LEN_event_data);
2022-11-18 22:06:06 +08:00
break;
case ID_supply_projectile_action: // 0x0102
memcpy(&referee_info.SupplyProjectileAction, (buff + DATA_Offset), LEN_supply_projectile_action);
2022-11-18 22:06:06 +08:00
break;
case ID_game_robot_state: // 0x0201
memcpy(&referee_info.GameRobotState, (buff + DATA_Offset), LEN_game_robot_state);
2022-11-18 22:06:06 +08:00
break;
case ID_power_heat_data: // 0x0202
memcpy(&referee_info.PowerHeatData, (buff + DATA_Offset), LEN_power_heat_data);
2022-11-18 22:06:06 +08:00
break;
case ID_game_robot_pos: // 0x0203
memcpy(&referee_info.GameRobotPos, (buff + DATA_Offset), LEN_game_robot_pos);
2022-11-18 22:06:06 +08:00
break;
case ID_buff_musk: // 0x0204
memcpy(&referee_info.BuffMusk, (buff + DATA_Offset), LEN_buff_musk);
2022-11-18 22:06:06 +08:00
break;
case ID_aerial_robot_energy: // 0x0205
memcpy(&referee_info.AerialRobotEnergy, (buff + DATA_Offset), LEN_aerial_robot_energy);
2022-11-18 22:06:06 +08:00
break;
case ID_robot_hurt: // 0x0206
memcpy(&referee_info.RobotHurt, (buff + DATA_Offset), LEN_robot_hurt);
2022-11-18 22:06:06 +08:00
break;
case ID_shoot_data: // 0x0207
memcpy(&referee_info.ShootData, (buff + DATA_Offset), LEN_shoot_data);
2022-11-18 22:06:06 +08:00
break;
case ID_student_interactive: // 0x0301 syhtodo接收代码未测试
memcpy(&referee_info.ReceiveData, (buff + DATA_Offset), LEN_receive_data);
2022-11-18 22:06:06 +08:00
break;
}
}
}
// 首地址加帧长度,指向CRC16下一字节,用来判断是否为0xA5,从而判断一个数据包是否有多帧数据
if (*(buff + sizeof(xFrameHeader) + LEN_CMDID + referee_info.FrameHeader.DataLength + LEN_TAIL) == 0xA5)
{ // 如果一个数据包出现了多帧数据,则再次调用解析函数,直到所有数据包解析完毕
JudgeReadData(buff + sizeof(xFrameHeader) + LEN_CMDID + referee_info.FrameHeader.DataLength + LEN_TAIL);
2022-11-18 22:06:06 +08:00
}
}
}