(六)CAN总线通讯
文章目录
- CAN总线回环测试
- 第一种基于板载CAN测试
- 第一步确认板载是否支持
- 第二步关闭 CAN 接口将 CAN 接口置于非活动状态
- 第三步 配置 CAN 接口
- 第一步 设置 CAN 接口比特率
- 第二步 设置 CAN 启用回环模式
- 第三步 启用 CAN 接口
- 第四步 测试CAN总线回环
- 捕获 CAN 消息
- 发送 CAN 消息
- 第二种基于Linux库的原生开发
- 1. 确认硬件支持
- 2. 配置 CAN 接口
- 3. 使用 SocketCAN API 编写应用程序
- 创建 CAN 套接字
- 绑定 CAN 接口
- 发送和接收 CAN 消息
- 示例代码
- 4. 应用程序测试
CAN总线回环测试
第一种基于板载CAN测试
第一步确认板载是否支持
确保你的硬件支持 CAN,并且已经正确安装了驱动程序。可以通过以下命令检查是否检测到了 CAN 设备:
ip link show type can
可以看到板载自带了can0和can1.
can0 和 can1:是接口的名称,分别表示系统中的第一个和第二个 CAN 接口。
<NOARP,ECHO>:
NOARP:表示该接口不使用 ARP(地址解析协议)。CAN 总线不需要 ARP 因为它不是基于 IP 的网络。
ECHO:可能启用了回环模式,发送的消息会被自己的 CAN 控制器重新接收。这里不确定是否启用了,可以用指令ip -details link show 查看到:
或者单看can0,can1的状态,以can0为例子:
ip -details link show can0
mtu 16:最大传输单元(Maximum Transmission Unit),对于 CAN 接口来说MTU 通常是 16 字节,这是因为 CAN 消息的最大有效载荷是 8 字节,加上一些额外的头部信息。
qdisc noop:队列调度算法(Queueing Discipline),这里使用的是 noop,即空操作队列调度器。这意味着没有任何包调度策略被应用,所有数据包将直接传递而不会排队。
state DOWN:当前接口状态为关闭(DOWN),意味着接口未激活,无法进行通信。要激活接口,可以使用 ip link set can0 up 或 ip link set can1 up。
mode DEFAULT:表示接口的工作模式,默认情况下是标准模式。
group default:指明这个接口属于哪个组,default 是默认组。
qlen 10:队列长度(Queue Length),表示可以排队等待处理的数据包数量,在这里是 10。
链路类型
link/can:表明这是一个 CAN 类型的接口。
第二步关闭 CAN 接口将 CAN 接口置于非活动状态
如果没有激活的情况,直接跳过此步
ip link set can0 down
第三步 配置 CAN 接口
第一步 设置 CAN 接口比特率
ip link set can0 type can bitrate 500000
ip link set:这是一个用来配置网络接口的命令。
can0:指定要配置的网络接口名称,这里是 can0。
type can:指明该接口是 CAN 类型的。
bitrate 500000:设置 CAN 总线的数据传输速率(比特率)为 500 kbps。这个值应该根据你的硬件支持和需求进行调整。
设置完可以借助指令查看:
ip -details link show can0
第二步 设置 CAN 启用回环模式
指令:
ip link set can0 type can loopback on
loopback on:启用回环模式。在这种模式下,所有从 CAN 控制器发出的消息都会被重新送回到同一个控制器,而不会真正发送到物理总线上。这对于测试非常有用,因为它允许你在不连接任何其他设备的情况下验证软件是否正常工作。
设置完同样可以借助指令查看:
ip -details link show can0
第三步 启用 CAN 接口
指令:
ip link set can0 up
启动成功后同样可以借助指令查看:
ip -details link show can0
第四步 测试CAN总线回环
捕获 CAN 消息
使用 nohup 和 candump 捕获消息并记录到文件
nohup candump can0 > log.txt &
nohup:使程序在用户退出终端后继续运行。这对于长时间运行的任务很有用。
candump:这是 Linux 下的一个工具,用于捕获并显示 CAN 消息。它会实时监听指定的 CAN 接口,并输出接收到的消息。
can0:指定要监听的 CAN 接口。
log.txt:将 candump 的输出重定向到一个名为 log.txt 的文件中,而不是打印到屏幕上。
&:将命令放入后台执行,这样可以在同一终端窗口中继续输入其他命令。
发送 CAN 消息
标准数据帧发8位数据:
cansend can0 0B000123#00.00.00.00.00.00.00.01
#后面每一位数据不加.也行,注意别发错了
cansend:这是 Linux 下的一个工具,用于向指定的 CAN 接口发送 CAN 消息。
can0:指定要使用的 CAN 接口。
0B000123#00.00.00.00.00.00.00.01:这是要发送的 CAN 消息格式。
具体来说:0B000123是 CAN ID,其中 0B 表示标准帧格式(11位ID),000123是具体的 ID 值。
#分隔符,后面跟着的是数据字段。00.00.00.00.00.00.00.01 是数据字段的内容,表示 8 字节的数据。每个字节用两位十六进制数表示。
其实都用默认,就是标准数据帧:
cansend can0 123#00.00.00.00.00.00.00.02
数据也可以发0-8任意字节:
cansend can0 234#88
第二种基于Linux库的原生开发
在 Linux 下进行 CAN 总线应用开发时,通常需要通过系统调用和特定的套接字 API 来与 CAN 接口交互。Linux 内核提供了一个叫做 SocketCAN 的子系统,它使得 CAN 通信可以像普通的网络编程一样使用标准的 BSD 套接字接口来实现。下面是进行 CAN 应用开发的基本步骤:
1. 确认硬件支持
ip link show type can
加粗样式
可以看到板载自带了can0和can1.
2. 配置 CAN 接口
先关闭can接口
ip link set can0 down
配置波特率:
ip link set can0 type can bitrate 500000
如果使用回环如下操作,不使用或者使用正常模式就跳过此步,我这里采用回环测试因此执行此步骤:
ip link set can0 type can loopback on
启动can接口
ip link set can0 up
至此所有配置完成
3. 使用 SocketCAN API 编写应用程序
创建 CAN 套接字
在 C/C++ 中,可以通过 socket() 函数创建一个 CAN 套接字。例如:
#include <sys/socket.h>
#include <linux/can.h>
#include <linux/can/raw.h>int s;
if ((s = socket(PF_CAN, SOCK_RAW, CAN_RAW)) < 0) {perror("Socket creation failed");return -1;
}
绑定 CAN 接口
接下来,将套接字绑定到具体的 CAN 接口(如 can0)。这可以通过 bind() 函数完成:
struct ifreq ifr;
struct sockaddr_can addr;addr.can_family = AF_CAN;
strcpy(ifr.ifr_name, "can0");if (ioctl(s, SIOCGIFINDEX, &ifr) < 0) {perror("Interface index get failed");close(s);return -1;
}addr.can_ifindex = ifr.ifr_ifindex;if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) < 0) {perror("Bind to CAN interface failed");close(s);return -1;
}
发送和接收 CAN 消息
struct can_frame frame;// 构造要发送的 CAN 帧
frame.can_id = 0x123; // CAN ID
frame.can_dlc = 8; // 数据长度码(DLC),表示数据域的字节数
memset(frame.data, 0x01, frame.can_dlc); // 设置数据域内容// 发送 CAN 帧
if (write(s, &frame, sizeof(struct can_frame)) != sizeof(struct can_frame)) {perror("Write CAN frame failed");close(s);return -1;
}// 接收 CAN 帧
if (read(s, &frame, sizeof(struct can_frame)) != sizeof(struct can_frame)) {perror("Read CAN frame failed");close(s);return -1;
}printf("Received CAN ID=%x, DLC=%d\n", frame.can_id, frame.can_dlc);
for (int i = 0; i < frame.can_dlc; ++i)printf("Data[%d]: %02x\n", i, frame.data[i]);
示例代码
下面是一段完整的示例代码,展示了如何创建 CAN 套接字、绑定接口、发送和接收消息:
这里我开启了线程,父线程用来写,子线程用来读;代码被屏蔽掉的那部分是,禁止回环模式的,为了防止配置的时候关闭回环失败,因此软件上再关闭一次。
#include "can_config.h"#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <signal.h>
#include <libgen.h>
#include <getopt.h>
#include <limits.h>#include <sys/types.h>
#include <sys/socket.h>
#include <sys/uio.h>
#include <sys/ioctl.h>
#include <net/if.h>#include <linux/can.h>
#include <linux/can/raw.h>int main()
{int s; // 就是fdint n_read = 0; // 读取到的数据个数//创建 CAN 套接字if ((s = socket(PF_CAN, SOCK_RAW, CAN_RAW)) < 0) {perror("Socket creation failed");return -1;}printf("socket ok ========================\n");/*// 禁用回环模式(如果需要)int loopback = 0; // 0 表示关闭回环模式if (setsockopt(s, SOL_CAN_RAW, CAN_RAW_LOOPBACK, &loopback, sizeof(loopback)) < 0) {perror("Failed to disable loopback mode");close(s);return -1;}// 禁用监听自己的消息(可选)int recv_own_msgs = 0; // 0 表示关闭接收自己的消息if (setsockopt(s, SOL_CAN_RAW, CAN_RAW_RECV_OWN_MSGS, &recv_own_msgs, sizeof(recv_own_msgs)) < 0) {perror("Failed to disable receiving own messages");close(s);return -1;}*/// 绑定 CAN 接口struct ifreq ifr;struct sockaddr_can addr;addr.can_family = AF_CAN;strcpy(ifr.ifr_name, "can0");if (ioctl(s, SIOCGIFINDEX, &ifr) < 0) {perror("Interface index get failed");close(s);return -1;}addr.can_ifindex = ifr.ifr_ifindex;if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) < 0) {perror("Bind to CAN interface failed");close(s);return -1;}printf("bind ok ========================\n");// 构造要发送的 CAN 帧struct can_frame wirte_frame;struct can_frame recive_frame;wirte_frame.can_id = 0x123; // CAN IDwirte_frame.can_dlc = 8; // 数据长度码(DLC),表示数据域的字节数memset(wirte_frame.data, 0x01, wirte_frame.can_dlc); // 设置数据域内容int data = 0x01;//开启线程 一个接收一个发送;__pid_t pid;pid = fork();//返回的pid号 父进程是正数id号码,子进程是0// 父进程 write 子进程接收if(pid>0){printf("in farther ok ========================\n");while(1){// 发送 CAN 帧 间隔3秒写一次if (write(s, &wirte_frame, sizeof(struct can_frame)) != sizeof(struct can_frame)) {perror("Write CAN frame failed");close(s);return -1;}sleep(3);memset(wirte_frame.data,++data, wirte_frame.can_dlc); // 设置数据域内容} }else if(pid==0){printf("in son ok ========================\n");while(1){// 接收 CAN 帧if ((n_read = read(s, &recive_frame, sizeof(struct can_frame)) )!= sizeof(struct can_frame)) {perror("Read CAN frame failed");close(s);return -1;}if(n_read>0){printf("Received CAN ID=%x, DLC=%d\n", recive_frame.can_id, recive_frame.can_dlc);for (int i = 0; i < recive_frame.can_dlc; ++i)printf("Data[%d]: %02x\n", i, recive_frame.data[i]);}}}else{perror("fork error\n");}close(s);return 0 ;
}
4. 应用程序测试
先用交叉编译工具编译:
arm-linux-gnueabi-gcc mycan.c -o mycan
远程发送到板子:
scp ./mycan root@192.168.1.101:/root/zhua
执行:
适当改一改代码:
再测试:
这里有个小问题就是write一直写,FIFO文件队列会满,然后就是溢出报错,因此要学习错误处理等操作。
所以最后关于帧格式,位同步,仲裁,错误处理等可以深入了解。推荐在学习stm32的时候学习寄存器配置开发,可以深入的了解到。
相关文章:

