How to build your own emulator with the SPECTRUSTY library.
This project is maintained by aibolem
This is a part of the tutorial for the SPECTRUSTY library.
In this step, you can learn how to add different ZX Spectrum models with more peripherals.
128 means 128kb RAM. It also means:
I’m going to skip the RS-232
in this tutorial. But otherwise, we’ll add a 128k model to the emulator that features all of the above.
First, update imports:
use spectrusty::audio::{
AudioSample, EarMicAmps4, EarOutAmps4, EarInAmps2,
Blep, BlepStereo, FromSample, UlaAudioFrame,
synth::BandLimited,
carousel::AudioFrameResult,
host::cpal::AudioHandleAnyFormat
};
use spectrusty::bus::{
BusDevice, NullDevice,
joystick::{
MultiJoystickBusDevice, JoystickSelect,
JoystickInterface
}
};
use spectrusty::clock::FTs;
use spectrusty::chip::{
ControlUnit, HostConfig, MemoryAccess,
UlaCommon, Ula128MemFlags, UlaControl,
ThreadSyncTimer,
ula::UlaPAL,
ula128::Ula128
};
use spectrusty::peripherals::{
ZXKeyboardMap,
serial::{SerialKeypad, KeypadKeys},
ay::audio::AyAmps
};
use spectrusty_utils::{
tap::{Tape, Tap},
keyboard::$implementation::{
update_keymap, update_keypad_keys,
update_joystick_from_key_event
}
};
Where $implementation
is one of the available keyboard implementations for event loops.
Next, let’s refactor the ZxSpectrum
struct by replacing UlaPAL
with a generic U
parameter:
#[derive(Default)]
struct ZxSpectrum<C: Cpu, U> {
cpu: C,
ula: U,
nmi_request: bool,
reset_request: Option<bool>,
state: EmulatorState
}
The reset_request
property works similarly to nmi_request
.
Ula128 provides a chipset implementation for ZX Spectrum 128k models. But we also need an emulator of AY-3-8910 PSG connected to the keypad via its IOA port. As you already know, we can “plug” it in as a static bus device. Let’s create a new type Ula128AyKeypad<D>
, for convenience. It will be used as our 128k chipset type:
// a specialized AY-3-8910 bus device with a keypad
use spectrusty::bus::ay::serial128::Ay3_8912Keypad;
// define Ula128 with a static mandatory device
type Ula128AyKeypad<D=TerminatorDevice> = Ula128<Ay3_8912Keypad<D>>;
The exposed D
parameter can be substituted by a device connected to the PSG as its NextDevice.
By default, D
is our TerminatorDevice
.
Now, let’s again define generic models:
type ZxSpectrum16k<C, D> = ZxSpectrum<C, UlaPAL<Memory16k, D>>;
type ZxSpectrum48k<C, D> = ZxSpectrum<C, UlaPAL<Memory48k, D>>;
type ZxSpectrum128k<C, D> = ZxSpectrum<C, Ula128AyKeypad<D>>;
Now, let’s look again at our pluggable-multi-joystick device:
// a pluggable joystick with run-time selectable joystick types
type PluggableMultiJoyBusDevice = OptionalBusDevice<
MultiJoystickBusDevice<
TerminatorDevice>>;
The last device in the chain determines the type of the timestamp used by the whole device chain.
So, let’s look at our TerminatorDevice
from the previous chapter:
type TerminatorDevice = NullDevice<VFrameTs<UlaVideoFrame>>;
Aha! It appears that our device chain is using VFrameTs<UlaVideoFrame>
as its BusDevice::Timestamp.
This can be a problem with our new 128k model, because Ula128 uses different VideoFrame: Ula128VidFrame.
How do we solve this? Can’t we just use another one that can work with all of our models?
Let’s take a look at the ControlUnit implementation of Ula. It appears that we can use a different type for the timestamp. As long as it satisfies the condition: From<VFrameTs<V: VideoFrame>>
.
As always, you can define your own timestamp type, but we already have one that fulfills such a requirement: FTs.
type TerminatorDevice = NullDevice<FTs>;
That’s it. Now we can use either TerminatorDevice
or PluggableMultiJoyBusDevice
with any of our generic models.
Let’s now extend the model enum:
enum ZxSpectrumModel<C: Cpu, D: BusDevice=TerminatorDevice> {
Spectrum16(ZxSpectrum16k<C, D>),
Spectrum48(ZxSpectrum48k<C, D>),
Spectrum128(ZxSpectrum128k<C, D>),
}
… and ModelReq
:
#[derive(Debug, Clone, Copy)]
enum ModelReq {
Spectrum16,
Spectrum48,
Spectrum128,
}
Because the 128k model uses different ROM again, for the sake of simplicity, let’s embed the ROM binaries in the executable, instead of loading them in run time.
// add ROMS to the binary resources
static ROM48: &[u8] = include_bytes!("../resources/roms/48.rom");
static ROM128_0: &[u8] = include_bytes!("../resources/roms/128-0.rom");
static ROM128_1: &[u8] = include_bytes!("../resources/roms/128-1.rom");
You may download ROM files from here.
Let’s also create dedicated methods for initializing each of the model types, concerning different ROMs:
impl<C: Cpu, M: ZxMemory, D: BusDevice> ZxSpectrum<C, UlaPAL<M, D>>
where Self: Default
{
fn new_with_rom() -> Self {
let mut spectrum = Self::default();
let mem = spectrum.ula.memory_mut();
mem.fill_mem(.., random).unwrap();
mem.load_into_rom(ROM48).unwrap();
spectrum
}
}
impl<C: Cpu, D: BusDevice> ZxSpectrum<C, Ula128AyKeypad<D>>
where Self: Default
{
fn new_with_rom() -> Self {
let mut spectrum = Self::default();
let mem = spectrum.ula.memory_mut();
mem.fill_mem(.., random).unwrap();
mem.load_into_rom_bank(0, ROM128_0).unwrap();
mem.load_into_rom_bank(1, ROM128_1).unwrap();
spectrum
}
}
Perhaps it’s time to refactor access to joysticks. We also need a way to access the 128 keypad to make it work with the user keyboard.
The JoystickAccess
trait doesn’t need to be changed. I’ll just list it here for reference:
trait JoystickAccess {
type JoystickInterface: JoystickInterface + ?Sized;
// Universal joystick interface access
fn joystick_interface(&mut self) -> Option<&mut Self::JoystickInterface> {
None
}
// Does nothing by default.
fn select_joystick(&mut self, _joy: usize) {}
fn current_joystick(&self) -> Option<&str> {
None
}
}
Nevertheless, it has to be reimplemented. But instead of doing it for each ZxSpectrum
type, let’s write a single implementation using an intermediate:
type SerialKeypad128 = SerialKeypad<FTs>;
trait DeviceAccess {
type JoystickDevice;
fn joystick_bus_device_mut(&mut self) -> Option<&mut Self::JoystickDevice> {
None
}
fn joystick_bus_device_ref(&self) -> Option<&Self::JoystickDevice> {
None
}
fn keypad128_mut(&mut self) -> Option<&mut SerialKeypad128> {
None
}
}
DeviceAccess
will be implemented directly on the chipset instead of the model struct and will give us conditional access to the joystick device as well as to the 128 keypad.
Having defined an intermediate, we can now implement JoystickAccess
.
impl<C: Cpu, U: UlaCommon> JoystickAccess for ZxSpectrum<C, U>
where U: DeviceAccess<JoystickDevice = PluggableMultiJoyBusDevice>
{
type JoystickInterface = dyn JoystickInterface;
fn joystick_interface(
&mut self
) -> Option<&mut Self::JoystickInterface>
{
let sub_joy = self.state.sub_joy;
self.ula.joystick_bus_device_mut().and_then(|joy_bus_dev| {
joy_bus_dev.as_deref_mut()
.and_then(|j| j.joystick_interface(sub_joy))
})
}
fn select_joystick(&mut self, joy_index: usize) {
if let Some(joy_bus_dev) = self.ula.joystick_bus_device_mut() {
let (joy_dev, index) = JoystickSelect::new_with_index(joy_index)
.map(|(joy_sel, index)|
(Some(MultiJoystickBusDevice::new_with(joy_sel)), index)
)
.unwrap_or((None, 0));
**joy_bus_dev = joy_dev;
self.state.sub_joy = index;
}
}
fn current_joystick(&self) -> Option<&str> {
self.ula.joystick_bus_device_ref()
.and_then(|jbd| jbd.as_deref().map(Into::into))
}
}
The implementation is similar to the one used in step 4. But this time, we are using methods of DeviceAccess
trait to get access to the very specific DeviceAccess::JoystickDevice
implementation. This is reflected in the where
condition set for the generic type U
.
Time to get busy with DeviceAccess
:
// implement for Ula with a default device for completness
impl<M: ZxMemory> DeviceAccess for UlaPAL<M> {
type JoystickDevice = PluggableMultiJoyBusDevice;
}
// implement for Ula with a joystick device
impl<M: ZxMemory> DeviceAccess for UlaPAL<M, PluggableMultiJoyBusDevice> {
type JoystickDevice = PluggableMultiJoyBusDevice;
fn joystick_bus_device_mut(
&mut self
) -> Option<&mut Self::JoystickDevice>
{
Some(self.bus_device_mut())
}
fn joystick_bus_device_ref(&self) -> Option<&Self::JoystickDevice> {
Some(self.bus_device_ref())
}
}
// implement for Ula128 with a default device for completness
impl DeviceAccess for Ula128AyKeypad {
type JoystickDevice = PluggableMultiJoyBusDevice;
fn keypad128_mut(&mut self) -> Option<&mut SerialKeypad128> {
Some(&mut self.bus_device_mut().ay_io.port_a.serial1)
}
}
// implement for Ula128 with a joystick device
impl DeviceAccess for Ula128AyKeypad<PluggableMultiJoyBusDevice> {
type JoystickDevice = PluggableMultiJoyBusDevice;
fn joystick_bus_device_mut(
&mut self
) -> Option<&mut Self::JoystickDevice>
{
Some(self.bus_device_mut().next_device_mut())
}
fn joystick_bus_device_ref(&self) -> Option<&Self::JoystickDevice> {
Some(self.bus_device_ref().next_device_ref())
}
fn keypad128_mut(&mut self) -> Option<&mut SerialKeypad128> {
Some(&mut self.bus_device_mut().ay_io.port_a.serial1)
}
}
As we already know, the joystick device for Ula128 is positioned slightly deeper in the device chain because we made its first device a sound processor. The 128 keypad is connected to the PSG’s IO port A
. 128k ROM routines are using this port for connecting to both the keypad (AUX - serial port 1) and the RS-232 (SER - serial port 2). But in this example, we won’t be using the second serial port for the RS-232 connection. You can check this example to see how to implement both.
Having dealt with devices, now we can focus on the hot-swap function, as it will be slightly more challenging.
use std::io::{self, Read};
impl<C, D, M> From<ZxSpectrumModel<C, D>> for ZxSpectrum<C, UlaPAL<M, D>>
where C: Cpu,
D: BusDevice<Timestamp=FTs> + Default,
M: ZxMemory,
Self: Default
{
fn from(model: ZxSpectrumModel<C, D>) -> Self {
let border = model.border_color();
let mut spectrum = Self::new_with_rom();
let mem_rd = model.read_ram();
let _ = spectrum.ula.memory_mut()
.load_into_mem(M::PAGE_SIZE as u16.., mem_rd);
let (cpu, dev, state) = model.into_cpu_device_and_state();
spectrum.cpu = cpu;
spectrum.state = state;
spectrum.ula.set_border_color(border);
*spectrum.ula.bus_device_mut() = dev;
spectrum
}
}
impl<C, D> From<ZxSpectrumModel<C, D>> for ZxSpectrum<C, Ula128AyKeypad<D>>
where C: Cpu,
D: BusDevice<Timestamp=FTs> + Default,
Self: Default
{
fn from(model: ZxSpectrumModel<C, D>) -> Self {
let border = model.border_color();
let mut spectrum = Self::new_with_rom();
let mem_rd = model.read_ram();
let _ = spectrum.ula.memory_mut().load_into_mem(
<Ula128 as MemoryAccess>::Memory::PAGE_SIZE as u16..,
mem_rd);
let (cpu, dev, state) = model.into_cpu_device_and_state();
spectrum.cpu = cpu;
spectrum.state = state;
spectrum.ula.set_border_color(border);
*spectrum.ula.bus_device_mut().next_device_mut() = dev;
// lock in 48k mode until reset
spectrum.ula.set_ula128_mem_port_value(Ula128MemFlags::ROM_BANK
|Ula128MemFlags::LOCK_MMU);
spectrum
}
}
We need to deal with the fact that the last bank of RAM can be swapped in the 128k model. Instead of copying slices of linear memory, we’ll use a reader to copy the content of the last visible 3 pages of RAM. So the reader can have different implementation concerning the model used.
Now, for the implementation of model enum helpers:
impl<C: Cpu, D> ZxSpectrumModel<C, D>
where D: BusDevice<Timestamp=FTs> + Default
{
fn into_cpu_device_and_state(self) -> (C, D, EmulatorState) {
match self {
ZxSpectrumModel::Spectrum16(spec16) => (
spec16.cpu, spec16.ula.into_bus_device(), spec16.state
),
ZxSpectrumModel::Spectrum48(spec48) => (
spec48.cpu, spec48.ula.into_bus_device(), spec48.state
),
ZxSpectrumModel::Spectrum128(spec128) => (
spec128.cpu,
spec128.ula.into_bus_device().into_next_device(),
spec128.state
),
}
}
// returns a dynamically dispatched reader from RAM
fn read_ram<'a>(&'a self) -> Box<dyn Read + 'a> {
match self {
ZxSpectrumModel::Spectrum16(spec16) =>
Box::new(spec16.ula.memory_ref().ram_ref()
.chain(io::repeat(!0))),
ZxSpectrumModel::Spectrum48(spec48) =>
Box::new(spec48.ula.memory_ref().ram_ref()),
ZxSpectrumModel::Spectrum128(spec128) => {
let mem = spec128.ula.memory_ref();
// returns paged in RAM banks as a chained reader
Box::new(mem.page_ref(1).unwrap()
.chain(mem.page_ref(2).unwrap())
.chain(mem.page_ref(3).unwrap()))
}
}
}
fn border_color(&self) -> BorderColor {
match self {
ZxSpectrumModel::Spectrum16(spec16) =>
spec16.ula.border_color(),
ZxSpectrumModel::Spectrum48(spec48) =>
spec48.ula.border_color(),
ZxSpectrumModel::Spectrum128(spec128) =>
spec128.ula.border_color(),
}
}
// hot-swap hardware models
fn change_model(self, request: ModelReq) -> Self {
use ZxSpectrumModel::*;
match (&self, request) {
(Spectrum16(..), ModelReq::Spectrum16)|
(Spectrum48(..), ModelReq::Spectrum48)|
(Spectrum128(..), ModelReq::Spectrum128) => return self,
_ => {}
}
match request {
ModelReq::Spectrum16 => Spectrum16(self.into()),
ModelReq::Spectrum48 => Spectrum48(self.into()),
ModelReq::Spectrum128 => Spectrum128(self.into())
}
}
}
There are a few things that need to be updated in the run
method:
fn run<C: Cpu, U>(
spectrum: &mut ZxSpectrum<C, U>,
env: HostEnvironment,
) -> Result<Action>
where U: UlaCommon
+ UlaAudioFrame<BandLim>
+ DeviceAccess
+ HostConfig,
ZxSpectrum<C, U>: JoystickAccess
{
//... ✂
// ensure the Blep implementation is prepared for pulses
spectrum.ula.ensure_audio_frame_time(
blep, audio.sample_rate(), U::CPU_HZ as f64);
//... ✂
let mut sync = ThreadSyncTimer::new(U::frame_duration_nanos());
//... ✂
'main: while is_running() {
process_keyboard_events(
|KeyEvent { key, pressed, shift_down, ctrl_down }| {
if !update_joystick_from_key_event(key, pressed, FIRE_KEY,
|| spectrum.joystick_interface())
{
spectrum.update_keyboard(|keymap|
update_keymap(
keymap, key, pressed, shift_down, ctrl_down)
);
spectrum.update_keypad128_keys(|padmap|
update_keypad_keys(
padmap, key, pressed, shift_down || ctrl_down)
);
}
});
//... ✂
}
//... ✂
}
The method’s signature has changed, and the more elaborate constraint is needed as we are now using a generic U
type instead of UlaPAL
struct. Next, we have to make room for the audio frames in the blep
buffer. Different chipsets can have a different number of cycles per frame, and so the duration of the single-frame pass can change. Lastly, we need to add a way to provide keyboard events to the 128 keypad.
UlaCommon is a trait that groups various commonly used functions for ULA family chipsets. UlaAudioFrame is a trait that groups audio-related functions.
A few methods in ZxSpectrum
only need to be changed slightly, and we have a new function - update_keypad128_keys
.
The implementation header, of course, reflects the changes:
impl<C: Cpu, U> ZxSpectrum<C, U>
where U: UlaCommon,
Self: JoystickAccess
{
//... ✂
fn update_keypad128_keys<F: FnOnce(KeypadKeys) -> KeypadKeys>(
&mut self,
update_keys: F
)
where U: DeviceAccess
{
if let Some(keypad) = self.ula.keypad128_mut() {
let padmap = update_keys( keypad.get_key_state() );
keypad.set_key_state(padmap);
}
}
fn run_frame(&mut self) -> Result<(FTs, bool)> {
//... ✂
if self.nmi_request && self.ula.nmi(&mut self.cpu) {
// clear nmi_request only if the triggering succeeded
self.nmi_request = false;
}
if let Some(hard) = self.reset_request.take() {
self.ula.reset(&mut self.cpu, hard);
}
self.ula.execute_next_frame(&mut self.cpu);
//... ✂
}
fn reset(&mut self, hard: bool) {
self.reset_request = Some(hard);
}
fn update_on_user_request(
&mut self,
input: InputRequest
) -> Result<Option<Action>>
{
match menu_id {
//... ✂
Spectrum128 => return Ok(Some(Action::ChangeModel(
ModelReq::Spectrum128))),
//... ✂
}
}
//... ✂
}
Because we have an additional source of the sound - a PSG, we need to update the render_audio
method.
impl<C: Cpu, U> ZxSpectrum<C, U>
//... ✂
{
fn render_audio<B: Blep<SampleDelta=BlepDelta>>(
&mut self, blep: &mut B
) -> usize
where U: UlaAudioFrame<B>
{
self.ula.render_ay_audio_frame::<AyAmps<BlepDelta>>(blep,
[0, 1, 2]);
// (1) add some amplitude steps to the BLEP that correspond to the EAR/MIC line changes
if self.state.audible_tape {
// render both EAR/MIC OUT channel
self.ula.render_earmic_out_audio_frame::<
EarMicAmps4<BlepDelta>
>(blep, 2);
// and the EAR IN channel
self.ula.render_ear_in_audio_frame::<
EarInAmps2<BlepDelta>
>(blep, 2);
}
else {
// render only EAR OUT channel
self.ula.render_earmic_out_audio_frame::<
EarOutAmps4<BlepDelta>
>(blep, 2);
}
// (2) finalize the BLEP frame
self.ula.end_audio_frame(blep)
}
//... ✂
}
You might have noticed that our BandLim
is initialized with a single channel only. But now, we somehow need 3?
Well… we need to change it:
// the type of the Blep implementation
type BandLim = BlepStereo<BandLimited<BlepDelta>>;
and initialize it:
fn main() -> Result<()> {
//... ✂
let mut blep = BlepStereo::build(0.8)(
BandLimited::<BlepDelta>::new(2)
);
//... ✂
}
Wait… why only 2? What is the 3rd channel then?
Thanks to BlepStereo, which wraps a 2-channel Blep. Any channel indexed as 2 or higher will be directed to both channels simultaneously. So if the 1st channel (0
) is left, the 2nd channel (1
) is right, then the 3rd (2
) is the center.
When calling AyAudioFrame::render_ay_audio_frame we can assign the Blep
audio channels to A, B, C
channels of the sound generator. In our example, we have: [0, 1, 2]
, which is A=0
(left), B=1
(right), C=2
(center).
This is the equivalent of ACB
setting used in most of the emulators. For example if you want ABC
you’ll have to specify A=0
(left), B=2
(center), C=1
(right), that is [0, 2, 1]
. For monophonic output, you can use [2, 2, 2]
.
Another thing to consider is that AyAmps can be replaced with AyFuseAmps, or for that matter your own implementation of AmpLevels. Just remember that 4 bits will be used - values 0 to 15.
The last audio-related part to update is produce_audio_frame
:
fn produce_audio_frame<T: AudioSample + FromSample<BlepDelta>>(
output_channels: usize,
outbuf: &mut Vec<T>,
blep: &mut BandLim,
)
{
// the BLEP buffer summing iterator of the channel 0
let sample_iter = blep.sum_iter::<T>(0);
// the number of samples that the iterator will generate
let frame_sample_count = sample_iter.len();
// ensure the size of the audio frame buffer is exactly as we need it
outbuf.resize(frame_sample_count * output_channels, T::silence());
// zip with the other channel
let sample_iter = sample_iter.zip(blep.sum_iter::<T>(1));
// render each sample
for (chans, (lsmp, rsmp)) in outbuf.chunks_mut(output_channels)
.zip(sample_iter) {
// write each sample to each channel
for (ch, sample) in chans.iter_mut().zip(&[lsmp, rsmp]) {
*ch = *sample;
}
}
}
We somewhat assume that the number of output channels for the host audio is at least 2.
And at last the main
function. Let’s also change the default model to 128k.
fn main() -> Result<()> {
//... ✂
// build the hardware
let mut spec128 = ZxSpectrum128k::<Z80NMOS,
PluggableMultiJoyBusDevice
>::new_with_rom();
// if the user provided the file name
if let Some(file_name) = tap_file_name {
//... ✂
spec128.state.tape.tap = Some(Tap::Reader(iter_pulse));
// or instead we could just write:
// spec128.tape.insert_as_reader(tap_file);
spec128.state.audible_tape = true;
spec128.state.flash_tape = true;
}
//... ✂
// width and height of the rendered frame image area in pixels
let (width, height) = <Ula128 as Video>::render_size_pixels(border);
//... ✂
// initialize audio
let frame_duration_nanos =
<Ula128 as HostConfig>::frame_duration_nanos();
//... ✂
let mut spectrum = ZxSpectrumModel::Spectrum128(spec128);
loop {
//... ✂
let req = match &mut spectrum {
Spectrum16(spec16) => run(spec16, env)?,
Spectrum48(spec48) => run(spec48, env)?,
Spectrum128(spec128) => run(spec128, env)?
};
//... ✂
}
Ok(())
}
The example program using minifb and cpal, covering the scope of this tutorial can be run with:
cargo run --bin step5 --release -- resources/iana128.tap
Press [ENTER]
and enjoy the 128k game.
Back to index.