Parse one line of code.
144 const char END_CHAR =
'#';
145 const int INIT_INDEX = -1;
147 result.word.setField(5);
148 result.parsedType = ParsedType::INSTRUCTION;
149 auto state = hasLocation ? ParseState::START : ParseState::BEFORE_OP;
150 int locationStart = INIT_INDEX,
151 operationStart = INIT_INDEX,
152 addressStart = INIT_INDEX,
153 indexStart = INIT_INDEX,
154 fieldStart = INIT_INDEX,
155 commentStart = INIT_INDEX,
156 defaultField = INIT_INDEX;
157 std::unordered_map<std::string, AtomicValue> emptyDict;
158 for (
int i = 0; i <= static_cast<int>(line.size()); ++i) {
159 char ch = i < static_cast<int>(line.size()) ? line[i] : END_CHAR;
161 case ParseState::START:
164 state = ParseState::BEFORE_OP;
165 }
else if (ch ==
'*') {
167 state = ParseState::COMMENT;
168 result.parsedType = ParsedType::EMPTY;
170 }
else if (ch == END_CHAR) {
171 state = ParseState::END;
172 result.parsedType = ParsedType::EMPTY;
173 }
else if (isalnum(ch)) {
175 state = ParseState::LOC;
178 throw ParseError(i,
"Unexpected character encountered while parsing location: " + std::string(1, ch));
182 case ParseState::LOC:
184 state = ParseState::BEFORE_OP;
185 result.rawLocation = line.substr(locationStart, i - locationStart);
186 }
else if (!isalnum(ch)) {
187 throw ParseError(i,
"Unexpected character encountered while parsing location: " + std::string(1, ch));
191 case ParseState::BEFORE_OP:
194 }
else if (ch == END_CHAR) {
195 if (locationStart != INIT_INDEX) {
196 throw ParseError(i,
"No operation found after location");
198 state = ParseState::END;
199 result.parsedType = ParsedType::EMPTY;
200 }
else if (isalnum(ch)) {
201 state = ParseState::OP;
204 throw ParseError(i,
"Unexpected character encountered while finding operation: " + std::string(1, ch));
209 if (ch ==
' ' || ch == END_CHAR) {
210 result.operation = line.substr(operationStart, i - operationStart);
214 state = ParseState::BEFORE_ADDRESS;
216 state = ParseState::BEFORE_COMMENT;
219 state = ParseState::END;
221 if (operation == Instructions::INVALID) {
222 throw ParseError(i,
"Unknown operation: " + result.operation);
223 }
else if (operation <= Instructions::LAST) {
224 result.word.setOperation(
static_cast<uint8_t
>(operation));
227 result.parsedType = ParsedType::PSEUDO;
228 result.word.setOperation(
static_cast<uint8_t
>(operation - Instructions::PSEUDO));
229 if (operation == Instructions::ALF) {
231 result.rawAddress =
" ";
233 for (
int shift = 0; shift < 5 && i < static_cast<int>(line.size()); ++shift) {
234 result.rawAddress[shift] = line[++i];
236 int32_t charsValue = ComputerWord(result.rawAddress).value();
238 if (i <
static_cast<int>(line.size())) {
239 state = ParseState::BEFORE_COMMENT;
241 state = ParseState::END;
245 }
else if (!isalnum(ch)) {
246 throw ParseError(i,
"Unexpected character encountered while parsing operation: " + std::string(1, ch));
250 case ParseState::BEFORE_ADDRESS:
253 }
else if (ch == END_CHAR) {
254 state = ParseState::END;
256 state = ParseState::ADDRESS;
259 throw ParseError(i,
"Unexpected character encountered while finding address: " + std::string(1, ch));
263 case ParseState::ADDRESS:
264 if (ch ==
' ' || ch ==
',' || ch ==
'(' || ch == END_CHAR) {
266 state = ParseState::BEFORE_COMMENT;
267 }
else if (ch ==
',') {
268 state = ParseState::BEFORE_INDEX;
269 }
else if (ch ==
'(') {
270 state = ParseState::FIELD_OPEN;
272 state = ParseState::END;
274 result.rawAddress = line.substr(addressStart, i - addressStart);
276 result.address.parse(result.rawAddress, lineSymbol);
277 }
catch (
const ExpressionError& e) {
278 throw ParseError(addressStart + e.index(), e.what());
280 result.evaluateAddress(emptyDict);
282 throw ParseError(i,
"Unexpected character encountered while parsing address: " + std::string(1, ch));
286 case ParseState::BEFORE_INDEX:
288 state = ParseState::INDEX;
290 }
else if (ch == END_CHAR) {
291 throw ParseError(i,
"No index found after comma");
293 throw ParseError(i,
"Unexpected character encountered while finding index: " + std::string(1, ch));
297 case ParseState::INDEX:
298 if (ch ==
' ' || ch ==
'(' || ch == END_CHAR) {
300 state = ParseState::BEFORE_COMMENT;
301 }
else if (ch ==
'(') {
302 state = ParseState::FIELD_OPEN;
304 state = ParseState::END;
306 result.rawIndex = line.substr(indexStart, i - indexStart);
308 result.index.parse(result.rawIndex, lineSymbol);
309 }
catch (
const ExpressionError& e) {
310 throw ParseError(addressStart + e.index(), e.what());
312 result.evaluateIndex(emptyDict);
314 throw ParseError(i,
"Unexpected character encountered while parsing index: " + std::string(1, ch));
318 case ParseState::FIELD_OPEN:
320 state = ParseState::FIELD;
323 throw ParseError(i,
"Unexpected character encountered "
324 "while parsing modification: " + std::string(1, ch));
328 case ParseState::FIELD:
330 state = ParseState::FIELD_CLOSE;
331 result.rawField = line.substr(fieldStart, i - fieldStart);
333 result.field.parse(result.rawField, lineSymbol);
334 }
catch (
const ExpressionError& e) {
335 throw ParseError(addressStart + e.index(), e.what());
337 result.evaluateField(emptyDict);
339 throw ParseError(i,
"Unexpected character encountered while parsing index: " + std::string(1, ch));
343 case ParseState::FIELD_CLOSE:
344 if (ch ==
' ' || ch == END_CHAR) {
346 state = ParseState::BEFORE_COMMENT;
348 state = ParseState::END;
351 throw ParseError(i,
"Unexpected character encountered while parsing field: " + std::string(1, ch));
355 case ParseState::BEFORE_COMMENT:
356 if (ch == END_CHAR) {
357 state = ParseState::END;
358 }
else if (ch !=
' ') {
359 state = ParseState::COMMENT;
364 case ParseState::COMMENT:
365 if (ch == END_CHAR) {
366 state = ParseState::END;
367 result.comment = line.substr(commentStart, i - commentStart);
371 case ParseState::END:
375 if (result.rawField.empty()) {
376 if (result.word.operation() == Instructions::MOVE) {
379 }
else if (result.word.operation() == Instructions::NOP) {
384 if (defaultField >= 0) {
385 result.word.setField(defaultField);
387 assert(state == ParseState::END);