mirror of
https://github.com/KrisKennaway/ii-vision.git
synced 2025-01-02 20:29:38 +00:00
Clean up module and abort import if we are unable to find the address
of an opcode.
This commit is contained in:
parent
531a6ae345
commit
13fb9c3125
@ -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()
|
||||||
|
Loading…
Reference in New Issue
Block a user