转自:http://blog.csdn.net/dfysy/article/details/5959451

  1. 版权声明:本文为博主原创文章,未经博主允许不得转载。
  2. 说句老实话,我不太喜欢现在Linux 2.6这套bus, platform, device,device driver 的模式。我觉得这种模式破坏了Linux的“简单就是美”的哲学,原来那套驱动已经可以包容所有驱动,也可以直接注册驱动文件和管理,而且以前的驱动在现在的结构上也还可以使用,把它在注册到bus这棵树上又有什么用呢?虽然可以看到一点对于移植性和平台管理方面的优点,但是我认为现在这种驱动编程的风格越来越像Windows的风格,很不直观和简约,让人理解起来相当的困惑。
  3. 牢骚发完了,bus结构还得继续,说说platform_deviceplatform_driver的匹配吧!一般来说系统上来在init_init_machine的对应函数ap_init中注册一些这个平台的设备,如下:
  4. static struct resource cbp_sdmmc_resource[]=
  5. {
  6. [0]={
  7. .start=HWD_MMC_BASE,
  8. .end=HWD_MMC_BASE+0xff0,
  9. .flags=IORESOURCE_MEM
  10. },
  11. [1]={
  12. .start=IRQ_SDMMC,
  13. .end=IRQ_SDMMC,
  14. .flags=IORESOURCE_IRQ
  15. },
  16. [2]={
  17. .start=IRQ_SDMMC_CD,
  18. .end=IRQ_SDMMC_CD,
  19. .flags=IORESOURCE_IRQ
  20. }
  21. };
  22. struct platform_device cbp_device_sdmmc = {
  23. .name = "cbp-sdmmc",
  24. .id = -1,
  25. .num_resources = ARRAY_SIZE(cbp_sdmmc_resource),
  26. .resource = cbp_sdmmc_resource,
  27. .dev = {
  28. .coherent_dma_mask = 0xffffffffUL
  29. }
  30. };
  31. static struct platform_device * cbp_devices[] __initdata = {
  32. &cbp_device_sdmmc
  33. };
  34. static void __init ap_init(void)
  35. {
  36. platform_add_devices(cbp_devices,ARRAY_SIZE(cbp_devices));
  37. }
  38. 说明这个平台使用的SD/MMC驱动的名字叫"cbp-sdmmc",然后在驱动中用platform_driver_register声明对应的platform_driver来使用上面声明的平台资源,如下:
  39. static int __init cbpmci_init(void)
  40. {
  41. return platform_driver_register(&cbpmci_driver);
  42. }
  43. platform_driverplatform_driver的匹配方式有两种:
  44. 1)直接根据名字来进行匹配,这种方式是比较常用的方式,比如如下申明cbpmci_driver
  45. static struct platform_driver cbpmci_driver = {
  46. .probe = cbpmci_probe,
  47. .driver = {
  48. .name = "cbp-sdmmc",
  49. },
  50. };
  51. 2)通过id_talbe来实现,这种实现的最终还是通过名字对应来匹配,但是匹配的名字被列在一个表中,platform_devicename和这个表中的每一个值进行比较,知道找到相同的那一个,如下申明:
  52. static struct platform_device_id cbpmci_driver_ids[] = {
  53. {
  54. .name = "other-sdmmc",
  55. .driver_data = 0,
  56. },
  57. {
  58. .name = "cbp-sdmmc",
  59. .driver_data = 1,
  60. },
  61. { }
  62. };
  63. static struct platform_driver cbpmci_driver = {
  64. .driver = {
  65. .name = "vtc_sdmmc",
  66. .owner = THIS_MODULE,
  67. .pm = &cbpmci_pm_ops,
  68. },
  69. .id_table = cbpmci_driver_ids,
  70. .probe = cbpmci_probe,
  71. .remove = __devexit_p(cbpmci_remove),
  72. .shutdown = cbpmci_shutdown,
  73. };
  74. id_table不为空的时候,.driver.name中的名字“vtc_sdmmc”就不管用了,会按照id_table中的内容进行匹配,同时匹配后的会把匹配上的platform_device_id保存在platform_device结构的id_entry中,在probe的时候就可以通过id_entry中的driver_data判断匹配的到底是cbpmci_driver_ids中的哪一组ID
  75. 比如上面例子中的"cbp-sdmmc"中匹配的是第二组,这样就可以再Probe的判断到底是什么平台了
  76. ifplatform_get_device_id(pdev)->driver_data==1) printk("cgp SD/MMC support/n");
  77. 这种匹配方式在三星的SD/MMC中有使用,由于24122440的地址是一样的,而2410不一样,所以通过driver_data 是否为1来区分。
  78. static struct platform_device_id s3cmci_driver_ids[] = {
  79. {
  80. .name = "s3c2410-sdi",
  81. .driver_data = 0,
  82. }, {
  83. .name = "s3c2412-sdi",
  84. .driver_data = 1,
  85. }, {
  86. .name = "s3c2440-sdi",
  87. .driver_data = 1,
  88. },
  89. { }
  90. };
  91. 一下是Linux匹配的源代码,在platform.c中,一看就一目了然了,当然要跟到这步,中间还有好多指针要走:
  92. static int platform_match(struct device *dev, struct device_driver *drv)
  93. {
  94. struct platform_device *pdev = to_platform_device(dev);
  95. struct platform_driver *pdrv = to_platform_driver(drv);
  96. /* match against the id table first */
  97. if (pdrv->id_table)
  98. return platform_match_id(pdrv->id_table, pdev) != NULL;
  99. /* fall-back to driver name match */
  100. return (strcmp(pdev->name, drv->name) == 0);
  101. }
  102. 此外platform_add_devices最终会调用platform_device_register,而platform_device_registerplatform_driver_register应该先调用哪一个理论上讲是先调用device然后driver,但其实是无所谓的,最后都会调用device_attach()来probe

 

版权声明:本文为sky-heaven原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/sky-heaven/p/6869591.html