Pluggable tracing disassembler with CPU, renderer and environment extension points.
Project description
Dasmos
A pluggable tracing disassembler for retro CPUs, version 0.1.2.
From Ancient Greek δασμός (dasmós, "division"), from δαίω (daíō, "to divide, share").
Dasmos is a ground-up rewrite and reimagining of a heavily modified fork of py8dis — Steven Flintham's original programmable tracing disassembler for the 6502 family. Dasmos owes the whole core idea to Steven and to the py8dis project; this project organises a tracing disassembler as a core algorithm customised through plug-in extensions which provide knowledge of CPUs, different assembly syntaxes, and target environments. The core of the essential design vocabulary — driver scripts, traced classification, label/comment/banner annotations — is all inspired by py8dis.
Driver scripts written for py8dis can be ported automatically to
Dasmos with the bundled scripts/py8dis2dasmos.py.
Install
The
uvanduvxcommands shown below come from Astral's uv. If you don't have it yet, see the uv installation guide — one-line installers are available for macOS, Linux, and Windows.
For one-shot CLI use, no install needed — uvx fetches and runs in
a transient environment:
uvx dasmos disassemble myrom.bin --load-addr '&8000'
To add dasmos to a project (required for driver scripts that
import dasmos):
uv add dasmos
Or with pip:
pip install dasmos
Dasmos's round-trip / parity tests assemble back to bytes via
beebasm; beebasm on PATH
(or via the BEEBASM env var) activates them. CI builds beebasm
from source per matrix cell so the round-trip is part of the gate.
Programmatic API
Every CLI capability is also reachable through the package. The typical driver-script flow is: pick a CPU plug-in, load a binary, register entry points / labels / classifications / annotations, disassemble, then render via a renderer plug-in.
from dasmos import Disassembler, Align
d = Disassembler.create(cpu="nmos6502")
d.load("rom.bin", 0x8000)
d.entry(0x8000, name="start")
d.label(0x8006, "show", description="Display routine")
d.comment(0x8000, "Entry point.")
d.comment(0x8000, "magic", align=Align.INLINE)
ir = d.disassemble()
print(str(ir.render("beebasm")))
That produces beebasm-assemblable source. Re-assembling it via the
real beebasm binary yields a binary byte-identical to the input
— the load-bearing round-trip property Dasmos's test suite
exercises against real Acorn ROMs (the 8 KB Econet Bridge and the
2 KB-mapped 6502 Tube Client both round-trip end-to-end with full
py8dis annotation-content parity).
CLI
$ dasmos --help
Usage: dasmos [OPTIONS] COMMAND [ARGS]...
A pluggable tracing disassembler.
Options:
--version Show the version and exit.
--help Show this message and exit.
Commands:
describe-cpu Describe a specific CPU plug-in.
describe-environment Describe a specific environment plug-in.
describe-renderer Describe a specific renderer plug-in.
disassemble Disassemble ROM and write the rendered output.
init Scaffold a starter dasmos driver at DRIVER_PATH.
list-cpus List the available CPU plug-ins.
list-environments List the available environment plug-ins.
list-renderers List the available renderer plug-ins.
The CLI commands inherit a uniform --as display | tsv | json story
(plus --report, --header, --detailed) from
asyoulikeit, so any
command's structured output drives downstream tooling cleanly.
Discovering plug-ins
Two namespaces are populated by the bundled extensions; third-party packages register additional entries the same way.
$ dasmos list-cpus
CPUs registered under 'dasmos.cpu'
┏━━━━━━━━━━━┳━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
┃ Name ┃ Description ┃
┡━━━━━━━━━━━╇━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┩
│ cmos65c02 │ The CMOS 65C02 — NMOS 6502 superset with 8 new mnemonics, 2 new │
│ nmos6502 │ The classic NMOS 6502. │
└───────────┴─────────────────────────────────────────────────────────────────┘
$ dasmos list-renderers
Renderers registered under 'dasmos.renderer'
┏━━━━━━━━━┳━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
┃ Name ┃ Description ┃
┡━━━━━━━━━╇━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┩
│ beebasm │ Beebasm-syntax renderer. │
│ json │ JSON structured-output renderer. │
└─────────┴──────────────────────────────────┘
$ dasmos list-environments
Environments registered under 'dasmos.environment'
┏━━━━━━━━━━━━━━━━━━━━┳━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
┃ Name ┃ Description ┃
┡━━━━━━━━━━━━━━━━━━━━╇━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┩
│ acorn_bbc_hardware │ Acorn BBC Micro hardware-register Environment. │
│ acorn_mos │ Acorn MOS environment. │
│ acorn_sideways_rom │ Acorn sideways ROM environment. │
└────────────────────┴────────────────────────────────────────────────┘
Environments layer onto a disassembler additively — a driver can
activate any number of them, in either the constructor's
environments=[…] kwarg or via repeated
d.use_environment(…) calls.
describe-cpu (and the matching describe-renderer) shows the full
docstring of a single plug-in:
$ dasmos describe-cpu nmos6502
The classic NMOS 6502.
16-bit address space; the 56 documented mnemonics across 13
addressing modes; 151 documented opcodes (undocumented opcodes
deliberately omitted).
$ dasmos list-cpus --help
Usage: dasmos list-cpus [OPTIONS]
List the available CPU plug-ins.
Produces reports:
cpus Registered CPU (processor) plug-ins with one-line descriptions.
Options:
Report Output Options:
--no-reports Suppress all report output. The handler still runs
(useful for action commands whose reports are
incidental); only rendering is skipped. Mutually
exclusive with --report and --all-reports.
--all-reports Render every report the handler returns,
regardless of the command's default_reports.
Useful for commands whose default is a subset (or
silent) but where you want the full picture this
time. Mutually exclusive with --report and --no-
reports.
--report [cpus] Report name(s) to display (can be specified
multiple times). Shows all if omitted. Valid
values: cpus.
--header / --no-header Include column headers in output. Overrides each
report's default. Format-specific: TSV prefixes
first cell with '#', display omits
headers/title/caption, JSON ignores this flag.
--detailed / --essential Include detailed columns or only essential
columns. Auto-detects based on output format if
not specified.
--as [display|json|tsv] Output format for tabular data. Defaults to
'display' for terminals, 'tsv' for pipes.
--help Show this message and exit.
Migrating a py8dis driver
scripts/py8dis2dasmos.py is an AST-based porter that translates a
py8dis driver script into the equivalent Dasmos call shape:
uv run python scripts/py8dis2dasmos.py path/to/disasm_<rom>.py > ported.py
It rewrites the wildcard import, swaps load(addr, file, cpu) order,
maps move() → d.add_move(...), threads inline=True →
align=Align.INLINE, recognises subroutine(..., is_entry_point=False)
as a label-plus-banner pair, expands hook_subroutine and the bundled
hooks (stringhi_hook, …) through dasmos.hooks, and configures the
render() call so the output matches py8dis's defaults
(boundary_label_prefix='pydis_', byte_column=True).
Every transformation is covered by unit tests, plus a load-bearing end-to-end test that ports the unmodified Econet Bridge driver and asserts the resulting beebasm source re-assembles to the original ROM bytes.
Testing
pytest -v runs the suite. Tests marked @pytest.mark.beebasm
auto-skip when beebasm isn't on PATH; the rest run anywhere
Python and uv are installed. CI exercises the full matrix
(ubuntu/macos/windows × earliest+latest declared Python) against
the installed wheel, not the source tree, so packaging regressions
(missing entry points, omitted py.typed markers, unshipped
sub-packages) fail loud.
Layout
src/dasmos/ the package
src/dasmos/cli.py Click entry point + asyoulikeit reports
src/dasmos/disassembler.py Disassembler (driver-script API)
src/dasmos/core/ Memory / labels / moves / classifications
src/dasmos/cpu.py Cpu base + Opcode shape
src/dasmos/renderer.py Renderer base
src/dasmos/ext/cpus/ Bundled CPU plug-ins (nmos6502, cmos65c02)
src/dasmos/ext/renderers/ Bundled renderer plug-ins (beebasm)
src/dasmos/hooks.py Subroutine hooks (stringhi_hook, …)
scripts/py8dis2dasmos.py py8dis → dasmos AST porter
scripts/generate_readme.py This README's generator
docs/design/ Architecture decisions & sweep memos
tests/ Unit + round-trip + py8dis-parity tests
tests/fixtures/ Vendored ROM + driver + reference output
Related projects
- py8dis (fork) — the predecessor Dasmos is replacing.
Driver scripts written against this fork port via
scripts/py8dis2dasmos.py. - The four sibling Acorn ROM disassembly repositories under the
acornaeology umbrella that drive Dasmos's
round-trip / parity validation:
acorn-econet-bridge,acorn-6502-tube-client,acorn-nfs,acorn-adfs. - beebasm — the BBC-Micro-style assembler used as the round-trip oracle.
- asyoulikeit — the CLI-output framework Dasmos's reports are built on.
This README is generated from scripts/readme_template.md.j2 by
scripts/generate_readme.py. Do not edit it directly — edit the
template (or the generator, or the source files whose output it
captures) and re-run uv run python scripts/generate_readme.py. The
pre-commit hook and the readme-check CI job both run the
generator's --check mode and will refuse stale READMEs.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file dasmos-0.1.2.tar.gz.
File metadata
- Download URL: dasmos-0.1.2.tar.gz
- Upload date:
- Size: 189.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.8 {"installer":{"name":"uv","version":"0.11.8","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d69a97e4457a82696a1035f8a1fc0d8533639d5847a37d072bc26ac0dd351a68
|
|
| MD5 |
f6bb6b8f93467daef4e2696c1fe9ca07
|
|
| BLAKE2b-256 |
0fbb1b09ae326db165d041d61bdd1d7f266791bfcb7adcad80ec9e5865011c8a
|
File details
Details for the file dasmos-0.1.2-py3-none-any.whl.
File metadata
- Download URL: dasmos-0.1.2-py3-none-any.whl
- Upload date:
- Size: 130.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.8 {"installer":{"name":"uv","version":"0.11.8","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2fa692665c1d749b04df6559a2ec55dcb557bc49a05e8afbe1fa682d04319379
|
|
| MD5 |
7f982f9de0468fe006faa13f765db38f
|
|
| BLAKE2b-256 |
e3e2a49c29895a7fe988e151ad881d324ab4b6631b508b5063e9b123ba76d38c
|