Logo Search packages:      
Sourcecode: jgraph version File versions  Download package

token.c

/* 
 * $Source: /tmp_mnt/n/fs/grad1/jsp/src/jgraph/RCS/token.c,v $
 * $Revision: 8.3 $
 * $Date: 92/11/30 11:42:42 $
 * $Author: jsp $
 */

#include "jgraph.h"
#include "list.h"

#define CNULL ((char *)0)

typedef struct iostack {
  struct iostack *flink;
  struct iostack *blink;
  char *filename;
  FILE *stream;
  int oldcharvalid;
  char oldchar;
  char pipe;
  int line;
} *Iostack;

static char INPUT[1000];
static int getnew = 1;
static char oldchar = '\0';
static int oldcharvalid = 0;
static char pipe = 0;
static int eof = 0;
static int init = 0;
static Iostack stack;
static char real_eof = EOF;

static FILE *IOSTREAM;
static char FILENAME[300];
static int line = 1;

void set_input_file(char *s)
{
  Iostack n;

  if (init == 0) {
    stack = (Iostack) make_list(sizeof(struct iostack));
    if (s == CNULL) {
      IOSTREAM = stdin;
      strcpy(FILENAME, "<stdin>");
    } else {
      IOSTREAM = fopen(s, "r");
      if (IOSTREAM == NULL) {
        fprintf(stderr, "Error: cannot open file \"%s\"\n", s);
        exit(1);
      }
      strcpy(FILENAME, s);
    }
    init = 1;
  } else {
    n = (Iostack) get_node((List)stack);
    n->stream = NULL;
    n->filename = (char *) malloc (sizeof(char)*(strlen(s)+2));
    strcpy(n->filename, s);
    n->oldchar = oldchar;
    n->oldcharvalid = oldcharvalid;
    n->pipe = pipe;
    n->line = line;
    insert((List)n, (List)stack->flink);
  }
}

void error_header(void)
{
  fprintf(stderr, "%s,%d: ", FILENAME, line);
}
  
int gettokenchar(void)
{
  if (oldcharvalid == 0) oldchar = getc(IOSTREAM);
  oldcharvalid = 0;
  if (oldchar == '\n') line++;
  return oldchar;
}

void ungettokenchar(void)
{
  oldcharvalid = 1;
  if (oldchar == '\n') line--;
}

int gettoken(char *s)
{
  int i;
  char c;

  for (c = gettokenchar(); 
       c == ' ' || c == '\t' || c == '\n';
       c = gettokenchar()) ;
  for (i = 0;
       c != real_eof && c != ' ' && c != '\t' && c != '\n';
       c = gettokenchar()) {
    s[i++] = c;
  }
  s[i] = '\0';
  ungettokenchar();
  return i;
}

void get_comment(void)
{
  if (eof) return;
  while (1) {
    if (gettoken(INPUT) == 0) return;
    else if (strcmp(INPUT, "(*") == 0)
      get_comment();
    else if (strcmp(INPUT, "*)") == 0) 
      return;
  }
}

static int iostackempty(void)
{
  return (first(stack) == nil(stack));
}

static void push_iostack(int p)
{
  Iostack n;

  n = (Iostack) get_node((List)stack);
  n->stream = IOSTREAM;
  n->filename = (char *) malloc (sizeof(char)*(strlen(FILENAME)+2));
  n->oldchar = oldchar;
  n->oldcharvalid = oldcharvalid;
  n->pipe = pipe;
  n->line = line;
  strcpy(n->filename, FILENAME);
  insert((List)n, (List)stack);
  if (p) {
    IOSTREAM = (FILE *) popen(INPUT, "r");
  } else {
    IOSTREAM = fopen(INPUT, "r");
  }
  pipe = p;
  line = 1;
  if (IOSTREAM == NULL) {
    error_header();
    fprintf(stderr, "Include file \"%s\" does not exist\n", INPUT);
    exit(1);
  }
  strcpy(FILENAME, INPUT);
}

static void pop_iostack(void)
{
  Iostack n;

  fflush(IOSTREAM);
  if (pipe) {
    if (pclose(IOSTREAM)) {
      /*error_header();
      fprintf(stderr, "\n\nPipe returned a non-zero error code.\n");
      exit(1); */
    }
  } else {
    fclose(IOSTREAM);
  }
  n = last(stack);
  if (n->stream == NULL) {
    n->stream = fopen(n->filename, "r");
    if (n->stream == NULL) {
      fprintf(stderr, "Error: cannot open file \"%s\"\n", n->filename);
      exit(1);
    }
  }
  IOSTREAM = n->stream;
  strcpy(FILENAME, n->filename);
  free(n->filename);
  pipe = n->pipe;
  line = n->line;
  oldchar = n->oldchar;
  oldcharvalid = n->oldcharvalid;
  delete_item((List)n);
  free_node((List)n, (List)stack);
}

