278 | 278 |
def encountered_gotos(self):
|
279 | 279 |
return self._gotos_encountered
|
280 | 280 |
|
|
281 |
def assert_types_for_read_table(self, instr, src, dest, type_):
|
|
282 |
if (not TableType.is_a_table_type(src.ref.type, type_)) or (not dest.type == type_):
|
|
283 |
raise TypeMismatchError(instr, '{} and {}'.format(src.ref.name, dest.name))
|
|
284 |
self.assert_meaningful(src, src.index)
|
|
285 |
self.assert_in_range(src.index, src.ref)
|
|
286 |
|
|
287 |
def assert_types_for_update_table(self, instr, dest, type_):
|
|
288 |
if not TableType.is_a_table_type(dest.ref.type, type_):
|
|
289 |
raise TypeMismatchError(instr, '{}'.format(dest.ref.name))
|
|
290 |
self.assert_meaningful(dest.index)
|
|
291 |
self.assert_in_range(dest.index, dest.ref)
|
|
292 |
self.set_written(dest.ref)
|
|
293 |
|
281 | 294 |
|
282 | 295 |
class Analyzer(object):
|
283 | 296 |
|
|
385 | 398 |
|
386 | 399 |
if opcode == 'ld':
|
387 | 400 |
if isinstance(src, IndexedRef):
|
388 | |
if TableType.is_a_table_type(src.ref.type, TYPE_BYTE) and dest.type == TYPE_BYTE:
|
389 | |
pass
|
390 | |
else:
|
391 | |
raise TypeMismatchError(instr, '{} and {}'.format(src.ref.name, dest.name))
|
392 | |
context.assert_meaningful(src, src.index)
|
393 | |
context.assert_in_range(src.index, src.ref)
|
|
401 |
context.assert_types_for_read_table(instr, src, dest, TYPE_BYTE)
|
394 | 402 |
elif isinstance(src, IndirectRef):
|
395 | 403 |
# copying this analysis from the matching branch in `copy`, below
|
396 | 404 |
if isinstance(src.ref.type, PointerType) and dest.type == TYPE_BYTE:
|
|
406 | 414 |
context.set_written(dest, FLAG_Z, FLAG_N)
|
407 | 415 |
elif opcode == 'st':
|
408 | 416 |
if isinstance(dest, IndexedRef):
|
409 | |
if src.type == TYPE_BYTE and TableType.is_a_table_type(dest.ref.type, TYPE_BYTE):
|
410 | |
pass
|
411 | |
else:
|
412 | |
raise TypeMismatchError(instr, (src, dest))
|
413 | |
context.assert_meaningful(dest.index)
|
414 | |
context.assert_in_range(dest.index, dest.ref)
|
415 | |
context.set_written(dest.ref)
|
|
417 |
if src.type != TYPE_BYTE:
|
|
418 |
raise TypeMismatchError(instr, (src, dest))
|
|
419 |
context.assert_types_for_update_table(instr, dest, TYPE_BYTE)
|
416 | 420 |
elif isinstance(dest, IndirectRef):
|
417 | 421 |
# copying this analysis from the matching branch in `copy`, below
|
418 | 422 |
if isinstance(dest.ref.type, PointerType) and src.type == TYPE_BYTE:
|
|
429 | 433 |
context.assert_meaningful(src)
|
430 | 434 |
elif opcode == 'add':
|
431 | 435 |
context.assert_meaningful(src, dest, FLAG_C)
|
432 | |
if src.type == TYPE_BYTE:
|
|
436 |
if isinstance(src, IndexedRef):
|
|
437 |
context.assert_types_for_read_table(instr, src, dest, TYPE_BYTE)
|
|
438 |
elif src.type == TYPE_BYTE:
|
433 | 439 |
self.assert_type(TYPE_BYTE, src, dest)
|
434 | |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
|
435 | 440 |
else:
|
436 | 441 |
self.assert_type(TYPE_WORD, src)
|
437 | 442 |
if dest.type == TYPE_WORD:
|
438 | |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
|
439 | 443 |
context.set_touched(REG_A)
|
440 | 444 |
context.set_unmeaningful(REG_A)
|
441 | 445 |
elif isinstance(dest.type, PointerType):
|
442 | |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
|
443 | 446 |
context.set_touched(REG_A)
|
444 | 447 |
context.set_unmeaningful(REG_A)
|
445 | 448 |
else:
|
446 | 449 |
self.assert_type(TYPE_WORD, dest)
|
|
450 |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
|
447 | 451 |
context.invalidate_range(dest)
|
448 | 452 |
elif opcode == 'sub':
|
449 | 453 |
context.assert_meaningful(src, dest, FLAG_C)
|
450 | |
if src.type == TYPE_BYTE:
|
|
454 |
if isinstance(src, IndexedRef):
|
|
455 |
context.assert_types_for_read_table(instr, src, dest, TYPE_BYTE)
|
|
456 |
elif src.type == TYPE_BYTE:
|
451 | 457 |
self.assert_type(TYPE_BYTE, src, dest)
|
452 | |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
|
453 | 458 |
else:
|
454 | 459 |
self.assert_type(TYPE_WORD, src, dest)
|
455 | |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
|
456 | 460 |
context.set_touched(REG_A)
|
457 | 461 |
context.set_unmeaningful(REG_A)
|
458 | |
context.invalidate_range(dest)
|
459 | |
elif opcode in ('inc', 'dec'):
|
460 | |
self.assert_type(TYPE_BYTE, dest)
|
461 | |
context.assert_meaningful(dest)
|
462 | |
context.set_written(dest, FLAG_Z, FLAG_N)
|
|
462 |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
|
463 | 463 |
context.invalidate_range(dest)
|
464 | 464 |
elif opcode == 'cmp':
|
465 | |
self.assert_type(TYPE_BYTE, src, dest)
|
466 | 465 |
context.assert_meaningful(src, dest)
|
|
466 |
if isinstance(src, IndexedRef):
|
|
467 |
context.assert_types_for_read_table(instr, src, dest, TYPE_BYTE)
|
|
468 |
else:
|
|
469 |
self.assert_type(TYPE_BYTE, src, dest)
|
467 | 470 |
context.set_written(FLAG_Z, FLAG_N, FLAG_C)
|
468 | 471 |
elif opcode == 'and':
|
469 | |
self.assert_type(TYPE_BYTE, src, dest)
|
|
472 |
if isinstance(src, IndexedRef):
|
|
473 |
context.assert_types_for_read_table(instr, src, dest, TYPE_BYTE)
|
|
474 |
else:
|
|
475 |
self.assert_type(TYPE_BYTE, src, dest)
|
470 | 476 |
context.assert_meaningful(src, dest)
|
471 | 477 |
context.set_written(dest, FLAG_Z, FLAG_N)
|
472 | 478 |
# If you AND the A register with a value V, the resulting value of A
|
473 | 479 |
# cannot exceed the value of V; i.e. the maximum value of A becomes
|
474 | 480 |
# the maximum value of V.
|
475 | |
context.set_top_of_range(dest, context.get_top_of_range(src))
|
|
481 |
if not isinstance(src, IndexedRef):
|
|
482 |
context.set_top_of_range(dest, context.get_top_of_range(src))
|
476 | 483 |
elif opcode in ('or', 'xor'):
|
477 | |
self.assert_type(TYPE_BYTE, src, dest)
|
|
484 |
if isinstance(src, IndexedRef):
|
|
485 |
context.assert_types_for_read_table(instr, src, dest, TYPE_BYTE)
|
|
486 |
else:
|
|
487 |
self.assert_type(TYPE_BYTE, src, dest)
|
478 | 488 |
context.assert_meaningful(src, dest)
|
479 | 489 |
context.set_written(dest, FLAG_Z, FLAG_N)
|
480 | 490 |
context.invalidate_range(dest)
|
|
491 |
elif opcode in ('inc', 'dec'):
|
|
492 |
context.assert_meaningful(dest)
|
|
493 |
if isinstance(dest, IndexedRef):
|
|
494 |
context.assert_types_for_update_table(instr, dest, TYPE_BYTE)
|
|
495 |
context.set_written(dest.ref, FLAG_Z, FLAG_N)
|
|
496 |
#context.invalidate_range(dest)
|
|
497 |
else:
|
|
498 |
self.assert_type(TYPE_BYTE, dest)
|
|
499 |
context.set_written(dest, FLAG_Z, FLAG_N)
|
|
500 |
context.invalidate_range(dest)
|
481 | 501 |
elif opcode in ('shl', 'shr'):
|
482 | |
self.assert_type(TYPE_BYTE, dest)
|
483 | 502 |
context.assert_meaningful(dest, FLAG_C)
|
484 | |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C)
|
485 | |
context.invalidate_range(dest)
|
|
503 |
if isinstance(dest, IndexedRef):
|
|
504 |
context.assert_types_for_update_table(instr, dest, TYPE_BYTE)
|
|
505 |
context.set_written(dest.ref, FLAG_Z, FLAG_N, FLAG_C)
|
|
506 |
#context.invalidate_range(dest)
|
|
507 |
else:
|
|
508 |
self.assert_type(TYPE_BYTE, dest)
|
|
509 |
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C)
|
|
510 |
context.invalidate_range(dest)
|
486 | 511 |
elif opcode == 'call':
|
487 | 512 |
type = instr.location.type
|
488 | 513 |
if isinstance(type, VectorType):
|