mmap 函数

头文件:#include <unistd.h>    #include <sys/mman.h>

定义函数:void *mmap(void *start, size_t length, int prot, int flags, int fd, off_t offsize);

函数说明:mmap()用来将某个文件内容映射到内存中,对该内存区域的存取即是直接对该文件内容的读写。

参数说明:

参数 说明
start 指向欲对应的内存起始地址,通常设为NULL,代表让系统自动选定地址,对应成功后该地址会返回。
length 代表将文件中多大的部分对应到内存。
prot  代表映射区域的保护方式,有下列组合:

  • PROT_EXEC  映射区域可被执行;
  • PROT_READ  映射区域可被读取;
  • PROT_WRITE  映射区域可被写入;
  • PROT_NONE  映射区域不能存取。
flags 会影响映射区域的各种特性:

  • MAP_FIXED  如果参数 start 所指的地址无法成功建立映射时,则放弃映射,不对地址做修正。通常不鼓励用此旗标。
  • MAP_SHARED  对应射区域的写入数据会复制回文件内,而且允许其他映射该文件的进程共享。
  • MAP_PRIVATE  对应射区域的写入操作会产生一个映射文件的复制,即私人的"写入时复制" (copy on write)对此区域作的任何修改都不会写回原来的文件内容。
  • MAP_ANONYMOUS  建立匿名映射,此时会忽略参数fd,不涉及文件,而且映射区域无法和其他进程共享。
  • MAP_DENYWRITE  只允许对应射区域的写入操作,其他对文件直接写入的操作将会被拒绝。
  • MAP_LOCKED  将映射区域锁定住,这表示该区域不会被置换(swap)。

在调用mmap()时必须要指定MAP_SHARED 或MAP_PRIVATE。

fd open()返回的文件描述词,代表欲映射到内存的文件。
offset 文件映射的偏移量,通常设置为0,代表从文件最前方开始对应,offset必须是分页大小的整数倍。

返回值:若映射成功则返回映射区的内存起始地址,否则返回MAP_FAILED(-1),错误原因存于errno 中。

错误代码:

    • EBADF  参数fd 不是有效的文件描述词。
    • EACCES  存取权限有误。如果是MAP_PRIVATE 情况下文件必须可读,使用MAP_SHARED 则要有PROT_WRITE 以及该文件要能写入。
    • EINVAL  参数start、length 或offset 有一个不合法。
    • EAGAIN  文件被锁住,或是有太多内存被锁住。
    • ENOMEM  内存不足。

mmap将一个文件或者其它对象映射进内存。文件被映射到多个页上,如果文件的大小不是所有页的大小之和,最后一个页不被使用的空间将会清零。munmap执行相反的操作,删除特定地址区域的对象映射。

当使用mmap映射文件到进程后,就可以直接操作这段虚拟地址进行文件的读写等操作,不必再调用read,write等系统调用.但需注意,直接对该段内存写时不会写入超过当前文件大小的内容.

采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要任何数据的拷贝。对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只拷贝两次数据:一次从输入文件到共享内存区,另一次从共享内存区到输出文件。实际上,进程之间在共享内存时,并不总是读写少量数据后就解除映射,有新的通信时,再重新建立共享内存区域。而是保持共享区域,直到通信完毕为止,这样,数据内容一直保存在共享内存中,并没有写回文件。共享内存中的内容往往是在解除映射时才写回文件的。因此,采用共享内存的通信方式效率是非常高的。

基于文件的映射,在mmap和munmap执行过程的任何时刻,被映射文件的st_atime可能被更新。如果st_atime字段在前述的情况下没有得到更新,首次对映射区的第一个页索引时会更新该字段的值。用PROT_WRITE 和 MAP_SHARED标志建立起来的文件映射,其st_ctime 和 st_mtime在对映射区写入之后,但在msync()通过MS_SYNC 和 MS_ASYNC两个标志调用之前会被更新。

时间: 2024-10-27 07:10:12

mmap 函数的相关文章

mmap函数使用

