Linux ROS与嵌入式的串口通信

简介: 1.根据ros wiki的官方教程 学习即可,并且有许多例子可供学习       http://wiki.ros.org/rosserial_arduino/Tutorials 2.用ASIO读写设备串行口       ASIO不仅支持网络通信还能支持串口通信。

1.根据ros wiki的官方教程 学习即可,并且有许多例子可供学习

      http://wiki.ros.org/rosserial_arduino/Tutorials

2.用ASIO读写设备串行口

      ASIO不仅支持网络通信还能支持串口通信。如何让两个设备使用串口通讯,最重要的是设置好正确的参数,那么串口的参数就包括:波特率、奇偶校验位、停止位、字符大小和流量控制,两个设备只有设置相同的参数才能相互间进行数据交换。

ASIO提供了boost::asio::serial_port类, 它有一个set_option(const SettableSerialPortOption& option)方法就是用于设置上面列举的这些参数的,其中的option 可以是:

     serial_port::baud_rate    波特率,构造参数为unsigned int

     serial_port::parity        奇偶校验,构造参数为serial_port::parity::type,enum类型,可以是none, odd, even。

     serial_port::flow_control    流量控制,构造参数为serial_port::flow_control::type,enum类型,可以是none software hardware

     serial_port::stop_bits    停止位,构造参数为serial_port::stop_bits::type,enum类型,可以是one onepointfive two

     serial_port::character_size    字符大小,构造参数为unsigned int

 这里介绍了关于ASIO的参数设置函数,向串口发送数据时,采用boost::asio:serial_port下含有write字样的函数将数据写入串口,接受串口数据时,用read函数从串口读取数据,那么比如用串口对象调用write_some(),read_some(),这类函数属于serial_port的成员函数,还有在函数内部指明串口对象的write(),read()函数,这些函数是不属于serial_port类的成员函数,但他们是boost::asio成员的函数,一般情况下我们都会用指明串口的函数比如:

      write(sp, buffer("Hello world", 12));第一个参数SP表示serial_port对象,第二个参数是写向串口的数据,第三个参数数据长度,但如果write函数在传输数据错误时会自动抛出异常boost::sysytem::error_code 那么这句话的意思就是向串口的写入长度为12 的字符串“Helloworld”

       char buf[12]    这句话的额意思很明显是声明数据的格式及长度;

       read(sp, buffer(buf));  本句话的意思就是从串口读取数据,这里在读取数据一般是我们知道发送数据的长度,就是说这句话的是从串口读取12个字符(读满才返回)才会返回,

   那么用read读取数据流必须读满内存变量后才会返回,然而返回的有时候是乱码,就会阻塞后面程序的执行,此时可以采用异步读取/接受串口的方式,就算没有完全读取/接受串口数据,异步读取函数依旧会马上返回执行后面的代码,等串口数据读取完毕或者发生异常时,io_service::run()函数会等待异步读取串口的数据操作,然后调用异步函数制定的回调函数。 

      那么一步读写操作包含三个部分:异步操作函数,异步函数以形参的方式指定到回调函数,io_service::run()函数的调用,这三个部分的执行流程是:当程序执行到异步操作函数时,异步操作立即返回,程序继续执行后续的代码,异步操作函数功能完成或者异常时,io_service::run()函数会自动的调用异步操作函数指定到回调函数。

   3.这里用STM32作为主控器,那么我用的ALIENTEK  MiniSTM32开发板作为例子,(任何一个STM32或者嵌入式有串口的开发板都可以应用),我们直接下载开发板的自带的例子实验3——串口实验,下载后,它发送的是汉字,因为是在Ubuntu系统下,所以我把它改为英文,此时在win串口助手的调试会有如下图的结果:

            

说明串口一切正常。

4  ,我们的目的是要结合ROS与STM32,其中串口是一个重要的连接桥梁,一个重要的通讯工具,那么我们可以依照wiki.ros.org/ROS/Tutorials官网上的C++写的发布接受节点的例子

http://wiki.ros.org/ROS/Tutorials/WritingPublisherSubscriber%28c%2B%2B%29

然后包含ASIO串口的库函数即可接受一个STM32发送的字符串,并且通过一个节点发布出去,现在我们就具体分析这个程序,

#include <string>
#include <ros/ros.h>                           // 包含ROS的头文件
#include <sensor_msgs/JointState.h>
#include <tf/transform_broadcaster.h>
#include <nav_msgs/Odometry.h>
#include <boost/asio.hpp>                  //包含boost库函数
#include <boost/bind.hpp>
#include <math.h>
#include "std_msgs/String.h"              //ros定义的String数据类型

using namespace std;
using namespace boost::asio;           //定义一个命名空间,用于后面的读写操作

unsigned char buf[24];                      //定义字符串长度

