我是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;
}
答案 0 :(得分:5)
我猜您还需要在mmap中使用读取权限:PROT_WRITE | PROT_READ
答案 1 :(得分:2)
分段错误是指您尝试访问不属于您的内存空间时。 通常会发生(就像你的情况一样)数组。如果你有一个5个元素长的数组,并且你试图访问第6个,那么你将遇到分段错误,你的程序将会停止。
因此,在行printf("%d\n", (int) data[sk]);
中,您的错误可能是sk中的值太大,您应该尝试打印它以进行检查。
你也应该总是检查系统功能的返回(不属于你的功能),例如你的open()或atoi()。如果打开失败并且你没有停止你的计划,那你将度过一个糟糕的时光:)