(六)CAN总线通讯
文章目录 CAN总线回环测试第一种基于板载CAN测试第一步确认板载是否支持第二步关闭 CAN 接口将 CAN 接口置于非活动状态第三步 配置 CAN 接口第一步 设置 CAN 接口比特率第二步 设置 CAN 启用回环模式第三步 启用 CAN 接口 第四步 测试CAN总线回环捕获 CAN 消息发送 CAN 消息 第…...

新一代智能工控系统网络安全合规解决方案
01.新一代智能工控系统概述 新一代智能工控系统是工业自动化的核心,它通过集成人工智能、工业大模型、物联网、5G等技术,实现生产过程的智能化管理和控制。这些系统具备实时监控、自动化优化、灵活调整等特点,能够提升生产效率、保证产品质量…...

Vivado中Tri_mode_ethernet_mac的时序约束、分析、调整——(一)时序约束的基本概念
1、基本概念 推荐阅读,Ally Zhou编写的《Vivado使用误区与进阶》系列文章,熟悉基本概念、tcl语句的使用。 《Vivado使用误区与进阶》电子书开放下载!! 2、Vivado中的语法例程 1)语法例程 约束的语句可以参考vivado…...

车载网络:现代汽车的数字心跳
在汽车领域,“智能汽车”一词毫不夸张。如今的汽车已不再是原始的机械工程,而是通过先进的车载网络无缝连接的精密数字生态系统。这些滚动计算机由复杂的电子控制单元(ECU)网络提供动力,ECU是负责管理从发动机性能到信息娱乐系统等一切事务的…...

