【问题标题】:Linux mmap() errorLinux mmap() 错误
【发布时间】:2009-07-23 14:06:57
【问题描述】:

我有一个内存映射文件,我希望从中解析缓冲区的内容。 mmap() 返回成功,我可以使用 fprintf 成功将缓冲区内容打印到文件中。但是,当我尝试直接在程序中将缓冲区作为数组访问时,会出现分段错误。为什么会这样? 代码如下:

  #define PTT_DUMP "/home/dhruv/somefile"     
  .
  .
  .

  int fd_ptt_dump = open(PTT_DUMP, O_RDONLY);
  struct stat struct_ptt_dump;
  fstat(fd_ptt_dump, &struct_ptt_dump);
  printf("\n\n\t\t\t --- The size of the dump is = %d -----\n\n",    struct_ptt_dump.st_size);
  char *membuffer;
  char pid_num[100];
  char cycles[100], instr[100], cpi[100] ;
  int pid_index =0;
  int cycles_index = 0;
  int instr_index = 0;
  int cpi_index = 0 ;
  int len = (int)struct_ptt_dump.st_size;
  int newline_count = 0;
  int n = 0;
  if( (membuffer = mmap(0, struct_ptt_dump.st_size, PROT_READ, MAP_FILE | MAP_PRIVATE, fd_ptt_dump, 0)) == (caddr_t) -1)
   err_sys("mmap error");

  /* If the following is uncommented, it prints fine */

  /*
  for ( n =0; n < struct_ptt_dump.st_size ; n++)      
  fprintf(fp_logfile,"%c", membuffer[n]);
  */

  /* But, I really want to access the buffer as an array for speed if possible */
  /* Here is where everything goes haywire */

  while (newline_count != 5)
   if ( membuffer[n++] == '\n' )
    newline_count++ ;

  /* printf returns OK, and I am able to skip newlines */

  printf("\n\n newlines = %d\n\n 10 buffer characters", newline_count);

  int k =0;


  /* All code from here gives segmentation fault */

  while ( membuffer[n++] != ' ' )
pid_num[pid_index++] = membuffer[n] ;

  /* Even if I comment out everything from here on, the above assignment itself results in a segmentation fault */


  pid_num[pid_index] = '\0';

  printf("\n\n pid = %s", pid_num);



  while ( membuffer[len--] != ' ' )
if ( membuffer[len] != '\n' )
  cpi[cpi_index++] = membuffer[len];

  cpi[cpi_index] = '\0';

  for( ; membuffer[len] == ' ' ; len-- )
;

  for(n=0; membuffer[len-n] != ' '; n++)
instr[instr_index++] = membuffer[len-n] ;

  instr[instr_index] = '\0' ;
  n++;

  for( ; membuffer[len-n] != ' ' ; n++)
cycles[cycles_index++] = membuffer[len-n];

  cycles[cycles_index] = '\0';

  printf("\n\n\t\t\t\t ********** buffer values *************\n\n");
  printf("\t\t\t\tdominant pid = %s\t cycles = %s\t instructions = %s\t cpi = %s \n\n", pid_num, cycles, instr, cpi);



  fflush(STDOUT_FILENO);

【问题讨论】:

  • 请使用 gdb 运行并指定堆栈(gdb 中的 where 命令)。
  • 直接使用缓冲区有什么限制吗?我可能会在whiles和ifs中搞砸并存储超出我分配的东西,但我想知道是否可以直接访问内存。我知道第一次访问缓冲区会引发页面错误,但这并不重要。作为一个附带问题,如果在 printf("\n\n newlines = %d ..") 之后由于上下文切换而调度了其他进程,那么当再次调度该进程时是否会恢复缓冲区的内容?
  • @Drakosha 是的,我会这样做并带着堆栈信息返回这里。

标签: c linux memory segmentation-fault mmap


【解决方案1】:

pid_index 有多大?您是否阅读了超过 100 个字符却找不到空格?

顺便说一句,这个

while ( membuffer[n++] != ' ' )
    pid_num[pid_index++] = membuffer[n] ;

应该是这样的

while ( membuffer[n] != ' ' )
    pid_num[pid_index++] = membuffer[n++] ;

没有?

【讨论】:

  • 感谢您指出这一点。 pid_index 确实溢出了。该程序现在运行良好。
【解决方案2】:
  • 您没有检查 n 是否保持
  • 您没有检查 pid_index 是否保持

【讨论】:

  • 谢谢,我将这些界限放入程序中以提高稳定性。
猜你喜欢
  • 1970-01-01
  • 2018-04-01
  • 2010-11-06
  • 2018-06-06
  • 1970-01-01
  • 1970-01-01
  • 2012-08-08
  • 1970-01-01
  • 2014-11-05
相关资源
最近更新 更多