0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

ROS與移動底盤通信教程

3D視覺工坊 ? 來源:CSDN-白鳥無言 ? 2023-03-14 10:27 ? 次閱讀

本實驗是實現(xiàn)機器人自主導航的重要步驟,對于輪式機器人,可以通過在底盤加裝輪式里程計的方式來獲得機器人的速度數(shù)據(jù),這些數(shù)據(jù)可以用來輔助機器人實現(xiàn)自主定位,同時機器人還需要將控制指令發(fā)送給移動底盤,實現(xiàn)自主控制,本實驗就將實現(xiàn)ROS與移動底盤的通信。

實驗環(huán)境:

· 軟件環(huán)境:Ubuntu18.04 + ROS melodic、Windows + Keil 5、VSCode

· 硬件環(huán)境:Jetson Nano(以下稱為ROS端)、小車(以下稱為STM32端)

01 實驗內容

ROS與STM32的通信流程如圖所示

1cd04234-c1a0-11ed-bfe3-dac502259ad0.png

主要包含兩個方面:

· 小車里程計數(shù)據(jù)的上傳與接收

· 控制指令的下發(fā)與接收

1.1 原始消息內容

在ROS中,里程計數(shù)據(jù)主要包括機器人的位姿(位置和姿態(tài)),以及機器人的速度(線速度和角速度)。對于本實驗所用到的麥輪地面機器人,只需要知道機器人的x軸與y軸線速度、x軸與y軸位置、z軸角速度、偏航角即可。

由于對速度積分可以得到位置,對角速度積分可以得到角度,所以STM32端上傳的里程計數(shù)據(jù)只需要包括機器人的x軸與y軸線速度、z軸角速度,ROS端在接收到這些數(shù)據(jù)后,進行積分即可得到位置和角度。

另外,在本實驗用到的STM32端集成了一個ICM20602姿態(tài)傳感器,其中內置了姿態(tài)解算算法,可以獲得準確的機器人姿態(tài)數(shù)據(jù),因此本實驗使用STM32端上傳的偏航角來代替對角速度積分得到的航向角。

所以STM32上傳的里程計數(shù)據(jù)包括機器人的x軸線速度、y軸線速度、z軸角速度、偏航角。

與里程計數(shù)據(jù)類似,對于麥輪地面機器人,控制指令只需要包括機器人的x軸速度、y軸速度、z軸角速度即可,機器人坐標系如圖所示:
1ce12d4c-c1a0-11ed-bfe3-dac502259ad0.png

1.2 轉換為字節(jié)數(shù)組

知道了消息的原始數(shù)據(jù),還需要將它轉換成傳輸效率更高的字節(jié)數(shù)組,如圖:

1cf38b0e-c1a0-11ed-bfe3-dac502259ad0.png

在C/C++中,有很多種將原始數(shù)據(jù)轉換為字節(jié)數(shù)組的方法,其中一種常用的方法是使用聯(lián)合體(union)。

聯(lián)合體的所有成員占用同一段內存,修改一個成員會影響其余成員,如果要實現(xiàn)一個float數(shù)據(jù)與字節(jié)數(shù)組的互相轉換,我們可以定義如下的聯(lián)合體:

typedefunion{
float data;
uint8_t data8[4];
}data_u;
?

這個聯(lián)合體中有兩個成員,一個是32位的float數(shù)據(jù)data,另一個同樣是占據(jù)了32位字長的字節(jié)數(shù)組data8,根據(jù)聯(lián)合體的性質,這兩個成員所在的內存位置是一樣的,也就是說,改變其中任何一個成員的值,另一個也會被改變。

利用這個性質,我們就可以實現(xiàn)float與字節(jié)數(shù)據(jù)的互相轉換。

1.3 添加幀頭和校驗碼

本實驗選擇常用的0xAA 0x55作為幀頭,同時對原始數(shù)據(jù)轉換得到的字節(jié)數(shù)組進行求和,將結果保存在1字節(jié)數(shù)據(jù)中,作為校驗碼。

