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"
2023-01-01 17:32:22 +08:00
# include "stdlib.h"
2023-01-04 21:26:11 +08:00
# include "daemon.h"
2022-10-20 17:13:02 +08:00
2023-04-23 14:51:02 +08:00
# define RC_MOUSE_SMOOTH_COEF 0.9 // 鼠标平滑滤波器的阶数
2022-11-01 22:32:15 +08:00
# define REMOTE_CONTROL_FRAME_SIZE 18u // 遥控器接收的buffer大小
// 遥控器数据
2023-02-22 13:53:25 +08:00
static RC_ctrl_t rc_ctrl [ 2 ] ; //[0]:当前数据TEMP,[1]:上一次的数据LAST.用于按键持续按下和切换的判断
static uint8_t rc_init_flag = 0 ; // 遥控器初始化标志位
2023-01-01 17:32:22 +08:00
// 遥控器拥有的串口实例,因为遥控器是单例,所以这里只有一个,就不封装了
2022-12-05 21:01:26 +08:00
static USARTInstance * rc_usart_instance ;
2023-01-04 21:26:11 +08:00
static DaemonInstance * rc_daemon_instance ;
2022-12-11 14:59:45 +08:00
/**
2023-01-03 22:53:49 +08:00
* @ brief 矫 正 遥 控 器 摇 杆 的 值 , 超 过 660 或 者 小 于 - 660 的 值 都 认 为 是 无 效 值 , 置 0
2022-12-11 20:48:24 +08:00
*
2022-12-11 14:59:45 +08:00
*/
static void RectifyRCjoystick ( )
{
2022-12-13 19:40:03 +08:00
for ( uint8_t i = 0 ; i < 5 ; + + i )
2022-12-11 14:59:45 +08:00
{
2023-01-01 17:32:22 +08:00
if ( abs ( * ( & rc_ctrl [ TEMP ] . rc . rocker_l_ + i ) ) > 660 )
* ( & rc_ctrl [ TEMP ] . rc . rocker_l_ + i ) = 0 ;
2022-12-11 14:59:45 +08:00
}
}
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
*/
2023-04-14 18:17:46 +08:00
2023-01-01 17:32:22 +08:00
static void sbus_to_rc ( const uint8_t * sbus_buf )
2022-10-20 17:13:02 +08:00
{
2023-03-27 22:03:35 +08:00
2022-12-08 17:36:12 +08:00
// 摇杆,直接解算时减去偏置
2022-12-11 20:48:24 +08:00
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 ; // 左侧拨轮
2022-12-11 14:59:45 +08:00
RectifyRCjoystick ( ) ;
2022-12-05 21:01:26 +08:00
// 开关,0左1右
2023-01-03 22:53:49 +08:00
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
2022-12-05 21:01:26 +08:00
// 鼠标解析
2023-04-23 14:51:02 +08:00
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
2022-12-08 12:08:49 +08:00
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
2023-03-27 22:03:35 +08:00
// 位域的按键值解算,直接memcpy即可,注意小端低字节在前,即lsb在第一位,msb在最后. 尚未测试
* ( uint16_t * ) & rc_ctrl [ TEMP ] . key [ KEY_PRESS ] = ( uint16_t ) ( sbus_buf [ 14 ] | ( sbus_buf [ 15 ] < < 8 ) ) ;
2022-12-05 21:01:26 +08:00
2023-03-27 22:03:35 +08:00
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 + + )
2022-10-20 17:13:02 +08:00
{
2023-03-27 22:03:35 +08:00
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 ;
}
}
2022-10-20 17:13:02 +08:00
}
2023-03-27 22:03:35 +08:00
memcpy ( & rc_ctrl [ 1 ] , & rc_ctrl [ TEMP ] , sizeof ( RC_ctrl_t ) ) ; // 保存上一次的数据,用于按键持续按下和切换的判断
2022-10-20 17:13:02 +08:00
}
2022-11-01 22:32:15 +08:00
/**
2023-01-04 21:26:11 +08:00
* @ brief 对 sbus_to_rc的简单封装 , 用 于 注 册 到 bsp_usart的回调函数中
2023-02-22 13:53:25 +08:00
*
2022-11-01 22:32:15 +08:00
*/
2022-12-06 22:58:42 +08:00
static void RemoteControlRxCallback ( )
2022-11-01 22:32:15 +08:00
{
2023-02-22 13:53:25 +08:00
DaemonReload ( rc_daemon_instance ) ; // 先喂狗
2023-01-04 21:26:11 +08:00
sbus_to_rc ( rc_usart_instance - > recv_buff ) ; // 进行协议解析
}
/**
2023-03-23 18:22:24 +08:00
* @ brief 遥 控 器 离 线 的 回 调 函 数 , 注 册 到 守 护 进 程 中 , 串 口 掉 线 时 调 用
2023-02-22 13:53:25 +08:00
*
2023-01-04 21:26:11 +08:00
*/
2023-03-23 18:22:24 +08:00
static void RCLostCallback ( void * id )
2023-01-04 21:26:11 +08:00
{
// @todo 遥控器丢失的处理
2023-03-23 18:22:24 +08:00
USARTServiceInit ( rc_usart_instance ) ; // 尝试重新启动接收
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 ) ;
2023-01-04 21:26:11 +08:00
// 进行守护进程的注册,用于定时检查遥控器是否正常工作
// @todo 当前守护进程直接在这里注册,后续考虑将其封装到遥控器的初始化函数中,即可以让用户决定reload_count的值(是否有必要?)
Daemon_Init_Config_s daemon_conf = {
2023-03-23 18:22:24 +08:00
. reload_count = 10 , // 100ms未收到数据视为离线,遥控器的接收频率实际上是1000/14Hz(大约70Hz)
. callback = RCLostCallback ,
. owner_id = NULL , // 只有1个遥控器,不需要owner_id
2023-01-04 21:26:11 +08:00
} ;
rc_daemon_instance = DaemonRegister ( & daemon_conf ) ;
2023-02-22 13:53:25 +08:00
rc_init_flag = 1 ;
2022-12-11 20:48:24 +08:00
return ( RC_ctrl_t * ) & rc_ctrl ;
2023-02-22 13:53:25 +08:00
}
2023-05-06 16:23:52 +08:00
uint8_t RemoteControlIsOnline ( )
2023-02-22 13:53:25 +08:00
{
if ( rc_init_flag )
return DaemonIsOnline ( rc_daemon_instance ) ;
return 0 ;
2022-11-01 22:32:15 +08:00
}