驱动程序辨认设备时,有以下两种办法:
(1)驱动程序自身带有设备信息,比方开端地址、中止号等;加载驱动程序时,就能够依据这些信息来辨认设备。
(2)驱动程序自身没有设备信息,可是内核中现已(或今后)依据其他办法确认了许多设备的信息;加载驱动程序时,将驱动程序与这些设备逐一比较,确认两者是否匹配(math)。假如驱动程序与某个设备匹配,就能够经过该驱动程序来操作这个设备了。
内核常运用第二种办法来辨认设备,这能够将各种设备会集在一个文件中办理,当开发板的装备改动时,便于修正代码。在内核文件include/linux/platform_device.h中,界说了两个数据结构来表明这些设备和驱动程序:platform_device结构用来描绘设备的称号、ID、所占用的资源(比方内存地址/巨细、中止号)等;platform_driver结构用来描绘各种操作函数,比方枚举函数、移除设备函数、驱动称号等。
内核发动后,首要结构链表将描绘设备的platform_device结构组织起来,得到一个设备的列表;当加载某个驱动程序的platform_driver结构时,运用一些匹配函数来查看驱动程序能否支撑这些设备,常用的查看办法很简单:比较驱动程序和设备的称号。
以S3C2410开发板为例,在include/arch/arm/mach-s3c2410/mach-smdk2410.c中界说了如下设备:
89 static struct platform_device *smdk2410_devices[] __initdata = {
90 &s3c_device_usb,
91 &s3c_device_lcd,
92 &s3c_device_wdt,
93 &s3c_device_i2c,
94 &s3c_device_iis,
95 };
104 static void __init smdk2410_init(void)
105 {
106 platform_add_devices(smdk2410_devices, ARRAY_SIZE(smdk2410_devices));
107 smdk_machine_init();
108 }
在include/arch/arm/plat-s3c24xx/common-smdk.c中界说了如下设备:
175 static struct platform_device __initdata *smdk_devs[] = {
176 &s3c_device_nand,
177 &smdk_led4,
178 &smdk_led5,
179 &smdk_led6,
180 &smdk_led7,
181 };
183 void __init smdk_machine_init(void)
184 {
185 /* Configure the LEDs (even if we have no LED support)*/
186
187 s3c2410_gpio_cfgpin(S3C2410_GPF4, S3C2410_GPF4_OUTP);
188 s3c2410_gpio_cfgpin(S3C2410_GPF5, S3C2410_GPF5_OUTP);
189 s3c2410_gpio_cfgpin(S3C2410_GPF6, S3C2410_GPF6_OUTP);
190 s3c2410_gpio_cfgpin(S3C2410_GPF7, S3C2410_GPF7_OUTP);
191
192 s3c2410_gpio_setpin(S3C2410_GPF4, 1);
193 s3c2410_gpio_setpin(S3C2410_GPF5, 1);
194 s3c2410_gpio_setpin(S3C2410_GPF6, 1);
195 s3c2410_gpio_setpin(S3C2410_GPF7, 1);
196
197 if (machine_is_smdk2443())
198 smdk_nand_info.twrph0 = 50;
199
200 s3c_device_nand.dev.platform_data = &smdk_nand_info;
201
202 platform_add_devices(smdk_devs, ARRAY_SIZE(smdk_devs));
203
204 s3c2410_pm_init();
205 }
627 int __init s3c2410_pm_init(void)
628 {
629 printk("S3C2410 Power Management, (c) 2004 Simtec Electronics\n");
630
631 suspend_set_ops(&s3c2410_pm_ops);
632 return 0;
633 }
这些设备在smdk2410_init函数(对应S3C2410)或smdk2440_machine_init函数(对应S3C2440)中,经过plat_add_devices函数注册进内核中。
NAND Flash设备s3c2410_device_nand在include/arch/arm/plat-s3c24xx/devs.c中的界说如下:
201 struct platform_device s3c_device_nand = {
202 .name = "s3c2410-nand",
203 .id = -1,
204 .num_resources = ARRAY_SIZE(s3c_nand_resource),
205 .resource = s3c_nand_resource,
206 };
207
208 EXPORT_SYMBOL(s3c_device_nand); /*导出符号,以便外部文件引证*/
关于S3C2440开发板,s3c_device_nand结构的姓名会在s3c244x_map_io函数中修正为“s3c2440-nand”,这个函数在include/arch/arm/plat-s3c24xx/s3c244x.c中界说如下:
59 void __init s3c244x_map_io(struct map_desc *mach_desc, int size)
…
68 s3c_device_i2c.name = "s3c2440-i2c";
69 s3c_device_nand.name = "s3c2440-nand";
70 s3c_device_usbgadget.name = "s3c2440-usbgadget";
71 }
有了NAND Flash设备,还要有NAND Flash驱动程序,内核针对S3C2410、S3C2412、S3C2440界说了3个驱动。它们在include/drivers/mtd/nand/s3c2410.c中的s3c2410_nand_init函数中注册进内核,如下所示:
890 static int __init s3c2410_nand_init(void)
891 {
892 printk("S3C24XX NAND Driver, (c) 2004 Simtec Electronics\n");
893
894 platform_driver_register(&s3c2412_nand_driver);
895 platform_driver_register(&s3c2440_nand_driver);
896 return platform_driver_register(&s3c2410_nand_driver);
897 }
其间的S3C2410_nand_driver结构是在include/drivers/mtd/nand/s3c2410.c中界说,如下所示:
857 static struct platform_driver s3c2410_nand_driver = {
858 .probe = s3c2410_nand_probe,
859 .remove = s3c2410_nand_remove,
860 .suspend = s3c24xx_nand_suspend,
861 .resume = s3c24xx_nand_resume,
862 .driver = {
863 .name = "s3c2410-nand",
864 .owner = THIS_MODULE,
865 },
866 };
可见,s3c_device_nand结构和s3c2410_nand_driver结构中的name成员相同,都是“s3c2410-nand”。platform_driver_register函数便是依据这点确认它们是匹配的,所以调用s3c2410_nand_probe函数枚举NAND Flash设备s3c2410_device_nand。