1 mmap的注意事项1
1 可以open的时候O_CREAT一个新文件来创建映射区吗?
可以,但是:
- 1)当创建映射区的文件为0时,mmap的len为非0,出总线错误SIGBUS。
- 2)当创建映射区的文件为0时,mmap的len也为0,出无效参数错误Invaild xxx。
2 如果open时O_RDONLY, mmap时PROT参数指定PROT_READ|PROT_WRITE会怎样?
- 1)结果是出无效参数错误Invaild xxx,因为open的fd根本没有写权限,mmap无法使用。所以这里应该mmap的权限要小于等于open的权限。但是即使满足小于等于也可能报错,例如下面的同是写权限,会报权限不允许。
- 2)并且当文件大小为0时,需要扩展大小,ftruncate扩展时需要写权限,否则同样出错。
- 3)当文件大小不为0时,因为不调用ftruncate,所以open时可以只传只读权限。并且当mmap的PROT也只传只读,mmap是可以正常执行,但是写数据时报段错误,因为映射区被创建后是只读,使用strcpy写的时候是非法的,故报段错误。
- 4)当open权限只写,mmap也只写,报权限不允许,说明mmap必须要有读权限。
3 文件描述符先关闭,对mmap映射有没有影响?
- 1)没有影响,并且更安全。后续访问只需要使用指针即可。
4 如果文件偏移量为1000会怎样?
- 1)不行,出无效参数错误Invaild xxx。因为偏移量必须是4K的整数倍,更深点就是因为mmu的工作,因为mmu映射时的单位必须是4K的整数倍。
5 对mem越界操作会怎样?
- 1)当越界的长度比较少时,可能不报错,但是越界的内存稍大时,就会出现段错误。例如越界100个字节不出错,4096*4个字节就会报段错误。所以必须保证不越界。
6 如果mem++,munmap可否成功?
- 1)不能,因为对mem++后,导致映射区的首地址改变,munmap是无法回收的,类似free,传错地址也是回收失败。若一样要操作该指针,可以定义第三方变量先保存首地址。
7 mmap什么情况下会调用失败?
- 1)所有参数中,某个参数传错都有可能出错。
8 如果不检测mmap的返回值,会怎样?
- 1)会导致后面的程序有可能因mmap出错而崩溃。
9 当mmap的标志位为MAP_PRIVATE时会怎么样?
- 1)MAP_PRIVATE: 映射区所做的修改不会反映到物理设备,只会在内存中改变。可以通过0d -tcx 文件名,查看空洞,结果为一串0。
10 特殊情况
- 1)上面第2点我们总结过说,mmap的权限应该小于open的权限,但是不一定,也有特殊,当open以只读打开,mmap为读写并且标志位为私有,是可以成功的。因为当标志位为私有,在内存的读写是不会影响到磁盘文件,所以不需要文件的写权限,只需要内存的写权限即可,故需要注意。
11 总结mmap的保险用法
- 1)open时给读写权限,mmap时也给读写权限,并且标志位最好是共享。
- 2)参数2的len一定需要注意是文件有效的大小,即len不能大于文件大小。
2 测试代码
大家可以通过这些测试代码来测试上面的结论。
#include <stdio.h>
#include <string.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/mman.h>
void sys_err(char *str)
{
perror(str);
exit(1);
}
int main(void)
{
char *mem = NULL;
int len = 0;
int fd = open("hello244", O_RDWR|O_CREAT|O_TRUNC, 0644);
if (fd < 0){
sys_err("open error");
}
/*
注意:由于mmap的参2是开辟文件大小的映射区,完美open时是一个新文件,
此时大小为0,所以必须保证文件大小充足才能成功调用mmap开辟映射区。
这就是下面两个方法的原因。
方法1
lseek(fd, 19, SEEK_END); //将文件下标移至19个字节
write(fd, "\0", 1); //往文件写0,补充结尾。19+1=20个字节
len = lseek(fd, 0, SEEK_END); //利用lseek的返回值获取当前文件下标即文件大小
printf("The length of file = %d\n", len);
*/
//方法2
ftruncate(fd, 20);//该函数和lseek+write作用一样,但是需要写权限
len = lseek(fd, 0, SEEK_END);
mem = mmap(NULL, len, PROT_WRITE, MAP_SHARED, fd, 0);
if (mem == MAP_FAILED){
sys_err("mmap err: ");
}
//此时可以直接关闭描述符,因为mem已经代替其作用
close(fd);
//使用mem(内存)对磁盘文件进行读写操作。
strcpy(mem, "hello mmap");//写
printf("%s\n", mem);//读
if (munmap(mem, mem) < 0){
sys_err("munmap");
}
return 0;
}