存储映射I/O
存储映射I/O(Memory-mapped I/O)使一个磁盘文件与存储空间中的一个缓冲区相映射。于是当从缓冲区取数据,就相当于读文件中的相应字节。与次类似,将数据存入缓冲区,则相应的字节就自动地写入文件。这样就可以在不使用read和write的情况下执行I/O。
一、存储映射I/O函数
#include <sys/mman.h>
void *mmap(void *addr, size_t len, int prot, int flag, int filedes, off_t off );
Returns: starting address of mapped region if OK, MAP_FAILED on error
addr:指定映射存储区的起始地址
len:映射存储区的长度
filedes:指定要被映射的文件描述符
prot:说明对映射存储区的保护要求
flag:影响存储区的多种属性
off:需要映射文件的偏移量
在调用fork之后,子进程继承存储映射区,但是由于同样的理由,调用exec后的新程序则不继承此存储映射区。
二、mprotect函数
功能:修改现存映射存储区权限
#include <sys/mman.h>
int mprotect(void *addr, size_t len, int prot);
Returns: 0 if OK, -1 on error
三、msync函数
功能:将页面冲洗到被映射的文件中
#include <sys/mman.h>
int msync(void *addr, size_t len, int flags);
Returns: 0 if OK, -1 on error
四、munmap函数
功能:解除存储映射
#include <sys/mman.h>
int munmap(caddr_t addr, size_t len);
Returns: 0 if OK, -1 on error
进程终止时,或调用了munmap之后,存储映射区就被自动解除映射。但是关闭文件描述符并不解除映射。
五、示例:存储映射I/O复制一个文件
#include "apue.h"
#include <fcntl.h>
#include <sys/mman.h>
int
main(int argc, char *argv[])
{
int fdin, fdout;
void *src, *dst;
struct stat statbuf;
if (argc != 3)
err_quit("usage: %s <fromfile> <tofile>", argv[0]);
if ((fdin = open(argv[1], O_RDONLY)) < 0)
err_sys("can't open %s for reading", argv[1]);
if ((fdout = open(argv[2], O_RDWR | O_CREAT | O_TRUNC,
FILE_MODE)) < 0)
err_sys("can't creat %s for writing", argv[2]);
if (fstat(fdin, &statbuf) < 0) /* need size of input file */
err_sys("fstat error");
/* set size of output file */
if (lseek(fdout, statbuf.st_size - 1, SEEK_SET) == -1)
err_sys("lseek error");
if (write(fdout, "", 1) != 1)
err_sys("write error");
if ((src = mmap(0, statbuf.st_size, PROT_READ, MAP_SHARED,
fdin, 0)) == MAP_FAILED)
err_sys("mmap error for input");
if ((dst = mmap(0, statbuf.st_size, PROT_READ | PROT_WRITE,
MAP_SHARED, fdout, 0)) == MAP_FAILED)
err_sys("mmap error for output");
memcpy(dst, src, statbuf.st_size); /* does the file copy */
exit(0);
}
将一个普通文件复制到另一个普通文件时,存储映射I/O比较快。但是有一些限制,例如不能用其在某些设备(例如网络设备或者终端设备)之间进行复制,并且在对被复制的文件进行映射后,也要注意该文件的长度是否改变。