準備工作:

1. 在ROS端安裝serial功能包
sudoapt-getinstallros-melodic-serial
2. 在ROS端創(chuàng)建一個功能包,命名為xrobot,添加依賴項roscpp rospy tf serial

02 里程計數(shù)據(jù)的上傳與接收

2.1 通信協(xié)議

里程計數(shù)據(jù)格式(19字節(jié))

1d03fe4e-c1a0-11ed-bfe3-dac502259ad0.png

2.2 STM32端
/**
 * @brief 發(fā)送里程計數(shù)據(jù)
 */
void DataTrans_Odom(void)
{
uint8_t _cnt = 0;
  data_u _temp; // 聲明一個聯(lián)合體實例,使用它將待發(fā)送數(shù)據(jù)轉換為字節(jié)數(shù)組
uint8_t data_to_send[100] = {0}; // 待發(fā)送的字節(jié)數(shù)組


  data_to_send[_cnt++]=0xAA;
  data_to_send[_cnt++]=0x55;


uint8_t _start = _cnt;


float datas[] = {kinematics.odom.vel.linear_x, 
                     kinematics.odom.vel.linear_y, 
                     kinematics.odom.vel.angular_z, 
                     kinematics.odom.pose.theta
                    };


for(int i = 0; i < sizeof(datas) / sizeof(float); i++)
  {
// 將要發(fā)送的數(shù)據(jù)賦值給聯(lián)合體的float成員
// 相應的就能更改字節(jié)數(shù)組成員的值
    _temp.data = datas[i];
    data_to_send[_cnt++]=_temp.data8[0];
    data_to_send[_cnt++]=_temp.data8[1];
    data_to_send[_cnt++]=_temp.data8[2];
    data_to_send[_cnt++]=_temp.data8[3]; // 最高位
  }


uint8_t checkout = 0;
for(int i = _start; i < _cnt; i++)
  {
    checkout += data_to_send[i];
  }
  data_to_send[_cnt++] = checkout;
// 串口發(fā)送
  SendData(data_to_send, _cnt); 
}
2.3 ROS端

采用狀態(tài)機的方式來接收STM32端上傳的里程計數(shù)據(jù),每讀取一字節(jié)數(shù)據(jù),則在狀態(tài)機中處理一次,部分程序如下:
uint8_tbuffer=0;
ser.read(&buffer, 1); // ser是串口類的一個實例,該語句表示從串口中讀取一個字節(jié)
if(state == 0 && buffer == 0xAA)
{
    state++;
}
else if(state == 1 && buffer == 0x55)
{
    state++;
}
else if(state == 2)
{
    data_receive[data_cnt++]=buffer;
if(data_cnt == 17)
    {
/* 進行數(shù)據(jù)校驗 */
uint8_t checkout = 0;
for(int k = 0; k < data_cnt - 1; k++)
        {
            checkout += data_receive[k];
        }
if(checkout == data_receive[data_cnt - 1]) // 串口接收到的最后一個字節(jié)是校驗碼 
        {
/* 校驗通過,進行解碼 */
float vx, vy, vth, th; // x軸線速度,y軸線速度,z軸角速度,偏航角
float* datas_ptr[] = {&vx, &vy, &vth, &th};
            data_u temp;
for(int i = 0; i < sizeof(datas_ptr) / sizeof(float*); i++)
            {
                temp.data8[0] = data_receive[4 * i + 0];
                temp.data8[1] = data_receive[4 * i + 1];
                temp.data8[2] = data_receive[4 * i + 2];
                temp.data8[3] = data_receive[4 * i + 3];              
                *(datas_ptr[i]) = temp.data;
            }
            th *= D2R; // 轉換為弧度
        }
        data_cnt = 0;
        state = 0;
    }
}
else state = 0;
? ROS端在運行時可能會提示串口打開失敗,有兩種原因,一是串口號不對,使用dmesg | grep ttyS*列出檢測到的串口號,逐個測試;

