【C语言】RDMACM、Verbs API与epoll一起使用的示例

news2024/12/24 3:42:03

一、epoll介绍

epoll是Linux内核为处理大批量文件描述符而作了改进的poll,是Linux下多路复用IO接口select/poll的增强版本,它能显著提高程序在大量并发连接中只有少量活跃的情况下的系统CPU利用率。

以下是epoll的主要使用方法和优点:

  1. epoll的创建和使用主要涉及到三个函数:epoll_create、epoll_ctl和epoll_wait。首先,使用epoll_create创建一个新的epoll实例,并返回一个引用该实例的文件描述符。然后,通过epoll_ctl注册对感兴趣的文件描述符。最后,使用epoll_wait等待I/O事件。
  2. epoll的优点主要体现在:支持一个进程打开大数目的socket描述符(FD);IO效率不随FD数目增加而线性下降;支持边缘触发模式;使用mmap加速内核与用户空间的消息传递。
  3. 水平触发和边沿触发是epoll的两种事件分发机制。二者的区别在于,水平触发模式下,只要文件描述符处于就绪状态,无论应用程序是否读取或者写入数据,每次调用epoll_wait都会返回该文件描述符;而在边缘触发模式下,只有当文件描述符状态发生变化时(比如从非就绪变为就绪),epoll_wait才会返回该文件描述符。

总的来说,epoll是Linux内核中处理大批量文件描述符的高效工具,特别适用于大量并发连接中只有少量活跃的情况,能显著提高系统CPU利用率。

二、示例

以下是一个使用C语言的示例,展示了如何在使用RDMACM和Verbs API时与epoll一起使用:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/epoll.h>
#include <rdma/rdma_cma.h>

#define MAX_EVENTS 10

struct connection {
    struct rdma_cm_id *id;
    struct ibv_qp *qp;
    // 其他连接相关的数据
};

void handle_cm_event(struct rdma_cm_event *event) {
    // 处理RDMACM事件的逻辑
}

void handle_cq_event(struct ibv_wc *wc) {
    // 处理完成队列事件的逻辑
}

int main() {
    struct rdma_event_channel *cm_channel;
    struct rdma_cm_event *event;
    struct epoll_event epoll_events[MAX_EVENTS];
    struct connection *conn;
    struct ibv_cq *cq;
    struct ibv_wc wc;
    int cm_fd, cq_fd, epoll_fd, i, n;

    // 创建RDMACM事件通知文件描述符
    cm_channel = rdma_create_event_channel();
    if (!cm_channel) {
        perror("Failed to create RDMACM event channel");
        return 1;
    }

    // 创建完成队列(CQ)
    cq = ibv_create_cq(NULL, 10, NULL, NULL, 0);
    if (!cq) {
        perror("Failed to create completion queue");
        return 1;
    }

    // 获取RDMACM事件通知文件描述符和完成队列的文件描述符
    cm_fd = rdma_event_channel_fd(cm_channel);
    cq_fd = cq->channel->fd;

    // 创建epoll实例
    epoll_fd = epoll_create1(0);
    if (epoll_fd == -1) {
        perror("Failed to create epoll instance");
        return 1;
    }

    // 将RDMACM事件通知文件描述符添加到epoll的事件集合中
    struct epoll_event cm_event;
    cm_event.events = EPOLLIN | EPOLLET;
    cm_event.data.fd = cm_fd;
    if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, cm_fd, &cm_event) == -1) {
        perror("Failed to add RDMACM event channel to epoll");
        return 1;
    }

    // 将完成队列的文件描述符添加到epoll的事件集合中
    struct epoll_event cq_event;
    cq_event.events = EPOLLIN | EPOLLET;
    cq_event.data.fd = cq_fd;
    if (epoll_ctl(epoll_fd, EPOLL_CTL_ADD, cq_fd, &cq_event) == -1) {
        perror("Failed to add completion queue to epoll");
        return 1;
    }

    // 进入事件循环
    while (1) {
        // 等待事件发生
        n = epoll_wait(epoll_fd, epoll_events, MAX_EVENTS, -1);
        if (n == -1) {
            perror("Failed to wait for events");
            return 1;
        }

        // 处理所有事件
        for (i = 0; i < n; i++) {
            if (epoll_events[i].data.fd == cm_fd) {
                // 有RDMACM事件发生
                event = rdma_get_cm_event(cm_channel);
                handle_cm_event(event);
                rdma_ack_cm_event(event);
            } else if (epoll_events[i].data.fd == cq_fd) {
                // 有完成队列事件发生
                while (ibv_poll_cq(cq, 1, &wc) > 0) {
                    handle_cq_event(&wc);
                }
            }
        }
    }

    // 清理资源
    close(epoll_fd);
    ibv_destroy_cq(cq);
    rdma_destroy_event_channel(cm_channel);

    return 0;

}

