首页 > 系统相关 >linux下打满网卡的带宽和影响打满网卡的原因

linux下打满网卡的带宽和影响打满网卡的原因

时间:2024-07-01 09:59:07浏览次数:1  
标签:count usleep int 网卡 linux 打满 include port loop

 

1.打满网卡带宽的定义

网络传输场景很多,咱们这里用一个最常见的场景:

客户端发送数据,服务端接收数据,所以这里给一个打满的定义:

使用监控软件发现客户端的上行和服务端的下载带宽用满

 

2.测试软件

客户端:

// Client side C program to demonstrate Socket
// programming
#include <arpa/inet.h>
#include <stdio.h>
#include <string.h>
#include <sys/socket.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <sched.h>
#define PORT 8080

#define BUFF_SIZE (4*1024)
#define BUFF_SEND_SIZE (2*1024)


void loop_barrier(int loop_count)
{
        static count = 1;
        double sum = 0.0f;
        for(int i = 0 ; i < loop_count; i++)
                sum = sum * 1223491.3f * (double)(9401.f + i) * count++;
        
}


int main(int argc, char const* argv[])
{
    short port = atoi(argv[1]);
    int buff_send_size = atoi(argv[2]);
    int usleep_time = atoi(argv[3]);
    int loop_count = atoi(argv[4]);
    printf("%d %d %d %d\n", port, buff_send_size, usleep_time, loop_count);
    int status, valread, client_fd;
    struct sockaddr_in serv_addr;
    //char* hello = "Hello from client adsadfasdfasdfasdf0001234";
    char* hello = malloc(buff_send_size);
    for(int i = 0; i < buff_send_size; i++)
        hello[i] = '0' + i % 10;

    char buffer[BUFF_SIZE] = { 0 };
    if ((client_fd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
        printf("\n Socket creation error \n");
        return -1;
    }

    serv_addr.sin_family = AF_INET;
    //serv_addr.sin_port = htons(PORT);
    serv_addr.sin_port = htons(port);

    // Convert IPv4 and IPv6 addresses from text to binary
    // form
    if (inet_pton(AF_INET, "10.10.3.93", &serv_addr.sin_addr)
        <= 0) {
        printf(
            "\nInvalid address/ Address not supported \n");
        return -1;
    }

    if ((status
        = connect(client_fd, (struct sockaddr*)&serv_addr,
                sizeof(serv_addr)))
        < 0) {
        printf("\nConnection Failed \n");
        return -1;
    }

    while(1)
    {
        //usleep(usleep_time);
        //sched_yield();
        loop_barrier(loop_count);
        send(client_fd, hello, strlen(hello), 0);
        continue;
        //printf("[send] %s\n", hello);
        valread = read(client_fd, buffer, sizeof(buffer)); // subtract 1 for the null
        if (valread <= 0)
        {
            printf("recv %d %s", valread, errno);
            exit(-2);    
        }
        buffer[valread] = '\0';
        //printf("[recv] %s\n", buffer);
        //sleep(1);
    }

    // closing the connected socket
    close(client_fd);
    return 0;
}

测试命令:

nohup ./client 8080 2048 0 1000 &(连接服务端端口8080,一次发送buffer是2048字节,usleep 0,算术计算loop循环1000)
nohup ./client 8081 2048 0 1000 &
nohup ./client 8082 2048 0 1000 &

 

服务端:

// Server side C program to demonstrate Socket
// programming
#include <netinet/in.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <unistd.h>
#include <errno.h>
#include <sched.h> 

#define PORT 8080

#define BUFF_SIZE (4*1024)

void loop_barrier(int loop_count)
{
    static count = 1;
    double sum = 0.0f;
    for(int i = 0 ; i < loop_count; i++)
        sum = sum * 1223491.3f * (double)(9401.f + i) * count++;
        
}

int main(int argc, char const* argv[])
{
    short port = atoi(argv[1]);
    int usleep_time = atoi(argv[2]);
    int loop_count = atoi(argv[3]);
    printf("%d %d %d\n",port, usleep_time, loop_count);
    int server_fd, new_socket;
    ssize_t valread;
    struct sockaddr_in address;
    int opt = 1;
    socklen_t addrlen = sizeof(address);
    char buffer[BUFF_SIZE] = { 0 };
    char* hello = "Hello from server asdfasdfsadfasdfasdf009999999999";

    // Creating socket file descriptor
    if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
        perror("socket failed");
        exit(EXIT_FAILURE);
    }

    // Forcefully attaching socket to the port 8080
    if (setsockopt(server_fd, SOL_SOCKET,
                SO_REUSEADDR | SO_REUSEPORT, &opt,
                sizeof(opt))) {
        perror("setsockopt");
        exit(EXIT_FAILURE);
    }
    address.sin_family = AF_INET;
    address.sin_addr.s_addr = INADDR_ANY;
    //address.sin_port = htons(PORT);
    address.sin_port = htons(port);

    // Forcefully attaching socket to the port 8080
    if (bind(server_fd, (struct sockaddr*)&address,
            sizeof(address))
        < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }
    if (listen(server_fd, 3) < 0) {
        perror("listen");
        exit(EXIT_FAILURE);
    }
    if ((new_socket
        = accept(server_fd, (struct sockaddr*)&address,
                &addrlen))
        < 0) {
        perror("accept");
        exit(EXIT_FAILURE);
    }

    while(1)
    {
        valread = read(new_socket, buffer,
                sizeof(buffer)); // subtract 1 for the null
        //usleep(usleep_time);
        //sched_yield();  
        loop_barrier(loop_count);
        if (valread <= 0)
                {
                        printf("recv %d %s", valread, errno);
                        exit(-2);
                }
        continue;
        buffer[valread] = '\0';
    //    printf("[recv] %s\n", buffer);
        send(new_socket, hello, strlen(hello), 0);
    //    printf("[send] %s\n", hello);
    //    sleep(1);
    }
    // terminator at the end
    //printf("%s\n", buffer);
    //send(new_socket, hello, strlen(hello), 0);
    //printf("Hello message sent\n");

    // closing the connected socket
    close(new_socket);
    // closing the listening socket
    close(server_fd);
    return 0;
}

