view tests/agcl/contrib/yabasic.syn @ 20:bb115deb6fb2

Improve agfiles rule. (1) It didn't depend on $(AGCL) and it absolutely should have. (2) allow AGFORCE=1 to make it rebuild whether or not it looks out of date. (3) Document this.
author David A. Holland
date Mon, 13 Jun 2022 00:02:15 -0400
parents 13d2b8934445
children
line wrap: on
line source

{

/*
     YABASIC --- a tiny integrated Basic Compiler/Interpreter

     BISON - part

     this Program is subject to the GNU General Public License;
     see the file yabasic.c for details.
*/


//#undef WINDOWS
#include "yabasic.h"     /* definitions of yabasic */
#include <malloc.h>

#if HAVE_ALLOCA_H
#include <alloca.h>
#endif

void __yy_bcopy(char *,char *,int); /* prototype missing */

int yylineno=1;
int yylex(void);

char *popString(int);
void pushChar(int);

}

[
  ~case sensitive
  disregard white space
  lexeme {digit string, name, STRING, STRSYM, number}
  distinguish lexemes
  parser name = parseBasic
  line numbers
  escape backslashes
]

white space = ' ' + '\t'

//(double *) step_part


SEP
 -> ["REM", ~(eof + white space + '\n')?...], '\n'      ={
     yylineno++;
     if (interactive) PCB.exit_flag = AG_SUCCESS_CODE;
     return;
 }

(double) number
 -> simple real
 -> simple real:x, 'e'+'E', '+'?,exponent:e            =x*pow(10,e);
 -> simple real:x, 'e'+'E', '-',exponent:e            =x*pow(10,-e);

(double) simple real
 -> integer part:i, '.', fraction part:f                      = i+f;
 -> integer part, '.'?
 -> '.', fraction part:f                                        = f;

(double) integer part
 -> digit:d                                                 = d-'0';
 -> integer part:x, digit:d                          = 10*x + d-'0';

(double) fraction part
 -> digit:d                                            =(d-'0')/10.;
 -> digit:d, fraction part:f                       =(d-'0' + f)/10.;

(int) exponent
 -> digit:d                                                 = d-'0';
 -> exponent:x, digit:d                              = 10*x + d-'0';

(int) name           //value of name token is length of name string
 -> letter: c                                       =pushChar(c), 1;
 -> name:k, letter+digit: c                       =pushChar(c), k+1;


(char *) SYMBOL
  -> name:k                                   =strdup(popString(k));

(char *) STRSYM
  -> name:k, '$'                              =pushChar('$'), strdup(popString(k+1));

(char *) DIGITS
  -> digit string:k                                  =strdup(popString(k));

(int) digit string
 -> digit:d                                         =pushChar(d), 1;
 -> digit string:k, digit:d                         =pushChar(d), k+1;

(char *) STRING
 -> '"', string text:k, '"'                         =strdup(popString(k));
 -> '"', string text:k, '\n'                        =strdup(popString(k));

(int) string text
 ->                                                 =0;
 -> string text:k, ~(eof + '"' + '\n'):c            =pushChar(c), k+1;
 -> string text:k, "\\\""                           =pushChar('"'), k+1;

/*
[
  left {"OR"}
  left {"AND"}
  left {"NOT"}
  left {'-', '+'}
  left {'*', '/'}
  left {'^'}
  nonassoc {UMINUS}
]
*/

letter = 'a-z' + 'A-Z'
digit  = '0-9'
eof = 0 + -1


program $
 -> statement list, eof             =end_of_file = TRUE;

statement list
 ->
/*
 -> statement_list, !{
if (errorlevel<=ERROR) {YYABORT;}},
   SEP, !{
yylineno+=$3;},
   statement
*/
 -> statement list, SEP, statement
 -> statement list, SEP, label, statement

label
 -> "LABEL", SYMBOL:s                =create_label(s);
 -> DIGITS:s                         =create_label(s);


statement  /* empty */
 ->
 -> string_assignment
 -> assignment
 -> for_loop
 -> if_clause
 -> "GOTO", symbol_or_lineno:s ={create_goto(s);}
 -> "GOSUB", symbol_or_lineno:s ={create_gosub(s);}
 -> "ON", "INTERRUPT", "BREAK" ={create_exception(TRUE);}
 -> "ON", "INTERRUPT", "CONTINUE" ={create_exception(FALSE);}
 -> "ON", expression, "GOTO", !{create_skipper();}, goto_list ={create_nop();}
 -> "ON", expression, "GOSUB", !{create_skipper();}, gosub_list ={create_nop();}
