探究linux设备驱动模型之——platform虚拟总线(二)

  上回说到,platform_match是驱动和设备之间的媒人婆,那么platform_match是如何匹配驱动和设备的呢?platform总线定义的匹配条件很简单,主要就是查看驱动结构体和设备结构体的name成员变量是否相同,不同总线定义的匹配条件都不同的,例如USB总线的匹配条件就异常复杂,USB总线我们迟点会具体分析的。

  现在我们把驱动和设备分开来讲,首先会讲驱动,在最后我们会以一个实际的例子来说明,当然这个例子是没有任何的实际意义的,主要是拿来学习。在后面我们还会回过头来完善这个例子的。

  我们先看看platform总线定义的驱动结构,如下所示

  struct platform_driver {
     int (*probe)(struct platform_device *);
     int (*remove)(struct platform_device *);
     void (*shutdown)(struct platform_device *);
     int (*suspend)(struct platform_device *, pm_message_t state);
     int (*resume)(struct platform_device *);
     struct device_driver driver;
     struct platform_device_id *id_table;
  };

  当platform_match帮驱动找到匹配的设备的时候会在某个时候调用该结构体的probe函数,remove方法是在设备被移除前会被调用的(驱动移除时并不会调用该remove方法)。下面给出一个很简单的驱动模块,

 1 #include <linux/module.h>
 2 #include <linux/kernel.h>
 3 #include <linux/fs.h>
 4 #include <linux/init.h>
 5 #include <linux/platform_device.h>
 6 #include <linux/types.h>
 7 #include <asm/system.h>
 8 struct haoge_data
 9 {
10     char a[1024];
11 };
12 
13 
14 static int haoge_probe(struct platform_device *dev)
15 {
16     struct haoge_data * p =(dev->dev).platform_data;
17     
18     
19 
20     printk(KERN_ALERT "%s",p->a);
21     
22     return 0;
23 }
24 
25 
26 static struct platform_driver haoge_driver = {
27     .probe        = haoge_probe,
28 
29     .driver        = {
30         .name        = "haoge",
31         .owner        = THIS_MODULE,
32     },
33 };
34 
35 static int __init haoge_init(void)
36 {
37 
38     return platform_driver_register(&haoge_driver);
39 }
40 
41 static void __exit haoge_exit(void)
42 {
43     
44 
45     platform_driver_unregister(&haoge_driver);
46 }
47 
48 module_init(haoge_init);
49 module_exit(haoge_exit);
50 MODULE_LICENSE("GPL");
51 MODULE_AUTHOR("HaoGe");

  大家可以编译这个驱动模块,然后把他加载,一加载这个驱动platform总线就会把他挂接到该总线的驱动链表上。这个模块唯一值得一提的便是haoge_probe函数,这函数是我们定义的该驱动的probe方法。这个函数只有一个参数,便是struct platform_device *dev。从名字可以看出,这个参数便是指向设备结构体的一个指针。这个结构体如下所示

  

struct platform_device {

   const char * name;  

   int  id;  

   struct device dev;  

   u32  num_resources;  

   struct resource * resource;

   struct platform_device_id *id_entry;

       struct pdev_archdata archdata;

};

  我们提一下这个结构体的name成员变量,我们之前说过platform总线是以名字来匹配驱动和设备的,所以如果我们另外写一个设备模块,把设备结构体的的name成员赋值为"haoge",那么该设备就会匹配上面我们所写的那个驱动模块。

  我们再来看一下platform_device结构体的dev成员变量:struct device dev。其实device结构体才是总线、设备、驱动这三者中真正的设备一员,相应的device_driver是真正的驱动一员。platform_device和platform_driver是platform总线对device和device_driver这两个最底层结构体的封装。在device结构体中有个成员platform_data,他是专门用于驱动和设备之间传输数据的。在我们定义的haoge_probe函数中,我们便是通过platform_data把设备传输过来的数据用printk函数打印出来。

  下面我们给出设备模块的代码。

 

 1 #include <linux/module.h>
 2 #include <linux/kernel.h>
 3 #include <linux/fs.h>
 4 #include <linux/init.h>
 5 #include <linux/platform_device.h>
 6 #include <linux/types.h>
 7 #include <asm/system.h>
 8 
 9 struct haoge_data
10 {
11     char a[1024];
12 }s;
13 
14 struct platform_device haoge_device ={
15     .name= "haoge",
16     .id=1,
17     .dev = {
18             .platform_data = &s, 
19      },
20 };
21 
22 static int __init haoge_init(void)
23 {
24     sprintf(s.a,"haogeverygood!");
25 
26     platform_device_register(&haoge_device);
27     
28     return 0;
29 }
30 
31 static void __exit haoge_exit(void)
32 {
33     
34     platform_device_unregister(&haoge_device);
35 }
36 
37 module_init(haoge_init);
38 module_exit(haoge_exit);
39 MODULE_LICENSE("GPL");
40 MODULE_AUTHOR("HaoGe");

 一加载这个设备模块,platform总线就会把这个设备模块与驱动模块绑定起来,然后我们就会看到字符界面输出"haogeverygood!"的字样了。一般驱动和设备都是一对多的,一个驱动可以满足多个设备。所以我们可以加载另外一个设备,把上面的代码修改一下,如下:

 

 1 #include <linux/module.h>
 2 #include <linux/kernel.h>
 3 #include <linux/fs.h>
 4 #include <linux/init.h>
 5 #include <linux/platform_device.h>
 6 #include <linux/types.h>
 7 #include <asm/system.h>
 8 
 9 struct haoge_data
10 {
11     char a[1024];
12 }s2;
13 
14 struct platform_device haoge2_device ={
15     .name= "haoge",
16     .id = 2,
17     .dev = {
18             .platform_data = &s2, 
19      },
20 };
21 
22 static int __init haoge_init(void)
23 {
24     sprintf(s2.a,"haoge very handsone!");
25 
26     platform_device_register(&haoge2_device);
27     
28     return 0;
29 }
30 
31 static void __exit haoge_exit(void)
32 {
33     
34     platform_device_unregister(&haoge2_device);
35 }
36 
37 module_init(haoge_init);
38 module_exit(haoge_exit);
39 MODULE_LICENSE("GPL");
40 MODULE_AUTHOR("HaoGe");


  加载第二个设备模块,就会输出"haoge very handsome!"的字样。记住一点,这两个设备模块中的platform_device结构体中的id成员必须是不同的值,否则会发生kernel的错误。

  今天就写到这,下次我会再完善一下这个简单的例子

 

  

  

原文地址:https://www.cnblogs.com/Blue-Moon/p/3332385.html