Chinaunix首页 | 论坛 | 博客
  • 博客访问: 164949
  • 博文数量: 60
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 458
  • 用 户 组: 普通用户
  • 注册时间: 2014-04-04 10:01
文章分类

全部博文(60)

文章存档

2016年(3)

2014年(57)

我的朋友

分类: LINUX

2014-04-05 18:34:00


 platform_get_resource函数源码分析:
/**
*platform_get_resource-getaresourceforadevice*@dev:platformdevice*@type:resourcetype*@num:resourceindex*/
structresource*platform_get_resource(structplatform_device*dev,
unsignedinttype,unsignedintnum)
{
inti;
for(i=0;inum_resources;i++){
structresource*r=&dev->resource[i];
if(type==resource_type(r)&&num--==0)
returnr;
}
returnNULL;}
EXPORT_SYMBOL_GPL(platform_get_resource);
staticinlineunsignedlongresource_type(conststructresource*res){
returnres->flags&IORESOURCE_TYPE_BITS;}
函数分析:
structresource*r=&dev->resource[i];
这行代码使得不管你是想获取哪一份资源都从第一份资源开始搜索。if(type==resource_type(r)&&num--==0)
这行代码首先通过type==resource_type(r)判断当前这份资源的类型是否匹配,如果匹配则再通过num--==0判断是否是你要的,如果不匹配重新提取下一份资源而不会执行num--==0这一句代码。通过以上两步就能定位到你要找的资源了,接着把资源返回即可。如果都不匹配就返回NULL。


实例分析:
下面通过一个例子来看看它是如何拿到设备资源的。staticstructresources3c_buttons_resource[]={
[0]={
.start=S3C24XX_PA_GPIO,.end
=S3C24XX_PA_GPIO+S3C24XX_SZ_GPIO-1,
.flags=IORESOURCE_MEM,
},[1]={
.start=IRQ_EINT8,.end
=IRQ_EINT8,
.flags=IORESOURCE_IRQ,
},[2]={
.start=IRQ_EINT11,.end
=IRQ_EINT11,
.flags=IORESOURCE_IRQ,
},[3]={
.start=IRQ_EINT13,.end
=IRQ_EINT13,
.flags=IORESOURCE_IRQ,
},[4]={
.start=IRQ_EINT14,.end
=IRQ_EINT14,
.flags=IORESOURCE_IRQ,
},[5]={
.start=IRQ_EINT15,.end
=IRQ_EINT15,
.flags=IORESOURCE_IRQ,
},[6]={
.start=IRQ_EINT19,.end
=IRQ_EINT19,
.flags=IORESOURCE_IRQ,
}
};



 


驱动中通过下面代码拿到第一份资源:structresource*res;
res=platform_get_resource(pdev,IORESOURCE_MEM,0);
函数进入for里面,i=0,num_resources=7,拿出resource[0]资源。
resource_type(r)提取出该份资源的资源类型并与函数传递下来的资源类型进行比较,匹配。Num=0(这里先判断是否等于0再自减1)符合要求,从而返回该资源。
获取剩下资源的代码如下:for(i=0;i<6;i++){
buttons_irq=platform_get_resource(pdev,IORESOURCE_IRQ,i);if(buttons_irq==NULL){
dev_err(dev,"noirqresourcespecified\n");ret=-ENOENT;gotoerr_map;}
button_irqs[i]=buttons_irq->start;
}
分析如下:For第一次循环:
buttons_irq=platform_get_resource(pdev,IORESOURCE_IRQ,0);
在拿出第一份资源进行resource_type(r)判断资源类型时不符合(此时num--==0这句没有执行),进而拿出第二份资源,此时i=1,num_resources=7,num传递下来为0,资源类型判断时候匹配,num也等于0,从而确定资源并返回。
For第二次循环:
buttons_irq=platform_get_resource(pdev,IORESOURCE_IRQ,1);
拿出第二份资源的时候resource_type(r)资源类型匹配,但是num传递下来时候为1,执行num--==0时不符合(但num开始自减1,这导致拿出第三份资源时num==0),只好拿出第三份资源。剩下的以此类推。



 


总结:
structresource*platform_get_resource(structplatform_device*dev,
unsignedinttype,unsignedintnum)
unsignedinttype决定资源的类型,
unsignedintnum决定type类型的第几份资源(从0开始)。即使同类型资源在资源数组中不是连续排放也可以定位得到该资源。
比如第一份IORESOURCE_IRQ类型资源在resource[2],而第二份在resource[5],那platform_get_resource(pdev,IORESOURCE_IRQ,0);可以定位第一份IORESOURCE_IRQ资源;
platform_get_resource(pdev,IORESOURCE_IRQ,1);可以定位第二份IORESOURCE_IRQ资源。
之所以能定位到资源,在于函数实现中的这一行代码:if(type==resource_type(r)&&num--==0)
该行代码,如果没有匹配资源类型,num--==0不会执行而重新提取下一份资源,只有资源匹配了才会寻找该类型的第几份资源,即使这些资源排放不连续。
阅读(1372) | 评论(1) | 转发(0) |
给主人留下些什么吧!~~

7大爷2014-04-08 13:48:09

不错