测试命令:

nohup ./server 8080 0 1000 &(监听端口8080,usleep 0,算术计算loop循环1000)
nohup ./server 8081 0 1000 &
nohup ./server 8082 0 1000 &

 

3.测试结果:

说明:因为是在一台宿主机进行测试,所以其实不走宿主机的网卡,走的是虚拟机的网卡,但是对网络敏感度更高,对比实验更有说服力:

usleep 100 结果:300Mb
usleep 10 结果:600Mb
usleep 1 结果:690Mb
empty loop 结果:20Gb
usleep 0 结果:708Mb (deactivate_task调用)
sched_yield 结果:10Gb
loop 10000 calculate multipie_hard 结果: 2.7Gb (用了static 避免走缓存)
loop 1000 calculate multipie_hard 结果: 8.2Gb (用了static 避免走缓存)
loop 100 calculate multipie_hard 结果: 15Gb (用了static 避免走缓存)

 

4.初步分析结论:

empty loop : 很容易就把网络带宽打满。

usleep 0  :网络带宽完全利用不上,猜测是deactivate_task等调用,引发更大的耗时。

sched_yield  :出让cpu,网络带宽也能打的很高。

loop 10000 calculate multipie_hard : 就算进行很耗时的cpu计算,网络带宽依然可以打的很高。

 

5.总结:

所以网卡带宽打满,不仅仅是和其他操作耗时有关,还跟操作系统的一些进行内核态切换的api有关(例如usleep)

标签:count,usleep,int,网卡,linux,打满,include,port,loop
From: https://www.cnblogs.com/dodng/p/18266974

