linux中每个设备驱动由一个structdevice_driver描述:
structdevice_driver {
constchar*name;//设备驱动程序的名称
structbus_type *bus;//该驱动所管理的设备挂接的总线类型
structmodule *owner;
constchar*mod_name;/* used for built-in modules */
int(*probe) (structdevice *dev);
int(*remove) (structdevice *dev);
void(*shutdown) (structdevice *dev);
int(*suspend) (structdevice *dev, pm_message_t state);
int(*resume) (structdevice *dev);
structattribute_group **groups;
structdev_pm_ops *pm;
structdriver_private *p;
};
structdriver_private {
structkobject kobj;
structklist klist_devices;//该驱动所管理的设备链表头
structklist_node knode_bus;//挂入总线链表中的指针
structmodule_kobject *mkobj;
structdevice_driver *driver;
};
structdriver_attribute {
structattribute attr;
ssize_t (*show)(structdevice_driver *driver,char*buf);
ssize_t (*store)(structdevice_driver *driver,constchar*buf,
size_t count);
};
可以看出device结构体中也包含了一个kobject对象和连接设备的链表。
当设备从系统总删除是remove被调用。
当系统关机的时候shutdown被调用。注册一个设备驱动int driver_register(struct driver*drv)删除一个设备驱动void driver_unregister(struct driver*drv)
创建属性
int driver_create_file(struct driver *drv,struct driver_attribute *attr)
删除属性
intdriver_remove_file(struct driver*drv,struct driver_attribute *attr)
structdevice_driver中有个int(*probe) (structdevice *dev)函数,它什么时候调用呢?device_driver中name指向驱动的名字,前面的structdevice中也有一个名为bus_id的字符数组。查看一下,struct bus_type中有一个match函数,这个是干什么用的呢。设备有了驱动才可以工作,只有驱动没有设备也是不行,驱动和设备需要关联上,这就需要这个match函数。驱动和设备是通过name来管理的,所以在总线驱动match函数中要比较device的bus_id和driver中的name是否相等,什么时候比较?只要往总线添加设备或驱动时,总线都会把调用match函数对新添加的设备或驱动中名字与总线中已经注册的驱动或设备的名字一一比较。如果有相等,就说明驱动和设备互相找到了,这时device_driver中的probe函数就被调用。
添加设备驱动例子,添加到前面创建的my_bus总线上:
#include
#include
#include
#include
#include
MODULE_AUTHOR("David Xie");
MODULE_LICENSE("Dual BSD/GPL");
extern struct bus_type my_bus_type;
static int my_probe(struct device *dev)
{
printk("Driver found device which my driver can handle!\n");
return 0;
}
static int my_remove(struct device *dev)
{
printk("Driver found device unpluged!\n");
return 0;
}
struct device_driver my_driver = {
.name = "my_dev",
.bus = &my_bus_type,
.probe = my_probe,
.remove= my_remove,
};
/*
* Export a simple attribute.
*/
static ssize_t mydriver_show(struct device_driver *driver, char *buf)
{
return sprintf(buf, "%s\n", "This is my driver!");
}
static DRIVER_ATTR(drv, S_IRUGO, mydriver_show, NULL);
static int __init my_driver_init(void)
{
int ret = 0;
/*注册驱动*/
driver_register(&my_driver);
/*创建属性文件*/
driver_create_file(&my_driver, &driver_attr_drv);
return ret;
}
static void my_driver_exit(void)
{
driver_unregister(&my_driver);
}
module_init(my_driver_init);
module_exit(my_driver_exit);
创建一个名为“bus_dev”的驱动,并将bus成员指向第一步创建的my_bus_type总线
测试结果:加载驱动将会打印:Driver found device which my driver can handle!
查看/sys/bus/my_bus/driver
ls/sys/bus/my_bus/drivermy_dev