puppy居
puppy居士
posts - 41,comments - 27,trackbacks - 0
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 阅读(721) 评论(0)  编辑 收藏 引用
只有注册用户登录后才能发表评论。