int main(int argc, char** argv) {

    ros::init(argc, argv, "boost");       //初始化节点
    ros::NodeHandle n;
    
   ros::Publisher chatter_pub = n.advertise<std_msgs::String>("chatter", 1000);      //定义发布消息的名称及sulv

  ros::Rate loop_rate(10);


    io_service iosev;
    serial_port sp(iosev, "/dev/ttyUSB0");         //定义传输的串口
    sp.set_option(serial_port::baud_rate(9600));   
    sp.set_option(serial_port::flow_control());
    sp.set_option(serial_port::parity());
    sp.set_option(serial_port::stop_bits());
    sp.set_option(serial_port::character_size(8));

    while (ros::ok()) {
      // write(sp, buffer(buf1, 6));  //write the speed for cmd_val    
     //write(sp, buffer("Hello world", 12));  
     read (sp,buffer(buf));
    string str(&buf[0],&buf[22]);            //将数组转化为字符串
  //if(buf[0]=='p' && buf[21] == 'a')
   // {
       std_msgs::String msg;
       std::stringstream ss;
       ss <<str;
     
      msg.data = ss.str();
     
    ROS_INFO("%s", msg.data.c_str());//打印接受到的字符串
    chatter_pub.publish(msg);   //发布消息

    ros::spinOnce();

    loop_rate.sleep();
  //  }
    }

    iosev.run(); 
    return 0;
}

最后我们把STM32 开发板连接装好ROS的Ubuntu14.04的系统,并且相应的catkin_make一下,编译无错误,那么我们我们运行rosrun beginner_tutorials  yl_boost  就会有如下结果

 

如果出现错误,我们首先检查串口号  用命令 “ls  /dev/ttyUSB*”  一定要在程序中USB的参数完全一致。

  如果仍然有错误提示/dev/ttyUSB0  拒绝访问,此时我们就需要对USB的权限进行设置,可以输入

              sudo chmod 666 /dev/ttyUSB0或者

              sudo chmod a+rw /dev/ttyUSB0 

-------------------------------------------------------------------------------------------------------------

------------------------------------------------------------------------------------------------------------

版权所有,转载请注明出处

 

 

如果您觉得看完有所收获,欢迎扫一扫,可以资助一分,几分money,不在乎多少(我也是跟网上的大神们学的),不想挣钱娶媳妇的程序员不是好程序员(同时我也看看到底有没有人认可),谢谢

        

相关实践学习
Docker镜像管理快速入门
本教程将介绍如何使用Docker构建镜像,并通过阿里云镜像服务分发到ECS服务器,运行该镜像。
阿里云资源编排ROS使用教程
资源编排(Resource Orchestration)是一种简单易用的云计算资源管理和自动化运维服务。用户通过模板描述多个云计算资源的依赖关系、配置等,并自动完成所有资源的创建和配置,以达到自动化部署、运维等目的。编排模板同时也是一种标准化的资源和应用交付方式,并且可以随时编辑修改,使基础设施即代码(Infrastructure as Code)成为可能。 产品详情:https://www.aliyun.com/product/ros/
相关文章
|
16天前
|
JSON 机器人 Linux
推荐一款嵌入式Linux开源框架与封装-cpp-tbox
推荐一款嵌入式Linux开源框架与封装-cpp-tbox
48 3
|
1月前
|
Linux 开发工具 git
Linux嵌入式系统中如何使用U-Boot实例
Linux嵌入式系统中如何使用U-Boot实例
26 0
|
1月前
|
Linux Android开发
嵌入式linux中Framebuffer 驱动程序框架分析
嵌入式linux中Framebuffer 驱动程序框架分析
26 0
|
1月前
|
Linux C语言 SoC
嵌入式linux总线设备驱动模型分析
嵌入式linux总线设备驱动模型分析
32 1
|
1月前
|
Linux
嵌入式linux系统设备树实例分析
嵌入式linux系统设备树实例分析
34 0
|
4天前
|
Linux 编译器 测试技术
嵌入式 Linux 下的 LVGL 移植
嵌入式 Linux 下的 LVGL 移植
|
17天前
|
Linux
嵌入式Linux系统(NUC980)tf卡出错处理errors=remount-ro改为errors=continue
嵌入式Linux系统(NUC980)tf卡出错处理errors=remount-ro改为errors=continue
7 1
|
17天前
|
安全 Linux
嵌入式Linux系统关闭串口调试信息的输出
嵌入式Linux系统关闭串口调试信息的输出
13 1
|
17天前
|
Linux 编译器 网络安全
嵌入式Linux移植dropbear
嵌入式Linux移植dropbear
16 3
|
17天前
|
存储 Ubuntu Linux
制作一个嵌入式Linux的应用程序升级文件
制作一个嵌入式Linux的应用程序升级文件
12 2