-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathgenerator.py
102 lines (90 loc) · 3.38 KB
/
generator.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
from typing import List
from ..parser.ast import AST, Node
from ..scanner.tokens import Tokens
from ..semantic import COMPUTATION_NODES, CONSTANT_NODES
class CodeGenerator:
"""
Generates code for the dc language from an AST of the ac language.
"""
def __init__(self, ast: AST):
self.ast = ast
self.generated = list()
def visit_assignment(self, node: Node) -> None:
"""
Visits an assignment node and emits dc code for that assignment
:param node: the assignment node to visit and emit dc code for
"""
self.codegen(node.right())
self.emit("s")
self.emit(f"{node.left().value}") # emit ID
self.emit("0 k")
def visit_computation(self, node: Node) -> None:
"""
Visits a computation node and emits dc code for that computation
:param node: the computation node to visit and emit dc code for
"""
self.codegen(node.left())
self.codegen(node.right())
self.emit("+" if node.type == Tokens.PLUS else "-")
def visit_reference(self, node: Node) -> None:
"""
Visits a reference node and emits the ID of that node
:param node: the reference node to visit and emit dc code for
"""
self.emit("l")
self.emit(node.value)
def visit_print(self, node: Node) -> None:
"""
Visits a print node and emits the value of the symbol referenced in that node
:param node: the print node to visit and emit dc code for
"""
self.emit("l")
self.emit(node.value)
self.emit("p")
self.emit("si")
def visit_convert(self, node: Node) -> None:
"""
Visits a convert node, emit the value of child, and emit dc code to change the
precision level to five decimal places.
:param node: the convert node to visit and emit dc code for
"""
self.emit(node.child().value)
self.emit("5 k")
def visit_constant(self, node: Node) -> None:
"""
Visits a constant node and emits its value in dc code.
:param node: the constant node to visit and emit dc code for
"""
self.emit(node.value)
def codegen(self, node: Node) -> None:
"""
Generates dc code by calling the relevant visitor method for the given node.
:param node: the node to generate dc code for.
"""
for child in node.children:
self.codegen(child)
if node.type == Tokens.ASSIGN:
self.visit_assignment(node)
elif node.type in COMPUTATION_NODES:
self.visit_computation(node)
elif node.type == Tokens.ID:
self.visit_reference(node)
elif node.type == Tokens.PRINT:
self.visit_print(node)
elif node.type == Tokens.CONVERT:
self.visit_convert(node)
elif node.type in CONSTANT_NODES:
self.visit_constant(node)
def emit(self, code: str) -> None:
"""
Append generated code to the list of produced code.
:param code: the code string to append to the list of generated code
"""
self.generated.append(code)
def generate(self) -> List[str]:
"""
Generate dc code from the AST produced by the parser
:return: the list of generated dc code statements
"""
self.codegen(self.ast.root)
return self.generated