3a92d97ff0
I'm starting with the easy cases. The leftovers need to be looked at a bit more closely. Note that this change _does_ modify the code of the old tests. This is required in order to allow the code to locate the data files in the source directory instead of the current directory, because Kyua automatically changes the latter to a temporary directory. Also note that at least one test is known to be broken here. Actually, the test is not really broken: it's marked as a TODO but unfortunately Kyua's TAP parser currently does not understand that. Will have to be fixed separately.
113 lines
2.0 KiB
Plaintext
113 lines
2.0 KiB
Plaintext
%{
|
|
# include <stdio.h>
|
|
# include <ctype.h>
|
|
|
|
int regs[26];
|
|
int base;
|
|
|
|
int yylex(void);
|
|
static void yyerror(const char *s);
|
|
|
|
%}
|
|
|
|
%start list
|
|
|
|
%token OP_ADD "ADD"
|
|
%token OP_SUB "SUB"
|
|
%token OP_MUL "MUL"
|
|
%token OP_DIV "DIV"
|
|
%token OP_MOD "MOD"
|
|
%token OP_AND "AND"
|
|
|
|
%token DIGIT LETTER
|
|
|
|
%left '|'
|
|
%left '&'
|
|
%left '+' '-'
|
|
%left '*' '/' '%'
|
|
%left UMINUS /* supplies precedence for unary minus */
|
|
|
|
%% /* beginning of rules section */
|
|
|
|
list : /* empty */
|
|
| list stat '\n'
|
|
| list error '\n'
|
|
{ yyerrok ; }
|
|
;
|
|
|
|
stat : expr
|
|
{ printf("%d\n",$1);}
|
|
| LETTER '=' expr
|
|
{ regs[$1] = $3; }
|
|
;
|
|
|
|
expr : '(' expr ')'
|
|
{ $$ = $2; }
|
|
| expr "ADD" expr
|
|
{ $$ = $1 + $3; }
|
|
| expr "SUB" expr
|
|
{ $$ = $1 - $3; }
|
|
| expr "MUL" expr
|
|
{ $$ = $1 * $3; }
|
|
| expr "DIV" expr
|
|
{ $$ = $1 / $3; }
|
|
| expr "MOD" expr
|
|
{ $$ = $1 % $3; }
|
|
| expr "AND" expr
|
|
{ $$ = $1 & $3; }
|
|
| expr '|' expr
|
|
{ $$ = $1 | $3; }
|
|
| "SUB" expr %prec UMINUS
|
|
{ $$ = - $2; }
|
|
| LETTER
|
|
{ $$ = regs[$1]; }
|
|
| number
|
|
;
|
|
|
|
number: DIGIT
|
|
{ $$ = $1; base = ($1==0) ? 8 : 10; }
|
|
| number DIGIT
|
|
{ $$ = base * $1 + $2; }
|
|
;
|
|
|
|
%% /* start of programs */
|
|
|
|
int
|
|
main (void)
|
|
{
|
|
while(!feof(stdin)) {
|
|
yyparse();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
yyerror(const char *s)
|
|
{
|
|
fprintf(stderr, "%s\n", s);
|
|
}
|
|
|
|
int
|
|
yylex(void) {
|
|
/* lexical analysis routine */
|
|
/* returns LETTER for a lower case letter, yylval = 0 through 25 */
|
|
/* return DIGIT for a digit, yylval = 0 through 9 */
|
|
/* all other characters are returned immediately */
|
|
|
|
int c;
|
|
|
|
while( (c=getchar()) == ' ' ) { /* skip blanks */ }
|
|
|
|
/* c is now nonblank */
|
|
|
|
if( islower( c )) {
|
|
yylval = c - 'a';
|
|
return ( LETTER );
|
|
}
|
|
if( isdigit( c )) {
|
|
yylval = c - '0';
|
|
return ( DIGIT );
|
|
}
|
|
return( c );
|
|
}
|