当前位置:   article > 正文

通过mavros实现无人机航点飞行(mission模式)详细讲解_无人机通过mavros发布一系列航点

无人机通过mavros发布一系列航点

 通过mavros实现无人机航点飞行(mission模式)

垂起固定翼通过mavros实现航点飞行gazebo仿真演示与代码讲解20240524

mavros提供了mavros/mission/push这个service可以进行航点上传,mavros/mission/pull可以进行航点下载,mavros/mission/clear可以进行航点清除。
http://wiki.ros.org/mavros
 

输入图片说明


也可以借助于mavwp命令实现一些航点相关操作,这个比较小众,这里不详细展开。
 

输入图片说明


http://docs.ros.org/en/api/mavros_msgs/html/srv/WaypointPush.html
 

输入图片说明


WaypointPush类型里面,start_index值为0代表上传更新全部航点,似乎可以只更新部分航点,笔者没有这么用过。
mavros_msgs/Waypoint[] waypoints是一个个mavros_msgs/Waypoint类型组成的,mavros_msgs/Waypoint类型定义如下
http://docs.ros.org/en/api/mavros_msgs/html/msg/Waypoint.html
 

输入图片说明


全部内容如下

  1. # Waypoint.msg
  2. #
  3. # ROS representation of MAVLink MISSION_ITEM
  4. # See mavlink documentation
  5. # see enum MAV_FRAME
  6. uint8 frame
  7. uint8 FRAME_GLOBAL = 0
  8. uint8 FRAME_LOCAL_NED = 1
  9. uint8 FRAME_MISSION = 2
  10. uint8 FRAME_GLOBAL_REL_ALT = 3
  11. uint8 FRAME_LOCAL_ENU = 4
  12. uint8 FRAME_GLOBAL_INT = 5
  13. uint8 FRAME_GLOBAL_RELATIVE_ALT_INT = 6
  14. uint8 FRAME_LOCAL_OFFSET_NED = 7
  15. uint8 FRAME_BODY_NED = 8
  16. uint8 FRAME_BODY_OFFSET_NED = 9
  17. uint8 FRAME_GLOBAL_TERRAIN_ALT = 10
  18. uint8 FRAME_GLOBAL_TERRAIN_ALT_INT = 11
  19. uint8 FRAME_BODY_FRD = 12
  20. uint8 FRAME_RESERVED_13 = 13
  21. uint8 FRAME_RESERVED_14 = 14
  22. uint8 FRAME_RESERVED_15 = 15
  23. uint8 FRAME_RESERVED_16 = 16
  24. uint8 FRAME_RESERVED_17 = 17
  25. uint8 FRAME_RESERVED_18 = 18
  26. uint8 FRAME_RESERVED_19 = 19
  27. uint8 FRAME_LOCAL_FRD = 20
  28. uint8 FRAME_LOCAL_FLU = 21
  29. # see enum MAV_CMD and CommandCode.msg
  30. uint16 command
  31. bool is_current
  32. bool autocontinue
  33. # meaning of this params described in enum MAV_CMD
  34. float32 param1
  35. float32 param2
  36. float32 param3
  37. float32 param4
  38. float64 x_lat
  39. float64 y_long
  40. float64 z_alt

注释里有说明对应的MISSION_ITEM这个mavlink消息,MISSION_ITEM在2020年后被MISSION_ITEM_INT消息取代,所以实际发的MISSION_ITEM_INT消息,嵌套有MAV_CMD消息。
注意mavros消息类型里面经度和纬度是float64类型,我们就按照浮点数赋值即可,虽然MISSION_ITEM_INT消息里经度和纬度是int32类型,这个mavros自己会做相应转换。

输入图片说明

