2022-10-20 17:13:02 +08:00
|
|
|
#include "remote_control.h"
|
|
|
|
#include "string.h"
|
|
|
|
#include "bsp_usart.h"
|
2022-12-05 21:01:26 +08:00
|
|
|
#include "memory.h"
|
2022-10-20 17:13:02 +08:00
|
|
|
|
2022-11-01 22:32:15 +08:00
|
|
|
#define REMOTE_CONTROL_FRAME_SIZE 18u // 遥控器接收的buffer大小
|
|
|
|
// 遥控器数据
|
2022-12-05 21:01:26 +08:00
|
|
|
static RC_ctrl_t rc_ctrl[2]; //[0]:当前数据,[1]:上一次的数据.用于按键判断
|
2022-11-01 22:32:15 +08:00
|
|
|
// 遥控器拥有的串口实例
|
2022-12-05 21:01:26 +08:00
|
|
|
static USARTInstance *rc_usart_instance;
|
2022-10-20 17:13:02 +08:00
|
|
|
|
|
|
|
/**
|
2022-11-01 22:32:15 +08:00
|
|
|
* @brief remote control protocol resolution
|
|
|
|
* @param[in] sbus_buf: raw data point
|
|
|
|
* @param[out] rc_ctrl: remote control data struct point
|
|
|
|
* @retval none
|
2022-10-20 17:13:02 +08:00
|
|
|
*/
|
2022-12-05 21:01:26 +08:00
|
|
|
static void sbus_to_rc(volatile const uint8_t *sbus_buf)
|
2022-10-20 17:13:02 +08:00
|
|
|
{
|
2022-12-08 12:08:49 +08:00
|
|
|
memcpy(&rc_ctrl[1], &rc_ctrl[TEMP], sizeof(RC_ctrl_t)); // 保存上一次的数据
|
2022-12-08 17:36:12 +08:00
|
|
|
// 摇杆,直接解算时减去偏置
|
2022-12-09 18:25:35 +08:00
|
|
|
rc_ctrl[TEMP].rc.joystick[0] = ((sbus_buf[0] | (sbus_buf[1] << 8)) & 0x07ff)- RC_CH_VALUE_OFFSET ; //!< Channel 0
|
|
|
|
rc_ctrl[TEMP].rc.joystick[1] = (((sbus_buf[1] >> 3) | (sbus_buf[2] << 5)) & 0x07ff)- RC_CH_VALUE_OFFSET ; //!< Channel 1
|
|
|
|
rc_ctrl[TEMP].rc.joystick[2] = (((sbus_buf[2] >> 6) | (sbus_buf[3] << 2) | (sbus_buf[4] << 10)) & 0x07ff)- RC_CH_VALUE_OFFSET ; //!< Channel 2
|
|
|
|
rc_ctrl[TEMP].rc.joystick[3] = (((sbus_buf[4] >> 1) | (sbus_buf[5] << 7)) & 0x07ff)- RC_CH_VALUE_OFFSET ; //!< Channel 3
|
|
|
|
rc_ctrl[TEMP].rc.joystick[4] = ((sbus_buf[16] | (sbus_buf[17] << 8)) & 0x07FF)- RC_CH_VALUE_OFFSET; // 左侧拨轮
|
2022-12-05 21:01:26 +08:00
|
|
|
// 开关,0左1右
|
2022-12-08 12:08:49 +08:00
|
|
|
rc_ctrl[TEMP].rc.s[0] = ((sbus_buf[5] >> 4) & 0x0003); //!< Switch left
|
|
|
|
rc_ctrl[TEMP].rc.s[1] = ((sbus_buf[5] >> 4) & 0x000C) >> 2; //!< Switch right
|
2022-12-05 21:01:26 +08:00
|
|
|
|
|
|
|
// 鼠标解析
|
2022-12-08 12:08:49 +08:00
|
|
|
rc_ctrl[TEMP].mouse.x = sbus_buf[6] | (sbus_buf[7] << 8); //!< Mouse X axis
|
|
|
|
rc_ctrl[TEMP].mouse.y = sbus_buf[8] | (sbus_buf[9] << 8); //!< Mouse Y axis
|
|
|
|
rc_ctrl[TEMP].mouse.z = sbus_buf[10] | (sbus_buf[11] << 8); //!< Mouse Z axis
|
|
|
|
rc_ctrl[TEMP].mouse.press_l = sbus_buf[12]; //!< Mouse Left Is Press ?
|
|
|
|
rc_ctrl[TEMP].mouse.press_r = sbus_buf[13]; //!< Mouse Right Is Press ?
|
2022-12-05 21:01:26 +08:00
|
|
|
|
|
|
|
// 按键值,每个键1bit,key_temp共16位;按键顺序在remote_control.h的宏定义中可见
|
2022-12-06 22:58:42 +08:00
|
|
|
// 使用位域后不再需要这一中间操作
|
2022-12-08 12:08:49 +08:00
|
|
|
rc_ctrl[TEMP].key_temp = sbus_buf[14] | (sbus_buf[15] << 8); //!< KeyBoard value
|
2022-12-05 21:01:26 +08:00
|
|
|
|
2022-12-06 22:58:42 +08:00
|
|
|
// @todo 似乎可以直接用位域操作进行,把key_temp通过强制类型转换变成key类型? 位域方案在下面,尚未测试
|
2022-12-05 21:01:26 +08:00
|
|
|
// 按键值解算,利用宏+循环减少代码长度
|
2022-12-07 21:08:22 +08:00
|
|
|
for (uint16_t i = 0x0001, j = 0; i != 0x8000; i *= 2, j++) // 依次查看每一个键
|
2022-10-20 17:13:02 +08:00
|
|
|
{
|
2022-12-05 21:01:26 +08:00
|
|
|
// 如果键按下,对应键的key press状态置1,否则为0
|
2022-12-08 12:08:49 +08:00
|
|
|
rc_ctrl[TEMP].key[KEY_PRESS][j] = rc_ctrl[TEMP].key_temp & i;
|
|
|
|
// 如果当前按下且上一次没按下,切换按键状态.一些模式要通过按键状态而不是按键是否按下来确定(实际上是大部分)
|
|
|
|
rc_ctrl[TEMP].key[KEY_STATE][j] = rc_ctrl[TEMP].key[KEY_PRESS][j] && !rc_ctrl[1].key[KEY_PRESS][j];
|
2022-12-05 21:01:26 +08:00
|
|
|
// 检查是否有组合键按下
|
2022-12-08 12:08:49 +08:00
|
|
|
if (rc_ctrl[TEMP].key_temp & 0x0001u << Key_Shift) // 按下ctrl
|
|
|
|
rc_ctrl[TEMP].key[KEY_PRESS_WITH_SHIFT][j] = rc_ctrl[TEMP].key_temp & i;
|
|
|
|
if (rc_ctrl[TEMP].key_temp & 0x0001u << Key_Ctrl) // 按下shift
|
|
|
|
rc_ctrl[TEMP].key[KEY_PRESS_WITH_CTRL][j] = rc_ctrl[TEMP].key_temp & i;
|
2022-10-20 17:13:02 +08:00
|
|
|
}
|
|
|
|
|
2022-12-06 22:58:42 +08:00
|
|
|
// 位域的按键值解算,直接memcpy即可,注意小端低字节在前,即lsb在第一位
|
2022-12-08 12:08:49 +08:00
|
|
|
// *(uint16_t *)&rc_ctrl[TEMP].key_test[KEY_PRESS] = (uint16_t)(sbus_buf[14] | (sbus_buf[15] << 8));
|
|
|
|
// *(uint16_t *)&rc_ctrl[TEMP].key_test[KEY_STATE] = *(uint16_t *)&rc_ctrl[TEMP].key_test[KEY_PRESS] & ~(*(uint16_t *)&(rc_ctrl[1].key_test[KEY_PRESS]));
|
|
|
|
// if (rc_ctrl[TEMP].key_test[KEY_PRESS].ctrl)
|
|
|
|
// rc_ctrl[TEMP].key_test[KEY_PRESS_WITH_CTRL] = rc_ctrl[TEMP].key_test[KEY_PRESS];
|
|
|
|
// if (rc_ctrl[TEMP].key_test[KEY_PRESS].shift)
|
|
|
|
// rc_ctrl[TEMP].key_test[Key_Shift] = rc_ctrl[TEMP].key_test[KEY_PRESS];
|
2022-10-20 17:13:02 +08:00
|
|
|
}
|
|
|
|
|
2022-11-01 22:32:15 +08:00
|
|
|
/**
|
|
|
|
* @brief protocol resolve callback
|
|
|
|
* this func would be called when usart3 idle interrupt happens
|
|
|
|
* 对sbus_to_rc的简单封装
|
|
|
|
*
|
|
|
|
*/
|
2022-12-06 22:58:42 +08:00
|
|
|
static void RemoteControlRxCallback()
|
2022-11-01 22:32:15 +08:00
|
|
|
{
|
2022-12-05 21:01:26 +08:00
|
|
|
sbus_to_rc(rc_usart_instance->recv_buff);
|
2022-11-01 22:32:15 +08:00
|
|
|
}
|
|
|
|
|
2022-12-05 21:01:26 +08:00
|
|
|
RC_ctrl_t *RemoteControlInit(UART_HandleTypeDef *rc_usart_handle)
|
2022-11-01 22:32:15 +08:00
|
|
|
{
|
2022-11-28 17:54:07 +08:00
|
|
|
USART_Init_Config_s conf;
|
2022-12-06 22:58:42 +08:00
|
|
|
conf.module_callback = RemoteControlRxCallback;
|
2022-11-28 17:54:07 +08:00
|
|
|
conf.usart_handle = rc_usart_handle;
|
|
|
|
conf.recv_buff_size = REMOTE_CONTROL_FRAME_SIZE;
|
|
|
|
rc_usart_instance = USARTRegister(&conf);
|
2022-12-09 18:25:35 +08:00
|
|
|
return (RC_ctrl_t*)&rc_ctrl;
|
2022-11-01 22:32:15 +08:00
|
|
|
}
|