-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathParser.swift
180 lines (157 loc) · 4.93 KB
/
Parser.swift
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
//
// Parser.swift
// hasm
//
// Created by Tiago Lopes on 09/11/24.
//
struct Parser {
func instructions(from lines: [String]) throws -> [Instruction] {
return try lines.map(parse)
}
func parse(_ line: String) throws -> Instruction {
guard let char = line.first else {
throw Error.invalid(line: line)
}
switch char {
case "@":
return try parseAddress(line)
case "(":
return try parseSymbol(line)
default:
return try parseComputation(line)
}
}
private func parseAddress(_ line: String) throws -> Instruction {
guard line.count >= 2 else { // Accounts for @ sign
throw Error.invalid(address: line)
}
let valStartIndex = line.index(after: line.startIndex)
let val = String(line[valStartIndex...])
try validate(addressOrVariable: val)
return Instructions.Address(
val: val
)
}
private func parseComputation(_ line: String) throws -> Instruction {
let function: String
let destination: String?
let jump: String?
if let equalsIndex = line.firstIndex(of: "=") {
destination = String(line[..<equalsIndex])
let functionStartIndex = line.index(after: equalsIndex)
function = String(line[functionStartIndex...])
jump = nil
} else if let semicolonIndex = line.firstIndex(of: ";") {
destination = nil
function = String(line[..<semicolonIndex])
let jumpStartIndex = line.index(after: semicolonIndex)
jump = String(line[jumpStartIndex...])
} else {
throw Error.invalid(cInstruction: line)
}
try validate(function: function)
if let destination { try validate(destination: destination) }
if let jump { try validate(jump: jump) }
return Instructions.Computation(
function: function,
destination: destination,
jump: jump
)
}
private func parseSymbol(_ line: String) throws -> Instruction {
guard line.count >= 3, // (, some min identifier, )
line[line.startIndex] == "(" &&
line[line.index(before: line.endIndex)] == ")"
else {
throw Error.invalid(symbol: line)
}
var symbol = line
symbol.remove(at: symbol.startIndex)
symbol.remove(at: symbol.index(before: symbol.endIndex))
try validate(symbol: symbol)
return Instructions.Symbol(
name: symbol
)
}
}
extension Parser {
private func validate(addressOrVariable: String) throws {
guard addressOrVariable.isValidAddress ||
addressOrVariable.isValidLabel else {
throw Error.invalid(address: addressOrVariable)
}
}
private func validate(destination: String) throws {
switch destination {
case "A": break
case "D": break
case "M": break
case "AM": break
case "AD": break
case "MD": break
case "AMD": break
default: throw Error.invalid(destination: destination)
}
}
private func validate(jump: String) throws {
switch jump {
case "JGT": break
case "JEQ": break
case "JGE": break
case "JLT": break
case "JNE": break
case "JLE": break
case "JMP": break
default: throw Error.invalid(jump: jump)
}
}
private func validate(function: String) throws {
switch function {
case "0": break
case "1": break
case "-1": break
case "D": break
case "A": break
case "!D": break
case "!A": break
case "-D": break
case "-A": break
case "D+1": break
case "A+1": break
case "D-1": break
case "A-1": break
case "D+A": break
case "D-A": break
case "A-D": break
case "D&A": break
case "D|A": break
case "M": break
case "!M": break
case "-M": break
case "M+1": break
case "M-1": break
case "D+M": break
case "D-M": break
case "M-D": break
case "D&M": break
case "D|M": break
default: throw Error.invalid(function: function)
}
}
private func validate(symbol: String) throws {
guard symbol.isValidLabel else {
throw Error.invalid(symbol: symbol)
}
}
}
extension Parser {
enum Error: Swift.Error {
case invalid(line: String)
case invalid(address: String)
case invalid(cInstruction: String)
case invalid(destination: String)
case invalid(function: String)
case invalid(jump: String)
case invalid(symbol: String)
}
}