从mmaped文件中获取第N个字节

时间:2015-05-04 13:50:10

标签: c memory segmentation-fault mmap coredump

我是C的新人,现在我正在学习mmap。我想从mmaped文件中获取第N个字节,但是我收到此错误Segmentation Fault (core dumped)当我使用gdb测试我的程序时,我发现这行printf("%d\n", (int) data[sk]);出错了然后我print data我得到了

(gdb) print data[sk]
Cannot access memory at address 0xfe5f07d0
(gdb) print data
$1 = 0xfe5f0000 <Address 0xfe5f0000 out of bounds>

我不知道为什么会收到此错误。这是我的代码

int main( int argc, char * argv[] ){
    int sk;
    int d;
    char *data;
    size_t s;
    if(argc == 3){
        sk = atoi(argv[2]);
        d = da_open(argv[1]);
        s = da_fileSize(d);
        data = (char*)da_mmap(d, s);
        printf("File Size: %d\n", (int) s);
        printf("%d\n", (int) data[sk]); // this line is bad. But why?
        close(d);
    }
    return 0;
}

此处还有我的完整代码

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

int da_open(const char *name);
void *da_mmap(int d, size_t size);
size_t da_fileSize();

int da_open(const char *name){
   int dskr;
   dskr = open( name, O_RDWR );
   if( dskr == -1 ){
      perror( name );
      exit(1);
   }
   printf( "dskr1 = %d\n", dskr );
   return dskr;
}

void *da_mmap(int d, size_t size){
     void *a = NULL;
     a = mmap(NULL, size, PROT_WRITE, MAP_SHARED, d, 0);
     if( a == MAP_FAILED ){
          perror( "mmap failed" );
          abort();
     }
     return a;
}

size_t da_fileSize(int d){
    struct stat info;
    if(fstat(d, &info) == -1) {
        perror("fstat failed");
        exit(1);
    }
    return (size_t)info.st_size;
}

int main( int argc, char * argv[] ){
    int sk;
    int d;
    char *data;
    size_t s;
    if(argc == 3){
        sk = atoi(argv[2]);
        d = da_open(argv[1]);
        s = da_fileSize(d);
        data = (char*)da_mmap(d, s);
        printf("File Size: %d\n", (int) s);
        printf("%d\n", (int) data[sk]);
        close(d);
    }
    return 0;
}

2 个答案:

答案 0 :(得分:5)

我猜您还需要在mmap中使用读取权限:PROT_WRITE | PROT_READ

答案 1 :(得分:2)

分段错误是指您尝试访问不属于您的内存空间时。 通常会发生(就像你的情况一样)数组。如果你有一个5个元素长的数组,并且你试图访问第6个,那么你将遇到分段错误,你的程序将会停止。

因此,在行printf("%d\n", (int) data[sk]);中,您的错误可能是sk中的值太大,您应该尝试打印它以进行检查。

你也应该总是检查系统功能的返回(不属于你的功能),例如你的open()或atoi()。如果打开失败并且你没有停止你的计划,那你将度过一个糟糕的时光:)