Linux 内存映射函数 mmap()函数笔记

    xiaoxiao2022-07-04  175

    1.所需头文件:<sys/mman.h>

    2.函数原型:void * mmap(void *start, size_t length, int prot , int flags, int fd, off_t offset)

    (1)start:要映射到的内存区域的起始地址,通常都是用NULL(NULL即为0)。NULL表示由内核来指定该内存地址 ;

    (2)length:要映射的内存区域的大小 ;

    (3)prot:期望的内存保护标志,不能与文件的打开模式冲突。是以下的某个值,可以通过or运算合理地组合在一起 :

             PROT_EXEC //页内容可以被执行           PROT_READ //页内容可以被读取           PROT_WRITE //页可以被写入           PROT_NONE //页不可访问 

    (4)flags:指定映射对象的类型,映射选项和映射页是否可以共享。它的值可以是一个或者多个以下位的组合体 :

             MAP_FIXED :使用指定的映射起始地址,如果由start和len参数指定的内存区重叠于现存的映射空间,重叠部分将会被丢                                    弃。如果指定的起始地址不可用,操作将会失败。并且起始地址必须落在页的边界上。           MAP_SHARED :对映射区域的写入数据会复制回文件内, 而且允许其他映射该文件的进程共享。           MAP_PRIVATE :建立一个写入时拷贝的私有映射。内存区域的写入不会影响到原文件。这个标志和以上标志是互斥的,                                         只能使用其中一个。            MAP_DENYWRITE :这个标志被忽略。            MAP_EXECUTABLE :同上            MAP_NORESERVE :不要为这个映射保留交换空间。当交换空间被保留,对映射区修改的可能会得到保证。当交换空                                                   间不被保留,同时内存不足,对映射区的修改会引起段违例信号。            MAP_LOCKED :锁定映射区的页面,从而防止页面被交换出内存。            MAP_GROWSDOWN :用于堆栈,告诉内核VM系统,映射区可以向下扩展。            MAP_ANONYMOUS :匿名映射,映射区不与任何文件关联。            MAP_ANON :MAP_ANONYMOUS的别称,不再被使用。            MAP_FILE :兼容标志,被忽略。            MAP_32BIT :将映射区放在进程地址空间的低2GB,MAP_FIXED指定时会被忽略。当前这个标志只在x86-64平台上得                                      到 支持。            MAP_POPULATE :为文件映射通过预读的方式准备好页表。随后对映射区的访问不会被页违例阻塞。            MAP_NONBLOCK :仅和MAP_POPULATE一起使用时才有意义。不执行预读,只为已存在于内存中的页面建立入 口。

    (5)fd:文件描述符(由open函数返回) 

    (6)offset:表示被映射对象(即文件)从那里开始对映,通常都是用0。 该值应该为大小为PAGE_SIZE的整数倍 ;

    (7)返回值:成功执行时,mmap()返回被映射区的指针,失败时,mmap()返回MAP_FAILED

    errno被设为以下的某个值  EACCES:访问出错  EAGAIN:文件已被锁定,或者太多的内存已被锁定  EBADF:fd不是有效的文件描述词  EINVAL:一个或者多个参数无效  ENFILE:已达到系统对打开文件的限制  ENODEV:指定文件所在的文件系统不支持内存映射  ENOMEM:内存不足,或者进程已超出最大内存映射数量  EPERM:权能不足,操作不允许  ETXTBSY:已写的方式打开文件,同时指定MAP_DENYWRITE标志  SIGSEGV:试着向只读区写入  SIGBUS:试着访问不属于进程的内存区 

    (8)使用例子:

    int fd = open("/dev/mem",O_RDWR|O_SYNC); if(fd < 0) { printf("opent /dev/mem error! \r\n"); close(fd); return; } maped_addr = (unsigned long*)mmap(NULL, 0x1000, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0xF8001000);

    3. int munmap(void *start, size_t length) 

    start:要取消映射的内存区域的起始地址  length:要取消映射的内存区域的大小。  返回说明  成功执行时munmap()返回0。失败时munmap返回-1.

    最新回复(0)