This repository is deprecated. Refer to the refactored Dragon 32 emulator in new repository https://github.com/eyalabraham/dragon32-emu
This project implements the software and the hardware needed to emulate a Dragon 32 computer. This emulation is for a bare-metal Raspberry Pi (RPi) platform with some external peripherals. Development was done on a Raspberry Pi Linux distribution. The Dragon was my first computer when home/personal computers started to emerge in the mid 80s, and it is also one of the simplest to emulate. Other emulators exist, including XROR, but I decided to build my own as another exercise in RPi bare metal programming.
- The DRAGON Archives resources and software.
- DRAGON Data archives hardware schematics.
- Inside the Dragon by Duncan Smeed and Ian Sommerville.
- Computers based on 6809 CoCo Coding
- RPi BCM2835 GPIOs
- C library for Broadcom BCM 2835
The CPU module is cpu.c
, the memory module is mem.c
. The IO module is implemented as call-back functions hooked through the memory module to selected memory addresses, and emulate the response of a memory mapper IO device. The IO module call-backs implement the various IO device in the emulated computer. This design is flexible enough to allow the definition of any computer configuration, memory and IO, built around an MC6809 CPU. The examples and development steps show an SWTPC computer and Grant's 6-chip 6809 computer as two instances I used along the way.
The emulation will not all be done through software. In order to save time some IO devices will be implemented in hardware: keyboard interface (possibly PS2 through the Pi's SPI), audio DAC and an analog comparator for a successive approximation ADC that will reproduce sound and support an original (circa 1984) Dragon Computer joystick.
ASCII art depiction of the system for the RPi bare metal implementation:
+-------------------------------------+ |
| Monitor (via Auxiliary UART) | |
| Logger | |
| Monitor CLI | |
+-------------------------------------+ |
| | | |
+-----+ +----------+ +--------------+ Software emulation
| CPU |--| MEM |--| IO | |
+-----+ +----------+ | | |
| | | |
+----------+ | | | |
| Graphics | | SAM | PIA | |
| xlate |--| VGD | | |
+----------+ +-------+------+ |
| | ------
+----------+ +------+ |
| RPi | | RPi | |
| frame | | GPIO | RPi HW
| buff | | | |
+----------+ +------+ |
| | ------
+----------+ +---------------+ |
| VGA | | Devices: | |
| Monitor | | PS2 Keyboard | External HW
+----------+ | Joystick | |
| Audio DAC | |
| SD card | |
+---------------+ |
Schematics of external emulator hardware (KiCAD drawing)[https://github.com/eyalabraham/schematics/tree/master/dragon].
This repository contains all the intermediate implementation steps and tags them for easy retrieval. Each step builds on the functionality of its predecessors and maintains backward compatibility. The latest release is listed first:
- Release tag 1.0 Complete emulator for bare metal Raspberry Pi Zero.
The mem.c
and cpu.c
modules are all that are required for a basic MC6809E CPU emulator. The emu09.c
module implements such an emulator, which I used for testing. The implementation of function cpu_run()
in module cpu.c
provides a timing-accurate execution of MC6809E code in both single step and continuous execution. Each time the function is called another CPU command is emulated, with CPU state available for inspection when single-stepping.
The method of repeatedly calling cpu_run()
allow interruption for single stepping, break-point detection, and insertion of external events such as CPU interrupts.
To implement the full Dragon computer emulation, the cpu_run()
function is called from an endless loop and IO device call-backs are implemented to carry out IO device activities.
One of the goals is to achieve CPU cycle timing that is as close as possible to the 0.89MHz of MC6809E that was used in the Dragon Computer. A crude timing profile using profile.asm
test code and profile.c
emulation module yielded the following results under Raspberrypi Linux
Measurement | Setup | Total [uSec] | Command [uSec] | Cycles | Cycle time [uSec] |
---|---|---|---|---|---|
Baseline | Empty loop overhead | 2.8 | |||
NOP op-code | 10x NOP op-codes | 10 | 0.72 | 2 | 0.36 |
ORA | 10x ORA | 9.6 | 0.68 | 2 | 0.34 |
ABX op-code | 10x ABX op-codes | 9.8 | 0.7 | 3 | 0.23 |
STA <addr_ext> | 10x STA <add_ext> | 14.8 | 1.2 | 5 | 0.24 |
LBRA | 10x LBRA | 12.8 | 1.0 | 5 | 0.20 |
PSHS/PULS a,b,x,y | 5x PSHS / 5xPULS a,b,x,y | 27.5 | 2.47 | 11 | 0.22 |
The tests were run on a Raspberry Pi model B, single core (ARM1176JZF-S) 700MHz Broadcom BCM2835 with 512MB RAM, running a generic Raspberrypi Linux distribution.
The calculations show that the emulated CPU runs at an average rate of 3,754,978[Hz]
The Dragon computers where based on Motorola's MC6806E CPUs. The 6809 is an 8-bit microprocessor with some neat 16-bit features. The CPU module emulates the full set of CPU opcodes.
- The CPU module
cpu.c
interfaces with the memory module for reading opcode or data and writing to memory locations or to memory-mapped IO devices through themem.c
module - Includes an API for external controls:
- External interrupt emulation
- CPU halt command
- CPU Reset
- CPU state and registers
The Dragon computer supported a maximum of 64K Bytes of memory. The memory map was managed by the SAM chip and divided into RAM, ROM, expansion ROM (cartridge), and memory mapped IO address spaces. In its basic state the memory module emulates 64K Bytes of RAM that can be accessed with the mem_read()
and mem_write()
API calls. Memory address ranges can be configured with special attributes that change their behavior into ROM or memory mapped IO addresses:
mem_define_rom()
will define a memory address range as read-only after which amem_write()
call would trigger a debug exception.mem_define_io()
will define a memory address range as a memory mapped IO device and will register an IO device handler that will be called when a read or write calls are directed to addresses in the defined range.mem_load()
will load a memory range with data copied from an input buffer.mem_init()
will initialize memory.
typedef enum
{
MEM_TYPE_RAM,
MEM_TYPE_ROM,
MEM_TYPE_IO,
} memory_flag_t;
typedef struct
{
uint8_t data_byte;
memory_flag_t memory_type;
io_handler_callback io_handler;
} memory_t;
When the CPU emulation module reads a memory location is uses the mem_read()
call that returns the contents of the memory address passed with the call. For a memory write using mem_write()
call the following logic is applied:
- Check if address is in range 0x0000 to 0xffff. If not flag exception and return with no action
- Check memory location against MEM_FLAG_ROM flag. If memory location is ROM return with no action
- Write data to memory location.
For both mem_read()
and mem_write()
check memory location against MEM_FLAG_IO flag. If flag is set, invoke the callback with the accessed address, the data (if a write operation) and a read/write flag. This will give the IO callback the context it needs to emulate the IO behind the memory address.
The MC6809E CPU in the Dragon computer uses memory mapped IO devices. During initialization the emulation registers device callback functions that implement the IO devices' functionality. The callbacks are registered against memory address ranges associated with the device using the mem_define_io()
call. The callbacks are invoked when reads or writes are issued to memory locations registered to IO devices. The dragon.c
, mon09.c
and basic09.c
computer emulation modules use IO callbacks to emulate the SAM, VDG, MC6821 PIA and MC6850 ACIA etc.
The code in the call-backs redirect the IO request to the appropriate Raspberry Pi GPIO pin and function. The Dragon 32 emulation uses the following external GPIO:
Function | RPi model B | RPi Zero / ZeroW |
---|---|---|
Analog multiplexer select-0 | GPIO-02 | GPIO-02 |
Analog multiplexer select-1 | GPIO-03 | GPIO-03 |
RPi timing test point | GPIO-04 | GPIO-04 |
Emulator reset | na | GPIO-05 |
Analog comparator input | GPIO-07 | GPIO-07 |
Right joystick "fire" | GPIO-08 | GPIO-08 |
AVR ATtiny85 keyboard MISO | GPIO-09 | GPIO-09 |
AVR ATtiny85 keyboard MOSI | GPIO-10 | GPIO-10 |
AVR ATtiny85 keyboard SCLK | GPIO-11 | GPIO-11 |
Serial TxD (future) | GPIO-14 | GPIO-14 |
Serial RxD (future) | GPIO-15 | GPIO-15 |
SD card CE | na | GPIO-16 |
AVR ATtiny85 reset | GPIO-17 | GPIO-17 |
SD card MISO | na | GPIO-19 |
SD card MOSI | na | GPIO-20 |
SD card SCLK | na | GPIO-21 |
DAC bit.0 | GPIO-22 | GPIO-22 |
DAC bit.1 | GPIO-23 | GPIO-23 |
DAC bit.2 | GPIO-24 | GPIO-24 |
DAC bit.3 | GPIO-25 | GPIO-25 |
DAC bit.4 | GPIO-18 | GPIO-26 (1) |
DAC bit.5 | GPIO-27 | GPIO-27 |
Notes:
- (1) Changed on the Zero/ZeroW in order to form a contiguous bit order in the GPIO register.
The SAM chip in the Dragon computer is responsible for IO address decoding, dynamic RAM memory refresh, and video memory address generation for the Video Display Generator (VDG) chip. Of these three functions, only the last one requires implementation. Since the SAM chip does not generate video, the address scanning is implemented in the VDG module by the vdg_render()
function. The SAM device emulation only transfers offset address and video mode settings to the VDG module.
The VDG is Motorola's MC6847 video chip. Since the VDG's video memory is part of the 64K Bytes of the CPU's memory map, then writes to that region are reflected into the RPi's video frame buffer by the IO handler of the VDG. The handler will adapt the writes to the RPi frame buffer based on the VDG/SAM modes for text or graphics. The Dragon computer video display emulation is implemented in the VDG module by the vdg_render()
function, by accessing the Raspberry Pi Frame Buffer.
The Dragon computer's IO was provided by two MC6821 Peripheral Interface Adapters (PIAs).
The keyboard interface uses an ATtiny85 AVR coded with a PS2 to SPI interface. It implements a PS2 keyboard interface and an SPI serial interface. The AVR connects with the Raspberry Pi's SPI. The code configures the keyboard, accepts scan codes, converts the AT scan codes to ASCII make/break codes for the Dragon 32 emulation running on the Raspberry Pi. The AVR buffers the key codes in a small FIFO buffer, and the emulation periodically reads the buffer through the SPI interface.
+-----+ +-----+ +-------+
| | | | | |
| +----[ MOSI>----+ | | |
| | | | | |
| +----< MISO]----+ +--< Data >--+ Level |
| RPi | | AVR | | shift +---> PS2 keyboard
| +----[ SCL >----+ +--< CLK ]---+ |
| | | | | |
| +----[ RST >----+ | | |
| | | | | |
+-----+ +-----+ +-------+
The Dragon's sound system is built around a simple 6-bit Digital to Analog (DAC). The DAC is a common resistor ladder that is driven by a 6-bit MC4050 buffer. The Dragon emulator uses a similar setup with an 8-bit TTL buffer (74244) and similar resistor values. The GPIO pins of a Raspberry Pi B do not provide a contiguous set of output bits, so bit position translation is implemented.
DAC RPi B (RPi Zero)
+-------+
bit.0 GPIO22 P1.15 (GPIO22 P1.15 ) >----+ +-[330K]-+
bit.1 GPIO23 P1.16 (GPIO23 P1.16 ) >----+ +-[150K]-+
bit.2 GPIO24 P1.18 (GPIO24 P1.18 ) >----+ 74244 +-[ 82K]-+
bit.3 GPIO25 P1.22 (GPIO25 P1.22 ) >----+ +-[ 39K]-+
bit.4 GPIO18 P1.12 (GPIO26 P1.37 ) >----+ +-[ 20K]-+
bit.5 GPIO27 P1.13 (GPIO27 P1.13 ) >----+ +-[ 10K]-+---[Analog out>
+-------+
In the Dragon computer the audio multiplexer is controlled by PIA0-CA2 and CB2, with PA1-CB2 controlling the audio source inhibit line. The CD4052 user in this emulator is different from the 4529 device used in the original computer and some changes in the emulation call-back are implemented to account for the difference. The changes reduce the number of supported joysticks to one with only the right joystick, and only two audio sources: DAC, and one open source for future use.
The external hardware provides connectivity for the right joystick. The emulation software supports only one joystick. The external hardware is built with an analog multiplexer (CD4052) that routes the joystick output voltages to a comparator. The comparator works in conjunction with the DAC and the Dragon software to convert the analog joystick position to a number range between 0 and 63. The analog multiplexer is controlled by GPIO pins that represent PIA0-CA2 and PIA1-CB2 control lines, using low order select bit and the inhibit line instead of the high order select bit.
In the Dragon computer, the system generates an IRQ interrupt at the frame synchronization (FS) rate of 50 or 60Hz. The FS signal is routed through PIA0-CB1 (control register B-side) and generates an IRQ signal. Resetting the interrupt request by reading data register PIA0 B-side.
The software loader/manager interfaces with an SD card that holds Dragon 32 ROM cartridge images and CAS files. The loader/manager can be escaped into from the emulation using the F1 key. Within the loader one can brows ROM and CAS files to load and run on the Dragon 32 emulator.
This functionality is available only on RPi Zero/W and uses an SD card interface connected to the auxiliary SPI interface (SPI1).
ROM code files are loaded as-is into the Dragon's ROM cartridge memory address space. No auto start is provided, but the BASIC EXEC vector is modified to point to 0xC000, so a simple EXEC from the BASIC prompt will start the ROM code.
CAS files are digital images of old-style tape content and not memeory images. More on CAS file formats here, and Dragon 32 CAS format here. A cassette file can be mounted by the loader (like loading a cassette into a tape player), and then use the BASIC CLOAD or CLOADM commands to do the reading.
- Cartridge interrupt for cartridge code auto start
- Clock interrupts for CPU execution pacing
- Settable logging to serial console
- Exception generation, example: writing to a memory location that is defines as ROM.
- Dragon sound sources: single-bit
- Joystick drift: check reference voltage. Reduce to TTL output level from 74LS244
- dragon.c main module for Dragon Computer emulation.
- Emulation
- cpu.c 6809E emulation.
- mem.c memory emulation module.
- sam.c SAM emulation call-back functions.
- vdg.c VDG emulation.
- pia.c PIA emulation call-back functions.
- Utilities and drivers
- loader.c ROM and CAS file loader/manager.
- sdfat32.c SD card reader for FAT32 file system.
- printf.c printf() replacement for bare metal.
- RPi bare metal code modules
- rpibm.c Raspberry Pi hardware specific functions.
- gpio.c RPi GPIO manipulation.
- auxuart.c auxiliary UART driver.
- irq.c interrupt driver.
- irq_utils.S interrupt utilities.
- mailbox.c video core (VC) mailbox interface.
- spi0.c SPI0 driver.
- spi1.c auxiliary SPI (SPI1) driver.
- start.S bare metal startup code.
- timer.c system timer driver.
- Miscellaneous
- README.md this file.
- LICENSE.md license.
- Makefile make file.
- include/ include files.