设备驱动外传 – 虚拟总线和platform device「建议收藏」

设备驱动外传 – 虚拟总线和platform device「建议收藏」1.总论2.系统初始化platformdevice3.驱动程序使用platformdevice   1.总论Linux-2.6.11引入了设备模型的概念,将大部分设备驱动挂载到虚拟总线上。其目的在于:1)提供友好的用户接口,用户可以在sys/bus/platform/下找到相应的驱动和设备。2)更有利于电源管理。  2

大家好,又见面了,我是你们的朋友全栈君。

1. 总论

2. 系统初始化 platform device

3. 驱动程序使用 platform device

 

  

1. 总论

Linux-2.6.11引入了设备模型的概念,将大部分设备驱动挂载到虚拟总线上。

其目的在于:

1) 提供友好的用户接口,用户可以在sys/bus/platform/下找到相应的驱动和设备。

2) 更有利于电源管理。

 

 

2. 系统初始化 platform device

在驱动程序的初始化函数 xxx_probe(struct platform_device *pdev)中,其结构体参数platform_device,是在系统初始化过程中发现了该设备存在的前提条件下,通过platform_device_register(dev) 完成注册的,并且struct platform_device的结构变量被赋值。

以Xscal初始化sd卡控制器为例,

 

 /* linux/arch/arm/mach-pxa/starwood_p1.c */

MACHINE_START(SAAR, “PXA935 handheld Platform (Starwood P1)”)
 … …
 .init_machine   = saar_init,
MACHINE_END

 

static void __init saar_init(void)
{

 … …

 saar_init_mmc();

  … …
}

 

static void __init saar_init_mmc(void)
{

 … …
  pxa_set_mci_info(&saar_mci_platform_data); 
//saar_mci_platform_data完成该设备特有数据的赋值
 … …
 }

 

static struct pxamci_platform_data saar_mci_platform_data = {

 .detect_delay = 50,
 .ocr_mask = MMC_VDD_32_33|MMC_VDD_33_34,
 .init   = saar_mci_init,   //回调函数,设置sd卡控制器的探测中断pin脚 gpio_cd
 .exit  = saar_mci_exit,
};

 

void __init pxa_set_mci_info(struct pxamci_platform_data *info)
{

 pxa_register_device(&pxa_device_mci, info);
}

 

struct platform_device pxa_device_mci = {     //完成结构体platform_device的赋值
 .name  = “pxa2xx-mci”,    //芯片的设备名称,驱动通过匹配该名字找到设备
 .id  = 0,
 .dev  = {

  .dma_mask = &pxamci_dmamask,
  .coherent_dma_mask = 0xffffffff,
 },
 .num_resources = ARRAY_SIZE(pxamci_resources),
 .resource = pxamci_resources,  
//设置芯片的物理地址
};

 

void __init pxa_register_device(struct platform_device *dev, void *data)
{

 … …

 dev->dev.platform_data = data;     //将该设备特有的数据挂到 dev->dev.platform_data下

 platform_device_register(dev);  

 … …
}

 

 

 

3. 驱动程序使用 platform device

 驱动程序通过platform_driver_register()找到与该驱动对应的设备,完成驱动和设备的绑定,并且挂载到虚拟总线上。

具体过程如下,

 platform_driver_register(struct platform_driver *drv);

                 |

 driver_register(&drv->driver);

                 |

 bus_add_driver(drv); 

                 |

 driver_attach(drv);

int driver_attach(struct device_driver *drv)
{

 return bus_for_each_dev(drv->bus, NULL, drv, __driver_attach);  //__driver_attach最后调用driver_bound(dev);
}

 

int bus_for_each_dev(struct bus_type *bus, struct device *start,
       void *data, int (*fn)(struct device *, void *))
{

 … …
 while ((dev = next_device(&i)) && !error)
  fn(dev, data);  //fn即__driver_attach,完成设备与驱动的绑定

  … …
}

 static void driver_bound(struct device *dev)
{

 … …

 klist_add_tail(&dev->knode_driver, &dev->driver->p->klist_devices);
}

 

 

 

