- 论坛徽章:
- 0
|
然后听说ioremap的功能是把io空间映射出来,疑问: ioremap只能操作io空间而不能操作ram空间? 该函数实现原理是? 能否以最简要的句子概括,谢谢!wilbur512 发表于 2010-05-30 23:08 ![]()
看ioremap的代码实现,应该能操作ram空间吧(当开启了CONFIG_HIGHMEM时,能操作大于896M的RAM)
- void __iomem * __ioremap(unsigned long phys_addr, unsigned long size, unsigned long flags)
- {
- void __iomem * addr;
- struct vm_struct * area;
- unsigned long offset, last_addr;
- /* Don't allow wraparound or zero size */
- last_addr = phys_addr + size - 1;
- if (!size || last_addr < phys_addr)
- return NULL;
- /*
- * Don't remap the low PCI/ISA area, it's always mapped..
- */
- if (phys_addr >= ISA_START_ADDRESS && last_addr < ISA_END_ADDRESS)
- return (void __iomem *) phys_to_virt(phys_addr);
- /*
- * Don't allow anybody to remap normal RAM that we're using..
- */
- if (phys_addr <= virt_to_phys(high_memory - 1)) {
- char *t_addr, *t_end;
- struct page *page;
- t_addr = __va(phys_addr);
- t_end = t_addr + (size - 1);
-
- for(page = virt_to_page(t_addr); page <= virt_to_page(t_end); page++)
- if(!PageReserved(page))
- return NULL;
- }
- /*
- * Mappings have to be page-aligned
- */
- offset = phys_addr & ~PAGE_MASK;
- phys_addr &= PAGE_MASK;
- size = PAGE_ALIGN(last_addr+1) - phys_addr;
- /*
- * Ok, go for it..
- */
- area = get_vm_area(size, VM_IOREMAP | (flags << 20));
- if (!area)
- return NULL;
- area->phys_addr = phys_addr;
- addr = (void __iomem *) area->addr;
- if (ioremap_page_range((unsigned long) addr,
- (unsigned long) addr + size, phys_addr, flags)) {
- vunmap((void __force *) addr);
- return NULL;
- }
- return (void __iomem *) (offset + (char __iomem *)addr);
- }
复制代码 if判断中:
- if (phys_addr <= virt_to_phys(high_memory - 1))
复制代码 看high_memory的赋值:
- mem_init()
- #ifdef CONFIG_HIGHMEM
- high_memory = (void *) __va(highstart_pfn * PAGE_SIZE - 1) + 1;
- #else
- high_memory = (void *) __va(max_low_pfn * PAGE_SIZE - 1) + 1;
- #endif
复制代码 所以当物理内存大于896M且内核开启了CONFIG_HIGHMEM,ioremap传入的phys_addr参数可以为高端的ram地址(当然大部分IO空间本身就处于高端RAM地址范围中) |
|