ROS机械臂开发:MoveIt! + Gazebo仿真

一、 ROS中的控制器插件

ros_control的功能:

  • ROS为开发者提供的机器人控制中间件
  • 包含一系列控制器接口、传动装置接口、硬件接口、控制器工具箱等等
  • 可以帮助机器人应用功能包快速落地,提高开发效率
ros control

下面介绍一下采用ros_control实现的控制系统整体架构:


gazebo + ros + ros_control
  • 控制器管理器
    提供一种通用的接口来管理不同的控制器,比如小车的轮子需要速度指令,机械臂需要位置、扭矩指令,不同的控制器就需要一个controller manager来管理,以同时包含多种controller
  • 控制器
    读取硬件状态,发布控制命令,完成每个joint的控制
  • 硬件资源
    为上下两层提供硬件资源的接口
  • 机器人硬件抽象
    机器人硬件抽象和硬件资源直接打交道,通过write和read方法完成硬件操作
  • 真实机器人
    执行接收到的命令

controller中主要实现的是PID控制,完成速度、位置等闭环控制,并将指令发出去,发给谁呢,当然是机器人,如果是仿真就发给gazebo,如果是实际机器人,就发给真实机器人硬件。这里就有一个interface,通过串口,网口等接口的形式把数据发出去hardware_interface是为了让上层的接口屏蔽底层的硬件,作为一个硬件封装层封装了底层的电机、执行器。接收到上层的指令后,再分发到底层的硬件。比如发给小车车轮,上面通常会有一个嵌入式板卡来完成本地的闭环控制,比如发送电压信号给电机,电机如果配备编码器,还可以通过硬件抽象层反馈上来,发回controller,再形成闭环。
ros_control这个功能包包含了很多controllers,完成速度、位置、力控制。常用的有四个controller:

  • joint_state_controller
    这个controller其实和控制没关系,比较容易让人误解,监控机器人状态,比如编码器的反馈回来的位置速度等数据,它将其封装成topic发出来,功能和joint_state_publisher这个节点是一样的,只是数据输入不一样,输出都是话题形式。

  • joint_effort_controller

  • joint_position_controller
    机械臂经常会用这个controller

  • joint_velocity_controller

此处会介绍 joint_state_controller和joint_position_controller,最后会再介绍一个更加上层的controller。因此ros_control在整个ROS框架中的作用就是作为一个中间件,衔接上层应用和真实机器人或仿真模型。


image.png

2.完善机器人模型

要放到gazebo中仿真,之前的urdf模型文件还需要再进行完善。

2.1 模型文件

第一步:在可视化基础上为link添加惯性参数和碰撞属性
image.png

一个技巧是设置很小的质量和较大的惯性矩阵,这样导入gazebo后模型运行会比较稳定,当然前提是仿真应用对这些物理参数要求不高。

第二步:为joint添加传动装置

代表减速器等传动装置,传进来的是控制指令,比如下图的位置指令


image.png
第三步:添加gazebo控制器插件

电机如何转起来,需要添加一个控制器,对应真实机器人的伺服驱动器,就是前面讲的controller,完成电机的运动控制,输入位置,输出到transmission上面去。参数是比较多的,一般在模型中加载插件,再通过多个参数文件具体配置。
robotNamespace代表具体的机器人命名空间,仿真类型robotSimType,使用默认的硬件抽象层接口,完成上面的指令到机器人的转换。legacyModeNS用于兼容之前ROS版本的配置。


image.png

2.2 在gazebo中加载机器人

