38 case ParsedType::EMPTY: os <<
"EMPTY";
break;
39 case ParsedType::INSTRUCTION: os <<
"INSTRUCTION";
break;
40 case ParsedType::PSEUDO: os <<
"PSEUDO";
break;
46 if (
rawAddress.length() > 0 && !evaluateAddress(constants)) {
49 if (
rawIndex.length() > 0 && !evaluateIndex(constants)) {
52 if (
rawField.length() > 0 && !evaluateField(constants)) {
58 bool ParsedResult::evaluateAddress(
const std::unordered_map<std::string, AtomicValue>& constants, int32_t index) {
64 throw ParseError(
index,
"Address can not be represented in 2 bytes: " + std::to_string(value));
70 bool ParsedResult::evaluateIndex(
const std::unordered_map<std::string, AtomicValue>& constants, int32_t column) {
75 if (value < 0 || 6 < value) {
76 throw ParseError(column,
"Invalid index value: " + std::to_string(value));
82 bool ParsedResult::evaluateField(
const std::unordered_map<std::string, AtomicValue>& constants, int32_t index) {
88 if (defaultField >= 0 && value != defaultField) {
89 throw ParseError(
index,
"The given field value does not match the default one: " +
90 std::to_string(value) +
" != " + std::to_string(defaultField));
92 if (value < 0 || 64 <= value) {
93 throw ParseError(
index,
"Invalid field value: " + std::to_string(value));
130 out <<
',' << result.
index;
137 out <<
'(' << result.
field <<
')';
144 const char END_CHAR =
'#';
145 const int INIT_INDEX = -1;
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;
170 }
else if (ch == END_CHAR) {
171 state = ParseState::END;
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;
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) {
223 }
else if (operation <= Instructions::LAST) {
228 result.
word.
setOperation(
static_cast<uint8_t
>(operation - Instructions::PSEUDO));
229 if (operation == Instructions::ALF) {
233 for (
int shift = 0; shift < 5 && i < static_cast<int>(line.size()); ++shift) {
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);
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);
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);
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:
384 if (defaultField >= 0) {
387 assert(state == ParseState::END);