static void nexttoken(void)
{
  if (eof) return;
  if (getnew) {
    while (1) {
      if (gettoken(INPUT) == 0) {
        if (iostackempty()) {
          eof = 1;
          getnew = 0;
          return;
        } else {
          pop_iostack();
        }
      } else if (strcmp(INPUT, "(*") == 0) {
        get_comment();
      } else if (strcmp(INPUT, "include") == 0) {
        if (gettoken(INPUT) == 0) {
          error_header();
          fprintf(stderr, "Empty include statement\n");
          exit(1);
        } else {
          push_iostack(0);
        }
      } else if (strcmp(INPUT, "shell") == 0) {
        if (gettoken(INPUT) == 0 || strcmp(INPUT, ":") != 0) {
          error_header();
          fprintf(stderr, "'shell' must be followed by ':'\n");
          exit(1);
        } 
        if (getsystemstring() == 0) {
          fprintf(stderr, "Empty shell statement\n");
          exit(1);
        }
        push_iostack(1);
      } else {
        getnew = 1;
        return;
      }
    }
  }
  getnew = 1;
  return;
}

int getstring(char *s)
{
  nexttoken();
  if (eof) return 0;
  strcpy(s, INPUT);
  return 1;
}

int getint(int *i)
{
  int j;

  nexttoken();
  if (eof) return 0;
  *i = atoi(INPUT);
  if (*i == 0) {
    for (j = 0; INPUT[j] != '\0'; j++)
      if (INPUT[j] != '0') return 0;
  }
  return 1;
}

int getfloat(float *f)
{
  int j;

  nexttoken();
  if (eof) return 0;
  *f = (float) atof(INPUT);
  if (*f == 0.0) {
    for (j = 0; INPUT[j] == '-'; j++);
    while (INPUT[j] == '0') j++;
    if (INPUT[j] == '.') j++;
    while (INPUT[j] == '0') j++;
    if (INPUT[j] == 'e' || INPUT[j] == 'E') {
      j++;
      if (INPUT[j] == '+' || INPUT[j] == '-') j++;
      while (INPUT[j] == '0') j++;
    }
    return (INPUT[j] == '\0');
  } else return 1;
}

static char *new_printable_text(char *s)
{
  char *new_s;
  int to_pad, i, j;

  to_pad = 0;
  for (i = 0; s[i] != '\0'; i++) {
    if (s[i] == '\\' || s[i] == ')' || s[i] == '(') {
       to_pad++;
    }
  }

  j = sizeof(char) * (i + to_pad + 2);
  if ((j % 8) != 0) j += 8 - j % 8;
  new_s = (char *) malloc (j);
  j = 0;
  for (i = 0; s[i] != '\0'; i++) {
    if (s[i] == '\\' || s[i] == ')' || s[i] == '(') {
      new_s[j++] = '\\';
    }
    new_s[j++] = s[i];
  }
  new_s[j] = '\0';            /* added: tie off -hdd */
  return new_s;
}

char *getmultiline(void)
{
  char c;
  int i, done, len=0, started;
  char *out_str = NULL;

  if (getnew == 0) return CNULL;
  
  c = gettokenchar();
  if (c == real_eof) {
    ungettokenchar();
    return CNULL;
  }
  done = 0;
  started = 0;
  while (!done) {
    i = 0;
    for (c = gettokenchar(); c != real_eof && c != '\n';  c = gettokenchar()) {
      INPUT[i++] = c;
    }
    INPUT[i] = '\0';
    if (!started) {
      out_str = (char *) malloc (sizeof(char)*(i+1));
      strcpy(out_str, INPUT);
      len = i;
      started = 1;
    } else {
      out_str = (char *) realloc(out_str, (len + i + 3) * sizeof(char));
      sprintf(&(out_str[len]), "\n%s", INPUT);
      len += i+1;
    }
    if (c == '\n' && len != 0 && out_str[len-1] == '\\') {
      len--;
    } else {
      done = 1;
    }
  }
  ungettokenchar();
  return out_str;
}

char *getlabel(void)
{
  char *txt, *new;

  txt = getmultiline();
  if (txt == CNULL) return CNULL;
  new = new_printable_text(txt);
  free(txt);
  return new;
}

int getsystemstring(void)
{
  char c;
  int i;
  int done;

  if (getnew == 0) return 0;
  
  c = gettokenchar();
  if (c == real_eof) {
    ungettokenchar();
    return 0;
  }
  i = 0;
  done = 0;
  while (!done) {
    for (c = gettokenchar(); c != real_eof && c != '\n';  c = gettokenchar()) {
      INPUT[i++] = c;
    }
    if (c == '\n' && i > 0 && INPUT[i-1] == '\\') {
      INPUT[i++] = '\n';
    } else {
      done = 1;
    }
  }
  ungettokenchar();
  INPUT[i] = '\0';
  return 1;
}

void rejecttoken(void)
{
  getnew = 0;
}

Generated by  Doxygen 1.6.0   Back to index