一文搞定MAVLINK软件协议

2023-12-18 04:58

本文主要是介绍一文搞定MAVLINK软件协议,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

转载:https://mp.weixin.qq.com/s/iGURlSS7V-5iBCEtgpzT7w

一文搞定MAVLINK软件协议

原创 L君 TBUS社区 2019-11-06

本文纯属资深程序猿个人观点,旨在让大家从不同的角度理解MAVLINK协议,文中包含的一些玩笑话,大家不要当真,看着玩玩儿呗~


 

图片

搞开源无人机的朋友最耳熟能详的莫过于它的通讯协议MAVLINK了。

 

Mavlink----一个又好气又好笑的名词,仿佛自带光环,它一出场,就会附带两个小弟:ros、mavros。

图片

网络上mavros免费的课程一大堆,mavlink的倒是少之又少,犹如,未过门的媳妇,见不得人啊图片

 

 

很多人都会形成一种观念,难道是因为mavros要简单些?今天L君要告诉你们,其实直接使用mavlink比使用mavros简单的多啦。

 

对于一个老程序员来说,理解数据结构犹如吃饭,但是理解框架犹如登天!

 

先来好好说说这个大哥和两小弟的关系。

ROS

robot operation system,机器人操作系统。说是操作系统,其实跟硬件毫无关系,完全是用一大堆软件堆砌起来的彻头彻尾的----“骗局”。没有mmu、smp、cache,它唯一的功能就是----通讯。把所有的逻辑性代码转换为通讯数据,比如一个函数调用完成的事情,它会拆分成----publish数据、对端接受数据、执行代码。这里看起来,ros跟mavlink似乎毫无关系,直到说到mavros。

 

MAVROS

一个在ros下的mavlink软件适配包,它的作用其实就是将mavlink协议放到ros框架里去解析,统一使用ros框架,publish、subscribe进行操作。

 

MAVLINK

一种软件协议,说白了,就是一大堆的struct,做过服务器的朋友应该会搞懂,其实就是封包格式啦,只是mavlink由于介质不唯一,所以加了CRC效验而已。

 

 

那么从这里我们已经看出了端倪:学mavros其实是无端的增加了学习成本呀,本来我收发收发数据就完事儿了的东西,我为啥一定要去学一套ros呢?

 

而且就程序员的原则而言,我们学习时,一定要看到本质,用mavros的时候,中间经过了一个黑盒,里面到底干了什么,我们根本不知道呀。

 

L君以前也吃过很多亏,用mavros反跟ardupilot代码,发现,我publish出去的东西,人家ardupilot收到的压根儿就不一样!坐标系从NED变成了ENU,然而heading的坐标系还是NED。有些结构体少了几个参数,有的甚至比mavlink原来的还多了几个。

 

不仅mavros在给你制造着麻烦,ros也时不时给你制造麻烦:

CMakeList怎么改才能用PCL库?

TMD怎么有的包必须用catkin_make_isolated才能编译?

不是说好的catkin_make吗?

为什么我publish出去没效果啊?

catkin_make一分钟,看一次效果3秒。MD怎么用gdb啊?

我好想单步走啊,printf调试起来内心在滴血- -。。。

 

不!!!这简直就是地狱!我要砍掉它,这不是人干的事儿。我忍受着在linux下用记事本写代码的痛苦,你还给我搞这些幺蛾子?

 

算了算了,框架这玩意儿不是我等卑微程序员能玩儿的东西,我们还是不怕苦不怕累的啃啃数据流吧。

 

当我正打算回到小黑屋啃mavlink数据协议的时候,我发现了

图片

 

一个老程序员的直觉,既然mavlink是以modules存在的,那么一定在那神奇的网

图片

好吧好吧,其实我早就知道了,人家mavlink为了用户方便,sdk早就给我们准备好啦,根本不需要我们自己去做解析,我们快速的git clone下来:

图片

然后我们执行mavgenerate.py脚本:

图片

 

这里我们XML选择:message_definitions/v1.0/common.xml(其中还有ardupilotmega.xml表示支持APM固件扩展协议,以及其他无人机固件的协议,common表示通用协议,所有固件都支持。)

 

Out随意选择一个要生成代码的地方

 

语言选择c语言

 

协议簇2.0

 

点击生成:

图片

图片

 

一个通用的mavlink库就生成好啦

图片

我们现在只需要在编程的时候包含这些头文件就可以实现我们的mavlink通讯了。

 

 

现在赶快连上ALICE飞控,打开visual studio:

图片

 

将我们刚才生成的mav_inc文件夹整个丢进新创建的项目:

图片

 

并在解决方案资源管理器中添加mavlink的头文件和一个main.cpp(注意文件夹需要自己添加哦):

图片

 

 

写入如下代码:

