udev实现热插拔

一、UDEV是什么?

Udev是一个针对Linux内核2.6的可提供自动创建的设备节点和命名的解决方法的一个文件系统;其实与/etc/目录下的fstab文件类似

二、Udev如何获取内核这些模块的变化信息?

参考博客:http://blog.chinaunix.net/uid-24943863-id-3223000.html

设备节点的创建,是通过sysfs接口分析dev文件取得设备节点号,这个很显而易见。那么udevd是通过什么机制来得知内核里模块的变化情况,如何得知设备的插入移除情况呢?当然是通过hotplug机制了,那hotplug又是怎么实现的?或者说内核是如何通知用户空间一个事件的发生的呢?

答案是通过netlink socket通讯,在内核和用户空间之间传递信息。

新的Linux内核使用udev代替了hotplug作为热拔插管理,虽然有udevd管理热拔插,但有时候我们还是需要在应用程序中检测热拔插事件以便快速地处理,比如在读写SD卡的时候拔下SD卡,那么需要立即检测出该情况,然后结束读写线程,防止VFS崩溃。Netlink是面向数据包的服务,为内核与用户层搭建了一个高速通道,是udev实现的基础。该工作方式是异步的,用户空间程序不必使用轮询等技术来检测热拔插事件

内核中使用uevent事件通知用户空间,uevent首先在内核中调用netlink_kernel_create()函数创建一个socket套接字,该函数原型在netlink.h有定义,其类型是表示往用户空间发送消息的NETLINK_KOBJECT_UEVENT,groups=1,由于uevent只往用户空间发送消息而不接受,因此其输入回调函数input和cb_mutex都设置为NULL。

struct sock *netlink_kernel_create(struct net *net,int unit,unsigned int groups,

                                                  void (*input)(struct sk_buff *skb),

                                                  struct mutex *cb_mutex,

                                                  struct module *module);

ue_sk->sk = netlink_kernel_create(net, NETLINK_KOBJECT_UEVENT, 1, NULL, NULL, THIS_MODULE);

当有事件发生的时候,调用 kobject_uevent()函数,实际上最终是调用 netlink_broadcast_filtered(uevent_sock, skb , 0, 1, GFP_KERNEL , kobj_bcast_filter, kobj);

完成广播任务。

  用户空间程序只需要创建一个socket描述符,将描述符绑定到接收地址,就可以实现热拔插事件的监听了。

 1 #include <stdio.h>
 2 #include <stdlib.h>
 3 #include <string.h>
 4 #include <errno.h>
 5 #include <sys/types.h>
 6 #include <asm/types.h>
 7 //该头文件需要放在netlink.h前面防止编译出现__kernel_sa_family未定义
 8 #include <sys/socket.h>  
 9 #include <linux/netlink.h>
10 
11 void MonitorNetlinkUevent()
12 {
13     int sockfd;
14     struct sockaddr_nl sa;
15     int len;
16     char buf[4096];
17     struct iovec iov;
18     struct msghdr msg;
19     int i;
20 
21     memset(&sa,0,sizeof(sa));
22     sa.nl_family=AF_NETLINK;
23     sa.nl_groups=NETLINK_KOBJECT_UEVENT;
24     sa.nl_pid = 0;//getpid(); both is ok
25     memset(&msg,0,sizeof(msg));
26     iov.iov_base=(void *)buf;
27     iov.iov_len=sizeof(buf);
28     msg.msg_name=(void *)&sa;
29     msg.msg_namelen=sizeof(sa);
30     msg.msg_iov=&iov;
31     msg.msg_iovlen=1;
32 
33     sockfd=socket(AF_NETLINK,SOCK_RAW,NETLINK_KOBJECT_UEVENT);
34     if(sockfd==-1)
35         printf("socket creating failed:%s\n",strerror(errno));
36     if(bind(sockfd,(struct sockaddr *)&sa,sizeof(sa))==-1)
37         printf("bind error:%s\n",strerror(errno));
38 
39     len=recvmsg(sockfd,&msg,0);
40     if(len<0)
41         printf("receive error\n");
42     else if(len<32||len>sizeof(buf))
43         printf("invalid message");
44     for(i=0;i<len;i++)
45         if(*(buf+i)=='\0')
46             buf[i]='\n';
47     printf("received %d bytes\n%s\n",len,buf);
48 }
49 
50 int main(int argc,char **argv)
51 {
52     MonitorNetlinkUevent();
53     return 0;
54 }

