/* $Header$ */ /* Command grammar */ { #include #include #include #include "ops.h" #include "class.h" #include "position.h" #include "file.h" #include "idf.h" #include "symbol.h" #include "tree.h" #include "langdep.h" #include "token.h" #include "expr.h" extern char *Salloc(); extern char *strindex(); extern char *strcpy(); extern void signal_child(); extern FILE *db_in; extern int disable_intr; extern p_tree run_command, print_command; struct token tok, aside; int errorgiven = 0; int child_interrupted = 0; int interrupted = 0; int eof_seen = 0; static int shellescape(); static int extended_charset = 0; static int in_expression = 0; #define binprio(op) ((*(currlang->binop_prio))(op)) #define unprio(op) ((*(currlang->unop_prio))(op)) } %start Commands, commands; %lexical LLlex; commands { p_tree com, lastcom = 0; int give_prompt; } : { errorgiven = 0; } [ %persistent command_line(&com) [ '\n' { give_prompt = 1; } | %default ';' { give_prompt = 0; } ] { if (com) { if (lastcom) { freenode(lastcom); lastcom = 0; } if (errorgiven) { if (com != run_command) freenode(com); com = 0; } else { enterlog(com); eval(com); if (repeatable(com)) { lastcom = com; } else if (! in_status(com) && com != run_command && com != print_command) { freenode(com); com = 0; } } } else if (lastcom && ! errorgiven) { enterlog(lastcom); eval(lastcom); } if (give_prompt) { errorgiven = 0; interrupted = 0; prompt(); } } ]* ; command_line(p_tree *p;) : { *p = 0; } [ list_command(p) | file_command(p) | run_command(p) | stop_command(p) | when_command(p) | continue_command(p) | step_command(p) | next_command(p) | regs_command(p) | where_command(p) | STATUS { *p = mknode(OP_STATUS); } | DUMP { *p = mknode(OP_DUMP); } | RESTORE opt_num(p) { *p = mknode(OP_RESTORE, *p); } | delete_command(p) | print_command(p) | display_command(p) | trace_command(p) | set_command(p) | help_command(p) | FIND qualified_name(p){ *p = mknode(OP_FIND, *p); } | WHICH qualified_name(p){ *p = mknode(OP_WHICH, *p); } | able_command(p) | '!' { (void) shellescape(); *p = mknode(OP_SHELL); } | source_command(p) | log_command(p) | frame_command(p) | ] ; frame_command(p_tree *p;) : FRAME [ { *p = mknode(OP_FRAME, (p_tree) 0); } | count(p) { *p = mknode(OP_FRAME, *p); } | '-' count(p) { *p = mknode(OP_DOWN, *p); } | '+' count(p) { *p = mknode(OP_UP, *p); } ] ; source_command(p_tree *p;) : SOURCE { extended_charset = 1; } name(p) { (*p)->t_idf = str2idf((*p)->t_str, 0); } { *p = mknode(OP_SOURCE, *p); extended_charset = 0; } ; log_command(p_tree *p;) : LOG { extended_charset = 1; } [ name(p) { (*p)->t_idf = str2idf((*p)->t_str, 0); } | { *p = 0; } ] { *p = mknode(OP_LOG, *p); extended_charset = 0; } ; where_command(p_tree *p;) : WHERE opt_num(p) { *p = mknode(OP_WHERE, *p); } ; list_command(p_tree *p;) { p_tree t1 = 0, t2 = 0; } : LIST [ | position(&t1) | qualified_name(&t1) ] [ ',' count(&t2) | '-' [ count(&t2) { t2->t_ival = - t2->t_ival; } | { t2 = mknode(OP_INTEGER, -100000000L); } ] | ] { *p = mknode(OP_LIST, t1, t2); } ; file_command(p_tree *p;) : XFILE { extended_charset = 1; } [ { *p = 0; } | name(p) { (*p)->t_idf = str2idf((*p)->t_str, 0); } ] { *p = mknode(OP_FILE, *p); extended_charset = 0; } ; help_command(p_tree *p;) : [ HELP | '?' ] { *p = mknode(OP_HELP, (p_tree) 0); } [ name(&(*p)->t_args[0])? | '?' { (*p)->t_args[0] = mknode(OP_NAME, str2idf("help",0), (char *) 0); } | '!' { (*p)->t_args[0] = mknode(OP_NAME, (struct idf *) 0, "!"); } ] ; run_command(p_tree *p;) : RUN { extended_charset = 1; } args(p) { *p = mknode(OP_RUN, *p); extended_charset = 0; } | RERUN { if (! run_command) { error("no run command given yet"); } else *p = run_command; } [ '?' { *p = mknode(OP_PRCOMM, *p); } | ] ; stop_command(p_tree *p;) { p_tree whr = 0, cond = 0; } : STOP where(&whr)? condition(&cond)? { if (! whr && ! cond) { error("no position or condition"); *p = 0; } else *p = mknode(OP_STOP, whr, cond); } ; trace_command(p_tree *p;) { p_tree whr = 0, cond = 0, exp = 0; } : TRACE [ ON format_expression(&exp) ]? where(&whr)? condition(&cond)? { *p = mknode(OP_TRACE, whr, cond, exp); } ; continue_command(p_tree *p;) { long l; p_tree pos = 0; } : CONT [ INTEGER { l = tok.ival; } | { l = 1; } ] [ AT position(&pos) ]? { *p = mknode(OP_CONT, mknode(OP_INTEGER, l), pos); } ; when_command(p_tree *p;) { p_tree whr = 0, cond = 0; } : WHEN where(&whr)? condition(&cond)? { *p = mknode(OP_WHEN, whr, cond, (p_tree) 0); p = &(*p)->t_args[2]; } '{' command_line(p) [ ';' { if (*p) { *p = mknode(OP_LINK, *p, (p_tree) 0); p = &((*p)->t_args[1]); } } command_line(p) ]* '}' { if (! whr && ! cond) { error("no position or condition"); } else if (! *p) { error("no commands given"); } } ; step_command(p_tree *p;) : STEP { *p = mknode(OP_STEP, (p_tree) 0); } count(&(*p)->t_args[0])? ; next_command(p_tree *p;) : NEXT { *p = mknode(OP_NEXT, (p_tree) 0); } count(&(*p)->t_args[0])? ; regs_command(p_tree *p;) : REGS { *p = mknode(OP_REGS, (p_tree) 0); } count(&(*p)->t_args[0])? ; delete_command(p_tree *p;) : DELETE num_list(p)? { *p = mknode(OP_DELETE, *p); } ; print_command(p_tree *p;) : PRINT [ format_expression_list(p) { *p = mknode(OP_PRINT, *p); } | { *p = mknode(OP_PRINT, (p_tree) 0); } ] ; display_command(p_tree *p;) : DISPLAY format_expression_list(p) { *p = mknode(OP_DISPLAY, *p); } ; format_expression_list(p_tree *p;) : format_expression(p) [ ',' { *p = mknode(OP_LINK, *p, (p_tree) 0); p = &((*p)->t_args[1]); } format_expression(p) ]* ; format_expression(p_tree *p;) { p_tree p1; } : expression(p, 0) [ '\\' name(&p1) { register char *c = p1->t_str; while (*c) { if (! strindex("doshcax", *c)) { error("illegal format: %c", *c); break; } c++; } *p = mknode(OP_FORMAT, *p, p1); } | ] ; set_command(p_tree *p;) : SET expression(p, 0) { *p = mknode(OP_SET, *p, (p_tree) 0); } TO expression(&((*p)->t_args[1]), 0) ; able_command(p_tree *p;) : [ ENABLE { *p = mknode(OP_ENABLE, (p_tree) 0); } | DISABLE { *p = mknode(OP_DISABLE, (p_tree) 0); } ] num_list(&(*p)->t_args[0])? ; num_list(p_tree *p;) : num(p) [ ',' { *p = mknode(OP_LINK, *p, (p_tree) 0); } num(&(*p)->t_args[1]) ]* ; condition(p_tree *p;) : IF expression(p, 0) ; where(p_tree *p;) : IN qualified_name(p) { *p = mknode(OP_IN, *p, (p_tree) 0); } [ AT position(&((*p)->t_args[1])) ]? | AT position(p) ; expression(p_tree *p; int level;) { int currprio, currop; } : { in_expression++; } factor(p) [ %while ((currprio = binprio(currop = (int) tok.ival)) > level) [ BIN_OP | PREF_OR_BIN_OP ] { *p = mknode(OP_BINOP, *p, (p_tree) 0); (*p)->t_whichoper = currop; } expression(&((*p)->t_args[1]), currprio) | SEL_OP { *p = mknode(OP_BINOP, *p, (p_tree) 0); (*p)->t_whichoper = (int) tok.ival; } name(&(*p)->t_args[1]) | '[' { *p = mknode(OP_BINOP, *p, (p_tree) 0); (*p)->t_whichoper = E_ARRAY; } expression(&(*p)->t_args[1], 0) [ ',' { *p = mknode(OP_BINOP, *p, (p_tree) 0); (*p)->t_whichoper = E_ARRAY; } expression(&(*p)->t_args[1], 0) ]* ']' ]* { in_expression--; } ; factor(p_tree *p;) : [ %default EXPRESSION /* lexical analyzer will never return this token */ { *p = mknode(OP_INTEGER, 0L); } | '(' expression(p, 0) ')' | INTEGER { *p = mknode(OP_INTEGER, tok.ival); } | REAL { *p = mknode(OP_REAL, tok.fval); } | STRING { *p = mknode(OP_STRING, tok.str); } | qualified_name(p) | { *p = mknode(OP_UNOP, (p_tree) 0); (*p)->t_whichoper = (int) tok.ival; } [ PREF_OP | PREF_OR_BIN_OP { (*currlang->fix_bin_to_pref)(*p); } ] expression(&(*p)->t_args[0], unprio((*p)->t_whichoper)) ] [ %while(1) POST_OP { *p = mknode(OP_UNOP, *p); (*p)->t_whichoper = (int) tok.ival; } ]* ; position(p_tree *p;) { p_tree lin; char *str; } : [ STRING { str = tok.str; } ':' | { if (! listfile) str = 0; else str = listfile->sy_idf->id_text; } ] count(&lin) { *p = mknode(OP_AT, lin->t_ival, str); freenode(lin); } ; args(p_tree *p;) { int first_time = 1; } : [ { if (! first_time) { *p = mknode(OP_LINK, *p, (p_tree) 0); p = &((*p)->t_args[1]); } first_time = 0; } arg(p) ]* ; arg(p_tree *p;) : name(p) | '>' name(p) { (*p)->t_oper = OP_OUTPUT; } | '<' name(p) { (*p)->t_oper = OP_INPUT; } ; count(p_tree *p;) : INTEGER { *p = mknode(OP_INTEGER, tok.ival); } ; opt_num(p_tree *p;) : num(p) | { *p = 0; } ; num(p_tree *p;) : count(p) | '-' count(p) { (*p)->t_ival = - (*p)->t_ival; } ; qualified_name(p_tree *p;) : name(p) [ '`' { *p = mknode(OP_SELECT, *p, (p_tree) 0); } name(&((*p)->t_args[1])) ]* ; name(p_tree *p;) : [ XFILE | LIST | RUN | RERUN | STOP | WHEN | AT | IN | IF | %default NAME | CONT | STEP | NEXT | REGS | WHERE | STATUS | PRINT | DELETE | DUMP | RESTORE | TRACE | ON | SET | TO | FIND | DISPLAY | WHICH | HELP | DISABLE | ENABLE | SOURCE | FRAME | LOG ] { *p = mknode(OP_NAME, tok.idf, tok.str); } ; { int LLlex() { register int c; if (ASIDE) { tok = aside; ASIDE = 0; return TOK; } do { c = getc(db_in); } while (c != EOF && class(c) == STSKIP); if (c == EOF) { eof_seen = 1; return c; } if (extended_charset && in_ext(c)) { TOK = get_name(c); return TOK; } switch(class(c)) { case STSTR: TOK = (*currlang->get_string)(c); break; case STIDF: if (in_expression) TOK = (*currlang->get_name)(c); else TOK = get_name(c); break; case STNUM: TOK = (*currlang->get_number)(c); break; case STNL: TOK = c; break; case STSIMP: if (! in_expression) { TOK = c; break; } /* Fall through */ default: TOK = (*currlang->get_token)(c); break; } return TOK; } int get_name(c) register int c; { char buf[512+1]; register char *p = &buf[0]; register struct idf *id; do { if (p - buf < 512) *p++ = c; c = getc(db_in); } while ((extended_charset && in_ext(c)) || in_idf(c)); ungetc(c, db_in); *p++ = 0; if (extended_charset) { tok.idf = 0; tok.str = Salloc(buf, (unsigned) (p - buf)); return NAME; } id = str2idf(buf, 1); tok.idf = id; tok.str = id->id_text; return id->id_reserved ? id->id_reserved : NAME; } extern char *symbol2str(); LLmessage(t) { if (t > 0) { if (! errorgiven) { error("%s missing before %s", symbol2str(t), symbol2str(TOK)); } aside = tok; } else if (t == 0) { if (! errorgiven) { error("%s unexpected", symbol2str(TOK)); } } else if (! errorgiven) { error("EOF expected"); } errorgiven = 1; } static void catch_del() { signal(SIGINT, catch_del); if (! disable_intr) { signal_child(SIGEMT); child_interrupted = 1; } interrupted = 1; } void init_del() { signal(SIGINT, catch_del); } static void ctch() { /* Only for shell escapes ... */ signal(SIGINT, ctch); } #define SHBUFSIZ 512 static int shellescape() { register char *p; /* walks through command */ static char previous[SHBUFSIZ]; /* previous command */ char comm[SHBUFSIZ]; /* space for command */ register int cnt; /* prevent array bound errors */ register int c; /* current char */ register int lastc = 0; /* will contain the previous char */ p = comm; cnt = SHBUFSIZ-2; while (c = getc(db_in), c != '\n') { switch(c) { case '!': /* * An unescaped ! expands to the previous * command, but disappears if there is none */ if (lastc != '\\') { if (*previous) { int len = strlen(previous); if ((cnt -= len) <= 0) break; strcpy(p,previous); p += len; } } else { *p++ = c; } continue; case '%': /* * An unescaped % will expand to the current * filename, but disappears is there is none */ if (lastc != '\\') { if (listfile) { int len = strlen(listfile->sy_idf->id_text); if ((cnt -= len) <= 0) break; strcpy(p,listfile->sy_idf->id_text); p += len; } } else { *p++ = c; } continue; default: lastc = c; if (cnt-- <= 0) break; *p++ = c; continue; } break; } *p = '\0'; if (c != '\n') { warning("shell command too long"); while (c != '\n') c = getc(db_in); } ungetc(c, db_in); strcpy(previous, comm); signal(SIGINT, ctch); cnt = system(comm); signal(SIGINT, catch_del); return cnt; } }