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<<PAGE_SHIFT。最一个参数prot是新页所要求的保护属性。
在驱动程序中,一般能使用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;

}

}




  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值