13LinuxC进程间通信之mmap的注意事项总结(重要)

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;
}
  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值