相关文章

  • Linux Redis 服务设置开机自启动
    @目录前言一、准备工作二、操作步骤2.1修改redis.conf文件2.2创建启动脚本2.3设置redis脚本权限2.4设置开机启动2.5验证总结前言请各大网友尊重本人原创知识分享,谨记本人博客:南国以南i、提示:以下是本篇文章正文内容,下面案例可供参考一、准备工作注意:准备已安装好的Re......
  • Linux进程间的通信方式(一)System V 消息队列
    文章目录前言1.消息队列概念2.消息队列的应用场景3.消息队列接口分类3.1SystemV消息队列3.2POSIX消息队列4.消息队列相关操作函数4.1ftok函数(获取一个key值)4.2msgget函数(根据key值获取一个消息队列操作符)4.3msgctl函数(设置消息队列属性)4.4msgsnd函......
  • Linux和windows网络配置文件的修改
    Linux和windows网络配置文件的修改网络配置文件是计算机网络管理中至关重要的一部分。正确配置网络文件可以确保计算机与网络设备之间的通信顺畅,避免网络故障。本文将详细介绍网络配置文件的修改方法,包括常见命令、使用方法举例,以及一些优化建议。一、网络配置文件概述网......
  • Linux运行等级
    Linux运行级别Linuxsystem存在7个运行级别运行级别0:所有进程终止,机器将有序停止,关机时就处于这个运行级别运行级别1:单用户模式(root用户进行维护),系统中所有的服务也不会启动运行级别2:多用户模式(网络文件系统NFS服务没有启动)运行级别3:完全多用户模式(有NFS网络文件系统......
  • Linux---文件的权限
    在Linux中的权限采用的是三位十进制的数表示权限,如:0755,0644权限的使用是由四个部分组成,ABCD(为了方便说明,采用这四个字母代表四个部分)A~0:表示十进制B:表示用户的权限C:表示组用户的权限D:表示其他用户的权限—:0,表示不能读,不能写,不可以执行–x:1,表示不能读,不能写,可以执......
  • Linux---open和close函数
    open:这是对文件权限的说明。注意:返回上一个工作目录:cd-close函数:关闭文件注意:在对C语言代码进行了修改时,必须要都运行的文件重新编译,然后在重新运行。不然,输出的结果不会发生改变。今日标语“努力不一定成功,但不努力一定不会成功。”......
  • Linux系统用户组管理
    用户管理​和Windows一样在Linux中也存在许多用户,可以登陆Linux,和Windows不同的是,在Windows中同一时刻只可以存在一个用户登录系统,而在Linux中是允许多个用户同时登陆进入Linux中的,然后可以同时进行操作。​因为在Linux中一切皆文件,所以说,用户也是以文件的形......
  • 在 Rocky Linux 中,你可以使用 firewalld 来管理防火墙规则。
    在RockyLinux中,你可以使用`firewalld`来管理防火墙规则。以下是关闭所有端口并重新开放所需端口的步骤:###步骤1:关闭所有端口首先,你需要关闭所有端口。你可以使用`firewall-cmd`命令来实现这一点:```bashsudofirewall-cmd--permanent--add-rich-rule='rulefamily="i......
  • 41、linux-yum源管理-阿里云仓库配置
    ·yum的管理1、清理原有的yum配置·把本地或者官方的/etc/yum.repos.d/路径下的所有repo配置文件移走·确保/etc/yum.repos.d/这里没有其它文件2、下载配置阿里巴巴开源镜像站官网配置:https://developer.aliyun.com/mirror/·在这个位置/e......
  • 如何在Windows11下部署Linux子系统中安装GCC编译器
    GCC编译器安装:1:gcc出现命令找不到2.直接按照提示来安装。会发现链接找不到服务器原因是因为默认的服务器在国外,无法直接进行访问,需要切换成国内的服务器3.切换软件源——换成国内的服务器注意:软件源要与版本号一致!演示所用均为22.04版本号,可根据版本号找对应的软件......