2023-06-25 21:35:30 +02:00
|
|
|
===========================
|
2018-08-06 03:35:43 +02:00
|
|
|
Target system specification
|
2023-06-25 21:35:30 +02:00
|
|
|
===========================
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2018-09-15 16:21:05 +02:00
|
|
|
Prog8 targets the following hardware:
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2020-08-27 18:18:29 +02:00
|
|
|
- 8 bit MOS 6502/65c02/6510 CPU
|
2018-08-06 03:35:43 +02:00
|
|
|
- 64 Kb addressable memory (RAM or ROM)
|
2020-08-27 18:18:29 +02:00
|
|
|
- optional use of memory mapped I/O registers
|
|
|
|
- optional use of system ROM routines
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2022-02-15 01:39:12 +01:00
|
|
|
Currently these machines can be selected as a compilation target (via the ``-target`` compiler argument):
|
2020-08-27 18:18:29 +02:00
|
|
|
|
2021-12-21 19:08:33 +01:00
|
|
|
- 'c64': the Commodore 64
|
|
|
|
- 'cx16': the `Commander X16 <https://www.commanderx16.com/>`_
|
2022-02-21 23:38:53 +01:00
|
|
|
- 'c128': the Commodore 128 (*limited support*)
|
2023-08-14 15:16:46 +02:00
|
|
|
- 'pet32': the Commodore PET 4032 (*limited support*)
|
2022-02-21 23:38:53 +01:00
|
|
|
- 'atari': the Atari 800 XL (*experimental support*)
|
2022-09-20 04:04:47 +02:00
|
|
|
- 'virtual': a builtin virtual machine
|
2020-08-30 18:32:16 +02:00
|
|
|
|
2021-12-30 18:33:26 +01:00
|
|
|
This chapter explains some relevant system details of the c64 and cx16 machines.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2020-10-17 20:35:36 +02:00
|
|
|
.. hint::
|
2022-10-29 14:10:11 +02:00
|
|
|
If you only use standard Kernal and prog8 library routines,
|
2022-09-20 04:04:47 +02:00
|
|
|
it is often possible to compile the *exact same program* for
|
|
|
|
different machines (just change the compilation target flag)!
|
2020-09-22 00:47:02 +02:00
|
|
|
|
2018-08-06 03:35:43 +02:00
|
|
|
|
|
|
|
Memory Model
|
|
|
|
============
|
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
Generic 6502 Physical address space layout
|
|
|
|
------------------------------------------
|
2018-08-06 03:35:43 +02:00
|
|
|
|
|
|
|
The 6502 CPU can address 64 kilobyte of memory.
|
2018-09-15 16:21:05 +02:00
|
|
|
Most of the 64 kilobyte address space can be used by Prog8 programs.
|
2023-04-03 20:47:31 +02:00
|
|
|
This is a hard limit: there is no support for RAM expansions or bank switching built natively into the language.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
|
|
|
====================== ================== ========
|
|
|
|
memory area type note
|
|
|
|
====================== ================== ========
|
2022-10-29 14:12:10 +02:00
|
|
|
``$00``--``$ff`` zeropage contains many sensitive system variables
|
2018-08-06 03:35:43 +02:00
|
|
|
``$100``--``$1ff`` Hardware stack used by the CPU, normally not accessed directly
|
|
|
|
``$0200``--``$ffff`` Free RAM or ROM free to use memory area, often a mix of RAM and ROM
|
2023-04-03 20:47:31 +02:00
|
|
|
depending on the specific computer system
|
2018-08-06 03:35:43 +02:00
|
|
|
====================== ================== ========
|
|
|
|
|
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
Memory map for the C64 and the X16
|
|
|
|
----------------------------------
|
|
|
|
|
|
|
|
This is the default memory map of the 64 Kb addressable memory for those two systems.
|
|
|
|
Both systems have ways to alter the memory map and/or to switch memory banks, but that is not shown here.
|
|
|
|
|
|
|
|
.. image:: memorymap.svg
|
|
|
|
|
|
|
|
Footnotes for the Commander X16
|
|
|
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
|
|
|
*Golden Ram $0400 - $07FF*
|
2023-07-15 15:25:32 +02:00
|
|
|
*free to use.*
|
2023-04-03 20:47:31 +02:00
|
|
|
|
|
|
|
*Zero Page $0000 - $00FF*
|
|
|
|
$00 and $01 are hardwired as Rom and Ram banking registers.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
$02 - $21 are the 16 virtual cx16 registers R0-R15.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2023-07-29 13:04:35 +02:00
|
|
|
$22 - $7F are used by Prog8 to put variables in.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
The top half of the ZP ($80-$FF) is reserved for use by the Kernal and Basic in normal operation.
|
|
|
|
Zero page use by Prog8 can be manipulated with the ``%zeropage`` directive, various options
|
2023-07-29 13:04:35 +02:00
|
|
|
may free up more locations for use by Prog8 or to reserve them for other things.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
Footnotes for the Commodore 64
|
|
|
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
*RAM $C000-$CFFF*
|
2023-07-15 15:25:32 +02:00
|
|
|
*free to use:* $C000 - $CFDF
|
|
|
|
*reserved:* $CFE0 - $CFFF for the 16 virtual cx16 registers R0-R15
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
*Zero Page $0000 - $00FF*
|
|
|
|
Consider the full zero page to be reserved for use by the Kernal and Basic in normal operation.
|
|
|
|
Zero page use by Prog8 can be manipulated with the ``%zeropage`` directive, various options
|
2023-07-29 13:04:35 +02:00
|
|
|
may free up more locations for use by Prog8 or to reserve them for other things.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
|
|
|
|
2023-04-03 20:47:31 +02:00
|
|
|
Zero page usage by the Prog8 compiler
|
|
|
|
-------------------------------------
|
2018-09-15 16:21:05 +02:00
|
|
|
Prog8 knows what addresses are safe to use in the various ZP handling configurations.
|
2018-09-06 21:13:49 +02:00
|
|
|
It will use the free ZP addresses to place its ZP variables in,
|
2018-08-06 03:35:43 +02:00
|
|
|
until they're all used up. If instructed to output a program that takes over the entire
|
|
|
|
machine, (almost) all of the ZP addresses are suddenly available and will be used.
|
2018-09-06 21:13:49 +02:00
|
|
|
|
2022-10-29 14:12:10 +02:00
|
|
|
**zeropage handling is configurable:**
|
2018-09-06 21:13:49 +02:00
|
|
|
There's a global program directive to specify the way the compiler
|
|
|
|
treats the ZP for the program. The default is to be reasonably restrictive to use the
|
2022-10-29 14:10:11 +02:00
|
|
|
part of the ZP that is not used by the C64's Kernal routines.
|
|
|
|
It's possible to claim the whole ZP as well (by disabling the operating system or Kernal).
|
2022-10-28 22:39:15 +01:00
|
|
|
If you want, it's also possible to be more restrictive and stay clear of the addresses used by BASIC routines too.
|
2019-02-03 00:14:56 +01:00
|
|
|
This allows the program to exit cleanly back to a BASIC ready prompt - something that is not possible in the other modes.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
|
|
|
|
2022-10-29 14:12:10 +02:00
|
|
|
IRQs and the zeropage
|
2018-08-06 03:35:43 +02:00
|
|
|
^^^^^^^^^^^^^^^^^^^^^
|
|
|
|
|
2022-10-29 14:12:10 +02:00
|
|
|
The normal IRQ routine in the C64's Kernal will read and write several addresses in the ZP
|
2018-08-06 03:35:43 +02:00
|
|
|
(such as the system's software jiffy clock which sits in ``$a0 - $a2``):
|
|
|
|
|
|
|
|
``$a0 - $a2``; ``$91``; ``$c0``; ``$c5``; ``$cb``; ``$f5 - $f6``
|
|
|
|
|
2018-09-06 21:13:49 +02:00
|
|
|
These addresses will *never* be used by the compiler for ZP variables, so variables will
|
2018-08-06 03:35:43 +02:00
|
|
|
not interfere with the IRQ routine and vice versa. This is true for the normal ZP mode but also
|
|
|
|
for the mode where the whole system and ZP have been taken over.
|
|
|
|
So the normal IRQ vector can still run and will be when the program is started!
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CPU
|
|
|
|
===
|
|
|
|
|
|
|
|
Directly Usable Registers
|
|
|
|
-------------------------
|
|
|
|
|
2020-07-24 22:57:19 +02:00
|
|
|
The hardware CPU registers are not directly accessible from regular Prog8 code.
|
|
|
|
If you need to mess with them, you'll have to use inline assembly.
|
2018-08-06 03:35:43 +02:00
|
|
|
|
2020-07-24 22:57:19 +02:00
|
|
|
The status register (P) carry flag and interrupt disable flag can be written via a couple of special
|
|
|
|
builtin functions (``set_carry()``, ``clear_carry()``, ``set_irqd()``, ``clear_irqd()``),
|
|
|
|
and read via the ``read_flags()`` function.
|
2019-01-24 02:43:25 +01:00
|
|
|
|
2020-12-22 13:29:16 +01:00
|
|
|
The 16 'virtual' 16-bit registers that are defined on the Commander X16 machine are not real hardware
|
|
|
|
registers and are just 16 memory-mapped word values that you *can* access directly.
|
2019-03-06 22:11:16 +01:00
|
|
|
|
|
|
|
|
|
|
|
IRQ Handling
|
|
|
|
============
|
|
|
|
|
|
|
|
Normally, the system's default IRQ handling is not interfered with.
|
2021-02-21 22:17:28 +01:00
|
|
|
You can however install your own IRQ handler (for clean separation, it is advised to define it inside its own block).
|
2022-10-29 14:12:10 +02:00
|
|
|
There are a few library routines available to make setting up C64 60hz IRQs and Raster IRQs a lot easier (no assembly code required).
|
2019-03-06 22:11:16 +01:00
|
|
|
|
2023-04-28 23:13:03 +02:00
|
|
|
These routines are::
|
2019-03-06 22:11:16 +01:00
|
|
|
|
2023-11-21 22:33:37 +01:00
|
|
|
sys.set_irq(uword handler_address)
|
|
|
|
sys.set_rasterirq(uword handler_address, uword rasterline)
|
2023-04-28 23:13:03 +02:00
|
|
|
sys.restore_irq() ; set everything back to the systems default irq handler
|
2020-07-24 22:57:19 +02:00
|
|
|
|
2023-11-21 22:33:37 +01:00
|
|
|
The IRQ handler routine must return a boolean value (0 or 1) in the A register.
|
|
|
|
0 means do *not* run the system IRQ handler routine afterwards, 1 means run the system IRQ handler routine afterwards.
|
|
|
|
|
|
|
|
|
2023-11-21 21:31:50 +01:00
|
|
|
**CommanderX16 specific notes**
|
|
|
|
|
2023-11-21 22:33:37 +01:00
|
|
|
Note that for the CommanderX16 the set_rasterirq() will disable VSYNC irqs and never call the system IRQ handler regardless
|
|
|
|
of the return value of the user handler routine. This also means the default sys.wait() routine won't work anymore,
|
|
|
|
when using this handler.
|
|
|
|
|
|
|
|
These two helper routines are not particularly suited to handle multiple IRQ sources on the Commander X16.
|
|
|
|
It's possible but it requires correct fiddling with IRQ enable bits, acknowledging the IRQs, and properly calling
|
|
|
|
or not calling the system IRQ handler routine.
|
2021-02-21 23:41:50 +01:00
|
|
|
|
2023-04-28 23:13:03 +02:00
|
|
|
The Commander X16 syslib provides two additional routines that should be used *in your IRQ handler routine* if it uses the Vera registers.
|
2023-01-26 00:37:30 +01:00
|
|
|
They take care of saving and restoring the Vera state of the interrupted main program, otherwise the IRQ handler's manipulation
|
|
|
|
will corrupt any Vera operations that were going on in the main program. The routines are::
|
2022-07-02 23:10:15 +02:00
|
|
|
|
2023-04-17 23:37:15 +02:00
|
|
|
cx16.save_vera_context()
|
2023-11-19 18:59:03 +01:00
|
|
|
; perhaps also cx16.save_virtual_registers() here...
|
|
|
|
; ... do your work that uses vera here!...
|
|
|
|
; perhaps also cx16.restore_virtual_registers() here...
|
2023-04-17 23:37:15 +02:00
|
|
|
cx16.restore_vera_context()
|
2022-07-02 23:10:15 +02:00
|
|
|
|
2023-01-20 03:10:41 +01:00
|
|
|
.. caution::
|
2023-03-19 20:58:02 +01:00
|
|
|
The Commander X16's 16 'virtual registers' R0-R15 are located in zeropage and *are not preserved* in the IRQ handler!
|
|
|
|
So you should make sure that the handler routine does NOT use these registers, or do some sort of saving/restoring yourself
|
|
|
|
of the ones that you do need in the IRQ handler.
|
2023-06-24 21:04:47 +02:00
|
|
|
There are two utility routines in cx16 that save and restore *all* 16 registers so it's a bit inefficient but safe.
|
2023-11-19 18:59:03 +01:00
|
|
|
(these are ``save_virtual_registers()`` and ``restore_virtual_registers()``)
|
2023-03-19 20:58:02 +01:00
|
|
|
|
|
|
|
It is also advised to not use floating point calculations inside IRQ handler routines.
|
2023-01-20 03:10:41 +01:00
|
|
|
Beside them being very slow, there are intricate requirements such as having the
|
|
|
|
correct ROM bank enabled to be able to successfully call them (and making sure the correct
|
|
|
|
ROM bank is reset at the end of the handler), and the possibility
|
|
|
|
of corrupting variables and floating point calculations that are being executed
|
|
|
|
in the interrupted main program. These memory locations should be backed up
|
|
|
|
and restored at the end of the handler, further increasing its execution time...
|