mdev的基本工作原理【转】

简介:

转自:http://www.cnblogs.com/hnrainll/archive/2011/06/10/2077435.html

分析过mdev(udev的BusyBox简化版)源码的都知道mdev的基本原理: 

a、执行mdev -s命令时,mdev扫描/sys/block(块设备保存在/sys/block目录下,内核2.6.25版本以后,块设备也保存在/sys /class/block目录下。mdev扫描/sys/block是为了实现向后兼容)和/sys/class两个目录下的dev属性文件,从该dev 属性文件中获取到设备编号(dev属性文件以"major:minor\n"形式保存设备编号),并以包含该dev属性文件的目录名称作为设备名 device_name(即包含dev属性文件的目录称为device_name,而/sys/class和device_name之间的那部分目录称为 subsystem。也就是每个dev属性文件所在的路径都可表示为/sys/class/subsystem/device_name/dev),在 /dev目录下创建相应的设备文件。例如,cat /sys/class/tty/tty0/dev会得到4:0,subsystem为tty,device_name为tty0。

b、当mdev因uevnet事件(以前叫hotplug事件)被调用时,mdev通过由uevent事件传递给它的环境变量获取到:引起该uevent 事件的设备action及该设备所在的路径device path。然后判断引起该uevent事件的action是什么。若该action是add,即有新设备加入到系统中,不管该设备是虚拟设备还是实际物理 设备,mdev都会通过device path路径下的dev属性文件获取到设备编号,然后以device path路径最后一个目录(即包含该dev属性文件的目录)作为设备名,在/dev目录下创建相应的设备文件。若该action是remote,即设备已 从系统中移除,则删除/dev目录下以device path路径最后一个目录名称作为文件名的设备文件。如果该action既不是add也不是remove,mdev则什么都不做。 

由上面可知,如果我们想在设备加入到系统中或从系统中移除时,由mdev自动地创建和删除设备文件,那么就必须做到以下三点:1、在/sys/class 的某一subsystem目录下,2、创建一个以设备名device_name作为名称的目录,3、并且在该device_name目录下还必须包含一个 dev属性文件,该dev属性文件以"major:minor\n"形式输出设备编号。 

思路有了,剩下的就是如何实现了。下面通过介绍linux设备驱动模型里两个重要的数据结构:class和class_device,顺带讲述如何架构mdev驱动程序,并给出2个例子:HelloWorld和udev_demo。 

注意:下面讲到的结构体、函数以及例子都是针对linux-2.6.24.7版本内核,并不适合于所有版本的linux内核。各版本内核的结构体和函数原型可能有变。比如我在Ubuntu9.04下make一下第二个驱动例子,出现了好几个错误,后来发现Ubuntu9.04的内核linux-2.6.28里面关于linux设备驱动模型的数据结构有了很大改变,没有了class_device。所以当你跑下面两个例子时,如不是linux-2.6.24.7版本内核,那就要注意一下例子里使用数据结构与你内核里的数据结构是否一致。 

1、class 

一个类是一个设备的高层视图,它抽象掉了底层的实现细节。例如,在驱动层面时,你可能会见到SCSI磁盘或者ATA磁盘;但在类层面时,它们都是磁盘。类允许用户空间基于它们做什么来使用设备,而不是它们如何被连接或者它们如何工作。 

class表示一类设备,所有class都属于class_subsys(class子系统),即出现在/sys/class目录下,除了块设备(可能出现在/sys/block/或/sys/class/block,上面讲过了)。 

其实,class在/sys/class下生成的目录也就是上面提到subsystem。这样第1点就有了。 

