-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathyacc.py
109 lines (74 loc) · 1.88 KB
/
yacc.py
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
103
104
105
106
107
108
109
import ply.yacc as yacc
from compiler import (Assignment, Call, DoBlock, Expression, Function, Guards, IfBlock)
from lex import tokens
def p_PROG(p):
"""PROG : FUNC PROG
| FUNC"""
p[0] = [p[1]] + p[2] if len(p) == 3 else [p[1]]
def p_FUNC(p):
"""FUNC : NAME PARMS '=' EXPR
| NAME PARMS GUARDS"""
if len(p) == 5:
p[0] = Function(p[1], p[2], p[4])
else:
p[0] = Function(p[1], p[2], Guards(p[3]))
def p_PARMS(p):
"""PARMS : NAME PARMS
| empty"""
p[0] = [p[1]] + p[2] if len(p) == 3 else []
def p_GUARDS(p):
"""GUARDS : GUARD GUARDS
| GUARD"""
p[0] = [p[1]] + p[2] if len(p) == 3 else [p[1]]
def p_GUARD(p):
"GUARD : '|' EXPR '=' EXPR"
p[0] = (p[2], p[4])
def p_EXPR(p):
"""EXPR : DO '{' STMTS '}'
| IF EXPR THEN EXPR ELSE EXPR
| CALL"""
if len(p) == 5:
p[0] = Expression(DoBlock(p[3]))
elif len(p) == 7:
p[0] = Expression(IfBlock(p[2], p[4], p[6]))
else:
p[0] = Expression(p[1])
def p_STMTS(p):
"""STMTS : STMT ';' STMTS
| EXPR"""
p[0] = [p[1]] + p[3] if len(p) == 4 else [p[1]]
def p_STMT(p):
"""STMT : ASSG
| EXPR"""
p[0] = p[1]
def p_ASSG(p):
"ASSG : NAME ASSIGN EXPR"
p[0] = Assignment(p[1], p[3])
def p_CALL(p):
"CALL : NAME ARGS"
p[0] = Call(p[1], p[2])
def p_ARGS(p):
"""ARGS : ARG ARGS
| empty"""
p[0] = [p[1]] + p[2] if len(p) == 3 else []
def p_ARG_CALL(p):
"ARG : '(' CALL ')'"
p[0] = p[2]
def p_ARG_NAME(p):
"ARG : NAME"
p[0] = {
"type": "NAME",
"value": str(p[1])
}
def p_ARG_CONST(p):
"ARG : CONST"
p[0] = {
"type": "CONST",
"value": int(p[1])
}
def p_empty(p):
"empty :"
pass
def p_error(p):
print("Syntax error in input!")
parser = yacc.yacc()