mmap函数详解

来源:互联网 发布:人体工学椅推荐 知乎 编辑:程序博客网 时间:2024/06/14 06:11

       mmap可以把磁盘文件的一部分直接映射到内存,这样文件中的位置直接就有对应的内存地址,对文件的读写可以直接用指针来做而不需要read/write函数。
原型:#include <sys/mman.h>
void *mmap(void *addr, size_t len, int prot, int flag, int filedes, off_t off);
int munmap(void *addr, size_t len);

       参数解释如下:

a,如果addr参数为NULL,内核会自己在进程地址空间中选择合适的地址建立映射。如果addr不是NULL,则给内核一个提示,应该从什么地址开始映射,内核会选择addr之上的某个合适的地址开始映射。建立映射后,真正的映射首地址通过返回值可以得到。

b,len参数是需要映射的那一部分文件的长度。

c,prot参数有四种取值:
PROT_EXEC表示映射的这一段可执行,例如映射共享库
PROT_READ表示映射的这一段可读
PROT_WRITE表示映射的这一段可写
PROT_NONE表示映射的这一段不可访问

d,flag参数有很多种取值,这里只讲两种,
MAP_SHARED多个进程对同一个文件的映射是共享的,一个进程对映射的内存做了修改,另一个进程也会看到这种变化。
MAP_PRIVATE多个进程对同一个文件的映射不是共享的,一个进程对映射的内存做了修改,另一个进程并不会看到这种变化,也不会真的写到文件中去。
e,filedes是代表该打开文件的描述符。
f,off参数是从文件的什么位置开始映射,必须是页大小的整数倍(在32位体系统结构上通常是4K)。
      如果mmap成功则返回映射首地址,如果出错则返回常数MAP_FAILED。当进程终止时,该进程的映射内存会自动解除,也可以调用munmap解除映射。munmap成功返回0,出错返回-1。模型如下:

                       
      实验如下:可以手工建立一个hello.txt文件文件,编辑其内容。

#include <stdlib.h>#include <sys/mman.h>#include <fcntl.h>int main(void){      int *p;      int fd = open("hello.txt", O_RDWR);      if (fd < 0) {                 perror("open hello");                 exit(1);      }      p = mmap(NULL, 6, PROT_WRITE, MAP_SHARED, fd, 0);      if (p == MAP_FAILED) {                 perror("mmap");                 exit(1);      }      close(fd);   //关掉fd不影响映射的内存,除非munmap掉。      p[0] = 0x30313233; //改写其内容      munmap(p, 6);      return 0;}

用gcc编译,运行./a.out,在cat hello.txt,可以看到其内容已发生改变。

         需要注意的是:在hello.txt为空时,运行a.out会提示“段错误 (核心已转储)”。如果非空就不会,比如内容是adffsd,执行a.out后就是3210sd(这个结果跟CPU的endian模式有关,另一个结果是0123sd)。

 

 

 原文地址:http://learn.akae.cn/media/ch28s08.html

原创粉丝点击