Netlink通信
前言
- Netlink 是 Linux 内核与用户空间进程之间进行通信的机制之一,一种特殊的进程间通信(IPC) 。
- 它是一种全双工、异步的通信机制,允许内核与用户空间之间传递消息。
- Netlink 主要用于内核模块与用户空间程序之间进行通信,也被一些用户空间工具用于与内核进行交互,比如 ip 命令。
Netlink 的主要特点
- 异步通信:发送方和接收方是独立的,不需要等待对方的响应,因此是异步通信。
- 多播支持:可以通过多播组的方式实现一对多的通信。
- 全双工:可以同时进行收发消息,不像一些其他通信机制是单向的。
- 支持多种消息类型:Netlink 支持多种消息类型,不同类型的消息有不同的用途。
Netlink 的基本使用流程
- 创建 Netlink 套接字: 在内核或用户空间中创建 Netlink 套接字。在内核空间,可以使用netlink_kernel_create 函数;在用户空间,使用 socket 函数创建 Netlink 套接字。
- 绑定套接字到 Netlink 组: 对于用户空间的 Netlink 套接字,可以通过 bind 函数将套接字绑定到一个 Netlink 组。内核空间的 Netlink 套接字通常使用 NETLINK_KERNEL 类型。
- 发送消息: 应用层使用 sendmsg 函数发送消息给内核驱动。消息的格式通常是 struct nlmsghdr,你可以在消息中携带不同的有效负载。
- 内核驱动接收方可以注册一个回调函数,当消息到达时,内核会调用这个回调函数。在用户空间,可以使用 recvmsg 函数接收消息。
什么是netlink组
- 在 Netlink 中,组(Group)是一种机制,它允许多个进程注册到同一个 Netlink类型并接收相同类型的消息。使用组的概念,可以实现一对多的通信机制。
- 每个 Netlink 套接字可以加入一个或多个组,然后发送到该组的消息将被组中的所有套接字接收。这使得多个进程可以同时监听并处理相同类型的消息。
- 在 Netlink 中,组是通过一个 32 位的标志位来表示的,这个标志位被称为 Multicast Group ID。通过使用这个标志位,可以方便地控制消息的分发。
Netlink 组的一些常用的组 ID
- 0: 表示不加入任何组,也就是说消息只会发送给明确指定的套接字。
- 1: 通常用于指定默认的组,允许接收一些通用的消息。
- 2 到 32: 这些组的含义是由具体的协议或模块定义的。
- 33 到 1023: 保留给用户使用。
- 1024 到 2047: 保留给内核使用。
Netlink 组的使用
- 创建 Netlink 套接字时指定组: 在用户空间使用 socket 函数创建 Netlink 套接字时,可以通过 bind函数将套接字绑定到指定的组。
- 内核中使用 mc_list 字段: 在内核模块中使用 struct netlink_kernel_cfg 结构体的 mc_list 字段来指定加入的组。
static struct netlink_kernel_cfg my_netlink_cfg = {.input = my_netlink_receive,.mc_list = my_netlink_groups, // 这是一个包含组 ID 的数组
};
这里,my_netlink_groups 是一个包含组 ID 的数组,通过将套接字绑定到这些组,可以实现一对多的消息分发。
本文目标
- 内核驱动与应用层完成数据的发送与接收,单进程通信
代码正文
用户层代码:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <linux/netlink.h>#define NETLINK_USER 31struct sockaddr_nl src_addr, dest_addr;
struct nlmsghdr *nlh = NULL;
struct iovec iov;
struct msghdr msg;void handle_msg(struct nlmsghdr *nlh) {printf("Netlink Message Header:\n");printf(" Length: %u\n", nlh->nlmsg_len);printf(" Type: %u\n", nlh->nlmsg_type);printf(" Flags: 0x%X\n", nlh->nlmsg_flags);printf(" Sequence Number: %u\n", nlh->nlmsg_seq);printf(" PID: %u\n", nlh->nlmsg_pid);printf("Received message: %s\n", (char *)NLMSG_DATA(nlh));
}int main() {int sock_fd = socket(PF_NETLINK, SOCK_RAW, NETLINK_USER);if (sock_fd < 0) {perror("Error creating socket");return -1;}memset(&src_addr, 0, sizeof(src_addr));src_addr.nl_family = AF_NETLINK;src_addr.nl_pid = getpid(); // Process IDsrc_addr.nl_groups = 0; // Unicastbind(sock_fd, (struct sockaddr *)&src_addr, sizeof(src_addr));memset(&dest_addr, 0, sizeof(dest_addr));dest_addr.nl_family = AF_NETLINK;dest_addr.nl_pid = 0; // To kernel.如果消息是发送给另一个进程的单播消息,nl_pid是另外一个进程的pid值而nl_groups为零。dest_addr.nl_groups = 0; // Unicast// 分配消息缓冲区并设置消息头nlh = malloc(NLMSG_SPACE(strlen("Hello from user space!") + 1));strcpy(NLMSG_DATA(nlh), "Hello from user space!");nlh->nlmsg_len = NLMSG_SPACE(strlen("Hello from user space!") + 1);nlh->nlmsg_pid = getpid();nlh->nlmsg_flags = 0;iov.iov_base = (void *)nlh;iov.iov_len = nlh->nlmsg_len;msg.msg_name = (void *)&dest_addr;msg.msg_namelen = sizeof(dest_addr);msg.msg_iov = &iov;msg.msg_iovlen = 1;// Send a message to the kernelsendmsg(sock_fd, &msg, 0);recvmsg(sock_fd, &msg, 0);handle_msg(nlh);free(nlh);close(sock_fd);return 0;
}
驱动代码:
#include <linux/module.h>
#include <net/sock.h>
#include <linux/netlink.h>#define NETLINK_USER 31struct sock *nl_sk = NULL;
void send_to_user(const char *message, u32 pid) {struct sk_buff *skb;struct nlmsghdr *nlh;int len = strlen(message) + 1;skb = nlmsg_new(NLMSG_SPACE(len), GFP_KERNEL);if (!skb) {printk(KERN_ERR "Failed to allocate skb\n");return;}nlh = nlmsg_put(skb, 0, 0, NLMSG_DONE, len, 0);strncpy(nlmsg_data(nlh), message, len);netlink_unicast(nl_sk, skb, pid, MSG_DONTWAIT);
}void nl_recv_msg(struct sk_buff *skb) {struct nlmsghdr *nlh = nlmsg_hdr(skb);
/*
Netlink 消息头的数据结构struct nlmsghdr {__u32 nlmsg_len; // 消息长度(包括头部长度)__u16 nlmsg_type; // 消息类型__u16 nlmsg_flags; // 消息标志__u32 nlmsg_seq; // 序列号__u32 nlmsg_pid; // 发送方进程的进程ID
};
*/ printk("-------------驱动收到消息-------------\n");printk("Netlink Message Header:\n");printk(" Length: %u\n", nlh->nlmsg_len);printk(" Type: %u\n", nlh->nlmsg_type);printk(" Flags: 0x%X\n", nlh->nlmsg_flags);printk(" Sequence Number: %u\n", nlh->nlmsg_seq);printk(" PID: %u\n", nlh->nlmsg_pid);printk("Received message: %s\n", (char *)nlmsg_data(nlh));int pid = nlh->nlmsg_pid; // Sending process IDsend_to_user("Hello from kernel!", pid);
}static struct netlink_kernel_cfg cfg = {.input = nl_recv_msg,//回调函数
};int init_module() {nl_sk = netlink_kernel_create(&init_net, NETLINK_USER, &cfg);if (!nl_sk) {printk(KERN_ALERT "Error creating socket.\n");return -10;}printk(KERN_INFO "Netlink module loaded.\n");return 0;
}void cleanup_module() {netlink_kernel_release(nl_sk);printk(KERN_INFO "Netlink module unloaded.\n");
}MODULE_LICENSE("GPL");
编译与验证
驱动Makefile:
KERNEL_DIR = /lib/modules/$(shell uname -r)/build
#PWD = $(shell pwd)
obj-m += netlink_socket_dirver.oall:make -C $(KERNEL_DIR) M=$(PWD) modulesclean:make -C $(KERNEL_DIR) M=$(PWD) clean
驱动编译:
@ubuntu:$ make all
make -C /lib/modules/5.4.0-150-generic/build M=/netlink_socket modules
make[1]: Entering directory '/usr/src/linux-headers-5.4.0-150-generic'CC [M] netlink_socket/netlink_socket_dirver.o
/netlink_socket/netlink_socket_dirver.c: In function ‘nl_recv_msg’:
netlink_socket/netlink_socket_dirver.c:51:5: warning: ISO C90 forbids mixed declarations and code [-Wdeclaration-after-statement]int pid = nlh->nlmsg_pid; // Sending process ID^~~Building modules, stage 2.MODPOST 1 modulesCC [M] netlink_socket/netlink_socket_dirver.mod.oLD [M] netlink_socket/netlink_socket_dirver.ko
make[1]: Leaving directory '/usr/src/linux-headers-5.4.0-150-generic'
应用层编译:
@ubuntu:netlink_socket$ gcc netlink_socket_user.c -o netlink_socket_user.o
验证执行
驱动加载:
sudo insmod netlink_socket_dirver.ko
dmesg看log:
[890093.196626] -------------驱动收到消息-------------
[890093.196627] Netlink Message Header:
[890093.196627] Length: 40
[890093.196628] Type: 0
[890093.196628] Flags: 0x0
[890093.196628] Sequence Number: 0
[890093.196628] PID: 96003
[890093.196629] Received message: Hello from user space!
应用层打印:
@ubuntu:$ sudo ./netlink_socket_user.o
Netlink Message Header:Length: 35Type: 3Flags: 0x0Sequence Number: 0PID: 0
Received message: Hello from kernel!
结果
成功验证
PS:
希望能抛砖引玉
相关文章:
Netlink通信
前言 Netlink 是 Linux 内核与用户空间进程之间进行通信的机制之一,一种特殊的进程间通信(IPC) 。它是一种全双工、异步的通信机制,允许内核与用户空间之间传递消息。Netlink 主要用于内核模块与用户空间程序之间进行通信,也被一些用户空间工具用于与内…...
Python打造简单而强大的聊天机器人:详解与实例代码
更多资料获取 📚 个人网站:ipengtao.com 聊天机器人在现代应用中扮演着重要的角色,能够与用户进行自然语言交互。本篇博客将带领读者通过Python,使用自然语言处理库构建一个简单而强大的聊天机器人。我们将深入讨论处理用户输入、…...
Unity 通过代码将一张大图切成多个小图的方法
在Unity 中要通过代码将一张贴图切割成多张小图,可以使用以下方法: /// <summary>/// 把一张图片切割成多张使用/// </summary>/// <param name"texture">原图</param>/// <param name"rows">切割的行…...
Spring Cloud Gateway请求路径修改指南:详解ServerWebExchange的完美解决方案及代码示例
🌷🍁 博主猫头虎 带您 Go to New World.✨🍁 🦄 博客首页——猫头虎的博客🎐 🐳《面试题大全专栏》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺 &a…...
AI+爬虫 爬虫宝
场景 在很多时候,有很多爬虫的需求,其实需求都是大同小异,不过是在某某网页上爬取某某东西。把这些东西给到业务。以往的做法是:每一个需求写一个c#或者python脚本或者应用,进行爬取,这相当的费时费力&…...
【matlab】绘制横状双组渐变柱状图
【matlab】绘制横状双组渐变柱状图...
如何下载知网论文、专利的PDF格式
知网的论文格式有其特有的格式:CAJ。将CAJ格式转化为Word或者PDF非常麻烦,且会出现乱码的情况,直接用知网官方的CAJ浏览器也不太方便。为此,困扰了许久。 其实,知网可以直接下载PDF格式,只需在浏览器上安装…...
SMARTFORMS打印问题
一、问题概述 项目中碰到EPSON针式打印机,SAP打印出来的三单合一,是想横向打印,结果不管怎么设置打印出来都是纵向的; 二、调整 1、首先调整SMARTFORMS的打印方向,使用T-CODE:SMARTFORMS,如下图…...
uniapp组件和周期的知识点以及怎么使用
UniApp是一个使用Vue.js开发跨平台应用的框架,它可以将代码编译成微信小程序、支付宝小程序、H5、App等多个平台的应用。在UniApp中,组件是构建应用界面的基本单位,而生命周期则是组件在实例化、挂载、更新和销毁过程中的一系列方法。 以下是…...
【Java JVM】实例对象的创建
当我们涉及 Java 编程时, 对象的创建是一个基础而关键的概念。在 Java 中, 一切皆为对象, 而对象的创建方式直接影响代码的结构和性能。 本博客将探讨一下 Java 实例对象的创建过程。 1 创建对象的方法有哪些 在 Java 中如果要创建一个对象, 有哪些方式呢? 运用 new 关键字创…...
sectigo多域名通配符证书
多域名通配符SSL证书是一种特殊的SSL证书,它综合了多域名SSL证书和通配符SSL证书的特点,可以同时保护多个泛域名以及它的主域名,适合域名多,子域名也比较多的单个或多个站点使用。今天就随SSL盾小编了解Sectigo旗下的DV多域名通配…...
vue 流光边框矩形圆形容器
实现流光边框一般是用渐变背景加动画实现,然后使用内部盒子遮挡内部空间,达到边框流光的效果 思路:背景渐变旋转动画 功能: 自定义渐变(是否渐变<不渐变没有流光效果>,渐变颜色,渐变角…...
明理信息科技打造专属个人或企业知识付费平台,核心功能设计
在当今信息爆炸的时代,知识管理已经成为了每个人必须面对的问题。然而,市面上的知识付费平台大多数都是通用的,无法满足个性化需求。 因此,明理信息科技提供了一款专属定制的适合个人的知识付费平台。核心产品能力如下࿱…...
超结MOS/低压MOS在5G基站电源上的应用-REASUNOS瑞森半导体
一、前言 5G基站是5G网络的核心设备,实现有线通信网络与无线终端之间的无线信号传输,5G基站主要分为宏基站和小基站。5G基站由于通信设备功耗大,采用由电源插座、交直流配电、防雷器、整流模块和监控模块组成的电气柜。所以顾名思义…...
字节8年经验之谈 —— 如何从0开始做自动化测试?
自动化测试是使用软件工具在应用程序上自动运行测试的过程,无需任何人为干预。这可以通过减少手动测试的需要来保存时间并提高软件开发过程的效率。由于人为错误或不一致性,手动测试可能容易出错,这可能导致错误未被检测到。自动化测试通过提…...
深入探索Zookeeper的ZAB协议:分布式系统的核心解析
引言 自我进入软件开发领域以来,我一直对分布式系统充满着浓厚的兴趣。在这个领域中,Zookeeper无疑是一个备受关注的重要组件。作为一名资深的Java工程师,我有幸深入探索过Zookeeper的许多方面,其中最让我着迷的部分莫过于其核心机…...
安捷伦Agilent 34970A数据采集
易学易用 从34972A简化的配置到内置的图形Web界面,我们都投入了非常多的时间和精力,以帮助您节约宝贵的时间。一些非常简单的东西,例如模块上螺旋型端子连接器内置热电偶参考结、包括众多实例和提示的完整用户文档,以及使您能够在开机数分钟后…...
Apache Flume(5):多个agent模型
可以将多个Flume agent 程序连接在一起,其中一个agent的sink将数据发送到另一个agent的source。Avro文件格式是使用Flume通过网络发送数据的标准方法。 从多个Web服务器收集日志,发送到一个或多个集中处理的agent,之后再发往日志存储中心&…...
如何在Ubuntu系统中安装VNC并结合内网穿透实现远程访问桌面
文章目录 前言1. ubuntu安装VNC2. 设置vnc开机启动3. windows 安装VNC viewer连接工具4. 内网穿透4.1 安装cpolar【支持使用一键脚本命令安装】4.2 创建隧道映射4.3 测试公网远程访问 5. 配置固定TCP地址5.1 保留一个固定的公网TCP端口地址5.2 配置固定公网TCP端口地址5.3 测试…...
JaCoCo 统计度量
1、JaCoCo: 一个判断算2个Branch,最后一个括号算一行 2、IDEA:一个判断算一个Branch,最后一个括号不算一行...
我们经常使用的AI技术
窄人工智能 Narrow AI 窄人工智能主要集中于处理相对单一的任务,可以理解,科学家在研究如何模拟人类智能时,一种思路就是我们是不是可以先分别模拟人的不同能力,例如我们的视觉能力,文字识别能力等等。所以就演化出针…...
静态路由及动态路由
文章目录 静态路由及动态路由一、静态路由基础1. 静态路由配置2. 负载分担3. 路由备份4. 缺省路由5. 静态路由实操 二、RIP 动态路由协议1. RIP 协议概述2. RIP 协议版本对比2.1 有类路由及无类路由 3. RIP 路由协议原理4. RIP 计时器5. 度量值6. 收敛7. 示例 静态路由及动态路…...
MySQL——表的增删查改
目录 一.Create(创建) 1.单行数据 全列插入 2.多行数据 指定列插入 3.插入否则更新 4. 替换 二.Retrieve(读取) 1. select 列 查询 2.where 条件 3.结果排序 4.筛选分页结果 三.Update (修改)…...
javascript_1
3) string ⭐️ js 字符串三种写法 let a "hello"; // 双引号 let b "world"; // 单引号 let c hello; // 反引号 html 代码如下,用 java 和 js 中的字符串如何表示? <a href"1.html">超链接</a> …...
【ranger】CDP环境 更新 ranger 权限策略会发生低概率丢失权限策略的解决方法
一、问题描述: 我们的 kafka 服务在更新(添加) ranger 权限时,会有极低的概率导致 MM2 同步服务报错,报错内容 Not Authorized。但是查看 ranger 权限是赋予的,并且很早配置的权限策略也会报错。 相关组件…...
Python安装及配置
一、前置说明 Python的安装有两种方式:1. 访问Python官方网站下载安装;2. 使用Python的开源发行版进行安装。 Anaconda 是一个用于科学计算、数据分析和机器学习的开源发行版,它包含了许多常用的科学计算和数据分析库。Anaconda 不仅仅是 P…...
Instagram 外贸产品推广技巧
在Instagram上,外贸业务有许多独特的机会来展示其产品并吸引国际买家。成功的外贸产品推广要求细致的策略、引人入胜的创意内容和有针对性的市场洞察。下面的小节将详细解析如何在Instagram上进行外贸产品的有效推广。 1.创意与视觉呈现 Instagram是一个基于图片和…...
5款实用的小工具,让你的日常生活多姿多彩
简单而小巧的工具,经常能在日常中悄然发挥极大的作用。这五款小工具可能成为你生活中不可或缺的一部分。 1.网络浏览器——Brave Brave是一款基于Chromium内核的开源网络浏览器,它可以阻止网站的广告和跟踪程序,保护您的隐私和安全。…...
【改进YOLOv8】磁瓦缺陷分类系统:改进LSKNet骨干网络的YOLOv8
1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 研究背景与意义 近年来,随着智能制造产业的不断发展,基于人工智能与机器视觉的自动化产品缺陷检测技术在各行各业中得到了广泛应用。磁瓦作为永磁电机的主…...
Linux-VRRP
这里写自定义目录标题 一、VRRP简介1.1 什么是VRRP?1.2 keepalived是什么? 二、配置过程2.1 试验模型2.2. Keepalived监控和维护VRRP集群的步骤 一、VRRP简介 1.1 什么是VRRP? VRRP(Virtual Router Redundancy Protocolÿ…...
免费视频模板网站/百度下载老版本
设置服务器 获取ip ip addr过程设置服务器开放端口配置文件: /etc/sysconfig/iptables.save //修改后重启 service iptables restart 查看监听的端口: netstat -tlncentos7使用firewalld打包 //将当前路径文件和文件夹打包到xx.gztar czf xx.gz app.js public/ //解压文件到x…...
西安制作公司网站的公司/深圳seo排名优化
最近一直在找前端工作,前前后后面了几家公司,虽然不太顺利,但是收获还是有的,就过程中遇到的面试题总结一下 CSS1:标准盒模型/IE盒模型2:自适应三栏布局3:移动端自适应方案4:FLEX布局…...
免费空间做网站/免费python在线网站
本文适用于将MyEclipse上的项目projectA检出后重命名为projectB的情况,如果只是检出projectA到Eclipse,也可以部分参考 1、从svn上检出Myeclipse项目到Eclipse 检出后,首先断开svn连接。 2、更改项目名称 从svn检出的项目projectA,…...
做网站要学那些东西/ios aso优化工具
开头 这里是一些个人开发者接私活和自己做软件加广告的一些科普知识。可是做软件,需要服务器,需要后台,对于一些小的开发者,想赚点广告费而又不想做后台使用服务器的人来说,网上提供了一些免费的接口,可以…...
如何管理好一个网站/网站怎么做到秒收录
查看GPU使用情况: nvidia-smi 发现进程22140占用GPU0,运行下面命令删除进程22140 kill -9 22140...
做图标去什么网站找/站长工具端口检测
1444 “破锣摇滚”乐队 时间限制: 1 s空间限制: 256000 KB题目等级 : 白银 Silver题解题目描述 Description你刚刚继承了流行的“破锣摇滚”乐队录制的尚未发表的N(1 < N < 20)首歌的版权。你打算从中精选一些歌曲,发行M(1 < M < 20)张CD。每一张CD最多…...