在这个示例中,我们使用了rdma/rdma_cma.h头文件中提供的RDMACM和Verbs API。首先,我们创建了一个RDMACM事件通知文件描述符和一个完成队列(CQ),然后将它们的文件描述符添加到epoll的事件集合中。接下来,我们进入一个事件循环,等待事件发生。如果有RDMACM事件发生,我们调用handle_cm_event函数来处理该事件;如果有完成队列事件发生,我们调用handle_cq_event函数来处理该事件。
请注意,这只是一个简单的示例,实际使用时可能需要根据具体需求进行更多的处理和错误检查。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1317549.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

代码随想录刷题题Day15

刷题的第十五天&#xff0c;希望自己能够不断坚持下去&#xff0c;迎来蜕变。&#x1f600;&#x1f600;&#x1f600; 刷题语言&#xff1a;C Day15 任务 ● 513.找树左下角的值 ● 112. 路径总和 113.路径总和ii ● 106.从中序与后序遍历序列构造二叉树 105.从前序与中序遍历…

整理了上百个开源中文大语言模型,涵盖模型、应用、数据集、微调、部署、评测

自ChatGPT为代表的大语言模型&#xff08;Large Language Model, LLM&#xff09;出现以后&#xff0c;由于其惊人的类通用人工智能&#xff08;AGI&#xff09;的能力&#xff0c;掀起了新一轮自然语言处理领域的研究和应用的浪潮。 尤其是以ChatGLM、LLaMA等平民玩家都能跑起…

广州华锐互动:汽车电子线束加工VR仿真培训与实际生产场景相结合,提高培训效果

随着科技的不断发展&#xff0c;虚拟现实&#xff08;VR&#xff09;技术已经逐渐渗透到各个领域&#xff0c;为企业和个人带来了前所未有的便利。在汽车制造行业中&#xff0c;线束加工作为一项关键的生产工艺&#xff0c;其质量直接影响到汽车的性能和安全。因此&#xff0c;…

LCR 181. 字符串中的单词反转

解题思路&#xff1a; class Solution {public String reverseMessage(String message) {message message.trim(); // 删除首尾空格int j message.length() - 1, i j;StringBuilder res new StringBuilder();while (i > 0) {while (i >…

局域网环境下的ntp对时

服务端&#xff1a; 此处为v4-sp4服务器 安装ntp&#xff0c;apt-get install ntp -y ,若为离线环境&#xff0c;则安装ntp和libopts25两个包。 配置&#xff1a; 在/etc/ntp.conf的配置文件里 加入 restrict default nomodify notrap noquery restrict 127.0.0.1 rest…

时序预测 | Python实现GRU-XGBoost组合模型电力需求预测

时序预测 | Python实现GRU-XGBoost组合模型电力需求预测 目录 时序预测 | Python实现GRU-XGBoost组合模型电力需求预测预测效果基本描述程序设计参考资料预测效果 基本描述 该数据集因其每小时的用电量数据以及 TSO 对消耗和定价的相应预测而值得注意,从而可以将预期预测与当前…

【IDEA】关于快捷键行注释时会从行首开始注释

前言 每次在使用IDEA写代码时&#xff0c;用快捷键进行行注释的时候&#xff0c;//会处于行首位置 但是我们想要的是这种注释效果 修改 IDEA中打开settings 找到Editor - Code Style- Java - 找到Code Genneration&#xff0c;取消勾选Line comment at first column&#x…

如何批量获取CSDN文章数据并进行持久化

自己去看文章数据的话&#xff0c;比较慢&#xff0c;所以一直想通过程序来批量获取CSDN的文章数据&#xff0c;最近研究了一下&#xff0c;发现还是挺简单的&#xff0c;能够直接通过解析json来获取文章数据&#xff0c;跟大家分享一下。 文章目录 一、步骤1、首先我们到自己的…

VUE学习三、前端项目部署

1.前端项目打包 执行命令 npm run build:prod正常命令结束 , 会在前端项目里面出现dist文件夹 2.nginx下载安装 nginx下载 : http://nginx.org/en/download.html Windows 下载版本 Mainline version&#xff1a;Mainline 是 Nginx 目前主力在做的版本&#xff0c;可以说…

