Linux tun/tap 驱动多队列模式(C/C++)

2024-03-04 06:44

本文主要是介绍Linux tun/tap 驱动多队列模式(C/C++),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

LINUX 内核在3.X版本才支持,正式是到 4.1 版本内核才加入到内核之中被支持,所以相对来说距今时间上来说是比较新的一种扩展技术。

多队列模式的 tun/tap 可以显著的提高,单个 tun/tap 网卡的带宽吞吐速度,但肯定没有 DPDK 那种零拷贝不内核切换的效率高。

一般来说,tun/tap 的效率与内核直接处理网卡数据速度,大约差距至少5倍左右,即 1000Mbps,tun/tap 最大用户层可以达到 200Mbps,实际上大多数情况达不到。

以玩客云S805晶片为例,单核拉满在300Mbps 左右,三核拉满达到最大速度就是900Mbps,虽然是1000Mbps 的网卡口子,但是芯片处理不了那么大的数据吞吐量。

四核拉满跑 tun/tap 最大吞吐速度大约在 350Mbps 左右,即双工 160 ~ 170Mbps 下行速度,上传倒是可以爆表达到 190 ~ 200Mbps。

这里是指用多队列模式榨干CPU、并且需要跑AES-256-GCM/CFB这样的加密算法,如果不跑这些会更好一点,可能在 200 ~ 230Mbps 左右。

在这种机子上面 tun/tap 驱动损失的转发效率可能没有那么大,但是一般大约就是五倍的性能差距,反正按着算大体不会差太多。

重点:

1、多队列模式下,可以重复打开相同的 tun/tap 网卡

2、多队列模式下,每个tun/tap 设备句柄会随机收取到内核派发的包

(所以不存在主从这样的接口)

3、多队列模式下,每个 tun/tap 设备句柄可以被 epoll 附加监听读写事件

4、多队列模式下,每个 tun/tap 设备句柄都可以单独设置为 “非阻塞” 或 “阻塞模式”

5、多队列模式下,每个 tun/tap 设备句柄都可以单独被写入IP数据包

每个 tun/tap 设备句柄被单独的 read、write 都涉及到内核加锁的过程,所以如果多线程并行交叉读写这些 tun/tap 设备句柄效率会很低,因为内核会上锁等,所以最好的办法是,一个队列一个 tun/tap io 线程来操作,读写都在同个线程,跨线程会严重影响效率,建议结合 epoll 来用。

每个 tun/tap 设备句柄都应该设置为非阻塞,并且写入不建议用 epoll,直接write向内核写入数据包就行了,同时 tun/tap 驱动在Linux内核,非阻塞只作用在读,写是不存在丢包的,调用一次都会调用到内核驱动的 tun_write 函数,它是无视非阻塞设置的。

源实现:(打开多队列模式的 tun/tap、多队列可以重复打开 tun/tap 句柄)

        int TapLinux::OpenDriver(const char* ifrName) noexcept {if (NULL == ifrName || *ifrName == '\x0') {ifrName = "tun%d";}// __oflagint __open_flags = O_RDWR | O_NONBLOCK;
#if defined(O_CLOEXEC)__open_flags |= O_CLOEXEC;
#endifint tun = open("/dev/tun", __open_flags);if (tun == -1) {tun = open("/dev/net/tun", __open_flags);if (tun == -1) {return -1;}}Socket::SetNonblocking(tun, true);ppp::unix__::UnixAfx::set_fd_cloexec(tun);struct ifreq ifr;memset(&ifr, 0, sizeof(ifr));// By default, try to enable tun/tap-driver multi-queue mode, if not single-queue mode.// https://www.kernel.org/doc/Documentation/networking/tuntap.txtstrncpy(ifr.ifr_name, ifrName, IFNAMSIZ);#if defined(IFF_MULTI_QUEUE)ifr.ifr_flags = IFF_TUN | IFF_NO_PI | IFF_MULTI_QUEUE;bool fails = ioctl(tun, TUNSETIFF, &ifr) < 0;if (fails) {ifr.ifr_flags = IFF_TUN | IFF_NO_PI;fails = ioctl(tun, TUNSETIFF, &ifr) < 0;}
#elseifr.ifr_flags = IFF_TUN | IFF_NO_PI;bool fails = ioctl(tun, TUNSETIFF, &ifr) < 0;
#endifif (fails) {::close(tun);return -1;}else {return tun;}}

这篇关于Linux tun/tap 驱动多队列模式(C/C++)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/772319

相关文章

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

usb接口驱动异常问题常用解决方案

《usb接口驱动异常问题常用解决方案》当遇到USB接口驱动异常时,可以通过多种方法来解决,其中主要就包括重装USB控制器、禁用USB选择性暂停设置、更新或安装新的主板驱动等... usb接口驱动异常怎么办,USB接口驱动异常是常见问题,通常由驱动损坏、系统更新冲突、硬件故障或电源管理设置导致。以下是常用解决

Redis消息队列实现异步秒杀功能

《Redis消息队列实现异步秒杀功能》在高并发场景下,为了提高秒杀业务的性能,可将部分工作交给Redis处理,并通过异步方式执行,Redis提供了多种数据结构来实现消息队列,总结三种,本文详细介绍Re... 目录1 Redis消息队列1.1 List 结构1.2 Pub/Sub 模式1.3 Stream 结

Linux命令之firewalld的用法

《Linux命令之firewalld的用法》:本文主要介绍Linux命令之firewalld的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux命令之firewalld1、程序包2、启动firewalld3、配置文件4、firewalld规则定义的九大

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

Linux内核参数配置与验证详细指南

《Linux内核参数配置与验证详细指南》在Linux系统运维和性能优化中,内核参数(sysctl)的配置至关重要,本文主要来聊聊如何配置与验证这些Linux内核参数,希望对大家有一定的帮助... 目录1. 引言2. 内核参数的作用3. 如何设置内核参数3.1 临时设置(重启失效)3.2 永久设置(重启仍生效

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指