多旋翼无人机

    返回首页    发表留言
本文作者:李德强
          第一节 SBUS协议
 
 

        今天我们来一起学习遥控器常见通讯协议S-BUS的解析方法。S-BUS其实是一种串口通信协议,采用100000的波特率,数据位点8bits,停止位点2bits,偶效验,即8E2的串口通信。但是S-BUS采用的是反向电平传输,也就是说,在S-BUS的发送端高低电平是反向的,协议中的所有高电平都被转换成低电平,协议中的所有低电平都被转换成高电平。所以在S-BUS的接收端需要增加一个高低电平反向器来进行电平反转,如下图:

            实际上,我们使用的Pixhawk飞控板上已经集成了这个反向器,所以对于使用Pixhawk的用户来说,可以忽略掉S-BUS的反向机制,但是对于其它没有集成S-BUS反向器的硬件平台上,就需要使用者增加一个反向器来处理数据,否则将无法读取协议数据。

        另外,100000的波特率并不是标准的波特率,这在一些只支持标准波特率的系统上无法实现,好在Nuttx支持自定义的波特率,我们可以通过对设备节点的配置实现波特率的设定。在Pixhawk IO上,S-BUS总线的设备节点为/dev/ttyS2,于是我们可以编写一个程序对这个串口节点的波特率进行配置:

void sbus_config(void)
{
	int sbus_fd = open("/dev/ttyS2", O_RDWR | O_NONBLOCK);
	if (sbus_fd < 0)
	{
		return;
	}
	struct termios t;		
	//设置100000波特率
	tcgetattr(sbus_fd, &t);
	cfsetspeed(&t, 100000);
	t.c_cflag |= (CSTOPB | PARENB);
	tcsetattr(sbus_fd, TCSANOW, &t);
}

        设置好波特率就可以对标准文件设备进行读取了,也就是通过read()函数来读取串口当中的数据。S-BUS协议在传输过程中还有两种不同的传输模式:

1)高速模式:数据发送周期为4ms,发送频率为250Hz;

2)低速模式:数据发送周期为14ms,发送频率为71.4Hz。

        S-BUS协议数据格式如下:

        需要注意的是S-BUS中用11bits来表示一个遥控器通道的数值,22个字节就可以表示16通道(8 × 22 = 11 ×16)。11个bit可以表示的数值范围为0~2047。例如:我们的遥控器前4个通道数值分别为200、300、400和500,其它通道都为0。它们的二进制数据分别为:

200:  000 1100 1000

300:  001 0010 1100

400:  001 1001 0000

500:  001 1111 0100

其它通道都为:0

        S-BUS的16个遥控器通道,每一个通道用11个bit表示,那么这16个通道的二进制数值拼接起来则为:

0001 1001 0000 0100 1011 0000 1100 1000 0001 1111 0100

[0F] 19 04 B0 C8 1F 40 …… 00 [00]

        实际上遥控器发送每一个通道的数值在200~1800之间,用来表示遥控器通道的所有数值。但是PX4飞控程序中有效的通道值范围通常是1000~2000,所以就需要将原始数值进行一次转换。接下来我们就来编写S-BUS协议解析的驱动程序:

#define SIZE_BUFF	(100)

//S-BUS协议中遥控器通道数值范围
#define SBUS_RANGE_MIN 200.0f
#define SBUS_RANGE_MAX 1800.0f
//PX4中使用的遥控器通道数值范围
#define SBUS_TARGET_MIN 1000.0f
#define SBUS_TARGET_MAX 2000.0f
//数值放大因子
#define SBUS_SCALE_FACTOR ((SBUS_TARGET_MAX - SBUS_TARGET_MIN) / (SBUS_RANGE_MAX - SBUS_RANGE_MIN))
//数值放大偏移量
#define SBUS_SCALE_OFFSET (int)(SBUS_TARGET_MIN - (SBUS_SCALE_FACTOR * SBUS_RANGE_MIN + 0.5f))

//S-BUS解析函数
int sbus_read_parse(int _fd, uint16_t *val)
{
	//读取遥控器通道数据
	uint8_t _buf[SIZE_BUFF];
	int len = read(_fd, _buf, SIZE_BUFF);
	if (len < 0)
	{
		return -1;
	}

    //略过协议包头、包尾、长度判断过程

	//按11bits解析遥控器通道
	val[0] = ((buff[ind + 1] | buff[ind + 2] << 8) & 0x07FF);
	val[1] = ((buff[ind + 2] >> 3 | buff[ind + 3] << 5) & 0x07FF);
	val[2] = ((buff[ind + 3] >> 6 | buff[ind + 4] << 2 | buff[ind + 5] << 10) & 0x07FF);
	val[3] = ((buff[ind + 5] >> 1 | buff[ind + 6] << 7) & 0x07FF);
	val[4] = ((buff[ind + 6] >> 4 | buff[ind + 7] << 4) & 0x07FF);
	val[5] = ((buff[ind + 7] >> 7 | buff[ind + 8] << 1 | buff[ind + 9] << 9) & 0x07FF);
	val[6] = ((buff[ind + 9] >> 2 | buff[ind + 10] << 6) & 0x07FF);
	val[7] = ((buff[ind + 10] >> 5 | buff[ind + 11] << 3) & 0x07FF);
	val[8] = ((buff[ind + 12] | buff[ind + 13] << 8) & 0x07FF);
	val[9] = ((buff[ind + 13] >> 3 | buff[ind + 14] << 5) & 0x07FF);
	val[10] = ((buff[ind + 14] >> 6 | buff[ind + 15] << 2 | buff[ind + 16] << 10) & 0x07FF);
	val[11] = ((buff[ind + 16] >> 1 | buff[ind + 17] << 7) & 0x07FF);
	val[12] = ((buff[ind + 17] >> 4 | buff[ind + 18] << 4) & 0x07FF);
	val[13] = ((buff[ind + 18] >> 7 | buff[ind + 19] << 1 | buff[ind + 20] << 9) & 0x07FF);
	val[14] = ((buff[ind + 20] >> 2 | buff[ind + 21] << 6) & 0x07FF);
	val[15] = ((buff[ind + 21] >> 5 | buff[ind + 22] << 3) & 0x07FF);

	//将原始数值转换到PX4所使用的范围
	for (int i = 0; i < 16; i++)
	{
		val[i] = (uint16_t)(val[i] * SBUS_SCALE_FACTOR + .5f) + SBUS_SCALE_OFFSET;
	}

	return 0;
}

        当然,我们需要在驱动程序中启动一个线程来读取并解析S-BUS协议,此进程将使用循环的方式调用int sbus_read_parse(int _fd, uint16_t *val)函数,之后将val数组中的内容使用input_rc.msg消息发布到uORB总线上为其它进程所使用。

 

 

    返回首页    返回顶部
  看不清?点击刷新

 

  Copyright © 2015-2023 问渠网 辽ICP备15013245号