Linux基础 -- 用户态Generic Netlink库高性能接收与回调框架

用户态Generic Netlink库高性能接收与回调框架

一、概述

在 Linux 系统中,Netlink 是用户态与内核态通信的强大机制。libnl 是一个专为简化 Netlink 编程而设计的库,提供了接收和处理 Netlink 消息的高级接口。libnl-genl 是其通用 Netlink (Generic Netlink) 扩展库,专门用于处理自定义协议。本文将详细介绍如何通过 nl_socket_modify_cbnl_cb_set 两种方法实现高效的 Netlink 消息接收和处理框架。

Netlink 使用回调机制来处理接收到的消息。常用的回调设置方法有两种:

1. nl_socket_modify_cb

  • 仅在指定的 Netlink 套接字上设置回调。
  • 适合在每个套接字上使用不同回调。

2. nl_cb_set

  • 在回调上下文 (struct nl_cb *) 上设置回调。
  • 可以在多个套接字之间共享。

1. epoll + 非阻塞 + 批量接收

  • 使用 epoll 进行非阻塞监听。
  • 通过大缓冲区批量接收消息。
  • 结合 libnllibnl-genl 实现高效的 Netlink 消息接收。
c 复制代码
#include <netlink/netlink.h>
#include <netlink/msg.h>
#include <netlink/genl/genl.h>
#include <netlink/genl/ctrl.h>
#include <sys/epoll.h>
#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>

#define MY_GENL_FAMILY "my_custom_family"

// 自定义消息处理回调
int my_genl_handler(struct nl_msg *msg, void *arg) {
	struct nlmsghdr *nlh = nlmsg_hdr(msg);
	struct genlmsghdr *gnlh = nlmsg_data(nlh);
    
	printf("[Genl Handler] Received a Netlink message.\n");
	printf("Command: %d, Version: %d\n", gnlh->cmd, gnlh->version);
	
	struct nlattr *attrs[GENL_MAX_ATTRS + 1] = {};
	nla_parse(attrs, GENL_MAX_ATTRS, genlmsg_attrdata(gnlh, 0), genlmsg_attrlen(gnlh, 0), NULL);
	
	if (attrs[GENL_ATTR_MSG]) {
		printf("Message: %s\n", nla_get_string(attrs[GENL_ATTR_MSG]));
	}
	
    return NL_OK;
}

// 设置非阻塞模式
void set_nonblocking(int fd) {
    int flags = fcntl(fd, F_GETFL, 0);
    fcntl(fd, F_SETFL, flags | O_NONBLOCK);
}

int main() {
    struct nl_sock *sock = nl_socket_alloc();
    nl_connect(sock, NETLINK_GENERIC);

    // 查找 Generic Netlink family
    int family_id = genl_ctrl_resolve(sock, MY_GENL_FAMILY);
    if (family_id < 0) {
        printf("[Error] Could not resolve Generic Netlink family\n");
        return -1;
    }

    struct nl_cb *cb = nl_cb_alloc(NL_CB_DEFAULT);
    nl_cb_set(cb, NL_CB_VALID, NL_CB_CUSTOM, my_genl_handler, NULL);
    nl_socket_set_cb(sock, cb);

    int fd = nl_socket_get_fd(sock);
    set_nonblocking(fd);

    int epfd = epoll_create1(0);
    struct epoll_event ev, events[10];
    ev.events = EPOLLIN;
    ev.data.fd = fd;
    epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &ev);

    while (1) {
        int nfds = epoll_wait(epfd, events, 10, -1);
        for (int i = 0; i < nfds; i++) {
            if (events[i].data.fd == fd) {
                nl_recvmsgs(sock, cb); // 使用 libnl + epoll 处理消息
            }
        }
    }

    nl_cb_put(cb);
    nl_close(sock);
    nl_socket_free(sock);
    close(epfd);
    return 0;
}

3. 多线程处理(可选)

  • 通过线程池并行处理接收到的 Netlink 消息。
  • 每个线程从 epoll 事件队列中读取并解析 Netlink 消息。

四、最佳实践

  • 通过 nl_socket_modify_cb 灵活调整特定套接字的回调。
  • 通过 nl_cb_set 设置全局回调,并在多套接字间复用。
  • 对于高频 Netlink 通信,采用非阻塞模式 + epoll + 批量处理。
  • 使用 libnl-genl 处理 Generic Netlink,简化自定义协议实现。

五、总结

本框架提供了一种高效的 Netlink 消息接收和处理方式,适用于高并发和高性能场景,并支持自定义协议的灵活扩展。

相关推荐
wingaso19 分钟前
[经验总结]删除gitlab仓库分支报错:错误:无法推送一些引用到“http:”
linux·数据仓库·git
独行soc24 分钟前
2025年渗透测试面试题总结-阿里云[实习]阿里云安全-安全工程师(题目+回答)
linux·经验分享·安全·阿里云·面试·职场和发展·云计算
勤不了一点35 分钟前
小白上手RPM包制作
linux·运维·服务器·软件工程
麦a~M了M2 小时前
ansible
linux·运维·ansible
QQ_4376643143 小时前
Linux下可执行程序的生成和运行详解(编译链接汇编图解)
linux·运维·c语言·汇编·caffe
窦再兴4 小时前
来一个复古的技术FTP
linux·运维·服务器
xiaobin889994 小时前
【2025最新版】VMware虚拟机下载安装教程 保姆级图文详解(附安装包+常用镜像Linux,win11,ubuntu,centos)
linux·其他·ubuntu·centos
ALex_zry5 小时前
Ubuntu 20.04 C++开发环境搭建指南(2025版)
linux·c++·ubuntu
疯狂的挖掘机5 小时前
记一次从windows连接远程Linux系统来控制设备采集数据方法
linux·运维·windows