-
Notifications
You must be signed in to change notification settings - Fork 0
/
interpreter.py
126 lines (91 loc) · 3.96 KB
/
interpreter.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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
from tokens import Integer,Float,Reserved
class Interpreter:
def __init__(self,tree,base):
self.tree = tree
self.data = base
def read_INT(self,value):
return int(value)
def read_FLT(self,value):
return float(value)
def read_VAR(self,id):
variable = self.data.read(id)
variable_type =variable.type
return getattr(self,f"read_{variable_type}")(variable.value)
def compute_bin(self,left, op ,right):
left_type ="VAR" if str(left.type).startswith("VAR") else str(left.type)
right_type ="VAR" if str(right.type).startswith("VAR") else str(right.type)
if op.value == "=":
left.type = f"VAR({right_type})"
self.data.write(left, right)
return self.data.read_all()
left = getattr(self, f"read_{left_type}")(left.value)
right = getattr(self, f"read_{right_type}")(right.value)
if op.value == "+":
output = left + right
elif op.value == "-":
output = left - right
elif op.value == "*":
output = left * right
elif op.value == "/":
output = left / right
elif op.value == ">":
output = 1 if left > right else 0
elif op.value == ">=":
output = 1 if left >= right else 0
elif op.value == "<":
output = 1 if left < right else 0
elif op.value == "<=":
output = 1 if left <= right else 0
elif op.value == "?=":
output = 1 if left == right else 0
elif op.value == "and":
output = 1 if left and right else 0
elif op.value == "or":
output = 1 if left or right else 0
return Integer(output) if (left_type == "INT" and right_type == "INT") else Float(output)
def compute_unary(self,operator,operand):
operand_type ="VAR" if str(operand.type).startswith("VAR") else str(operand.type)
operand = getattr(self, f"read_{operand_type}")(operand.value)
if operator.value == "+":
return +operand
elif operator.value == "-":
return -operand
elif operator.value == "not":
output = 1 if not operand else 0
return Integer(output) if (operand_type == "INT") else Float(output)
def interpret(self,tree=None):
if tree is None :
tree = self.tree
if isinstance(tree,list):
if isinstance(tree[0],Reserved):
if tree [0].value == "if":
for idx,condition in enumerate(tree[1][0]):
evaluation = self.interpret(condition)
if evaluation.value == 1:
return self.interpret(tree[1][1][idx])
if len(tree[1]) == 3:
return self.interpret(tree[1][2])
else:
return
elif tree[0].value == "while":
condition = self.interpret(tree[1][0])
while condition.value == 1:
print(self.interpret(tree[1][1]))
condition = self.interpret(tree[1][0])
print(condition.value)
if isinstance(tree,list) and len(tree) == 2:
expression = tree[1]
if isinstance(expression,list):
expression = self.interpret(expression)
return self.compute_unary(tree[0],expression)
elif not isinstance(tree,list):
return tree
else:
left_node = tree[0]
if isinstance(left_node, list):
left_node = self.interpret(left_node)
right_node = tree[2]
if isinstance(right_node, list):
right_node = self.interpret(right_node)
operator = tree[1]
return self.compute_bin(left_node, operator ,right_node)