在文件上使用mmap

sam*_*moz 4 c mmap file

我试图允许两个不同的进程通过使用内存映射相同的文件进行通信.但是,我遇到了一些问题.我有一种感觉,这与我使用open()调用并将我的文件描述符传递给mmap的方式有关.

这是我的代码,你能看到它有什么问题吗?

对象1的代码:

 16     FILE* temp = fopen(theSharedFileName, "w");
 17     fseek(temp, fileSize-1, SEEK_SET);
 18     fprintf(temp, "0"); // make the file a certain size
 19     fseek(temp, 0, SEEK_CUR);
 20 
 21     int sharedFileName = fileno(temp);
 ...
 31     sharedArea = (MyStruct*)mmap(0, fileSize,
 32         PROT_READ | PROT_WRITE | PROT_EXEC, MAP_SHARED, sharedFileName, 0);
Run Code Online (Sandbox Code Playgroud)

我使用"w"文件模式,因为对象1只会被制作一次,我希望它重置以前存在的任何数据.

对象2的代码:

 130     FILE* tempFile = fopen(sharedFileName, "a");
 131     int theFile = fileno(tempFile);
 ...
 135     sharedArea = (MyStruct*)mmap(NULL, fileSize,
 136         PROT_READ | PROT_WRITE | PROT_EXEC, MAP_SHARED, theFile, 0);
Run Code Online (Sandbox Code Playgroud)

Jul*_*ano 24

一些问题:

  1. 避免混合高级I/O(fopen(),fseek())和一些低级操作,如mmap().虽然您可以使用fileno()获取低级文件描述符,但这就像走最长的路径到达同一个地方.此外,仅使用mmap()会破坏BSD和POSIX之外的兼容性,因此您无需使用标准CI/O函数.只需直接使用open()和lseek().
  2. 在内存映射的同一文件上使用流格式化的I/O(fprintf())没有意义.内存映射文件时,您隐式告诉系统您将其用作随机访问(直接索引)数据.fprintf()用于流输出,通常用于顺序访问.事实上,尽管可能,但在同一个描述符中看到fprintf()和fseek()是不常见的(这甚至不是可移植的,但由于之前的项目,我不考虑可移植性).
  3. 保护必须与打开文件保护相匹配.由于您将"w"传递给fopen()和PROT_READ | PROT_WRITE | PROT_EXECmmap(),因此您违反了此限制.这也突出了为什么你不应该将高级I/O与内存映射混合:你如何保证fopen(...,"w")用正确的标志打开文件?这应该是C库的"实现细节".如果要对具有读写权限的文件进行内存映射,则应使用低级别open(theSharedFileName, O_RDWR)来打开该文件.
  4. 不要使用PROT_WRITEPROT_EXEC在一起.它不便携,存在安全风险.阅读W ^ X可执行空间保护.