// -> LABEL, symbol_or_lineno ={create_label($2);}
 -> "OPEN", hashed_number:n, ',', string expression, ',', string expression ={create_myopen(n,'+');}
 -> "OPEN", hashed_number:n, ',', string expression ={create_myopen(n,'-');}
 -> "CLOSE", hashed_number:n ={create_myclose(n);}
 -> "PRINT", printintro, printlist, !{
create_revert(FALSE);},
   semicolon
 -> "INPUT", inputintro, inputlist =lastinput->args=0;
 -> "READ", readlist
 -> "DATA", datalist
 -> "RESTORE" =create_restore("");
 -> "RESTORE", symbol_or_lineno:s =create_restore(s);
 -> "RETURN" ={create_return();}
 -> "DIM", dimlist
 -> "OPEN", "WINDOW", expression, ',', expression =create_openwin(FALSE);
 -> "OPEN", "WINDOW", expression, ',', expression, ',', string expression ={create_openwin(TRUE);}
 -> "DOT", mapping                    =create_dot();
 -> "LINE", mapping, "TO", mapping =create_line('l');
 -> "CIRCLE", mapping, ',', expression =create_circle();
 -> "TEXT", string expression, ',', mapping =create_text(TRUE);
 -> "TEXT", mapping, ',', string expression =create_text(FALSE);
 -> "MAP", expression, ',', expression, ',', expression, ',', expression, "TO", expression, ',', expression, ',', expression, ',', expression =create_makemap();
 -> "ARROW", mapping, "TO", mapping =create_line('a');
 -> "XTICK", mapping, ',', string expression =create_tick(1);
 -> "YTICK", mapping, ',', string expression =create_tick(3);
 -> "XTICK", mapping =create_tick(0);
 -> "YTICK", mapping =create_tick(2);
 -> "CLOSE", "WINDOW" =create_closewin();
 -> "CLEAR", "WINDOW" =create_clearwin();
 -> "CLEAR", "SCREEN" =create_clearscreen();
 -> "OPEN", "PRINTER" =create_openprinter(0);
 -> "OPEN", "PRINTER", string expression =create_openprinter(1);
 -> "CLOSE", "PRINTER" =create_closeprinter();
 -> "WAIT", expression =create_mywait();
 -> "BELL" =create_bell();
 -> "INKEY" ={create_function(MYINKEY); create_popstrsym(NULL);}
 -> "SYSTEM2", '(', string expression, ')' ={create_function(MYSYSTEM2);
	create_popdblsym(NULL);}
 -> "POKE", string expression, ',', string expression ={create_poke('s');}
 -> "POKE", string expression, ',', expression ={create_poke('d');}
 -> "END" ={create_myend();}

string_assignment
 -> STRSYM:s, '=', string expression ={create_popstrsym(s);}
 -> "MID", '(', STRSYM:s, !{
create_pushstrptr(s);},
   ',', expression, ',', expression, ')', '=', string expression ={create_changestring(MYMID);}
 -> "LEFT", '(', STRSYM:s, !{
create_pushstrptr(s);},
   ',', expression, ')', '=', string expression ={create_changestring(MYLEFT);}
 -> "RIGHT", '(', STRSYM:s, !{
create_pushstrptr(s);},
   ',', expression, ')', '=', string expression ={create_changestring(MYRIGHT);}
 -> STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')', '=', string expression ={create_doarray(s,ASSIGNSTRINGARRAY);}
 -> "MID", '(', STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')', !{
create_doarray(s,GETSTRINGPOINTER);},
   ',', expression, ',', expression, ')', '=', string expression ={create_changestring(MYMID);}
 -> "LEFT", '(', STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')', !{
create_doarray(s,GETSTRINGPOINTER);},
   ',', expression, ')', '=', string expression ={create_changestring(MYLEFT);}
 -> "RIGHT", '(', STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')', !{
create_doarray(s,GETSTRINGPOINTER);},
   ',', expression, ')', '=', string expression ={create_changestring(MYRIGHT);}

string expression
 -> primary string expression
 -> string expression, '+', primary string expression ={create_concat();}

primary string expression
 -> STRSYM:s ={create_pushstrsym(s);}
 -> string_function
 -> STRING:s ={if (s==NULL) {error(ERROR,"String not terminated");create_pushstr("");} else {create_pushstr(s);}}
 -> STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_doarray(s,CALLSTRINGARRAY);}
 -> '(', string expression, ')'

