Clean up module and abort import if we are unable to find the address

of an opcode.
This commit is contained in:
kris 2019-03-23 22:01:49 +00:00
parent 531a6ae345
commit 13fb9c3125

View File

@ -6,22 +6,29 @@ from typing import Iterator, Tuple
import symbol_table import symbol_table
from machine import Machine from machine import Machine
_op_cmds = [
def _op_cmds():
"""Construct names of player opcodes."""
op_cmds = [
"TERMINATE", "TERMINATE",
"NOP", "NOP",
"ACK", "ACK",
] ]
for tick in range(4, 68, 2): for tick in range(4, 68, 2):
for page in range(32, 64): for page in range(32, 64):
_op_cmds.append("TICK_%d_PAGE_%d" % (tick, page)) op_cmds.append("TICK_%d_PAGE_%d" % (tick, page))
return op_cmds
OpcodeCommand = enum.Enum("OpcodeCommand", _op_cmds)
OpcodeCommand = enum.Enum("OpcodeCommand", _op_cmds())
class Opcode: class Opcode:
"""Base class for opcodes."""
COMMAND = None # type: OpcodeCommand COMMAND = None # type: OpcodeCommand
# Offset of start byte in decoder opcode # Offset of start byte of player opcode implementation
_START = None # type: int _START = None # type: int
def __repr__(self): def __repr__(self):
@ -50,6 +57,7 @@ class Opcode:
class Nop(Opcode): class Nop(Opcode):
"""NOP pad opcode that does nothing except vector to the next one."""
COMMAND = OpcodeCommand.NOP COMMAND = OpcodeCommand.NOP
def __data_eq__(self, other): def __data_eq__(self, other):
@ -57,6 +65,7 @@ class Nop(Opcode):
class Terminate(Opcode): class Terminate(Opcode):
"""Terminates video playback."""
COMMAND = OpcodeCommand.TERMINATE COMMAND = OpcodeCommand.TERMINATE
def __data_eq__(self, other): def __data_eq__(self, other):
@ -64,6 +73,7 @@ class Terminate(Opcode):
class Ack(Opcode): class Ack(Opcode):
"""Instructs player to perform TCP stream + buffer management."""
COMMAND = OpcodeCommand.ACK COMMAND = OpcodeCommand.ACK
def emit_data(self) -> Iterator[int]: def emit_data(self) -> Iterator[int]:
@ -76,6 +86,13 @@ class Ack(Opcode):
class BaseTick(Opcode): class BaseTick(Opcode):
"""Base class for "fat" audio + video opcode.
Each such opcode is specialized for a particular HiRes graphics page,
and speaker duty cycle count. The opcode also stores the provided
content byte at 4 offsets on this graphics page.
"""
def __init__(self, content: int, offsets: Tuple): def __init__(self, content: int, offsets: Tuple):
self.content = content self.content = content
if len(offsets) != 4: if len(offsets) != 4:
@ -90,18 +107,23 @@ class BaseTick(Opcode):
yield from self.offsets yield from self.offsets
TICK_OPCODES = {} def _make_tick_opcodes():
# Dynamically construct classes for each of the tick opcodes.
tick_opcodes = {}
for _tick in range(4, 68, 2): for _tick in range(4, 68, 2):
for _page in range(32, 64): for _page in range(32, 64):
_cls = type( tick_opcodes[(_tick, _page)] = type(
"Tick%dPage%d" % (_tick, _page), "Tick%dPage%d" % (_tick, _page),
(BaseTick,), (BaseTick,),
{ {
"COMMAND": OpcodeCommand["TICK_%d_PAGE_%d" % (_tick, _page)] "COMMAND": OpcodeCommand["TICK_%d_PAGE_%d" % (_tick, _page)]
} }
) )
TICK_OPCODES[(_tick, _page)] = _cls return tick_opcodes
TICK_OPCODES = _make_tick_opcodes()
def _parse_symbol_table(): def _parse_symbol_table():
@ -128,20 +150,28 @@ def _parse_symbol_table():
def _fill_opcode_addresses(): def _fill_opcode_addresses():
"""Populate _START on opcodes from symbol table.""" """Populate _START on opcodes from symbol table."""
_OPCODE_ADDRS = _parse_symbol_table()
_OPCODE_CLASSES = {
OpcodeCommand.TERMINATE: Terminate,
OpcodeCommand.NOP: Nop,
OpcodeCommand.ACK: Ack,
}
for _tick in range(4, 68, 2):
for _page in range(32, 64):
_tickop = OpcodeCommand["TICK_%d_PAGE_%d" % (_tick, _page)]
_OPCODE_CLASSES[_tickop] = TICK_OPCODES[(_tick, _page)]
for op, start in _OPCODE_ADDRS: for op, start in _OPCODE_ADDRS:
cls = _OPCODE_CLASSES[op] cls = _OPCODE_CLASSES[op]
cls._START = start cls._START = start
for op, cls in _OPCODE_CLASSES.items():
if not cls._START:
raise ValueError(
"Unable to find opcode address for %s in player debug symbols"
% op
)
_OPCODE_ADDRS = _parse_symbol_table()
_OPCODE_CLASSES = {
OpcodeCommand.TERMINATE: Terminate,
OpcodeCommand.NOP: Nop,
OpcodeCommand.ACK: Ack,
}
for _tick in range(4, 68, 2):
for _page in range(32, 64):
_tickop = OpcodeCommand["TICK_%d_PAGE_%d" % (_tick, _page)]
_OPCODE_CLASSES[_tickop] = TICK_OPCODES[(_tick, _page)]
_fill_opcode_addresses() _fill_opcode_addresses()