驱动程序找到了自己所对应的设备之后,就可以对该设备初始化了,即probe()操作,在probe()中,便可以使用platform_device结构体里的数据了。

 

仍然以sd卡控制器为例,

/* linux/drivers/mmc/host/pxamci.c */

 

static int __init pxamci_init(void)
{

 … …
 return platform_driver_register(&pxamci_driver);
}

 

static struct platform_driver pxamci_driver = {

 .probe  = pxamci_probe,
 … …
};

 

static int pxamci_probe(struct platform_device *pdev)
{

 

  struct resource *r;

 

/*

 *  得到sd卡控制器芯片的物理地址

 */

 r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
 r = request_mem_region(r->start, SZ_4K, DRIVER_NAME);

 host->res = r;

/*

 *  得到sd卡控制器芯片的内部中断源并申请中断 

 */

 irq = platform_get_irq(pdev, 0);

 host->irq = irq;
 request_irq(host->irq, pxamci_irq, 0, DRIVER_NAME, host);

 

   … …
 }

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/163805.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • js禁止浏览器后退按钮[通俗易懂]

    js禁止浏览器后退按钮[通俗易懂]js禁止浏览器后退按钮1.js//禁止浏览器后退按钮functionBanBack(ele){//禁止浏览器后退按钮if(window.history&&window.history.pushState){$(window).on(‘popstate’,function(){window.histo…

  • 数据结构——线索化二叉树和哈夫曼树[通俗易懂]

    数据结构——线索化二叉树和哈夫曼树[通俗易懂]线索化二叉树和哈夫曼树基础知识介绍与代码分析一、基础知识介绍二、代码分析:线索二叉树(采用中序遍历)#include “pch.h”#include <iostream>using namespace std;//定义线索二叉树typedef struct Tree{ int data, LTag, RTag; //定义数据域与标记域 Tre…

  • 黄聪:在C#中如何使用资源中的图片

    黄聪:在C#中如何使用资源中的图片

  • IntelliJ IDEA 整理代码格式 快捷键[通俗易懂]

    IntelliJ IDEA 整理代码格式 快捷键[通俗易懂]一、前言在开发的过程中,项目代码格式尤为重要,但是有些开发人员经常会不注意细节,导致代码阅读性很差,如下图所示:二、解决方案打开IDEA,按Ctrl+Alt+L键,进行整理代码格式,可以看到代码已经进行整理PS:如果和qq热键冲突的话,需要先将qq的热键Ctrl+Alt+L设置为无,如下图所示:…

    2022年10月12日
  • 创意的课程设计——憨憨钟设计(碾压)

    创意的课程设计——憨憨钟设计(碾压)目录写在前面的话设计的实际演示憨憨钟的主要功能设计的实现过程界面设计模型结构设计元件清单调试异常分析结语写在前面的话大家好,学习之余给大家分享一个这学期设计的课程设计,互相学习借鉴。课程设计,尤其是自定义的课程设计,自盘古开天地以来就困扰着每一位工科男。我,一枚工科男,觉得需要站出来为大家排忧解难了。这学期有自定义的课程设计,所以我就设计了一个很有创意的课程设计,是一个国外开源的设计,结合了国外开源资料和一些大佬的实践经验。设计的实际演示憨憨钟自动写时间的效果:憨憨钟的演示效果1憨憨钟自动画天气

  • jmeter的正则表达式提取器_正则表达式提取

    jmeter的正则表达式提取器_正则表达式提取应用场景:在一个线程组中,B请求需要使用A请求返回的数据,也就是常说的关联,将上一个请求的响应结果作为下一个请求的参数,则需要对A请求的响应报文使用后置处理器,其中最方便最常用的就是正则表达式提取器了。正则表达式提取器:允许用户从作用域内的sampler请求的服务器响应结果中通过正则表达式提取值所需值,生成模板字符串,并将结果存储到给定的变量名中。先上个图:各配置项介绍:APPlyto:作用范围…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号