分段故障11混乱

时间:2014-03-02 08:11:58

标签: c segmentation-fault

我的作业是读取test.m文件,解析它,并将相关信息放入类型语句数组中(我创建它)。我已经完成了大部分工作。唯一错误的是我在输出的最后得到了一个Segmentation Fault 11。认为我正在做的输出部分可能有问题,所以我评论它看出来。它仍然给了我一个错误。无论我评论什么,它总是在程序的最后给我一个seg错误。令我感到困惑的是它造成了什么。我该怎么办?由于其他一切正常,我应该忽略它吗?

这是代码

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#define LC_SIZE 16
#define FREAD_SIZE 1024
#define STATES_SIZE 1024

FILE *ifp;
int linetotal;
int locals[LC_SIZE];
int localspot = 0; // localspot = # of locals
int EP;

typedef char STRING[256];

typedef struct {
  STRING label;
  STRING opcode;
  STRING arg;
} statement;

int main(int argc, char *argv[]) {

  //statement states[STATES_SIZE] = {"(NULL)"};

  char str[256];
  const char newline[3] = "\n";
  const char space[2] = " ";
  char *token;
  unsigned long length;

  ifp = (argc > 1) ? fopen(argv[1], "r") : NULL;

  if (ifp == NULL) {
    printf("Input file not available\n");
    return -1; }

  fread(str, FREAD_SIZE, 1, ifp);

  token = strtok(str, space);

  statement states[STATES_SIZE];
   for (int i = 0; i < STATES_SIZE; i++) {
    strcpy(states[i].label, "(null)");
    strcpy(states[i].opcode, "(null)");
    strcpy(states[i].arg, "(null)"); }

  while (token != NULL) {

    length = strlen(token);
    if (length > 1 ) { strcpy(str, &token[length-1]); }

    // comment check
    if ((strncmp(token, "/", 1) == 0) || (strcmp(str, "/") == 0)) {
      token = strtok(NULL, newline);
      //printf("Comment :: %s\n", token);
      token = strtok(NULL, space);
    }

    // local check
    else if (strcmp(token, "LC:") == 0) { 
      token = strtok(NULL, space);
      //printf("==LC here==\n");
      locals[localspot] = atoi(token);
      localspot++;
      //printf("Local variable %i = %s\n", localspot, token);
      token = strtok(NULL, space);
    }

    // starting point check
    else if (strcmp(token, "EP:") == 0) {
      token = strtok(NULL, space);
      EP = atoi(token);
      //printf("\nEP: %i\n", EP);
    }

    // label check
    else if (strcmp(str, ":") == 0) {
      //printf("--Label found :: %s\n", token);
      strcpy(states[linetotal].label, token);
      token = strtok(NULL, space);
    }

    // a function
    else if ((strcmp(token, "") != 0) && (strcmp(token, "\n") != 0)){
      //printf("Function :: %s\n", token);
      strcpy(states[linetotal].opcode, token);
      if ((strcmp(token, "halt") != 0) && (strcmp(token, "mul") != 0) && (strcmp(token,     "sub") != 0)) {
        token = strtok(NULL, space); 
        strcpy(states[linetotal].arg, token);
        //printf("arg :: %s\n", token);
      }
      token = strtok(NULL, space);
      linetotal++;
    }
    else
      break;
  } 

  // the output process
  printf("System State:\n-------------\n");
  printf("prog.PC: %i\n", EP);
  printf("Number of locals: %i\n", localspot);
  for (int i = 0; i < localspot; i++) 
    printf("Local %i: %i\n", i, locals[i]);
  printf("prog.PROGSIZE: %i\n\n", linetotal);

  printf("       Program      \n--------------------\n\n");
  for (int i = 0; i < linetotal; i++) 
    printf("%.6s %.5s %.6s\n", states[i].label, states[i].opcode, states[i].arg);

  fclose(ifp);
  return(0);
}

1 个答案:

答案 0 :(得分:2)

使用fread,您可以从文件中读取原始数据。这意味着您读取的字符串不是以空值终止的。所有strxxx函数都以空终止字符串运行。因为str不是以空值终止的,所以它们将真实数据之后的垃圾视为字符串的一部分 - 未确定的长度。

你还应该注意WhozCraig所说的内容并使你的str足够大,这也意味着为终止空字符保留一个额外的字符。为缓冲区大小定义一个常量是个好主意,但是你应该在整个程序中一致地使用它。

无论如何:null-terminate你的字符串。 fread返回成功读取的项目,因此:

char str[FREAD_SIZE + 1];
int n;

/* ... */

n = fread(str, 1. FREAD_SIZE, ifp);
if (n < 0) exit(1);
str[n] = '\0';

请注意,我已将第二个和第三个参数交换为fread。第二个是每个项目的大小,这里是char。第三个是要读取的最大项目数。返回值将取决于它,因为它返回读取的项数,而不是读取的字节数。

编辑:字符缓冲区str当然必须大到能够终止'\0'。否则,如果读取最大字符数,则设置str[n]将写入缓冲区之外。 (感谢WhozCraig发现了这一点。我自己也应该这样做。)