/* class结构体 */ structclass{ 

constchar * name; /* class的名称 */ 

struct module * owner; /* 拥有该class的模块 */ 

struct kset subsys; /* 该class对应的子系统 */ 

struct list_head children; /* 该class的class_device列表 */ 

struct list_head devices; 

struct list_head interfaces; 

struct kset class_dirs; 

struct semaphore sem; /* locks both the children and interfaces lists */ 

struct class_attribute * class_attrs;/* 该class的默认属性,以NULL结尾 */ 

struct class_device_attribute * class_dev_attrs;/* 添加到class的class_device所拥有的默认属性 */ 

struct device_attribute * dev_attrs; 

/* 该函数提供在产生热插拔class_device事件时,添加环境变量的能力 */ 

int (*uevent)(struct class_device *dev,struct kobj_uevent_env *env); 

/* 该函数提供在产生热插拔device(物理设备)事件时,添加环境变量的能力 */ 

int (*dev_uevent)(struct device *dev,struct kobj_uevent_env *env); 

/* 添加到class的class_device移除时,调用该函数进行必要的清理工作 */ 

void(*release)(struct class_device *dev); 

/* class被移除时,调用该函数进行必要的清理工作 */ 

void(*class_release)(structclass*class); 

void(*dev_release)(struct device *dev); 

int (*suspend)(struct device *, pm_message_t state); 

int (*resume)(struct device *); 

}; 

/* class注册函数 */ 

int __must_check class_register(structclass*); 

void class_unregister(structclass*); 建立一个class有两种方法 

a、根据需要,填充一个struct class,然后再调用class_register注册该class就ok了(此法比较灵活,可以自己定制很多东西) 

b、就是通过下面的class_create来创建一个class,该函数会返回一个指向刚建立的class的指针(创建class的最简单方法) 

/* class_create用于创建一个名为name的class,其owner参数一般为THIS_MODULE。class_create内部调用了class_register */ 

structclass*class_create(struct module *owner,constchar*name); 

/* class_destroy用于删除一个class,实际上其内部只是简单调用了class_unregister(cls)来注销cls */ 

void class_destroy(structclass*cls); 一个class属性对应于/sys/class/class.name(class.name就是该class的名称)目录里的一个文件。通过这些文件, 可以向用户空间输出一些关于该class的信息,也可从用户空间获取到一些信息。 

/* class属性结构体 */ 

struct class_attribute { 

struct attribute attr; 

/* 当用户空间读取该属性时,调用show函数输出一个"属性值"给用户空间 */ 

ssize_t (*show)(structclass*,char* buf); 

/* 当用户空间写该属性时,调用store函数保存用户写入的"属性值" */ 

ssize_t (*store)(structclass*,constchar* buf,size_tcount); 

}; 

struct attribute { 

constchar * name; 

struct module * owner; 

mode_t mode; 

}; 

/* CLASS_ATTR可以在编译时创建一个class属性,该属性的名称为class_attr_name */ 

#define CLASS_ATTR(_name,_mode,_show,_store) \ 

struct class_attribute class_attr_##_name = __ATTR(_name,_mode,_show,_store) 

/* class_create_file与class_remove_file用于创建与删除class默认属性外的属性 */ 

int __must_check class_create_file(structclass*, 

conststruct class_attribute *); 

void class_remove_file(structclass*,conststruct class_attribute *); 

2、class_device 

一个class可以看成是一个容器(一个子系统subsystem),包含了很多的class_device,这些class_device是由class这个大的容器来管理的,而每个class_device都对应着一个具体的设备。 

每个class对象包括一个class_device链表,每个class_device对象表示一个逻辑设备并通过struct class_device中的dev成员(一个指向struct device的指针)关联一个物理设备。一个逻辑设备总是对应一个物理设备,而一个物理设备却可以对应多个逻辑设备。 

实际上,class_device在/sys/class/subsystem生成的目录就是上面提到的class_device。这样第2点也有了。 