string_function
 -> "LEFT", '(', string expression, ',', expression, ')' ={create_function(MYLEFT);}
 -> "RIGHT", '(', string expression, ',', expression, ')' ={create_function(MYRIGHT);}
 -> "MID", '(', string expression, ',', expression, ',', expression, ')' ={create_function(MYMID);}
 -> "STR", '(', expression, ')' ={create_function(MYSTR);}
 -> "STR", '(', expression, ',', string expression, ')' ={create_function(MYSTR2);}
 -> "INKEY" ={create_function(MYINKEY);}
 -> "CHR", '(', expression, ')' ={create_function(MYCHR);}
 -> "UPPER", '(', string expression, ')' ={create_function(MYUPPER);}
 -> "LOWER", '(', string expression, ')' ={create_function(MYLOWER);}
 -> "LTRIM", '(', string expression, ')' ={create_function(MYLTRIM);}
 -> "RTRIM", '(', string expression, ')' ={create_function(MYRTRIM);}
 -> "TRIM", '(', string expression, ')' ={create_function(MYTRIM);}
 -> "SYSTEM", '(', string expression, ')' ={create_function(MYSYSTEM);}
 -> "DATE" ={create_function(MYDATE);}
 -> "TIME" ={create_function(MYTIME);}
 -> "PEEK2", '(', string expression, ')' ={create_function(MYPEEK2);}

assignment
 -> SYMBOL:s, '=', expression ={create_popdblsym(s);}
 -> SYMBOL:s, '(', !{
pushcounter();},
   indexlist, ')', '=', expression ={create_doarray(s,ASSIGNARRAY);}

primary expression
 -> number:n                           =create_pushdbl(n);
 -> function
 -> SYMBOL:s                           =create_pushdblsym(s);
 -> SYMBOL:s, '(', !pushcounter();,
      indexlist, ')'                   ={create_doarray(s,CALLARRAY);}
 -> '(', expression, ')'
 -> '-', primary expression /* 1ef4:fa2arec <֠ */ ={create_negate();}

exponential expression
 -> primary expression
 -> exponential expression, '^', primary expression ={create_dblbin('^');}

multiplicative expression
 -> exponential expression
 -> multiplicative expression, '*', exponential expression ={create_dblbin('*');}
 -> multiplicative expression, '/', exponential expression ={create_dblbin('/');}

expression
 -> multiplicative expression
 -> expression, '+', multiplicative expression ={create_dblbin('+');}
 -> expression, '-', multiplicative expression ={create_dblbin('-');}

/*
 -> expression, '+', expression ={create_dblbin('+');}
 -> expression, '-', expression ={create_dblbin('-');}
 -> expression, '*', expression ={create_dblbin('*');}
 -> expression, '/', expression ={create_dblbin('/');}
 -> expression, '^', expression ={create_dblbin('^');}
 -> '-', expression /* 1ef4:fa2arec <֠ * / ={create_negate();}
*/

mapping
 -> expression, ',', expression
 -> "MAP", '(', expression, ',', expression, ')' ={create_map();}

function
 -> "SIN", '(', expression, ')' ={create_function(MYSIN);}
 -> "ASIN", '(', expression, ')' ={create_function(MYASIN);}
 -> "COS", '(', expression, ')' ={create_function(MYCOS);}
 -> "ACOS", '(', expression, ')' ={create_function(MYACOS);}
 -> "TAN", '(', expression, ')' ={create_function(MYTAN);}
 -> "ATAN", '(', expression, ')' ={create_function(MYATAN);}
 -> "ATAN", '(', expression, ',', expression, ')' ={create_function(MYATAN2);}
 -> "EXP", '(', expression, ')' ={create_function(MYEXP);}
 -> "LOG", '(', expression, ')' ={create_function(MYLOG);}
 -> "SQRT", '(', expression, ')' ={create_function(MYSQRT);}
 -> "INT", '(', expression, ')' ={create_function(MYINT);}
 -> "FRAC", '(', expression, ')' ={create_function(MYFRAC);}
 -> "MOD", '(', expression, ',', expression, ')' ={create_function(MYMOD);}
 -> "RAN", '(', expression, ')' ={create_function(MYRAN);}
 -> "RAN", '(', ')' ={create_function(MYRAN2);}
 -> "MIN", '(', expression, ',', expression, ')' ={create_function(MYMIN);}
 -> "MAX", '(', expression, ',', expression, ')' ={create_function(MYMAX);}
 -> "XMAP", '(', expression, ')' ={create_function(MYXMAP);}
 -> "YMAP", '(', expression, ')' ={create_function(MYYMAP);}
 -> "LEN", '(', string expression, ')' ={create_function(MYLEN);}
 -> "VAL", '(', string expression, ')' ={create_function(MYVAL);}
 -> "ASC", '(', string expression, ')' ={create_function(MYASC);}
 -> "INSTR", '(', string expression, ',', string expression, ')' ={create_function(MYINSTR);}
 -> "SYSTEM2", '(', string expression, ')' ={create_function(MYSYSTEM2);}
 -> "PEEK", '(', string expression, ')' ={create_function(MYPEEK);}