#include <math.h>
#include <thread>
#include <mutex>
#include "mav_inc/mavlink_types.h"
#define MAVLINK_USE_CONVENIENCE_FUNCTIONS
#define MAVLINK_SEND_UART_BYTES(a, b, c) send(a, b, c, 0)
mavlink_system_t mavlink_system = { 1,1 };
#include "mav_inc/common/mavlink.h"
int main(void){
        Socket sock;
        sock.Create();
        while (sock.Connect((char*)"127.0.0.1", 1244) != 0) 
       { 
               printf("connect error.waiting for 1 second...\n");
                usleep(1000000);
        } 
        while (1)
        { 
               mavlink_message_t msg; 
               mavlink_status_t status;
                uint8_t buf[1024]; 
               int recv_len = sock.Receive((char*)buf, 1024); 
               for (int i = 0; i < recv_len; i++) 
               { 
                       if (mavlink_parse_char(0, buf[i], &msg, &status))
                        { 
                               switch (msg.msgid)  
                              {  
                              case MAVLINK_MSG_ID_ATTITUDE: 
                                       mavlink_attitude_t attitude; 
                                       mavlink_msg_attitude_decode(&msg, &attitude);
                                        printf("rpy=%f,%f,%f\n", attitude.roll, attitude.pitch, attitude.yaw);  
                                      break; 
                               case MAVLINK_MSG_ID_HEARTBEAT: 
                                       mavlink_heartbeat_t heart; 
                                       mavlink_msg_heartbeat_decode(&msg, &heart);
                                        if (heart.base_mode) 
                                       {  
                                              bool armed = (heart.base_mode & MAV_MODE_FLAG_SAFETY_ARMED); 
                                               bool guided = (heart.base_mode & MAV_MODE_FLAG_GUIDED_ENABLED); 
                                               printf("armed=%d,guided=%d\n", armed, guided); 
                                       }  
                                      break; 
                               default: 
                                       break; 
                               } 
                       }
                } 
       }
         return 0;
}

 

 

编译并执行:

图片

 

可以看到我们已经从mavlink取到当前飞机的姿态信息啦!有着完美的代码变色、自动补全、智能分析排错的visual studio是不是已经让你享受起开发的乐趣了呢?再配合上Alice板卡的wifi网卡功能,你甚至都可以不需要任何辅助,直接实地调试你的程序哟(被逼着插进来的硬核广告图片)。

 

现在我们再回过头来开看看代码。

 

class Socket这个类就不必说了,他是一个tcp协议的公用库类,实现网络通讯功能,不懂的自己百度一下。Alice的ardupilot固件默认使用2号telem作为本地tcp通讯端口,方便开发mavlink外机控制程序。 现在我们先来看main函数里面做了什么,寥寥几行,基本就实现了通讯了:

//实例化网络对象
Socket sock;
//初始化网络对象
sock.Create();
        //循环连接本地ip端口1244,alice开机自启动ardupilot使用tcp协议侦听1244号端口发送mavlink 
         while (sock.Connect((char*)"127.0.0.1", 1244) != 0) 
         {
                //如果连接失败,打印消息,等待1s重试。
                printf("connect error.waiting for 1 second...\n");
                usleep(1000000);
        } 
        while (1)
       { 
              mavlink_message_t msg; 
               mavlink_status_t status; 
               uint8_t buf[1024];//从连接上的tcpsocket收包存到buf中
              int recv_len = sock.Receive((char*)buf, 1024); 
               for (int i = 0; i < recv_len; i++) 
               {
                      //这里其实就是最重要最重要的地方了,mavlink_parse_char函数
                      //第一个参数默认0就行
                      //第二个参数是一个字节,也就是我们收一次包,比如有20个自己,就循环
                      //调用20次mavlink_parse_char,把每个字节依次放入
      //第三个参数,返回的真实mavlink_raw包
                    //第四个参数,返回一些正在解析的状态
                     //如果mavlink_parse_char返回true,表示已经真正收到了一个包,并已
                     //填充到了msg中。但是我们还不能直接使用这个包,应为现在还是raw形式
                     //下面还会将它转换成结构体形式!
                        if (mavlink_parse_char(0, buf[i], &msg, &status)) 
                       {
                                //这个开关语句就是判断收到包的ID号到底是多少啦
                                //raw形式的msg只能取到ID,无法取到内容 
                               switch (msg.msgid) 
                               {
                                case MAVLINK_MSG_ID_ATTITUDE:
                                         //判断完ID以后我们就可以直接开始调用函数解析啦
                                        //这里解析的是attitude那么就调用mavlink_msg_attitude_decode
                                        //如果是local_position就是mavlink_msg_local_position_decode
                                        //具体用什么decode你们可以直接到头文件中去查
                                       //mavlink头文件名几乎都是自注释的
                                        mavlink_attitude_t attitude;
                                        mavlink_msg_attitude_decode(&msg, &attitude);
                                        printf("rpy=%f,%f,%f\n", attitude.roll, attitude.pitch, attitude.yaw);
                                        break;
                                case MAVLINK_MSG_ID_HEARTBEAT:
                                        mavlink_heartbeat_t heart;
                                        //那么这里id是心跳,那么解析函数就是mavlink_msg_heartbeat_decode啦
                                        mavlink_msg_heartbeat_decode(&msg, &heart);
                                        if (heart.base_mode) 
                                       { 
                                               bool armed = (heart.base_mode & MAV_MODE_FLAG_SAFETY_ARMED); 
                                               bool guided = (heart.base_mode & MAV_MODE_FLAG_GUIDED_ENABLED);
                                                printf("armed=%d,guided=%d\n", armed, guided);
                                        }
                                        break; 
                               default: 
                                       break; 
                               } 
                      }
                } 
       }

 

