Linux设备驱动开发 - 平台设备驱动

时间:2020-12-05 17:56:45

Linux2.6的内核中引入了一种新的设备驱动模型-平台(platform)设备驱动,平台设备驱动分为平台设备(platform_device)和平台驱动(platform_driver),平台设备的引入使得Linux设备驱动更加便于移植。

一、平台设备
平台设备结构体:

 1 struct platform_device {
2 const char * name; /* 设备名 */
3 int id;
4 struct device dev; /* 设备结构体 */
5 u32 num_resources; /* 设备资源数量 */
6 struct resource * resource; /* 设备资源 */
7
8 const struct platform_device_id *id_entry;
9
10 /* arch specific additions */
11 struct pdev_archdata archdata;
12 };

平台设备主要是提供设备资源和平台数据给平台驱动,resource为设备资源数组,类型有IORESOURCE_IO、IORESOURCE_MEM、IORESOURCE_IRQ、IORESOURCE_DMA、IORESOURCE_DMA。下面是一个网卡芯片DM9000的外设资源:

 1 static struct resource dm9000_resources[] = {
2 [0] = {
3 .start = S3C64XX_PA_DM9000,
4 .end = S3C64XX_PA_DM9000 + 3,
5 .flags = IORESOURCE_MEM,
6 },
7 [1] = {
8 .start = S3C64XX_PA_DM9000 + 4,
9 .end = S3C64XX_PA_DM9000 + S3C64XX_SZ_DM9000 - 1,
10 .flags = IORESOURCE_MEM,
11 },
12 [2] = {
13 .start = IRQ_EINT(7),
14 .end = IRQ_EINT(7),
15 .flags = IORESOURCE_IRQ | IRQF_TRIGGER_HIGH,
16 },
17 };

dm9000_resources里面有三个设备资源,第一个为IORESOURCE_MEM类型,指明了第一个资源内存的起始地址为S3C64XX_PA_DM9000结束地址为S3C64XX_PA_DM9000 + 3,第二个同样为IORESOURCE_MEM类型,指明了第二个资源内存的起始地址为S3C64XX_PA_DM9000 + 4结束地址为S3C64XX_PA_DM9000 + S3C64XX_SZ_DM9000 - 1,第三个为IORESOURCE_IRQ类型,指明了中断号为IRQ_EINT(7)。

 1 struct device {
2 struct device *parent;
3
4 struct device_private *p;
5
6 struct kobject kobj;
7 const char *init_name; /* initial name of the device */
8 struct device_type *type;
9
10 struct mutex mutex; /* mutex to synchronize calls to
11 * its driver.
12 */
13
14 struct bus_type *bus; /* type of bus device is on */
15 struct device_driver *driver; /* which driver has allocated this
16 device */
17 void *platform_data; /* Platform specific data, device
18 core doesn't touch it */
19 ...
20 };

 

struct device结构体里面有一个重要成员platform_data,它是平台设备和平台驱动进行数据传递的重要成员。

 

平台设备注册:

1 int platform_device_register(struct platform_device *pdev);

platform_device_register()会对平台设备进行相应的初始化之后调用platform_device_register()函数把它添加到子系统中。

 

平台设备注销:

1 void platform_device_unregister(struct platform_device *pdev);

platform_device_unregister()函数释放设备资源之后从子系统中将其移除。

 

平台设备模板:

 1 static struct resource xxx_resource = 
2 {
3 [0] =
4 {
5 .start = ...,
6 .end = ...,
7 .flags = ...,
8 },
9 [1] =
10 {
11 ...
12 }
13 ...
14 };
15
16 static struct xxx_plat_data xxx_data =
17 {
18 ...
19 };
20
21 static struct platform_device xxx_platform_device =
22 {
23 .name = NAME,
24 .num_resources = ARRAY_SIZE(xxx_resource),
25 .resource = xxx_resource,
26 .dev =
27 {
28 .platform_data = &xxx_data,
29 }
30 };
31
32 static int __init xxx_device_init(void)
33 {
34 ...
35 /* 注册平台设备 */
36 platform_device_register(&xxx_platform_device);
37 ...
38 }
39
40 static void __exit xxx_device_exit(void)
41 {
42 ...
43 /* 注销平台设备 */
44 platform_device_unregister(&xxx_platform_device);
45 ...
46 }

 

 

二、平台驱动
平台驱动结构体:

1 struct platform_driver {
2 int (*probe)(struct platform_device *);
3 int (*remove)(struct platform_device *);
4 void (*shutdown)(struct platform_device *);
5 int (*suspend)(struct platform_device *, pm_message_t state);
6 int (*resume)(struct platform_device *);
7 struct device_driver driver;
8 const struct platform_device_id *id_table;
9 };

平台驱动结构体driver成员里面的name必须与平台设备结构体里面的name成员一致,在系统注册一个设备的时候,会通过设备结构体里面的name成员和平台驱动driver里面的name成员匹配,当匹配成功则调用平台驱动的probe函数,
通常在probe函数中获取平台设备的资源和私有数据并进行设备的初始化。

获取设备资源:

1 struct resource *platform_get_resource(struct platform_device *dev, unsigned int type, unsigned int num);

platform_get_resource()函数用于获取平台设备的资源,dev为要平台设备,type为平台设备资源类型,num为平台资源号(比如同一个资源有两个则资源号为0,1)。

 

平台驱动注册:

1 int platform_driver_register(struct platform_driver *drv);

platform_driver_register()函数完成平台驱动的注册,在驱动模块加载时调用。

 

平台驱动注销:

1 void platform_driver_unregister(struct platform_driver *drv);

platform_driver_unregister()函数完成平台驱动的注销,在驱动模块卸载时调用。

 

平台驱动模板:

 1 static int __devinit xxx_probe(struct platform_device *pdev)
2 {
3 struct xxx_plat_data *pdata = pdev->dev.platform_data; /* 获取私有数据 */
4 platform_get_resource(pdev,xxx,x); /* 获取设备资源 */
5 ...
6 }
7
8 static struct platform_driver xxx_platform_driver =
9 {
10 .probe = xxx_probe,
11 .remove = __devexit_p(xxx_remove),
12 .driver =
13 {
14 .name = NAME, /* 跟平台设备名一致 */
15 ...
16 },
17 ...
18 };
19
20 static int __init xxx_driver_init(void)
21 {
22 ...
23 /* 驱动注册 */
24 platform_driver_register(&xxx_platform_driver);
25 ...
26 }
27
28 static void __exit xxx_driver_exit(void)
29 {
30 ...
31 /* 驱动注销 */
32 platform_driver_unregister(&xxx_platform_driver);
33 ...
34 }