(double) const
 -> number:n =n;
 -> '+', number:n =n;
 -> '-', number:n =-n;

/*
number
 -> FNUM ={$$=$1;}
 -> DIGITS ={$$=atoi($1);}
*/

(int) intnum
 -> DIGITS:d                            =atoi(d);

(char *) symbol_or_lineno
 -> DIGITS:s =s;
 -> SYMBOL:s =s;

dimlist
 -> SYMBOL:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_dim(s,'d');}
 -> dimlist, ',', SYMBOL:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_dim(s,'d');}
 -> STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_dim(s,'s');}
 -> dimlist, ',', STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_dim(s,'s');}

indexlist
 -> expression ={inccounter();}
 -> indexlist, ',', expression ={inccounter();}

for_loop
 -> "FOR", SYMBOL:s, '=', expression, !{
pushname(s);create_popdblsym(s);pushgoto();
             create_pushdblsym(s);},
   "TO", expression, step_part:p, !{

       create_dblrelop((p>0)?'{':'}');
             create_decide();
             pushlabel();},
/*
   SEP, !{
yylineno+=$10;},
*/ SEP,
   statement list, !{
             create_pushdbl(p);
       create_pushdblsym(s);
             create_dblbin('+');
       create_popdblsym(s);
             swap();popgoto();poplabel();},
   next_or_eofile, next_symbol                    ={/* cookie*/}

next_or_eofile
 -> "NEXT"
 -> eof ={end_of_file=TRUE;
      error(ERROR,"'next'-statement is missing"); PCB.exit_flag = AG_SYNTAX_ERROR_CODE;}

(double) step_part
 ->  /* can be omitted */ =1.0;
 -> "STEP", const:v =v;

next_symbol
 -> /* can be omitted */ ={pop();}
 -> SYMBOL:s ={if (strcmp(pop()->pointer,s))
             {error(ERROR,"'for' and 'next' do not match"); PCB.exit_flag = AG_SYNTAX_ERROR_CODE;}
           }

if_clause
 -> "IF", condition, !{
create_decide();pushlabel();},
   "THEN", statement list, !{
pushlabel();swap();poplabel();},
   else_part, !{
poplabel();},
   endif_or_eof

endif_or_eof
 -> "ENDIF"
 -> eof ={end_of_file=TRUE;
            error(ERROR,"'endif'-statement is missing"); PCB.exit_flag = AG_SYNTAX_ERROR_CODE;}
/*
condition
 -> '(', condition, ')'
 -> condition, "OR", condition ={create_boole('|');}
 -> "NOT", condition ={create_boole('!');}
*/

condition
 -> and condition
 -> condition, "OR", and condition ={create_boole('|');}


and condition
 -> primary condition
 -> and condition, "AND", primary condition ={create_boole('&');}

primary condition
 -> comparison
 -> '(', condition, ')'
 -> "NOT", primary condition

comparison
 -> string expression, '=', string expression ={create_strrelop('=');}
 -> string expression, "<>", string expression ={create_strrelop('!');}
 -> string expression, '<', string expression ={create_strrelop('<');}
 -> string expression, "<=", string expression ={create_strrelop('{');}
 -> string expression, '>', string expression ={create_strrelop('>');}
 -> string expression, ">=", string expression ={create_strrelop('}');}
 -> expression, '=', expression ={create_dblrelop('=');}
 -> expression, "<>", expression ={create_dblrelop('!');}
 -> expression, '<', expression ={create_dblrelop('<');}
 -> expression, "<=", expression ={create_dblrelop('{');}
 -> expression, '>', expression ={create_dblrelop('>');}
 -> expression, ">=", expression ={create_dblrelop('}');}
 //-> MYEOF, '(', hashed_number, ')' ={create_testeof($3);}

