1
0
mirror of https://github.com/mnaberez/py65.git synced 2025-01-15 12:31:16 +00:00
py65/docs/index.rst

509 lines
14 KiB
ReStructuredText
Raw Normal View History

2008-12-01 02:58:39 +00:00
***********************************************
Py65 - 6502 Microprocessor Simulation in Python
***********************************************
:Author: Mike Naberezny
:Version: |version|
.. topic:: Overview
2012-01-15 13:18:28 -08:00
Simulate 6502-based microcomputer systems in Python.
2008-12-01 02:58:39 +00:00
Using the Monitor
=================
Introduction
------------
Py65 includes a program called Py65Mon that functions as a machine language
monitor. This kind of program is sometimes also called a debugger. Py65Mon
provides a command line with many convenient commands for interacting with the
simulated 6502-based system.
The monitor is started using the ``py65mon`` command::
$ py65mon
Py65 Monitor
2012-11-30 11:43:35 -08:00
PC AC XR YR SP NV-BDIZC
6502: 0000 00 00 00 ff 00110000
2008-12-01 02:58:39 +00:00
.
Once the monitor has started, it will display a register dump and the
dot prompt. You can then enter commands for the monitor at this prompt.
Py65Mon uses commands that are very similar to those used by the
`VICE emulator's monitor <http://vice-emu.sourceforge.net/vice_12.html>`_
for Commodore computers. You can get a list of available commands
with ``help`` or help on a specific command with ``help command``.
2008-12-01 02:58:39 +00:00
Number Systems
--------------
When working with Py65Mon, you will frequently need to enter numbers, addresses,
2012-01-15 13:04:27 -08:00
and ranges of addresses. Almost all Py65 command support entering numbers in
2008-12-01 02:58:39 +00:00
binary, decimal, and hexadecimal.
2012-01-15 13:04:27 -08:00
Numbers can be entered with a prefix to specify the radix, e.g. ``$c000``
2008-12-01 02:58:39 +00:00
instructs Py65Mon that the number ``c000`` is hexadecimal. The following
prefixes are supported:
- ``$c000``: The dollar sign indicates hexadecimal.
- ``+828``: The plus sign indicates decimal.
- ``%0101``: The percent sign indicates binary.
Numbers can also be entered without a prefix. Most of the time, working in
hexadecimal will be the most convenient so this is the default radix. The
number ``c000`` will be assumed to be hexadecimal unless the default radix
is changed using the ``radix`` command.
Address Ranges
--------------
Some commands accept a range of memory addresses::
.disassemble ff80:ff84
$ff80 d8 CLD
$ff81 a2 ff LDX #$ff
$ff83 9a TXS
$ff84 a0 1c LDY #$1c
The syntax for a range is ``start:end``. Each of the two parts may have
2012-01-15 13:04:27 -08:00
a prefix to indicate the radix, or no prefix to use the default radix.
2008-12-01 02:58:39 +00:00
Sometimes it is useful to have the starting and ending address in a range
be the same, such as when you want to inspect a single byte of memory. In
this case, you can enter ``ff80:ff80`` or simply ``ff80``.
Assigning Labels
----------------
Large assembly language programs may have hundreds of procedures. It is
difficult to remember the memory address of each procedure and the addresses
2012-01-15 13:04:27 -08:00
may change if the program is reassembled.
2008-12-01 02:58:39 +00:00
You can add a label for any memory address using the ``add_label`` command.
This label can then be used anywhere the address could be used::
.add_label ff80 start
.disassemble start
$ff80 d8 CLD
When using labels, you can also specify an offset (plus or minus)::
.disassemble start:start+4
$ff80 d8 CLD
$ff81 a2 ff LDX #$ff
$ff83 9a TXS
$ff84 a0 1c LDY #$1c
Offsets are interpreted like any other numbers. In the example above,
``start+4`` implies that the offset (``4``) uses the default radix. This
could also be written as ``start+$04`` for explicit hexadecimal.
2014-10-01 13:31:31 +02:00
Breakpoints
-----------
It is possible to set breakpoints to stop execution when reaching a
given address or label. Breakpoints are added using the
``add_breakpoint`` command::
.disassemble start:start+4
$ff80 d8 CLD
$ff81 a2 ff LDX #$ff
$ff83 9a TXS
$ff84 a0 1c LDY #$1c
.add_breakpoint $ff84
Breakpoint 0 added at $FF84
.goto $ff80
Breakpoint 0 reached.
PC AC XR YR SP NV-BDIZC
6502: ff84 00 ff 00 ff 10110000
2014-10-01 13:31:31 +02:00
Note that a number is assigned to each breakpoint, similar to how
VICE operates. Deleting a breakpoint can be done via the
``delete_breakpoint`` command using the breakpoint identifier given
by ``add_breakpoint``::
.add_breakpoint $ff84
Breakpoint 0 added at $FF84
.delete_breakpoint 0
Breakpoint 0 removed
Breakpoints can be listed using the ``show_breakpoints`` command::
2014-10-01 13:31:31 +02:00
.add_breakpoint $1234
Breakpoint 0 added at $1234
.add_breakpoint $5678
Breakpoint 1 added at $5678
.add_breakpoint $9ABC
Breakpoint 2 added at $9ABC
.show_breakpoints
2014-10-01 13:31:31 +02:00
Breakpoint 0 : $1234
Breakpoint 1 : $5678
Breakpoint 2 : $9ABC
Keep in mind that breakpoint identifiers are not recycled throughout
a session, this means that if you add three breakpoints (#0, #1, #2)
and then delete breakpoint #1, the next breakpoint you add will be
breakpoint #3, not #1. Also, invoking ``reset`` clears breakpoints
too, not just labels.
2008-12-01 02:58:39 +00:00
Command Reference
=================
2014-10-01 13:31:31 +02:00
.. describe:: add_breakpoint <address|label>
Sets a breakpoint on execution at the given address or at the
address represented by the given label::
.add_breakpoint $1234
.add_label f000 start
.add_breakpoint start
Breakpoints get a numeric identifier to be used with
``delete_breakpoint``, the list of identifiers can be retrieved
with ``show_breakpoints``.
2014-10-01 13:31:31 +02:00
2008-12-01 02:58:39 +00:00
.. describe:: add_label <address> <label>
Assign a label to an address::
.add_label f000 start
Once defined, the label may be used in place of the address in other
commands. If a label already exists at the address, it will be silently
overwritten.
2012-01-15 13:04:27 -08:00
.. describe:: assemble <address> [<statement>]
Assemble a single statement at an address::
2008-12-01 02:58:39 +00:00
.assemble c000 lda $a0,x
$c000 b5 a0 LDA $a0,X
2012-01-15 13:04:27 -08:00
If no statement is given, interactive assembly mode will start::
.assemble c000
$c000
Enter a statement and it will be assembled at the current address. The
address will then be incremented and another statement may be entered.
Press Enter or Return without entering a statement to exit interactive
assembly mode.
If you have defined labels with add_label, you may use those labels in
the address and the operand.
2008-12-01 02:58:39 +00:00
.. describe:: cd <path>
Change the current working directory to the path specified::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.cd /path/to/my/files
/path/to/my/files
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
After changing the directory, the new working directory will be displayed.
The default working directory is the directory from which the monitor was
started.
.. describe:: cycles
Display the number of cycles that the microprocessor has run
since it was last reset::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.cycles
2012-01-15 13:04:27 -08:00
12
2008-12-01 02:58:39 +00:00
2014-10-01 13:31:31 +02:00
.. describe:: delete_breakpoint <breakpoint_id>
Removes the breakpoint associated with the given identifier::
.add_breakpoint $1234
Breakpoint 0 added at $1234
.add_label f000 start
.add_breakpoint start
Breakpoint 1 added at $F000
.delete_breakpoint 0
Breakpoint 0 removed
The list of identifiers added with ``add_breakpoint`` can be
retrieved with ``show_breakpoints``.
2014-10-01 13:31:31 +02:00
2008-12-01 02:58:39 +00:00
.. describe:: delete_label <label>
Delete a label that was previously defined with ``add_label``::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.delete_label foo
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
If the label does not exist, the command will fail silently.
.. describe:: disassemble <address_range>
2013-08-26 20:24:23 -07:00
Disassemble a range of memory::
2008-12-01 02:58:39 +00:00
.disassemble ff80:ff84
$ff80 d8 CLD
$ff81 a2 ff LDX #$ff
$ff83 9a TXS
$ff84 a0 1c LDY #$1c
2013-08-26 20:24:23 -07:00
The disassembly will use the instruction set of the selected MPU. For
example, the extra instructions of the 65C02 will only be displayed if
a 65C02 MPU is selected. On an NMOS 6502, those instructions would be
disassembled as ``???``.
2008-12-01 02:58:39 +00:00
If labels have been defined, they will be substituted for
addresses in the operands.
.. describe:: fill <address_range> <byte> [<byte> <byte> ...]
Fill a range of memory using one or more bytes from the list::
.fill c000:c003 aa bb
Wrote +4 bytes from $c000 to $c003
.mem c000:c003
c000: aa bb aa bb
If the range is larger than the number of bytes in the list, the list
will repeat as shown above.
.. describe:: goto <address>
Set the program counter to an address and resume execution::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.goto c000
.. describe:: help [<command>]
Display help for all commands or a single command. If no command is
given, a list of commands will be displayed::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.help
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
If a command is given, help for that comand is displayed::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.help disassemble
disassemble <address_range>
Disassemble instructions in the address range.
.. describe:: show_breakpoints
2014-10-01 13:31:31 +02:00
Lists all the breakpoints that have been set so far::
.add_breakpoint $1234
Breakpoint 0 added at $1234
.add_breakpoint $5678
Breakpoint 1 added at $5678
.add_breakpoint $9ABC
Breakpoint 2 added at $9ABC
.show_breakpoints
2014-10-01 13:31:31 +02:00
Breakpoint 0 : $1234
Breakpoint 1 : $5678
Breakpoint 2 : $9ABC
2008-12-01 02:58:39 +00:00
.. describe:: load <filename> <address>
Load a binary file into memory starting at the address specified::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.load hello.bin c000
Wrote +29 bytes from $c000 to $c01c
The file will be loaded relative to the current working directory. You
may also specify an absolute path. If the filename contains spaces, use
quotes around it::
.load "say hello.bin" c000
Wrote +29 bytes from $c000 to $c01c
2012-01-15 13:04:27 -08:00
.. note::
2008-12-01 02:58:39 +00:00
Unlike the VICE monitor, Py65Mon's ``load`` command does not expect
the first two bytes to be a Commodore-style load address. It will start
2012-01-15 13:37:25 -08:00
reading the data at byte 0, not byte 2.
2008-12-01 02:58:39 +00:00
2012-01-15 13:31:49 -08:00
If the filename is a URL, it will be retrieved::
.load https://github.com/mnaberez/py65/raw/0.11/examples/ehbasic.bin 0000
Wrote +65536 bytes from $0000 to $ffff
2008-12-01 02:58:39 +00:00
.. describe:: mem <address_range>
Display the contents of memory an address range::
.mem ff80:ffa0
ff80: d8 a2 ff 9a a0 1c b9 bb ff 99 04 02 88 d0 f7 b9 d8 ff
ff92: f0 06 20 a6 e0 c8 d0 f5 20 a3 e0 90 fb 29 df
2012-01-15 13:04:27 -08:00
The contents will be wrapped to the terminal width specified by the
2008-12-01 02:58:39 +00:00
``width`` command.
2012-01-15 13:18:28 -08:00
.. describe:: mpu [<mpu_name>]
Display or set the current microprocessor. If no argument is given, the
current microprocessor will be displayed::
.mpu
Current MPU is 6502
Available MPUs: 6502, 65C02, 65Org16
If an argument is given, the microprocessor will be changed::
.mpu 65C02
Reset with new MPU 65C02
The default microprocessor is ``6502``, the original NMOS 6502 from
MOS Technology.
2008-12-01 02:58:39 +00:00
.. describe:: pwd
Display the current working directory::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.pwd
/home/mnaberez
.. describe:: quit
Quit the monitor::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.quit
.. describe:: radix [<H|D|O|B>]
2012-01-15 13:04:27 -08:00
Display or set the default radix that is assumed for numbers that have no prefix.
2008-12-01 02:58:39 +00:00
If no argument is given, the default radix is displayed::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.radix
Default radix is Hexadecimal
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
If an argument is given, the default radix will be changed::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.radix d
Default radix is Decimal
2012-01-15 13:04:27 -08:00
The default radix may be changed to Hexadecimal, Decimal, Octal, or Binary.
2008-12-01 02:58:39 +00:00
.. describe:: registers [<name=value>, <name=value>, ...>]
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
Display or change the registers of the microprocessor. If no arguments are
given, the registers are displayed::
.registers
PC AC XR YR SP NV-BDIZC
6502: 0000 00 00 00 ff 00110000
2008-12-01 02:58:39 +00:00
Registers can changed giving ``name=value``, separated by commas if
multiple registers are to be changed::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.registers a=02, x=04
PC AC XR YR SP NV-BDIZC
6502: 0000 02 04 00 ff 00110000
2008-12-01 02:58:39 +00:00
.. describe:: reset
Reset the microprocessor to its default state. All memory will
also be cleared::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.reset
.. describe:: return
2012-01-15 13:04:27 -08:00
Continue execution and return to the monitor just before the next
2008-12-01 02:58:39 +00:00
RTS or RTI is executed::
2012-01-15 13:04:27 -08:00
.return
2008-12-01 02:58:39 +00:00
2012-01-15 13:37:25 -08:00
.. describe:: save <filename> <start_address> <end_address>
Save the specified memory range to disk as a binary file::
.save hello.bin c000 c01c
Wrote +29 bytes from $c000 to $c01c
The file will be saved relative to the current working directory. You
may also specify an absolute path. If the filename contains spaces, use
quotes around it::
.save "say hello.bin" c000 c01c
Wrote +29 bytes from $c000 to $c01c
.. note::
Unlike the VICE monitor, Py65Mon's ``save`` command does not write
the first two bytes as a Commodore-style load address. It will start
writing the data at byte 0, not byte 2.
2008-12-01 02:58:39 +00:00
.. describe:: show_labels
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
Display labels that have been defined with ``add_label``::
.show_labels
ffd2: charout
.. describe:: step
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
Execute a single instruction at the program counter. After the instruction
executes, the next instruction is disassembled and printed::
PC AC XR YR SP NV-BDIZC
6502: 0000 00 00 00 ff 00110000
2008-12-01 02:58:39 +00:00
.registers pc=c000
PC AC XR YR SP NV-BDIZC
6502: c000 00 00 00 ff 00110000
2008-12-01 02:58:39 +00:00
.step
$c002 a9 42 LDA #$42
PC AC XR YR SP NV-BDIZC
6502: c002 00 00 00 ff 00110000
2008-12-01 02:58:39 +00:00
.
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
In the example above, the instruction at ``$C000`` executes and the monitor
2012-01-15 13:04:27 -08:00
prompt returns.
2008-12-01 02:58:39 +00:00
2012-01-15 13:04:27 -08:00
.. note::
2008-12-01 02:58:39 +00:00
After the instruction executes, the disassembly of the **next** instruction
is printed. This allows you to see what will be executed on the next step.
.. describe:: tilde
Display a number in the supported number systems::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.~ c000
+49152
$c000
140000
1100000000000000
2012-01-15 13:04:27 -08:00
The number will be displayed in this order: decimal, hexadecimal, octal,
2008-12-01 02:58:39 +00:00
and then binary.
.. describe:: version
Display version information::
.version
Py65 Monitor
.. describe:: width [<columns>]
Display or set the terminal width. The width is used to wrap the output
of some commands like ``mem``. With no argument, the current width is
displayed::
2012-01-15 13:04:27 -08:00
2008-12-01 02:58:39 +00:00
.width
Terminal width is 78
If a column count is given, the width will be changed::
.width 130
Terminal width is 130
The number of columns is always specified as a decimal number.