二是沒有操作權限,使用sudo chmod 666 /dev/ttyACM0即可解決,也可以使用sudo usermod -aG dialout 用戶名來獲得永久權限,用戶名可使用whoami查看。

2.4 里程計數(shù)據(jù)可視化

以上步驟僅僅得到了機器人的x軸線速度、y軸線速度、z軸角速度、偏航角,還需要進一步處理來獲得完整的里程計數(shù)據(jù)。

STM32端返回的x軸線速度、y軸線速度是相對于自身的機體坐標系的速度,而機器人的位置信息是相對于世界坐標系的位置,所以在對速度進行積分前,要先將機體坐標系下的x軸線速度、y軸線速度轉換到世界坐標系,如圖:
1d196eaa-c1a0-11ed-bfe3-dac502259ad0.png

這個坐標變換可以通過一個簡單的旋轉矩陣來實現(xiàn)

1d2864c8-c1a0-11ed-bfe3-dac502259ad0.png

其中θ就是機器人的偏航角。相應的程序如下:
/*對速度進行積分得到位移*/
// 獲取當前時間
current_time = ros::now();
// 獲取積分間隔
double dt = (current_time - last_time).toSec();
last_time = current_time;
// 將機體系速度轉換到里程計坐標系
double delta_x = (vx * cos(th) - vy * sin(th)) * dt;
double delta_y = (vx * sin(th) + vy * cos(th)) * dt;
// 速度積分
x += delta_x;
y += delta_y;
? 在機器人中,一般使用四元數(shù)/旋轉矩陣的形式來表示機器人的姿態(tài),而不是歐拉角形式。所以需要將STM32返回的偏航角轉換為四元數(shù),程序如下:
/*對速度進行積分得到位移*/
// 獲取當前時間
current_time = ros::now();
// 獲取積分間隔
double dt = (current_time - last_time).toSec();
last_time = current_time;
// 將機體系速度轉換到里程計坐標系
double delta_x = (vx * cos(th) - vy * sin(th)) * dt;
double delta_y = (vx * sin(th) + vy * cos(th)) * dt;
// 速度積分
x += delta_x;
y += delta_y;
? 以上就獲取了完整的機器人里程計數(shù)據(jù),接下來需要將里程計數(shù)據(jù)發(fā)布到ROS中。
nav_msgs::Odometryodom;
geometry_msgs::TransformStamped odom_trans;


odom_trans.header.stamp = current_time;
odom_trans.header.frame_id = "odom";
odom_trans.child_frame_id = "base_link";


odom_trans.transform.translation.x = x;
odom_trans.transform.translation.y = y;
odom_trans.transform.translation.z = 0.0;
odom_trans.transform.rotation = odom_quat;
// 發(fā)布坐標變換
odom_broadcaster.sendTransform(odom_trans);


odom.header.stamp = current_time;
odom.header.frame_id = "odom";
odom.child_frame_id = "base_link";


// 設置機器人的位置和姿態(tài)
odom.pose.pose.position.x = x;
odom.pose.pose.position.y = y;
odom.pose.pose.position.z = 0.0;
odom.pose.pose.orientation = odom_quat;


// 設置機器人的速度
odom.twist.twist.linear.x = vx;
odom.twist.twist.linear.y = vy;
odom.twist.twist.angular.z = vth;


// 發(fā)布里程計消息
odom_pub.publish(odom);
? 運行后,打開PC上的Ubuntu,配置ip從而實現(xiàn)遠程連接嵌入式處理器上的ROS系統(tǒng)。

配置完成后,重新打開一個終端,輸入:rviz,打開ROS的可視化工具,按照下圖操作即可 1d3b4cbe-c1a0-11ed-bfe3-dac502259ad0.png


可視化結果如下:

1d5e82f6-c1a0-11ed-bfe3-dac502259ad0.png