else_part /* can be omitted */
 ->
 -> "ELSE", statement list

inputlist
 -> input
 -> input, ',', inputlist

input
 -> SYMBOL:s ={create_myread('d');create_popdblsym(s);}
 -> SYMBOL:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_myread('d');create_doarray(s,ASSIGNARRAY);}
 -> STRSYM:s ={create_myread('s');create_popstrsym(s);}
 -> STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_myread('s');create_doarray(s,ASSIGNSTRINGARRAY);}

readlist
 -> readitem
 -> readlist, ',', readitem

readitem
 -> SYMBOL:s ={create_readdata('d');create_popdblsym(s);}
 -> SYMBOL:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_readdata('d');create_doarray(s,ASSIGNARRAY);}
 -> STRSYM:s ={create_readdata('s');create_popstrsym(s);}
 -> STRSYM:s, '(', !{
pushcounter();},
   indexlist, ')' ={create_readdata('s');create_doarray(s,ASSIGNSTRINGARRAY);}

datalist
 -> STRING:s ={create_strdata(s);}
 -> const:v ={create_dbldata(v);}
 -> datalist, ',', STRING:s ={create_strdata(s);}
 -> datalist, ',', const:v ={create_dbldata(v);}

printlist  /* possible empty */
 ->
 -> expression ={create_print('d');}
 -> printlist, ',', expression ={create_print('d');}
 -> string expression ={create_print('s');}
 -> printlist, ',', string expression ={create_print('s');}

inputintro
 ->  ={create_myswitch(0);create_readline(NULL);}
 -> stream
 -> position ={create_myswitch(0);create_readline(NULL);}
 -> !{
create_myswitch(0);},
   prompt
 -> position, !{
create_myswitch(0);},
   prompt

printintro
 ->  /* can be empty */ ={create_myswitch(0);}
 -> stream
 -> "REVERSE" ={create_revert(TRUE);create_myswitch(0);}
 -> position ={create_myswitch(0);}
 -> "REVERSE", position ={create_revert(TRUE);create_myswitch(0);}

prompt
 -> STRING:s ={create_readline(s);}

position
 -> "AT", '(', expression, ',', expression, ')' ={create_mymove();}

stream
 -> '#', intnum:n ={create_myswitch(n);}

(int) hashed_number
 -> '#', intnum:n =n;
 -> intnum:n /* need not contain hash */ =n;

semicolon /* can be left out */
 ->  ={create_print('n');}
 -> ';'

goto_list
 -> symbol_or_lineno:s ={create_goto(s);create_findnop();}
 -> goto_list, ',', symbol_or_lineno:s ={create_goto(s);create_findnop();}

gosub_list
 -> symbol_or_lineno:s ={create_gosub(s);create_findnop();}
 -> gosub_list, ',', symbol_or_lineno:s ={create_gosub(s);create_findnop();}

{
#define CHAR_STACK_LENGTH 500

static char  charStack[CHAR_STACK_LENGTH+1];
static char *charStackTop = charStack;

/* Define an error record */
typedef struct {
  char *message;                 /* identifies error */
  int line;                      /* location of error */
  int column;
} ErrorRecord;

ErrorRecord errorRecord;           /* define an error record */

void diagnoseError(char *msg) {
  if (parseBasic_pcb.exit_flag == AG_RUNNING_CODE)
    parseBasic_pcb.exit_flag = AG_SEMANTIC_ERROR_CODE;      /* stop parse */
  errorRecord.message = msg;
  errorRecord.line    = parseBasic_pcb.line;
  errorRecord.column  = parseBasic_pcb.column;
}

void pushChar(int c) {              /* append char to name string */
  if (charStackTop < charStack+CHAR_STACK_LENGTH) {
    *charStackTop++ = (char) c;
    return;
  }
  /* buffer overflow, kill parse and issue diagnostic */
  diagnoseError("Character Stack Overflow");
}

static char *popString(int nChars) {                /* get string */
  *charStackTop = 0;
  return charStackTop -= nChars;
}

FILE *parserInputFile;
void switch_to_my_file(FILE *inputfile) /* switches lex input to given file */
{
  parserInputFile = inputfile;
  return;
}

#define GET_INPUT ((PCB).input_code = fgetc(parserInputFile))

int yyparse() {
  parseBasic();
  return !(PCB.exit_flag == AG_SUCCESS_CODE);
}


}