#include "remote_control.h" #include "string.h" #include "bsp_usart.h" #include "memory.h" #include "stdlib.h" #include "daemon.h" #define RC_MOUSE_SMOOTH_COEF 0.9 // 鼠标平滑滤波器的阶数 #define REMOTE_CONTROL_FRAME_SIZE 18u // 遥控器接收的buffer大小 // 遥控器数据 static RC_ctrl_t rc_ctrl[2]; //[0]:当前数据TEMP,[1]:上一次的数据LAST.用于按键持续按下和切换的判断 static uint8_t rc_init_flag = 0; // 遥控器初始化标志位 // 遥控器拥有的串口实例,因为遥控器是单例,所以这里只有一个,就不封装了 static USARTInstance *rc_usart_instance; static DaemonInstance *rc_daemon_instance; /** * @brief 矫正遥控器摇杆的值,超过660或者小于-660的值都认为是无效值,置0 * */ static void RectifyRCjoystick() { for (uint8_t i = 0; i < 5; ++i) { if (abs(*(&rc_ctrl[TEMP].rc.rocker_l_ + i)) > 660) *(&rc_ctrl[TEMP].rc.rocker_l_ + i) = 0; } } /** * @brief remote control protocol resolution * @param[in] sbus_buf: raw data point * @param[out] rc_ctrl: remote control data struct point * @retval none */ static void sbus_to_rc(const uint8_t *sbus_buf) { // 摇杆,直接解算时减去偏置 rc_ctrl[TEMP].rc.rocker_r_ = ((sbus_buf[0] | (sbus_buf[1] << 8)) & 0x07ff) - RC_CH_VALUE_OFFSET; //!< Channel 0 rc_ctrl[TEMP].rc.rocker_r1 = (((sbus_buf[1] >> 3) | (sbus_buf[2] << 5)) & 0x07ff) - RC_CH_VALUE_OFFSET; //!< Channel 1 rc_ctrl[TEMP].rc.rocker_l_ = (((sbus_buf[2] >> 6) | (sbus_buf[3] << 2) | (sbus_buf[4] << 10)) & 0x07ff) - RC_CH_VALUE_OFFSET; //!< Channel 2 rc_ctrl[TEMP].rc.rocker_l1 = (((sbus_buf[4] >> 1) | (sbus_buf[5] << 7)) & 0x07ff) - RC_CH_VALUE_OFFSET; //!< Channel 3 rc_ctrl[TEMP].rc.dial = ((sbus_buf[16] | (sbus_buf[17] << 8)) & 0x07FF) - RC_CH_VALUE_OFFSET; // 左侧拨轮 RectifyRCjoystick(); // 开关,0左1右 rc_ctrl[TEMP].rc.switch_right = ((sbus_buf[5] >> 4) & 0x0003); //!< Switch right rc_ctrl[TEMP].rc.switch_left = ((sbus_buf[5] >> 4) & 0x000C) >> 2; //!< Switch left // 鼠标解析 rc_ctrl[TEMP].mouse.x = (float)(sbus_buf[6] | (sbus_buf[7] << 8))*RC_MOUSE_SMOOTH_COEF + (1-RC_MOUSE_SMOOTH_COEF)*(float)(rc_ctrl[LAST].mouse.x); //!< Mouse X axis rc_ctrl[TEMP].mouse.y = (float)(sbus_buf[8] | (sbus_buf[9] << 8))*RC_MOUSE_SMOOTH_COEF + (1-RC_MOUSE_SMOOTH_COEF)*(float)(rc_ctrl[TEMP].mouse.y); //!< Mouse Y 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 ? // 位域的按键值解算,直接memcpy即可,注意小端低字节在前,即lsb在第一位,msb在最后. 尚未测试 *(uint16_t *)&rc_ctrl[TEMP].key[KEY_PRESS] = (uint16_t)(sbus_buf[14] | (sbus_buf[15] << 8)); if (rc_ctrl[TEMP].key[KEY_PRESS].ctrl) rc_ctrl[TEMP].key[KEY_PRESS_WITH_CTRL] = rc_ctrl[TEMP].key[KEY_PRESS]; else memset(&rc_ctrl[TEMP].key[KEY_PRESS_WITH_CTRL], 0, sizeof(Key_t)); if (rc_ctrl[TEMP].key[KEY_PRESS].shift) rc_ctrl[TEMP].key[KEY_PRESS_WITH_SHIFT] = rc_ctrl[TEMP].key[KEY_PRESS]; else memset(&rc_ctrl[TEMP].key[KEY_PRESS_WITH_SHIFT], 0, sizeof(Key_t)); for (uint32_t i = 0, j = 0x1; i < 16; j <<= 1, i++) { if (((*(uint16_t *)&rc_ctrl[TEMP].key[KEY_PRESS] & j) == j) && ((*(uint16_t *)&rc_ctrl[1].key[KEY_PRESS] & j) == 0) && ((*(uint16_t *)&rc_ctrl[TEMP].key[KEY_PRESS_WITH_CTRL] & j) != j) && ((*(uint16_t *)&rc_ctrl[TEMP].key[KEY_PRESS_WITH_SHIFT] & j) != j)) { rc_ctrl[TEMP].key_count[KEY_PRESS][i]++; if (rc_ctrl[TEMP].key_count[KEY_PRESS][i] >= 240) { rc_ctrl[TEMP].key_count[KEY_PRESS][i] = 0; } } if (((*(uint16_t *)&rc_ctrl[TEMP].key[KEY_PRESS_WITH_CTRL] & j) == j) && ((*(uint16_t *)&rc_ctrl[1].key[KEY_PRESS_WITH_CTRL] & j) == 0)) { rc_ctrl[TEMP].key_count[KEY_PRESS_WITH_CTRL][i]++; if (rc_ctrl[TEMP].key_count[KEY_PRESS_WITH_CTRL][i] >= 240) { rc_ctrl[TEMP].key_count[KEY_PRESS_WITH_CTRL][i] = 0; } } if (((*(uint16_t *)&rc_ctrl[TEMP].key[KEY_PRESS_WITH_SHIFT] & j) == j) && ((*(uint16_t *)&rc_ctrl[1].key[KEY_PRESS_WITH_SHIFT] & j) == 0)) { rc_ctrl[TEMP].key_count[KEY_PRESS_WITH_SHIFT][i]++; if (rc_ctrl[TEMP].key_count[KEY_PRESS_WITH_SHIFT][i] >= 240) { rc_ctrl[TEMP].key_count[KEY_PRESS_WITH_SHIFT][i] = 0; } } } memcpy(&rc_ctrl[1], &rc_ctrl[TEMP], sizeof(RC_ctrl_t)); // 保存上一次的数据,用于按键持续按下和切换的判断 } /** * @brief 对sbus_to_rc的简单封装,用于注册到bsp_usart的回调函数中 * */ static void RemoteControlRxCallback() { DaemonReload(rc_daemon_instance); // 先喂狗 sbus_to_rc(rc_usart_instance->recv_buff); // 进行协议解析 } /** * @brief 遥控器离线的回调函数,注册到守护进程中,串口掉线时调用 * */ static void RCLostCallback(void *id) { USARTServiceInit(rc_usart_instance); // 尝试重新启动接收 } RC_ctrl_t *RemoteControlInit(UART_HandleTypeDef *rc_usart_handle) { USART_Init_Config_s conf; conf.module_callback = RemoteControlRxCallback; conf.usart_handle = rc_usart_handle; conf.recv_buff_size = REMOTE_CONTROL_FRAME_SIZE; rc_usart_instance = USARTRegister(&conf); // 进行守护进程的注册,用于定时检查遥控器是否正常工作 Daemon_Init_Config_s daemon_conf = { .reload_count = 10, // 100ms未收到数据视为离线,遥控器的接收频率实际上是1000/14Hz(大约70Hz) .callback = RCLostCallback, .owner_id = NULL, // 只有1个遥控器,不需要owner_id }; rc_daemon_instance = DaemonRegister(&daemon_conf); rc_init_flag = 1; return (RC_ctrl_t *)&rc_ctrl; } uint8_t RemoteControlIsOnline() { if (rc_init_flag) return DaemonIsOnline(rc_daemon_instance); return 0; }