轉APUE:mmap函数
起初
看过一遍内存映射I/O,意思大概是懂了,就是直接操作文件再而直接通过缓冲区来操作,减少一些read、write调用所花费的时间。加上文中给出一个copy的例子,意思也好理解的。
不过困扰的来了,我琢磨着在映射两个文件后,再使用memcpy内存复制函数将文件写入完全可以,但是这是两个文件的操作,我想一个文件自己对自己的修改呢?只有一个文件的时候,该怎么做呢?于是自己就开始构建一个程序来验证一下。正好也顺手将书后的习题14.11也解决掉:)
猜想
首先设想创建一个空文件,先写入一点数据,接着映射到缓冲区区,接着再写一点数据进去。简单说就是open==> write ==> mmap ==> write 但是一想这边根本没有涉及到使用缓冲区,还是建立在文件的操作上,失败!
继续想一个思路,书上有一句话 此函数的返回地址是该映射区的返回地址 ,意思就是说mmap的返回值可以拿过来用,我们可以将它的内容输出,我个人猜想这个内容应该就是文件的内容。那么这样我不就可以对这个返回值进行相应的操作,这不就是在对映射的缓存区进行操作吗?
鉴于书中提到一个函数msync是将映射区的修改同步到文件中,那么对文件的修改也就能体现出来了。一气呵成的想法,最后还有一个释放映射区,这是一个基本的清理操作。大体上应该可以设计了。
思路整理下,首先是创建文件,写入一点数据,然后映射过去,读映射区的内容(返回值) 修改映射区,同步写入文件,清理工作。这个思路应该差不多了。
实现
编码之前的一个问题,mmap的返回值是void *, 而我们要展示的确是字符串之类的,于是想到得强制转换下这个返回值。这一步解决后,开始编码
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<fcntl.h> #include<unistd.h> #include<sys/mman.h> #include<errno.h>static void error(const char *msg) {fprintf(stderr,"%s:%s\n",msg,strerror(errno));exit(1); }int main(void) {int fd;char *dst;if((fd = open("mapfile",O_RDWR|O_CREAT|O_TRUNC,0666)) < 0)error("open error");if(write(fd,"abc",3) != 3)error("wirte error");/*let this file map to memory area*/if((dst =(char *)mmap(0,64,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0)) == MAP_FAILED)error("mmap error");close(fd);printf("%s\n",dst);dst[2]='d';printf("%s\n",dst);if(msync((void *)dst,64,MS_SYNC) < 0)error("msync error");if(munmap((void *)dst,64) < 0)error("munmap error");return 0;}
初期的一个编码大概是这样的,运行后的结果还是比较不错的.
细究下这个代码,我尝试的是去改已经写入的内容,这个确实有一个修改的意思了,看来对返回值的猜想是正确的,对返回值的修改就是对这个映射区的修改。同时,我也测试了将文件描述符关闭后,缓存区是不受影响的,照样可以操作,并且是能够同步到文件中的。这点确实很关键。
可我并不觉着这样就可以了,我的目的是继续往里面写数据,就像一开始想的那样,在映射后我还需要将其他数据写入, 看看这个是否是和write函数一样可以做到这一点。
修改代码
我们一开始只是写入了3个,而给的缓存区是64的,足够再写入的。但是运行的结果
和上面的一样,没有任何的改变。。。 按照道理来时应该会有一个x出现的。
看书,发现书上的copy函数的例子中,有一个偏移的设置,书中也有个说法映射文件的启示偏移量受系统虚存页的限制,如果映射区的长度不是页的整数倍,任何的变动,都不会再文件中反映出来,不能用mmap将数据添加到文件中,需要加长该文件 也就是说我这边需要一个偏移,看书上的copy的例子,上面是从文件开始偏移了inputfile的长度。也就是说现在这个文件从开始到偏移的位置有一个空洞,那么再映射过去后,显示的量应该就是能满足了,这样就可以将所更改的写入到文件中。回到我刚才的那个,虽然映射区是给64,但是文件的长度不足,这和write不一样,write可以直接接着后面写。所以必须文件的大小要满足才行。
继续修改代码
此刻运行的结果就符合我的猜想了,下面是结果的运行
可以看到,cat输出了我刚刚插入的一个'x',继而看文件的真实情况,也完全符合我的猜想,在第十一位放入了一个x,至于为什么输出的还是三个,因为字符串的结尾有一个'\0'则'x'没有连在一起。
其实还有一个细节,再形成文件空洞的时候,不仅仅是要偏移位置,还需要写入一个字符,来标识这个文件真实长度,否则结果和不使用偏移一样文件的长度还是原来的大小,并没有被扩充。
有一个不错的方法使用ftruncate,直接设置文件的大小,省却使用lseek和write的繁杂。
总结
mmap再IPC中提及到,但是在APUE中涉及了这个,这让我想起了再做摄像头驱动的时候,使用过这个函数,但是一点概念一点都没有,完全跟着老师后一个字一个字的敲,现在大概能理解使用存储映射的过程,至于什么二级缓存,一级缓存,感觉需要到实际项目中才能感受到。
(全文完)
转载于:https://www.cnblogs.com/sexybear/p/4510456.html
轉APUE:mmap函数相关推荐
- 共享内存简介和mmap 函数
一.共享内存简介 共享内存区是最快的IPC形式,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据. 即每个进程地址空间都有一个共享存储器的映射区,当这块区 ...
- 【Android 逆向】Android 进程注入工具开发 ( 注入代码分析 | 远程调用 目标进程中 libc.so 动态库中的 mmap 函数 三 | 等待远程函数执行完毕 | 寄存器获取返回值 )
文章目录 前言 一.等待远程进程 mmap 函数执行完毕 二.从寄存器中获取进程返回值 三.博客资源 前言 前置博客 : [Android 逆向]Android 进程注入工具开发 ( 注入代码分析 | ...
- 【Android 逆向】Android 进程注入工具开发 ( 注入代码分析 | 远程调用 目标进程中 libc.so 动态库中的 mmap 函数 二 | 准备参数 | 远程调用 mmap 函数 )
文章目录 一.准备 mmap 函数的参数 二.mmap 函数远程调用 一.准备 mmap 函数的参数 上一篇博客 [Android 逆向]Android 进程注入工具开发 ( 注入代码分析 | 远程调 ...
- 【Android 逆向】Android 进程注入工具开发 ( 注入代码分析 | 远程调用 目标进程中 libc.so 动态库中的 mmap 函数 一 | mmap 函数简介 )
文章目录 一.mmap 简介 二.mmap 函数作用 一.mmap 简介 mmap 函数的作用是 将 文件 映射到 内存中 , 映射的单位必须是 PAGE_SIZE ; mmap 函数引入头文件 : ...
- 【Android 逆向】Android 进程注入工具开发 ( 注入代码分析 | 获取 远程 目标进程 中的 /system/lib/libc.so 动态库中的 mmap 函数地址 )
文章目录 一.获取 远程 目标进程 中的 /system/lib/libc.so 动态库中的 mmap 函数地址 二.从 /proc/pid/maps 文件中获取 指定 进程 中的 /system/l ...
- Linux系统编程---5(共享存储映射,存储映射I/O,mmap函数,父子进程间通信,匿名映射)
共享存储映射 文件进程间通信 使用文件也可以完成 IPC,理论依据是,fork 后,父子进程共享文件描述符.也就共享打开的文件. 编程:父子进程共享打开的文件.借助文件进行进程间通信. 测试代码 /* ...
- 共享内存:mmap函数实现
内存映射的应用: 以页面为单位,将一个普通文件映射到内存中,通常在需要对文件进行频繁读写时使用,这样用内存读写取代I/O读写,以获得较高的性能; 将特殊文件进行匿名内存映射,可以为关联进程提供共享内存 ...
- mmap函数_分析由 mmap 导致的内存泄漏
背景 一个程序链接 TCMalloc ,同时调用 mmap / munmap 管理一部分较大的内存 通过 TCMalloc 的统计信息,判断内存泄漏不是由 new / malloc 等常规接口导致的 ...
- mmap函数用法及示例程序
UNIX网络编程第二卷进程间通信对mmap函数进行了说明.该函数主要用途有三个: 1.将一个普通文件映射到内存中,通常在需要对文件进行频繁读写时使用,这样用内存读写取代I/O读写,以获得较高的性能: ...
- linux 内存映射-ioremap和mmap函数
最近开始学习Linux驱动程序,将内存映射和ioremap,mmap函数相关资料进行了整理 一,内存映射 对于提供了MMU(存储管理器,辅助操作系统进行内存管理,提供虚实地址转换等硬件支持)的处理器 ...
最新文章
- JVM——虚拟机的简介【摘录】
- 假设检验 Hypothesis testing
- api与密度转换公式_API重度和密度换算公式
- LVM--逻辑卷管理
- 【牛客 - 157D】插排树(dfs,树形dp)
- Redis线上救命丸:01---误操作AOF、RDB恢复数据
- 第 3-1 课:集合详解(上) + 面试题
- 实战:自定义简易版SpringBoot
- Hibernate api 之常见的类(配置类,会话工厂类,会话类)
- Tensorflow中的tf.keras库
- CVTE 2017 秋季校招一面(C++ 后台)
- atlas 力矩计算_Atlas Copco基本拧紧技术
- 消息队列RabbitMQ的使用
- html5文本与段落简介,美化html段落文本 Ⅰ
- Flutter之Dialog使用和踩坑
- Linux运维工程师面试题库,Linux运维工程师面试题库
- 27条好赚钱副业的途径:自己也能不用上班月入2万+!
- 软件安装 | SolidWorks2016软件安装,SW2010-2016.Activator.GUI.SSQ激活闪退解决办法
- 魔兽三界血歌鸿蒙武器怎么合成,魔兽RPG剑域N3剧情攻略 特殊副本武器合成指南[多图]...
- 场景图生成论文阅读笔记 之 PCPL