ROS stm32 CAN通信

运行环境:

ubuntu18.04.melodic
STM32:DJI Robomaster C板
ROS:18.04
硬件:USB-CAN(选支持Linux驱动的)
在这里插入图片描述

原理

1.1 ros中的代码

1)socketcan_bridge

http://wiki.ros.org/socketcan_bridge

主要利用socketcan_bridge_node节点,相当于ros和stm32桥梁作用

在这里插入图片描述

原理解释:
Subscribed Topics
sent_messages (can_msgs/Frame)
它可以监听话题为sent_messages,消息类型为can_msgs/Frame的数据

原理解释:
Published Topics
received_messages (can_msgs/Frame)
将监听到的sent_messages话题,消息类型为can_msgs/Frame的数据发送到can总线上(以便stm32的can回调函数接收can数据)

2)测试的ros-python包

写一个python包。发布话题为sent_messages,消息类型为can_msgs/Frame的数据

注意标识符要和stm32的一样,这里都是设置成0x208
 msg.id = 0x208
        msg.dlc = 8  # 数据字段的大小,单位是字节
        msg.is_error = False
        msg.is_rtr = False
        msg.is_extended = False

再到stm32中的can回调函数解析这里的数据
data_to_send = [0x01,0x03,0x03,0x03,0x03,0x03,0x03,0x03]
如果要获取第一个数据,就data[0]这样,就可以得到0x01的值,也就算十进制的1, 我这里只是解析一位来控制电机转动

在这里插入图片描述
详细代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-

import rospy
from can_msgs.msg import Frame
import struct

def send_can_frame():
    # 初始化 ROS 节点
    rospy.init_node('can_frame_sender', anonymous=True)

    # 创建一个发布者,发布 can_msgs/Frame 类型的消息到目标 ROS 话题
    pub = rospy.Publisher('/sent_messages', Frame, queue_size=10)

    # 设置循环速率,这里设为 10Hz,根据需求可以调整
    rate = rospy.Rate(10)

    while not rospy.is_shutdown():
        # 创建一个 can_msgs/Frame 类型的消息
        msg = Frame()

        # 设置 can::Frame 对象的属性,这里以设置标识符为 0x205 为例
        msg.id = 0x208
        msg.dlc = 8  # 数据字段的大小,单位是字节
        msg.is_error = False
        msg.is_rtr = False
        msg.is_extended = False

        # 只解析第一位
        # data_to_send = [0x04,0x03,0x03,0x03,0x03,0x03,0x03,0x03]
        data_to_send = [0x01,0x03,0x03,0x03,0x03,0x03,0x03,0x03]

        # 将要发送的数据赋值给 msg.data
        msg.data = data_to_send

        # 发布消息到 ROS 话题
        pub.publish(msg)

        # 等待指定的循环速率
        rate.sleep()

        # 打印成功发布的提醒消息
        rospy.loginfo("Successfully published CAN frame with ID 0x208")

if __name__ == '__main__':
    try:
        send_can_frame()
    except rospy.ROSInterruptException:
        pass

在这里插入图片描述

在这里插入图片描述

3)keil5中数据解析

rx_data[0]解析得是ros发布得数据 data_to_send = [0x01,0x03,0x03,0x03,0x03,0x03,0x03,0x03],也就是得到十进制得0x01,代表数字1。