【手撕算法系列】BN

BN的计算公式 BN中均值与方差的计算 所以对于输入x: b,c,h,w 则 mean: 1,c,1,1var: 1,c,1,1代码 class BatchNorm(nn.Module):def __init__(self, num_features, num_dims):# num_features&#xff1a;完全连接层的输出数量或卷积层的输出通道数。# num_dims&#xff1a;2表示…

【JavaEE】多线程(4) -- 单例模式

目录 什么是设计模式? 1.饿汉模式 2.懒汉模式 线程安全问题 什么是设计模式? 设计模式好⽐象棋中的 "棋谱". 红⽅当头炮, ⿊⽅⻢来跳. 针对红⽅的⼀些⾛法, ⿊⽅应招的时候有⼀ 些固定的套路. 按照套路来⾛局势就不会吃亏. 软件开发中也有很多常⻅的 "问题…

接口优先于反射机制

在Java中&#xff0c;使用接口通常比反射机制更为优雅和安全。接口提供了一种声明性的方式来定义类的契约&#xff0c;并且能够在编译时进行类型检查&#xff0c;而反射则是在运行时动态获取和操作类的信息。下面是一个简单的例子&#xff0c;说明为什么在某些情况下接口比反射…

基于Java SSM框架实现体育竞赛运动会成绩裁判管系统项目【项目源码+论文说明】计算机毕业设计

基于java的SSM框架实现体育竞赛运动会成绩裁判管系统演示 摘要 体育竞赛是各种体育体育项目比赛的总称。是在裁判员的主持下&#xff0c;按统一的规则要求&#xff0c;组织与实施的体育员个体或体育队之间的竞技较量&#xff0c;是竞技体育与社会发生关联&#xff0c;并作用于…

鼠标点击展开详情

网页部分&#xff1a; HTML部分&#xff1a; <body><br><div id"nav"><div id"dang"></div><ul id"aaa"><li class"a1"><img src"img/U011.jpg" width"195" heig…

7+PPI+机器学习+实验,非肿瘤结合建模筛选生物标志物,可升级

今天给同学们分享一篇生信文章“Identification of diagnostic biomarkers and therapeutic targets in peripheral immune landscape from coronary artery disease”&#xff0c;这篇文章发表在J Transl Med期刊上&#xff0c;影响因子为7.4。 结果解读&#xff1a; 外周血中…

为什么要用 top_p 做文本生成采样

上一篇文章介绍了在大模型生成文本时&#xff0c;一个可以控制的参数 Temperature 的作用&#xff1a;查看温度参数与 softmax&#xff0c;今天继续来看另一个同样可以控制文本输出效果的参数。 或许你在使用 OpenAI 的接口时&#xff0c;会碰到这个参数&#xff0c;那就是 to…

尚硅谷Docker笔记-基础篇

B站视频&#xff1a;https://www.bilibili.com/video/BV1gr4y1U7CY 1.Docker简介 解决了运行环境和配置问题的软件容器 方便做持续集成并有助于整体发布的容器虚拟化技术 容器与虚拟机比较 Docker 容器是在操作系统层面上实现虚拟化&#xff0c;直接复用本地主机的操作系统…

线程终止,线程的相关方法,线程状态图以及线程的同步机制。

首先我们知道了什么是线程&#xff0c;以及线程的实现方法&#xff0c;接下来我们来了解一下继承Thread类和Runnable接口的区别&#xff0c;其实本质上是没有区别的&#xff0c;因为Thread也实现了Runnable的接口&#xff0c;唯一区别就是使用Runnable接口&#xff0c;可以实现…

准确无误的二手车估值API接口,帮您把握市场行情!

随着二手车市场的快速发展&#xff0c;越来越多的人开始关注二手车的估值问题。在购买二手车或者出售二手车时&#xff0c;了解车辆的真实价值是非常重要的。而准确无误的二手车估值API接口能够帮助我们更好地把握市场行情。 在挖数据平台上&#xff0c;我们提供了一个功能强大…

IntersectionObserver API的简单介绍及应用

前言 IntersectionObserver 之前就有所了解&#xff0c;但是一直没有用&#xff0c;当时觉得浏览器兼容性不好&#xff0c;直接用onscroll监听也能实现相关功能&#xff0c;但是随着浏览器支持性越来越好&#xff0c;这个属性到迄今为止&#xff0c;可以用在大部分的正式项目中…