STM32 解析futaba S-bus协议

转自:http://www.eefocus.com/taogashi/blog/14-05/303577_f140d.html

S-bus为futaba使用的串行通信协议。实际上为串口通信。但是有几点需要注意:

1.         在大端小端上,网上资料都说的不是很清楚;

2.         跟TTL串口信号相比,S-bus的逻辑电平是反的,需用如下电路对电平反相,再借到串口接收的Rx管脚就可以了;

STM32 解析futaba S-bus协议_第1张图片

一、协议说明:

串口配置为波特率100kbps,8位数据,偶校验(even),2位停止位,无流控。

链接https://mbed.org/users/Digixx/notebook/futaba-s-bus-controlled-by-mbed/说明了S-bus帧格式。每帧25个字节,按照如下顺序排列:

[startbyte] [data1] [data2] .... [data22] [flags][endbyte]

 

起始字节startbyte = 11110000b (0xF0),但实际上用STM32(据说ARM核)收到的是0x0F。中间22个字节就是16个通道的数据了,为什么是16个通道?因为22x8=11x16,每个通道用11bit表示,范围是0-2047。不信看波形图:

STM32 解析futaba S-bus协议_第2张图片

什么,还看不清?

STM32 解析futaba S-bus协议_第3张图片

两帧之间的时间间隔4ms(高速模式),约7ms一帧。

STM32 解析futaba S-bus协议_第4张图片

STM32 解析futaba S-bus协议_第5张图片

基本而言,data1为ch1的低8位,data2的低3位为ch1的高三位,data2的高5位是ch2的低5位,data3的低6位是ch2的高6位,以此类推,如下图所示:

STM32 解析futaba S-bus协议_第6张图片

flags的结构如下所示:

flags

bit7 = ch17 = digital channel (0x80)
bit6 = ch18 = digital channel (0x40)
bit5 = Frame lost, equivalent red LED on receiver (0x20)
bit4 = failsafe activated (0x10)
bit3 = n/a
bit2 = n/a
bit1 = n/a
bit0 = n/a

 

 

endbyte0x00

一、程序实现:

在STM32中的具体实现,除了如上述内容配置串口参数,还需要写好中断函数,写好解析函数。思路很简单,利用间隔时间来区分两帧,收到一帧数据后,做如下检查:

1.       字节数够不够?

2.       第一个字节是不是0x0f?

3.       最后一个字节是不是0x00?

4.       检查flag中的标志位

举个中断函数栗子:

void UART4_IRQHandler(void)

{

         static uint8_t byteCNT = 0;

        

         static uint32_t lastTime = 0;

         uint32_t curTime;

         uint32_t interval = 0;

        

         HAL_NVIC_ClearPendingIRQ(UART4_IRQn);

 

         //如果时间间隔大于3毫秒,则认为是新的一帧

         if(lastTime == 0)

         {

                   curTime = HAL_GetTick();

                   lastTime = curTime;

         }

         else

         {

                  curTime = HAL_GetTick();

                   interval = curTime - lastTime;

                   lastTime = curTime;

                  

                   if(interval >= 3)

                   {

                            if(byteCNT == 25 && uart4_cache1[0] == 0x0f && uart4_cache1[24] == 0x00)

                            {

                                     rc_captured = 0;

                                     memcpy(uart4_cache2, uart4_cache1, byteCNT);

                                     rc_captured = 1;

                            }

                            byteCNT = 0;

                   }

         }

        

         if(RESET != __HAL_UART_GET_FLAG(&huart4, UART_FLAG_ORE))

         {

                   __HAL_UART_CLEAR_FLAG(&huart4, UART_FLAG_ORE);

                   uart4_cache1[byteCNT++] = huart4.Instance->DR;

         }

        

         if(RESET != __HAL_UART_GET_FLAG(&huart4, UART_FLAG_RXNE))

         {

                   uart4_cache1[byteCNT++] = huart4.Instance->DR;

         }

 

}

你可能感兴趣的:(硬件相关)