-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathgrammar2.y
102 lines (86 loc) · 2.75 KB
/
grammar2.y
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
%{
#include <iostream>
#include <string>
#include <cmath>
#include <FlexLexer.h>
%}
%require "3.7.4"
%language "C++"
%defines "Parser2.hpp"
%output "Parser2.cpp"
%define api.parser.class {Parser}
%define api.namespace {calc}
%define api.value.type variant
%parse-param {Scanner* scanner}
%code requires
{
namespace calc {
class Scanner;
} // namespace calc
} // %code requires
%code
{
#include "Scanner2.hpp"
#define yylex(x) scanner->lex(x)
}
%token EOL LPAREN RPAREN
%token <long long> INT
%token <double> FLT
%token <char> INTVAR FLTVAR
%nterm <long long> iexp
%nterm <double> fexp
%nonassoc ASSIGN
%left PLUS MINUS
%left MULTIPLY DIVIDE MODULO
%precedence UMINUS
%precedence FACTORIAL
%right EXPONENT
%code
{
namespace calc {
long long ivars['Z' - 'A' + 1];
double fvars['z' - 'a' + 1];
long long factorial(long long n) {
if (n < 2) {
return 1;
}
return n * factorial(n - 1);
}
} // namespace calc
} // %code
%%
lines : %empty
| lines line
;
line : EOL { std::cerr << "Read an empty line.\n"; }
| iexp EOL { std::cout << $1 << '\n'; }
| fexp EOL { std::cout << $1 << '\n'; }
| INTVAR ASSIGN iexp EOL { ivars[$1 - 'A'] = $3; }
| FLTVAR ASSIGN fexp EOL { fvars[$1 - 'a'] = $3; }
| error EOL { yyerrok; }
;
iexp : INT { $$ = $1; }
| iexp PLUS iexp { $$ = $1 + $3; }
| iexp MINUS iexp { $$ = $1 - $3; }
| iexp MULTIPLY iexp { $$ = $1 * $3; }
| iexp DIVIDE iexp { $$ = $1 / $3; }
| iexp MODULO iexp { $$ = $1 % $3; }
| MINUS iexp %prec UMINUS { $$ = -$2; }
| iexp FACTORIAL { $$ = factorial($1); }
| LPAREN iexp RPAREN { $$ = $2; }
| INTVAR { $$ = ivars[$1 - 'A']; }
;
fexp : FLT { $$ = $1; }
| fexp PLUS fexp { $$ = $1 + $3; }
| fexp MINUS fexp { $$ = $1 - $3; }
| fexp MULTIPLY fexp { $$ = $1 * $3; }
| fexp DIVIDE fexp { $$ = $1 / $3; }
| fexp EXPONENT fexp { $$ = pow($1, $3); }
| MINUS fexp %prec UMINUS { $$ = -$2; }
| LPAREN fexp RPAREN { $$ = $2; }
| FLTVAR { $$ = fvars[$1 - 'a']; }
;
%%
void calc::Parser::error(const std::string& msg) {
std::cerr << msg << '\n';
}