Linux mmap学习及遇到的问题解决

Linux mmap的学习

本来以前mmap是很简单的,看下参数用法就ok,结果遇到了一大堆麻烦,google了很多博客,都写得很浅显,就介绍一下参数然后再给了一个简单得不得了的代码,就完了,所以纠结了我差不多一天时间。

mmap怎么用我就不说了,网上很多,在这里就把我犯的错误经验分享出来,先贴一下我用mmap实现拷贝任意文件的代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <sys/types.h>

int main()
{
	int fdin,fdout, len = 0, pages = 0, offset = 0, lastPage = 0;
	char *src, *dst;
	size_t pagesize;

	struct stat statbuf;

	if((fdin = open("mmmap.c", O_RDWR)) == -1)
	{
		printf("can't open fdin ");
		return 1;
	}

	if((fdout = open("mmmmap1.c",O_RDWR | O_CREAT | O_TRUNC,0644)) == -1)
	{
		if(EEXIST == errno)
		{
			fd = open("mmmmap1.c", O_RDWR);
			if(-1 == fd)
			{
				perror("open");
				exit(-1);
			}
		}
		else
		{
			perror("open");
			exit(-1);
		}	
	}
	
	//lseek(fd, 4096, SEEK_SET);
	//write(fd, " ", 1);

	pagesize = sysconf(_SC_PAGESIZE);
	printf("pagesize = %ld\n", pagesize);

	assert(fstat(fdin,&statbuf) == 0);
	printf("statbuf.st_size = %d\n", statbuf.st_size);

	pages = statbuf.st_size/pagesize;	
	if((statbuf.st_size)%pagesize != 0)
	{
		lastPage = (statbuf.st_size)%pagesize;
		pages += 1;
		printf("lastPage = %d\n", lastPage);
	}
	printf("pages = %d\n", pages);

	//assert(ftruncate(fdout, statbuf.st_size) == 0);
	if (lastPage != 0)
	{
		len += lastPage;
	}
	else
	{
		len += pagesize;
	}

	while(pages--)
	{
		assert(ftruncate(fdout, len) == 0);

		src = (char *)mmap(NULL, pagesize, PROT_READ | PROT_WRITE, MAP_SHARED, fdin, offset);
		if( src == MAP_FAILED )
		{
			printf("mmap error for fdin.\n");
			return 1;
		}

		dst = mmap(0,pagesize,PROT_READ | PROT_WRITE,MAP_SHARED,fdout,offset);
		if( dst == MAP_FAILED )
		{
			printf("mmap error for fdout.\n");
			return 1;
		}

		memcpy(dst,src, pagesize);
		
		offset += pagesize;
		
		if (pages == 1 && lastPage != 0)
		{
			len += lastPage;
		}
		else
		{
			len += pagesize;
		}

		munmap(src, pagesize);
		munmap(dst, pagesize);
	}
	
	close(fdin);
	close(fdout);

	return 0;
}


我遇到的问题:
1.报段错误,打开的读写方式和mmap里的参数不一致会报此错
2.bus error,文件为空,或者mmap了一页而你却要去操作第二页
3.新建的写入文件有时会打开异常,那是因为在mmap的时候往文件里写了不应该写入的字符,所以建议更新文件大小的时候不要用write,用ftruncate

反正记住mmap是一页一页映射的,len和offset参数要是页的倍数就行了...

有什么问题欢迎留言讨论..…^_^

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Linuxmmap是一种内存映射文件的方法,通过将一个文件或者其他对象映射到进程的地址空间,实现磁盘的地址和进程虚拟地址空间的一一对应关系。通过使用mmap系统调用,进程可以将同一个普通文件映射到它们的地址空间中,从而实现共享内存的效果。一旦文件被映射到进程的地址空间中,进程可以像访问普通内存一样对文件进行写操作。 在Linux中,通过调用内核的mmap函数来进行内存映射,其函数原型为int mmap(struct file *filp, struct vm_area_struct *vma)。需要注意的是,这里的mmap是内核态的mmap,并非用户态的mmap。 下面是一个演示映射到物理内存的案例: ```cpp #include <iostream> #include <sys/mman.h> #include <cstring> #include <cerrno> #include <cstdio> using namespace std; static const int SIZE = 4096; int main() { char *str = (char *)mmap(NULL, SIZE, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, 0, 0); // 注意MAP_PRIVATE和MAP_ANONYMOUS // 建立映射 if (str == MAP_FAILED) { printf("%s\n", strerror(errno)); return -2; } strcpy(str, "hello ksy"); puts(str); // 用于取消映射 munmap(str, SIZE); return 0; } ``` 以上代码使用mmap将一段物理内存映射到进程的地址空间中,然后将字符串"hello ksy"写入到这段映射的内存中,并 #### 引用[.reference_title] - *1* *2* *3* [Linuxmmap](https://blog.csdn.net/qq_56999918/article/details/127070280)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值