linux device driver 关系,[zt]linux resource, platform_device和驱动的关系

static struct map_desc smdk2410_iodesc[] __initdata = {

{

/* Map the ethernet controller CS8900A */

.virtual =  vSMDK2410_ETH_IO,

.pfn = __phys_to_pfn(pSMDK2410_ETH_IO),

.length = SZ_1M,

.type = MT_DEVICE

},

};

这样的定义,无非是希望系统将 pSMDK2410_ETH_IO这个物理地址给映射到虚地址vSMDK2410_ETH_IO上,占用可操作的长度是1M.

接下来在我的CS8900的驱动里就能利用这个转换后的地址操作我的CS8900了。例如一下的代码:

dev->base_addr = vSMDK2410_ETH_IO + 0x300;

dev->irq = SMDK2410_ETH_IRQ;

if ((result = check_mem_region (dev->base_addr, 0xfff))) {

printk (KERN_ERR "%s: can't get I/O port address 0x%lx\n",dev->name,dev->base_addr);

return (result);

}

request_mem_region (dev->base_addr, 0xff, dev->name);

这里直接就可以使用vSMDK2410_ETH_IO 这个事先定义好的地址进行操作了,后面的几个语句无非的检查和对齐这个地址资源什么的。

而用smdk2410_devices定义的资源,需要在驱动代码中动态向系统申请映射,并返回一个可操作的虚地址才能操作的。例如以下的代码:

host->mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);

if (!host->mem) {

printk(KERN_ERR PFX "failed to get io memory region resouce.\n");

ret = -ENOENT;

goto probe_free_host;

}

host->mem = request_mem_region(host->mem->start,

RESSIZE(host->mem), pdev->name);

if (!host->mem) {

printk(KERN_ERR PFX "failed to request io memory region.\n");

ret = -ENOENT;

goto probe_free_host;

}

host->base = ioremap(host->mem->start, RESSIZE(host->mem));

if (host->base == 0) {

printk(KERN_ERR PFX "failed to ioremap() io memory region.\n");

ret = -EINVAL;

goto probe_free_mem_region;

}

其中platform_get_resource是得到物理的IO或者寄存器的物理地址.

然后用request_mem_region申请一块区域,最后用host->base = ioremap(host->mem->start, RESSIZE(host->mem));将可操作的虚地址返回给驱动,驱动用这个转换后的地址就可以操作硬件的寄存器什么的了...这里面的物理和需地址的转化是怎样的。自己有兴趣自己了解.

posted on 2008-08-20 10:22 puppy 阅读(648) 评论(0)  编辑 收藏 引用

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值