linux中can,pcap数据录制与udp缓存空间配置

1.简单粗暴的数据录制

录制点云数据:

sudo tcpdump -i ens33 -w /home/cyun/my_lidar.pcap udp port 6699 or udp port 7788

录制can报文:

candump -l can2
canplayer -I candump-2023-xx-xx_xxxxxx.log
# 格式: canplayer -I <文件> 录制时can=现在想要的输出的can口
canplayer -I my_drive_data.log i can2=can0

2.创建虚拟can口与can消息的发布

# 1. 加载虚拟CAN模块
sudo modprobe vcan

# 2. 创建一个叫 can2 的虚拟接口 (名字必须和你 python 代码里的一致!)
sudo ip link add dev can2 type vcan

# 3. 启动接口
sudo ip link set up can2

发送测试

# -g 10: 间隔10ms发一次
# -I 32A: 指定发送 ID 为 32A (810)
# -D r: 数据内容随机
cangen can2 -g 10 -I 32A -D r

3.通信频率与缓存空间

rslidar(以及绝大多数激光雷达)使用的是 UDP 协议 进行大量数据的传输。
这个问题一般就是ros2 topic hz 打印出来的频率是非常低的(大概是6-7hz的样子)
如果发现频率问题,那就一定是这个缓存空间不够的问题了,我们需要设置一下linux的默认缓存空间(注意:这里有个误区,千万不要去配置dds的yaml给ros2!!!)

sudo gedit /etc/sysctl.d/10-cyclone-max.conf
# ------------- 雷达 UDP 缓冲区优化 -------------
# UDP 接收缓冲区(读缓冲区,工控机收雷达数据)
net.core.rmem_default = 26214400  # 默认 25MB
net.core.rmem_max = 67108864      # 最大 64MB

# UDP 发送缓冲区(写缓冲区,DDS 发点云数据)
net.core.wmem_default = 26214400  # 默认 25MB
net.core.wmem_max = 67108864      # 最大 64MB

使设置生效:

sudo sysctl -p /etc/sysctl.d/10-cyclone-max.conf

关机之后再次开机,然后查看设置是否保留了下来:

# 默认接收缓冲区大小(全局)
sysctl net.core.rmem_default

# 最大接收缓冲区大小(全局上限)
sysctl net.core.rmem_max
# 默认发送缓冲区大小(全局)
sysctl net.core.wmem_default

# 最大发送缓冲区大小(全局上限)
sysctl net.core.wmem_max

结果如下所示
在这里插入图片描述

Linux 系统下使用 `pcap` 库获取 UDP 数据包的大小,可以通过解析捕获到的数据包的结构来实现。`pcap` 库提供了一种机制来捕获网络接口上的原始数据包,并允许用户通过编程方式访问其内容。具体步骤如下: ### 1. 初始化 pcap 设备 首先需要选择一个网络接口来捕获数据包。可以通过 `pcap_lookupdev` 函数获取默认的网络接口名称,或者通过 `pcap_findalldevs` 函数获取所有可用的网络接口列表。例如: ```c pcap_t *handle; char errbuf[PCAP_ERRBUF_SIZE]; handle = pcap_open_live("eth0", BUFSIZ, 1, 1000, errbuf); ``` 这里的 `"eth0"` 是网络接口名称,可以根据实际情况进行替换。 ### 2. 设置过滤器 为了只捕获 UDP 数据包,可以使用 `pcap_compile` 和 `pcap_setfilter` 函数设置一个 BPF 过滤器。例如: ```c struct bpf_program fp; char filter_exp[] = "udp"; pcap_compile(handle, &fp, filter_exp, 0, PCAP_NETMASK_UNKNOWN); pcap_setfilter(handle, &fp); ``` ### 3. 捕获数据包 使用 `pcap_loop` 或 `pcap_next` 函数开始捕获数据包。每当捕获到一个数据包时,回调函数会被触发。例如: ```c pcap_loop(handle, 0, got_packet, NULL); ``` ### 4. 解析数据包 在回调函数中,可以解析数据包的以太网头部、IP 头部和 UDP 头部,以获取 UDP 数据包的大小。UDP 头部中包含了一个长度字段,该字段表示整个 UDP 数据包(包括头部和数据)的长度。 ```c void got_packet(u_char *args, const struct pcap_pkthdr *header, const u_char *packet) { struct ether_header *eth_hdr = (struct ether_header *)packet; if (ntohs(eth_hdr->ether_type) == ETHERTYPE_IP) { struct ip *ip_hdr = (struct ip *)(packet + sizeof(struct ether_header)); if (ip_hdr->ip_p == IPPROTO_UDP) { struct udphdr *udp_hdr = (struct udphdr *)((u_char *)ip_hdr + (ip_hdr->ip_hl << 2)); int udp_len = ntohs(udp_hdr->uh_ulen); printf("UDP packet length: %d\n", udp_len); } } } ``` ### 5. 清理资源 捕获完成后,需要关闭 pcap 句柄并释放相关资源: ```c pcap_close(handle); ``` 通过上述步骤,可以在 Linux 系统下使用 `pcap` 库获取 UDP 数据包的大小。这种方法不仅适用于 UDP 数据包,还可以扩展到其他类型的网络数据包分析[^1]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

白云千载尽

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

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

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

打赏作者

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

抵扣说明:

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

余额充值