Linux随着硬件设备的发展及内核版本的演进,设备模型也变得越来月复杂,早先看了《Linux设备驱动程序》觉得一头雾水,又看了许多资料和高手的帖子,总算有了一定认识,下面写出来和Linux内核爱好者分享一下。
一、底层数据结构:kobject 和 kset
1、kobject内核对象:Linux2.6设备模型的最底层核心结构,该数据结构使所有设备在底层都具有统一的接口,每一个在内核中注册的kobject对象都对应于sysfs文件系统中的一个目录。Kobject在内核中对应有一套申请,初始化,添加,注册,计数操作,释放等函数。
2、 kset内核对象集合:具有相同类型的kobject的集合。
3、 subsystem内核对象子系统:一系列kset的集合,描述了某一类子系统,如block_subsys表示所有的块设备,对应于susfs文件系统中的block目录。device_subsys对应于sysfs中的devices目录,描述系统中的所有的设备,说直白了其实也是kset,kset的类型由内嵌的ktype结构描述。
由此可以看出,kobject类似于面向对象程序设计中的基类,经过层层继承封装来实现上层的设备驱动模型。
二、设备模型层次关系:bus_type,device,device_driver
驱动核心可以注册多种类型的bus;每种bus可以挂载许多device(通过kset devices);每种bus下可以有很多device_driver(通过kset drivers);每个device_driver可以处理一组devices。
bus是处理器与一个或多个设备之间的通道。在设备模型中,所有的设备都通过总线相连。每个总线都有自己的子系统,一个总线中包含了两个kset,一个是总线的驱动程序,一个是插入总线的所有设备。
device描述设备相关的信息,设备之间的层次关系,以及设备与总线、驱动之间的关系。通常device结构体不单独使用,而是包含在更大的结构体中。
device_driver结构体描述系统中的每个驱动程序,通常也不直接使用,而是包含在更大的结构体中。
三、面向对象思想在Linux设备模型中的体现:
内核中常见到封装了数据和方法的结构体,这种结构体套结构体的结构体可以看成是面向对象封装特性的实现。而Linux设备模型展现的更多的是继承方面的实现。以《Linux设备驱动程序》中的pci_driver为例,它的父类是device_driver,而更上一层是一个kobject。在C++中,继承一个父类则子类中相应的包含父类的一个实例。内核中也是通过包含一个父类的实体来实现这种派生关系。因此,一个pci_driver内部必然包含一个device_driver,同样,device_driver内部必然包含一个kobject。注册一个模型的过程类似于面向对象中构造函数的调用。子类需要调用父类构造函数来完成自身的构造。而注册一个pci_driver的过程如下:
pci_register_driver(struct pci_driver *driver)
-->driver_register(&drv->driver);
-->kobject_register(&drv->kobj);
这正符合这面向对象的思路。所以鄙人在想,如果C++在在某些底层操作方面进行优化和该进,那么用c++改写的Linux会是会不会更高效,鄙人拙见。。。