2018-06-16 11:48:22 +00:00
# Aiie!
2017-02-19 18:46:13 +00:00
Aiie! is an Apple //e emulator, written ground-up for the Teensy
2020-12-29 04:36:25 +00:00
4.1 (originally for the Teensy 3.6).
2017-02-19 18:46:13 +00:00
The name comes from a game I used to play on the Apple //e back
around 1986 - Ali Baba and the Forty Thieves, published by Quality
Software in 1981.
2018-06-16 11:48:22 +00:00
[http://crpgaddict.blogspot.com/2013/07/game-103-ali-baba-and-forty-thieves-1981.html ](http://crpgaddict.blogspot.com/2013/07/game-103-ali-baba-and-forty-thieves-1981.html )
2017-02-19 18:46:13 +00:00
When characters in the game did damage to each other, they exclaimed
something like "HAH! JUST A SCRATCH!" or "AAARGH!" or "OH MA, I THINK
ITS MY TIME" [sic]. One of these exclamations was "AIIYEEEEE!!"
2018-06-16 11:48:22 +00:00
## Build log:
2017-02-19 18:46:13 +00:00
2018-06-16 11:48:22 +00:00
[https://hackaday.io/project/19925-aiie-an-embedded-apple-e-emulator ](https://hackaday.io/project/19925-aiie-an-embedded-apple-e-emulator )
2017-02-19 18:46:13 +00:00
2018-06-16 11:48:22 +00:00
# Getting the ROMs
2017-02-20 00:08:29 +00:00
As with many emulators, you have to go get the ROMs yourself. I've got
the ROMs that I dumped out of my Apple //e. You can probably get yours
a lot easier.
2017-12-29 20:35:47 +00:00
There are four files that you'll need:
2017-02-20 00:08:29 +00:00
2017-02-20 00:15:40 +00:00
* apple2e.rom -- a 32k dump of the entire Apple //e ROM
* disk.rom -- a 256 byte dump of the DiskII controller ROM (16-sector P5)
2017-02-21 01:37:11 +00:00
* parallel.rom -- a 256 byte dump of the Apple Parallel Card
2017-12-29 20:35:47 +00:00
* HDDRVR.BIN -- a 256 byte hard drive driver from AppleWin
(https://github.com/AppleWin/AppleWin/blob/master/firmware/HDD/HDDRVR.BIN)
2017-02-20 00:08:29 +00:00
2017-02-20 23:41:46 +00:00
The MD5 sums of those files are:
2017-02-20 00:08:29 +00:00
2017-02-20 00:15:40 +00:00
* 003a780b461c96ae3e72861ed0f4d3d9 apple2e.rom
* 2020aa1413ff77fe29353f3ee72dc295 disk.rom
2017-02-20 23:41:46 +00:00
* 5902996f16dc78fc013f6e1db14805b3 parallel.rom
2021-12-11 13:42:19 +00:00
(this is the "parallel mode" ROM, not the "Centronics mode" ROM,
and is availble on the Asimov mirror.)
* e91f379957d87aa0af0c7255f6ee6ba0 HDDRVR.BIN (from 2016) although
e0a40e9166af27b16f60beb83c9233f0 (from 2021) seems to be fine.
2017-02-20 23:41:46 +00:00
2017-02-20 00:08:29 +00:00
From those, the appropriate headers will be automatically generated by
"make roms" (or any other target that relies on the ROMs).
2018-06-16 11:48:22 +00:00
# Building (for the Teensy)
2017-02-20 18:55:16 +00:00
The directory 'teensy' contains 'teensy.ino' - the Arduino development
environment project file. You'll need to open that up and compile from
within.
However.
I built this on a Mac, and I used a lot of symlinks because of
limitations in the Arduino IDE. There's no reason that shouldn't work
under Linux, but I have absolutely no idea what Windows will make of
it. I would expect trouble. No, I won't accept pull requests that
remove the symlinks and replace them with the bare files. Sorry.
Also, you'll have to build the ROM headers (above) with 'make roms'
before you can build the Teensy .ino file.
If anyone knows how to make the Arduino development environment do any
form of scripting that could be used to generate those headers, I'd
gladly adopt that instead of forcing folks to run the Perl script via
Makefile. And if you have a better way of dealing with subfolders of
code, with the Teensy-specific code segregated as it is, I'm all ears...
2021-12-11 13:42:19 +00:00
I compile this with optimization set to "Faster" for the Teensy 4.1 at
600MHz. I've been successful underclocking it to increase the battery
life. YMMV.
2017-02-20 18:55:16 +00:00
2018-06-16 11:48:22 +00:00
## Environment and Libraries
2017-02-20 18:55:16 +00:00
2020-12-29 04:36:25 +00:00
I built this with arduino 1.8.13 and TeensyDuino 1.54b5.
2017-02-20 18:55:16 +00:00
https://www.pjrc.com/teensy/td_download.html
These libraries I'm using right from Teensy's environment: TimerOne;
2020-12-29 04:36:25 +00:00
SPI; EEPROM; Time; Keypad; SdFat (previously called "SdFat-beta" but
renamed in TeensyDuino 1.54).
2017-02-20 18:55:16 +00:00
2021-01-21 01:54:53 +00:00
You'll also need the ILI9341_t3n library from
https://github.com/KurtE/ILI9341_t3n/
2022-02-02 12:17:05 +00:00
As of this writing, the master branch does not work for Aiie; but the
branch "dma_new_fix" is fine. I'd recommend checking out that branch
if it exists.
2021-01-21 01:54:53 +00:00
2018-06-16 11:48:22 +00:00
# Running (on the Teensy)
2017-02-20 18:55:16 +00:00
2020-12-29 04:36:25 +00:00
The reset/menu button brings up a BIOS menu with options like:
2017-02-20 18:55:16 +00:00
Resume
Reset
Cold Reboot
Drop to Monitor
Debug: off
2020-12-29 04:36:25 +00:00
Suspend/Restore VM
Display: RGB
CPU Speed: Normal (1.023 MHz)
Paddle X/Y inverted
Configure paddles
Volume +/-
2017-02-20 18:55:16 +00:00
Insert/Eject Disk 1
Insert/Eject Disk 2
2018-01-03 14:30:51 +00:00
Insert/Eject HD 1
Insert/Eject HD 2
2017-02-20 18:55:16 +00:00
2018-06-16 11:48:22 +00:00
## Reset
2017-02-20 18:55:16 +00:00
This is the same as control-reset on the actual hardware. If you
want to execute the Apple //e self-test, then hold down the two
joystick buttons; hit the reset/menu key; and select "Reset".
2018-06-16 11:48:22 +00:00
## Cold Reboot
2017-02-20 18:55:16 +00:00
This resets much of the hardware to a default state and forces a
2020-12-29 04:36:25 +00:00
reboot. It ejects any inserted disks. (You can get the self-test using
this, too.)
2017-02-20 18:55:16 +00:00
2018-06-16 11:48:22 +00:00
## Drop to Monitor
2017-02-20 18:55:16 +00:00
"Drop to Monitor" tries fairly hard to get you back to a monitor
prompt. Useful for debugging, probably not for much else.
2018-06-16 11:48:22 +00:00
## Display
2017-02-20 18:55:16 +00:00
"Display" has four values, and they're only really implemented for
text and hi-res modes (not for lo-res modes). To describe them, I have
to talk about the details of the Apple II display system.
In hires modes, the Apple II can only display certain colors in
certain horizontal pixel columns. Because of how the composite video
out works, the color "carries over" from one pixel to its neighbor;
multiple pixels turned on in a row makes them all white. Which means
that, if you're trying to display a picture in hires mode, you get
color artifacts on the edges of white areas.
The Apple Color Composite Monitor had a button on it that turned on
"Monochrome" mode, with the finer resolution necessary to display the
pixels without the color cast. So its two display modes would be the
ones I call "NTSC-like" and "Black and White."
There are two other video modes. The "RGB" mode (the default, because
it's my preference) shows the color pixels as they're actually drawn
in to memory. That means there can't be a solid field of, for example,
orange; there can only be vertical stripes of orange with black
between them.
The last mode is "Monochrome" which looks like the original "Monitor
II", a black-and-green display.
2018-06-16 11:48:22 +00:00
## Debug
2017-02-20 18:55:16 +00:00
This has several settings:
off
Show FPS
Show mem free
Show paddles
2020-12-29 04:36:25 +00:00
Show PC (program counter)
Show cycles (CPU run cycle count)
Show battery (raw data and percentage)
Show time (clock time)
Show disk (selected drive / head position)
2017-02-20 18:55:16 +00:00
... these are all fairly self-explanatory.
2018-01-03 14:30:51 +00:00
2018-06-16 11:48:22 +00:00
## Insert/Eject Disk1/2 HD1/2
2018-01-03 14:30:51 +00:00
2020-12-29 04:36:25 +00:00
Fairly self-explanatory. Disks may be .dsk, .po, .nib, or .woz images
2018-01-03 14:30:51 +00:00
(although .nib images aren't very heavily tested, particularly for
write support). Hard drives are raw 32MB files, whose filenames must
end in .img.
2018-06-16 11:48:22 +00:00
## Suspend and Restore
2018-01-03 14:30:51 +00:00
The Teensy can be fully suspended and restored - including what disks
are inserted. It's a full VM hibernation. It currently writes to a
file named "suspend.vm" in the root of the MicroSD card. (I would like
to be able to select from multiple suspend/restore files
eventually. It wouldn't be terribly hard; it's just that the BIOS
interface is very limited.)
2018-06-16 11:48:22 +00:00
# Building (on a Mac)
2017-02-19 23:55:54 +00:00
While this isn't the purpose of the emulator, it is functional, and is
my first test target for most of the work. With MacOS 10.11.6 and
Homebrew, you can build and run it like this:
2018-06-16 11:48:22 +00:00
```
2017-02-20 00:18:42 +00:00
< pre >
2020-12-29 04:36:25 +00:00
$ make sdl
2018-01-03 14:30:51 +00:00
$ ./aiie-sdl /path/to/disk.dsk
2017-02-20 00:18:42 +00:00
< / pre >
2018-06-16 11:48:22 +00:00
```
2017-02-19 23:55:54 +00:00
2018-01-03 14:30:51 +00:00
As the name implies, this requires that SDL is installed and in
2017-02-19 23:55:54 +00:00
/usr/local/lib. I've done that with Homebrew like this
2018-06-16 11:48:22 +00:00
```
2017-02-20 00:18:42 +00:00
< pre >
2018-01-03 14:30:51 +00:00
$ brew install sdl2
2017-02-20 00:18:42 +00:00
< / pre >
2018-06-16 11:48:22 +00:00
```
2017-02-19 23:55:54 +00:00
2020-12-29 04:36:25 +00:00
When running, F10 enters the BIOS.
2018-06-16 11:48:22 +00:00
# Building (on Linux)
I've been experimenting with Aiie running under a handmade OS on a Raspberry Pi Zero W; the hardware is decent, and cheap. I just don't want Linux in the way. So I built JOSS (see [https://hackaday.io/project/19925-aiie-an-embedded-apple-e-emulator/log/87286-entry-18-pi-zero-w-and-joss ](my Hackaday page about JOSS )).
Well, performance under JOSS is poor, so I built a Linux framebuffer wrapper for Aiie so that I can do performance testing on the Zero W, directly between JOSS and Linux.
```
$ make linuxfb
$ ./linuxfb
```
# Mockingboard
2017-02-24 15:15:17 +00:00
Mockingboard support is slowly taking shape, based on the schematic in
the Apple II Documentation Project:
https://mirrors.apple2.org.za/Apple%20II%20Documentation%20Project/Interface%20Cards/Audio/Sweet%20Microsystems%20Mockingboard/Schematics/Mockingboard%20Schematic.gif
2020-12-29 04:36:25 +00:00
It was difficult to shoehorn this in to the Teensy 3.6, but with the
Teensy 4.1 it might be possible. More work required, though.
2018-01-03 14:30:51 +00:00
2018-06-16 11:48:22 +00:00
# VM
2017-02-19 23:55:54 +00:00
The virtual machine architecture is broken in half - the virtual and
physical pieces. There's the root VM object (vm.h), which ties
together the MMU, virtual keyboard, and virtual display.
Then there are the physical interfaces, which aren't as well
organized. They exist as globals in globals.cpp:
2017-02-20 00:18:42 +00:00
< pre >
FileManager *g_filemanager = NULL;
2017-02-19 23:55:54 +00:00
PhysicalDisplay *g_display = NULL;
PhysicalKeyboard *g_keyboard = NULL;
PhysicalSpeaker *g_speaker = NULL;
2017-02-20 00:18:42 +00:00
PhysicalPaddles *g_paddles = NULL;
< / pre >
2017-02-19 23:55:54 +00:00
There are the two globals that point to the VM and the virtual CPU:
2017-02-20 00:18:42 +00:00
< pre >
Cpu *g_cpu = NULL;
VM *g_vm = NULL;
< / pre >
2017-02-19 23:55:54 +00:00
And there are two global configuration values that probably belong in
some sort of Prefs class:
2017-02-20 00:18:42 +00:00
< pre >
int16_t g_volume;
uint8_t g_displayType;
< / pre >
2017-02-19 23:55:54 +00:00
2018-06-16 11:48:22 +00:00
# CPU
2017-02-19 18:46:13 +00:00
2018-06-16 11:48:22 +00:00
The CPU is a full and complete 65C02. It supports all of the 65C02
documented and undocumented opcodes:
2017-02-19 18:46:13 +00:00
http://www.oxyron.de/html/opcodes02.html
2018-06-16 11:48:22 +00:00
The timing of the CPU is close to, but not quite, correct. It
2017-02-19 18:46:13 +00:00
doesn't count cycles due to page boundary crossings during branch
2018-06-16 11:48:22 +00:00
instructions, for example. (See the "cycle count footnotes" in cpu.cpp.)
2017-02-19 18:46:13 +00:00
2018-06-16 11:48:22 +00:00
The CPU passes all of the the 6502 tests from here, including the undocumented ADC and SBC handling of Decimal mode and the overflow flag:
2017-02-19 18:46:13 +00:00
https://github.com/Klaus2m5/6502_65C02_functional_tests
2018-06-16 11:48:22 +00:00
Doing a
```
$ make test
```
will build the test harness and execute the three tests that encompass all others. (There are more tests in the **tests/** directory - only 3 of them are truly unique.) Two of the tests should emit
All tests successful!
while the third says
Test complete. Result: passed
2017-02-19 18:46:13 +00:00
2020-08-03 00:38:01 +00:00
# Caveats
2020-12-29 04:36:25 +00:00
This *requires* TeensyDuino 1.54 beta 5, which is the most recent
release as of this writing. There are two sets of functionality needed
here: first, SdFat with long file name support that works with the
Teensy 4; and second, raw USB keyboard scancode support.
Suspend/Restore is untested with the Teensy 4.1 hardware.
The Audio channel needs antialiasing and downsampling support for
cleaner audio. This wasn't a problem on the Teensy 3.6 because the
code was live toggling an actual speaker, just like the Apple did, in
real-time; but on the 4.1, we're using a digital audio interface so
we're experiencing real analog/digital conversion issues in some
situations (particularly code that exploits the physical hardware to
make really sophisticsated sounds).
CPU speed regulation isn't working at the moment; no matter what speed
you pick, you'll get normal full speed.
NIB disks are completely broken at the moment due to the Woz disk
format implementation. Internally, the NIB is converted to a WOZ and
apparently there's a bug somewhere.
The LinuxFB build is currently unmaintained, and definitely broken.
Many, but not all, copy-protected Woz disks work. There appears to be
a subtle timing bug in my disk driver code.
Disk write protection isn't implemented.
2020-08-03 00:38:01 +00:00
2020-12-29 04:36:25 +00:00
If you don't have an SD card inserted when you turn on Aiie, you can't
insert one and use it without power cycling. (The card driver is only
initialized on hardware startup.)
2020-08-03 00:38:01 +00:00
2020-12-29 04:36:25 +00:00
While I do have an ESP-01 wired in to the hardware, I don't have a
working driver written yet.
2017-02-19 18:46:13 +00:00