坐标系frame不同取值的含义如下

  1. FRAME_GLOBAL (0):使用全局坐标系(经度、纬度和海拔)来表示航点的位置。
  2. FRAME_LOCAL_NED (1):使用本地坐标系(北、东、地下偏移)来表示航点的位置。这个本地坐标系是以起始位置作为原点建立的。
  3. FRAME_MISSION (2):使用任务坐标系来表示航点的位置,任务坐标系是相对于导航目标点的坐标系。
  4. FRAME_GLOBAL_REL_ALT (3):使用全局坐标系(经度、纬度和相对海拔)来表示航点的位置。
  5. FRAME_LOCAL_ENU (4):使用本地坐标系(东、北、向上偏移)来表示航点的位置。
  6. FRAME_GLOBAL_INT (5):与FRAME_GLOBAL相似,但使用带有整数解析度的全局坐标。
  7. FRAME_GLOBAL_RELATIVE_ALT_INT (6):与FRAME_GLOBAL_REL_ALT相似,但使用带有整数解析度的全局坐标。
  8. FRAME_LOCAL_OFFSET_NED (7):使用本地坐标系(北、东、地下偏移)来表示相对于当前位置的航点位置。
  9. FRAME_BODY_NED (8):使用机体坐标系(前、右、下)来表示航点的位置。
  10. FRAME_BODY_OFFSET_NED (9):使用机体坐标系(前、右、下)来表示相对于当前位置的航点位置。
  11. FRAME_GLOBAL_TERRAIN_ALT (10):使用全局坐标系(经度、纬度和地形高度)来表示航点的位置。
  12. FRAME_GLOBAL_TERRAIN_ALT_INT (11):与FRAME_GLOBAL_TERRAIN_ALT相似,但使用带有整数解析度的全局坐标。
  13. FRAME_BODY_FRD (12):使用机体坐标系(前、右、向下)来表示航点的位置。
  14. FRAME_LOCAL_FRD (20): 使用本地坐标系(前、右、向下)来表示航点的位置。这个坐标系在无人机飞行控制中比较常见,它使用无人机当前位置为原点,以向前(前进方向)、向右(滑行方向)和向下(下降方向)作为坐标轴。
  15. FRAME_LOCAL_FLU (21): 使用本地坐标系(前、左、向上)来表示航点的位置。类似于FRAME_LOCAL_FRD,但使用向前、向左和向上作为坐标轴。

本质也还是MAV_CMD,所以也有7个参数可以配置的。
飞航线本质可以理解为执行一个又一个MAV_CMD命令,每个航点对应一个MAV_CMD命令,第一个都得默认是TAKE OFF(如果飞机没有起飞,则切到mission之后会自动起飞并飞往第一个航点,如果飞机已经起飞也没有关系,切mission后则是直接飞往第一个航点,注意这个点的位置并不需要在无人机当前位置正上方),这里的uint16 command也就是这个意思,值为对应要执行的MAV_CMD的消息ID值,就是每个航点实际对应执行的命令。

输入图片说明

一些航点可选的MAV_CMD举例如下
MAV_CMD_NAV_WAYPOINT
 

输入图片说明


MAV_CMD_NAV_TAKEOFF
 

输入图片说明


MAV_CMD_NAV_LAND
 

输入图片说明


MAV_CMD_NAV_VTOL_TAKEOFF
垂起固定翼的起飞航点可以用这个指令,垂直起飞后会自动切固定翼飞往第一个航点
 

输入图片说明


MAV_CMD_NAV_VTOL_LAND

输入图片说明

在上面的几个MAV_CMD中,除了经度纬度高度的赋值外,偏航的赋值也需要注意下,如果想让无人机偏航自动朝向下一个航点,则偏航赋值为NaN,而不是赋值为0。

通过 mavros/mission/push 这个服务进行航点上传

这里坐标系选择的 FRAME_GLOBAL_REL_ALT ,因为是使用相对海拔。
关于给的经纬度需要精确到小数点后几位,可以参考下面表格,一般精确到小数点后7位。
https://blog.csdn.net/lang_niu/article/details/123550453

输入图片说明

下面代码的主体流程是,先上传航点,再下载航点确认,再解锁无人机,再切mission,这样无人机就可以开始自动起飞执行航点任务了。

代码来自:https://github.com/Mohit505Git/Mavros-AUTO.MISSION-mode

以此航点的赋值为例做下讲解

 w = wayp0.setWaypoints(3,22,True,True,0.0,0.0,0.0,float('nan'),47.397713,8.547605,50)


frame = 3是指坐标系frame的值为3,也就是用的FRAME_GLOBAL_REL_ALT坐标系,这项也对应着MISSION_ITEM_INT这个mavlink消息里的frame这一项。
command = 22指此航点执行的id是22的MAV_CMD,也就是MAV_CMD_NAV_TAKEOFF,这项也对应着MISSION_ITEM_INT这个mavlink消息里的command这一项。这里坐标系选择的 FRAME_GLOBAL_REL_ALT ,因为是使用相对海拔。
is_current = True 意思是此航点作为当前要执行的航点,也就是从此航点开始执行,注意上传的航点里面就让一个航点的is_current为true就行,其他都为false,一般第一个航点的is_current为true,代表从第一个航点开始飞。这项也对应着MISSION_ITEM_INT这个mavlink消息里的current这一项。
autocontinue = True 意思是到达此航点后自动飞往下一个航点,这项也对应着MISSION_ITEM_INT这个mavlink消息里的autocontinue这一项。
param1 = 0 此参数无定义
param2 = 0 此参数无定义
param3 = 0 此参数无定义
param4 = NAN 偏航自动朝向下一个航点
x_lat = 47.397713 航点纬度
y_long = 8.547605 航点经度
z_alt = 50 高度,是在FRAME_GLOBAL_REL_ALT坐标系下的高度,是相对海拔