/* class_device结构体 */ struct class_device { 

struct list_head node; /* 仅供驱动核心内部使用 */ 

struct kobject kobj; /* 该class_device相应的kobject,仅供驱动核心内部使用 */ 

structclass *class; /* 该class_device所属的class,必须有 */ 

dev_t devt; /* 该class_device的设备编号,用于创建其dev属性文件,仅供驱动核心内部使用 */ 

struct device * dev; /* 指向该class_device相关的device结构体(物理设备),可选.若不为NULL,用于创建一个从class入口到/sys/devices 下相应入口的符号连接,以便用户空间查找设备入口 */ 

void * class_data; /* 该class_device的私有数据指针 */ 

struct class_device *parent; /* parent of this child device, if there is one */ 

struct attribute_group ** groups; /* optional groups */ 

void (*release)(struct class_device *dev); 

int (*uevent)(struct class_device *dev,struct kobj_uevent_env *env); 

char class_id[BUS_ID_SIZE]; /* 该class_device的名称,在其所属class中应是唯一的,不可重名 */ 

}; 

/* class_devic注册函数 */ 

int __must_check class_device_register(struct class_device *); 

void class_device_unregister(struct class_device *); 与class一样,建立一个class_device也有两种方法 

a、根据需要,填充一个struct class_device,然后再调用class_device_register注册该class_device就ok了(此法比较灵活,可以自己定制很多东西) 

b、就是通过下面的class_device_create来创建一个class_device,该函数会返回一个指向刚建立的class_device的指针(创建class_device的最简单方法) 

/* class_device_create用于创建一个class_device,其名称最后两个参数决定(类似于printf的格式化字符串) 

* cls指明了其所属的class,可以是自己填充的class或由class_create返回的class 

* parent指明了该class_device的父class_device,若没有,则为NULL 

* 该class_device的设备编号,用于创建其dev属性文件,必须指明 

* device指明了该class_device(逻辑设备)对应的device(物理设备),可有可无,无则为NULL 

* 实际上,class_device_create也就是填充一个class_device,然后调用了class_device_register注册该class_device 

*/ 

struct class_device *class_device_create(structclass*cls, 

struct class_device *parent, 

dev_t devt, 

struct device *device, 

constchar*fmt,...) 

__attribute__((format(printf,5,6))); 

/* class_destroy用于删除一个class,内部调用了class_unregister(cls)来注销cls */ 

void class_device_destroy(structclass*cls, dev_t devt); class_device属性对应于/sys/class/class.name/class_device.class_id目录下一个文件。通过这些 文件,可以向用户空间输出一些关于该class_device的信息,也可从用户空间获取到一些信息。 

/* class_device属性,其show和store函数类似于class属性的show和store函数 */ 

struct class_device_attribute { 

struct attribute attr; 

ssize_t (*show)(struct class_device *,char* buf); 

ssize_t (*store)(struct class_device *,constchar* buf,size_tcount); 

}; 

/* CLASS_DEVICE_ATTR可以在编译时创建一个class_device属性,该属性的名称为class_device_attr_name */ 

#define CLASS_DEVICE_ATTR(_name,_mode,_show,_store) \ 

struct class_device_attribute class_device_attr_##_name = \ 

__ATTR(_name,_mode,_show,_store) 

/* class_device_create_file与class_device_remove_file用于创建与删除class_device默认属性外的属性 */ 

int __must_check class_device_create_file(struct class_device *, 

conststruct class_device_attribute *); 

void class_device_remove_file(struct class_device * class_dev, 

conststruct class_device_attribute * attr); 其实,在调用class_device_register注册一个class_device时,该函数内部调用了 

int class_device_add(struct class_device *class_dev) 

在class_device_add内,通过class_device_create_file创建dev、uevent和该class_device 所拥有的默认属性(由class_device.class->class_dev_attrs指定)等属性文件。这样第3点也有了。 

dev属性文件用于向用户空间输出该class_device的设备编号。 

uevent属性文件使用户可以手动触发uevent事件(通过向该文件写,如add、remove等字符串)。 

接下来是两个基于mdev的驱动例子。 

