-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathCodeGenerator.java
257 lines (241 loc) · 7.53 KB
/
CodeGenerator.java
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
255
256
257
import analysis.DepthFirstAdapter;
import node.*;
import java.util.HashMap;
/**
* Need to test:
* - UnaryPlus not necessary?
*/
public class CodeGenerator extends DepthFirstAdapter {
private String code = "";
private HashMap<String, Integer> symbolTable = new HashMap<String, Integer>();
private HashMap<String, String> typeTable;
private String type = "";
private int countLabels = 0;
private int compareCounter = 0;
private int breakCounter = 0;
private int stackHeight = 1;
public CodeGenerator(HashMap<String, String> symbolTable) {
typeTable = symbolTable;
}
// Boolean expressions
@Override
public void outAAndExpr(AAndExpr node) {
code += "\tiand\n";
stackHeight--;
type = "boolean";
}
@Override
public void outAOrExpr(AOrExpr node) {
code += "\tior\n";
stackHeight--;
type = "boolean";
}
@Override
public void outAXorExpr(AXorExpr node) {
code += "\tixor\n";
stackHeight--;
type = "boolean";
}
@Override
public void outANotExpr(ANotExpr node) {
int temp = countLabels++;
code += "\tifeq LabelNotTrue"+temp+"\n";
code += "\tbipush 0\n";
code += "\tgoto LabelNotEnd"+temp+"\n";
code += "LabelNotTrue"+temp+":\n";
code += "\tbipush 1\n";
code += "\tLabelNotEnd"+temp+":\n";
type = "boolean";
}
// Arithmetic operations
@Override
public void outAPlusExpr(APlusExpr node) {
code += "\tiadd\n";
stackHeight--;
type = "integer";
}
@Override
public void outAMinusExpr(AMinusExpr node) {
code += "\tisub\n";
stackHeight--;
type = "integer";
}
// Arithmetic expressions
@Override // NEED TO TEST THIS!
public void outAUnaryMinusExpr(AUnaryMinusExpr node) {
code += "\tineg\n";
type = "integer";
}
@Override
public void outAMultExpr(AMultExpr node) {
code += "\timul\n";
stackHeight--;
type = "integer";
}
@Override
public void outADivExpr(ADivExpr node) {
code += "\tidiv\n";
stackHeight--;
type = "integer";
}
@Override
public void outAModExpr(AModExpr node) {
code += "\tirem\n";
type = "integer";
}
// Check identifier
@Override
public void outAIdentifierExpr(AIdentifierExpr node) {
boolean check = true;
String identifier = node.getIdentifier().toString().toLowerCase().replaceAll(" ","");
Node parent = node;
String parentName;
do { // Needed to check, if we are in a declaration-context
parent = parent.parent();
parentName = parent.getClass().getSimpleName().replaceAll(" ","");
if (parentName.equals("ADeclarationExpr")) check = false;
} while (!parentName.equals("AStartExpr"));
if (check) {
code += "\tiload "+symbolTable.get(identifier)+"\n";
stackHeight++;
type = typeTable.get(identifier);
}
}
// Check assignment and save expression in the variable
@Override
public void caseAAssignmentExpr(AAssignmentExpr node) {
node.getExpr().apply(this);
code += "\tistore "+symbolTable.get(node.getIdentifier().toString().toLowerCase().replaceAll(" ",""))+"\n";
stackHeight--;
}
@Override
public void caseANumberExpr(ANumberExpr node) {
int number = Integer.parseInt(node.getNumber().toString().replaceAll(" ", ""));
// if (number > 255)
code += "\tldc "+number+"\n";
// else
// code += "\tbipush "+number+"\n";
stackHeight++;
type = "integer";
}
@Override
public void caseATrueExpr(ATrueExpr node) {
code += "\tbipush 1\n";
stackHeight++;
type = "boolean";
}
@Override
public void caseAFalseExpr(AFalseExpr node) {
code += "\tbipush 0\n";
stackHeight++;
type = "boolean";
}
@Override
public void caseAPrintExpr(APrintExpr node) {
String intOrBoolean = "I";
code += "\tgetstatic java/lang/System/out Ljava/io/PrintStream;\n";
node.getExpr().apply(this);
if (type.equals("boolean")) intOrBoolean = "Z";
code += "\tinvokevirtual java/io/PrintStream/println("+intOrBoolean+")V\n";
stackHeight++;
}
// While loop
@Override
public void caseAWhileExpr(AWhileExpr node) {
int temp = countLabels++;
code += "LabelWhileUp"+temp+":\n";
node.getLeft().apply(this);
code += "\tifeq LabelWhileDown"+temp+"\n";
stackHeight--;
node.getRight().apply(this);
code += "\tgoto LabelWhileUp"+temp+"\n";
code += "LabelWhileDown"+temp+":\n";
code += "LabelBreakDown"+(breakCounter++)+":\n";
}
// Break
@Override
public void caseABreakExpr(ABreakExpr node) {
code += "\tgoto LabelBreakDown"+breakCounter+"\n";
}
// If-then Part
@Override
public void caseAIfThenExpr(AIfThenExpr node) {
int temp = countLabels++;
node.getLeft().apply(this);
code += "\tifeq LabelIfDown"+temp+"\n";
stackHeight--;
node.getRight().apply(this);
code += "LabelIfDown"+temp+":\n";
}
// If-then-else Part
@Override
public void caseAIfThenElseExpr(AIfThenElseExpr node) {
int temp = countLabels++;
node.getIf().apply(this);
code += "\tifeq LabelIfElse"+temp+"\n";
stackHeight--;
node.getThen().apply(this);
code += "\tgoto LabelIfElseEnd"+temp+"\n";
code += "LabelIfElse"+temp+":\n";
node.getElse().apply(this);
code += "LabelIfElseEnd"+temp+":\n";
}
// Comparisons
@Override
public void caseAComparisonExpr(AComparisonExpr node) {
int temp = 0;
node.getLeft().apply(this);
node.getRight().apply(this);
code += "\tisub\n"; // To check with zero
stackHeight--;
temp = compareCounter;
node.getComparison().apply(this); // Add line matching to corresponding symbol
code += "\tbipush 0\n";
stackHeight++;
code += "\tgoto LabelCompEnd"+temp+"\n";
code += "LabelTrue"+temp+":\n";
code += "\tbipush 1\n";
stackHeight++;
code += "LabelCompEnd"+temp+":\n";
}
@Override
public void outAGtExpr(AGtExpr node) {
code += "\tifgt LabelTrue"+(compareCounter++)+"\n";
stackHeight--;
}
@Override
public void outAGeExpr(AGeExpr node) {
code += "\tifge LabelTrue"+(compareCounter++)+"\n";
stackHeight--;
}
@Override
public void outALtExpr(ALtExpr node) {
code += "\tiflt LabelTrue"+(compareCounter++)+"\n";
stackHeight--;
}
@Override
public void outALeExpr(ALeExpr node) {
code += "\tifle LabelTrue"+(compareCounter++)+"\n";
stackHeight--;
}
@Override
public void outANeExpr(ANeExpr node) {
code += "\tifne LabelTrue"+(compareCounter++)+"\n";
stackHeight--;
}
@Override
public void outAEqExpr(AEqExpr node) {
code += "\tifeq LabelTrue"+(compareCounter++)+"\n";
stackHeight--;
}
/********************************* Getter and Setter **************************************/
public HashMap<String, Integer> getSymbolTable() {
return symbolTable;
}
public String getCode() {
return code;
}
public int getStackHeight() {
return stackHeight;
}
}