wayPointMission.py

  1. #!/usr/bin/env python
  2. # ROS python API
  3. import rospy
  4. # 3D point & Stamped Pose msgs
  5. from geometry_msgs.msg import Point, PoseStamped
  6. # import all mavros messages and services
  7. from mavros_msgs.msg import *
  8. from mavros_msgs.srv import *
  9. class Modes:
  10. def __init__(self):
  11. pass
  12. def setArm(self):
  13. rospy.wait_for_service('mavros/cmd/arming')
  14. try:
  15. armService = rospy.ServiceProxy('mavros/cmd/arming', mavros_msgs.srv.CommandBool)
  16. armService(True)
  17. except rospy.ServiceException, e:
  18. print "Service arming call failed: %s"%e
  19. def auto_set_mode(self):
  20. rospy.wait_for_service('mavros/set_mode')
  21. try:
  22. # setModeService = rospy.ServiceProxy('mavros/set_mode', mavros_msgs.srv.set_mode.request.custom_mode)
  23. setModeService = rospy.ServiceProxy('mavros/set_mode', mavros_msgs.srv.SetMode)
  24. setModeService(custom_mode="AUTO.MISSION")
  25. except rospy.ServiceException, e:
  26. print "Service takeoff call failed: %s"%e
  27. def wpPush(self,index,wps):
  28. rospy.wait_for_service('mavros/mission/push')
  29. try:
  30. wpPushService = rospy.ServiceProxy('mavros/mission/push', WaypointPush,persistent=True)
  31. wpPushService(start_index=0,waypoints=wps)#start_index = the index at which we want the mission to start
  32. print "Waypoint Pushed"
  33. except rospy.ServiceException, e:
  34. print "Service takeoff call failed: %s"%e
  35. def wpPull(self,wps):
  36. rospy.wait_for_service('mavros/mission/pull')
  37. try:
  38. wpPullService = rospy.ServiceProxy('mavros/mission/pull', WaypointPull,persistent=True)
  39. print wpPullService().wp_received
  40. print "Waypoint Pulled"
  41. except rospy.ServiceException, e:
  42. print "Service Puling call failed: %s"%e
  43. class stateMoniter:
  44. def __init__(self):
  45. self.state = State()
  46. # Instantiate a setpoints message
  47. self.sp = PositionTarget()
  48. # set the flag to use position setpoints and yaw angle
  49. self.sp.type_mask = int('010111111000', 2)
  50. def stateCb(self, msg):
  51. self.state = msg
  52. class wpMissionCnt:
  53. def __init__(self):
  54. self.wp =Waypoint()
  55. def setWaypoints(self,frame,command,is_current,autocontinue,param1,param2,param3,param4,x_lat,y_long,z_alt):
  56. self.wp.frame =frame # FRAME_GLOBAL_REL_ALT = 3 for more visit http://docs.ros.org/api/mavros_msgs/html/msg/Waypoint.html
  57. self.wp.command = command #'''VTOL TAKEOFF = 84,NAV_WAYPOINT = 16, TAKE_OFF=22 for checking out other parameters go to https://github.com/mavlink/mavros/blob/master/mavros_msgs/msg/CommandCode.msg'''
  58. self.wp.is_current= is_current
  59. self.wp.autocontinue = autocontinue # enable taking and following upcoming waypoints automatically
  60. self.wp.param1=param1 # no idea what these are for but the script will work so go ahead
  61. self.wp.param2=param2
  62. self.wp.param3=param3
  63. self.wp.param4=param4
  64. self.wp.x_lat= x_lat
  65. self.wp.y_long=y_long
  66. self.wp.z_alt= z_alt #relative altitude.
  67. return self.wp
  68. def main():
  69. rospy.init_node('waypointMission', anonymous=True)
  70. rate = rospy.Rate(20.0)
  71. stateMt = stateMoniter()
  72. md = Modes()
  73. wayp0 = wpMissionCnt()
  74. wayp1 = wpMissionCnt()
  75. wayp2 = wpMissionCnt()
  76. wayp3 = wpMissionCnt()
  77. wps = [] #List to story waypoints
  78. w = wayp0.setWaypoints(3,22,True,True,0.0,0.0,0.0,float('nan'),47.397713,8.547605,50)
  79. wps.append(w)
  80. w = wayp1.setWaypoints(3,16,False,True,0.0,0.0,0.0,float('nan'),47.398621,8.547745,50)
  81. wps.append(w)
  82. w = wayp2.setWaypoints(3,16,False,True,0.0,0.0,0.0,float('nan'),47.399151,8.545320,50)
  83. wps.append(w)
  84. print wps
  85. md.wpPush(0,wps)
  86. md.wpPull(0)
  87. rospy.Subscriber("/mavros/state",State, stateMt.stateCb)
  88. # Arming the drone
  89. while not stateMt.state.armed:
  90. md.setArm()
  91. rate.sleep()
  92. # Switching the state to auto mode
  93. while not stateMt.state.mode=="AUTO.MISSION":
  94. md.auto_set_mode()
  95. rate.sleep()
  96. print "AUTO.MISSION"
  97. # rospy.spin()
  98. if __name__ == '__main__':
  99. try:
  100. main()
  101. except rospy.ROSInterruptException:
  102. pass

