remap_pfn_range使用详解

remap_pfn_range和io_remap_pfn_range负责为一段物理地址建立新的页表,他们的原型如下

      int remap_pfn_range(struct vm_area_struct *vma, unsigned long virt_addr, unsigned long pfn, unsigned long size, pgprot_t prot);

     int io_remap_pfn_range(struct vm_area_struct *vma, unsigned long virt_addr, unsigned long pfn, unsigned long size, pgprot_t prot);

函数的返回值是0或者是个负的错误碼。

第一个函数是在pfn指向实际的RAM的时候使用,后面一个函数实在phy_addr指向I/O内存时候使用,在实际使用中,除了SPARC外,对每个体系架构这两个函数都是等价的。大多数情况下使用第一个函数,对于有移植性的要求的驱动程序,要使用与特定情形相符的remap_pfn_range函数变种。

该函数的功能是创建页表。其中参数vma是内核根据用户的请求自己填写的,而参数addr表示内存映射开始处的虚拟地址,因此,该函数为addr~addr+size之间的虚拟地址构造页表。

   另外,pfn(Page Fram Number)是虚拟地址应该映射到的物理地址的页面号,实际上就是物理地址右移PAGE_SHIFT位。如果PAGE_SHIFT为4kb,则PAGE_SHIFT为12,因为PAGE_SHIFT等于1<    在驱动程序中,一般能使用remap_pfn_range()映射内存中的保留页(如X86系统中的640KB~1MB区域)和设备I/O内存。因此,如果想把kmalloc()申请的内存映射到用户空间,则可以通过mem_map_reserve()把相应的内存设置为保留后就可以。

一个简单的实现

如果驱动程序要将设备内存线性的映射到用户空间,基本上自需要调用第一个函数即可。

static int simple_remap_mmap(struct file *file, struct vm_area_struct *vma)

{

 if(remap_pfn_range(vma, vma->vm_start, vm->pgoff, vma->vm_end-vma->vm_start, vma->vm_page_port))

{

return -EAGAIN;

}

}




你可能感兴趣的:(linux,Linux)