python基础和redis
1. Map函数 2. filter函数 numbers generate_numbers() filtered_numbers filter(lambda x: x % 2 0, numbers) for _ in range(5):print(next(filtered_numbers)) # 输出: 0 2 4 6 83. filter map 和 reduce 4. picking and unpicking 5. python 没有函数的重载࿰…...

w~自动驾驶~合集16
我自己的原文哦~ https://blog.51cto.com/whaosoft/12765612 #SIMPL 用于自动驾驶的简单高效的多智能体运动预测基准 原标题:SIMPL: A Simple and Efficient Multi-agent Motion Prediction Baseline for Autonomous Driving 论文链接:https://ar…...
最长的指定瑕疵度的元音子串
一、题目 最长的指定瑕疵度的元音子串 定义:开头和结尾都是元音字母(aeiouAEIOU)的字符串为 元音字符串 ,其中混杂的非元音字母数量为其 瑕疵度 。比如: “a” 、 "aa"是元音字符串,其瑕疵度都为0 "aiu…...

每日算法Day15【组合、组合总和III、电话号码的字母组合】
77. 组合 算法链接: 77. 组合 - 力扣(LeetCode) 类型: 回溯 难度: 中等 回溯三步法: 1、确定参数返回值 2、确定终止条件 3、单层搜索逻辑 剪枝操作: 当path容量超过k时的数据可以不用遍历,故遍历边界条件判断: …...

