yacc example taken from lex & yacc simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17...

16
YACC Example • Taken from LEX & YACC • Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

Upload: calvin-burke

Post on 03-Jan-2016

220 views

Category:

Documents


0 download

TRANSCRIPT

Page 1: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

YACC Example

• Taken from LEX & YACC• Simple calculatora = 4 + 6aa=10b = 7c = a + bcc = 17pressure = (78 + 34) * 16.4$

Page 2: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

Grammarexpression ::= expression '+' term |

expression '-' term |

term

term ::= term '*' factor |

term '/' factor |

factor

factor ::= '(' expression ')' |

'-' factor |

NUMBER |

NAME

Page 3: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

parser.h

Page 4: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

/*

*Header for calculator program

*/

#define NSYMS 20 /* maximum number

of symbols */

struct symtab {

char *name;

double value;

} symtab[NSYMS];

struct symtab *symlook();

parser.h

name value0

name value1

name value2

name value3

name value4

name value5

name value6

name value7

name value8

name value9

name value10

name value11

name value12

name value13

name value14

Page 5: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

parser.y

Page 6: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

%{

#include "parser.h"

#include <string.h>

%}

%union {

double dval;

struct symtab *symp;

}

%token <symp> NAME

%token <dval> NUMBER

%type <dval> expression

%type <dval> term

%type <dval> factor

%%parser.y

Page 7: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

statement_list: statement '\n'

| statement_list statement '\n‘

;

statement: NAME '=' expression { $1->value = $3; }

| expression { printf("= %g\n", $1); }

;

expression: expression '+' term { $$ = $1 + $3; }

| expression '-' term { $$ = $1 - $3; }

term

;

parser.y

Page 8: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

term: term '*' factor { $$ = $1 * $3; }

| term '/' factor { if($3 == 0.0)

yyerror("divide by zero");

else

$$ = $1 / $3;

}

| factor

;

factor: '(' expression ')' { $$ = $2; }

| '-' factor { $$ = -$2; }

| NUMBER

| NAME { $$ = $1->value; }

;

%% parser.y

Page 9: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

/* look up a symbol table entry, add if not present */

struct symtab *symlook(char *s) {

char *p;

struct symtab *sp;

for(sp = symtab; sp < &symtab[NSYMS]; sp++) {

/* is it already here? */

if(sp->name && !strcmp(sp->name, s))

return sp;

if(!sp->name) { /* is it free */

sp->name = strdup(s);

return sp;

}

/* otherwise continue to next */

}

yyerror("Too many symbols");

exit(1); /* cannot continue */

} /* symlook */parser.y

Page 10: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

yyerror(char *s)

{

printf( "yyerror: %s\n", s);

}

parser.y

Page 11: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

typedef union

{

double dval;

struct symtab *symp;

} YYSTYPE;

extern YYSTYPE yylval;

# define NAME 257

# define NUMBER 258

y.tab.h

Page 12: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

calclexer.l

Page 13: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

%{

#include "y.tab.h"

#include "parser.h"

#include <math.h>

%}

%%

calclexer.l

Page 14: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

%%

([0-9]+|([0-9]*\.[0-9]+)([eE][-+]?[0-9]+)?) {

yylval.dval = atof(yytext);

return NUMBER;

}

[ \t] ; /* ignore white space */

[A-Za-z][A-Za-z0-9]* { /* return symbol pointer */

yylval.symp = symlook(yytext);

return NAME;

}

"$" { return 0; /* end of input */ }

\n|. return yytext[0];

%%calclexer.l

Page 15: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

Makefile

Page 16: YACC Example Taken from LEX & YACC Simple calculator a = 4 + 6 a a=10 b = 7 c = a + b c c = 17 pressure = (78 + 34) * 16.4 $

MakefileLEX = lexYACC = yaccCC = gcc

calcu: y.tab.o lex.yy.o$(CC) -o calcu y.tab.o lex.yy.o -ly -ll

y.tab.c y.tab.h: parser.y$(YACC) -d parser.y

y.tab.o: y.tab.c parser.h$(CC) -c y.tab.c

lex.yy.o: y.tab.h lex.yy.c$(CC) -c lex.yy.c

lex.yy.c: calclexer.l parser.h$(LEX) calclexer.l

clean:rm *.orm *.crm calcu