来自:http://blog.csdn.net/scorpio16/article/details/2059623 UNIX网络编程第二卷进程间通信对mmap函数进行了说明.该函数主要用途有三个:1.将一个普通文件映射到内存中,通常在需要对文件进行频繁读写时使用,这样用内存读写取代I/O读写,以获得较高的性能:2.将特殊文件进行匿名内存映射,可以为关联进程提供共享内存空间:3.为无关联的进程提供共享内存空间,一般也是将一个普通文件映射到内存中. 函数:void *mmap(void *start

共享内存:mmap函数实现

内存映射的应用: 以页面为单位,将一个普通文件映射到内存中,通常在需要对文件进行频繁读写时使用,这样用内存读写取代I/O读写,以获得较高的性能; 将特殊文件进行匿名内存映射,可以为关联进程提供共享内存空间; 为无关联的进程提供共享内存空间,一般也是将一个普通文件映射到内存中. 相关API #include <sys/mman.h> void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset);

mmap函数实现共享内存

mmap将一个文件或者其他对象映射进内存.mmap也可以实现共享内存.mmap函数调用使得进程之间通过映射同一个文件实现共享内存.文件被映射到进程地址空间后,进程可以像读写内存一样对文件进行操作. 函数原型:void* mmap(void* addr,size_t length,int prot,int flags,int fd,off_t offset); addr:映射区的开始地址,设置为0时表示系统决定映射区的起始地址 length:映射区的长度.长度单位为字节 prot:期望的内存保护标

3Linux内存映射,mmap()函数

 1mmap()依赖的头文件 #include <sys/mman.h> 2函数声明: void *mmap(void *addr, size_t length, intprot, int flags, intfd, off_t offset); int munmap(void *addr, size_t length); 函数说明: mmap可以把磁盘文件的一部分直接映射到内存,这样文件中的位置直接就有对应的内存 地址,对文件的读写可以直接用指针来做而不需要read/write函数. 3

Linux mmap函数简介

一.简介 Linux提供了内存映射函数mmap, 它把文件内容映射到一段内存上(准确说是虚拟内存上), 通过对这段内存的读取和修改, 实现对文件的读取和修改, 先来看一下mmap的函数声明: 头文件: <unistd.h> <sys/mman.h> 原型: void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offsize); 返回值: 成功则返回映射区起始地址, 失败则返回MAP_FAI

&lt;轉&gt;APUE:mmap函数

起初 看过一遍内存映射I/O,意思大概是懂了,就是直接操作文件再而直接通过缓冲区来操作,减少一些read.write调用所花费的时间.加上文中给出一个copy的例子,意思也好理解的.不过困扰的来了,我琢磨着在映射两个文件后,再使用memcpy内存复制函数将文件写入完全可以,但是这是两个文件的操作,我想一个文件自己对自己的修改呢?只有一个文件的时候,该怎么做呢?于是自己就开始构建一个程序来验证一下.正好也顺手将书后的习题14.11也解决掉:) 猜想 首先设想创建一个空文件,先写入一点数据,接着映射

mmap函数

1.mmap系统调用 通过man手册可以看出: 1 void *mmap(void *addr, size_t length, int prot, int flags, 2 int fd, off_t offset); 功能:负责把文件内容映射到进程的虚拟地址空间,通过对这段内存的读取和修改来实现对文件的读取和修改,而不需要再调用read和write: 参数: addr:映射的起始地址,设为NULL由系统指定:len:映射到内存的文件长度:prot:期望的内存保护标志,不能与文件的打开模式冲突.

Linux网络编程--文件空间映射mmap函数

/*使用mmap对文件进行操作*/ #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <sys/mman.h>/*mmap*/ #include <string.h>/*memset warning*/ #include <stdio.h> #define FILELENGTH 80 int main(void) { int fd = -1

mmap存储映射IO

mmap,munmap-- #include <sys/mman.h> void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset); int munmap(void *addr, size_t length); int mprotect(void *addr, size_t len, int prot); int msync(void *addr, size_t length, int fla