C语言教程——指针进阶(2)
目录 一、函数指针数组 1.1函数指针数组写法 1.2函数指针用途 二、指向函数指针数组的指针 2.1概念 三、回调函数 3.1用法 3.2qsort排序 总结 前言 我们接着上一篇的函数指针往下学习。 一、函数指针数组 1.1函数指针数组写法 我们都知道指针数组,里面可以…...
调和级数不为整数的证明
文章目录 1. 问题引入2. 证明2.1 引理12.2 引理22.3 引理3:2.4 核心证明: 3. 参考 1. 问题引入 s ( n ) 1 1 2 1 3 ⋯ 1 n , n ∈ N ∗ , n ≥ 2 s(n) 1\frac{1}{2}\frac{1}{3}\cdots\frac{1}{n}, \quad \\n \in N^*, n \ge2 s(n)12131⋯n1,…...

基于微信小程序的在线学习系统springboot+论文源码调试讲解
第4章 系统设计 一个成功设计的系统在内容上必定是丰富的,在系统外观或系统功能上必定是对用户友好的。所以为了提升系统的价值,吸引更多的访问者访问系统,以及让来访用户可以花费更多时间停留在系统上,则表明该系统设计得比较专…...

基于 Boost.Asio 和 Boost.Beast 的异步 HTTP 服务器(学习记录)
已完成功能: 支持 GET 和 POST 请求的路由与回调处理。 解析URL请求。 单例模式 管理核心业务逻辑。 异步 I/O 技术和 定时器 控制超时。 通过回调函数注册机制,可以灵活地为不同的 URL 路由注册处理函数。 1. 项目背景 1.1 项目简介 本项目是一个基于…...

有机物谱图信息的速查技巧有哪些?
谱图信息是化学家解读分子世界的“语言”,它们在化学研究的各个领域都发挥着不可或缺的作用。它们是理解和确定分子结构的关键,对化学家来说极为重要,每一种谱学技术都提供了不同的视角来观察分子,从而揭示其独特的化学和物理特性…...
Eureka缓存机制
一、Eureka的CAP特性 Eureka是一个AP系统,它优先保证可用性(A)和分区容错性(P),而不保证强一致性(C)。这种设计使得Eureka在分布式系统中能够应对各种故障和分区情况,保…...
【LC】78. 子集
题目描述: 给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。 解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。 示例 1: 输入:nums [1,2,3] 输出࿱…...