void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan)
{
  // 定义CAN消息的头部和数据
  CAN_RxHeaderTypeDef rx_header;
  uint8_t rx_data[8];

  // 检查CAN总线实例是否为CAN1
  if (hcan->Instance == CAN1)
  {
    // 获取CAN消息的头部信息和数据
    HAL_CAN_GetRxMessage(hcan, CAN_RX_FIFO0, &rx_header, rx_data); // rx_data接收CAN总线上发送来的数据,不同的位代表不同的信息

    // 根据接收到的消息的标准帧ID执行相应的处理
    switch (rx_header.StdId) //如果接收的rx_header.StdId值等于0x205,则接收can发送过来的数据。可以设置ros发送的can标识符为为0x208 (ros如何发送信息到can中)
    {
    case 0x205: // 如果标准帧ID为0x205
    {
      // 解析接收到的数据,并存储到相应的数据结构中
      motor_yaw_info.rotor_angle = ((rx_data[0] << 8) | rx_data[1]);
      motor_yaw_info.rotor_speed = ((rx_data[2] << 8) | rx_data[3]);
      motor_yaw_info.torque_current = ((rx_data[4] << 8) | rx_data[5]);
      motor_yaw_info.temp = rx_data[6];	
			
      break;
    }
		
		case 0x208: 
   {
     
	  // 处理接收到的 target_yaw_angle 数据
      uint8_t received_target_yaw_angle = rx_data[0];  // 假设 target_yaw_angle 在数据的第一个字节
      // 在这里添加处理 received_target_yaw_angle 的代码
		  target_yaw_angle =  received_target_yaw_angle;
      break;
    }
 
		
    }
  }
}

4)USB-CAN连接

# 安装gs_usb 内核模块
sudo modprobe gs_usb
# 插入USB-CAN后执行下面步骤:
 查看can设备
ifconfig -a

# 设置波特率100M(can设备参数)-和stm32cubemx can配置的波特率一样
sudo ip link set can0 up type can bitrate 1000000

捕捉can信号
candump can0 

在这里插入图片描述
在这里插入图片描述

5)启动指令

#启动socketcan_bridge_node.cpp节点
roscore

duduzai@duduzai:~/Downloads/ros_can_ws$ source ./devel/setup.bash

duduzai@duduzai:~/Downloads/ros_can_ws$ rosrun socketcan_bridge socketcan_bridge_node
#启动python包
duduzai@duduzai:~/Downloads/target_yaw_angle_ws/src/target_yaw_angle/scripts$ chmod +x target_yaw_angle_pub.py

duduzai@duduzai:~/Downloads/target_yaw_angle_ws$ source ./devel/setup.bash

duduzai@duduzai:~/Downloads/target_yaw_angle_ws$ rosrun target_yaw_angle target_yaw_angle_pub.py

⭐⭐⭐ 嘟嘟崽 ⭐⭐⭐
⭐⭐⭐ 祝你成功 ⭐⭐⭐
  • 2
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
ROS(机器人操作系统)是一个用于开发机器人软件的框架,而STM32是一款常见的嵌入式微控制器。它们可以进行通信以实现机器人系统的控制和数据传输。 要实现ROSSTM32通信,通常有两种方式:串口通信ROS网络通信。 一种常见的方式是通过串口连接ROS主机和STM32ROS主机可以使用基于Linux系统的计算机,如Ubuntu等。使用ROS提供的串口通信库,可以在ROS主机上编写节点程序,通过串口与STM32进行数据交换。在ROS主机上,可以将STM32作为一个外设设备接入ROS系统,通过串口读取STM32发送的数据,并将ROS主机的控制指令发送给STM32。 另一种方式是通过ROS网络通信。在STM32上运行一个ROS节点,该节点通过TCP/IP协议与ROS主机上的其他ROS节点进行通信ROS节点可以通过STM32上的网卡或Wi-Fi模块连接到ROS主机所在的局域网。在ROS主机上,可以使用ROS提供的网络通信库与STM32节点进行通信,发送控制指令或接收传感器数据。 不论是串口通信还是网络通信ROSSTM32通信都需要定义消息格式。可以根据具体的需求,定义自己的ROS消息类型,包括控制指令、传感器数据等。在ROS主机上,可以使用ROS消息库来解析和处理这些消息。 总之,通过串口通信ROS网络通信,可以实现ROSSTM32通信,实现机器人系统的控制和数据传输。这种通信方式可以用于各种机器人应用,如无人车、机器人臂等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

圆嘟嘟2019

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值