struct class_interface { 

struct list_head node; 

struct class *class; /* 该class_interface所属的class */ 

int (*add) (struct class_device *, struct class_interface *); /* class属性 */ 

void (*remove) (struct class_device *, struct class_interface *); /* class属性 */ 

int (*add_dev) (struct device *, struct class_interface *); 

void (*remove_dev) (struct device *, struct class_interface *); 

}; 

/* class_interface注册函数 */ 

int __must_check class_interface_register(struct class_interface *); 

void class_interface_unregister(struct class_interface *);

转自:http://hi.baidu.com/ss0ss70/blog/item/b62aaeef4376a6dd2f2e21a1.html













本文转自张昺华-sky博客园博客,原文链接:http://www.cnblogs.com/sky-heaven/p/5688109.html,如需转载请自行联系原作者

相关文章
|
8月前
|
编译器 网络性能优化 C语言
VPP DPDK,不是翻墙!!
VPP DPDK,不是翻墙!!
|
调度 C++
NR HARQ (四)dynamic codebook
上篇提到type-1 HARQ-ACK codebook,即semi-static codebook,UE要为每个PDSCH候选位置生成反馈,也会包含实际没有下行传输的PDSCH,再加上配置CBG的场景,HARQ-ACK 码本中包含的无用信息会更多,开销确实很大。因而,Type-2 HARQ-ACK 码本即dynamic 码本就出现了,目前看实网中一般都用的dynamic 码本,还没有见过semi-static 码本,长话短说,下面就看下相关内容。
EMC基本概念之电流环路 (Current Loop)
EMC基本概念之电流环路 (Current Loop)
|
缓存 监控 网络协议
Vibro-meter VM600 IOCN 用于冗余Modbus TCP通信
Vibro-meter VM600 IOCN 用于冗余Modbus TCP通信
103 0
Vibro-meter  VM600 IOCN 用于冗余Modbus TCP通信
|
5G 文件存储 数据安全/隐私保护
【OpenAirInterface5g】RRC NR解析(一)
RRC(Radio Resource Control)称为无线资源控制,是5G系统的核心模块,其主要作用是给下层(PHY,MAC,RLC,PDCP)控制或配置所有无线资源,从而保证UE与基站之间进行通信。 OAI RRC模块位于openair2\RRC下,其下有NR,LTE,NR UE等几个源码目录,我们只分析NR。
779 0
【OpenAirInterface5g】RRC NR解析(一)
|
文件存储
【OpenAirInterface5g】RRC NR解析之RrcSetupRequest
Rrc setup Request消息由UE发起,携带RRC建立原因和UE标识,用于UE向gNB请求建立RRC 连接,传输信道为CCCH。
695 0
【OpenAirInterface5g】RRC NR解析之RrcSetupRequest
|
安全 文件存储 数据安全/隐私保护
【OpenAirInterface5g】RRC NR解析之RrcSetupComplete
RRC SetupComplete是终端收到网络侧分配到无线资源的"RRC SETUP“消息的应答。UE在应答消息中对分配资源进行了确认,并回复了其5G_S_TMSI_Value Part2的内容,消息中携带selectedPLMN-Identity、registeredAMF、snssai-list和NAS,NAS信息为上下文建立和注册请求。基站(gNB)收到后根据其AMF标识,与核心网一同进行认证和注册流程。
633 0
【OpenAirInterface5g】RRC NR解析之RrcSetupComplete
|
安全 算法 5G
5G NR RRC协议总体介绍
此文基于3GPP协议进行总结,概括RRC入网过程,5G UE三种状态,SRB0~SRB3
1625 0
5G NR RRC协议总体介绍
|
缓存 自然语言处理 监控
PHY的基本知识
写DM9000网卡芯片驱动的预备知识 from:http://blog.sina.com.cn/s/blog_522a41b201009ha7.html ETHERNET的接口实质是MAC通过MII总线控制PHY的过程。
848 0

热门文章

最新文章

下一篇
开通oss服务