协同过滤算法私人诊所系统|Java|SpringBoot|VUE|
【技术栈】 1⃣️:架构: B/S、MVC 2⃣️:系统环境:Windowsh/Mac 3⃣️:开发环境:IDEA、JDK1.8、Maven、Mysql5.7 4⃣️:技术栈:Java、Mysql、SpringBoot、Mybatis-Plus、VUE、jquery,html 5⃣️…...
Docker部署Naocs-- 超细教程
Docker 拉取镜像 docker pull nacos/nacos-server:v2.2.0 挂载目录 如果不是root账号 前面加sudo 或者 切换root账号 su root(命令) mkdir -p /mydata/nacos/logs/ #新建logs目录 mkdir -p /mydata/nacos/conf/ #新建conf目录 启动容器…...

[java基础-集合篇]优先队列PriorityQueue结构与源码解析
优先队列PriorityQueue 优先级队列表示为平衡二进制堆: queue[n] 的两个子级是 queue[2*n1] 和 queue[2*(n1)]。 注:左子节点index2*parentIndex1,右子节点index2*parentIndex2,源码中计算parent位置时就是这样反过来计算的 优…...
12. C语言 数组与指针(深入理解)
本章目录: 前言1. 什么是数组?2. 数组的声明与初始化声明数组初始化数组 3. 访问数组元素遍历数组 4. 获取数组长度使用 sizeof 获取长度使用宏定义简化 5. 数组与指针数组名与指针的区别使用指针操作数组 6. 多维数组遍历多维数组 7. 数组作为函数参数8. 高级技巧与…...

Postman接口测试基本操作
🍅 点击文末小卡片 ,免费获取软件测试全套资料,资料在手,涨薪更快 Postman-获取验证码 需求:使用Postman访问验证码接口,并查看响应结果。 地址:http://kdtx-test.itheima.net/api/captchaIm…...

国防科技大学计算机基础课程笔记02信息编码
1.机内码和国标码 国标码就是我们非常熟悉的这个GB2312,但是因为都是16进制,因此这个了16进制的数据既可以翻译成为这个机器码,也可以翻译成为这个国标码,所以这个时候很容易会出现这个歧义的情况; 因此,我们的这个国…...
FastAPI 教程:从入门到实践
FastAPI 是一个现代、快速(高性能)的 Web 框架,用于构建 API,支持 Python 3.6。它基于标准 Python 类型提示,易于学习且功能强大。以下是一个完整的 FastAPI 入门教程,涵盖从环境搭建到创建并运行一个简单的…...

dedecms 织梦自定义表单留言增加ajax验证码功能
增加ajax功能模块,用户不点击提交按钮,只要输入框失去焦点,就会提前提示验证码是否正确。 一,模板上增加验证码 <input name"vdcode"id"vdcode" placeholder"请输入验证码" type"text&quo…...

转转集团旗下首家二手多品类循环仓店“超级转转”开业
6月9日,国内领先的循环经济企业转转集团旗下首家二手多品类循环仓店“超级转转”正式开业。 转转集团创始人兼CEO黄炜、转转循环时尚发起人朱珠、转转集团COO兼红布林CEO胡伟琨、王府井集团副总裁祝捷等出席了开业剪彩仪式。 据「TMT星球」了解,“超级…...

HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...

C# 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...

[免费]微信小程序问卷调查系统(SpringBoot后端+Vue管理端)【论文+源码+SQL脚本】
大家好,我是java1234_小锋老师,看到一个不错的微信小程序问卷调查系统(SpringBoot后端Vue管理端)【论文源码SQL脚本】,分享下哈。 项目视频演示 【免费】微信小程序问卷调查系统(SpringBoot后端Vue管理端) Java毕业设计_哔哩哔哩_bilibili 项…...

Golang——7、包与接口详解
包与接口详解 1、Golang包详解1.1、Golang中包的定义和介绍1.2、Golang包管理工具go mod1.3、Golang中自定义包1.4、Golang中使用第三包1.5、init函数 2、接口详解2.1、接口的定义2.2、空接口2.3、类型断言2.4、结构体值接收者和指针接收者实现接口的区别2.5、一个结构体实现多…...
小木的算法日记-多叉树的递归/层序遍历
🌲 从二叉树到森林:一文彻底搞懂多叉树遍历的艺术 🚀 引言 你好,未来的算法大神! 在数据结构的世界里,“树”无疑是最核心、最迷人的概念之一。我们中的大多数人都是从 二叉树 开始入门的,它…...

算法打卡第18天
从中序与后序遍历序列构造二叉树 (力扣106题) 给定两个整数数组 inorder 和 postorder ,其中 inorder 是二叉树的中序遍历, postorder 是同一棵树的后序遍历,请你构造并返回这颗 二叉树 。 示例 1: 输入:inorder [9,3,15,20,7…...