最后將該rviz配置保存至文件,點擊File→Save Config As,將配置保存為xxxx.rviz。下次打開時,在命令行運行:rosrun rviz rviz -d xxxx.rviz即可。

03 控制指令的下發(fā)與接收

3.1 通信協(xié)議

控制指令格式(15字節(jié))
1d6caeda-c1a0-11ed-bfe3-dac502259ad0.png

3.2 ROS端

在ROS端,首先需要接收從其他節(jié)點的控制消息,在ROS中常常使用geometry_msgs::Twist來傳遞控制指令,該消息格式包括兩個三維向量,如下,分別是三軸線速度和三軸角速度:
geometry_msgs/Vector3linear
geometry_msgs/Vector3 angular
? 我們在控制指令的消息回調函數(shù)中,將控制指令下發(fā)給STM32,部分程序如下,其中使用了C++的lambda表達式來替換回調函數(shù)
ros::Subscribersub=nh.subscribe("/cmd_vel",10,[&](constgeometry_msgs::ConstPtr&cmd_vel){
uint8_t _cnt = 0;
    data_u _temp; // 聲明一個聯(lián)合體實例,使用它將待發(fā)送數(shù)據(jù)轉換為字節(jié)數(shù)組
uint8_t data_to_send[100]; // 待發(fā)送的字節(jié)數(shù)組    
    data_to_send[_cnt++]=0xAA;
    data_to_send[_cnt++]=0x55;
uint8_t _start = _cnt;
float datas[] = {(float)cmd_vel->linear.x,
                     (float)cmd_vel->linear.y,
                     (float)cmd_vel->angular.z};    
for(int i = 0; i < sizeof(datas) / sizeof(float); i++){
// 將要發(fā)送的數(shù)據(jù)賦值給聯(lián)合體的float成員
// 相應的就能更改字節(jié)數(shù)組成員的值
        _temp.data = datas[i];
        data_to_send[_cnt++]=_temp.data8[0];
        data_to_send[_cnt++]=_temp.data8[1];
        data_to_send[_cnt++]=_temp.data8[2];
        data_to_send[_cnt++]=_temp.data8[3]; // 最高位
    }
// 添加校驗碼
char checkout = 0;
for(int i = _start; i < _cnt; i++)
        checkout += data_to_send[i];
    data_to_send[_cnt++] = checkout;
// 串口發(fā)送
    ser.write(data_to_send, _cnt);
});
? 最后,創(chuàng)建一個控制指令發(fā)布節(jié)點,用來發(fā)布cmd_vel話題,在xrobot功能包下新建一個scripts文件夾,添加remote_ctrl.py,內容如下:
#!/usr/bin/envpython
# #-*- coding: UTF-8 -*- 


import rospy
from geometry_msgs.msg import Twist
from std_msgs.msg import String
import os, sys
import  tty, termios


pub = rospy.Publisher("cmd_vel", Twist)
rospy.init_node("remote_ctrl")
rate = rospy.Rate(rospy.get_param("-hz", 20))


cmd = Twist()
cmd.linear.x = 0.0
cmd.linear.y = 0.0
cmd.angular.z = 0.0


while not rospy.is_shutdown():    
    tty.setraw(sys.stdin.fileno())  
    ch = sys.stdin.read( 1 )  
if ch == "w":
        cmd.linear.x = 0.2
        cmd.linear.y = 0
        cmd.angular.z = 0
elif ch == "s":
        cmd.linear.x = -0.2
        cmd.linear.y = 0
        cmd.angular.z = 0
elif ch == "a":
        cmd.linear.x = 0
        cmd.linear.y = 0
        cmd.angular.z = 0.5
elif ch == "d":
        cmd.linear.x = 0
        cmd.linear.y = 0
        cmd.angular.z = -0.5