那么,这篇就讲到这里啦,看完全篇,是不是觉得mavlink开发其实是一件非常惬意的事情呢,用对了工具,用对了硬件,才能真正方便我们快速解决问题。

 

本文选自TBUS论坛_L君原创文章

 

这篇关于一文搞定MAVLINK软件协议的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/507111

相关文章

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

Ubuntu 怎么启用 Universe 和 Multiverse 软件源?

《Ubuntu怎么启用Universe和Multiverse软件源?》在Ubuntu中,软件源是用于获取和安装软件的服务器,通过设置和管理软件源,您可以确保系统能够从可靠的来源获取最新的软件... Ubuntu 是一款广受认可且声誉良好的开源操作系统,允许用户通过其庞大的软件包来定制和增强计算体验。这些软件

一文带你搞懂Nginx中的配置文件

《一文带你搞懂Nginx中的配置文件》Nginx(发音为“engine-x”)是一款高性能的Web服务器、反向代理服务器和负载均衡器,广泛应用于全球各类网站和应用中,下面就跟随小编一起来了解下如何... 目录摘要一、Nginx 配置文件结构概述二、全局配置(Global Configuration)1. w

Java如何接收并解析HL7协议数据

《Java如何接收并解析HL7协议数据》文章主要介绍了HL7协议及其在医疗行业中的应用,详细描述了如何配置环境、接收和解析数据,以及与前端进行交互的实现方法,文章还分享了使用7Edit工具进行调试的经... 目录一、前言二、正文1、环境配置2、数据接收:HL7Monitor3、数据解析:HL7Busines

MySQL数据库宕机,启动不起来,教你一招搞定!

作者介绍:老苏,10余年DBA工作运维经验,擅长Oracle、MySQL、PG、Mongodb数据库运维(如安装迁移,性能优化、故障应急处理等)公众号:老苏畅谈运维欢迎关注本人公众号,更多精彩与您分享。 MySQL数据库宕机,数据页损坏问题,启动不起来,该如何排查和解决,本文将为你说明具体的排查过程。 查看MySQL error日志 查看 MySQL error日志,排查哪个表(表空间

软件设计师备考——计算机系统

学习内容源自「软件设计师」 上午题 #1 计算机系统_哔哩哔哩_bilibili 目录 1.1.1 计算机系统硬件基本组成 1.1.2 中央处理单元 1.CPU 的功能 1)运算器 2)控制器 RISC && CISC 流水线控制 存储器  Cache 中断 输入输出IO控制方式 程序查询方式 中断驱动方式 直接存储器方式(DMA)  ​编辑 总线 ​编辑

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

【Linux】应用层http协议

一、HTTP协议 1.1 简要介绍一下HTTP        我们在网络的应用层中可以自己定义协议,但是,已经有大佬定义了一些现成的,非常好用的应用层协议,供我们直接使用,HTTP(超文本传输协议)就是其中之一。        在互联网世界中,HTTP(超文本传输协议)是一个至关重要的协议,他定义了客户端(如浏览器)与服务器之间如何进行通信,以交换或者传输超文本(比如HTML文档)。

免费也能高质量!2024年免费录屏软件深度对比评测

我公司因为客户覆盖面广的原因经常会开远程会议,有时候说的内容比较广需要引用多份的数据,我记录起来有一定难度,所以一般都用录屏工具来记录会议内容。这次我们来一起探索有什么免费录屏工具可以提高我们的工作效率吧。 1.福晰录屏大师 链接直达:https://www.foxitsoftware.cn/REC/  录屏软件录屏功能就是本职,这款录屏工具在录屏模式上提供了多种选项,可以选择屏幕录制、窗口

HomeBank:开源免费的个人财务管理软件

在个人财务管理领域,找到一个既免费又开源的解决方案并非易事。HomeBank&nbsp;正是这样一个项目,它不仅提供了强大的功能,还拥有一个活跃的社区,不断推动其发展和完善。 开源免费:HomeBank 是一个完全开源的项目,用户可以自由地使用、修改和分发。用户友好的界面:提供直观的图形用户界面,使得非技术用户也能轻松上手。数据导入支持:支持从 Quicken、Microsoft Money