<launch>

  <!-- these are the arguments you can pass this launch file, for example paused:=true -->
  <arg name="paused" default="false"/>
  <arg name="use_sim_time" default="true"/>
  <arg name="gui" default="true"/>
  <arg name="headless" default="false"/>
  <arg name="debug" default="false"/>

  <!-- 启动gazebo仿真环境 -->
  <include file="$(find gazebo_ros)/launch/empty_world.launch">
    <arg name="debug" value="$(arg debug)" />
    <arg name="gui" value="$(arg gui)" />
    <arg name="paused" value="$(arg paused)"/>
    <arg name="use_sim_time" value="$(arg use_sim_time)"/>
    <arg name="headless" value="$(arg headless)"/>
  </include>

  <!-- Load the URDF into the ROS Parameter Server -->
  <param name="robot_description" command="$(find xacro)/xacro --inorder '$(find probot_description)/urdf/probot_anno.xacro'" /> 


  <!-- 在gazebo中加载模型--> 
  <!--Run a python script to the send a service call to gazebo_ros to spawn a URDF robot -->
  <node name="urdf_spawner" pkg="gazebo_ros" type="spawn_model" respawn="false" output="screen"
    args="-urdf -model probot_anno -param robot_description"/> 


</launch>

launch文件完成的功能:一是启动gazebo仿真环境,二是将模型加载到仿真环境中去。

i5@i5-ThinkPad-T470:~/catkin_ws$ roslaunch probot_gazebo probot_anno_gazebo_world.launch
image.png

现在的机械臂还是不能运动的,可以看一下话题:

i5@i5-ThinkPad-T470:~/catkin_ws$ rostopic list
/clock
/gazebo/link_states
/gazebo/model_states
/gazebo/parameter_descriptions
/gazebo/parameter_updates
/gazebo/set_link_state
/gazebo/set_model_state
/gazebo_gui/parameter_descriptions
/gazebo_gui/parameter_updates
/rosout
/rosout_agg

3. 构建MoveIt!+Gazebo仿真

首先介绍一下moveit机器人控制框架,moveit输入用户指令,比如起始位置,输出轨迹数据,每个点通过位置速度加速度描述,为了通过话题发出来,还需要一个Follow Joint Trajectory功能,通过通信接口发给机器人,机器人还需要对轨迹进行精插补,再驱动电机同步运动。机器人控制器还需要将每个电机的状态数据反馈给moveit来确定机器人是否到达指定位置,这样完成一个闭环。


moveit!机器人控制框架

主要是理解三个模块的作用:Follow Joint Trajectory,Joint Trajectory Controller, Joint State Controller


moveit!+gazebo仿真框架

先看gazebo这一端的,轨迹通过Action机制发出来,因此gazebo这边是一个action server,moveit那边是一个action client,sever这一端接收轨迹,完成插补运算,再发到每一个电机位置控制接口上。
ROS提供的关节轨迹控制器插补运算(具体可以看ros_controllers的源码)
  • 线性样条:位置连续,速度、加速度不连续
  • 三次样条:位置和速度连续,加速度不连续
  • 五次样条:位置、速度、加速度都连续(ROS默认使用的)
    image.png

    电机转起来涉及到两个配置文件,一个是config/probot_anno_trajectory_control.yaml,用于明确控制器接口接收哪几个关节信息,要与urdf模型对应,文件中还包括每个关节的PID参数
probot_anno:
  arm_joint_controller:
    type: "position_controllers/JointTrajectoryController"
    joints:
      - joint_1
      - joint_2
      - joint_3
      - joint_4
      - joint_5
      - joint_6

    gains:
      joint_1:   {p: 1000.0, i: 0.0, d: 0.1, i_clamp: 0.0}
      joint_2:   {p: 1000.0, i: 0.0, d: 0.1, i_clamp: 0.0}
      joint_3:   {p: 1000.0, i: 0.0, d: 0.1, i_clamp: 0.0}
      joint_4:   {p: 1000.0, i: 0.0, d: 0.1, i_clamp: 0.0}
      joint_5:   {p: 1000.0, i: 0.0, d: 0.1, i_clamp: 0.0}
      joint_6:   {p: 1000.0, i: 0.0, d: 0.1, i_clamp: 0.0}

另一个文件是控制器启动文件:probot_anno_trajectory_controller.launch,首先将上面的文件加载到参数服务器,再调用spawner根据参数文件内容启动arm_joint_controller,这个控制器完成每个轴的插补运算,并让每个轴运动。

