-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathservusInterpreter.py
254 lines (201 loc) · 7.86 KB
/
servusInterpreter.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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
from servusYacc import *
from servusSymbolTable import *
import sys
import os
# sys.exit("ERROR")
# ---------------------------- TYPES OF CUDRUPLES ------------------------------
# For arithmetic and logic expressions:
# - Type A : CONST + CONST (len = 5)
# - Type B : CONST + TEMP (len = 5)
# - Type C : TEMP + CONST (len = 5)
# - Type D : TEMP + TEMP (len = 5)
# - Type E : VAR = CONST (len = 4)
# - Type F : VAR = TEMP (len = 4)
# ------------------------------------------------------------------------------
# ---------------------------- GLOBAL VARIABLES --------------------------------
instructionIndex = 0
# ------------------------------------------------------------------------------
def cleanOpernd(operand):
numeric = [int, float]
if type(operand) in numeric:
return operand
else:
tVal = servusSymbolTable.getValue(operand)
return float(tVal)
def computeValue(op, n1, n2):
if op == '+':
return n1 + n2
elif op == '-':
return n1 - n2
elif op == '*':
return n1 * n2
elif op == '/':
return n1 / n2
elif op == '%':
return n1 % n2
elif op == '<':
return n1 < n2
elif op == '<=':
return n1 <= n2
elif op == '>':
return n1 > n2
elif op == '>=':
return n1 >= n2
elif op == '||':
if type(n1) != bool or type(n2) != bool:
sys.exit("Invalid data type for logic comparison")
return n1 or n2
elif op == '&&':
if type(n1) != bool or type(n2) != bool:
sys.exit("Invalid data type for logic comparison")
return n1 and n2
elif op == '!=':
return n1 != n2
elif op == '==':
return n1 == n2
def executeArithmetic(instruction):
typeOfOperation = instruction[-1]
# print("Type of arithmLogic operation:= ", typeOfOperation)
if typeOfOperation == 'A':
operand1 = instruction[1]
operand2 = instruction[2]
# Befor start, clean both operands (a.k.a. get the valu of the variable)
operand1 = cleanOpernd(operand1)
operand2 = cleanOpernd(operand2)
newValue = computeValue(instruction[0], operand1, operand2)
newType = type(instruction[1]) # or type(instruction[2])
elif typeOfOperation == 'B':
operand1 = instruction[1]
operand2 = instruction[2]
# Befor start, clean 1st operand (a.k.a. get the valu of the variable)
operand1 = cleanOpernd(operand1)
newValue = computeValue(instruction[0], operand1, operand2.value)
newType = instruction[2].valueType
elif typeOfOperation == 'C':
operand1 = instruction[1]
operand2 = instruction[2]
# Befor start, clean 2nd operand (a.k.a. get the valu of the variable)
operand2 = cleanOpernd(operand2)
newValue = computeValue(instruction[0], operand1.value, operand2)
newType = instruction[1].valueType
elif typeOfOperation == 'D':
operand1 = instruction[1]
operand2 = instruction[2]
newValue = computeValue(instruction[0], operand1.value, operand2.value)
newType = instruction[1].valueType
instruction[-2].set(newType, newValue)
def executeLogic(instruction):
typeOfOperation = instruction[-1]
newType = bool
if typeOfOperation == 'A':
# print(instruction)
operand1 = instruction[1]
operand2 = instruction[2]
# Befor start, clean both operands (a.k.a. get the valu of the variable)
operand1 = cleanOpernd(operand1)
operand2 = cleanOpernd(operand2)
newValue = computeValue(instruction[0], operand1, operand2)
elif typeOfOperation == 'B':
operand1 = instruction[1]
operand2 = instruction[2]
# Befor start, clean 1st operand (a.k.a. get the valu of the variable)
operand1 = cleanOpernd(operand1)
newValue = computeValue(instruction[0], operand1, operand2.value)
elif typeOfOperation == 'C':
operand1 = instruction[1]
operand2 = instruction[2]
# Befor start, clean 2nd operand (a.k.a. get the valu of the variable)
operand2 = cleanOpernd(operand2)
newValue = computeValue(instruction[0], operand1.value, operand2)
elif typeOfOperation == 'D':
operand1 = instruction[1]
operand2 = instruction[2]
newValue = computeValue(instruction[0], operand1.value, operand2.value)
instruction[-2].set(newType, newValue)
def modifyInstructionIndex(val):
global instructionIndex
instructionIndex = val
def executeInstruction(instruction):
global instructionIndex
opCode = instruction[0]
arithmeticOperators = ['+','-','*','/','%']
logicOperators = ['<','>','<=','>=','!=','==','&&','||']
# print("OpCode:= ", opCode)
if opCode in arithmeticOperators:
executeArithmetic(instruction)
elif opCode == "=": # VAL -> variable
typeOfOperation = instruction[-1]
# symb = servusSymbolTable.get(instruction[2])
symbolToBeAssigned = instruction[2]
# TODO assign a new data-type to the variable, if it changes
if typeOfOperation == 'E':
newValue = servusSymbolTable.getValue(instruction[1])
elif typeOfOperation == 'F':
newValue = instruction[1].value
# symb.val = newValue
servusSymbolTable.setValue(symbolToBeAssigned, newValue)
elif opCode == "++": # A + 1 -> A
symb = servusSymbolTable.getSymbolFromTable(instruction[1])
symb.val += 1
elif opCode in logicOperators:
executeLogic(instruction)
elif opCode == "p":
# tVal = servusSymbolTable.getValue(instruction[1])
tSymbol = servusSymbolTable.getSymbolFromTable(instruction[1])
if tSymbol != None:
# print(instruction[1], '= ', tVal)
tSymbol.printValue()
else:
print(instruction[1][1:-1]) # Remove "" with [1:-1]
print("")
elif opCode == "gF":
if not instruction[1].value:
jumpInstr = instruction[-1]
instructionIndex = jumpInstr
elif opCode == "g":
jumpInstr = instruction[-1]
instructionIndex = jumpInstr
elif opCode == "gT":
if instruction[1].value:
jumpInstr = instruction[-1]
instructionIndex = jumpInstr
elif opCode == "cls":
os.system('cls')
elif opCode == "inp":
newValue = input("")
symbolToBeAssigned = instruction[1]
servusSymbolTable.setValue(symbolToBeAssigned, newValue)
elif opCode == "return":
global stReturn
modifyInstructionIndex(stReturn.pop())
elif opCode == "gSub":
# Store the index of return in stack stReturn
returnIndex = instruction[-1]
appendReturnIndex(returnIndex)
# Move to first cuadruple of the subroutine
modifyInstructionIndex(instruction[1])
# elif opCode == "":
# elif opCode == "":
# elif opCode == "":
def printBeginOfProgram():
# os.system('cls')
print("\n###################################################################")
print("##################### ANFANG DES PROGRAMMS ########################")
print("###################################################################\n")
def printEndOfProgram():
print("\n###################### ENDE DES PROGRAMMS #########################\n")
def executeIntermediateCode():
global instructionIndex
global servusSubroutines
while instructionIndex < len(intermediateCode):
instruction = intermediateCode[instructionIndex]
instructionIndex += 1
if instruction[0] == "start":
printBeginOfProgram()
# instructionIndex = servusSubroutines.get("main")
elif instruction[0] == "end":
printEndOfProgram()
else:
# print("Cuad: ", instructionIndex)
executeInstruction(instruction)
# print(instructionIndex)