mmap实现(remap_pfn_range)



mmap设备操作
映射一个设备是指把用户空间的一段地址关联到设备内存上,当程序读写这段用户空间的地址时,它实际上是在访问设备。

这里需要做的两个操作:
1.找到可以用来关联的虚拟地址区间
2.关联
其中找到可以用来关联的虚拟地址区间是由内核完成的,mmap只要关联这个操作。
mmap方法是file_operations结构的成员,在mmap系统调用发出时被调用。在此之前,内核已经完成了很多工作。mmap设备方法所需要做的就是建立虚拟地址到物理地址的页表


void *mmap(void *addr, size_t len, int prot, int flags, int fd, 
           off_t offset);  
addr:指定映射的起始地址,通常设为NULL,由系统指定。
length:映射到内存的文件长度。
prot:映射的保护方式,可以是:
PROT_EXEC:映射区可被执行
PROT_READ:映射区可被读取
PROT_WRITE:映射区可被写入
PROT_NONE:映射区不能存取
Flags:映射区的特性,可以是:
MAP_SHARED:
写入映射区的数据会复制回文件,且允许其他映射该文件的进程共享。
MAP_PRIVATE:
对映射区的写入操作会产生一个映射区的复制(copy_on_write),对此区域所做的修改不会写回原文件。
fd:由open返回的文件描述符,代表要映射的文件。
offset:以文件开始处的偏移量,必须是分页大小的整数倍,通常为0,表示从文件头开始映射。

int munmap(void *start, size_t length)

功能:取消参数start所指向的映射内存,参数length表示欲取消的内存大小。
返回值:解除成功返回0,否则返回-1,错误原因存在于errno中。
虚拟地址区域:vm_area_struct
Linux内核使用结构vm_area_struct(<linux/mm_types.h>)描述虚拟内存区域,其中几个主要成员如下:
unsigned long vm_start 虚拟内存区域起始地址
unsigned long vm_end 虚拟内存区域结束地址
unsigned long vm_flags 该区域的标志
如:VM_IO和VM_RESERVED。VM_IO将该VMA标记为内存映射的IO区域,VM_IO会阻止系统将该区域包含在进程的存放转存(core dump)中,VM_RESERVED标志内存区域不能被换出。


mmap在内核中实现方法为:

 void (*mmap)(struct file *,struct vm_area_struct *)//file_operations里面声明

 而mmap具体实现方法有两种:

1.使用remap_pfn_range一次建立所有页表;
2.使用nopage VMA方法每次建立一个页表;

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


注释:

1.#define io_remap_pfn_range(vma,from,pfn,size,prot) \

                remap_pfn_range(vma, from, pfn, size, prot)  //arch/arm/include/asm/pgtable.h

2.remap_pfn_range(vma, from, pfn, size, prot) //在mm目录下memory.c下面定义

vma->vm_flags |= VM_IO | VM_RESERVED | VM_PFNMAP;   //已经定义过了一次权限

3.struct vm_area_struct {
struct mm_struct * vm_mm;/* The address space we belong to. */
unsigned long vm_start;/* Our start address within vm_mm. */
unsigned long vm_end;/* The first byte after our end address
}

addr:内存映射的开始虚拟地址(即为vma->vm_start)

pfn(Page Fram Number):是虚拟地址映射到物理地址的页面号(4k为一页,pfn即为物理地址<<4k)

size:即为大小(vma->vm_end - vma->end)

prot:权限 一般设置为VM_IO | VM_RESERVED

这样应用层就可以获得重新建立页表内存的虚拟地址了,vm_area_struct祈祷传递作用




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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值