<launch>

    <rosparam file="$(find probot_gazebo)/config/probot_anno_trajectory_control.yaml" command="load"/>

    <node name="arm_controller_spawner" pkg="controller_manager" type="spawner" respawn="false"
          output="screen" ns="/probot_anno" args="arm_joint_controller"/>

</launch>

以上完成的就是框架中绿色部分的配置。

joint state controller配置是类似的,先进行参数配置:probot_anno_gazebo_joint_states.yaml

probot_anno:
  # Publish all joint states -----------------------------------
  joint_state_controller:
    type: joint_state_controller/JointStateController
    publish_rate: 50

用于监控机器人实时状态,joint_state_controller/JointStateController就是ros cotrollers四种当中的一种,参数就是反馈的频率。配置好后,需要启动它probot_anno_gazebo_states.launch:

<launch>
    <!-- 将关节控制器的配置参数加载到参数服务器中 -->
    <rosparam file="$(find probot_gazebo)/config/probot_anno_gazebo_joint_states.yaml" command="load"/>

    <node name="joint_controller_spawner" pkg="controller_manager" type="spawner" respawn="false"
          output="screen" ns="/probot_anno" args="joint_state_controller" />

    <!-- 运行robot_state_publisher节点,发布tf  -->
    <node name="robot_state_publisher" pkg="robot_state_publisher" type="robot_state_publisher"
        respawn="false" output="screen">
        <remap from="/joint_states" to="/probot_anno/joint_states" />
    </node>

</launch>

先加载文件,再通过spawner启动控制器,后面还启动了robot_state_publisher,用于根据joint_states发布TF坐标系.

第三个配置是框图中左边的,Follow Joint Trajectory,也是配参数controllers_gazebo.yaml:

controller_manager_ns: controller_manager
controller_list:
  - name: probot_anno/arm_joint_controller
    action_ns: follow_joint_trajectory
    type: FollowJointTrajectory
    default: true
    joints:
      - joint_1
      - joint_2
      - joint_3
      - joint_4
      - joint_5
      - joint_6

再启动probot_anno_moveit_controller_manager.launch.xml:

<launch>
    <arg name="moveit_controller_manager" default="moveit_simple_controller_manager/MoveItSimpleControllerManager"/>
    <param name="moveit_controller_manager" value="$(arg moveit_controller_manager)"/>

        <!-- gazebo Controller -->
        <rosparam file="$(find probot_anno_moveit_config)/config/controllers_gazebo.yaml"/>

</launch>

下面梳理一下所有文件,首先是最顶层的probot_anno_bringup_moveit.launch:

<launch>
  
    <!-- Launch Gazebo  -->
    <include file="$(find probot_gazebo)/launch/probot_anno/probot_anno_gazebo_world.launch" />

    <!-- ros_control arm launch file -->
    <include file="$(find probot_gazebo)/launch/probot_anno/probot_anno_gazebo_states.launch" />   

    <!-- ros_control trajectory control dof arm launch file -->
    <include file="$(find probot_gazebo)/launch/probot_anno/probot_anno_trajectory_controller.launch" />

    <!-- moveit launch file -->
    <include file="$(find probot_anno_moveit_config)/launch/moveit_planning_execution.launch" />

</launch>

首先启动仿真环境,接着启动Joint state反馈控制器,接着启动完成插补功能的控制器,最后启动moveit,里面会自动包含刚才的moveit一端封装action数据接口的功能。
总结一下几个核心的文件功能:
probot_anno_bringup_moveit.launch:最顶层的,启动下面三个文件和moveit里的文件
probot_anno_gazebo_world.launch: 启动仿真环境
probot_anno_gazebo_states.launch: 启动joint state controller
probot_anno_trajectory_controller.launch:启动插补运算的controller
probot_anno_moveit_controller_manager.launch.xml: 启动follow joint trajectory功能

i5@i5-ThinkPad-T470:~$ roslaunch probot_gazebo probot_anno_bringup_moveit.launch

最后梳理一下启动仿真的各个文件包含关系:


gazebo仿真启动文件
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343

推荐阅读更多精彩内容