创建socket描述符的时候指定协议族为AF_NETLINK或者PF_NETLINK,套接字type选择SOCK_RAW或者SOCK_DGRAM,Netlink协议并不区分这两种类型,第三个参数协议填充NETLINK_KOBJECT_UEVENT表示接收内核uevent信息。接着就绑定该文件描述符到sockadd_nl,注意该结构体nl_groups是接收掩码,取~0是将接收所有来自内核的消息,我们接收热拔插只需要NETLINK_KOBJECT_UEVENT即可。接下来调用recvmsg开始接收内核消息,recvmsg函数需要我们填充message报头,包括指定接收缓存等工作。该函数会阻塞直到有热拔插事件产生。

运行程序,然后我插入一个U盘,得到下面的结果:

$ ./netlink 

received 289 bytes

add@/devices/pci0000:00/0000:00:1a.0/usb1/1-1/1-1.1

ACTION=add

DEVPATH=/devices/pci0000:00/0000:00:1a.0/usb1/1-1/1-1.1

SUBSYSTEM=usb

MAJOR=189

MINOR=8

DEVNAME=bus/usb/001/009

DEVTYPE=usb_device

DEVICE=/proc/bus/usb/001/009

PRODUCT=781/5530/100

TYPE=0/0/0

BUSNUM=001

DEVNUM=009

SEQNUM=2306

运行程序,拔掉U盘

$ ./netlink 

received 294 bytes

remove@/devices/pci0000:00/0000:00:1a.0/usb1/1-1/1-1.1/1-1.1:1.0/host10/target10:0:0/10:0:0:0/bsg/10:0:0:0

ACTION=remove

DEVPATH=/devices/pci0000:00/0000:00:1a.0/usb1/1-1/1-1.1/1-1.1:1.0/host10/target10:0:0/10:0:0:0/bsg/10:0:0:0

SUBSYSTEM=bsg

MAJOR=253

MINOR=2

DEVNAME=bsg/10:0:0:0

SEQNUM=2345

程序正确地接收到了U盘热拔插事件,通过该信息用户程序可以在第一时间得到事件通知。事实上热拔插的时候产生的消息可不止一条呢,可以在revmsg的时候用一个循环接收更多的消息。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Coding01

一步一步搭建 PHP 服务器环境

这两天翻了翻之前的技术文档,发现有一篇文档值得拿出来和大家分享:如何给一台全新的阿里云服务器,搭建环境 —— 来部署 PHP 项目,Node.js 项目等。

3052
来自专栏云极客Frey

腾讯云容器微服务API设计实践

本文会在腾讯云容器服务上面构造微服务基础小项目, 通过搭建ELK集群,实现利用Logstash 采集Nginx日志,收纳及利用kibana展示的功能。

2.5K22
来自专栏散尽浮华

Tomcat自带的Session共享方案实施记录

一般来说,在多个tomcat集群业务中,session会话共享是必须的需求,不然前端nginx转发过来的请求不知道之前请求在哪台tomcat节点上,从而就找不到...

2242
来自专栏云计算教程系列

如何在Ubuntu 16.04使用Buildbot建立持续集成系统

Buildbot是一个基于Python的持续集成系统,用于自动化软件构建,测试和发布过程。

2083
来自专栏哎_小羊

Git Review + Gerrit 安装及使用完成 Code-Review

目录 Code Review 介绍 Gerrit 介绍 环境、软件准备 Git-Review 安装 Gerrit 安装 简单的 Demo 示例 1、Code R...

1.2K9
来自专栏Aloys的开发之路

Linux压缩与解压常用命令

tar的相关参数 -c: 建立压缩档案 -x:解压 -t:查看内容 -r:向压缩归档文件末尾追加文件 -u:更新...

2007
来自专栏Java开发

ajaxFileUpload.js 的一些Bug

这里以前提到过 http://blog.csdn.net/qq_30930805/article/details/62427726

1171
来自专栏Jerry的SAP技术分享

使用VM Tools让VMware虚拟机里的ubuntu能够共享Windows系统的文件夹

我们经常有这样的使用场景,在宿主机Windows(Linux)操作系统上通过VMware软件安装了一个Linux(Windows)的虚拟机, 然后需要在两种操作...

1193
来自专栏魏艾斯博客www.vpsss.net

centOS 如何安装 lnmp 环境

3304
来自专栏哎_小羊

minikube 安装 Kubernetes Dashboard 并集成 Heapster

目录 Kubernetes Dashboard 介绍 环境、软件准备 Kubernetes Dashboard 安装 Heapster 插件安装 简单演示使用 ...

1.3K8

扫码关注云+社区