在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

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

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

3天內不再提示

ROS與移動底盤通信教程

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

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

實驗環境:

· 軟件環境:Ubuntu18.04 + ROS melodic、Windows + Keil 5、VSCode

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

01 實驗內容

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

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

主要包含兩個方面:

· 小車里程計數據的上傳與接收

· 控制指令的下發與接收

1.1 原始消息內容

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

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

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

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

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

1.2 轉換為字節數組

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

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

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

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

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

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

利用這個性質,我們就可以實現float與字節數據的互相轉換。

1.3 添加幀頭和校驗碼

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

準備工作:

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

02 里程計數據的上傳與接收

2.1 通信協議

里程計數據格式(19字節)

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

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


  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++)
  {
// 將要發送的數據賦值給聯合體的float成員
// 相應的就能更改字節數組成員的值
    _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;
// 串口發送
  SendData(data_to_send, _cnt); 
}
2.3 ROS端

采用狀態機的方式來接收STM32端上傳的里程計數據,每讀取一字節數據,則在狀態機中處理一次,部分程序如下:
uint8_tbuffer=0;
ser.read(&buffer, 1); // ser是串口類的一個實例,該語句表示從串口中讀取一個字節
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)
    {
/* 進行數據校驗 */
uint8_t checkout = 0;
for(int k = 0; k < data_cnt - 1; k++)
        {
            checkout += data_receive[k];
        }
if(checkout == data_receive[data_cnt - 1]) // 串口接收到的最后一個字節是校驗碼 
        {
/* 校驗通過,進行解碼 */
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 里程計數據可視化

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

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

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

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;
? 在機器人中,一般使用四元數/旋轉矩陣的形式來表示機器人的姿態,而不是歐拉角形式。所以需要將STM32返回的偏航角轉換為四元數,程序如下:
/*對速度進行積分得到位移*/
// 獲取當前時間
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;
? 以上就獲取了完整的機器人里程計數據,接下來需要將里程計數據發布到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;
// 發布坐標變換
odom_broadcaster.sendTransform(odom_trans);


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


// 設置機器人的位置和姿態
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;


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

配置完成后,重新打開一個終端,輸入: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 控制指令的下發與接收

3.1 通信協議

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

3.2 ROS端

在ROS端,首先需要接收從其他節點的控制消息,在ROS中常常使用geometry_msgs::Twist來傳遞控制指令,該消息格式包括兩個三維向量,如下,分別是三軸線速度和三軸角速度:
geometry_msgs/Vector3linear
geometry_msgs/Vector3 angular
? 我們在控制指令的消息回調函數中,將控制指令下發給STM32,部分程序如下,其中使用了C++的lambda表達式來替換回調函數
ros::Subscribersub=nh.subscribe("/cmd_vel",10,[&](constgeometry_msgs::ConstPtr&cmd_vel){
uint8_t _cnt = 0;
    data_u _temp; // 聲明一個聯合體實例,使用它將待發送數據轉換為字節數組
uint8_t data_to_send[100]; // 待發送的字節數組    
    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++){
// 將要發送的數據賦值給聯合體的float成員
// 相應的就能更改字節數組成員的值
        _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;
// 串口發送
    ser.write(data_to_send, _cnt);
});
? 最后,創建一個控制指令發布節點,用來發布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節點,可以得到如下的節點圖:
1d7d1234-c1a0-11ed-bfe3-dac502259ad0.png

3.3 STM32端

部分程序如下:
/**
 * @brief 從串口讀取單個字節
 * @param  data             讀取的字節數據
 */
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 數據解碼
 * @param  data             待解碼數組
 */
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;  
}

審核編輯:湯梓紅

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

    關注

    211

    文章

    28557

    瀏覽量

    207683
  • 通信
    +關注

    關注

    18

    文章

    6046

    瀏覽量

    136214
  • STM32
    +關注

    關注

    2270

    文章

    10915

    瀏覽量

    356741
  • Ubuntu
    +關注

    關注

    5

    文章

    564

    瀏覽量

    29908
  • ROS
    ROS
    +關注

    關注

    1

    文章

    279

    瀏覽量

    17039

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

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

收藏 人收藏

    評論

    相關推薦

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

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

    ROS與STM32是如何進行通信

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

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

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

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

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

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

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

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

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

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

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

    通信教程03_I2C簡史 基礎原理及協議

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

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

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

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

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

    ROS與STM32通信

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

    Arduino長距離通信教程–LoRaLib庫

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

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

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

    ROS移動底盤通信試驗內容

    ROS與STM32的通信流程如圖所示 主要包含兩個方面: 小車里程計數據的上傳與接收 控制指令的下發與接收 1.原始消息內容 在ROS中,里程計數據主要包括機器人的位姿(位置和姿態),以及機器人
    的頭像 發表于 11-16 16:36 ?454次閱讀
    <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有三種常用的
    的頭像 發表于 12-01 15:03 ?886次閱讀
    <b class='flag-5'>ROS</b><b class='flag-5'>通信</b>接口機制介紹
    主站蜘蛛池模板: 亚色在线| 国产情侣自拍小视频| 日处女穴| www.夜| 午夜大片男女免费观看爽爽爽尤物| 俺来也久久| 经典三级一区在线播放| 久久夜色精品国产噜噜| 久久婷婷成人综合色| 国产资源站| 免费一级欧美片片线观看| 特级一级毛片免费看| 男人j桶进女人免费视频| 欧美猛性| 亚欧精品一区二区三区| 欧美精品区| 日本免费不卡视频| 亚洲一区二区三区中文字幕| 亚洲专区一路线二| 免费亚洲视频在线观看| 香蕉视频在线观看黄| 日本拍拍| 日本黄色美女视频| 色吧综合| 精品在线视频一区| 激情五月播播| 中文字幕久久精品波多野结| 奇米影视四色7777| 日本黄色大全| 在线播放免费人成毛片乱码| 性猛交毛片| 男男生子大肚play做到生| 五月婷婷丁香久久| 四虎影院欧美| 黄色三级视频| 天天夜夜久久| 美女福利在线观看| 国产精品福利一区二区亚瑟 | 欧美综合国产精品日韩一| 色多多视频在线观看免费大全| 美女一区二区三区|