两次调用mmap时的SIGSEGV

时间:2014-12-02 19:08:09

标签: c linux gcc mmap sigsegv

我试图在Ubuntu上运行以下程序,但它因分段错误而崩溃。

我尝试做的是两次调用mmap:

p1 = mmap(null, size: 16 * 4k, offset: 0);
p2 = mmap(p1+(16*4K), 136 * 4k , offset: 16 * 4k);

基本上,尝试创建两个连续的内存区域,镜像文件中的两个连续区域。 如果第二个mmap失败可以,但我想了解它导致分段错误的原因。

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

int main()
{
    int fd, ret;
    void* p1;
    void* p2;
    unlink ("test.file");// don't care if it doesn't exists

    fd = open("test.file", O_RDWR | O_CREAT | O_SYNC, ALLPERMS);

    if(fd == -1)
        return errno;

    ret = ftruncate(fd, 4096*16);
    if(ret != 0)
        return errno;

    p1 = mmap(NULL, 4096*16, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
    if(p1 == (void*)-1)
        return errno;

    ret = ftruncate(fd, 4096*150);
    if(ret != 0)
        return errno;

    // dies here!
    p2 = mmap(p1 + (4096*16), 4096*(150-16), PROT_READ | PROT_WRITE, MAP_SHARED |MAP_FIXED, fd, 4096*16);
    if(p2 == (void*)-1)
        return errno;


    return 0;
}

1 个答案:

答案 0 :(得分:3)

它可以在映射之前和之后分配防护页面以防止映射溢出,它们应该在读写时发生段错误。另一个选择是你在堆栈下面的防护页面。在第一个pmap之后检查mmap输出以确定。

请尝试使用mremap