协议数据按照通信方向可以分为两大类:
1 反馈信息:包含各个机构传感器反馈信息、底层计算出来的一些反馈信息;
2 底层状态信息:包含底层设备运行状态、底层对上层数据的一些响应等;
3 转发数据:包含裁判系统的全部信息、服务器端的自定义信息;
1 控制信息:上层对底层 3 个执行机构的控制信息;
一帧协议数据分为 4 个部分,分别是帧头数据、命令码ID、数据、帧尾校验数据。
/**
* @brief frame header structure definition
*/
/* This Struct Is Used To Describe A Package Header */
typedef struct
{
uint8_t sof; /*!< Identify Of A Package */
union {
struct
{
uint16_t data_len : 10; /*!< Data Length, Include Header And Crc */
uint16_t version : 6; /*!< Protocol Version */
};
uint16_t ver_data_len;
};
union {
struct
{
uint8_t session : 5; /*!< Need(0~1) Or Not Need(2~63) Ack */
uint8_t pack_type : 1; /*!< Ack Package Or Normal Package */
uint8_t res : 2; /*!< Reserve */
};
uint8_t S_A_R_c;
};
uint8_t sender; /*!< Sender Module Information */
uint8_t reciver; /*!< Receiver Module Information */
uint16_t res1; /*!< Reserve 1 */
uint16_t seq_num; /*!< Sequence Number */
uint16_t crc_16; /*!< CRC16 */
uint8_t pdata[];
} protocol_pack_desc_t;
帧头数据 | 占用字节 | 详细描述 |
---|---|---|
sof | 1 | 数据的域ID |
ver_data_len | 2 | 每帧内数据的长度和协议版本号 |
session | 1 | 包序号,在0xA0域中保留 |
sender | 1 | 发送者地址 |
reciver | 1 | 发送者地址 |
res | 2 | 保留位 |
seq | 2 | 包序号 |
crc16 | 2 | 帧头的crc校验结果 |
命令码 | 占用字节 |
---|---|
cmdid | 2 |
一个命令码对应一帧包含具体信息的数据,下面是现有所有数据对应的命令码。
命令码对应的数据传输方向和具体功能如下:
命令码 | 传输方向 | 功能介绍 | 频率 |
---|---|---|---|
0x0001 | 主控-->PC | 比赛时机器人状态 | 裁判系统10Hz |
0x0002 | 主控-->PC | 实时伤害数据 | 受到攻击时发送 |
0x0003 | 主控-->PC | 实时射击数据 | 裁判系统 |
0x0004 | 主控-->PC | 实时功率、热量数据 | ICRA不使用,不发送 |
0x0005 | 主控-->PC | 场地 RFID 数据 | 检测到 IC 卡发送 |
0x0006 | 主控-->PC | 比赛结果数据 | 比赛结束时发送 |
0x0007 | 主控-->PC | 获得 buff 数据 | 裁判系统 |
0x0008 | 主控-->PC | 场地 UWB 数据 | 裁判系统 |
0x0204 | 主控-->PC | 机器人底盘相关信息 | 100Hz定频 |
0x0304 | 主控-->PC | 机器人云台相关信息 | 100Hz定频 |
0x0402 | 主控-->PC | UWB相关信息 | UWB更新频率 |
0x0206 | PC-->主控 | 设置底盘速度 | |
0x0208 | PC-->主控 | 设置底盘速度(有加速度) | |
0x0309 | PC-->主控 | 控制摩擦轮转速 | 开启摩擦轮使用 |
0x030A | PC-->主控 | 控制射击 | |
0x0403 | PC-->主控 | 云台相关校准信息 | 需要校准云台时发送 |
为命令码 ID 对应的数据结构,数据长度即这个结构体的大小。
数据 | 占用字节 |
---|---|
data | data_length |
裁判系统学生串口信息,详情查阅裁判系统手册
控制信息与推送信息: 详细指令查看application/infantry_cmd.h
struct cmd_chassis_info
{
int16_t gyro_angle;
int16_t gyro_palstance;
int32_t position_x_mm;
int32_t position_y_mm;
int16_t angle_deg;
int16_t v_x_mm;
int16_t v_y_mm;
};
struct cmd_gimbal_info
{
uint8_t mode;
/* unit: degree */
int16_t pitch_ecd_angle;
int16_t yaw_ecd_angle;
int16_t pitch_gyro_angle;
int16_t yaw_gyro_angle;
/* uint: degree/s */
int16_t yaw_rate;
int16_t pitch_rate;
};
struct cmd_gimbal_angle
{
union{
uint8_t flag;
struct{
uint8_t yaw_mode:1; // 0 code angle
uint8_t pitch_mode:1;
}bit;
} ctrl;
int16_t pitch;
int16_t yaw;
};
struct cmd_chassis_speed
{
int16_t vx; // forward/back
int16_t vy; // left/right
int16_t vw; // anticlockwise/clockwise
int16_t rotate_x_offset;
int16_t rotate_y_offset;
};
struct cmd_chassis_spd_acc
{
int16_t vx;
int16_t vy;
int16_t vw;
int16_t ax;
int16_t ay;
int16_t wz;
int16_t rotate_x_offset;
int16_t rotate_y_offset;
};
struct cmd_firction_speed
{
uint16_t left;
uint16_t right;
};
struct cmd_shoot_num
{
uint8_t shoot_cmd;
uint32_t shoot_add_num;
uint16_t shoot_freq;
};
使用如下API配置协议和发送的数据:
int32_t protocol_send_cmd_config(uint16_t cmd,
uint8_t resend_times,
uint16_t resend_timeout,
uint8_t ack_enable,
ack_handle_fn_t ack_callback,
no_ack_handle_fn_t no_ack_callback);
int32_t protocol_rcv_cmd_register(uint16_t cmd, rcv_handle_fn_t rcv_callback);
int32_t protocol_rcv_cmd_unregister(uint16_t cmd);
int32_t protocol_send_cmd_unregister(uint16_t cmd);
uint32_t protocol_send_flush(void);
uint32_t protocol_unpack_flush(void);
uint32_t protocol_send_list_add_callback_reg(void_fn_t fn);
int32_t protocol_can_interface_register(char *interface_name,
uint16_t rcv_buf_size,
uint8_t broadcast_output_enable,
uint8_t can_port,
uint32_t can_tx_id,
uint32_t can_rx_id,
int (*can_send_fn)(uint32_t std_id, uint8_t *p_data, uint32_t len));
int32_t protocol_uart_interface_register(char *interface_name,
uint16_t rcv_buf_size,
uint8_t broadcast_output_enable,
uint8_t com_port,
int (*com_send_fn)(uint8_t *p_data, uint32_t len));
int32_t protocol_set_route(uint8_t tar_add, const char *name);
uint32_t protocol_can_rcv_data(uint8_t can_port, uint32_t rcv_id, void *p_data, uint32_t data_len);
uint32_t protocol_uart_rcv_data(uint8_t com_port, void *p_data, uint32_t data_len);
使用下面的方式解决裁判系统粘包的问题
void read_and_unpack_thread(void *argu)
{
uint8_t byte = 0;
int32_t read_len;
int32_t buff_read_index;
uint16_t data_len;
unpack_step_e unpack_step;
int32_t index;
uint8_t protocol_packet[PROTOCAL_FRAME_MAX_SIZE];
while (1)
{
read_len = uart_recv(uart_fd, computer_rx_buf, UART_BUFF_SIZE);
buff_read_index = 0;
while (read_len--)
{
byte = computer_rx_buf[buff_read_index++];
switch(unpack_step)
{
case STEP_HEADER_SOF:
{
if(byte == UP_REG_ID)
{
unpack_step = STEP_LENGTH_LOW;
protocol_packet[index++] = byte;
}
else
{
index = 0;
}
}break;
case STEP_LENGTH_LOW:
{
data_len = byte;
protocol_packet[index++] = byte;
unpack_step = STEP_LENGTH_HIGH;
}break;
case STEP_LENGTH_HIGH:
{
data_len |= (byte << 8);
protocol_packet[index++] = byte;
if(data_len < (PROTOCAL_FRAME_MAX_SIZE - HEADER_LEN - CRC_LEN))
{
unpack_step = STEP_FRAME_SEQ;
}
else
{
unpack_step = STEP_HEADER_SOF;
index = 0;
}
}break;
case STEP_FRAME_SEQ:
{
protocol_packet[index++] = byte;
unpack_step = STEP_HEADER_CRC8;
}break;
case STEP_HEADER_CRC8:
{
protocol_packet[index++] = byte;
if (index == HEADER_LEN)
{
if ( verify_crc8_check_sum(protocol_packet, HEADER_LEN) )
{
unpack_step = STEP_DATA_CRC16;
}
else
{
unpack_step = STEP_HEADER_SOF;
index = 0;
}
}
}break;
case STEP_DATA_CRC16:
{
if (index < (HEADER_LEN + CMD_LEN + data_len + CRC_LEN))
{
protocol_packet[index++] = byte;
}
if (index >= (HEADER_LEN + CMD_LEN + data_len + CRC_LEN))
{
unpack_step = STEP_HEADER_SOF;
index = 0;
if ( verify_crc16_check_sum(protocol_packet, HEADER_LEN + CMD_LEN + data_len + CRC_LEN) )
{
data_handle(protocol_packet);
}
}
}break;
default:
{
unpack_step = STEP_HEADER_SOF;
index = 0;
}break;
}
}
}
}
void data_handle(uint8_t *p_frame)
{
frame_header_t *p_header = (frame_header_t*)p_frame;
memcpy(p_header, p_frame, HEADER_LEN);
uint16_t data_length = p_header->data_length;
uint16_t cmd_id = *(uint16_t *)(p_frame + HEADER_LEN);
uint8_t *data_addr = p_frame + HEADER_LEN + CMD_LEN;
switch (cmd_id)
{
case GAME_INFO_ID:
memcpy(&game_information, data_addr, data_length);
break;
//............
//............
}
}