1
0
mirror of https://github.com/catseye/SixtyPical.git synced 2025-04-02 21:31:37 +00:00

Refactor Analyzer. Get analysis tests passing once again.

This commit is contained in:
Chris Pressey 2015-10-21 15:45:14 +01:00
parent 49d07cee5f
commit bb6ad5d3cf
3 changed files with 178 additions and 149 deletions

View File

@ -19,7 +19,7 @@ import traceback
from sixtypical.parser import Parser
from sixtypical.evaluator import Evaluator
from sixtypical.analyzer import analyze_program
from sixtypical.analyzer import Analyzer
from sixtypical.emitter import Emitter, Byte, Word
from sixtypical.compiler import Compiler
@ -50,12 +50,13 @@ if __name__ == '__main__':
for filename in args:
text = open(filename).read()
p = Parser(text)
program = p.program()
parser = Parser(text)
program = parser.program()
if options.analyze:
try:
analyze_program(program)
analyzer = Analyzer()
analyzer.analyze_program(program)
except Exception as e:
if options.traceback:
raise

View File

@ -41,6 +41,10 @@ class IncompatibleConstraintsError(StaticAnalysisError):
pass
class IllegalJumpError(StaticAnalysisError):
pass
class Context():
"""
A location is touched if it was changed (or even potentially
@ -126,153 +130,163 @@ class Context():
self.set_touched(*refs)
self.set_meaningful(*refs)
def analyze_program(program):
assert isinstance(program, Program)
routines = {r.name: r for r in program.routines}
for routine in program.routines:
analyze_routine(routine, routines)
class Analyzer(object):
def analyze_routine(routine, routines):
assert isinstance(routine, Routine)
if routine.block is None:
# it's an extern, that's fine
return
type = routine.location.type
context = Context(type.inputs, type.outputs, type.trashes)
analyze_block(routine.block, context, routines)
for ref in type.outputs:
context.assert_meaningful(ref, exception_class=UninitializedOutputError)
for ref in context.each_touched():
if ref not in type.outputs and ref not in type.trashes:
raise IllegalWriteError(ref.name)
def __init__(self):
self.current_routine = None
self.has_encountered_goto = False
def analyze_program(self, program):
assert isinstance(program, Program)
routines = {r.name: r for r in program.routines}
for routine in program.routines:
self.analyze_routine(routine, routines)
def analyze_block(block, context, routines):
assert isinstance(block, Block)
for i in block.instrs:
analyze_instr(i, context, routines)
def analyze_instr(instr, context, routines):
assert isinstance(instr, Instr)
opcode = instr.opcode
dest = instr.dest
src = instr.src
if opcode == 'ld':
if instr.index:
if src.type == TYPE_BYTE_TABLE and dest.type == TYPE_BYTE:
pass
else:
raise TypeMismatchError((src, dest))
elif src.type != dest.type:
raise TypeMismatchError((src, dest))
context.assert_meaningful(src)
context.set_written(dest, FLAG_Z, FLAG_N)
elif opcode == 'st':
if instr.index:
if src.type == TYPE_BYTE and dest.type == TYPE_BYTE_TABLE:
pass
else:
raise TypeMismatchError((src, dest))
elif src.type != dest.type:
raise TypeMismatchError((src, dest))
context.assert_meaningful(src)
context.set_written(dest)
elif opcode in ('add', 'sub'):
context.assert_meaningful(src, dest, FLAG_C)
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
elif opcode in ('inc', 'dec'):
context.assert_meaningful(dest)
context.set_written(dest, FLAG_Z, FLAG_N)
elif opcode == 'cmp':
context.assert_meaningful(src, dest)
context.set_written(FLAG_Z, FLAG_N, FLAG_C)
elif opcode in ('and', 'or', 'xor'):
context.assert_meaningful(src, dest)
context.set_written(dest, FLAG_Z, FLAG_N)
elif opcode in ('shl', 'shr'):
context.assert_meaningful(dest, FLAG_C)
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C)
elif opcode == 'call':
type = instr.location.type
for ref in type.inputs:
context.assert_meaningful(ref)
def analyze_routine(self, routine, routines):
assert isinstance(routine, Routine)
self.current_routine = routine
self.has_encountered_goto = False
if routine.block is None:
# it's an extern, that's fine
return
type = routine.location.type
context = Context(type.inputs, type.outputs, type.trashes)
self.analyze_block(routine.block, context, routines)
for ref in type.outputs:
context.set_written(ref)
for ref in type.trashes:
context.assert_writeable(ref)
context.set_touched(ref)
context.set_unmeaningful(ref)
elif opcode == 'if':
context1 = context.clone()
context2 = context.clone()
analyze_block(instr.block1, context1, routines)
if instr.block2 is not None:
analyze_block(instr.block2, context2, routines)
# TODO may we need to deal with touched separately here too?
# probably not; if it wasn't meaningful in the first place, it
# doesn't really matter if you modified it or not, coming out.
for ref in context1.each_meaningful():
context2.assert_meaningful(ref, exception_class=InconsistentInitializationError)
for ref in context2.each_meaningful():
context1.assert_meaningful(ref, exception_class=InconsistentInitializationError)
context.set_from(context1)
elif opcode == 'repeat':
# it will always be executed at least once, so analyze it having
# been executed the first time.
analyze_block(instr.block, context, routines)
context.assert_meaningful(ref, exception_class=UninitializedOutputError)
for ref in context.each_touched():
if ref not in type.outputs and ref not in type.trashes:
raise IllegalWriteError(ref.name)
self.current_routine = None
# now analyze it having been executed a second time, with the context
# of it having already been executed.
analyze_block(instr.block, context, routines)
def analyze_block(self, block, context, routines):
assert isinstance(block, Block)
for i in block.instrs:
if self.has_encountered_goto:
raise IllegalJumpError(i)
self.analyze_instr(i, context, routines)
# NB I *think* that's enough... but it might not be?
elif opcode == 'copy':
# check that their types are basically compatible
if src.type == dest.type:
pass
elif isinstance(src.type, ExecutableType) and \
isinstance(dest.type, VectorType):
pass
def analyze_instr(self, instr, context, routines):
assert isinstance(instr, Instr)
opcode = instr.opcode
dest = instr.dest
src = instr.src
if opcode == 'ld':
if instr.index:
if src.type == TYPE_BYTE_TABLE and dest.type == TYPE_BYTE:
pass
else:
raise TypeMismatchError((src, dest))
elif src.type != dest.type:
raise TypeMismatchError((src, dest))
context.assert_meaningful(src)
context.set_written(dest, FLAG_Z, FLAG_N)
elif opcode == 'st':
if instr.index:
if src.type == TYPE_BYTE and dest.type == TYPE_BYTE_TABLE:
pass
else:
raise TypeMismatchError((src, dest))
elif src.type != dest.type:
raise TypeMismatchError((src, dest))
context.assert_meaningful(src)
context.set_written(dest)
elif opcode in ('add', 'sub'):
context.assert_meaningful(src, dest, FLAG_C)
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C, FLAG_V)
elif opcode in ('inc', 'dec'):
context.assert_meaningful(dest)
context.set_written(dest, FLAG_Z, FLAG_N)
elif opcode == 'cmp':
context.assert_meaningful(src, dest)
context.set_written(FLAG_Z, FLAG_N, FLAG_C)
elif opcode in ('and', 'or', 'xor'):
context.assert_meaningful(src, dest)
context.set_written(dest, FLAG_Z, FLAG_N)
elif opcode in ('shl', 'shr'):
context.assert_meaningful(dest, FLAG_C)
context.set_written(dest, FLAG_Z, FLAG_N, FLAG_C)
elif opcode == 'call':
type = instr.location.type
for ref in type.inputs:
context.assert_meaningful(ref)
for ref in type.outputs:
context.set_written(ref)
for ref in type.trashes:
context.assert_writeable(ref)
context.set_touched(ref)
context.set_unmeaningful(ref)
elif opcode == 'if':
context1 = context.clone()
context2 = context.clone()
self.analyze_block(instr.block1, context1, routines)
if instr.block2 is not None:
self.analyze_block(instr.block2, context2, routines)
# TODO may we need to deal with touched separately here too?
# probably not; if it wasn't meaningful in the first place, it
# doesn't really matter if you modified it or not, coming out.
for ref in context1.each_meaningful():
context2.assert_meaningful(ref, exception_class=InconsistentInitializationError)
for ref in context2.each_meaningful():
context1.assert_meaningful(ref, exception_class=InconsistentInitializationError)
context.set_from(context1)
elif opcode == 'repeat':
# it will always be executed at least once, so analyze it having
# been executed the first time.
self.analyze_block(instr.block, context, routines)
# now analyze it having been executed a second time, with the context
# of it having already been executed.
self.analyze_block(instr.block, context, routines)
# NB I *think* that's enough... but it might not be?
elif opcode == 'copy':
# check that their types are basically compatible
if src.type == dest.type:
pass
elif isinstance(src.type, ExecutableType) and \
isinstance(dest.type, VectorType):
pass
else:
raise TypeMismatchError((src, dest))
# if dealing with routines and vectors,
# check that they're not incompatible
if isinstance(src.type, ExecutableType) and \
isinstance(dest.type, VectorType):
if not (src.type.inputs <= dest.type.inputs):
raise IncompatibleConstraintsError(src.type.inputs - dest.type.inputs)
if not (src.type.outputs <= dest.type.outputs):
raise IncompatibleConstraintsError(src.type.outputs - dest.type.outputs)
if not (src.type.trashes <= dest.type.trashes):
raise IncompatibleConstraintsError(src.type.trashes - dest.type.trashes)
context.assert_meaningful(src)
context.set_written(dest)
context.set_touched(REG_A, FLAG_Z, FLAG_N)
context.set_unmeaningful(REG_A, FLAG_Z, FLAG_N)
elif opcode == 'with-sei':
self.analyze_block(instr.block, context, routines)
elif opcode == 'goto':
location = instr.location
type = location.type
if not isinstance(type, ExecutableType):
raise TypeMismatchError(location)
# assert that the dest routine's inputs are all initialized
for ref in type.inputs:
context.assert_meaningful(ref)
# and that this routine's trashes and output constraints are a
# superset of the called routine's
current_type = self.current_routine.location.type
if not (type.outputs <= current_type.outputs):
raise IncompatibleConstraintsError(type.outputs - current_type.outputs)
if not (type.trashes <= current_type.trashes):
raise IncompatibleConstraintsError(type.trashes - current_type.trashes)
self.has_encountered_goto = True
else:
raise TypeMismatchError((src, dest))
# if dealing with routines and vectors,
# check that they're not incompatible
if isinstance(src.type, ExecutableType) and \
isinstance(dest.type, VectorType):
if not (src.type.inputs <= dest.type.inputs):
raise IncompatibleConstraintsError(src.type.inputs - dest.type.inputs)
if not (src.type.outputs <= dest.type.outputs):
raise IncompatibleConstraintsError(src.type.outputs - dest.type.outputs)
if not (src.type.trashes <= dest.type.trashes):
raise IncompatibleConstraintsError(src.type.trashes - dest.type.trashes)
context.assert_meaningful(src)
context.set_written(dest)
context.set_touched(REG_A, FLAG_Z, FLAG_N)
context.set_unmeaningful(REG_A, FLAG_Z, FLAG_N)
elif opcode == 'with-sei':
analyze_block(instr.block, context, routines)
elif opcode == 'goto':
location = instr.location
type = location.type
if not isinstance(type, ExecutableType):
raise TypeMismatchError(location)
# assert that the dest routine's inputs are all initialized
for ref in type.inputs:
context.assert_meaningful(ref)
# and that the called routine's output constraints are a
# superset of this routine's
current_type = self.current_routine.type
if not (current_type.outputs <= type.outputs):
raise IncompatibleConstraintsError(current_type.outputs - type.outputs)
if not (current_type.trashes <= type.trashes):
raise IncompatibleConstraintsError(current_type.trashes - type.trashes)
else:
raise NotImplementedError(opcode)
raise NotImplementedError(opcode)

View File

@ -1149,7 +1149,7 @@ Calling the vector has indeed trashed stuff etc,
| }
? UninitializedOutputError: x
`goto`, if present, must be the final instruction in a routine.
`goto`, if present, must be in tail position (the final instruction in a routine.)
| routine bar trashes x, z, n {
| ld x, 200
@ -1182,6 +1182,20 @@ Calling the vector has indeed trashed stuff etc,
| goto bar
| }
| }
= ok
| routine bar trashes x, z, n {
| ld x, 200
| }
|
| routine main trashes x, z, n {
| ld x, 0
| if z {
| ld x, 1
| goto bar
| }
| ld x, 0
| }
? IllegalJumpError
Can't `goto` a routine that outputs or trashes more than the current routine.
@ -1195,7 +1209,7 @@ Can't `goto` a routine that outputs or trashes more than the current routine.
| ld x, 0
| goto bar
| }
? IllegalWriteError: y
? IncompatibleConstraintsError
| routine bar outputs y trashes z, n {
| ld y, 200
@ -1205,4 +1219,4 @@ Can't `goto` a routine that outputs or trashes more than the current routine.
| ld x, 0
| goto bar
| }
? IllegalWriteError: y
? IncompatibleConstraintsError