wayPointMission.py使用方法很简单,基本的px4 gazebo mavros仿真环境起起来之后,直接运行python wayPointMission.py即可,无人机会自动解锁切mission,然后起飞依次飞往三个航点,到达最后一个航点后会在此航点保持悬停。wayPointMission.py内容如上,不用建工作空间,python脚本起ros节点确实非常方便,一个Python文件即可。
 

输入图片说明


可以同时地面站查看POSITION_TARGET_GLOABL_INT mavlink消息,看名称可知数值是整型,看当前的目标点就是所发的第一个航点的经纬高。
 

输入图片说明


无人机到达第一个航点后飞往第二个航点
 

输入图片说明


查看POSITION_TARGET_GLOABL_INT mavlink消息,可以看到当前的目标点就是所发的第二个航点的经纬高。
 

输入图片说明


抵达第三个航点
 

输入图片说明


地面站查看GLOBAL_POSITION_INT消息,可以看到无人机当前的经纬高就是第三个航点的经纬高,数据是整型显示的。

输入图片说明

mavros的终端打印,可以看到航点上传,以及依次到达各个航点的提示。

输入图片说明

我们也可以把最后一个航点的命令改为MAV_CMD_NAV_LAND(21),则无人机到达最后一个航点后会在Mission模式下自动降落,落地后自动上锁。

  1. w = wayp0.setWaypoints(3,22,True,True,0.0,0.0,0.0,float('nan'),47.397713,8.547605,50)
  2. wps.append(w)
  3. w = wayp1.setWaypoints(3,16,False,True,0.0,0.0,0.0,float('nan'),47.398621,8.547745,50)
  4. wps.append(w)
  5. w = wayp2.setWaypoints(3,21,False,True,0.0,0.0,0.0,float('nan'),47.399151,8.545320,50)
  6. wps.append(w)

输入图片说明

对于垂起固定翼,我们可以把起飞航点的MAV_CMD改为MAV_CMD_NAV_TAKEOFF(84),垂起会先以旋翼模式垂直起飞,然后切到固定翼依次到达各个航点,轨迹如下图所示
同时可以明显感受到垂起固定翼飞完这三个航点比四旋翼快太多了。

  1. w = wayp0.setWaypoints(3,84,True,True,0.0,0.0,0.0,float('nan'),47.397713,8.547605,50)
  2. wps.append(w)
  3. w = wayp1.setWaypoints(3,16,False,True,0.0,0.0,0.0,float('nan'),47.398621,8.547745,50)
  4. wps.append(w)
  5. w = wayp2.setWaypoints(3,16,False,True,0.0,0.0,0.0,float('nan'),47.399151,8.545320,50)
  6. wps.append(w)

输入图片说明

我们可以进一步地把最后一个航点改为MAV_CMD_NAV_VTOL_TAKEOFF(85),垂起则会再最后一个航点自动切为旋翼并垂直降落,轨迹如下图所示。

  1. w = wayp0.setWaypoints(3,84,True,True,0.0,0.0,0.0,float('nan'),47.397713,8.547605,50)
  2. wps.append(w)
  3. w = wayp1.setWaypoints(3,16,False,True,0.0,0.0,0.0,float('nan'),47.398621,8.547745,50)
  4. wps.append(w)
  5. w = wayp2.setWaypoints(3,85,False,True,0.0,0.0,0.0,float('nan'),47.399151,8.545320,50)
  6. wps.append(w)

输入图片说明

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/知新_RL/article/detail/688592
推荐阅读
相关标签
  

闽ICP备14008679号