ioremap和vmalloc一样,都会返回vmalloc区的虚拟地址,但是ioremap并不会分配任何物理内存,ioremap也需要建立新的页表来访问物理页面。

ioremap获取的虚拟地址空间需要使用iounmap来释放。

 

ioremap主要用来映射设备的物理内存到内核的虚拟地址空间,这样,内核就可以通过这些虚拟地址空间访问设备内存。

使用ioremap需要注意几点:

1. 尽量不要直接访问这些虚拟地址,使用专用的io操作函数可以增加移植性,当然有些特殊情况,比如android pmem是对物理内存进行的ioremap,所以可以直接访问内存

2. 有些情况下,可能不想让系统cache发挥作用,这时要使用ioremap_nocache进行映射

 

今天偶然打了下ioremap返回的地址,居然是0x9c820000,而不是0xCxxx xxxx

一再确定打印地址代码没问题后,查了代码也没发现问题出在哪里,最后dmesg找到了答案    

 Virtual kernel memory layout:
    vector  : 0xffff0000 - 0xffff1000   (   4 kB)
    fixmap  : 0xfff00000 - 0xfffe0000   ( 896 kB)
    DMA     : 0xf9e00000 - 0xffe00000   (  96 MB)
    vmalloc : 0x9c800000 - 0xf4000000   (1400 MB)
    lowmem  : 0x80000000 - 0x9c000000   ( 448 MB)
    pkmap   : 0x7fe00000 - 0x80000000   (   2 MB)
    modules : 0x7f000000 - 0x7fe00000   (  14 MB)
      .init : 0x80008000 - 0x8004e000   ( 280 kB)
      .text : 0x8004e000 - 0x8086a000   (8304 kB)

      .data : 0x80892000 - 0x808dfba0   ( 311 kB)

 

看来是把2G~4G定义为kernel空间,而ioremap的地址正好返回的是vmalloc的地址范围内。

一般情况下,内核的地址空间都是3G~4G,所以ioremap返回的地址应该也在这个范围内.
但是我用的开发板返回的是0x9c820000,这个地址位于2G~3G之间,这是因为我的开发板定义的内核地址空间为2G~4G
dmesg打出的vmalloc area地址范围为:
vmalloc : 0x9c800000 - 0xf4000000
ioremap函数要在vmalloc area内分配空间
ioremap函数要在vmalloc area内分配空间