elif ch == "q":
break
else:
        cmd.linear.x = 0
        cmd.linear.y = 0
        cmd.angular.z = 0
    rospy.loginfo(str( cmd.linear.x) + " " + str(cmd.linear.y) + " " + str(cmd.angular.z) + "
")
    pub.publish(cmd)
    rate.sleep()
? 運行robot_node和remote_ctrl節(jié)點,可以得到如下的節(jié)點圖:
1d7d1234-c1a0-11ed-bfe3-dac502259ad0.png

3.3 STM32端

部分程序如下:
/**
 * @brief 從串口讀取單個字節(jié)
 * @param  data             讀取的字節(jié)數(shù)據(jù)
 */
void GetOneByte(uint8_t data)
{
  static u8 state = 0;
  static u8 cnt = 0;
if(state == 0 && data == 0xAA)
  {
    state++;
  }
else if(state == 1 && data == 0x55)
  {
    state++;
  }
else if(state == 2)
  {
    data_receive[cnt++] = data;
if(cnt >= 13)
    {
// 校驗
      u8 checkout = 0;
for(int i = 0; i < cnt - 1; i++)
      {
        checkout += data_receive[i];
      }
if(checkout == data_receive[cnt - 1])
      {
// 校驗通過,進行解碼
        DataDecoder(data_receive);
      }
      state = 0;
      cnt = 0;
    }
  }
else state = 0;
}


/**
 * @brief 數(shù)據(jù)解碼
 * @param  data             待解碼數(shù)組
 */
void DataDecoder(u8 *data)
{
    data_u temp;
// x軸線速度
    temp.data8[0] = data[0];
    temp.data8[1] = data[1];
    temp.data8[2] = data[2];
    temp.data8[3] = data[3];
    kinematics.exp_vel.linear_x = temp.data;
// y軸線速度
    temp.data8[0] = data[4];
    temp.data8[1] = data[5];
    temp.data8[2] = data[6];
    temp.data8[3] = data[7];
    kinematics.exp_vel.linear_y = temp.data;
// z軸角速度
    temp.data8[0] = data[8];
    temp.data8[1] = data[9];
    temp.data8[2] = data[10];
    temp.data8[3] = data[11];
    kinematics.exp_vel.angular_z = temp.data;  
}

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 機器人
    +關注

    關注

    211

    文章

    28582

    瀏覽量

    207818
  • 通信
    +關注

    關注

    18

    文章

    6056

    瀏覽量

    136254
  • STM32
    +關注

    關注

    2270

    文章

    10918

    瀏覽量

    356897
  • Ubuntu
    +關注

    關注

    5

    文章

    565

    瀏覽量

    29930
  • ROS
    ROS
    +關注

    關注

    1

    文章

    279

    瀏覽量

    17042

原文標題:ROS與移動底盤通信

文章出處:【微信號:3D視覺工坊,微信公眾號:3D視覺工坊】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    rosserial、ros_lib移植到STM32講解 精選資料分享

    這邊博客主要是對前面兩篇博客的一個補充(ROS使用STM32F407ZGT6作為底盤控制器、ros下使用rosserial和STM32F1/STM32F4系列進行通信(MDK5工程))
    發(fā)表于 08-04 06:13

    ROS與STM32是如何進行通信

    ROS與STM32通信2020.8.1主要內容制作ROS包,將控制命令傳給STM32,并將接收到的數(shù)據(jù)作為話題進行發(fā)布STM32接收數(shù)據(jù)并將姿態(tài)數(shù)據(jù)傳回給ROS接收:期望角速度、期望線
    發(fā)表于 08-11 07:25

    最實用的STM32和ROS機器人的串口通信方案

    全網最實用的STM32和ROS機器人的串口通信方案小白學移動機器人同名公眾號:小白學移動機器人創(chuàng)作聲明:內容包含虛構創(chuàng)作內容中的情節(jié)存在虛構加工,僅供參考全網最實用的STM32和
    發(fā)表于 08-20 06:33

    移動機器人底盤主要包含哪些設備

    移動機器人底盤主要包含電機,電機驅動器,底盤控制器和其它設備。底盤控制器與電腦通信,把電腦指令解析后發(fā)送給電機驅動器,同時控制器
    發(fā)表于 09-07 06:15

    如何完成ROS與STM32之間的串口通信

    如何去實現(xiàn)ROS與STM32串口通信測試功能?如何完成ROS與STM32之間的串口通信呢?
    發(fā)表于 12-10 06:54

    如何搭建實體機器人ros底盤

    目錄介紹一、底盤主控板二、嵌入式開發(fā)板1. 與上位機pc的關系2. 與STM32主控板的關系介紹自下而上的分析實體機器人(差分輪速機器人)搭建中的關鍵過程。一、底盤主控板本部分搭建實體機器人ros
    發(fā)表于 01-20 07:36

    通信教程的04_SPI接口說明及原理

    通信教程04_SPI接口說明及原理
    的頭像 發(fā)表于 02-05 12:29 ?3986次閱讀

    通信教程03_I2C簡史 基礎原理及協(xié)議

    通信教程03_I2C簡史,基礎原理及協(xié)議
    的頭像 發(fā)表于 02-05 13:14 ?3028次閱讀

    通信教程02 幾種常見串行通信及基礎原理

    通信教程02_幾種常見串行通信及基礎原理
    的頭像 發(fā)表于 02-26 16:12 ?9932次閱讀

    通信教程01 什么是并行通信?什么是串行通信?

    通信教程01_什么是并行通信?什么是串行通信?
    的頭像 發(fā)表于 02-26 16:27 ?1.2w次閱讀

    ROS與STM32通信

    ROS與STM32通信2020.8.1主要內容制作ROS包,將控制命令傳給STM32,并將接收到的數(shù)據(jù)作為話題進行發(fā)布STM32接收數(shù)據(jù)并將姿態(tài)數(shù)據(jù)傳回給ROS接收:期望角速度、期望線
    發(fā)表于 12-24 19:00 ?12次下載
    <b class='flag-5'>ROS</b>與STM32<b class='flag-5'>通信</b>

    Arduino長距離通信教程–LoRaLib庫

    為了控制 Arduino長距離通信教程–LoRenz 開發(fā)板中構建的LoRenz開發(fā)板,我開發(fā)了LoRaLib——用于SX1278芯片的開源Arduino庫。
    的頭像 發(fā)表于 02-24 09:51 ?1704次閱讀
    Arduino長距離<b class='flag-5'>通信教</b>程–LoRaLib庫

    ROS1的通信架構的基礎通信方式及相關概念

    ROS通信架構是ROS的靈魂所在,它包括數(shù)據(jù)處理,進程運行,消息傳遞等** 。這篇文章主要介紹ROS1的通信架構的基礎
    的頭像 發(fā)表于 05-19 17:23 ?3506次閱讀
    <b class='flag-5'>ROS</b>1的<b class='flag-5'>通信</b>架構的基礎<b class='flag-5'>通信</b>方式及相關概念

    ROS移動底盤通信試驗內容

    ROS與STM32的通信流程如圖所示 主要包含兩個方面: 小車里程計數(shù)據(jù)的上傳與接收 控制指令的下發(fā)與接收 1.原始消息內容 在ROS中,里程計數(shù)據(jù)主要包括機器人的位姿(位置和姿態(tài)),以及機器人
    的頭像 發(fā)表于 11-16 16:36 ?456次閱讀
    <b class='flag-5'>ROS</b>與<b class='flag-5'>移動</b><b class='flag-5'>底盤</b>的<b class='flag-5'>通信</b>試驗內容

    ROS通信接口機制介紹

    ROS通信接口 接口可以讓程序之間的依賴降低,便于我們使用別人的代碼,也方便別人使用我們的代碼,這就是ROS的核心目標,減少重復造輪子。 ROS有三種常用的
    的頭像 發(fā)表于 12-01 15:03 ?889次閱讀
    <b class='flag-5'>ROS</b><b class='flag-5'>通信</b>接口機制介紹