git @ Cat's Eye Technologies Castile / b4188cd
Proper label generation in stack backend. catseye 12 years ago
1 changed file(s) with 41 addition(s) and 23 deletion(s). Raw diff Collapse all Expand all
1717 class Compiler(object):
1818 def __init__(self, out):
1919 self.out = out
20 self.labels = {}
21 self.current_loop_end = None
22
23 def get_label(self, pref):
24 count = self.labels.get(pref, 0)
25 label = '%s_%d' % (pref, count)
26 self.labels[pref] = count + 1
27 return label
2028
2129 def compile(self, ast):
2230 if ast.type == 'Program':
2735 for child in ast.children:
2836 self.compile(child)
2937 self.out.write("""\
38 ; call main
39 push main
40 call
3041 """)
3142 elif ast.type == 'Defn':
3243 self.out.write('%s:\n' % ast.value)
3445 elif ast.type in ('StructDefn', 'Forward'):
3546 pass
3647 elif ast.type == 'FunLit':
37 self.out.write('jmp next\n')
38 self.out.write('funlit:\n')
48 l = self.get_label('past_fun')
49 self.out.write('jmp %s\n' % l)
50 f = self.get_label('fun_lit')
51 self.out.write('%s:\n' % f)
3952 self.compile(ast.children[0])
4053 self.compile(ast.children[1])
41 self.out.write('push nil\n')
42 self.out.write('rts\n')
54 self.out.write('%s:\n' % l)
55 self.out.write('push %s\n' % f)
4356 elif ast.type == 'Args':
4457 for child in ast.children:
4558 self.compile(child)
5265 self.out.write('; push variable %s onto stack\n' % ast.value)
5366 self.compile(ast.children[0])
5467 elif ast.type == 'While':
55 self.out.write('loop_start:\n')
68 start = self.get_label('loop_start')
69 end = self.get_label('loop_end')
70 save = self.current_loop_end
71 self.current_loop_end = end
72 self.out.write('%s:\n' % start)
5673 self.compile(ast.children[0])
57 self.out.write('beq loop_end\n')
74 self.out.write('beq %s\n' % end)
5875 self.compile(ast.children[1])
59 self.out.write('jmp loop_start\n')
60 self.out.write('loop_end:\n')
76 self.out.write('jmp %s\n' % start)
77 self.out.write('%s:\n' % end)
78 self.current_loop_end = self.current_loop_end
6179 elif ast.type == 'Op':
6280 self.compile(ast.children[0])
6381 self.compile(ast.children[1])
7290 self.compile(ast.children[0])
7391 self.out.write('call\n')
7492 elif ast.type == 'If':
75 #~ self.out.write('if (')
76 #~ self.compile(ast.children[0])
77 #~ self.out.write(') then\n')
78 #~ if len(ast.children) == 3: # if-else
79 #~ self.compile(ast.children[1])
80 #~ self.out.write(' else\n')
81 #~ self.compile(ast.children[2])
82 #~ else: # just-if
83 #~ self.compile(ast.children[1])
84 #~ self.out.write('end\n')
85 pass
93 else_part = self.get_label('else_part')
94 end_if = self.get_label('end_if')
95 self.compile(ast.children[0])
96 self.out.write('beq %s\n' % else_part)
97 self.compile(ast.children[1])
98 self.out.write('jmp %s\n' % end_if)
99 self.out.write('%s:\n' % else_part)
100 if len(ast.children) == 3:
101 self.compile(ast.children[2])
102 self.out.write('%s:\n' % end_if)
86103 elif ast.type == 'Return':
87104 self.compile(ast.children[0])
88105 self.out.write('rts\n')
89106 elif ast.type == 'Break':
90 self.out.write('jmp loop_end\n')
107 self.out.write('jmp %s\n' % self.current_loop_end)
91108 elif ast.type == 'Not':
92109 self.compile(ast.children[0])
93110 self.out.write('not\n')
95112 self.out.write('push nil\n')
96113 elif ast.type == 'BoolLit':
97114 if ast.value:
98 self.out.write("push true\n")
115 self.out.write("push -1\n")
99116 else:
100 self.out.write("push false\n")
117 self.out.write("push 0\n")
101118 elif ast.type == 'IntLit':
102119 self.out.write('push %s\n' % ast.value)
103120 elif ast.type == 'StrLit':
104121 self.out.write('push %r\n' % ast.value)
105122 elif ast.type == 'Assignment':
106123 self.compile(ast.children[1])
107 self.out.write('; assign to...')
124 self.out.write('; assign to...\n')
108125 self.compile(ast.children[0])
126 self.out.write('assign\n')
109127 elif ast.type == 'Make':
110128 #~ self.out.write('{')
111129 #~ self.commas(ast.children[1:])