Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1174804
  • 博文数量: 173
  • 博客积分: 4048
  • 博客等级:
  • 技术积分: 2679
  • 用 户 组: 普通用户
  • 注册时间: 2010-09-12 18:53
文章分类

全部博文(173)

文章存档

2018年(1)

2016年(1)

2013年(1)

2012年(118)

2011年(52)

分类: LINUX

2012-03-31 09:41:58

最近用ioremap 和 phys_to_virt 做物理地址于虚拟地址的转换发现
addr = (unsigned int volatile *)ioremap(0x56000088,12);
printk(KERN_ALERT"%x\n",addr);
addr = (unsigned int volatile *) phys_to_virt(0x56000088);
printk(KERN_ALERT"%x\n",addr);
两个很函数返回的addr值不一样看了一下(arm2410)内核的源码发现

在内核中phys_to_virt只是给地址减去一个固定的偏移 :


点击(此处)折叠或打开

  1. #ifndef __virt_to_phys
  2. #define __virt_to_phys(x) ((x) - PAGE_OFFSET + PHYS_OFFSET)
  3. #define __phys_to_virt(x) ((x) - PHYS_OFFSET + PAGE_OFFSET)
  4. #endif

因此phys_to_virt只是提供在内核空间内,默认的情况下,物理地址的转换。如果页表有所改变,phys_to_virt并不会得到正确的结果。如果要用phys_to_virt得到的地址进行IO,则也要注意IO地址的偏移是否和内核一致。

而ioremap()的原则就是内核会根据指定的物理地址新建映射页表,物理地址和虚拟地址的关系就由这些页表来搭建!:

点击(此处)折叠或打开

  1. void __iomem *
  2. __ioremap(unsigned long phys_addr, size_t size, unsigned long flags,
  3. unsigned long align)
  4. {
  5. void * addr;
  6. struct vm_struct * area;
  7. unsigned long offset, last_addr;
  8. /* Don't allow wraparound or zero size */
  9. last_addr = phys_addr + size - 1;
  10. if (!size || last_addr < phys_addr)
  11. return NULL;
  12. /*
  13. * Mappings have to be page-aligned
  14. */
  15. offset = phys_addr & ~PAGE_MASK;
  16. phys_addr &= PAGE_MASK;
  17. size = PAGE_ALIGN(last_addr + 1) - phys_addr;
  18. /*
  19. * Ok, go for it..
  20. */
  21. area = get_vm_area(size, VM_IOREMAP);
  22. if (!area)
  23. return NULL;
  24. addr = area->addr;
  25. if (remap_area_pages((unsigned long) addr, phys_addr, size, flags)) {
  26. vfree(addr);
  27. return NULL;
  28. }
  29. return (void __iomem *) (offset + (char *)addr);
而且在一般平台下,使用ioremap得到的虚拟地址进行IO时,会bypass cache,直接进行IO。
还可以使用ioremap_nocache确保bypass cache
阅读(2493) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~