From d25d7d7d408e7d32d7c603fc0b059981435fa881 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 29 Jul 2017 21:56:33 -0400 Subject: [PATCH 01/70] Added the Amstrad CPC as a named target and declared support for its CDT file format. --- OSBindings/Mac/Clock Signal/Info.plist | 18 ++++++++++++++ StaticAnalyser/StaticAnalyser.cpp | 34 +++++++++++--------------- 2 files changed, 32 insertions(+), 20 deletions(-) diff --git a/OSBindings/Mac/Clock Signal/Info.plist b/OSBindings/Mac/Clock Signal/Info.plist index eb8a76d93..cd9bb6f73 100644 --- a/OSBindings/Mac/Clock Signal/Info.plist +++ b/OSBindings/Mac/Clock Signal/Info.plist @@ -224,6 +224,24 @@ Tape Image CFBundleTypeRole Viewer + LSTypeIsPackage + 0 + NSDocumentClass + $(PRODUCT_MODULE_NAME).MachineDocument + + + CFBundleTypeExtensions + + cdt + + CFBundleTypeIconFile + cassette + CFBundleTypeName + Amstrad CPC Tape Image + CFBundleTypeRole + Viewer + LSTypeIsPackage + 0 NSDocumentClass $(PRODUCT_MODULE_NAME).MachineDocument diff --git a/StaticAnalyser/StaticAnalyser.cpp b/StaticAnalyser/StaticAnalyser.cpp index 0d437a692..c8ef57b3a 100644 --- a/StaticAnalyser/StaticAnalyser.cpp +++ b/StaticAnalyser/StaticAnalyser.cpp @@ -40,30 +40,28 @@ typedef int TargetPlatformType; enum class TargetPlatform: TargetPlatformType { Acorn = 1 << 0, - Atari2600 = 1 << 1, - Commodore = 1 << 2, - Oric = 1 << 3, - ZX8081 = 1 << 4, + AmstradCPC = 1 << 1, + Atari2600 = 1 << 2, + Commodore = 1 << 3, + Oric = 1 << 4, + ZX8081 = 1 << 5, - AllTape = Acorn | Commodore | Oric | ZX8081, + AllTape = Acorn | Commodore | Oric | ZX8081 | AmstradCPC, }; using namespace StaticAnalyser; -std::list StaticAnalyser::GetTargets(const char *file_name) -{ +std::list StaticAnalyser::GetTargets(const char *file_name) { std::list targets; // Get the extension, if any; it will be assumed that extensions are reliable, so an extension is a broad-phase // test as to file format. const char *mixed_case_extension = strrchr(file_name, '.'); char *lowercase_extension = nullptr; - if(mixed_case_extension) - { + if(mixed_case_extension) { lowercase_extension = strdup(mixed_case_extension+1); char *parser = lowercase_extension; - while(*parser) - { + while(*parser) { *parser = (char)tolower(*parser); parser++; } @@ -86,18 +84,17 @@ std::list StaticAnalyser::GetTargets(const char *file_name) } catch(...) {} #define Format(extension, list, class, platforms) \ - if(!strcmp(lowercase_extension, extension)) \ - { \ + if(!strcmp(lowercase_extension, extension)) { \ TryInsert(list, class, platforms) \ } - if(lowercase_extension) - { + if(lowercase_extension) { Format("80", tapes, Tape::ZX80O81P, TargetPlatform::ZX8081) // 80 Format("81", tapes, Tape::ZX80O81P, TargetPlatform::ZX8081) // 81 Format("a26", cartridges, Cartridge::BinaryDump, TargetPlatform::Atari2600) // A26 Format("adf", disks, Disk::AcornADF, TargetPlatform::Acorn) // ADF Format("bin", cartridges, Cartridge::BinaryDump, TargetPlatform::Atari2600) // BIN + Format("cdt", tapes, Tape::TZX, TargetPlatform::AmstradCPC) // CDT Format("csw", tapes, Tape::CSW, TargetPlatform::AllTape) // CSW Format("d64", disks, Disk::D64, TargetPlatform::Commodore) // D64 Format("dsd", disks, Disk::SSD, TargetPlatform::Acorn) // DSD @@ -108,14 +105,11 @@ std::list StaticAnalyser::GetTargets(const char *file_name) Format("p81", tapes, Tape::ZX80O81P, TargetPlatform::ZX8081) // P81 // PRG - if(!strcmp(lowercase_extension, "prg")) - { + if(!strcmp(lowercase_extension, "prg")) { // try instantiating as a ROM; failing that accept as a tape try { Insert(cartridges, Cartridge::PRG, TargetPlatform::Commodore) - } - catch(...) - { + } catch(...) { try { Insert(tapes, Tape::PRG, TargetPlatform::Commodore) } catch(...) {} From 5b5720fac036806730f4c96b5fd8a6e363ad72c0 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 30 Jul 2017 21:15:20 -0400 Subject: [PATCH 02/70] Added to the static analyser the most basic through-path for Amstrad CPC content. --- .../Clock Signal.xcodeproj/project.pbxproj | 14 ++++++++++ StaticAnalyser/AmstradCPC/StaticAnalyser.cpp | 23 ++++++++++++++++ StaticAnalyser/AmstradCPC/StaticAnalyser.hpp | 27 +++++++++++++++++++ StaticAnalyser/StaticAnalyser.cpp | 12 +++++---- StaticAnalyser/StaticAnalyser.hpp | 3 ++- 5 files changed, 73 insertions(+), 6 deletions(-) create mode 100644 StaticAnalyser/AmstradCPC/StaticAnalyser.cpp create mode 100644 StaticAnalyser/AmstradCPC/StaticAnalyser.hpp diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 2412376ca..2a74ab3f0 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -46,6 +46,7 @@ 4B30512D1D989E2200B4FED8 /* Drive.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B30512B1D989E2200B4FED8 /* Drive.cpp */; }; 4B3051301D98ACC600B4FED8 /* Plus3.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B30512E1D98ACC600B4FED8 /* Plus3.cpp */; }; 4B37EE821D7345A6006A09A4 /* BinaryDump.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B37EE801D7345A6006A09A4 /* BinaryDump.cpp */; }; + 4B38F3441F2EB3E900D9235D /* StaticAnalyser.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B38F3421F2EB3E900D9235D /* StaticAnalyser.cpp */; }; 4B3940E71DA83C8300427841 /* AsyncTaskQueue.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B3940E51DA83C8300427841 /* AsyncTaskQueue.cpp */; }; 4B3BA0C31D318AEC005DD7A7 /* C1540Tests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4B3BA0C21D318AEB005DD7A7 /* C1540Tests.swift */; }; 4B3BA0CE1D318B44005DD7A7 /* C1540Bridge.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B3BA0C61D318B44005DD7A7 /* C1540Bridge.mm */; }; @@ -532,6 +533,8 @@ 4B30512F1D98ACC600B4FED8 /* Plus3.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = Plus3.hpp; path = Electron/Plus3.hpp; sourceTree = ""; }; 4B37EE801D7345A6006A09A4 /* BinaryDump.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = BinaryDump.cpp; sourceTree = ""; }; 4B37EE811D7345A6006A09A4 /* BinaryDump.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = BinaryDump.hpp; sourceTree = ""; }; + 4B38F3421F2EB3E900D9235D /* StaticAnalyser.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = StaticAnalyser.cpp; path = ../../StaticAnalyser/AmstradCPC/StaticAnalyser.cpp; sourceTree = ""; }; + 4B38F3431F2EB3E900D9235D /* StaticAnalyser.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = StaticAnalyser.hpp; path = ../../StaticAnalyser/AmstradCPC/StaticAnalyser.hpp; sourceTree = ""; }; 4B3940E51DA83C8300427841 /* AsyncTaskQueue.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = AsyncTaskQueue.cpp; path = ../../Concurrency/AsyncTaskQueue.cpp; sourceTree = ""; }; 4B3940E61DA83C8300427841 /* AsyncTaskQueue.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = AsyncTaskQueue.hpp; path = ../../Concurrency/AsyncTaskQueue.hpp; sourceTree = ""; }; 4B3BA0C21D318AEB005DD7A7 /* C1540Tests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = C1540Tests.swift; sourceTree = ""; }; @@ -1249,6 +1252,15 @@ name = Outputs; sourceTree = ""; }; + 4B38F3451F2EB41800D9235D /* AmstradCPC */ = { + isa = PBXGroup; + children = ( + 4B38F3421F2EB3E900D9235D /* StaticAnalyser.cpp */, + 4B38F3431F2EB3E900D9235D /* StaticAnalyser.hpp */, + ); + name = AmstradCPC; + sourceTree = ""; + }; 4B3940E81DA83C8700427841 /* Concurrency */ = { isa = PBXGroup; children = ( @@ -2157,6 +2169,7 @@ 4B5A12581DD55873007A2231 /* Disassembler */, 4BCF1FAC1DADD41F0039D2E7 /* Oric */, 4B14978C1EE4AC6200CE2596 /* ZX80/81 */, + 4B38F3451F2EB41800D9235D /* AmstradCPC */, ); name = StaticAnalyser; sourceTree = ""; @@ -2676,6 +2689,7 @@ 4B8805FE1DD02552003085B1 /* Tape.cpp in Sources */, 4B9CCDA11DA279CA0098B625 /* Vic20OptionsPanel.swift in Sources */, 4B8805F01DCFC99C003085B1 /* Acorn.cpp in Sources */, + 4B38F3441F2EB3E900D9235D /* StaticAnalyser.cpp in Sources */, 4B3051301D98ACC600B4FED8 /* Plus3.cpp in Sources */, 4B30512D1D989E2200B4FED8 /* Drive.cpp in Sources */, 4BCA6CC81D9DD9F000C2D7B2 /* CommodoreROM.cpp in Sources */, diff --git a/StaticAnalyser/AmstradCPC/StaticAnalyser.cpp b/StaticAnalyser/AmstradCPC/StaticAnalyser.cpp new file mode 100644 index 000000000..b66621960 --- /dev/null +++ b/StaticAnalyser/AmstradCPC/StaticAnalyser.cpp @@ -0,0 +1,23 @@ +// +// AmstradCPC.cpp +// Clock Signal +// +// Created by Thomas Harte on 30/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#include "StaticAnalyser.hpp" + +void StaticAnalyser::AmstradCPC::AddTargets( + const std::list> &disks, + const std::list> &tapes, + const std::list> &cartridges, + std::list &destination) { + Target target; + target.machine = Target::AmstradCPC; + target.probability = 1.0; + target.disks = disks; + target.tapes = tapes; + target.cartridges = cartridges; + destination.push_back(target); +} diff --git a/StaticAnalyser/AmstradCPC/StaticAnalyser.hpp b/StaticAnalyser/AmstradCPC/StaticAnalyser.hpp new file mode 100644 index 000000000..f798f5bcd --- /dev/null +++ b/StaticAnalyser/AmstradCPC/StaticAnalyser.hpp @@ -0,0 +1,27 @@ +// +// StaticAnalyser.hpp +// Clock Signal +// +// Created by Thomas Harte on 30/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#ifndef StaticAnalyser_AmstradCPC_StaticAnalyser_hpp +#define StaticAnalyser_AmstradCPC_StaticAnalyser_hpp + +#include "../StaticAnalyser.hpp" + +namespace StaticAnalyser { +namespace AmstradCPC { + +void AddTargets( + const std::list> &disks, + const std::list> &tapes, + const std::list> &cartridges, + std::list &destination +); + +} +} + +#endif /* StaticAnalyser_AmstradCPC_StaticAnalyser_hpp */ diff --git a/StaticAnalyser/StaticAnalyser.cpp b/StaticAnalyser/StaticAnalyser.cpp index c8ef57b3a..ff1502617 100644 --- a/StaticAnalyser/StaticAnalyser.cpp +++ b/StaticAnalyser/StaticAnalyser.cpp @@ -12,6 +12,7 @@ // Analysers #include "Acorn/StaticAnalyser.hpp" +#include "AmstradCPC/StaticAnalyser.hpp" #include "Atari/StaticAnalyser.hpp" #include "Commodore/StaticAnalyser.hpp" #include "Oric/StaticAnalyser.hpp" @@ -129,11 +130,12 @@ std::list StaticAnalyser::GetTargets(const char *file_name) { // Hand off to platform-specific determination of whether these things are actually compatible and, // if so, how to load them. (TODO) - if(potential_platforms & (TargetPlatformType)TargetPlatform::Acorn) Acorn::AddTargets(disks, tapes, cartridges, targets); - if(potential_platforms & (TargetPlatformType)TargetPlatform::Atari2600) Atari::AddTargets(disks, tapes, cartridges, targets); - if(potential_platforms & (TargetPlatformType)TargetPlatform::Commodore) Commodore::AddTargets(disks, tapes, cartridges, targets); - if(potential_platforms & (TargetPlatformType)TargetPlatform::Oric) Oric::AddTargets(disks, tapes, cartridges, targets); - if(potential_platforms & (TargetPlatformType)TargetPlatform::ZX8081) ZX8081::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::Acorn) Acorn::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::AmstradCPC) AmstradCPC::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::Atari2600) Atari::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::Commodore) Commodore::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::Oric) Oric::AddTargets(disks, tapes, cartridges, targets); + if(potential_platforms & (TargetPlatformType)TargetPlatform::ZX8081) ZX8081::AddTargets(disks, tapes, cartridges, targets); free(lowercase_extension); } diff --git a/StaticAnalyser/StaticAnalyser.hpp b/StaticAnalyser/StaticAnalyser.hpp index 982dc1315..ac1aca5ee 100644 --- a/StaticAnalyser/StaticAnalyser.hpp +++ b/StaticAnalyser/StaticAnalyser.hpp @@ -52,10 +52,11 @@ enum class ZX8081MemoryModel { */ struct Target { enum { + AmstradCPC, Atari2600, Electron, - Vic20, Oric, + Vic20, ZX8081 } machine; float probability; From c0f131383066cb008c20abb8cab25192be934475 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 30 Jul 2017 22:05:29 -0400 Subject: [PATCH 03/70] Performed sufficient wiring to get to the point where attempting to load a CDT creates an instance of the Amstrad CPC and then fails only because the thing vends a `nullptr` CRT. --- Machines/AmstradCPC/AmstradCPC.cpp | 38 +++++++++++++++ Machines/AmstradCPC/AmstradCPC.hpp | 41 ++++++++++++++++ .../Clock Signal.xcodeproj/project.pbxproj | 32 +++++++++++++ .../Base.lproj/AmstradCPCOptions.xib | 48 +++++++++++++++++++ .../StaticAnalyser/CSStaticAnalyser.mm | 47 ++++++++---------- .../Machine/Wrappers/CSAmstradCPC.h | 13 +++++ .../Machine/Wrappers/CSAmstradCPC.mm | 23 +++++++++ 7 files changed, 216 insertions(+), 26 deletions(-) create mode 100644 Machines/AmstradCPC/AmstradCPC.cpp create mode 100644 Machines/AmstradCPC/AmstradCPC.hpp create mode 100644 OSBindings/Mac/Clock Signal/Base.lproj/AmstradCPCOptions.xib create mode 100644 OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h create mode 100644 OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp new file mode 100644 index 000000000..e3890bf17 --- /dev/null +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -0,0 +1,38 @@ +// +// AmstradCPC.cpp +// Clock Signal +// +// Created by Thomas Harte on 30/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#include "AmstradCPC.hpp" + +using namespace AmstradCPC; + +HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle) { + return HalfCycles(0); +} + +void Machine::flush() { +} + +void Machine::setup_output(float aspect_ratio) { +} + +void Machine::close_output() { +} + +std::shared_ptr Machine::get_crt() { + return nullptr; +} + +std::shared_ptr Machine::get_speaker() { + return nullptr; +} + +void Machine::run_for(const Cycles cycles) { +} + +void Machine::configure_as_target(const StaticAnalyser::Target &target) { +} diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp new file mode 100644 index 000000000..fc1e74794 --- /dev/null +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -0,0 +1,41 @@ +// +// AmstradCPC.hpp +// Clock Signal +// +// Created by Thomas Harte on 30/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#ifndef AmstradCPC_hpp +#define AmstradCPC_hpp + +#include "../ConfigurationTarget.hpp" +#include "../CRTMachine.hpp" + +#include "../../Processors/Z80/Z80.hpp" +#include "../../Components/AY38910/AY38910.hpp" + +namespace AmstradCPC { + +class Machine: + public CPU::Z80::Processor, + public CRTMachine::Machine, + public ConfigurationTarget::Machine { + public: + HalfCycles perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle); + void flush(); + + void setup_output(float aspect_ratio); + void close_output(); + + std::shared_ptr get_crt(); + std::shared_ptr get_speaker(); + + void run_for(const Cycles cycles); + + void configure_as_target(const StaticAnalyser::Target &target); +}; + +} + +#endif /* AmstradCPC_hpp */ diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 2a74ab3f0..6646ad483 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -47,6 +47,9 @@ 4B3051301D98ACC600B4FED8 /* Plus3.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B30512E1D98ACC600B4FED8 /* Plus3.cpp */; }; 4B37EE821D7345A6006A09A4 /* BinaryDump.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B37EE801D7345A6006A09A4 /* BinaryDump.cpp */; }; 4B38F3441F2EB3E900D9235D /* StaticAnalyser.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B38F3421F2EB3E900D9235D /* StaticAnalyser.cpp */; }; + 4B38F3481F2EC11D00D9235D /* AmstradCPC.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B38F3461F2EC11D00D9235D /* AmstradCPC.cpp */; }; + 4B38F34C1F2EC3CA00D9235D /* CSAmstradCPC.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B38F34B1F2EC3CA00D9235D /* CSAmstradCPC.mm */; }; + 4B38F34F1F2EC6BA00D9235D /* AmstradCPCOptions.xib in Resources */ = {isa = PBXBuildFile; fileRef = 4B38F34D1F2EC6BA00D9235D /* AmstradCPCOptions.xib */; }; 4B3940E71DA83C8300427841 /* AsyncTaskQueue.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B3940E51DA83C8300427841 /* AsyncTaskQueue.cpp */; }; 4B3BA0C31D318AEC005DD7A7 /* C1540Tests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4B3BA0C21D318AEB005DD7A7 /* C1540Tests.swift */; }; 4B3BA0CE1D318B44005DD7A7 /* C1540Bridge.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B3BA0C61D318B44005DD7A7 /* C1540Bridge.mm */; }; @@ -535,6 +538,11 @@ 4B37EE811D7345A6006A09A4 /* BinaryDump.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = BinaryDump.hpp; sourceTree = ""; }; 4B38F3421F2EB3E900D9235D /* StaticAnalyser.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = StaticAnalyser.cpp; path = ../../StaticAnalyser/AmstradCPC/StaticAnalyser.cpp; sourceTree = ""; }; 4B38F3431F2EB3E900D9235D /* StaticAnalyser.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = StaticAnalyser.hpp; path = ../../StaticAnalyser/AmstradCPC/StaticAnalyser.hpp; sourceTree = ""; }; + 4B38F3461F2EC11D00D9235D /* AmstradCPC.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = AmstradCPC.cpp; path = AmstradCPC/AmstradCPC.cpp; sourceTree = ""; }; + 4B38F3471F2EC11D00D9235D /* AmstradCPC.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = AmstradCPC.hpp; path = AmstradCPC/AmstradCPC.hpp; sourceTree = ""; }; + 4B38F34A1F2EC3CA00D9235D /* CSAmstradCPC.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = CSAmstradCPC.h; sourceTree = ""; }; + 4B38F34B1F2EC3CA00D9235D /* CSAmstradCPC.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = CSAmstradCPC.mm; sourceTree = ""; }; + 4B38F34E1F2EC6BA00D9235D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = "Clock Signal/Base.lproj/AmstradCPCOptions.xib"; sourceTree = SOURCE_ROOT; }; 4B3940E51DA83C8300427841 /* AsyncTaskQueue.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = AsyncTaskQueue.cpp; path = ../../Concurrency/AsyncTaskQueue.cpp; sourceTree = ""; }; 4B3940E61DA83C8300427841 /* AsyncTaskQueue.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = AsyncTaskQueue.hpp; path = ../../Concurrency/AsyncTaskQueue.hpp; sourceTree = ""; }; 4B3BA0C21D318AEB005DD7A7 /* C1540Tests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = C1540Tests.swift; sourceTree = ""; }; @@ -1193,11 +1201,13 @@ 4B2A53981D117D36003C6002 /* Wrappers */ = { isa = PBXGroup; children = ( + 4B38F34A1F2EC3CA00D9235D /* CSAmstradCPC.h */, 4B2A53991D117D36003C6002 /* CSAtari2600.h */, 4B2A539B1D117D36003C6002 /* CSElectron.h */, 4BCF1FA61DADC5250039D2E7 /* CSOric.h */, 4B2A539D1D117D36003C6002 /* CSVic20.h */, 4B14978D1EE4B4D200CE2596 /* CSZX8081.h */, + 4B38F34B1F2EC3CA00D9235D /* CSAmstradCPC.mm */, 4B2A539A1D117D36003C6002 /* CSAtari2600.mm */, 4B2A539C1D117D36003C6002 /* CSElectron.mm */, 4BCF1FA71DADC5250039D2E7 /* CSOric.mm */, @@ -1261,6 +1271,15 @@ name = AmstradCPC; sourceTree = ""; }; + 4B38F3491F2EC12000D9235D /* AmstradCPC */ = { + isa = PBXGroup; + children = ( + 4B38F3461F2EC11D00D9235D /* AmstradCPC.cpp */, + 4B38F3471F2EC11D00D9235D /* AmstradCPC.hpp */, + ); + name = AmstradCPC; + sourceTree = ""; + }; 4B3940E81DA83C8700427841 /* Concurrency */ = { isa = PBXGroup; children = ( @@ -1342,6 +1361,7 @@ 4B2A332E1DB86869002876E3 /* OricOptionsPanel.swift */, 4B9CCDA01DA279CA0098B625 /* Vic20OptionsPanel.swift */, 4B95FA9C1F11893B0008E395 /* ZX8081OptionsPanel.swift */, + 4B38F34D1F2EC6BA00D9235D /* AmstradCPCOptions.xib */, 4B8FE2131DA19D5F0090D3CE /* Atari2600Options.xib */, 4B8FE2171DA19D5F0090D3CE /* ElectronOptions.xib */, 4B8FE2151DA19D5F0090D3CE /* MachineDocument.xib */, @@ -1923,6 +1943,7 @@ 4B8E4ECD1DCE483D003716C3 /* KeyboardMachine.hpp */, 4B2A33291DB8544D002876E3 /* MemoryFuzzer.hpp */, 4B1E85741D170228001EF87D /* Typer.hpp */, + 4B38F3491F2EC12000D9235D /* AmstradCPC */, 4B2E2D961C3A06EC00138695 /* Atari2600 */, 4B4DC81D1D2C2425003C5BF8 /* Commodore */, 4B2E2D9E1C3A070900138695 /* Electron */, @@ -2307,6 +2328,7 @@ 4BB73EAC1B587A5100552FC2 /* MainMenu.xib in Resources */, 4B8FE21D1DA19D5F0090D3CE /* ElectronOptions.xib in Resources */, 4B79E4461E3AF38600141F11 /* floppy525.png in Resources */, + 4B38F34F1F2EC6BA00D9235D /* AmstradCPCOptions.xib in Resources */, 4BC9DF451D044FCA00F44158 /* ROMImages in Resources */, 4B1497981EE4B97F00CE2596 /* ZX8081Options.xib in Resources */, ); @@ -2627,6 +2649,7 @@ 4B2BFDB21DAEF5FF001A68B8 /* Video.cpp in Sources */, 4B4DC82B1D2C27A4003C5BF8 /* SerialBus.cpp in Sources */, 4BC3B74F1CD194CC00F86E85 /* Shader.cpp in Sources */, + 4B38F3481F2EC11D00D9235D /* AmstradCPC.cpp in Sources */, 4B8FE2221DA19FB20090D3CE /* MachinePanel.swift in Sources */, 4BBB14311CD2CECE00BDB55C /* IntermediateShader.cpp in Sources */, 4BD5F1951D13528900631CD1 /* CSBestEffortUpdater.m in Sources */, @@ -2689,6 +2712,7 @@ 4B8805FE1DD02552003085B1 /* Tape.cpp in Sources */, 4B9CCDA11DA279CA0098B625 /* Vic20OptionsPanel.swift in Sources */, 4B8805F01DCFC99C003085B1 /* Acorn.cpp in Sources */, + 4B38F34C1F2EC3CA00D9235D /* CSAmstradCPC.mm in Sources */, 4B38F3441F2EB3E900D9235D /* StaticAnalyser.cpp in Sources */, 4B3051301D98ACC600B4FED8 /* Plus3.cpp in Sources */, 4B30512D1D989E2200B4FED8 /* Drive.cpp in Sources */, @@ -2797,6 +2821,14 @@ name = OricOptions.xib; sourceTree = ""; }; + 4B38F34D1F2EC6BA00D9235D /* AmstradCPCOptions.xib */ = { + isa = PBXVariantGroup; + children = ( + 4B38F34E1F2EC6BA00D9235D /* Base */, + ); + name = AmstradCPCOptions.xib; + sourceTree = ""; + }; 4B8FE2131DA19D5F0090D3CE /* Atari2600Options.xib */ = { isa = PBXVariantGroup; children = ( diff --git a/OSBindings/Mac/Clock Signal/Base.lproj/AmstradCPCOptions.xib b/OSBindings/Mac/Clock Signal/Base.lproj/AmstradCPCOptions.xib new file mode 100644 index 000000000..39bd7024b --- /dev/null +++ b/OSBindings/Mac/Clock Signal/Base.lproj/AmstradCPCOptions.xib @@ -0,0 +1,48 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/OSBindings/Mac/Clock Signal/Machine/StaticAnalyser/CSStaticAnalyser.mm b/OSBindings/Mac/Clock Signal/Machine/StaticAnalyser/CSStaticAnalyser.mm index 72a5dee94..7a3b12b09 100644 --- a/OSBindings/Mac/Clock Signal/Machine/StaticAnalyser/CSStaticAnalyser.mm +++ b/OSBindings/Mac/Clock Signal/Machine/StaticAnalyser/CSStaticAnalyser.mm @@ -14,6 +14,7 @@ #include "StaticAnalyser.hpp" +#import "CSAmstradCPC.h" #import "CSAtari2600.h" #import "CSElectron.h" #import "CSOric.h" @@ -22,16 +23,13 @@ #import "Clock_Signal-Swift.h" -@implementation CSStaticAnalyser -{ +@implementation CSStaticAnalyser { StaticAnalyser::Target _target; } -- (instancetype)initWithFileAtURL:(NSURL *)url -{ +- (instancetype)initWithFileAtURL:(NSURL *)url { self = [super init]; - if(self) - { + if(self) { std::list targets = StaticAnalyser::GetTargets([url fileSystemRepresentation]); if(!targets.size()) return nil; _target = targets.front(); @@ -42,34 +40,31 @@ return self; } -- (NSString *)optionsPanelNibName -{ - switch(_target.machine) - { - case StaticAnalyser::Target::Atari2600: return @"Atari2600Options"; - case StaticAnalyser::Target::Electron: return @"ElectronOptions"; - case StaticAnalyser::Target::Oric: return @"OricOptions"; - case StaticAnalyser::Target::Vic20: return @"Vic20Options"; - case StaticAnalyser::Target::ZX8081: return @"ZX8081Options"; +- (NSString *)optionsPanelNibName { + switch(_target.machine) { + case StaticAnalyser::Target::AmstradCPC: return @"AmstradCPCOptions"; + case StaticAnalyser::Target::Atari2600: return @"Atari2600Options"; + case StaticAnalyser::Target::Electron: return @"ElectronOptions"; + case StaticAnalyser::Target::Oric: return @"OricOptions"; + case StaticAnalyser::Target::Vic20: return @"Vic20Options"; + case StaticAnalyser::Target::ZX8081: return @"ZX8081Options"; default: return nil; } } -- (CSMachine *)newMachine -{ - switch(_target.machine) - { - case StaticAnalyser::Target::Atari2600: return [[CSAtari2600 alloc] init]; - case StaticAnalyser::Target::Electron: return [[CSElectron alloc] init]; - case StaticAnalyser::Target::Oric: return [[CSOric alloc] init]; - case StaticAnalyser::Target::Vic20: return [[CSVic20 alloc] init]; - case StaticAnalyser::Target::ZX8081: return [[CSZX8081 alloc] init]; +- (CSMachine *)newMachine { + switch(_target.machine) { + case StaticAnalyser::Target::AmstradCPC: return [[CSAmstradCPC alloc] init]; + case StaticAnalyser::Target::Atari2600: return [[CSAtari2600 alloc] init]; + case StaticAnalyser::Target::Electron: return [[CSElectron alloc] init]; + case StaticAnalyser::Target::Oric: return [[CSOric alloc] init]; + case StaticAnalyser::Target::Vic20: return [[CSVic20 alloc] init]; + case StaticAnalyser::Target::ZX8081: return [[CSZX8081 alloc] init]; default: return nil; } } -- (void)applyToMachine:(CSMachine *)machine -{ +- (void)applyToMachine:(CSMachine *)machine { [machine applyTarget:_target]; } diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h new file mode 100644 index 000000000..0457ffdf9 --- /dev/null +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h @@ -0,0 +1,13 @@ +// +// CSAmstradCPC.h +// Clock Signal +// +// Created by Thomas Harte on 30/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#import "CSMachine.h" + +@interface CSAmstradCPC : CSMachine + +@end diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm new file mode 100644 index 000000000..f243f9cdc --- /dev/null +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm @@ -0,0 +1,23 @@ +// +// CSAmstradCPC.m +// Clock Signal +// +// Created by Thomas Harte on 30/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#import "CSAmstradCPC.h" + +#include "AmstradCPC.hpp" + +@implementation CSAmstradCPC { + AmstradCPC::Machine _amstradCPC; +} + +- (CRTMachine::Machine * const)machine { + return &_amstradCPC; +} + +- (NSString *)userDefaultsPrefix { return @"amstradCPC"; } + +@end From 1d6fe119066a422499b1a7c1411c77edbb2c8371 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 07:16:51 -0400 Subject: [PATCH 04/70] Added an instance of `Outputs::CRT::CRT`. So progress is now: select CDT, up comes a blank window. --- Machines/AmstradCPC/AmstradCPC.cpp | 8 +++++++- Machines/AmstradCPC/AmstradCPC.hpp | 3 +++ 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index e3890bf17..01cf8bec6 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -18,13 +18,19 @@ void Machine::flush() { } void Machine::setup_output(float aspect_ratio) { + crt_.reset(new Outputs::CRT::CRT(256, 1, Outputs::CRT::DisplayType::PAL50, 1)); + crt_->set_rgb_sampling_function( + "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" + "{" + "return vec3(1.0);" + "}"); } void Machine::close_output() { } std::shared_ptr Machine::get_crt() { - return nullptr; + return crt_; } std::shared_ptr Machine::get_speaker() { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index fc1e74794..905f16cea 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -34,6 +34,9 @@ class Machine: void run_for(const Cycles cycles); void configure_as_target(const StaticAnalyser::Target &target); + + private: + std::shared_ptr crt_; }; } From 9c04d851e4877e72bdb5b0e42f163bb49da0886d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 07:29:50 -0400 Subject: [PATCH 05/70] Added the basics necessary to get the CPU ticking over, at a nominal 4Mhz but with the wait states that I currently believe to be accurate. --- ClockReceiver/ClockReceiver.hpp | 8 ++++++++ Machines/AmstradCPC/AmstradCPC.cpp | 10 ++++++++++ Machines/AmstradCPC/AmstradCPC.hpp | 3 +++ 3 files changed, 21 insertions(+) diff --git a/ClockReceiver/ClockReceiver.hpp b/ClockReceiver/ClockReceiver.hpp index 515d3216b..893ce2875 100644 --- a/ClockReceiver/ClockReceiver.hpp +++ b/ClockReceiver/ClockReceiver.hpp @@ -95,9 +95,17 @@ template class WrappedInt { return *static_cast(this); } + inline T &operator &=(const T &rhs) { + length_ &= rhs.length_; + return *static_cast(this); + } + inline T operator +(const T &rhs) const { return T(length_ + rhs.length_); } inline T operator -(const T &rhs) const { return T(length_ - rhs.length_); } + inline T operator %(const T &rhs) const { return T(length_ % rhs.length_); } + inline T operator &(const T &rhs) const { return T(length_ & rhs.length_); } + inline T operator -() const { return T(- length_); } inline bool operator <(const T &rhs) const { return length_ < rhs.length_; } diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 01cf8bec6..424328fef 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -10,7 +10,16 @@ using namespace AmstradCPC; +Machine::Machine() { + // primary clock is 4Mhz + set_clock_rate(4000000); +} + HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle) { + // Amstrad CPC timing scheme: assert WAIT for three out of four cycles + clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); + set_wait_line(clock_offset_ >= HalfCycles(2)); + return HalfCycles(0); } @@ -38,6 +47,7 @@ std::shared_ptr Machine::get_speaker() { } void Machine::run_for(const Cycles cycles) { + CPU::Z80::Processor::run_for(cycles); } void Machine::configure_as_target(const StaticAnalyser::Target &target) { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 905f16cea..293ca06f7 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -22,6 +22,8 @@ class Machine: public CRTMachine::Machine, public ConfigurationTarget::Machine { public: + Machine(); + HalfCycles perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle); void flush(); @@ -37,6 +39,7 @@ class Machine: private: std::shared_ptr crt_; + HalfCycles clock_offset_; }; } From 26b6c03a2a8833071ad672c0c8d1603d7d6004b8 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 07:30:07 -0400 Subject: [PATCH 06/70] Re-enabled the address sanitiser as a development tool. --- .../xcshareddata/xcschemes/Clock Signal.xcscheme | 1 + 1 file changed, 1 insertion(+) diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme index 6609a35c0..0bbefe859 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -75,6 +75,7 @@ useCustomWorkingDirectory = "NO" ignoresPersistentStateOnLaunch = "NO" debugDocumentVersioning = "YES" + enableAddressSanitizer = "YES" debugServiceExtension = "internal" allowLocationSimulation = "NO"> Date: Mon, 31 Jul 2017 18:44:49 -0400 Subject: [PATCH 07/70] Ensured that ROM images are loaded and passed to the Amstrad CPC. --- Machines/AmstradCPC/AmstradCPC.cpp | 4 +++ Machines/AmstradCPC/AmstradCPC.hpp | 8 +++++ .../Machine/Wrappers/CSAmstradCPC.mm | 35 +++++++++++++++++++ 3 files changed, 47 insertions(+) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 424328fef..35fdd52a7 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -26,6 +26,9 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c void Machine::flush() { } +void Machine::set_rom(ROMType type, std::vector data) { +} + void Machine::setup_output(float aspect_ratio) { crt_.reset(new Outputs::CRT::CRT(256, 1, Outputs::CRT::DisplayType::PAL50, 1)); crt_->set_rgb_sampling_function( @@ -36,6 +39,7 @@ void Machine::setup_output(float aspect_ratio) { } void Machine::close_output() { + crt_.reset(); } std::shared_ptr Machine::get_crt() { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 293ca06f7..239892d75 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -17,6 +17,12 @@ namespace AmstradCPC { +enum ROMType: uint8_t { + OS464, OS664, OS6128, + BASIC464, BASIC664, BASIC6128, + AMSDOS +}; + class Machine: public CPU::Z80::Processor, public CRTMachine::Machine, @@ -37,6 +43,8 @@ class Machine: void configure_as_target(const StaticAnalyser::Target &target); + void set_rom(ROMType type, std::vector data); + private: std::shared_ptr crt_; HalfCycles clock_offset_; diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm index f243f9cdc..48b797154 100644 --- a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm @@ -10,6 +10,10 @@ #include "AmstradCPC.hpp" +#import "CSMachine+Subclassing.h" +#import "NSData+StdVector.h" +#import "NSBundle+DataResource.h" + @implementation CSAmstradCPC { AmstradCPC::Machine _amstradCPC; } @@ -18,6 +22,37 @@ return &_amstradCPC; } +- (instancetype)init { + self = [super init]; + if(self) { + NSDictionary *roms = @{ + @(AmstradCPC::ROMType::OS464) : @"os464", + @(AmstradCPC::ROMType::OS664) : @"os664", + @(AmstradCPC::ROMType::OS6128) : @"os6128", + @(AmstradCPC::ROMType::BASIC464) : @"basic464", + @(AmstradCPC::ROMType::BASIC664) : @"basic664", + @(AmstradCPC::ROMType::BASIC6128) : @"basic6128", + @(AmstradCPC::ROMType::AMSDOS) : @"amsdos", + }; + + for(NSNumber *key in roms.allKeys) { + AmstradCPC::ROMType type = (AmstradCPC::ROMType)key.integerValue; + NSString *name = roms[key]; + NSData *data = [self rom:name]; + if(data) { + _amstradCPC.set_rom(type, data.stdVector8); + } else { + NSLog(@"Amstrad CPC ROM missing: %@", name); + } + } + } + return self; +} + +- (NSData *)rom:(NSString *)name { + return [[NSBundle mainBundle] dataForResource:name withExtension:@"rom" subdirectory:@"ROMImages/AmstradCPC"]; +} + - (NSString *)userDefaultsPrefix { return @"amstradCPC"; } @end From 3c90218c3d71d5d071572be68e96c7cbbfaef482 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 19:15:43 -0400 Subject: [PATCH 08/70] With a very basic stab at something a bit like the memory map (sans paging), execution begins. --- Machines/AmstradCPC/AmstradCPC.cpp | 40 ++++++++++++++++++++++++++++++ Machines/AmstradCPC/AmstradCPC.hpp | 3 +++ 2 files changed, 43 insertions(+) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 35fdd52a7..bd3635564 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -20,6 +20,40 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); set_wait_line(clock_offset_ >= HalfCycles(2)); + // Stop now if no action is strictly required. + if(!cycle.is_terminal()) return HalfCycles(0); + + uint16_t address = cycle.address ? *cycle.address : 0x0000; + switch(cycle.operation) { + case CPU::Z80::PartialMachineCycle::ReadOpcode: + case CPU::Z80::PartialMachineCycle::Read: + switch(address >> 14) { + case 0: *cycle.value = os_[address & 16383]; break; + case 1: case 2: + *cycle.value = ram_[address]; + break; + case 3: *cycle.value = basic_[address & 16383]; break; + } + break; + + case CPU::Z80::PartialMachineCycle::Write: + ram_[address] = *cycle.value; + break; + + case CPU::Z80::PartialMachineCycle::Output: + printf("Output %02x -> %04x?\n", *cycle.value, address); + break; + case CPU::Z80::PartialMachineCycle::Input: + printf("Input %04x?\n", address); + break; + + case CPU::Z80::PartialMachineCycle::Interrupt: + *cycle.value = 0xff; + break; + + default: break; + } + return HalfCycles(0); } @@ -27,6 +61,12 @@ void Machine::flush() { } void Machine::set_rom(ROMType type, std::vector data) { + // Keep only the two ROMs that are currently of interest. + switch(type) { + case ROMType::OS464: os_ = data; break; + case ROMType::BASIC464: basic_ = data; break; + default: break; + } } void Machine::setup_output(float aspect_ratio) { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 239892d75..292ddd434 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -48,6 +48,9 @@ class Machine: private: std::shared_ptr crt_; HalfCycles clock_offset_; + + uint8_t ram_[65536]; + std::vector os_, basic_; }; } From d88ca151f40a43ecd38989461a2ff2513475f19f Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 19:25:10 -0400 Subject: [PATCH 09/70] Added a first attempt at output port decoding. Just logging for now. --- Machines/AmstradCPC/AmstradCPC.cpp | 30 +++++++++++++++++++++++++++++- 1 file changed, 29 insertions(+), 1 deletion(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index bd3635564..cc303395e 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -41,7 +41,35 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c break; case CPU::Z80::PartialMachineCycle::Output: - printf("Output %02x -> %04x?\n", *cycle.value, address); + // Check for a gate array access. + if((address & 0xc000) == 0x4000) { + switch(*cycle.value >> 6) { + case 0: printf("Select pen %02x\n", *cycle.value & 0x1f); break; + case 1: printf("Select colour %02x\n", *cycle.value & 0x1f); break; + case 2: printf("Set mode %d, other flags %02x\n", *cycle.value & 3, (*cycle.value >> 2)&7); break; + case 3: printf("RAM paging?\n"); break; + } + } + + // Check for a CRTC access + if(!(address & 0x4000)) { + switch((address >> 8) & 3) { + case 0: printf("Select CRTC register %d\n", *cycle.value); break; + case 1: printf("Set CRTC value %d\n", *cycle.value); break; + case 2: case 3: printf("Illegal CRTC write?\n"); break; + } + } + + // Check for a PIO access + if(!(address & 0x800)) { + switch((address >> 8) & 3) { + case 0: printf("PSG data: %d\n", *cycle.value); break; + case 1: printf("Vsync, etc: %02x\n", *cycle.value); break; + case 2: printf("Key row, etc: %02x\n", *cycle.value); break; + case 3: printf("PIO control: %02x\n", *cycle.value); break; + } + } +// printf("Output %02x -> %04x?\n", *cycle.value, address); break; case CPU::Z80::PartialMachineCycle::Input: printf("Input %04x?\n", address); From 68dca9d04714496c58de49f90829f1a639bc31e8 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 19:37:28 -0400 Subject: [PATCH 10/70] Made a first attempt at ROM paging, with pretty much the same scheme that'll be needed for 128kb support. --- Machines/AmstradCPC/AmstradCPC.cpp | 47 +++++++++++++++++++++++------- Machines/AmstradCPC/AmstradCPC.hpp | 3 ++ 2 files changed, 39 insertions(+), 11 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index cc303395e..195561726 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -27,17 +27,11 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c switch(cycle.operation) { case CPU::Z80::PartialMachineCycle::ReadOpcode: case CPU::Z80::PartialMachineCycle::Read: - switch(address >> 14) { - case 0: *cycle.value = os_[address & 16383]; break; - case 1: case 2: - *cycle.value = ram_[address]; - break; - case 3: *cycle.value = basic_[address & 16383]; break; - } + *cycle.value = read_pointers_[address >> 14][address & 16383]; break; case CPU::Z80::PartialMachineCycle::Write: - ram_[address] = *cycle.value; + write_pointers_[address >> 14][address & 16383] = *cycle.value; break; case CPU::Z80::PartialMachineCycle::Output: @@ -46,7 +40,11 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c switch(*cycle.value >> 6) { case 0: printf("Select pen %02x\n", *cycle.value & 0x1f); break; case 1: printf("Select colour %02x\n", *cycle.value & 0x1f); break; - case 2: printf("Set mode %d, other flags %02x\n", *cycle.value & 3, (*cycle.value >> 2)&7); break; + case 2: + printf("Set mode %d, other flags %02x\n", *cycle.value & 3, (*cycle.value >> 2)&7); + read_pointers_[0] = (*cycle.value & 4) ? &ram_[0] : os_.data(); + read_pointers_[3] = (*cycle.value & 8) ? &ram_[49152] : basic_.data(); + break; case 3: printf("RAM paging?\n"); break; } } @@ -69,10 +67,28 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c case 3: printf("PIO control: %02x\n", *cycle.value); break; } } -// printf("Output %02x -> %04x?\n", *cycle.value, address); break; case CPU::Z80::PartialMachineCycle::Input: - printf("Input %04x?\n", address); + // Check for a CRTC access + if(!(address & 0x4000)) { + switch((address >> 8) & 3) { + case 0: case 1: printf("Illegal CRTC read?\n"); break; + case 2: printf("CRTC status\n"); break; + case 3: printf("CRTC data in\n"); break; + } + } + + // Check for a PIO access + if(!(address & 0x800)) { + switch((address >> 8) & 3) { + case 0: printf("[In] PSG data\n"); break; + case 1: printf("[In] Vsync, etc\n"); break; + case 2: printf("[In] Key row, etc\n"); break; + case 3: printf("[In] PIO control\n"); break; + } + } + + *cycle.value = 0xff; break; case CPU::Z80::PartialMachineCycle::Interrupt: @@ -123,4 +139,13 @@ void Machine::run_for(const Cycles cycles) { } void Machine::configure_as_target(const StaticAnalyser::Target &target) { + read_pointers_[0] = os_.data(); + read_pointers_[1] = &ram_[16384]; + read_pointers_[2] = &ram_[32768]; + read_pointers_[3] = basic_.data(); + + write_pointers_[0] = &ram_[0]; + write_pointers_[1] = &ram_[16384]; + write_pointers_[2] = &ram_[32768]; + write_pointers_[3] = &ram_[49152]; } diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 292ddd434..322027415 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -51,6 +51,9 @@ class Machine: uint8_t ram_[65536]; std::vector os_, basic_; + + uint8_t *read_pointers_[4]; + uint8_t *write_pointers_[4]; }; } From 68ceeab6104c91e2487bc7fbd8b006457d95c97a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 19:56:59 -0400 Subject: [PATCH 11/70] Created a 6845 class and started pushing data at it and clocking it. It doesn't currently have the concept of a bus but will do, hence the in-header implementation. --- Components/6845/CRTC6845.cpp | 12 +++++ Components/6845/CRTC6845.hpp | 46 +++++++++++++++++++ Machines/AmstradCPC/AmstradCPC.cpp | 17 +++++-- Machines/AmstradCPC/AmstradCPC.hpp | 3 ++ .../Clock Signal.xcodeproj/project.pbxproj | 14 ++++++ 5 files changed, 88 insertions(+), 4 deletions(-) create mode 100644 Components/6845/CRTC6845.cpp create mode 100644 Components/6845/CRTC6845.hpp diff --git a/Components/6845/CRTC6845.cpp b/Components/6845/CRTC6845.cpp new file mode 100644 index 000000000..f8290dae7 --- /dev/null +++ b/Components/6845/CRTC6845.cpp @@ -0,0 +1,12 @@ +// +// CRTC6845.cpp +// Clock Signal +// +// Created by Thomas Harte on 31/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#include "CRTC6845.hpp" + +using namespace Motorola; + diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp new file mode 100644 index 000000000..a0ba250ac --- /dev/null +++ b/Components/6845/CRTC6845.hpp @@ -0,0 +1,46 @@ +// +// CRTC6845.hpp +// Clock Signal +// +// Created by Thomas Harte on 31/07/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#ifndef CRTC6845_hpp +#define CRTC6845_hpp + +#include "../../ClockReceiver/ClockReceiver.hpp" + +#include + +namespace Motorola { + +class CRTC6845 { + public: + void run_for(Cycles cycles) { + } + + void select_register(uint8_t r) { + selected_register_ = (int)r & 15; + } + + uint8_t get_status() { + return 0xff; + } + + uint8_t get_register() { + return registers_[selected_register_]; + } + + void set_register(uint8_t value) { + registers_[selected_register_] = value; + } + + private: + uint8_t registers_[16]; + int selected_register_; +}; + +} + +#endif /* CRTC6845_hpp */ diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 195561726..2b351bac4 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -20,6 +20,15 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); set_wait_line(clock_offset_ >= HalfCycles(2)); + // Update the CRTC on the final two cycles of the clock period; + // this gives properly serialised memory accesses without having + // to emulate a buffer. + crtc_offset_ += cycle.length; + while(crtc_offset_ >= HalfCycles(8)) { + crtc_.run_for(Cycles(2)); + crtc_offset_ -= HalfCycles(8); + } + // Stop now if no action is strictly required. if(!cycle.is_terminal()) return HalfCycles(0); @@ -52,8 +61,8 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c // Check for a CRTC access if(!(address & 0x4000)) { switch((address >> 8) & 3) { - case 0: printf("Select CRTC register %d\n", *cycle.value); break; - case 1: printf("Set CRTC value %d\n", *cycle.value); break; + case 0: crtc_.select_register(*cycle.value); break; + case 1: crtc_.set_register(*cycle.value); break; case 2: case 3: printf("Illegal CRTC write?\n"); break; } } @@ -73,8 +82,8 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c if(!(address & 0x4000)) { switch((address >> 8) & 3) { case 0: case 1: printf("Illegal CRTC read?\n"); break; - case 2: printf("CRTC status\n"); break; - case 3: printf("CRTC data in\n"); break; + case 2: *cycle.value = crtc_.get_status(); break; + case 3: *cycle.value = crtc_.get_register(); break; } } diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 322027415..94cccf5bc 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -14,6 +14,7 @@ #include "../../Processors/Z80/Z80.hpp" #include "../../Components/AY38910/AY38910.hpp" +#include "../../Components/6845/CRTC6845.hpp" namespace AmstradCPC { @@ -47,7 +48,9 @@ class Machine: private: std::shared_ptr crt_; + Motorola::CRTC6845 crtc_; HalfCycles clock_offset_; + HalfCycles crtc_offset_; uint8_t ram_[65536]; std::vector os_, basic_; diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 6646ad483..9b3f804ab 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -424,6 +424,7 @@ 4BDDBA991EF3451200347E61 /* Z80MachineCycleTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4BDDBA981EF3451200347E61 /* Z80MachineCycleTests.swift */; }; 4BE77A2E1D84ADFB00BC3827 /* File.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE77A2C1D84ADFB00BC3827 /* File.cpp */; }; 4BE7C9181E3D397100A5496D /* TIA.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE7C9161E3D397100A5496D /* TIA.cpp */; }; + 4BE845211F2FF7F100A5EA22 /* CRTC6845.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE8451F1F2FF7F100A5EA22 /* CRTC6845.cpp */; }; 4BE9A6B11EDE293000CBCB47 /* zexdoc.com in Resources */ = {isa = PBXBuildFile; fileRef = 4BE9A6B01EDE293000CBCB47 /* zexdoc.com */; }; 4BEA525E1DF33323007E74F2 /* Tape.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BEA525D1DF33323007E74F2 /* Tape.cpp */; }; 4BEA52631DF339D7007E74F2 /* Speaker.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BEA52611DF339D7007E74F2 /* Speaker.cpp */; }; @@ -1001,6 +1002,8 @@ 4BE77A2D1D84ADFB00BC3827 /* File.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = File.hpp; path = ../../StaticAnalyser/Commodore/File.hpp; sourceTree = ""; }; 4BE7C9161E3D397100A5496D /* TIA.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = TIA.cpp; sourceTree = ""; }; 4BE7C9171E3D397100A5496D /* TIA.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = TIA.hpp; sourceTree = ""; }; + 4BE8451F1F2FF7F100A5EA22 /* CRTC6845.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = CRTC6845.cpp; path = 6845/CRTC6845.cpp; sourceTree = ""; }; + 4BE845201F2FF7F100A5EA22 /* CRTC6845.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = CRTC6845.hpp; path = 6845/CRTC6845.hpp; sourceTree = ""; }; 4BE9A6B01EDE293000CBCB47 /* zexdoc.com */ = {isa = PBXFileReference; lastKnownFileType = file; name = zexdoc.com; path = Zexall/zexdoc.com; sourceTree = ""; }; 4BEA525D1DF33323007E74F2 /* Tape.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = Tape.cpp; path = Electron/Tape.cpp; sourceTree = ""; }; 4BEA525F1DF333D8007E74F2 /* Tape.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; name = Tape.hpp; path = Electron/Tape.hpp; sourceTree = ""; }; @@ -2031,6 +2034,7 @@ 4B1E85791D174DEC001EF87D /* 6532 */, 4BC9DF4C1D04691600F44158 /* 6560 */, 4B4A762D1DB1A35C007AAE2E /* AY38910 */, + 4BE845221F2FF7F400A5EA22 /* 6845 */, ); name = Components; path = ../../Components; @@ -2128,6 +2132,15 @@ path = Resources; sourceTree = ""; }; + 4BE845221F2FF7F400A5EA22 /* 6845 */ = { + isa = PBXGroup; + children = ( + 4BE8451F1F2FF7F100A5EA22 /* CRTC6845.cpp */, + 4BE845201F2FF7F100A5EA22 /* CRTC6845.hpp */, + ); + name = 6845; + sourceTree = ""; + }; 4BE9A6B21EDE294200CBCB47 /* Zexall */ = { isa = PBXGroup; children = ( @@ -2732,6 +2745,7 @@ 4B2A53A01D117D36003C6002 /* CSMachine.mm in Sources */, 4B14978F1EE4B4D200CE2596 /* CSZX8081.mm in Sources */, 4BC91B831D1F160E00884B76 /* CommodoreTAP.cpp in Sources */, + 4BE845211F2FF7F100A5EA22 /* CRTC6845.cpp in Sources */, 4BCF1FAB1DADD41B0039D2E7 /* StaticAnalyser.cpp in Sources */, 4B2A539F1D117D36003C6002 /* CSAudioQueue.m in Sources */, 4B37EE821D7345A6006A09A4 /* BinaryDump.cpp in Sources */, From e28829bd1b225a5b5d0363c8a5f38846767508ef Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 20:14:46 -0400 Subject: [PATCH 12/70] Corrected CRTC timing, gave it someone to talk to and a means with which to talk. --- Components/6845/CRTC6845.hpp | 26 +++++++++++++++++++++++++- Machines/AmstradCPC/AmstradCPC.cpp | 19 ++++++++++--------- Machines/AmstradCPC/AmstradCPC.hpp | 12 ++++++++++-- 3 files changed, 45 insertions(+), 12 deletions(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index a0ba250ac..15e6b9201 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -12,12 +12,31 @@ #include "../../ClockReceiver/ClockReceiver.hpp" #include +#include namespace Motorola { +namespace CRTC { -class CRTC6845 { +struct BusState { + bool display_enable; + bool hsync; + bool vsync; + bool cursor; + uint16_t refresh_address; + uint16_t row_address; +}; + +template class CRTC6845 { public: + CRTC6845(T &bus_handler) : bus_handler_(bus_handler) {} + void run_for(Cycles cycles) { + int cyles_remaining = cycles.as_int(); + while(cyles_remaining--) { + // TODO: update state (!) + + bus_handler_.perform_bus_cycle(bus_state_); + } } void select_register(uint8_t r) { @@ -34,13 +53,18 @@ class CRTC6845 { void set_register(uint8_t value) { registers_[selected_register_] = value; + if(!selected_register_) printf("Horizontal total: %d\n", value); } private: + T &bus_handler_; + BusState bus_state_; + uint8_t registers_[16]; int selected_register_; }; +} } #endif /* CRTC6845_hpp */ diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 2b351bac4..40b554b27 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -10,7 +10,9 @@ using namespace AmstradCPC; -Machine::Machine() { +Machine::Machine() : + crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses + crtc_(crtc_bus_handler_) { // primary clock is 4Mhz set_clock_rate(4000000); } @@ -20,14 +22,13 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); set_wait_line(clock_offset_ >= HalfCycles(2)); - // Update the CRTC on the final two cycles of the clock period; - // this gives properly serialised memory accesses without having - // to emulate a buffer. - crtc_offset_ += cycle.length; - while(crtc_offset_ >= HalfCycles(8)) { - crtc_.run_for(Cycles(2)); - crtc_offset_ -= HalfCycles(8); - } + // Update the CRTC once every eight half cycles; aiming for half-cycle 4 as + // per the initial seed to the crtc_counter_, but any time in the final four + // will do as it's safe to conclude that nobody else has touched video RAM + // during that whole window + crtc_counter_ += cycle.length; + int crtc_cycles = crtc_counter_.divide(HalfCycles(8)).as_int(); + if(crtc_cycles) crtc_.run_for(Cycles(1)); // Stop now if no action is strictly required. if(!cycle.is_terminal()) return HalfCycles(0); diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 94cccf5bc..540151a6f 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -24,6 +24,11 @@ enum ROMType: uint8_t { AMSDOS }; +struct CRTCBusHandler { + inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { + } +}; + class Machine: public CPU::Z80::Processor, public CRTMachine::Machine, @@ -48,9 +53,12 @@ class Machine: private: std::shared_ptr crt_; - Motorola::CRTC6845 crtc_; + + CRTCBusHandler crtc_bus_handler_; + Motorola::CRTC::CRTC6845 crtc_; + HalfCycles clock_offset_; - HalfCycles crtc_offset_; + HalfCycles crtc_counter_; uint8_t ram_[65536]; std::vector os_, basic_; From cb0dc7b434db6c73837d2c4aaac91f771fa78c53 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:01:54 -0400 Subject: [PATCH 13/70] I'm sure it's not going to be this easy, but this is a genuine attempt at full horizontal and vertical timing. --- Components/6845/CRTC6845.hpp | 87 +++++++++++++++++++++++++++++++++++- 1 file changed, 85 insertions(+), 2 deletions(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index 15e6b9201..289f25b1c 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -33,8 +33,82 @@ template class CRTC6845 { void run_for(Cycles cycles) { int cyles_remaining = cycles.as_int(); while(cyles_remaining--) { - // TODO: update state (!) + // check for end of horizontal sync + if(hsync_down_counter_) { + hsync_down_counter_--; + if(!hsync_down_counter_) { + bus_state_.hsync = false; + } + } + // advance horizontal counter + character_counter_++; + + // check for start of horizontal sync + if(character_counter_ == registers_[2]) { + hsync_down_counter_ = registers_[3] & 15; + if(hsync_down_counter_) bus_state_.hsync = true; + } + + // check for end of visible characters + if(character_counter_ == registers_[1]) { + character_is_visible_ = false; + } + + // check for end-of-line + if(character_counter_ == registers_[0]+1) { + character_counter_ = 0; + character_is_visible_ = true; + + // check for end of vertical sync + if(vsync_down_counter_) { + vsync_down_counter_--; + if(!vsync_down_counter_) { + bus_state_.vsync = false; + } + } + + if(is_in_adjustment_period_) { + line_counter_++; + if(line_counter_ == registers_[5]) { + line_is_visible_ = true; + line_counter_ = 0; + is_in_adjustment_period_ = false; + line_address_ = (registers_[12] << 8) | registers_[13]; + bus_state_.refresh_address = line_address_; + } + } else { + // advance vertical counter + bus_state_.row_address++; + if(bus_state_.row_address == registers_[9]) { + line_address_ = bus_state_.refresh_address; + bus_state_.row_address = 0; + line_counter_++; + + // check for end of visible lines + if(line_counter_ == registers_[6]) { + line_is_visible_ = false; + } + + // check for start of vertical sync + if(line_counter_ == registers_[7]) { + bus_state_.vsync = true; + vsync_down_counter_ = 16; // TODO + } + + // check for entry into the overflow area + if(line_counter_ == registers_[4]) { + is_in_adjustment_period_ = true; + bus_state_.row_address = 0; + line_counter_ = 0; + } + } else { + bus_state_.refresh_address = line_address_; + } + } + } + + bus_state_.display_enable = line_is_visible_ && line_is_visible_; bus_handler_.perform_bus_cycle(bus_state_); } } @@ -53,7 +127,6 @@ template class CRTC6845 { void set_register(uint8_t value) { registers_[selected_register_] = value; - if(!selected_register_) printf("Horizontal total: %d\n", value); } private: @@ -62,6 +135,16 @@ template class CRTC6845 { uint8_t registers_[16]; int selected_register_; + + uint8_t character_counter_; + uint8_t line_counter_; + + bool character_is_visible_, line_is_visible_; + + int hsync_down_counter_; + int vsync_down_counter_; + bool is_in_adjustment_period_; + uint16_t line_address_; }; } From 5a396f6787204be02dfc1a41fcd9960d1094d53c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:04:31 -0400 Subject: [PATCH 14/70] Added an explicit cast. --- Components/6845/CRTC6845.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index 289f25b1c..c074794c5 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -74,7 +74,7 @@ template class CRTC6845 { line_is_visible_ = true; line_counter_ = 0; is_in_adjustment_period_ = false; - line_address_ = (registers_[12] << 8) | registers_[13]; + line_address_ = (uint16_t)((registers_[12] << 8) | registers_[13]); bus_state_.refresh_address = line_address_; } } else { From 69b99fe127d519f9aa1e4be50b4f216df6c56e06 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:04:52 -0400 Subject: [PATCH 15/70] Transferred ownership of the CRT to the CRTC bus handler, to give it easy access. --- Machines/AmstradCPC/AmstradCPC.cpp | 10 +++++----- Machines/AmstradCPC/AmstradCPC.hpp | 9 +++++---- 2 files changed, 10 insertions(+), 9 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 40b554b27..a9c33348f 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -124,20 +124,20 @@ void Machine::set_rom(ROMType type, std::vector data) { } void Machine::setup_output(float aspect_ratio) { - crt_.reset(new Outputs::CRT::CRT(256, 1, Outputs::CRT::DisplayType::PAL50, 1)); - crt_->set_rgb_sampling_function( + crtc_bus_handler_.crt_.reset(new Outputs::CRT::CRT(256, 1, Outputs::CRT::DisplayType::PAL50, 1)); + crtc_bus_handler_.crt_->set_rgb_sampling_function( "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" "{" - "return vec3(1.0);" + "return vec3(float(texture(texID, coordinate).r) / 255.0);" "}"); } void Machine::close_output() { - crt_.reset(); + crtc_bus_handler_.crt_.reset(); } std::shared_ptr Machine::get_crt() { - return crt_; + return crtc_bus_handler_.crt_; } std::shared_ptr Machine::get_speaker() { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 540151a6f..9584cf285 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -25,8 +25,11 @@ enum ROMType: uint8_t { }; struct CRTCBusHandler { - inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { - } + public: + inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { + } + + std::shared_ptr crt_; }; class Machine: @@ -52,8 +55,6 @@ class Machine: void set_rom(ROMType type, std::vector data); private: - std::shared_ptr crt_; - CRTCBusHandler crtc_bus_handler_; Motorola::CRTC::CRTC6845 crtc_; From f742fd5d4a959a41d229f10e2a016e961924529e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:13:20 -0400 Subject: [PATCH 16/70] Made basic attempt to get something on screen: white where the display is enabled, black for the border. --- Machines/AmstradCPC/AmstradCPC.cpp | 2 +- Machines/AmstradCPC/AmstradCPC.hpp | 22 ++++++++++++++++++++++ 2 files changed, 23 insertions(+), 1 deletion(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index a9c33348f..5beee29b2 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -124,7 +124,7 @@ void Machine::set_rom(ROMType type, std::vector data) { } void Machine::setup_output(float aspect_ratio) { - crtc_bus_handler_.crt_.reset(new Outputs::CRT::CRT(256, 1, Outputs::CRT::DisplayType::PAL50, 1)); + crtc_bus_handler_.crt_.reset(new Outputs::CRT::CRT(1024, 8, Outputs::CRT::DisplayType::PAL50, 1)); crtc_bus_handler_.crt_->set_rgb_sampling_function( "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" "{" diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 9584cf285..ca650b367 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -26,10 +26,32 @@ enum ROMType: uint8_t { struct CRTCBusHandler { public: + CRTCBusHandler() : cycles_(0), was_enabled_(false), was_sync_(false) { + } + inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { + cycles_++; + bool is_sync = state.hsync || state.vsync; + if(state.display_enable != was_enabled_ || is_sync != was_sync_) { + if(was_sync_) { + crt_->output_sync((unsigned int)(cycles_ * 2)); + } else { + uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); + if(colour_pointer) *colour_pointer = was_enabled_ ? 0xff : 0x00; + crt_->output_level((unsigned int)(cycles_ * 2)); + } + + cycles_ = 0; + was_sync_ = is_sync; + was_enabled_ = state.display_enable; + } } std::shared_ptr crt_; + + private: + int cycles_; + bool was_enabled_, was_sync_; }; class Machine: From ffaa627820897e70e0ee66d27951820acc3b21ba Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:13:45 -0400 Subject: [PATCH 17/70] Fixed frame restart when there is no adjustment period. --- Components/6845/CRTC6845.hpp | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index c074794c5..f4c7aba7f 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -98,7 +98,12 @@ template class CRTC6845 { // check for entry into the overflow area if(line_counter_ == registers_[4]) { - is_in_adjustment_period_ = true; + if(registers_[5]) { + is_in_adjustment_period_ = true; + } else { + line_address_ = (uint16_t)((registers_[12] << 8) | registers_[13]); + bus_state_.refresh_address = line_address_; + } bus_state_.row_address = 0; line_counter_ = 0; } @@ -108,7 +113,7 @@ template class CRTC6845 { } } - bus_state_.display_enable = line_is_visible_ && line_is_visible_; + bus_state_.display_enable = character_is_visible_ && line_is_visible_; bus_handler_.perform_bus_cycle(bus_state_); } } From 5c68b6cc210e7f1253847074f3449331dd926b0b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:16:08 -0400 Subject: [PATCH 18/70] Fixed display enable reset when there's no adjustment area. A practical lesson in failure to factor. --- Components/6845/CRTC6845.hpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index f4c7aba7f..f2ef5b93b 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -71,9 +71,9 @@ template class CRTC6845 { if(is_in_adjustment_period_) { line_counter_++; if(line_counter_ == registers_[5]) { - line_is_visible_ = true; line_counter_ = 0; is_in_adjustment_period_ = false; + line_is_visible_ = true; line_address_ = (uint16_t)((registers_[12] << 8) | registers_[13]); bus_state_.refresh_address = line_address_; } @@ -101,6 +101,7 @@ template class CRTC6845 { if(registers_[5]) { is_in_adjustment_period_ = true; } else { + line_is_visible_ = true; line_address_ = (uint16_t)((registers_[12] << 8) | registers_[13]); bus_state_.refresh_address = line_address_; } From c2253c1e0fa4a6d830d47de834236607096bcf2a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:17:46 -0400 Subject: [PATCH 19/70] Fixed multiplier: the dot clock I've used to instantiate the CRT is the pixel clock, not the character clock. --- Machines/AmstradCPC/AmstradCPC.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index ca650b367..054ac211c 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -34,11 +34,11 @@ struct CRTCBusHandler { bool is_sync = state.hsync || state.vsync; if(state.display_enable != was_enabled_ || is_sync != was_sync_) { if(was_sync_) { - crt_->output_sync((unsigned int)(cycles_ * 2)); + crt_->output_sync((unsigned int)(cycles_ * 2) * 8); } else { uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); if(colour_pointer) *colour_pointer = was_enabled_ ? 0xff : 0x00; - crt_->output_level((unsigned int)(cycles_ * 2)); + crt_->output_level((unsigned int)(cycles_ * 2) * 8); } cycles_ = 0; From e3f677fa37b3446ebf23ea8e42ee952cff0b1ccb Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:21:46 -0400 Subject: [PATCH 20/70] I was under-counting row lines. Adjusted comparison. The emulator now produces a solid white square of approximately correct proportions. I'm sure that filling in pixels will reveal the next set of bugs. --- Components/6845/CRTC6845.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index f2ef5b93b..97130dd2a 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -79,7 +79,6 @@ template class CRTC6845 { } } else { // advance vertical counter - bus_state_.row_address++; if(bus_state_.row_address == registers_[9]) { line_address_ = bus_state_.refresh_address; bus_state_.row_address = 0; @@ -109,6 +108,7 @@ template class CRTC6845 { line_counter_ = 0; } } else { + bus_state_.row_address++; bus_state_.refresh_address = line_address_; } } From 3df13cddd4063efaa33bba414570758564c559e3 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:32:04 -0400 Subject: [PATCH 21/70] As per my keenness for cleanliness improvements corresponding to my ever-increasing C++ ability: turned the Amstrad into something that a factory produces, allowing me completely to hide a bunch of implementation details. --- Machines/AmstradCPC/AmstradCPC.cpp | 94 ++++++++++++++++--- Machines/AmstradCPC/AmstradCPC.hpp | 66 +------------ .../Machine/Wrappers/CSAmstradCPC.mm | 10 +- 3 files changed, 92 insertions(+), 78 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 5beee29b2..7fd1534a0 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -8,16 +8,88 @@ #include "AmstradCPC.hpp" +#include "../../Processors/Z80/Z80.hpp" +#include "../../Components/AY38910/AY38910.hpp" +#include "../../Components/6845/CRTC6845.hpp" + using namespace AmstradCPC; -Machine::Machine() : +struct CRTCBusHandler { + public: + CRTCBusHandler() : cycles_(0), was_enabled_(false), was_sync_(false) {} + + inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { + cycles_++; + bool is_sync = state.hsync || state.vsync; + if(state.display_enable != was_enabled_ || is_sync != was_sync_) { + if(was_sync_) { + crt_->output_sync((unsigned int)(cycles_ * 2) * 8); + } else { + uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); + if(colour_pointer) *colour_pointer = was_enabled_ ? 0xff : 0x00; + crt_->output_level((unsigned int)(cycles_ * 2) * 8); + } + + cycles_ = 0; + was_sync_ = is_sync; + was_enabled_ = state.display_enable; + } + } + + std::shared_ptr crt_; + + private: + int cycles_; + bool was_enabled_, was_sync_; +}; + +class ConcreteMachine: + public CPU::Z80::Processor, + public Machine { + public: + ConcreteMachine(); + + HalfCycles perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle); + void flush(); + + void setup_output(float aspect_ratio); + void close_output(); + + std::shared_ptr get_crt(); + std::shared_ptr get_speaker(); + + void run_for(const Cycles cycles); + + void configure_as_target(const StaticAnalyser::Target &target); + + void set_rom(ROMType type, std::vector data); + + private: + CRTCBusHandler crtc_bus_handler_; + Motorola::CRTC::CRTC6845 crtc_; + + HalfCycles clock_offset_; + HalfCycles crtc_counter_; + + uint8_t ram_[65536]; + std::vector os_, basic_; + + uint8_t *read_pointers_[4]; + uint8_t *write_pointers_[4]; +}; + +Machine *Machine::AmstradCPC() { + return new ConcreteMachine; +} + +ConcreteMachine::ConcreteMachine() : crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses crtc_(crtc_bus_handler_) { // primary clock is 4Mhz set_clock_rate(4000000); } -HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle) { +HalfCycles ConcreteMachine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle) { // Amstrad CPC timing scheme: assert WAIT for three out of four cycles clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); set_wait_line(clock_offset_ >= HalfCycles(2)); @@ -111,10 +183,10 @@ HalfCycles Machine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &c return HalfCycles(0); } -void Machine::flush() { +void ConcreteMachine::flush() { } -void Machine::set_rom(ROMType type, std::vector data) { +void ConcreteMachine::set_rom(ROMType type, std::vector data) { // Keep only the two ROMs that are currently of interest. switch(type) { case ROMType::OS464: os_ = data; break; @@ -123,7 +195,7 @@ void Machine::set_rom(ROMType type, std::vector data) { } } -void Machine::setup_output(float aspect_ratio) { +void ConcreteMachine::setup_output(float aspect_ratio) { crtc_bus_handler_.crt_.reset(new Outputs::CRT::CRT(1024, 8, Outputs::CRT::DisplayType::PAL50, 1)); crtc_bus_handler_.crt_->set_rgb_sampling_function( "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" @@ -132,23 +204,23 @@ void Machine::setup_output(float aspect_ratio) { "}"); } -void Machine::close_output() { +void ConcreteMachine::close_output() { crtc_bus_handler_.crt_.reset(); } -std::shared_ptr Machine::get_crt() { +std::shared_ptr ConcreteMachine::get_crt() { return crtc_bus_handler_.crt_; } -std::shared_ptr Machine::get_speaker() { +std::shared_ptr ConcreteMachine::get_speaker() { return nullptr; } -void Machine::run_for(const Cycles cycles) { - CPU::Z80::Processor::run_for(cycles); +void ConcreteMachine::run_for(const Cycles cycles) { + CPU::Z80::Processor::run_for(cycles); } -void Machine::configure_as_target(const StaticAnalyser::Target &target) { +void ConcreteMachine::configure_as_target(const StaticAnalyser::Target &target) { read_pointers_[0] = os_.data(); read_pointers_[1] = &ram_[16384]; read_pointers_[2] = &ram_[32768]; diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 054ac211c..a8c43293b 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -12,10 +12,6 @@ #include "../ConfigurationTarget.hpp" #include "../CRTMachine.hpp" -#include "../../Processors/Z80/Z80.hpp" -#include "../../Components/AY38910/AY38910.hpp" -#include "../../Components/6845/CRTC6845.hpp" - namespace AmstradCPC { enum ROMType: uint8_t { @@ -24,70 +20,12 @@ enum ROMType: uint8_t { AMSDOS }; -struct CRTCBusHandler { - public: - CRTCBusHandler() : cycles_(0), was_enabled_(false), was_sync_(false) { - } - - inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { - cycles_++; - bool is_sync = state.hsync || state.vsync; - if(state.display_enable != was_enabled_ || is_sync != was_sync_) { - if(was_sync_) { - crt_->output_sync((unsigned int)(cycles_ * 2) * 8); - } else { - uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); - if(colour_pointer) *colour_pointer = was_enabled_ ? 0xff : 0x00; - crt_->output_level((unsigned int)(cycles_ * 2) * 8); - } - - cycles_ = 0; - was_sync_ = is_sync; - was_enabled_ = state.display_enable; - } - } - - std::shared_ptr crt_; - - private: - int cycles_; - bool was_enabled_, was_sync_; -}; - class Machine: - public CPU::Z80::Processor, public CRTMachine::Machine, public ConfigurationTarget::Machine { public: - Machine(); - - HalfCycles perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle); - void flush(); - - void setup_output(float aspect_ratio); - void close_output(); - - std::shared_ptr get_crt(); - std::shared_ptr get_speaker(); - - void run_for(const Cycles cycles); - - void configure_as_target(const StaticAnalyser::Target &target); - - void set_rom(ROMType type, std::vector data); - - private: - CRTCBusHandler crtc_bus_handler_; - Motorola::CRTC::CRTC6845 crtc_; - - HalfCycles clock_offset_; - HalfCycles crtc_counter_; - - uint8_t ram_[65536]; - std::vector os_, basic_; - - uint8_t *read_pointers_[4]; - uint8_t *write_pointers_[4]; + static Machine *AmstradCPC(); + virtual void set_rom(ROMType type, std::vector data) = 0; }; } diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm index 48b797154..f5c612fed 100644 --- a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm @@ -15,16 +15,20 @@ #import "NSBundle+DataResource.h" @implementation CSAmstradCPC { - AmstradCPC::Machine _amstradCPC; + std::unique_ptr _amstradCPC; } - (CRTMachine::Machine * const)machine { - return &_amstradCPC; + if(!_amstradCPC) { + _amstradCPC.reset(AmstradCPC::Machine::AmstradCPC()); + } + return _amstradCPC.get(); } - (instancetype)init { self = [super init]; if(self) { + [self machine]; NSDictionary *roms = @{ @(AmstradCPC::ROMType::OS464) : @"os464", @(AmstradCPC::ROMType::OS664) : @"os664", @@ -40,7 +44,7 @@ NSString *name = roms[key]; NSData *data = [self rom:name]; if(data) { - _amstradCPC.set_rom(type, data.stdVector8); + _amstradCPC->set_rom(type, data.stdVector8); } else { NSLog(@"Amstrad CPC ROM missing: %@", name); } From 0536f089e1e623b785d14aafa6c820e80ffb8ab7 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:32:26 -0400 Subject: [PATCH 22/70] Eliminated old-[personal-]fashioned line break. --- OSBindings/Mac/Clock Signal/Machine/CSMachine.mm | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm b/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm index 6ed1e1d79..9fbc7ffbb 100644 --- a/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm +++ b/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm @@ -43,8 +43,7 @@ struct MachineDelegate: CRTMachine::Machine::Delegate { - (instancetype)init { self = [super init]; - if(self) - { + if(self) { _machineDelegate.machine = self; self.machine->set_delegate(&_machineDelegate); _speakerDelegate.machine = self; From 2b168f7383d47d00c766b307fca0f9b08163b123 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:32:56 -0400 Subject: [PATCH 23/70] Disabled the address sanitiser as an every-time run again, as it just pushes my computer a bit too far. --- .../xcshareddata/xcschemes/Clock Signal.xcscheme | 1 - 1 file changed, 1 deletion(-) diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme index 0bbefe859..6609a35c0 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -75,7 +75,6 @@ useCustomWorkingDirectory = "NO" ignoresPersistentStateOnLaunch = "NO" debugDocumentVersioning = "YES" - enableAddressSanitizer = "YES" debugServiceExtension = "internal" allowLocationSimulation = "NO"> Date: Mon, 31 Jul 2017 22:39:25 -0400 Subject: [PATCH 24/70] Shunted method bodies inline, given that there's no need for a declaration/definition distinction. --- Machines/AmstradCPC/AmstradCPC.cpp | 316 ++++++++++++++--------------- 1 file changed, 155 insertions(+), 161 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 7fd1534a0..ae1b5c861 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -36,33 +36,177 @@ struct CRTCBusHandler { } } - std::shared_ptr crt_; + void setup_output(float aspect_ratio) { + crt_.reset(new Outputs::CRT::CRT(1024, 8, Outputs::CRT::DisplayType::PAL50, 1)); + crt_->set_rgb_sampling_function( + "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" + "{" + "return vec3(float(texture(texID, coordinate).r) / 255.0);" + "}"); + } + + void close_output() { + crt_.reset(); + } + + std::shared_ptr get_crt() { + return crt_; + } private: int cycles_; bool was_enabled_, was_sync_; + std::shared_ptr crt_; }; class ConcreteMachine: public CPU::Z80::Processor, public Machine { public: - ConcreteMachine(); + ConcreteMachine() : + crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses + crtc_(crtc_bus_handler_) { + // primary clock is 4Mhz + set_clock_rate(4000000); + } - HalfCycles perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle); - void flush(); + inline HalfCycles perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle) { + // Amstrad CPC timing scheme: assert WAIT for three out of four cycles + clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); + set_wait_line(clock_offset_ >= HalfCycles(2)); - void setup_output(float aspect_ratio); - void close_output(); + // Update the CRTC once every eight half cycles; aiming for half-cycle 4 as + // per the initial seed to the crtc_counter_, but any time in the final four + // will do as it's safe to conclude that nobody else has touched video RAM + // during that whole window + crtc_counter_ += cycle.length; + int crtc_cycles = crtc_counter_.divide(HalfCycles(8)).as_int(); + if(crtc_cycles) crtc_.run_for(Cycles(1)); - std::shared_ptr get_crt(); - std::shared_ptr get_speaker(); + // Stop now if no action is strictly required. + if(!cycle.is_terminal()) return HalfCycles(0); - void run_for(const Cycles cycles); + uint16_t address = cycle.address ? *cycle.address : 0x0000; + switch(cycle.operation) { + case CPU::Z80::PartialMachineCycle::ReadOpcode: + case CPU::Z80::PartialMachineCycle::Read: + *cycle.value = read_pointers_[address >> 14][address & 16383]; + break; - void configure_as_target(const StaticAnalyser::Target &target); + case CPU::Z80::PartialMachineCycle::Write: + write_pointers_[address >> 14][address & 16383] = *cycle.value; + break; - void set_rom(ROMType type, std::vector data); + case CPU::Z80::PartialMachineCycle::Output: + // Check for a gate array access. + if((address & 0xc000) == 0x4000) { + switch(*cycle.value >> 6) { + case 0: printf("Select pen %02x\n", *cycle.value & 0x1f); break; + case 1: printf("Select colour %02x\n", *cycle.value & 0x1f); break; + case 2: + printf("Set mode %d, other flags %02x\n", *cycle.value & 3, (*cycle.value >> 2)&7); + read_pointers_[0] = (*cycle.value & 4) ? &ram_[0] : os_.data(); + read_pointers_[3] = (*cycle.value & 8) ? &ram_[49152] : basic_.data(); + break; + case 3: printf("RAM paging?\n"); break; + } + } + + // Check for a CRTC access + if(!(address & 0x4000)) { + switch((address >> 8) & 3) { + case 0: crtc_.select_register(*cycle.value); break; + case 1: crtc_.set_register(*cycle.value); break; + case 2: case 3: printf("Illegal CRTC write?\n"); break; + } + } + + // Check for a PIO access + if(!(address & 0x800)) { + switch((address >> 8) & 3) { + case 0: printf("PSG data: %d\n", *cycle.value); break; + case 1: printf("Vsync, etc: %02x\n", *cycle.value); break; + case 2: printf("Key row, etc: %02x\n", *cycle.value); break; + case 3: printf("PIO control: %02x\n", *cycle.value); break; + } + } + break; + case CPU::Z80::PartialMachineCycle::Input: + // Check for a CRTC access + if(!(address & 0x4000)) { + switch((address >> 8) & 3) { + case 0: case 1: printf("Illegal CRTC read?\n"); break; + case 2: *cycle.value = crtc_.get_status(); break; + case 3: *cycle.value = crtc_.get_register(); break; + } + } + + // Check for a PIO access + if(!(address & 0x800)) { + switch((address >> 8) & 3) { + case 0: printf("[In] PSG data\n"); break; + case 1: printf("[In] Vsync, etc\n"); break; + case 2: printf("[In] Key row, etc\n"); break; + case 3: printf("[In] PIO control\n"); break; + } + } + + *cycle.value = 0xff; + break; + + case CPU::Z80::PartialMachineCycle::Interrupt: + *cycle.value = 0xff; + break; + + default: break; + } + + return HalfCycles(0); + } + + void flush() {} + + void setup_output(float aspect_ratio) { + crtc_bus_handler_.setup_output(aspect_ratio); + } + + void close_output() { + crtc_bus_handler_.close_output(); + } + + std::shared_ptr get_crt() { + return crtc_bus_handler_.get_crt(); + } + + std::shared_ptr get_speaker() { + return nullptr; + } + + void run_for(const Cycles cycles) { + CPU::Z80::Processor::run_for(cycles); + } + + void configure_as_target(const StaticAnalyser::Target &target) { + // Establish reset memory map as per machine model (or, for now, as a hard-wired 464) + read_pointers_[0] = os_.data(); + read_pointers_[1] = &ram_[16384]; + read_pointers_[2] = &ram_[32768]; + read_pointers_[3] = basic_.data(); + + write_pointers_[0] = &ram_[0]; + write_pointers_[1] = &ram_[16384]; + write_pointers_[2] = &ram_[32768]; + write_pointers_[3] = &ram_[49152]; + } + + void set_rom(ROMType type, std::vector data) { + // Keep only the two ROMs that are currently of interest. + switch(type) { + case ROMType::OS464: os_ = data; break; + case ROMType::BASIC464: basic_ = data; break; + default: break; + } + } private: CRTCBusHandler crtc_bus_handler_; @@ -81,153 +225,3 @@ class ConcreteMachine: Machine *Machine::AmstradCPC() { return new ConcreteMachine; } - -ConcreteMachine::ConcreteMachine() : - crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses - crtc_(crtc_bus_handler_) { - // primary clock is 4Mhz - set_clock_rate(4000000); -} - -HalfCycles ConcreteMachine::perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle) { - // Amstrad CPC timing scheme: assert WAIT for three out of four cycles - clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); - set_wait_line(clock_offset_ >= HalfCycles(2)); - - // Update the CRTC once every eight half cycles; aiming for half-cycle 4 as - // per the initial seed to the crtc_counter_, but any time in the final four - // will do as it's safe to conclude that nobody else has touched video RAM - // during that whole window - crtc_counter_ += cycle.length; - int crtc_cycles = crtc_counter_.divide(HalfCycles(8)).as_int(); - if(crtc_cycles) crtc_.run_for(Cycles(1)); - - // Stop now if no action is strictly required. - if(!cycle.is_terminal()) return HalfCycles(0); - - uint16_t address = cycle.address ? *cycle.address : 0x0000; - switch(cycle.operation) { - case CPU::Z80::PartialMachineCycle::ReadOpcode: - case CPU::Z80::PartialMachineCycle::Read: - *cycle.value = read_pointers_[address >> 14][address & 16383]; - break; - - case CPU::Z80::PartialMachineCycle::Write: - write_pointers_[address >> 14][address & 16383] = *cycle.value; - break; - - case CPU::Z80::PartialMachineCycle::Output: - // Check for a gate array access. - if((address & 0xc000) == 0x4000) { - switch(*cycle.value >> 6) { - case 0: printf("Select pen %02x\n", *cycle.value & 0x1f); break; - case 1: printf("Select colour %02x\n", *cycle.value & 0x1f); break; - case 2: - printf("Set mode %d, other flags %02x\n", *cycle.value & 3, (*cycle.value >> 2)&7); - read_pointers_[0] = (*cycle.value & 4) ? &ram_[0] : os_.data(); - read_pointers_[3] = (*cycle.value & 8) ? &ram_[49152] : basic_.data(); - break; - case 3: printf("RAM paging?\n"); break; - } - } - - // Check for a CRTC access - if(!(address & 0x4000)) { - switch((address >> 8) & 3) { - case 0: crtc_.select_register(*cycle.value); break; - case 1: crtc_.set_register(*cycle.value); break; - case 2: case 3: printf("Illegal CRTC write?\n"); break; - } - } - - // Check for a PIO access - if(!(address & 0x800)) { - switch((address >> 8) & 3) { - case 0: printf("PSG data: %d\n", *cycle.value); break; - case 1: printf("Vsync, etc: %02x\n", *cycle.value); break; - case 2: printf("Key row, etc: %02x\n", *cycle.value); break; - case 3: printf("PIO control: %02x\n", *cycle.value); break; - } - } - break; - case CPU::Z80::PartialMachineCycle::Input: - // Check for a CRTC access - if(!(address & 0x4000)) { - switch((address >> 8) & 3) { - case 0: case 1: printf("Illegal CRTC read?\n"); break; - case 2: *cycle.value = crtc_.get_status(); break; - case 3: *cycle.value = crtc_.get_register(); break; - } - } - - // Check for a PIO access - if(!(address & 0x800)) { - switch((address >> 8) & 3) { - case 0: printf("[In] PSG data\n"); break; - case 1: printf("[In] Vsync, etc\n"); break; - case 2: printf("[In] Key row, etc\n"); break; - case 3: printf("[In] PIO control\n"); break; - } - } - - *cycle.value = 0xff; - break; - - case CPU::Z80::PartialMachineCycle::Interrupt: - *cycle.value = 0xff; - break; - - default: break; - } - - return HalfCycles(0); -} - -void ConcreteMachine::flush() { -} - -void ConcreteMachine::set_rom(ROMType type, std::vector data) { - // Keep only the two ROMs that are currently of interest. - switch(type) { - case ROMType::OS464: os_ = data; break; - case ROMType::BASIC464: basic_ = data; break; - default: break; - } -} - -void ConcreteMachine::setup_output(float aspect_ratio) { - crtc_bus_handler_.crt_.reset(new Outputs::CRT::CRT(1024, 8, Outputs::CRT::DisplayType::PAL50, 1)); - crtc_bus_handler_.crt_->set_rgb_sampling_function( - "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" - "{" - "return vec3(float(texture(texID, coordinate).r) / 255.0);" - "}"); -} - -void ConcreteMachine::close_output() { - crtc_bus_handler_.crt_.reset(); -} - -std::shared_ptr ConcreteMachine::get_crt() { - return crtc_bus_handler_.crt_; -} - -std::shared_ptr ConcreteMachine::get_speaker() { - return nullptr; -} - -void ConcreteMachine::run_for(const Cycles cycles) { - CPU::Z80::Processor::run_for(cycles); -} - -void ConcreteMachine::configure_as_target(const StaticAnalyser::Target &target) { - read_pointers_[0] = os_.data(); - read_pointers_[1] = &ram_[16384]; - read_pointers_[2] = &ram_[32768]; - read_pointers_[3] = basic_.data(); - - write_pointers_[0] = &ram_[0]; - write_pointers_[1] = &ram_[16384]; - write_pointers_[2] = &ram_[32768]; - write_pointers_[3] = &ram_[49152]; -} From 763e3b65d18d66a565ac1dbb651d17556c402146 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 31 Jul 2017 22:46:06 -0400 Subject: [PATCH 25/70] Ensured a proper initial value for `delegate_`. --- Machines/CRTMachine.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/CRTMachine.hpp b/Machines/CRTMachine.hpp index 5ea8d856e..48f7b7991 100644 --- a/Machines/CRTMachine.hpp +++ b/Machines/CRTMachine.hpp @@ -22,7 +22,7 @@ namespace CRTMachine { */ class Machine { public: - Machine() : clock_is_unlimited_(false) {} + Machine() : clock_is_unlimited_(false), delegate_(nullptr) {} /*! Causes the machine to set up its CRT and, if it has one, speaker. The caller guarantees From 9d953421d8282b6499635956585f16917bc9f7d4 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 07:07:43 -0400 Subject: [PATCH 26/70] After a quick check, added a couple of other _delegate initialisations. I should probably find a way to template this. --- Machines/CRTMachine.hpp | 2 +- Storage/Disk/DigitalPhaseLockedLoop.cpp | 3 ++- Storage/Tape/Parsers/Acorn.cpp | 3 ++- 3 files changed, 5 insertions(+), 3 deletions(-) diff --git a/Machines/CRTMachine.hpp b/Machines/CRTMachine.hpp index 48f7b7991..96f2ba7fd 100644 --- a/Machines/CRTMachine.hpp +++ b/Machines/CRTMachine.hpp @@ -57,7 +57,7 @@ class Machine { virtual void machine_did_change_clock_rate(Machine *machine) = 0; virtual void machine_did_change_clock_is_unlimited(Machine *machine) = 0; }; - void set_delegate(Delegate *delegate) { this->delegate_ = delegate; } + void set_delegate(Delegate *delegate) { delegate_ = delegate; } protected: void set_clock_rate(double clock_rate) { diff --git a/Storage/Disk/DigitalPhaseLockedLoop.cpp b/Storage/Disk/DigitalPhaseLockedLoop.cpp index 9e99aa054..8348049c9 100644 --- a/Storage/Disk/DigitalPhaseLockedLoop.cpp +++ b/Storage/Disk/DigitalPhaseLockedLoop.cpp @@ -18,7 +18,8 @@ DigitalPhaseLockedLoop::DigitalPhaseLockedLoop(int clocks_per_bit, size_t length window_length_(clocks_per_bit), offset_history_pointer_(0), offset_history_(length_of_history, 0), - offset_(0) {} + offset_(0), + delegate_(nullptr) {} void DigitalPhaseLockedLoop::run_for(const Cycles cycles) { offset_ += cycles.as_int(); diff --git a/Storage/Tape/Parsers/Acorn.cpp b/Storage/Tape/Parsers/Acorn.cpp index 3cf875198..519fa3fe7 100644 --- a/Storage/Tape/Parsers/Acorn.cpp +++ b/Storage/Tape/Parsers/Acorn.cpp @@ -69,7 +69,8 @@ Shifter::Shifter() : pll_(PLLClockRate / 4800, 15), was_high_(false), input_pattern_(0), - input_bit_counter_(0) { + input_bit_counter_(0), + delegate_(nullptr) { pll_.set_delegate(this); } From 3ae699964fc674337765efc34afdb6012cda84db Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 07:24:29 -0400 Subject: [PATCH 27/70] Ensured an actual pixel stream is supplied for pixel regions. Though it's just a long stream of white pixels for now. So visual output is unchanged. --- Machines/AmstradCPC/AmstradCPC.cpp | 36 +++++++++++++++++++++++++----- 1 file changed, 30 insertions(+), 6 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index ae1b5c861..b9c56f2cd 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -16,18 +16,41 @@ using namespace AmstradCPC; struct CRTCBusHandler { public: - CRTCBusHandler() : cycles_(0), was_enabled_(false), was_sync_(false) {} + CRTCBusHandler() : cycles_(0), was_enabled_(false), was_sync_(false), pixel_data_(nullptr), pixel_pointer_(nullptr) {} inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { cycles_++; + bool is_sync = state.hsync || state.vsync; + // collect some more pixels if output is ongoing + if(!is_sync && state.display_enable) { + if(!pixel_data_) { + pixel_pointer_ = pixel_data_ = crt_->allocate_write_area(320); + } + if(pixel_pointer_) { + *pixel_pointer_++ = 0xff; + + // flush the current buffer if full + if(pixel_pointer_ == pixel_data_ + 320) { + crt_->output_data(320, 16); + pixel_pointer_ = pixel_data_ = nullptr; + } + } + } + + // if a transition between sync/border/pixels just occurred, announce it if(state.display_enable != was_enabled_ || is_sync != was_sync_) { if(was_sync_) { - crt_->output_sync((unsigned int)(cycles_ * 2) * 8); + crt_->output_sync((unsigned int)(cycles_ * 16)); } else { - uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); - if(colour_pointer) *colour_pointer = was_enabled_ ? 0xff : 0x00; - crt_->output_level((unsigned int)(cycles_ * 2) * 8); + if(was_enabled_) { + crt_->output_data((unsigned int)(cycles_ * 16), 16); + pixel_pointer_ = pixel_data_ = nullptr; + } else { + uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); + if(colour_pointer) *colour_pointer = 0x00; + crt_->output_level((unsigned int)(cycles_ * 16)); + } } cycles_ = 0; @@ -37,7 +60,7 @@ struct CRTCBusHandler { } void setup_output(float aspect_ratio) { - crt_.reset(new Outputs::CRT::CRT(1024, 8, Outputs::CRT::DisplayType::PAL50, 1)); + crt_.reset(new Outputs::CRT::CRT(1024, 16, Outputs::CRT::DisplayType::PAL50, 1)); crt_->set_rgb_sampling_function( "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" "{" @@ -57,6 +80,7 @@ struct CRTCBusHandler { int cycles_; bool was_enabled_, was_sync_; std::shared_ptr crt_; + uint8_t *pixel_data_, *pixel_pointer_; }; class ConcreteMachine: From 10a5581aeae819215118eac3d648871f514c6e4b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 07:34:12 -0400 Subject: [PATCH 28/70] Made first attempt at offering some sort of pictographic of actual RAM contents. --- Machines/AmstradCPC/AmstradCPC.cpp | 24 +++++++++++++++++++++--- 1 file changed, 21 insertions(+), 3 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index b9c56f2cd..a99259a18 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -16,7 +16,13 @@ using namespace AmstradCPC; struct CRTCBusHandler { public: - CRTCBusHandler() : cycles_(0), was_enabled_(false), was_sync_(false), pixel_data_(nullptr), pixel_pointer_(nullptr) {} + CRTCBusHandler(uint8_t *ram) : + cycles_(0), + was_enabled_(false), + was_sync_(false), + pixel_data_(nullptr), + pixel_pointer_(nullptr), + ram_(ram) {} inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { cycles_++; @@ -28,7 +34,16 @@ struct CRTCBusHandler { pixel_pointer_ = pixel_data_ = crt_->allocate_write_area(320); } if(pixel_pointer_) { - *pixel_pointer_++ = 0xff; + // the CPC shuffles output lines as: + // MA12 MA11 RA2 RA1 RA0 MA9 MA8 MA7 MA6 MA5 MA4 MA3 MA2 MA1 MA0 CCLK + uint16_t address = + (uint16_t)( + ((state.refresh_address & 0x3FF) << 1) | + ((state.row_address & 7) << 11) | + ((state.refresh_address & 0x1800) << 3) + ); + + *pixel_pointer_++ = ram_[address]; // flush the current buffer if full if(pixel_pointer_ == pixel_data_ + 320) { @@ -81,6 +96,8 @@ struct CRTCBusHandler { bool was_enabled_, was_sync_; std::shared_ptr crt_; uint8_t *pixel_data_, *pixel_pointer_; + + uint8_t *ram_; }; class ConcreteMachine: @@ -89,7 +106,8 @@ class ConcreteMachine: public: ConcreteMachine() : crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses - crtc_(crtc_bus_handler_) { + crtc_(crtc_bus_handler_), + crtc_bus_handler_(ram_) { // primary clock is 4Mhz set_clock_rate(4000000); } From 3b1db148171ab8ea306fe51b2232e2377016eba1 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 07:36:03 -0400 Subject: [PATCH 29/70] Made a quick attempt at properly updating the refresh address. --- Components/6845/CRTC6845.hpp | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index 97130dd2a..b35c69d34 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -55,6 +55,11 @@ template class CRTC6845 { character_is_visible_ = false; } + // update refresh address + if(character_is_visible_) { + bus_state_.refresh_address++; + } + // check for end-of-line if(character_counter_ == registers_[0]+1) { character_counter_ = 0; From 933d69a256bf1cb787090c24f395bb2ad23520df Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 07:51:13 -0400 Subject: [PATCH 30/70] Fixed slightly: the CPC wiki has a typo. It's 12 and 13 that move up to 14 and 15. --- Machines/AmstradCPC/AmstradCPC.cpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index a99259a18..1d065b697 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -35,12 +35,12 @@ struct CRTCBusHandler { } if(pixel_pointer_) { // the CPC shuffles output lines as: - // MA12 MA11 RA2 RA1 RA0 MA9 MA8 MA7 MA6 MA5 MA4 MA3 MA2 MA1 MA0 CCLK + // MA13 MA12 RA2 RA1 RA0 MA9 MA8 MA7 MA6 MA5 MA4 MA3 MA2 MA1 MA0 CCLK uint16_t address = (uint16_t)( - ((state.refresh_address & 0x3FF) << 1) | - ((state.row_address & 7) << 11) | - ((state.refresh_address & 0x1800) << 3) + ((state.refresh_address & 0x3ff) << 1) | + ((state.row_address & 0x7) << 11) | + ((state.refresh_address & 0x3000) << 2) ); *pixel_pointer_++ = ram_[address]; From ca42abab70b61c7b3fa722296ef3ad7376edbdcf Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 07:56:44 -0400 Subject: [PATCH 31/70] Doubled up to ensure that every byte that should be inspected is represented. This makes it clearer that I'm on the right road. A garbled version of 'Amstrad 64k Microcomputer' can be discerned, in a weird grayscale and with the right-hand column missing and skewed output as a result. --- Machines/AmstradCPC/AmstradCPC.cpp | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 1d065b697..994edfd4a 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -43,11 +43,13 @@ struct CRTCBusHandler { ((state.refresh_address & 0x3000) << 2) ); - *pixel_pointer_++ = ram_[address]; + pixel_pointer_[0] = ram_[address]; + pixel_pointer_[1] = ram_[address+1]; + pixel_pointer_ += 2; // flush the current buffer if full if(pixel_pointer_ == pixel_data_ + 320) { - crt_->output_data(320, 16); + crt_->output_data(320, 8); pixel_pointer_ = pixel_data_ = nullptr; } } @@ -59,7 +61,7 @@ struct CRTCBusHandler { crt_->output_sync((unsigned int)(cycles_ * 16)); } else { if(was_enabled_) { - crt_->output_data((unsigned int)(cycles_ * 16), 16); + crt_->output_data((unsigned int)(cycles_ * 16), 8); pixel_pointer_ = pixel_data_ = nullptr; } else { uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); From 6ac71327997062a0b9a531dd63ae5318495e22c8 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 15:16:13 -0400 Subject: [PATCH 32/70] Had a quick go at properly outputting Mode 1, adding wiring to communicate palette and mode changes to the CRTC bus handler. Colours are off but it's sufficient for now. --- Machines/AmstradCPC/AmstradCPC.cpp | 127 ++++++++++++++++++++++++++--- 1 file changed, 114 insertions(+), 13 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 994edfd4a..c41758649 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -22,6 +22,7 @@ struct CRTCBusHandler { was_sync_(false), pixel_data_(nullptr), pixel_pointer_(nullptr), + was_hsync_(false), ram_(ram) {} inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { @@ -43,14 +44,38 @@ struct CRTCBusHandler { ((state.refresh_address & 0x3000) << 2) ); - pixel_pointer_[0] = ram_[address]; - pixel_pointer_[1] = ram_[address+1]; - pixel_pointer_ += 2; + switch(mode_) { + case 0: + ((uint16_t *)pixel_pointer_)[0] = mode0_output_[ram_[address]]; + ((uint16_t *)pixel_pointer_)[1] = mode0_output_[ram_[address+1]]; + pixel_pointer_ += 4; + break; + + case 1: + ((uint32_t *)pixel_pointer_)[0] = mode1_output_[ram_[address]]; + ((uint32_t *)pixel_pointer_)[1] = mode1_output_[ram_[address+1]]; + pixel_pointer_ += 8; + break; + + case 2: + ((uint64_t *)pixel_pointer_)[0] = mode2_output_[ram_[address]]; + ((uint64_t *)pixel_pointer_)[1] = mode2_output_[ram_[address+1]]; + pixel_pointer_ += 16; + break; + + case 3: + ((uint32_t *)pixel_pointer_)[0] = mode3_output_[ram_[address]]; + ((uint32_t *)pixel_pointer_)[1] = mode3_output_[ram_[address+1]]; + pixel_pointer_ += 8; + break; + + } // flush the current buffer if full if(pixel_pointer_ == pixel_data_ + 320) { - crt_->output_data(320, 8); + crt_->output_data(cycles_ * 16, pixel_divider_); pixel_pointer_ = pixel_data_ = nullptr; + cycles_ = 0; } } } @@ -58,15 +83,15 @@ struct CRTCBusHandler { // if a transition between sync/border/pixels just occurred, announce it if(state.display_enable != was_enabled_ || is_sync != was_sync_) { if(was_sync_) { - crt_->output_sync((unsigned int)(cycles_ * 16)); + crt_->output_sync(cycles_ * 16); } else { if(was_enabled_) { - crt_->output_data((unsigned int)(cycles_ * 16), 8); + crt_->output_data(cycles_ * 16, pixel_divider_); pixel_pointer_ = pixel_data_ = nullptr; } else { uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); - if(colour_pointer) *colour_pointer = 0x00; - crt_->output_level((unsigned int)(cycles_ * 16)); + if(colour_pointer) *colour_pointer = border_; + crt_->output_level(cycles_ * 16); } } @@ -74,6 +99,20 @@ struct CRTCBusHandler { was_sync_ = is_sync; was_enabled_ = state.display_enable; } + + // check for a leading hsync, now that pixels have safely been flushed + if(!was_hsync_ && state.hsync) { + if(mode_ != next_mode_) { + mode_ = next_mode_; + switch(mode_) { + default: + case 0: pixel_divider_ = 4; break; + case 1: pixel_divider_ = 2; break; + case 2: pixel_divider_ = 1; break; + } + } + } + was_hsync_ = state.hsync; } void setup_output(float aspect_ratio) { @@ -81,8 +120,10 @@ struct CRTCBusHandler { crt_->set_rgb_sampling_function( "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" "{" - "return vec3(float(texture(texID, coordinate).r) / 255.0);" + "uint sample = texture(texID, coordinate).r;" + "return vec3(float(sample & 3u), float((sample >> 2) & 3u), float((sample >> 4) & 3u)) / 3.0;" "}"); + // TODO: better vectorise the above. } void close_output() { @@ -93,13 +134,72 @@ struct CRTCBusHandler { return crt_; } + void set_next_mode(int mode) { + next_mode_ = mode; + } + + void select_pen(int pen) { + pen_ = pen; + } + + void set_colour(uint8_t colour) { + if(pen_ & 16) { + printf("border: %d -> %02x\n", colour, mapped_palette_value(colour)); + border_ = mapped_palette_value(colour); + // TODO: should flush any border currently in progress + } else { + palette_[pen_] = mapped_palette_value(colour); + + for(int c = 0; c < 16; c++) { + printf("%02x ", palette_[c]); + } + printf("\n"); + + // TODO: no need for a full regeneration, of every mode, every time + for(int c = 0; c < 256; c++) { + // prepare mode 0 +// uint8_t *pixels = (uint8_t *)&mode0_output_[c]; +// pixels[0] = palette_[((c & 0x80) >> 4) | ((c & 0x08) >> 3)]; +// pixels[1] = palette_[((c & 0x40) >> 5) | ((c & 0x04) >> 2)]; + + // prepare mode 1 + uint8_t *pixels = (uint8_t *)&mode1_output_[c]; + pixels[0] = palette_[((c & 0x80) >> 6) | ((c & 0x08) >> 3)]; + pixels[1] = palette_[((c & 0x40) >> 5) | ((c & 0x04) >> 2)]; + pixels[2] = palette_[((c & 0x20) >> 4) | ((c & 0x02) >> 1)]; + pixels[3] = palette_[((c & 0x10) >> 3) | ((c & 0x01) >> 0)]; + +// mode2_output_[c] = 0xffffff; + } + } + } + private: - int cycles_; - bool was_enabled_, was_sync_; + uint8_t mapped_palette_value(uint8_t colour) { + uint8_t r = (colour / 3) % 3; + uint8_t g = (colour / 9) % 3; + uint8_t b = colour % 3; + return (uint8_t)(r | (g << 2) | (b << 4)); + } + + unsigned int cycles_; + bool was_enabled_, was_sync_, was_hsync_; std::shared_ptr crt_; uint8_t *pixel_data_, *pixel_pointer_; uint8_t *ram_; + + int next_mode_, mode_; + + unsigned int pixel_divider_; + uint16_t mode0_output_[256]; + uint32_t mode1_output_[256]; + uint64_t mode2_output_[256]; + uint32_t mode3_output_[256]; + + int pen_; + uint8_t palette_[16]; + uint8_t border_; }; class ConcreteMachine: @@ -145,12 +245,13 @@ class ConcreteMachine: // Check for a gate array access. if((address & 0xc000) == 0x4000) { switch(*cycle.value >> 6) { - case 0: printf("Select pen %02x\n", *cycle.value & 0x1f); break; - case 1: printf("Select colour %02x\n", *cycle.value & 0x1f); break; + case 0: crtc_bus_handler_.select_pen(*cycle.value & 0x1f); break; + case 1: crtc_bus_handler_.set_colour(*cycle.value & 0x1f); break; case 2: printf("Set mode %d, other flags %02x\n", *cycle.value & 3, (*cycle.value >> 2)&7); read_pointers_[0] = (*cycle.value & 4) ? &ram_[0] : os_.data(); read_pointers_[3] = (*cycle.value & 8) ? &ram_[49152] : basic_.data(); + crtc_bus_handler_.set_next_mode(*cycle.value & 3); break; case 3: printf("RAM paging?\n"); break; } From ee27e16fb1e697405e492f9ca9334f2a6995f15d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 15:19:25 -0400 Subject: [PATCH 33/70] Switched to post-tests increment. Seems to give proper screen width, but also eliminates that 'compare to +1' step that felt unlikely. --- Components/6845/CRTC6845.hpp | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index b35c69d34..b1c0c58a8 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -41,9 +41,6 @@ template class CRTC6845 { } } - // advance horizontal counter - character_counter_++; - // check for start of horizontal sync if(character_counter_ == registers_[2]) { hsync_down_counter_ = registers_[3] & 15; @@ -61,7 +58,7 @@ template class CRTC6845 { } // check for end-of-line - if(character_counter_ == registers_[0]+1) { + if(character_counter_ == registers_[0]) { character_counter_ = 0; character_is_visible_ = true; @@ -117,6 +114,9 @@ template class CRTC6845 { bus_state_.refresh_address = line_address_; } } + } else { + // advance horizontal counter + character_counter_++; } bus_state_.display_enable = character_is_visible_ && line_is_visible_; From 1d99c116e75e3faced842a2a657e71b30af80880 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 15:29:37 -0400 Subject: [PATCH 34/70] Actually, this is probably more correct: increment and then compare, but increment the refresh address once more after the final character, to avoid repeating it. --- Components/6845/CRTC6845.hpp | 22 +++++++++++++--------- 1 file changed, 13 insertions(+), 9 deletions(-) diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index b1c0c58a8..006defa23 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -41,6 +41,12 @@ template class CRTC6845 { } } + // check for end of line + bool is_end_of_line = character_counter_ == registers_[0]; + + // increment counter + character_counter_++; + // check for start of horizontal sync if(character_counter_ == registers_[2]) { hsync_down_counter_ = registers_[3] & 15; @@ -49,16 +55,17 @@ template class CRTC6845 { // check for end of visible characters if(character_counter_ == registers_[1]) { - character_is_visible_ = false; - } - - // update refresh address - if(character_is_visible_) { bus_state_.refresh_address++; + character_is_visible_ = false; + } else { + // update refresh address + if(character_is_visible_) { + bus_state_.refresh_address++; + } } // check for end-of-line - if(character_counter_ == registers_[0]) { + if(is_end_of_line) { character_counter_ = 0; character_is_visible_ = true; @@ -114,9 +121,6 @@ template class CRTC6845 { bus_state_.refresh_address = line_address_; } } - } else { - // advance horizontal counter - character_counter_++; } bus_state_.display_enable = character_is_visible_ && line_is_visible_; From a27946102a2aab0fd627353dc3c5cb78afe2260c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 15:49:16 -0400 Subject: [PATCH 35/70] Took a shot at the interrupt counter. Attempts at keyboard reading now recur so it'll probably do for now. I think that next puts me into the realm of needing to implement the 8255. --- Machines/AmstradCPC/AmstradCPC.cpp | 109 +++++++++++++++++++++-------- 1 file changed, 78 insertions(+), 31 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index c41758649..e21245b3e 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -23,12 +23,38 @@ struct CRTCBusHandler { pixel_data_(nullptr), pixel_pointer_(nullptr), was_hsync_(false), - ram_(ram) {} + ram_(ram), + interrupt_counter_(0), + interrupt_request_(false) {} inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { + bool is_sync = state.hsync || state.vsync; + + // if a transition between sync/border/pixels just occurred, announce it + if(state.display_enable != was_enabled_ || is_sync != was_sync_) { + if(was_sync_) { + crt_->output_sync(cycles_ * 16); + } else { + if(was_enabled_) { + if(cycles_) { + crt_->output_data(cycles_ * 16, pixel_divider_); + pixel_pointer_ = pixel_data_ = nullptr; + } + } else { + uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); + if(colour_pointer) *colour_pointer = border_; + crt_->output_level(cycles_ * 16); + } + } + + cycles_ = 0; + was_sync_ = is_sync; + was_enabled_ = state.display_enable; + } + + // increment cycles since state changed cycles_++; - bool is_sync = state.hsync || state.vsync; // collect some more pixels if output is ongoing if(!is_sync && state.display_enable) { if(!pixel_data_) { @@ -80,28 +106,8 @@ struct CRTCBusHandler { } } - // if a transition between sync/border/pixels just occurred, announce it - if(state.display_enable != was_enabled_ || is_sync != was_sync_) { - if(was_sync_) { - crt_->output_sync(cycles_ * 16); - } else { - if(was_enabled_) { - crt_->output_data(cycles_ * 16, pixel_divider_); - pixel_pointer_ = pixel_data_ = nullptr; - } else { - uint8_t *colour_pointer = (uint8_t *)crt_->allocate_write_area(1); - if(colour_pointer) *colour_pointer = border_; - crt_->output_level(cycles_ * 16); - } - } - - cycles_ = 0; - was_sync_ = is_sync; - was_enabled_ = state.display_enable; - } - - // check for a leading hsync, now that pixels have safely been flushed - if(!was_hsync_ && state.hsync) { + // check for a trailing hsync + if(was_hsync_ && !state.hsync) { if(mode_ != next_mode_) { mode_ = next_mode_; switch(mode_) { @@ -111,10 +117,41 @@ struct CRTCBusHandler { case 2: pixel_divider_ = 1; break; } } + + interrupt_counter_++; + if(interrupt_counter_ == 52) { + interrupt_request_ = true; + interrupt_counter_ = false; + } + + if(interrupt_reset_counter_) { + interrupt_reset_counter_--; + if(!interrupt_reset_counter_) { + if(interrupt_counter_ < 32) { + interrupt_request_ = true; + } + interrupt_counter_ = 0; + } + } } + + if(!was_vsync_ && state.vsync) { + interrupt_reset_counter_ = 2; + } + + was_vsync_ = state.vsync; was_hsync_ = state.hsync; } + bool get_interrupt_request() { + return interrupt_request_; + } + + void reset_interrupt_request() { + interrupt_request_ = false; + interrupt_counter_ &= ~32; + } + void setup_output(float aspect_ratio) { crt_.reset(new Outputs::CRT::CRT(1024, 16, Outputs::CRT::DisplayType::PAL50, 1)); crt_->set_rgb_sampling_function( @@ -144,16 +181,16 @@ struct CRTCBusHandler { void set_colour(uint8_t colour) { if(pen_ & 16) { - printf("border: %d -> %02x\n", colour, mapped_palette_value(colour)); +// printf("border: %d -> %02x\n", colour, mapped_palette_value(colour)); border_ = mapped_palette_value(colour); // TODO: should flush any border currently in progress } else { palette_[pen_] = mapped_palette_value(colour); - for(int c = 0; c < 16; c++) { - printf("%02x ", palette_[c]); - } - printf("\n"); +// for(int c = 0; c < 16; c++) { +// printf("%02x ", palette_[c]); +// } +// printf("\n"); // TODO: no need for a full regeneration, of every mode, every time for(int c = 0; c < 256; c++) { @@ -174,6 +211,10 @@ struct CRTCBusHandler { } } + void reset_interrupt_counter() { + interrupt_counter_ = 0; + } + private: uint8_t mapped_palette_value(uint8_t colour) { uint8_t r = (colour / 3) % 3; @@ -183,7 +224,7 @@ struct CRTCBusHandler { } unsigned int cycles_; - bool was_enabled_, was_sync_, was_hsync_; + bool was_enabled_, was_sync_, was_hsync_, was_vsync_; std::shared_ptr crt_; uint8_t *pixel_data_, *pixel_pointer_; @@ -200,6 +241,10 @@ struct CRTCBusHandler { int pen_; uint8_t palette_[16]; uint8_t border_; + + int interrupt_counter_; + bool interrupt_request_; + int interrupt_reset_counter_; }; class ConcreteMachine: @@ -226,6 +271,7 @@ class ConcreteMachine: crtc_counter_ += cycle.length; int crtc_cycles = crtc_counter_.divide(HalfCycles(8)).as_int(); if(crtc_cycles) crtc_.run_for(Cycles(1)); + set_interrupt_line(crtc_bus_handler_.get_interrupt_request()); // Stop now if no action is strictly required. if(!cycle.is_terminal()) return HalfCycles(0); @@ -248,9 +294,9 @@ class ConcreteMachine: case 0: crtc_bus_handler_.select_pen(*cycle.value & 0x1f); break; case 1: crtc_bus_handler_.set_colour(*cycle.value & 0x1f); break; case 2: - printf("Set mode %d, other flags %02x\n", *cycle.value & 3, (*cycle.value >> 2)&7); read_pointers_[0] = (*cycle.value & 4) ? &ram_[0] : os_.data(); read_pointers_[3] = (*cycle.value & 8) ? &ram_[49152] : basic_.data(); + if(*cycle.value & 15) crtc_bus_handler_.reset_interrupt_counter(); crtc_bus_handler_.set_next_mode(*cycle.value & 3); break; case 3: printf("RAM paging?\n"); break; @@ -301,6 +347,7 @@ class ConcreteMachine: case CPU::Z80::PartialMachineCycle::Interrupt: *cycle.value = 0xff; + crtc_bus_handler_.reset_interrupt_request(); break; default: break; From ace71280a087f36ebc03105d128b4db0621448c5 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 16:00:17 -0400 Subject: [PATCH 36/70] Removed implementation file; this is only ever going to be a template. --- Components/6845/CRTC6845.cpp | 12 ------------ .../Mac/Clock Signal.xcodeproj/project.pbxproj | 4 ---- 2 files changed, 16 deletions(-) delete mode 100644 Components/6845/CRTC6845.cpp diff --git a/Components/6845/CRTC6845.cpp b/Components/6845/CRTC6845.cpp deleted file mode 100644 index f8290dae7..000000000 --- a/Components/6845/CRTC6845.cpp +++ /dev/null @@ -1,12 +0,0 @@ -// -// CRTC6845.cpp -// Clock Signal -// -// Created by Thomas Harte on 31/07/2017. -// Copyright © 2017 Thomas Harte. All rights reserved. -// - -#include "CRTC6845.hpp" - -using namespace Motorola; - diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 9b3f804ab..7b3f32991 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -424,7 +424,6 @@ 4BDDBA991EF3451200347E61 /* Z80MachineCycleTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4BDDBA981EF3451200347E61 /* Z80MachineCycleTests.swift */; }; 4BE77A2E1D84ADFB00BC3827 /* File.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE77A2C1D84ADFB00BC3827 /* File.cpp */; }; 4BE7C9181E3D397100A5496D /* TIA.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE7C9161E3D397100A5496D /* TIA.cpp */; }; - 4BE845211F2FF7F100A5EA22 /* CRTC6845.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE8451F1F2FF7F100A5EA22 /* CRTC6845.cpp */; }; 4BE9A6B11EDE293000CBCB47 /* zexdoc.com in Resources */ = {isa = PBXBuildFile; fileRef = 4BE9A6B01EDE293000CBCB47 /* zexdoc.com */; }; 4BEA525E1DF33323007E74F2 /* Tape.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BEA525D1DF33323007E74F2 /* Tape.cpp */; }; 4BEA52631DF339D7007E74F2 /* Speaker.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BEA52611DF339D7007E74F2 /* Speaker.cpp */; }; @@ -1002,7 +1001,6 @@ 4BE77A2D1D84ADFB00BC3827 /* File.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = File.hpp; path = ../../StaticAnalyser/Commodore/File.hpp; sourceTree = ""; }; 4BE7C9161E3D397100A5496D /* TIA.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = TIA.cpp; sourceTree = ""; }; 4BE7C9171E3D397100A5496D /* TIA.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = TIA.hpp; sourceTree = ""; }; - 4BE8451F1F2FF7F100A5EA22 /* CRTC6845.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = CRTC6845.cpp; path = 6845/CRTC6845.cpp; sourceTree = ""; }; 4BE845201F2FF7F100A5EA22 /* CRTC6845.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = CRTC6845.hpp; path = 6845/CRTC6845.hpp; sourceTree = ""; }; 4BE9A6B01EDE293000CBCB47 /* zexdoc.com */ = {isa = PBXFileReference; lastKnownFileType = file; name = zexdoc.com; path = Zexall/zexdoc.com; sourceTree = ""; }; 4BEA525D1DF33323007E74F2 /* Tape.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = Tape.cpp; path = Electron/Tape.cpp; sourceTree = ""; }; @@ -2135,7 +2133,6 @@ 4BE845221F2FF7F400A5EA22 /* 6845 */ = { isa = PBXGroup; children = ( - 4BE8451F1F2FF7F100A5EA22 /* CRTC6845.cpp */, 4BE845201F2FF7F100A5EA22 /* CRTC6845.hpp */, ); name = 6845; @@ -2745,7 +2742,6 @@ 4B2A53A01D117D36003C6002 /* CSMachine.mm in Sources */, 4B14978F1EE4B4D200CE2596 /* CSZX8081.mm in Sources */, 4BC91B831D1F160E00884B76 /* CommodoreTAP.cpp in Sources */, - 4BE845211F2FF7F100A5EA22 /* CRTC6845.cpp in Sources */, 4BCF1FAB1DADD41B0039D2E7 /* StaticAnalyser.cpp in Sources */, 4B2A539F1D117D36003C6002 /* CSAudioQueue.m in Sources */, 4B37EE821D7345A6006A09A4 /* BinaryDump.cpp in Sources */, From 58b98267fcae5de7190d3ac9e86157d4db07adfc Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 16:15:19 -0400 Subject: [PATCH 37/70] Formally transferred ownership of PIO accesses to an incoming template, and decided to start being explicit about how to specify the interfaces and provide fallbacks for optional behaviour for the new, clean generation of interfaces. A full-project sweep will inevitably occur but I'll try to tie off this branch first. --- Components/6845/CRTC6845.hpp | 5 +++ Components/8255/i8255.hpp | 43 +++++++++++++++++++ Machines/AmstradCPC/AmstradCPC.cpp | 22 +++++----- .../Clock Signal.xcodeproj/project.pbxproj | 12 ++++++ 4 files changed, 70 insertions(+), 12 deletions(-) create mode 100644 Components/8255/i8255.hpp diff --git a/Components/6845/CRTC6845.hpp b/Components/6845/CRTC6845.hpp index 006defa23..13d1a2ffa 100644 --- a/Components/6845/CRTC6845.hpp +++ b/Components/6845/CRTC6845.hpp @@ -26,6 +26,11 @@ struct BusState { uint16_t row_address; }; +class BusHandler { + public: + void perform_bus_cycle(const BusState &) {} +}; + template class CRTC6845 { public: CRTC6845(T &bus_handler) : bus_handler_(bus_handler) {} diff --git a/Components/8255/i8255.hpp b/Components/8255/i8255.hpp new file mode 100644 index 000000000..2aa73f635 --- /dev/null +++ b/Components/8255/i8255.hpp @@ -0,0 +1,43 @@ +// +// i8255.hpp +// Clock Signal +// +// Created by Thomas Harte on 01/08/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#ifndef i8255_hpp +#define i8255_hpp + +namespace Intel { +namespace i8255 { + +class PortHandler { +}; + +template class i8255 { + public: + void set_register(int address, uint8_t value) { + switch((address >> 8) & 3) { + case 0: printf("PSG data: %d\n", value); break; + case 1: printf("Vsync, etc: %02x\n", value); break; + case 2: printf("Key row, etc: %02x\n", value); break; + case 3: printf("PIO control: %02x\n", value); break; + } + } + + uint8_t get_register(int address) { + switch((address >> 8) & 3) { + case 0: printf("[In] PSG data\n"); break; + case 1: printf("[In] Vsync, etc\n"); break; + case 2: printf("[In] Key row, etc\n"); break; + case 3: printf("[In] PIO control\n"); break; + } + return 0xff; + } +}; + +} +} + +#endif /* i8255_hpp */ diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index e21245b3e..598171786 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -9,6 +9,8 @@ #include "AmstradCPC.hpp" #include "../../Processors/Z80/Z80.hpp" + +#include "../../Components/8255/i8255.hpp" #include "../../Components/AY38910/AY38910.hpp" #include "../../Components/6845/CRTC6845.hpp" @@ -247,6 +249,9 @@ struct CRTCBusHandler { int interrupt_reset_counter_; }; +struct i8255PortHandler : public Intel::i8255::PortHandler { +}; + class ConcreteMachine: public CPU::Z80::Processor, public Machine { @@ -314,12 +319,7 @@ class ConcreteMachine: // Check for a PIO access if(!(address & 0x800)) { - switch((address >> 8) & 3) { - case 0: printf("PSG data: %d\n", *cycle.value); break; - case 1: printf("Vsync, etc: %02x\n", *cycle.value); break; - case 2: printf("Key row, etc: %02x\n", *cycle.value); break; - case 3: printf("PIO control: %02x\n", *cycle.value); break; - } + i8255_.set_register((address >> 8) & 3, *cycle.value); } break; case CPU::Z80::PartialMachineCycle::Input: @@ -334,12 +334,7 @@ class ConcreteMachine: // Check for a PIO access if(!(address & 0x800)) { - switch((address >> 8) & 3) { - case 0: printf("[In] PSG data\n"); break; - case 1: printf("[In] Vsync, etc\n"); break; - case 2: printf("[In] Key row, etc\n"); break; - case 3: printf("[In] PIO control\n"); break; - } + *cycle.value = i8255_.get_register((address >> 8) & 3); } *cycle.value = 0xff; @@ -404,6 +399,9 @@ class ConcreteMachine: CRTCBusHandler crtc_bus_handler_; Motorola::CRTC::CRTC6845 crtc_; + i8255PortHandler i8255_port_handler_; + Intel::i8255::i8255 i8255_; + HalfCycles clock_offset_; HalfCycles crtc_counter_; diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 7b3f32991..ff15f9843 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -421,6 +421,7 @@ 4BD4A8D01E077FD20020D856 /* PCMTrackTests.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4BD4A8CF1E077FD20020D856 /* PCMTrackTests.mm */; }; 4BD5F1951D13528900631CD1 /* CSBestEffortUpdater.m in Sources */ = {isa = PBXBuildFile; fileRef = 4BD5F1941D13528900631CD1 /* CSBestEffortUpdater.m */; }; 4BD69F941D98760000243FE1 /* AcornADF.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BD69F921D98760000243FE1 /* AcornADF.cpp */; }; + 4BD9137B1F3115A8009BCF85 /* i8255.hpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BD913791F3115A8009BCF85 /* i8255.hpp */; }; 4BDDBA991EF3451200347E61 /* Z80MachineCycleTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4BDDBA981EF3451200347E61 /* Z80MachineCycleTests.swift */; }; 4BE77A2E1D84ADFB00BC3827 /* File.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE77A2C1D84ADFB00BC3827 /* File.cpp */; }; 4BE7C9181E3D397100A5496D /* TIA.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE7C9161E3D397100A5496D /* TIA.cpp */; }; @@ -996,6 +997,7 @@ 4BD5F1941D13528900631CD1 /* CSBestEffortUpdater.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = CSBestEffortUpdater.m; path = Updater/CSBestEffortUpdater.m; sourceTree = ""; }; 4BD69F921D98760000243FE1 /* AcornADF.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = AcornADF.cpp; sourceTree = ""; }; 4BD69F931D98760000243FE1 /* AcornADF.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = AcornADF.hpp; sourceTree = ""; }; + 4BD913791F3115A8009BCF85 /* i8255.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = i8255.hpp; path = 8255/i8255.hpp; sourceTree = ""; }; 4BDDBA981EF3451200347E61 /* Z80MachineCycleTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Z80MachineCycleTests.swift; sourceTree = ""; }; 4BE77A2C1D84ADFB00BC3827 /* File.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = File.cpp; path = ../../StaticAnalyser/Commodore/File.cpp; sourceTree = ""; }; 4BE77A2D1D84ADFB00BC3827 /* File.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = File.hpp; path = ../../StaticAnalyser/Commodore/File.hpp; sourceTree = ""; }; @@ -2033,6 +2035,7 @@ 4BC9DF4C1D04691600F44158 /* 6560 */, 4B4A762D1DB1A35C007AAE2E /* AY38910 */, 4BE845221F2FF7F400A5EA22 /* 6845 */, + 4BD9137C1F3115AC009BCF85 /* 8255 */, ); name = Components; path = ../../Components; @@ -2121,6 +2124,14 @@ name = Updater; sourceTree = ""; }; + 4BD9137C1F3115AC009BCF85 /* 8255 */ = { + isa = PBXGroup; + children = ( + 4BD913791F3115A8009BCF85 /* i8255.hpp */, + ); + name = 8255; + sourceTree = ""; + }; 4BE5F85A1C3E1C2500C43F01 /* Resources */ = { isa = PBXGroup; children = ( @@ -2673,6 +2684,7 @@ 4B2A332A1DB8544D002876E3 /* MemoryFuzzer.cpp in Sources */, 4B55CE5F1C3B7D960093A61B /* MachineDocument.swift in Sources */, 4B2A332F1DB86869002876E3 /* OricOptionsPanel.swift in Sources */, + 4BD9137B1F3115A8009BCF85 /* i8255.hpp in Sources */, 4B7913CC1DFCD80E00175A82 /* Video.cpp in Sources */, 4B2A53A11D117D36003C6002 /* CSAtari2600.mm in Sources */, 4BF829661D8F732B001BAE39 /* Disk.cpp in Sources */, From 08ad35efd9dab72a1a871269c01d8870f9e15fda Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 16:34:13 -0400 Subject: [PATCH 38/70] It's barely an implementation of the 8255, but ensured that data is bounced into the PortHandler, conveniently assuming the interaction mode used by the CPC. --- Components/8255/i8255.hpp | 49 +++++++++++++++---- Machines/AmstradCPC/AmstradCPC.cpp | 21 +++++++- .../Clock Signal.xcodeproj/project.pbxproj | 6 +-- 3 files changed, 60 insertions(+), 16 deletions(-) diff --git a/Components/8255/i8255.hpp b/Components/8255/i8255.hpp index 2aa73f635..96e4e0a0a 100644 --- a/Components/8255/i8255.hpp +++ b/Components/8255/i8255.hpp @@ -13,28 +13,57 @@ namespace Intel { namespace i8255 { class PortHandler { + public: + void set_value(int port, uint8_t value) {} + uint8_t get_value(int port) { return 0xff; } }; +// TODO: most of the implementation below. Right now it just blindly passes data in all directions, +// ignoring operation mode. But at least it establishes proper ownership and hand-off of decision making. template class i8255 { public: + i8255() : control_(0), outputs_{0, 0, 0} {} + void set_register(int address, uint8_t value) { - switch((address >> 8) & 3) { - case 0: printf("PSG data: %d\n", value); break; - case 1: printf("Vsync, etc: %02x\n", value); break; - case 2: printf("Key row, etc: %02x\n", value); break; - case 3: printf("PIO control: %02x\n", value); break; + switch(address & 3) { + case 0: outputs_[0] = value; port_handler_.set_value(0, value); break; + case 1: outputs_[1] = value; port_handler_.set_value(1, value); break; + case 2: outputs_[2] = value; port_handler_.set_value(2, value); break; + case 3: + if(value & 0x80) { + control_ = value; + } else { + if(value & 1) { + outputs_[2] |= 1 << ((value >> 1)&7); + } else { + outputs_[2] &= ~(1 << ((value >> 1)&7)); + } + } + update_outputs(); + break; } } uint8_t get_register(int address) { - switch((address >> 8) & 3) { - case 0: printf("[In] PSG data\n"); break; - case 1: printf("[In] Vsync, etc\n"); break; - case 2: printf("[In] Key row, etc\n"); break; - case 3: printf("[In] PIO control\n"); break; + switch(address & 3) { + case 0: return port_handler_.get_value(0); + case 1: return port_handler_.get_value(1); + case 2: return port_handler_.get_value(2); + case 3: return control_; } return 0xff; } + + private: + void update_outputs() { + port_handler_.set_value(0, outputs_[0]); + port_handler_.set_value(1, outputs_[1]); + port_handler_.set_value(2, outputs_[2]); + } + + uint8_t control_; + uint8_t outputs_[3]; + T port_handler_; }; } diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 598171786..249653cca 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -16,7 +16,7 @@ using namespace AmstradCPC; -struct CRTCBusHandler { +class CRTCBusHandler { public: CRTCBusHandler(uint8_t *ram) : cycles_(0), @@ -249,7 +249,24 @@ struct CRTCBusHandler { int interrupt_reset_counter_; }; -struct i8255PortHandler : public Intel::i8255::PortHandler { +class i8255PortHandler : public Intel::i8255::PortHandler { + public: + void set_value(int port, uint8_t value) { + switch(port) { + case 0: printf("PSG data: %d\n", value); break; + case 1: printf("Vsync, etc: %02x\n", value); break; + case 2: printf("Key row, etc: %02x\n", value); break; + } + } + + uint8_t get_value(int port) { + switch(port) { + case 0: printf("PSG data\n"); break; + case 1: printf("[In] Vsync, etc\n"); break; + case 2: printf("[In] Key row, etc\n"); break; + } + return 0xff; + } }; class ConcreteMachine: diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index ff15f9843..31873f6f7 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -421,7 +421,6 @@ 4BD4A8D01E077FD20020D856 /* PCMTrackTests.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4BD4A8CF1E077FD20020D856 /* PCMTrackTests.mm */; }; 4BD5F1951D13528900631CD1 /* CSBestEffortUpdater.m in Sources */ = {isa = PBXBuildFile; fileRef = 4BD5F1941D13528900631CD1 /* CSBestEffortUpdater.m */; }; 4BD69F941D98760000243FE1 /* AcornADF.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BD69F921D98760000243FE1 /* AcornADF.cpp */; }; - 4BD9137B1F3115A8009BCF85 /* i8255.hpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BD913791F3115A8009BCF85 /* i8255.hpp */; }; 4BDDBA991EF3451200347E61 /* Z80MachineCycleTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4BDDBA981EF3451200347E61 /* Z80MachineCycleTests.swift */; }; 4BE77A2E1D84ADFB00BC3827 /* File.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE77A2C1D84ADFB00BC3827 /* File.cpp */; }; 4BE7C9181E3D397100A5496D /* TIA.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4BE7C9161E3D397100A5496D /* TIA.cpp */; }; @@ -997,7 +996,7 @@ 4BD5F1941D13528900631CD1 /* CSBestEffortUpdater.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = CSBestEffortUpdater.m; path = Updater/CSBestEffortUpdater.m; sourceTree = ""; }; 4BD69F921D98760000243FE1 /* AcornADF.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = AcornADF.cpp; sourceTree = ""; }; 4BD69F931D98760000243FE1 /* AcornADF.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = AcornADF.hpp; sourceTree = ""; }; - 4BD913791F3115A8009BCF85 /* i8255.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = i8255.hpp; path = 8255/i8255.hpp; sourceTree = ""; }; + 4BD9137D1F311BC5009BCF85 /* i8255.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = i8255.hpp; path = 8255/i8255.hpp; sourceTree = ""; }; 4BDDBA981EF3451200347E61 /* Z80MachineCycleTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Z80MachineCycleTests.swift; sourceTree = ""; }; 4BE77A2C1D84ADFB00BC3827 /* File.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = File.cpp; path = ../../StaticAnalyser/Commodore/File.cpp; sourceTree = ""; }; 4BE77A2D1D84ADFB00BC3827 /* File.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = File.hpp; path = ../../StaticAnalyser/Commodore/File.hpp; sourceTree = ""; }; @@ -2127,7 +2126,7 @@ 4BD9137C1F3115AC009BCF85 /* 8255 */ = { isa = PBXGroup; children = ( - 4BD913791F3115A8009BCF85 /* i8255.hpp */, + 4BD9137D1F311BC5009BCF85 /* i8255.hpp */, ); name = 8255; sourceTree = ""; @@ -2684,7 +2683,6 @@ 4B2A332A1DB8544D002876E3 /* MemoryFuzzer.cpp in Sources */, 4B55CE5F1C3B7D960093A61B /* MachineDocument.swift in Sources */, 4B2A332F1DB86869002876E3 /* OricOptionsPanel.swift in Sources */, - 4BD9137B1F3115A8009BCF85 /* i8255.hpp in Sources */, 4B7913CC1DFCD80E00175A82 /* Video.cpp in Sources */, 4B2A53A11D117D36003C6002 /* CSAtari2600.mm in Sources */, 4BF829661D8F732B001BAE39 /* Disk.cpp in Sources */, From 64da8e17d1f9114f1ba6a104ae993594e1e58c39 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 17:01:20 -0400 Subject: [PATCH 39/70] Fixed: of course this should take a reference to an existing port handler rather than hatching its own; otherwise additional communication with a port handler by an i8255 owner doesn't work as intended. --- Components/8255/i8255.hpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Components/8255/i8255.hpp b/Components/8255/i8255.hpp index 96e4e0a0a..e188d669c 100644 --- a/Components/8255/i8255.hpp +++ b/Components/8255/i8255.hpp @@ -22,7 +22,7 @@ class PortHandler { // ignoring operation mode. But at least it establishes proper ownership and hand-off of decision making. template class i8255 { public: - i8255() : control_(0), outputs_{0, 0, 0} {} + i8255(T &port_handler) : control_(0), outputs_{0, 0, 0}, port_handler_(port_handler) {} void set_register(int address, uint8_t value) { switch(address & 3) { @@ -63,7 +63,7 @@ template class i8255 { uint8_t control_; uint8_t outputs_[3]; - T port_handler_; + T &port_handler_; }; } From 2d4e202be3317b20092fc87d7b915967b9aa7202 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 17:01:36 -0400 Subject: [PATCH 40/70] Completed dangling comment. --- Components/AY38910/AY38910.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Components/AY38910/AY38910.hpp b/Components/AY38910/AY38910.hpp index 4b35e395a..34aefe1da 100644 --- a/Components/AY38910/AY38910.hpp +++ b/Components/AY38910/AY38910.hpp @@ -38,7 +38,7 @@ class AY38910: public ::Outputs::Filter { /// Gets the value that would appear on the data lines if only the AY is outputting. uint8_t get_data_output(); - /// Sets the + /// Sets the current control line state, as a bit field. void set_control_lines(ControlLines control_lines); /*! From cb732e5d5f7c54d7b0375315b76ad0a44b4fe85c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 17:01:58 -0400 Subject: [PATCH 41/70] Made an attempt to wire in an [unclocked] AY, in an endeavour to get to keyboard reading. --- Machines/AmstradCPC/AmstradCPC.cpp | 40 +++++++++++++++++++++++++----- 1 file changed, 34 insertions(+), 6 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 249653cca..b6da9b5b8 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -253,20 +253,40 @@ class i8255PortHandler : public Intel::i8255::PortHandler { public: void set_value(int port, uint8_t value) { switch(port) { - case 0: printf("PSG data: %d\n", value); break; + case 0: + ay_->set_data_input(value); + break; case 1: printf("Vsync, etc: %02x\n", value); break; - case 2: printf("Key row, etc: %02x\n", value); break; + case 2: + // TODO: set key row: (value & 15) + // TODO: set casette motor control: ((value >> 4) & 1) + // TODO: set casette output: ((value >> 5) & 1) + ay_->set_control_lines( + (GI::AY38910::ControlLines)( + ((value & 0x80) ? GI::AY38910::BCDIR : 0) | + ((value & 0x40) ? GI::AY38910::BC1 : 0) + )); + break; } } uint8_t get_value(int port) { switch(port) { - case 0: printf("PSG data\n"); break; + case 0: + return ay_->get_data_output(); + break; case 1: printf("[In] Vsync, etc\n"); break; case 2: printf("[In] Key row, etc\n"); break; } return 0xff; } + + void set_ay(std::shared_ptr ay) { + ay_ = ay; + } + + private: + std::shared_ptr ay_; }; class ConcreteMachine: @@ -276,7 +296,8 @@ class ConcreteMachine: ConcreteMachine() : crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses crtc_(crtc_bus_handler_), - crtc_bus_handler_(ram_) { + crtc_bus_handler_(ram_), + i8255_(i8255_port_handler_) { // primary clock is 4Mhz set_clock_rate(4000000); } @@ -368,14 +389,19 @@ class ConcreteMachine: return HalfCycles(0); } - void flush() {} + void flush() { +// i8255_port_handler_.flush(); + } void setup_output(float aspect_ratio) { crtc_bus_handler_.setup_output(aspect_ratio); + ay_.reset(new GI::AY38910); + i8255_port_handler_.set_ay(ay_); } void close_output() { crtc_bus_handler_.close_output(); + ay_.reset(); } std::shared_ptr get_crt() { @@ -383,7 +409,7 @@ class ConcreteMachine: } std::shared_ptr get_speaker() { - return nullptr; + return ay_; } void run_for(const Cycles cycles) { @@ -416,11 +442,13 @@ class ConcreteMachine: CRTCBusHandler crtc_bus_handler_; Motorola::CRTC::CRTC6845 crtc_; + std::shared_ptr ay_; i8255PortHandler i8255_port_handler_; Intel::i8255::i8255 i8255_; HalfCycles clock_offset_; HalfCycles crtc_counter_; + HalfCycles half_cycles_since_ay_update_; uint8_t ram_[65536]; std::vector os_, basic_; From 3b292273c7b9caf8580c5998d4a10e3c4189c146 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 17:05:11 -0400 Subject: [PATCH 42/70] Fixed: BC2 is always implicitly set. The machine is now periodically checking the AY's register 14 (i.e. the first input port), so probably there's enough here now to implement keyboard input. --- Machines/AmstradCPC/AmstradCPC.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index b6da9b5b8..d59bd610b 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -264,7 +264,8 @@ class i8255PortHandler : public Intel::i8255::PortHandler { ay_->set_control_lines( (GI::AY38910::ControlLines)( ((value & 0x80) ? GI::AY38910::BCDIR : 0) | - ((value & 0x40) ? GI::AY38910::BC1 : 0) + ((value & 0x40) ? GI::AY38910::BC1 : 0) | + GI::AY38910::BC2 )); break; } From e6854ff8dbec5284ee4f0039f5cf1072710387cf Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 17:06:57 -0400 Subject: [PATCH 43/70] Corrected typo: the input to an AY is BDIR, not BCDIR. --- Components/AY38910/AY38910.cpp | 6 +++--- Components/AY38910/AY38910.hpp | 2 +- Machines/AmstradCPC/AmstradCPC.cpp | 2 +- Machines/Oric/Oric.cpp | 2 +- 4 files changed, 6 insertions(+), 6 deletions(-) diff --git a/Components/AY38910/AY38910.cpp b/Components/AY38910/AY38910.cpp index 22769b805..c05dad2e6 100644 --- a/Components/AY38910/AY38910.cpp +++ b/Components/AY38910/AY38910.cpp @@ -244,12 +244,12 @@ void AY38910::set_control_lines(ControlLines control_lines) { switch((int)control_lines) { default: new_state = Inactive; break; - case (int)(BCDIR | BC2 | BC1): - case BCDIR: + case (int)(BDIR | BC2 | BC1): + case BDIR: case BC1: new_state = LatchAddress; break; case (int)(BC2 | BC1): new_state = Read; break; - case (int)(BCDIR | BC2): new_state = Write; break; + case (int)(BDIR | BC2): new_state = Write; break; } if(new_state != control_state_) { diff --git a/Components/AY38910/AY38910.hpp b/Components/AY38910/AY38910.hpp index 34aefe1da..6a6d9d2f9 100644 --- a/Components/AY38910/AY38910.hpp +++ b/Components/AY38910/AY38910.hpp @@ -29,7 +29,7 @@ class AY38910: public ::Outputs::Filter { enum ControlLines { BC1 = (1 << 0), BC2 = (1 << 1), - BCDIR = (1 << 2) + BDIR = (1 << 2) }; /// Sets the value the AY would read from its data lines if it were not outputting. diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index d59bd610b..7724b524e 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -263,7 +263,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { // TODO: set casette output: ((value >> 5) & 1) ay_->set_control_lines( (GI::AY38910::ControlLines)( - ((value & 0x80) ? GI::AY38910::BCDIR : 0) | + ((value & 0x80) ? GI::AY38910::BDIR : 0) | ((value & 0x40) ? GI::AY38910::BC1 : 0) | GI::AY38910::BC2 )); diff --git a/Machines/Oric/Oric.cpp b/Machines/Oric/Oric.cpp index 10e4f6b7e..74160be0d 100644 --- a/Machines/Oric/Oric.cpp +++ b/Machines/Oric/Oric.cpp @@ -244,7 +244,7 @@ void Machine::VIA::run_for(const Cycles cycles) { void Machine::VIA::update_ay() { ay8910->run_for(cycles_since_ay_update_.flush()); - ay8910->set_control_lines( (GI::AY38910::ControlLines)((ay_bdir_ ? GI::AY38910::BCDIR : 0) | (ay_bc1_ ? GI::AY38910::BC1 : 0) | GI::AY38910::BC2)); + ay8910->set_control_lines( (GI::AY38910::ControlLines)((ay_bdir_ ? GI::AY38910::BDIR : 0) | (ay_bc1_ ? GI::AY38910::BC1 : 0) | GI::AY38910::BC2)); } #pragma mark - TapePlayer From f5b278d6832859910b2ad455054942c30995005d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 17:31:56 -0400 Subject: [PATCH 44/70] Added enough stuff to put the emulated Amstrad CPC in a state of knowing whether its '0' key is pressed. --- Machines/AmstradCPC/AmstradCPC.cpp | 6 ++ Machines/AmstradCPC/AmstradCPC.hpp | 28 ++++++ .../Machine/Wrappers/CSAmstradCPC.h | 3 +- .../Machine/Wrappers/CSAmstradCPC.mm | 88 +++++++++++++++++++ 4 files changed, 124 insertions(+), 1 deletion(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 7724b524e..4e8e29c19 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -439,6 +439,12 @@ class ConcreteMachine: } } + void set_key_state(uint16_t key, bool isPressed) { + } + + void clear_all_keys() { + } + private: CRTCBusHandler crtc_bus_handler_; Motorola::CRTC::CRTC6845 crtc_; diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index a8c43293b..eef144fad 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -20,12 +20,40 @@ enum ROMType: uint8_t { AMSDOS }; +enum Key: uint16_t { +#define Line(l, k1, k2, k3, k4, k5, k6, k7, k8) \ + k1 = (l << 4) | 0x08,\ + k2 = (l << 4) | 0x07,\ + k3 = (l << 4) | 0x06,\ + k4 = (l << 4) | 0x05,\ + k5 = (l << 4) | 0x04,\ + k6 = (l << 4) | 0x03,\ + k7 = (l << 4) | 0x02,\ + k8 = (l << 4) | 0x01, + + Line(0, KeyFDot, KeyEnter, KeyF3, KeyF6, KeyF9, KeyDown, KeyRight, KeyUp) + Line(1, KeyF0, KeyF2, KeyF1, KeyF5, KeyF8, KeyF7, KeyCopy, KeyLeft) + Line(2, KeyControl, KeyBackSlash, KeyShift, KeyF4, KeyRightSquareBracket, KeyReturn, KeyLeftSquareBracket, KeyClear) + Line(3, KeyFullStop, KeyForwardSlash, KeyColon, KeySemicolon, KeyP, KeyAt, KeyMinus, KeyCaret) + Line(4, KeyComma, KeyM, KeyK, KeyL, KeyI, KeyO, Key9, Key0) + Line(5, KeySpace, KeyN, KeyJ, KeyH, KeyY, KeyU, Key7, Key8) + Line(6, KeyV, KeyB, KeyF, KeyG, KeyT, KeyR, Key5, Key6) + Line(7, KeyX, KeyC, KeyD, KeyS, KeyW, KeyE, Key3, Key4) + Line(8, KeyZ, KeyCapsLock, KeyA, KeyTab, KeyQ, KeyEscape, Key2, Key1) + Line(9, KeyDelete, KeyJoy1Fire3, KeyJoy2Fire2, KeyJoy1Fire1, KeyJoy1Right, KeyJoy1Left, KeyJoy1Down, KeyJoy1Up) + +#undef Line +}; + class Machine: public CRTMachine::Machine, public ConfigurationTarget::Machine { public: static Machine *AmstradCPC(); + virtual void set_rom(ROMType type, std::vector data) = 0; + virtual void set_key_state(uint16_t key, bool isPressed) = 0; + virtual void clear_all_keys() = 0; }; } diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h index 0457ffdf9..1286a6046 100644 --- a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.h @@ -7,7 +7,8 @@ // #import "CSMachine.h" +#import "CSKeyboardMachine.h" -@interface CSAmstradCPC : CSMachine +@interface CSAmstradCPC : CSMachine @end diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm index f5c612fed..20ca21194 100644 --- a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm @@ -59,4 +59,92 @@ - (NSString *)userDefaultsPrefix { return @"amstradCPC"; } +#pragma mark - Keyboard Mapping + +- (void)clearAllKeys { + @synchronized(self) { + _amstradCPC->clear_all_keys(); + } +} + +- (void)setKey:(uint16_t)key isPressed:(BOOL)isPressed { + @synchronized(self) { + switch(key) + { + case VK_ANSI_0: _amstradCPC->set_key_state(AmstradCPC::Key::Key0, isPressed); break; +/* case VK_ANSI_1: _electron.set_key_state(Electron::Key::Key1, isPressed); break; + case VK_ANSI_2: _electron.set_key_state(Electron::Key::Key2, isPressed); break; + case VK_ANSI_3: _electron.set_key_state(Electron::Key::Key3, isPressed); break; + case VK_ANSI_4: _electron.set_key_state(Electron::Key::Key4, isPressed); break; + case VK_ANSI_5: _electron.set_key_state(Electron::Key::Key5, isPressed); break; + case VK_ANSI_6: _electron.set_key_state(Electron::Key::Key6, isPressed); break; + case VK_ANSI_7: _electron.set_key_state(Electron::Key::Key7, isPressed); break; + case VK_ANSI_8: _electron.set_key_state(Electron::Key::Key8, isPressed); break; + case VK_ANSI_9: _electron.set_key_state(Electron::Key::Key9, isPressed); break; + + case VK_ANSI_Q: _electron.set_key_state(Electron::Key::KeyQ, isPressed); break; + case VK_ANSI_W: _electron.set_key_state(Electron::Key::KeyW, isPressed); break; + case VK_ANSI_E: _electron.set_key_state(Electron::Key::KeyE, isPressed); break; + case VK_ANSI_R: _electron.set_key_state(Electron::Key::KeyR, isPressed); break; + case VK_ANSI_T: _electron.set_key_state(Electron::Key::KeyT, isPressed); break; + case VK_ANSI_Y: _electron.set_key_state(Electron::Key::KeyY, isPressed); break; + case VK_ANSI_U: _electron.set_key_state(Electron::Key::KeyU, isPressed); break; + case VK_ANSI_I: _electron.set_key_state(Electron::Key::KeyI, isPressed); break; + case VK_ANSI_O: _electron.set_key_state(Electron::Key::KeyO, isPressed); break; + case VK_ANSI_P: _electron.set_key_state(Electron::Key::KeyP, isPressed); break; + case VK_ANSI_A: _electron.set_key_state(Electron::Key::KeyA, isPressed); break; + case VK_ANSI_S: _electron.set_key_state(Electron::Key::KeyS, isPressed); break; + case VK_ANSI_D: _electron.set_key_state(Electron::Key::KeyD, isPressed); break; + case VK_ANSI_F: _electron.set_key_state(Electron::Key::KeyF, isPressed); break; + case VK_ANSI_G: _electron.set_key_state(Electron::Key::KeyG, isPressed); break; + case VK_ANSI_H: _electron.set_key_state(Electron::Key::KeyH, isPressed); break; + case VK_ANSI_J: _electron.set_key_state(Electron::Key::KeyJ, isPressed); break; + case VK_ANSI_K: _electron.set_key_state(Electron::Key::KeyK, isPressed); break; + case VK_ANSI_L: _electron.set_key_state(Electron::Key::KeyL, isPressed); break; + case VK_ANSI_Z: _electron.set_key_state(Electron::Key::KeyZ, isPressed); break; + case VK_ANSI_X: _electron.set_key_state(Electron::Key::KeyX, isPressed); break; + case VK_ANSI_C: _electron.set_key_state(Electron::Key::KeyC, isPressed); break; + case VK_ANSI_V: _electron.set_key_state(Electron::Key::KeyV, isPressed); break; + case VK_ANSI_B: _electron.set_key_state(Electron::Key::KeyB, isPressed); break; + case VK_ANSI_N: _electron.set_key_state(Electron::Key::KeyN, isPressed); break; + case VK_ANSI_M: _electron.set_key_state(Electron::Key::KeyM, isPressed); break; + + case VK_Space: _electron.set_key_state(Electron::Key::KeySpace, isPressed); break; + case VK_ANSI_Grave: + case VK_ANSI_Backslash: + _electron.set_key_state(Electron::Key::KeyCopy, isPressed); break; + case VK_Return: _electron.set_key_state(Electron::Key::KeyReturn, isPressed); break; + case VK_ANSI_Minus: _electron.set_key_state(Electron::Key::KeyMinus, isPressed); break; + + case VK_RightArrow: _electron.set_key_state(Electron::Key::KeyRight, isPressed); break; + case VK_LeftArrow: _electron.set_key_state(Electron::Key::KeyLeft, isPressed); break; + case VK_DownArrow: _electron.set_key_state(Electron::Key::KeyDown, isPressed); break; + case VK_UpArrow: _electron.set_key_state(Electron::Key::KeyUp, isPressed); break; + + case VK_Delete: _electron.set_key_state(Electron::Key::KeyDelete, isPressed); break; + case VK_Escape: _electron.set_key_state(Electron::Key::KeyEscape, isPressed); break; + + case VK_ANSI_Comma: _electron.set_key_state(Electron::Key::KeyComma, isPressed); break; + case VK_ANSI_Period: _electron.set_key_state(Electron::Key::KeyFullStop, isPressed); break; + + case VK_ANSI_Semicolon: + _electron.set_key_state(Electron::Key::KeySemiColon, isPressed); break; + case VK_ANSI_Quote: _electron.set_key_state(Electron::Key::KeyColon, isPressed); break; + + case VK_ANSI_Slash: _electron.set_key_state(Electron::Key::KeySlash, isPressed); break; + + case VK_Shift: _electron.set_key_state(Electron::Key::KeyShift, isPressed); break; + case VK_Control: _electron.set_key_state(Electron::Key::KeyControl, isPressed); break; + case VK_Command: + case VK_Option: _electron.set_key_state(Electron::Key::KeyFunc, isPressed); break; + + case VK_F12: _electron.set_key_state(Electron::Key::KeyBreak, isPressed); break; + + default: +// printf("%02x\n", key); + break;*/ + } + } +} + @end From 2e4577f7418fa13ea83b6d8fd13ff0620428f840 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 17:52:05 -0400 Subject: [PATCH 45/70] Made a game attempt at implementing a (sticky) keyboard. No effect yet. --- Machines/AmstradCPC/AmstradCPC.cpp | 29 +++++++++++++++++++++++++---- 1 file changed, 25 insertions(+), 4 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 4e8e29c19..a21d0be07 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -249,16 +249,30 @@ class CRTCBusHandler { int interrupt_reset_counter_; }; +struct KeyboardState { + KeyboardState() : rows{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff} {} + uint8_t rows[10]; +}; + class i8255PortHandler : public Intel::i8255::PortHandler { public: + i8255PortHandler(const KeyboardState &key_state) : key_state_(key_state) {} + void set_value(int port, uint8_t value) { switch(port) { case 0: ay_->set_data_input(value); break; case 1: printf("Vsync, etc: %02x\n", value); break; - case 2: - // TODO: set key row: (value & 15) + case 2: { + // TODO: the AY really should allow port communications to be active. Work needed. +// printf("%d\n", value & 15); + int key_row = value & 15; + if(key_row < 10) { + ay_->set_port_input(false, key_state_.rows[key_row]); + } else { + ay_->set_port_input(false, 0xff); + } // TODO: set casette motor control: ((value >> 4) & 1) // TODO: set casette output: ((value >> 5) & 1) ay_->set_control_lines( @@ -267,7 +281,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { ((value & 0x40) ? GI::AY38910::BC1 : 0) | GI::AY38910::BC2 )); - break; + } break; } } @@ -288,6 +302,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { private: std::shared_ptr ay_; + const KeyboardState &key_state_; }; class ConcreteMachine: @@ -298,7 +313,8 @@ class ConcreteMachine: crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses crtc_(crtc_bus_handler_), crtc_bus_handler_(ram_), - i8255_(i8255_port_handler_) { + i8255_(i8255_port_handler_), + i8255_port_handler_(key_state_) { // primary clock is 4Mhz set_clock_rate(4000000); } @@ -440,9 +456,12 @@ class ConcreteMachine: } void set_key_state(uint16_t key, bool isPressed) { +// if(isPressed) key_state_.rows[key >> 4] &= ~(key&7); else key_state_.rows[key >> 4] |= (key&7); +key_state_.rows[key >> 4] &= ~(key&7); } void clear_all_keys() { + memset(key_state_.rows, 0xff, 10); } private: @@ -462,6 +481,8 @@ class ConcreteMachine: uint8_t *read_pointers_[4]; uint8_t *write_pointers_[4]; + + KeyboardState key_state_; }; Machine *Machine::AmstradCPC() { From 0267bc237fe72239cdda975c056c5b49c9c4bbcb Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 18:04:51 -0400 Subject: [PATCH 46/70] Added the ability to set a port input, and relaxed bus state testing. I think my on-demand bus reactions here are inappropriate, so more work to do here probably. --- Components/AY38910/AY38910.cpp | 8 ++++++-- Components/AY38910/AY38910.hpp | 6 ++++++ 2 files changed, 12 insertions(+), 2 deletions(-) diff --git a/Components/AY38910/AY38910.cpp b/Components/AY38910/AY38910.cpp index c05dad2e6..d3d7329c4 100644 --- a/Components/AY38910/AY38910.cpp +++ b/Components/AY38910/AY38910.cpp @@ -231,6 +231,10 @@ uint8_t AY38910::get_port_output(bool port_b) { return registers_[port_b ? 15 : 14]; } +void AY38910::set_port_input(bool port_b, uint8_t value) { + registers_[port_b ? 15 : 14] = value; +} + void AY38910::set_data_input(uint8_t r) { data_input_ = r; } @@ -252,7 +256,7 @@ void AY38910::set_control_lines(ControlLines control_lines) { case (int)(BDIR | BC2): new_state = Write; break; } - if(new_state != control_state_) { +// if(new_state != control_state_) { control_state_ = new_state; switch(new_state) { default: break; @@ -260,5 +264,5 @@ void AY38910::set_control_lines(ControlLines control_lines) { case Write: set_register_value(data_input_); break; case Read: data_output_ = get_register_value(); break; } - } +// } } diff --git a/Components/AY38910/AY38910.hpp b/Components/AY38910/AY38910.hpp index 6a6d9d2f9..c079d7fce 100644 --- a/Components/AY38910/AY38910.hpp +++ b/Components/AY38910/AY38910.hpp @@ -47,6 +47,12 @@ class AY38910: public ::Outputs::Filter { */ uint8_t get_port_output(bool port_b); + /*! + Sets the value that would appear on the requested interface port if it were in output mode. + @parameter port_b @c true to get the value for Port B, @c false to get the value for Port A. + */ + void set_port_input(bool port_b, uint8_t value); + // to satisfy ::Outputs::Speaker (included via ::Outputs::Filter; not for public consumption void get_samples(unsigned int number_of_samples, int16_t *target); From eca9586a0f1e10c0a636af9e4e1d831dca4b8997 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 20:19:02 -0400 Subject: [PATCH 47/70] Fixed: input value is no longer overwritten by 0xff. The '0' key now works. --- Machines/AmstradCPC/AmstradCPC.cpp | 12 ++++-------- 1 file changed, 4 insertions(+), 8 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index a21d0be07..c4eef898a 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -266,7 +266,6 @@ class i8255PortHandler : public Intel::i8255::PortHandler { case 1: printf("Vsync, etc: %02x\n", value); break; case 2: { // TODO: the AY really should allow port communications to be active. Work needed. -// printf("%d\n", value & 15); int key_row = value & 15; if(key_row < 10) { ay_->set_port_input(false, key_state_.rows[key_row]); @@ -287,9 +286,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { uint8_t get_value(int port) { switch(port) { - case 0: - return ay_->get_data_output(); - break; + case 0: return ay_->get_data_output(); case 1: printf("[In] Vsync, etc\n"); break; case 2: printf("[In] Key row, etc\n"); break; } @@ -378,6 +375,8 @@ class ConcreteMachine: } break; case CPU::Z80::PartialMachineCycle::Input: + *cycle.value = 0xff; + // Check for a CRTC access if(!(address & 0x4000)) { switch((address >> 8) & 3) { @@ -391,8 +390,6 @@ class ConcreteMachine: if(!(address & 0x800)) { *cycle.value = i8255_.get_register((address >> 8) & 3); } - - *cycle.value = 0xff; break; case CPU::Z80::PartialMachineCycle::Interrupt: @@ -456,8 +453,7 @@ class ConcreteMachine: } void set_key_state(uint16_t key, bool isPressed) { -// if(isPressed) key_state_.rows[key >> 4] &= ~(key&7); else key_state_.rows[key >> 4] |= (key&7); -key_state_.rows[key >> 4] &= ~(key&7); + if(isPressed) key_state_.rows[key >> 4] &= ~(key&7); else key_state_.rows[key >> 4] |= (key&7); } void clear_all_keys() { From 9d43784c65b4f69e252c13b37b9b1679ada9e2fd Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 20:37:55 -0400 Subject: [PATCH 48/70] Significantly increased quantity of keys forwarded. --- .../Machine/Wrappers/CSAmstradCPC.mm | 116 +++++++++--------- 1 file changed, 56 insertions(+), 60 deletions(-) diff --git a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm index 20ca21194..ad42574bd 100644 --- a/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm +++ b/OSBindings/Mac/Clock Signal/Machine/Wrappers/CSAmstradCPC.mm @@ -69,80 +69,76 @@ - (void)setKey:(uint16_t)key isPressed:(BOOL)isPressed { @synchronized(self) { - switch(key) - { + switch(key) { case VK_ANSI_0: _amstradCPC->set_key_state(AmstradCPC::Key::Key0, isPressed); break; -/* case VK_ANSI_1: _electron.set_key_state(Electron::Key::Key1, isPressed); break; - case VK_ANSI_2: _electron.set_key_state(Electron::Key::Key2, isPressed); break; - case VK_ANSI_3: _electron.set_key_state(Electron::Key::Key3, isPressed); break; - case VK_ANSI_4: _electron.set_key_state(Electron::Key::Key4, isPressed); break; - case VK_ANSI_5: _electron.set_key_state(Electron::Key::Key5, isPressed); break; - case VK_ANSI_6: _electron.set_key_state(Electron::Key::Key6, isPressed); break; - case VK_ANSI_7: _electron.set_key_state(Electron::Key::Key7, isPressed); break; - case VK_ANSI_8: _electron.set_key_state(Electron::Key::Key8, isPressed); break; - case VK_ANSI_9: _electron.set_key_state(Electron::Key::Key9, isPressed); break; + case VK_ANSI_1: _amstradCPC->set_key_state(AmstradCPC::Key::Key1, isPressed); break; + case VK_ANSI_2: _amstradCPC->set_key_state(AmstradCPC::Key::Key2, isPressed); break; + case VK_ANSI_3: _amstradCPC->set_key_state(AmstradCPC::Key::Key3, isPressed); break; + case VK_ANSI_4: _amstradCPC->set_key_state(AmstradCPC::Key::Key4, isPressed); break; + case VK_ANSI_5: _amstradCPC->set_key_state(AmstradCPC::Key::Key5, isPressed); break; + case VK_ANSI_6: _amstradCPC->set_key_state(AmstradCPC::Key::Key6, isPressed); break; + case VK_ANSI_7: _amstradCPC->set_key_state(AmstradCPC::Key::Key7, isPressed); break; + case VK_ANSI_8: _amstradCPC->set_key_state(AmstradCPC::Key::Key8, isPressed); break; + case VK_ANSI_9: _amstradCPC->set_key_state(AmstradCPC::Key::Key9, isPressed); break; - case VK_ANSI_Q: _electron.set_key_state(Electron::Key::KeyQ, isPressed); break; - case VK_ANSI_W: _electron.set_key_state(Electron::Key::KeyW, isPressed); break; - case VK_ANSI_E: _electron.set_key_state(Electron::Key::KeyE, isPressed); break; - case VK_ANSI_R: _electron.set_key_state(Electron::Key::KeyR, isPressed); break; - case VK_ANSI_T: _electron.set_key_state(Electron::Key::KeyT, isPressed); break; - case VK_ANSI_Y: _electron.set_key_state(Electron::Key::KeyY, isPressed); break; - case VK_ANSI_U: _electron.set_key_state(Electron::Key::KeyU, isPressed); break; - case VK_ANSI_I: _electron.set_key_state(Electron::Key::KeyI, isPressed); break; - case VK_ANSI_O: _electron.set_key_state(Electron::Key::KeyO, isPressed); break; - case VK_ANSI_P: _electron.set_key_state(Electron::Key::KeyP, isPressed); break; - case VK_ANSI_A: _electron.set_key_state(Electron::Key::KeyA, isPressed); break; - case VK_ANSI_S: _electron.set_key_state(Electron::Key::KeyS, isPressed); break; - case VK_ANSI_D: _electron.set_key_state(Electron::Key::KeyD, isPressed); break; - case VK_ANSI_F: _electron.set_key_state(Electron::Key::KeyF, isPressed); break; - case VK_ANSI_G: _electron.set_key_state(Electron::Key::KeyG, isPressed); break; - case VK_ANSI_H: _electron.set_key_state(Electron::Key::KeyH, isPressed); break; - case VK_ANSI_J: _electron.set_key_state(Electron::Key::KeyJ, isPressed); break; - case VK_ANSI_K: _electron.set_key_state(Electron::Key::KeyK, isPressed); break; - case VK_ANSI_L: _electron.set_key_state(Electron::Key::KeyL, isPressed); break; - case VK_ANSI_Z: _electron.set_key_state(Electron::Key::KeyZ, isPressed); break; - case VK_ANSI_X: _electron.set_key_state(Electron::Key::KeyX, isPressed); break; - case VK_ANSI_C: _electron.set_key_state(Electron::Key::KeyC, isPressed); break; - case VK_ANSI_V: _electron.set_key_state(Electron::Key::KeyV, isPressed); break; - case VK_ANSI_B: _electron.set_key_state(Electron::Key::KeyB, isPressed); break; - case VK_ANSI_N: _electron.set_key_state(Electron::Key::KeyN, isPressed); break; - case VK_ANSI_M: _electron.set_key_state(Electron::Key::KeyM, isPressed); break; + case VK_ANSI_Q: _amstradCPC->set_key_state(AmstradCPC::Key::KeyQ, isPressed); break; + case VK_ANSI_W: _amstradCPC->set_key_state(AmstradCPC::Key::KeyW, isPressed); break; + case VK_ANSI_E: _amstradCPC->set_key_state(AmstradCPC::Key::KeyE, isPressed); break; + case VK_ANSI_R: _amstradCPC->set_key_state(AmstradCPC::Key::KeyR, isPressed); break; + case VK_ANSI_T: _amstradCPC->set_key_state(AmstradCPC::Key::KeyT, isPressed); break; + case VK_ANSI_Y: _amstradCPC->set_key_state(AmstradCPC::Key::KeyY, isPressed); break; + case VK_ANSI_U: _amstradCPC->set_key_state(AmstradCPC::Key::KeyU, isPressed); break; + case VK_ANSI_I: _amstradCPC->set_key_state(AmstradCPC::Key::KeyI, isPressed); break; + case VK_ANSI_O: _amstradCPC->set_key_state(AmstradCPC::Key::KeyO, isPressed); break; + case VK_ANSI_P: _amstradCPC->set_key_state(AmstradCPC::Key::KeyP, isPressed); break; + case VK_ANSI_A: _amstradCPC->set_key_state(AmstradCPC::Key::KeyA, isPressed); break; + case VK_ANSI_S: _amstradCPC->set_key_state(AmstradCPC::Key::KeyS, isPressed); break; + case VK_ANSI_D: _amstradCPC->set_key_state(AmstradCPC::Key::KeyD, isPressed); break; + case VK_ANSI_F: _amstradCPC->set_key_state(AmstradCPC::Key::KeyF, isPressed); break; + case VK_ANSI_G: _amstradCPC->set_key_state(AmstradCPC::Key::KeyG, isPressed); break; + case VK_ANSI_H: _amstradCPC->set_key_state(AmstradCPC::Key::KeyH, isPressed); break; + case VK_ANSI_J: _amstradCPC->set_key_state(AmstradCPC::Key::KeyJ, isPressed); break; + case VK_ANSI_K: _amstradCPC->set_key_state(AmstradCPC::Key::KeyK, isPressed); break; + case VK_ANSI_L: _amstradCPC->set_key_state(AmstradCPC::Key::KeyL, isPressed); break; + case VK_ANSI_Z: _amstradCPC->set_key_state(AmstradCPC::Key::KeyZ, isPressed); break; + case VK_ANSI_X: _amstradCPC->set_key_state(AmstradCPC::Key::KeyX, isPressed); break; + case VK_ANSI_C: _amstradCPC->set_key_state(AmstradCPC::Key::KeyC, isPressed); break; + case VK_ANSI_V: _amstradCPC->set_key_state(AmstradCPC::Key::KeyV, isPressed); break; + case VK_ANSI_B: _amstradCPC->set_key_state(AmstradCPC::Key::KeyB, isPressed); break; + case VK_ANSI_N: _amstradCPC->set_key_state(AmstradCPC::Key::KeyN, isPressed); break; + case VK_ANSI_M: _amstradCPC->set_key_state(AmstradCPC::Key::KeyM, isPressed); break; - case VK_Space: _electron.set_key_state(Electron::Key::KeySpace, isPressed); break; - case VK_ANSI_Grave: - case VK_ANSI_Backslash: - _electron.set_key_state(Electron::Key::KeyCopy, isPressed); break; - case VK_Return: _electron.set_key_state(Electron::Key::KeyReturn, isPressed); break; - case VK_ANSI_Minus: _electron.set_key_state(Electron::Key::KeyMinus, isPressed); break; + case VK_Space: _amstradCPC->set_key_state(AmstradCPC::Key::KeySpace, isPressed); break; + case VK_ANSI_Grave: _amstradCPC->set_key_state(AmstradCPC::Key::KeyCopy, isPressed); break; + case VK_Return: _amstradCPC->set_key_state(AmstradCPC::Key::KeyReturn, isPressed); break; + case VK_ANSI_Minus: _amstradCPC->set_key_state(AmstradCPC::Key::KeyMinus, isPressed); break; - case VK_RightArrow: _electron.set_key_state(Electron::Key::KeyRight, isPressed); break; - case VK_LeftArrow: _electron.set_key_state(Electron::Key::KeyLeft, isPressed); break; - case VK_DownArrow: _electron.set_key_state(Electron::Key::KeyDown, isPressed); break; - case VK_UpArrow: _electron.set_key_state(Electron::Key::KeyUp, isPressed); break; + case VK_RightArrow: _amstradCPC->set_key_state(AmstradCPC::Key::KeyRight, isPressed); break; + case VK_LeftArrow: _amstradCPC->set_key_state(AmstradCPC::Key::KeyLeft, isPressed); break; + case VK_DownArrow: _amstradCPC->set_key_state(AmstradCPC::Key::KeyDown, isPressed); break; + case VK_UpArrow: _amstradCPC->set_key_state(AmstradCPC::Key::KeyUp, isPressed); break; - case VK_Delete: _electron.set_key_state(Electron::Key::KeyDelete, isPressed); break; - case VK_Escape: _electron.set_key_state(Electron::Key::KeyEscape, isPressed); break; + case VK_Delete: _amstradCPC->set_key_state(AmstradCPC::Key::KeyDelete, isPressed); break; + case VK_Escape: _amstradCPC->set_key_state(AmstradCPC::Key::KeyEscape, isPressed); break; - case VK_ANSI_Comma: _electron.set_key_state(Electron::Key::KeyComma, isPressed); break; - case VK_ANSI_Period: _electron.set_key_state(Electron::Key::KeyFullStop, isPressed); break; + case VK_ANSI_Comma: _amstradCPC->set_key_state(AmstradCPC::Key::KeyComma, isPressed); break; + case VK_ANSI_Period: _amstradCPC->set_key_state(AmstradCPC::Key::KeyFullStop, isPressed); break; case VK_ANSI_Semicolon: - _electron.set_key_state(Electron::Key::KeySemiColon, isPressed); break; - case VK_ANSI_Quote: _electron.set_key_state(Electron::Key::KeyColon, isPressed); break; + _amstradCPC->set_key_state(AmstradCPC::Key::KeySemicolon, isPressed); break; + case VK_ANSI_Quote: _amstradCPC->set_key_state(AmstradCPC::Key::KeyColon, isPressed); break; - case VK_ANSI_Slash: _electron.set_key_state(Electron::Key::KeySlash, isPressed); break; + case VK_ANSI_Slash: _amstradCPC->set_key_state(AmstradCPC::Key::KeyForwardSlash, isPressed); break; + case VK_ANSI_Backslash: _amstradCPC->set_key_state(AmstradCPC::Key::KeyBackSlash, isPressed); break; - case VK_Shift: _electron.set_key_state(Electron::Key::KeyShift, isPressed); break; - case VK_Control: _electron.set_key_state(Electron::Key::KeyControl, isPressed); break; - case VK_Command: - case VK_Option: _electron.set_key_state(Electron::Key::KeyFunc, isPressed); break; + case VK_Shift: _amstradCPC->set_key_state(AmstradCPC::Key::KeyShift, isPressed); break; + case VK_Control: _amstradCPC->set_key_state(AmstradCPC::Key::KeyControl, isPressed); break; - case VK_F12: _electron.set_key_state(Electron::Key::KeyBreak, isPressed); break; + case VK_F12: _amstradCPC->set_key_state(AmstradCPC::Key::KeyFDot, isPressed); break; default: // printf("%02x\n", key); - break;*/ + break; } } } From 7e04d00cc1391c8ffcf05b48eca9411e5da7a7d0 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 20:39:10 -0400 Subject: [PATCH 49/70] Fixed key values, causing the new set of keys to work, decreased quantity of output and ensured that pixels appear in modes 0 and 2. --- Machines/AmstradCPC/AmstradCPC.cpp | 46 +++++++++++++++++++++--------- Machines/AmstradCPC/AmstradCPC.hpp | 16 +++++------ 2 files changed, 40 insertions(+), 22 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index c4eef898a..538ef437f 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -27,7 +27,8 @@ class CRTCBusHandler { was_hsync_(false), ram_(ram), interrupt_counter_(0), - interrupt_request_(false) {} + interrupt_request_(false), + pixel_divider_(1) {} inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { bool is_sync = state.hsync || state.vsync; @@ -197,18 +198,27 @@ class CRTCBusHandler { // TODO: no need for a full regeneration, of every mode, every time for(int c = 0; c < 256; c++) { // prepare mode 0 -// uint8_t *pixels = (uint8_t *)&mode0_output_[c]; -// pixels[0] = palette_[((c & 0x80) >> 4) | ((c & 0x08) >> 3)]; -// pixels[1] = palette_[((c & 0x40) >> 5) | ((c & 0x04) >> 2)]; + uint8_t *mode0_pixels = (uint8_t *)&mode0_output_[c]; + mode0_pixels[0] = palette_[((c & 0x80) >> 4) | ((c & 0x20) >> 3) | ((c & 0x08) >> 2) | ((c & 0x02) >> 1)]; + mode0_pixels[1] = palette_[((c & 0x40) >> 3) | ((c & 0x10) >> 2) | ((c & 0x04) >> 1) | ((c & 0x01) >> 0)]; // prepare mode 1 - uint8_t *pixels = (uint8_t *)&mode1_output_[c]; - pixels[0] = palette_[((c & 0x80) >> 6) | ((c & 0x08) >> 3)]; - pixels[1] = palette_[((c & 0x40) >> 5) | ((c & 0x04) >> 2)]; - pixels[2] = palette_[((c & 0x20) >> 4) | ((c & 0x02) >> 1)]; - pixels[3] = palette_[((c & 0x10) >> 3) | ((c & 0x01) >> 0)]; + uint8_t *mode1_pixels = (uint8_t *)&mode1_output_[c]; + mode1_pixels[0] = palette_[((c & 0x80) >> 6) | ((c & 0x08) >> 3)]; + mode1_pixels[1] = palette_[((c & 0x40) >> 5) | ((c & 0x04) >> 2)]; + mode1_pixels[2] = palette_[((c & 0x20) >> 4) | ((c & 0x02) >> 1)]; + mode1_pixels[3] = palette_[((c & 0x10) >> 3) | ((c & 0x01) >> 0)]; -// mode2_output_[c] = 0xffffff; + // prepare mode 2 + uint8_t *mode2_pixels = (uint8_t *)&mode2_output_[c]; + mode2_pixels[0] = palette_[((c & 0x80) >> 7)]; + mode2_pixels[1] = palette_[((c & 0x40) >> 6)]; + mode2_pixels[2] = palette_[((c & 0x20) >> 5)]; + mode2_pixels[3] = palette_[((c & 0x10) >> 4)]; + mode2_pixels[4] = palette_[((c & 0x08) >> 3)]; + mode2_pixels[5] = palette_[((c & 0x04) >> 2)]; + mode2_pixels[6] = palette_[((c & 0x03) >> 1)]; + mode2_pixels[7] = palette_[((c & 0x01) >> 0)]; } } } @@ -263,7 +273,9 @@ class i8255PortHandler : public Intel::i8255::PortHandler { case 0: ay_->set_data_input(value); break; - case 1: printf("Vsync, etc: %02x\n", value); break; + case 1: +// printf("Vsync, etc: %02x\n", value); + break; case 2: { // TODO: the AY really should allow port communications to be active. Work needed. int key_row = value & 15; @@ -287,8 +299,12 @@ class i8255PortHandler : public Intel::i8255::PortHandler { uint8_t get_value(int port) { switch(port) { case 0: return ay_->get_data_output(); - case 1: printf("[In] Vsync, etc\n"); break; - case 2: printf("[In] Key row, etc\n"); break; + case 1: +// printf("[In] Vsync, etc\n"); + break; + case 2: +// printf("[In] Key row, etc\n"); + break; } return 0xff; } @@ -453,7 +469,9 @@ class ConcreteMachine: } void set_key_state(uint16_t key, bool isPressed) { - if(isPressed) key_state_.rows[key >> 4] &= ~(key&7); else key_state_.rows[key >> 4] |= (key&7); + int line = key >> 4; + uint8_t mask = (uint8_t)(1 << (key & 7)); + if(isPressed) key_state_.rows[line] &= ~mask; else key_state_.rows[line] |= mask; } void clear_all_keys() { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index eef144fad..0abe34dd7 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -22,14 +22,14 @@ enum ROMType: uint8_t { enum Key: uint16_t { #define Line(l, k1, k2, k3, k4, k5, k6, k7, k8) \ - k1 = (l << 4) | 0x08,\ - k2 = (l << 4) | 0x07,\ - k3 = (l << 4) | 0x06,\ - k4 = (l << 4) | 0x05,\ - k5 = (l << 4) | 0x04,\ - k6 = (l << 4) | 0x03,\ - k7 = (l << 4) | 0x02,\ - k8 = (l << 4) | 0x01, + k1 = (l << 4) | 0x07,\ + k2 = (l << 4) | 0x06,\ + k3 = (l << 4) | 0x05,\ + k4 = (l << 4) | 0x04,\ + k5 = (l << 4) | 0x03,\ + k6 = (l << 4) | 0x02,\ + k7 = (l << 4) | 0x01,\ + k8 = (l << 4) | 0x00, Line(0, KeyFDot, KeyEnter, KeyF3, KeyF6, KeyF9, KeyDown, KeyRight, KeyUp) Line(1, KeyF0, KeyF2, KeyF1, KeyF5, KeyF8, KeyF7, KeyCopy, KeyLeft) From 31c71533013de247438105af595e42f9c7c3e5aa Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 20:52:42 -0400 Subject: [PATCH 50/70] Corrected bit to colour mapping for modes 0 and 1. The total palette is still way off but there's consistency between modes now. --- Machines/AmstradCPC/AmstradCPC.cpp | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 538ef437f..a50fc6c35 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -199,15 +199,15 @@ class CRTCBusHandler { for(int c = 0; c < 256; c++) { // prepare mode 0 uint8_t *mode0_pixels = (uint8_t *)&mode0_output_[c]; - mode0_pixels[0] = palette_[((c & 0x80) >> 4) | ((c & 0x20) >> 3) | ((c & 0x08) >> 2) | ((c & 0x02) >> 1)]; - mode0_pixels[1] = palette_[((c & 0x40) >> 3) | ((c & 0x10) >> 2) | ((c & 0x04) >> 1) | ((c & 0x01) >> 0)]; + mode0_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x20) >> 3) | ((c & 0x08) >> 2) | ((c & 0x02) << 2)]; + mode0_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x10) >> 2) | ((c & 0x04) >> 1) | ((c & 0x01) << 3)]; // prepare mode 1 uint8_t *mode1_pixels = (uint8_t *)&mode1_output_[c]; - mode1_pixels[0] = palette_[((c & 0x80) >> 6) | ((c & 0x08) >> 3)]; - mode1_pixels[1] = palette_[((c & 0x40) >> 5) | ((c & 0x04) >> 2)]; - mode1_pixels[2] = palette_[((c & 0x20) >> 4) | ((c & 0x02) >> 1)]; - mode1_pixels[3] = palette_[((c & 0x10) >> 3) | ((c & 0x01) >> 0)]; + mode1_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x08) >> 2)]; + mode1_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x04) >> 1)]; + mode1_pixels[2] = palette_[((c & 0x20) >> 5) | ((c & 0x02) >> 0)]; + mode1_pixels[3] = palette_[((c & 0x10) >> 4) | ((c & 0x01) << 1)]; // prepare mode 2 uint8_t *mode2_pixels = (uint8_t *)&mode2_output_[c]; From c6e340a8a22c168536302e0a16bcb0c3f2ac347c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 21:21:59 -0400 Subject: [PATCH 51/70] Wired up the vsync signal. Pen 15 no longer flashes like crazy. Still can't figure out why the palette is so askew; was looking for perhaps some sort of detection of a green screen rather than a colour one, but there's no obvious input for that. --- Machines/AmstradCPC/AmstradCPC.cpp | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index a50fc6c35..53a725d5c 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -178,13 +178,19 @@ class CRTCBusHandler { next_mode_ = mode; } + bool get_vsync() const { + return was_vsync_; + } + void select_pen(int pen) { + printf("Pen %d:\n", pen); pen_ = pen; } void set_colour(uint8_t colour) { + printf("Colour %d\n", colour); if(pen_ & 16) { -// printf("border: %d -> %02x\n", colour, mapped_palette_value(colour)); + printf("border: %d -> %02x\n", colour, mapped_palette_value(colour)); border_ = mapped_palette_value(colour); // TODO: should flush any border currently in progress } else { @@ -266,7 +272,8 @@ struct KeyboardState { class i8255PortHandler : public Intel::i8255::PortHandler { public: - i8255PortHandler(const KeyboardState &key_state) : key_state_(key_state) {} + i8255PortHandler(const KeyboardState &key_state, const CRTCBusHandler &crtc_bus_handler) : + key_state_(key_state), crtc_bus_handler_(crtc_bus_handler) {} void set_value(int port, uint8_t value) { switch(port) { @@ -299,9 +306,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { uint8_t get_value(int port) { switch(port) { case 0: return ay_->get_data_output(); - case 1: -// printf("[In] Vsync, etc\n"); - break; + case 1: return (crtc_bus_handler_.get_vsync() ? 1 : 0) | 0xfe; case 2: // printf("[In] Key row, etc\n"); break; @@ -316,6 +321,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { private: std::shared_ptr ay_; const KeyboardState &key_state_; + const CRTCBusHandler &crtc_bus_handler_; }; class ConcreteMachine: @@ -327,7 +333,7 @@ class ConcreteMachine: crtc_(crtc_bus_handler_), crtc_bus_handler_(ram_), i8255_(i8255_port_handler_), - i8255_port_handler_(key_state_) { + i8255_port_handler_(key_state_, crtc_bus_handler_) { // primary clock is 4Mhz set_clock_rate(4000000); } From 4b6370eb862b1c449b8d2e6a82b207197d2624e0 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 21:47:52 -0400 Subject: [PATCH 52/70] Realised my colour error: mapping the ROM numbers as though they were the hardware numbers. Having fixed that, spotted that I was deserialising R and B the wrong way around and dividing by too much. Colours now appear to be correct. --- Machines/AmstradCPC/AmstradCPC.cpp | 27 ++++++++++++++------------- 1 file changed, 14 insertions(+), 13 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 53a725d5c..bacb2955c 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -161,7 +161,7 @@ class CRTCBusHandler { "vec3 rgb_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate)" "{" "uint sample = texture(texID, coordinate).r;" - "return vec3(float(sample & 3u), float((sample >> 2) & 3u), float((sample >> 4) & 3u)) / 3.0;" + "return vec3(float((sample >> 4) & 3u), float((sample >> 2) & 3u), float(sample & 3u)) / 2.0;" "}"); // TODO: better vectorise the above. } @@ -183,24 +183,16 @@ class CRTCBusHandler { } void select_pen(int pen) { - printf("Pen %d:\n", pen); pen_ = pen; } void set_colour(uint8_t colour) { - printf("Colour %d\n", colour); if(pen_ & 16) { - printf("border: %d -> %02x\n", colour, mapped_palette_value(colour)); border_ = mapped_palette_value(colour); // TODO: should flush any border currently in progress } else { palette_[pen_] = mapped_palette_value(colour); -// for(int c = 0; c < 16; c++) { -// printf("%02x ", palette_[c]); -// } -// printf("\n"); - // TODO: no need for a full regeneration, of every mode, every time for(int c = 0; c < 256; c++) { // prepare mode 0 @@ -235,10 +227,19 @@ class CRTCBusHandler { private: uint8_t mapped_palette_value(uint8_t colour) { - uint8_t r = (colour / 3) % 3; - uint8_t g = (colour / 9) % 3; - uint8_t b = colour % 3; - return (uint8_t)(r | (g << 2) | (b << 4)); +#define COL(r, g, b) (r << 4) | (g << 2) | b + static const uint8_t mapping[32] = { + COL(1, 1, 1), COL(1, 1, 1), COL(0, 2, 1), COL(2, 2, 1), + COL(0, 0, 1), COL(2, 0, 1), COL(0, 1, 1), COL(2, 1, 1), + COL(2, 0, 1), COL(2, 2, 1), COL(2, 2, 0), COL(2, 2, 2), + COL(2, 0, 0), COL(2, 0, 2), COL(2, 1, 0), COL(2, 1, 1), + COL(0, 0, 1), COL(0, 2, 1), COL(0, 2, 0), COL(0, 2, 2), + COL(0, 0, 0), COL(0, 0, 2), COL(0, 1, 0), COL(0, 1, 2), + COL(1, 0, 1), COL(1, 2, 1), COL(1, 2, 0), COL(1, 2, 2), + COL(1, 0, 0), COL(1, 0, 2), COL(1, 1, 0), COL(1, 1, 2), + }; +#undef COL + return mapping[colour]; } unsigned int cycles_; From 8d39a2008831be7ae60c2a98b429121585c2836e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 21:51:41 -0400 Subject: [PATCH 53/70] Added proper output of mode 3, were anything ever to try to use it. --- Machines/AmstradCPC/AmstradCPC.cpp | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index bacb2955c..baf366e7a 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -93,9 +93,9 @@ class CRTCBusHandler { break; case 3: - ((uint32_t *)pixel_pointer_)[0] = mode3_output_[ram_[address]]; - ((uint32_t *)pixel_pointer_)[1] = mode3_output_[ram_[address+1]]; - pixel_pointer_ += 8; + ((uint16_t *)pixel_pointer_)[0] = mode3_output_[ram_[address]]; + ((uint16_t *)pixel_pointer_)[1] = mode3_output_[ram_[address+1]]; + pixel_pointer_ += 4; break; } @@ -217,6 +217,11 @@ class CRTCBusHandler { mode2_pixels[5] = palette_[((c & 0x04) >> 2)]; mode2_pixels[6] = palette_[((c & 0x03) >> 1)]; mode2_pixels[7] = palette_[((c & 0x01) >> 0)]; + + // prepare mode 3 + uint8_t *mode3_pixels = (uint8_t *)&mode3_output_[c]; + mode3_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x08) >> 2)]; + mode3_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x04) >> 1)]; } } } @@ -255,7 +260,7 @@ class CRTCBusHandler { uint16_t mode0_output_[256]; uint32_t mode1_output_[256]; uint64_t mode2_output_[256]; - uint32_t mode3_output_[256]; + uint16_t mode3_output_[256]; int pen_; uint8_t palette_[16]; From 6ca07f1e28865070ddb4fb3a91c0ebc88b195da1 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 22:04:58 -0400 Subject: [PATCH 54/70] I guess it might end up living somewhere else, but introduced a header with the compiler-specific stuff to allow me to force things inline. --- ClockReceiver/ForceInline.h | 18 ++++++++++++++++++ .../Mac/Clock Signal.xcodeproj/project.pbxproj | 2 ++ 2 files changed, 20 insertions(+) create mode 100644 ClockReceiver/ForceInline.h diff --git a/ClockReceiver/ForceInline.h b/ClockReceiver/ForceInline.h new file mode 100644 index 000000000..786e41c2d --- /dev/null +++ b/ClockReceiver/ForceInline.h @@ -0,0 +1,18 @@ +// +// ForceInline.h +// Clock Signal +// +// Created by Thomas Harte on 01/08/2017. +// Copyright © 2017 Thomas Harte. All rights reserved. +// + +#ifndef ForceInline_h +#define ForceInline_h + +#ifdef __GNUC__ +#define forceinline __attribute__((always_inline)) inline +#elif _MSC_VER +#define forceinline __forceinline +#endif + +#endif /* ForceInline_h */ diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 31873f6f7..ac1236c91 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -655,6 +655,7 @@ 4BAB62B41D327F7E00DF5BA0 /* G64.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = G64.hpp; sourceTree = ""; }; 4BAB62B61D3302CA00DF5BA0 /* PCMTrack.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = PCMTrack.cpp; sourceTree = ""; }; 4BAB62B71D3302CA00DF5BA0 /* PCMTrack.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = PCMTrack.hpp; sourceTree = ""; }; + 4BB06B211F316A3F00600C7A /* ForceInline.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = ForceInline.h; sourceTree = ""; }; 4BB17D4C1ED7909F00ABD1E1 /* tests.expected.json */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.json; name = tests.expected.json; path = FUSE/tests.expected.json; sourceTree = ""; }; 4BB17D4D1ED7909F00ABD1E1 /* tests.in.json */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.json; name = tests.in.json; path = FUSE/tests.in.json; sourceTree = ""; }; 4BB297E51B587D8300A49093 /* start */ = {isa = PBXFileReference; lastKnownFileType = file; path = " start"; sourceTree = ""; }; @@ -2219,6 +2220,7 @@ isa = PBXGroup; children = ( 4BF6606A1F281573002CB053 /* ClockReceiver.hpp */, + 4BB06B211F316A3F00600C7A /* ForceInline.h */, ); name = ClockReceiver; path = ../../ClockReceiver; From 587eb3a67c94069157bac9c517e79f36c17deb08 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 22:15:39 -0400 Subject: [PATCH 55/70] Factored interrupt counting out of the CRTCBusHandler. --- Machines/AmstradCPC/AmstradCPC.cpp | 96 ++++++++++++++++++------------ 1 file changed, 57 insertions(+), 39 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index baf366e7a..cc58ead3c 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -16,9 +16,55 @@ using namespace AmstradCPC; +class InterruptTimer { + public: + InterruptTimer() : timer_(0), interrupt_request_(false) {} + + inline void increment() { + timer_++; + if(timer_ == 52) { + timer_ = 0; + interrupt_request_ = true; + } + + if(reset_counter_) { + reset_counter_--; + if(!reset_counter_) { + if(timer_ < 32) { + interrupt_request_ = true; + } + timer_ = 0; + } + } + } + + inline void signal_vsync() { + reset_counter_ = 2; + } + + inline void reset_request() { + interrupt_request_ = false; + timer_ &= ~32; + } + + inline bool get_request() { + return interrupt_request_; + } + + inline void reset_count() { + timer_ = 0; + } + + private: + + int reset_counter_; + bool interrupt_request_; + int timer_; +}; + class CRTCBusHandler { public: - CRTCBusHandler(uint8_t *ram) : + CRTCBusHandler(uint8_t *ram, InterruptTimer &interrupt_timer) : cycles_(0), was_enabled_(false), was_sync_(false), @@ -26,8 +72,7 @@ class CRTCBusHandler { pixel_pointer_(nullptr), was_hsync_(false), ram_(ram), - interrupt_counter_(0), - interrupt_request_(false), + interrupt_timer_(interrupt_timer), pixel_divider_(1) {} inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { @@ -121,40 +166,17 @@ class CRTCBusHandler { } } - interrupt_counter_++; - if(interrupt_counter_ == 52) { - interrupt_request_ = true; - interrupt_counter_ = false; - } - - if(interrupt_reset_counter_) { - interrupt_reset_counter_--; - if(!interrupt_reset_counter_) { - if(interrupt_counter_ < 32) { - interrupt_request_ = true; - } - interrupt_counter_ = 0; - } - } + interrupt_timer_.increment(); } if(!was_vsync_ && state.vsync) { - interrupt_reset_counter_ = 2; + interrupt_timer_.signal_vsync(); } was_vsync_ = state.vsync; was_hsync_ = state.hsync; } - bool get_interrupt_request() { - return interrupt_request_; - } - - void reset_interrupt_request() { - interrupt_request_ = false; - interrupt_counter_ &= ~32; - } - void setup_output(float aspect_ratio) { crt_.reset(new Outputs::CRT::CRT(1024, 16, Outputs::CRT::DisplayType::PAL50, 1)); crt_->set_rgb_sampling_function( @@ -226,10 +248,6 @@ class CRTCBusHandler { } } - void reset_interrupt_counter() { - interrupt_counter_ = 0; - } - private: uint8_t mapped_palette_value(uint8_t colour) { #define COL(r, g, b) (r << 4) | (g << 2) | b @@ -266,9 +284,7 @@ class CRTCBusHandler { uint8_t palette_[16]; uint8_t border_; - int interrupt_counter_; - bool interrupt_request_; - int interrupt_reset_counter_; + InterruptTimer &interrupt_timer_; }; struct KeyboardState { @@ -337,7 +353,7 @@ class ConcreteMachine: ConcreteMachine() : crtc_counter_(HalfCycles(4)), // This starts the CRTC exactly out of phase with the memory accesses crtc_(crtc_bus_handler_), - crtc_bus_handler_(ram_), + crtc_bus_handler_(ram_, interrupt_timer_), i8255_(i8255_port_handler_), i8255_port_handler_(key_state_, crtc_bus_handler_) { // primary clock is 4Mhz @@ -356,7 +372,7 @@ class ConcreteMachine: crtc_counter_ += cycle.length; int crtc_cycles = crtc_counter_.divide(HalfCycles(8)).as_int(); if(crtc_cycles) crtc_.run_for(Cycles(1)); - set_interrupt_line(crtc_bus_handler_.get_interrupt_request()); + set_interrupt_line(interrupt_timer_.get_request()); // Stop now if no action is strictly required. if(!cycle.is_terminal()) return HalfCycles(0); @@ -381,7 +397,7 @@ class ConcreteMachine: case 2: read_pointers_[0] = (*cycle.value & 4) ? &ram_[0] : os_.data(); read_pointers_[3] = (*cycle.value & 8) ? &ram_[49152] : basic_.data(); - if(*cycle.value & 15) crtc_bus_handler_.reset_interrupt_counter(); + if(*cycle.value & 15) interrupt_timer_.reset_count(); crtc_bus_handler_.set_next_mode(*cycle.value & 3); break; case 3: printf("RAM paging?\n"); break; @@ -422,7 +438,7 @@ class ConcreteMachine: case CPU::Z80::PartialMachineCycle::Interrupt: *cycle.value = 0xff; - crtc_bus_handler_.reset_interrupt_request(); + interrupt_timer_.reset_request(); break; default: break; @@ -498,6 +514,8 @@ class ConcreteMachine: i8255PortHandler i8255_port_handler_; Intel::i8255::i8255 i8255_; + InterruptTimer interrupt_timer_; + HalfCycles clock_offset_; HalfCycles crtc_counter_; HalfCycles half_cycles_since_ay_update_; From 4d5d5041dff44a90a39a9f4f151708ee0606eebf Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 1 Aug 2017 22:18:42 -0400 Subject: [PATCH 56/70] Attempted to ensure a clean startup. --- Machines/AmstradCPC/AmstradCPC.cpp | 69 ++++++++++++++++-------------- 1 file changed, 38 insertions(+), 31 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index cc58ead3c..3ec61b7f1 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -73,7 +73,11 @@ class CRTCBusHandler { was_hsync_(false), ram_(ram), interrupt_timer_(interrupt_timer), - pixel_divider_(1) {} + pixel_divider_(1), + mode_(2), + next_mode_(2) { + build_mode_tables(); + } inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { bool is_sync = state.hsync || state.vsync; @@ -214,41 +218,44 @@ class CRTCBusHandler { // TODO: should flush any border currently in progress } else { palette_[pen_] = mapped_palette_value(colour); - // TODO: no need for a full regeneration, of every mode, every time - for(int c = 0; c < 256; c++) { - // prepare mode 0 - uint8_t *mode0_pixels = (uint8_t *)&mode0_output_[c]; - mode0_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x20) >> 3) | ((c & 0x08) >> 2) | ((c & 0x02) << 2)]; - mode0_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x10) >> 2) | ((c & 0x04) >> 1) | ((c & 0x01) << 3)]; - - // prepare mode 1 - uint8_t *mode1_pixels = (uint8_t *)&mode1_output_[c]; - mode1_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x08) >> 2)]; - mode1_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x04) >> 1)]; - mode1_pixels[2] = palette_[((c & 0x20) >> 5) | ((c & 0x02) >> 0)]; - mode1_pixels[3] = palette_[((c & 0x10) >> 4) | ((c & 0x01) << 1)]; - - // prepare mode 2 - uint8_t *mode2_pixels = (uint8_t *)&mode2_output_[c]; - mode2_pixels[0] = palette_[((c & 0x80) >> 7)]; - mode2_pixels[1] = palette_[((c & 0x40) >> 6)]; - mode2_pixels[2] = palette_[((c & 0x20) >> 5)]; - mode2_pixels[3] = palette_[((c & 0x10) >> 4)]; - mode2_pixels[4] = palette_[((c & 0x08) >> 3)]; - mode2_pixels[5] = palette_[((c & 0x04) >> 2)]; - mode2_pixels[6] = palette_[((c & 0x03) >> 1)]; - mode2_pixels[7] = palette_[((c & 0x01) >> 0)]; - - // prepare mode 3 - uint8_t *mode3_pixels = (uint8_t *)&mode3_output_[c]; - mode3_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x08) >> 2)]; - mode3_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x04) >> 1)]; - } + build_mode_tables(); } } private: + void build_mode_tables() { + for(int c = 0; c < 256; c++) { + // prepare mode 0 + uint8_t *mode0_pixels = (uint8_t *)&mode0_output_[c]; + mode0_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x20) >> 3) | ((c & 0x08) >> 2) | ((c & 0x02) << 2)]; + mode0_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x10) >> 2) | ((c & 0x04) >> 1) | ((c & 0x01) << 3)]; + + // prepare mode 1 + uint8_t *mode1_pixels = (uint8_t *)&mode1_output_[c]; + mode1_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x08) >> 2)]; + mode1_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x04) >> 1)]; + mode1_pixels[2] = palette_[((c & 0x20) >> 5) | ((c & 0x02) >> 0)]; + mode1_pixels[3] = palette_[((c & 0x10) >> 4) | ((c & 0x01) << 1)]; + + // prepare mode 2 + uint8_t *mode2_pixels = (uint8_t *)&mode2_output_[c]; + mode2_pixels[0] = palette_[((c & 0x80) >> 7)]; + mode2_pixels[1] = palette_[((c & 0x40) >> 6)]; + mode2_pixels[2] = palette_[((c & 0x20) >> 5)]; + mode2_pixels[3] = palette_[((c & 0x10) >> 4)]; + mode2_pixels[4] = palette_[((c & 0x08) >> 3)]; + mode2_pixels[5] = palette_[((c & 0x04) >> 2)]; + mode2_pixels[6] = palette_[((c & 0x03) >> 1)]; + mode2_pixels[7] = palette_[((c & 0x01) >> 0)]; + + // prepare mode 3 + uint8_t *mode3_pixels = (uint8_t *)&mode3_output_[c]; + mode3_pixels[0] = palette_[((c & 0x80) >> 7) | ((c & 0x08) >> 2)]; + mode3_pixels[1] = palette_[((c & 0x40) >> 6) | ((c & 0x04) >> 1)]; + } + } + uint8_t mapped_palette_value(uint8_t colour) { #define COL(r, g, b) (r << 4) | (g << 2) | b static const uint8_t mapping[32] = { From bda944162034eb1020e837608c5b47bb50bb2368 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 07:20:59 -0400 Subject: [PATCH 57/70] Made an attempt to clock the AY. --- Outputs/Speaker.hpp | 3 +++ 1 file changed, 3 insertions(+) diff --git a/Outputs/Speaker.hpp b/Outputs/Speaker.hpp index 7d9215f50..4dee97da4 100644 --- a/Outputs/Speaker.hpp +++ b/Outputs/Speaker.hpp @@ -87,6 +87,7 @@ class Speaker { Ensures any deferred processing occurs now. */ void flush() { + if(!queued_functions_) return; std::shared_ptr>> queued_functions = queued_functions_; queued_functions_.reset(); _queue->enqueue([queued_functions] { @@ -143,7 +144,9 @@ template class Filter: public Speaker { void run_for(const Cycles cycles) { enqueue([=]() { + printf("... %d ...\n", (unsigned int)cycles.as_int()); unsigned int cycles_remaining = (unsigned int)cycles.as_int(); + printf("!!!\n", (unsigned int)cycles.as_int()); if(coefficients_are_dirty_) update_filter_coefficients(); // if input and output rates exactly match, just accumulate results and pass on From f7e66dea61f610a5ce958b4269257b737e2f424d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 07:21:21 -0400 Subject: [PATCH 58/70] Added a compound divide and convert. --- ClockReceiver/ClockReceiver.hpp | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/ClockReceiver/ClockReceiver.hpp b/ClockReceiver/ClockReceiver.hpp index 893ce2875..097b1c6db 100644 --- a/ClockReceiver/ClockReceiver.hpp +++ b/ClockReceiver/ClockReceiver.hpp @@ -175,6 +175,17 @@ class HalfCycles: public WrappedInt { length_ &= 1; return result; } + + /*! + Severs from @c this the effect of dividing by @c divisor — @c this will end up with + the value of @c this modulo @c divisor and @c divided by @c divisor is returned. + */ + inline Cycles divide_cycles(const Cycles &divisor) { + HalfCycles half_divisor = HalfCycles(divisor); + Cycles result(length_ / half_divisor.length_); + length_ %= half_divisor.length_; + return result; + } }; /*! From f602f9b6ec0b11e0c8d2133a7630a2c56667efa8 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 07:21:33 -0400 Subject: [PATCH 59/70] Adds an attempt to clock the AY. --- Machines/AmstradCPC/AmstradCPC.cpp | 69 ++++++++++++++++++++++-------- 1 file changed, 50 insertions(+), 19 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 3ec61b7f1..82fd4a2cc 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -62,6 +62,38 @@ class InterruptTimer { int timer_; }; +class AYDeferrer { + public: + inline void setup_output() { + ay_.reset(new GI::AY38910); + ay_->set_input_rate(1000000); + } + + inline void close_output() { + ay_.reset(); + } + + inline void run_for(HalfCycles half_cycles) { + cycles_since_update_ += half_cycles; + } + + inline void flush() { + ay_->run_for(cycles_since_update_.divide_cycles(Cycles(4))); + } + + std::shared_ptr get_speaker() { + return ay_; + } + + GI::AY38910 *ay() { + return ay_.get(); + } + + private: + std::shared_ptr ay_; + HalfCycles cycles_since_update_; +}; + class CRTCBusHandler { public: CRTCBusHandler(uint8_t *ram, InterruptTimer &interrupt_timer) : @@ -301,13 +333,14 @@ struct KeyboardState { class i8255PortHandler : public Intel::i8255::PortHandler { public: - i8255PortHandler(const KeyboardState &key_state, const CRTCBusHandler &crtc_bus_handler) : - key_state_(key_state), crtc_bus_handler_(crtc_bus_handler) {} + i8255PortHandler(const KeyboardState &key_state, const CRTCBusHandler &crtc_bus_handler, AYDeferrer &ay) : + key_state_(key_state), crtc_bus_handler_(crtc_bus_handler), ay_(ay) {} void set_value(int port, uint8_t value) { switch(port) { case 0: - ay_->set_data_input(value); + ay_.flush(); + ay_.ay()->set_data_input(value); break; case 1: // printf("Vsync, etc: %02x\n", value); @@ -316,13 +349,13 @@ class i8255PortHandler : public Intel::i8255::PortHandler { // TODO: the AY really should allow port communications to be active. Work needed. int key_row = value & 15; if(key_row < 10) { - ay_->set_port_input(false, key_state_.rows[key_row]); + ay_.ay()->set_port_input(false, key_state_.rows[key_row]); } else { - ay_->set_port_input(false, 0xff); + ay_.ay()->set_port_input(false, 0xff); } // TODO: set casette motor control: ((value >> 4) & 1) // TODO: set casette output: ((value >> 5) & 1) - ay_->set_control_lines( + ay_.ay()->set_control_lines( (GI::AY38910::ControlLines)( ((value & 0x80) ? GI::AY38910::BDIR : 0) | ((value & 0x40) ? GI::AY38910::BC1 : 0) | @@ -334,7 +367,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { uint8_t get_value(int port) { switch(port) { - case 0: return ay_->get_data_output(); + case 0: return ay_.ay()->get_data_output(); case 1: return (crtc_bus_handler_.get_vsync() ? 1 : 0) | 0xfe; case 2: // printf("[In] Key row, etc\n"); @@ -343,12 +376,8 @@ class i8255PortHandler : public Intel::i8255::PortHandler { return 0xff; } - void set_ay(std::shared_ptr ay) { - ay_ = ay; - } - private: - std::shared_ptr ay_; + AYDeferrer &ay_; const KeyboardState &key_state_; const CRTCBusHandler &crtc_bus_handler_; }; @@ -362,7 +391,7 @@ class ConcreteMachine: crtc_(crtc_bus_handler_), crtc_bus_handler_(ram_, interrupt_timer_), i8255_(i8255_port_handler_), - i8255_port_handler_(key_state_, crtc_bus_handler_) { + i8255_port_handler_(key_state_, crtc_bus_handler_, ay_) { // primary clock is 4Mhz set_clock_rate(4000000); } @@ -381,6 +410,9 @@ class ConcreteMachine: if(crtc_cycles) crtc_.run_for(Cycles(1)); set_interrupt_line(interrupt_timer_.get_request()); + // Pump the AY. + ay_.run_for(cycle.length); + // Stop now if no action is strictly required. if(!cycle.is_terminal()) return HalfCycles(0); @@ -455,18 +487,17 @@ class ConcreteMachine: } void flush() { -// i8255_port_handler_.flush(); + ay_.flush(); } void setup_output(float aspect_ratio) { crtc_bus_handler_.setup_output(aspect_ratio); - ay_.reset(new GI::AY38910); - i8255_port_handler_.set_ay(ay_); + ay_.setup_output(); } void close_output() { crtc_bus_handler_.close_output(); - ay_.reset(); + ay_.close_output(); } std::shared_ptr get_crt() { @@ -474,7 +505,7 @@ class ConcreteMachine: } std::shared_ptr get_speaker() { - return ay_; + return ay_.get_speaker(); } void run_for(const Cycles cycles) { @@ -517,7 +548,7 @@ class ConcreteMachine: CRTCBusHandler crtc_bus_handler_; Motorola::CRTC::CRTC6845 crtc_; - std::shared_ptr ay_; + AYDeferrer ay_; i8255PortHandler i8255_port_handler_; Intel::i8255::i8255 i8255_; From 633d8965e2f23ee6a1ad19502881c32664bdfa7f Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 07:38:14 -0400 Subject: [PATCH 60/70] Removed accidental logging commit. --- Outputs/Speaker.hpp | 2 -- 1 file changed, 2 deletions(-) diff --git a/Outputs/Speaker.hpp b/Outputs/Speaker.hpp index 4dee97da4..a64082d42 100644 --- a/Outputs/Speaker.hpp +++ b/Outputs/Speaker.hpp @@ -144,9 +144,7 @@ template class Filter: public Speaker { void run_for(const Cycles cycles) { enqueue([=]() { - printf("... %d ...\n", (unsigned int)cycles.as_int()); unsigned int cycles_remaining = (unsigned int)cycles.as_int(); - printf("!!!\n", (unsigned int)cycles.as_int()); if(coefficients_are_dirty_) update_filter_coefficients(); // if input and output rates exactly match, just accumulate results and pass on From dcab10f53e851441503fcae46f181bab0f174f19 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 07:38:35 -0400 Subject: [PATCH 61/70] Ensured the AY's async queue doesn't just fill and fill. --- Machines/AmstradCPC/AmstradCPC.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 82fd4a2cc..f152aa12b 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -79,6 +79,7 @@ class AYDeferrer { inline void flush() { ay_->run_for(cycles_since_update_.divide_cycles(Cycles(4))); + ay_->flush(); } std::shared_ptr get_speaker() { From e50adf1cc8390715470645a7351411e0f4c7e95a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 13:50:14 -0400 Subject: [PATCH 62/70] Were my TZX support up to it, this would likely be sufficient for tape emulation. --- Machines/AmstradCPC/AmstradCPC.cpp | 28 ++++++++++++++++++++++------ 1 file changed, 22 insertions(+), 6 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index f152aa12b..6430828bb 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -14,6 +14,8 @@ #include "../../Components/AY38910/AY38910.hpp" #include "../../Components/6845/CRTC6845.hpp" +#include "../../Storage/Tape/Tape.hpp" + using namespace AmstradCPC; class InterruptTimer { @@ -334,8 +336,8 @@ struct KeyboardState { class i8255PortHandler : public Intel::i8255::PortHandler { public: - i8255PortHandler(const KeyboardState &key_state, const CRTCBusHandler &crtc_bus_handler, AYDeferrer &ay) : - key_state_(key_state), crtc_bus_handler_(crtc_bus_handler), ay_(ay) {} + i8255PortHandler(const KeyboardState &key_state, const CRTCBusHandler &crtc_bus_handler, AYDeferrer &ay, Storage::Tape::BinaryTapePlayer &tape_player) : + key_state_(key_state), crtc_bus_handler_(crtc_bus_handler), ay_(ay), tape_player_(tape_player) {} void set_value(int port, uint8_t value) { switch(port) { @@ -354,8 +356,8 @@ class i8255PortHandler : public Intel::i8255::PortHandler { } else { ay_.ay()->set_port_input(false, 0xff); } - // TODO: set casette motor control: ((value >> 4) & 1) - // TODO: set casette output: ((value >> 5) & 1) + tape_player_.set_motor_control(!!((value >> 4) & 1)); + tape_player_.set_tape_output(!!((value >> 5) & 1)); ay_.ay()->set_control_lines( (GI::AY38910::ControlLines)( ((value & 0x80) ? GI::AY38910::BDIR : 0) | @@ -369,7 +371,10 @@ class i8255PortHandler : public Intel::i8255::PortHandler { uint8_t get_value(int port) { switch(port) { case 0: return ay_.ay()->get_data_output(); - case 1: return (crtc_bus_handler_.get_vsync() ? 1 : 0) | 0xfe; + case 1: return + (crtc_bus_handler_.get_vsync() ? 0x01 : 0x00) | + (tape_player_.get_input() ? 0x80 : 0x00) | + 0x7e; case 2: // printf("[In] Key row, etc\n"); break; @@ -381,6 +386,7 @@ class i8255PortHandler : public Intel::i8255::PortHandler { AYDeferrer &ay_; const KeyboardState &key_state_; const CRTCBusHandler &crtc_bus_handler_; + Storage::Tape::BinaryTapePlayer &tape_player_; }; class ConcreteMachine: @@ -392,7 +398,8 @@ class ConcreteMachine: crtc_(crtc_bus_handler_), crtc_bus_handler_(ram_, interrupt_timer_), i8255_(i8255_port_handler_), - i8255_port_handler_(key_state_, crtc_bus_handler_, ay_) { + i8255_port_handler_(key_state_, crtc_bus_handler_, ay_, tape_player_), + tape_player_(8000000) { // primary clock is 4Mhz set_clock_rate(4000000); } @@ -411,6 +418,10 @@ class ConcreteMachine: if(crtc_cycles) crtc_.run_for(Cycles(1)); set_interrupt_line(interrupt_timer_.get_request()); + // TODO (in the player, not here): adapt it to accept an input clock rate and + // run_for as HalfCycles + tape_player_.run_for(cycle.length.as_int()); + // Pump the AY. ay_.run_for(cycle.length); @@ -524,6 +535,10 @@ class ConcreteMachine: write_pointers_[1] = &ram_[16384]; write_pointers_[2] = &ram_[32768]; write_pointers_[3] = &ram_[49152]; + + if(!target.tapes.empty()) { + tape_player_.set_tape(target.tapes.front()); + } } void set_rom(ROMType type, std::vector data) { @@ -554,6 +569,7 @@ class ConcreteMachine: Intel::i8255::i8255 i8255_; InterruptTimer interrupt_timer_; + Storage::Tape::BinaryTapePlayer tape_player_; HalfCycles clock_offset_; HalfCycles crtc_counter_; From 819761f9fb21d475e91bee73285fe91b08d5815c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 13:56:35 -0400 Subject: [PATCH 63/70] Fixed another uninitialised pointer. --- Storage/Tape/Tape.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Storage/Tape/Tape.cpp b/Storage/Tape/Tape.cpp index f59314889..280b02100 100644 --- a/Storage/Tape/Tape.cpp +++ b/Storage/Tape/Tape.cpp @@ -110,7 +110,7 @@ void TapePlayer::process_next_event() { #pragma mark - Binary Player BinaryTapePlayer::BinaryTapePlayer(unsigned int input_clock_rate) : - TapePlayer(input_clock_rate), motor_is_running_(false), input_level_(false) + TapePlayer(input_clock_rate), motor_is_running_(false), input_level_(false), delegate_(nullptr) {} void BinaryTapePlayer::set_motor_control(bool enabled) { From d7a5c3f49a7e9b1fccd588245272424f6fb1be20 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 14:12:34 -0400 Subject: [PATCH 64/70] Added support for the ID 20 block and fixed a minor error in my skip-the-contents version of block 11: length is three bytes long, not two. This gives me enough structure properly to get to the end of my current test CDT, albeit without making any of the noises. --- Storage/Tape/Formats/TZX.cpp | 16 +++++++++++++++- Storage/Tape/Formats/TZX.hpp | 1 + 2 files changed, 16 insertions(+), 1 deletion(-) diff --git a/Storage/Tape/Formats/TZX.cpp b/Storage/Tape/Formats/TZX.cpp index 6814b8bcd..44596f16f 100644 --- a/Storage/Tape/Formats/TZX.cpp +++ b/Storage/Tape/Formats/TZX.cpp @@ -56,12 +56,14 @@ void TZX::get_next_pulses() { return; } +// printf("TZX %ld\n", ftell(file_)); switch(chunk_id) { case 0x10: get_standard_speed_data_block(); break; case 0x11: get_turbo_speed_data_block(); break; case 0x12: get_pure_tone_data_block(); break; case 0x13: get_pulse_sequence(); break; case 0x19: get_generalised_data_block(); break; + case 0x20: get_pause(); break; case 0x30: { // Text description. Ripe for ignoring. @@ -183,7 +185,10 @@ void TZX::get_turbo_speed_data_block() { __unused uint16_t length_of_pilot_tone = fgetc16le(); __unused uint8_t number_of_bits_in_final_byte = (uint8_t)fgetc(file_); __unused uint16_t pause_after_block = fgetc16le(); - uint16_t data_length = fgetc16le(); + long data_length = fgetc16le(); + data_length |= (long)(fgetc(file_) << 16); + + printf("Skipping %lu bytes of turbo speed data\n", data_length); // TODO output as described fseek(file_, data_length, SEEK_CUR); @@ -203,6 +208,15 @@ void TZX::get_pulse_sequence() { } } +void TZX::get_pause() { + uint16_t duration = fgetc16le(); + if(!duration) { + // TODO (maybe): post a 'pause the tape' suggestion + } else { + post_gap(duration); + } +} + #pragma mark - Output void TZX::post_pulse(unsigned int length) { diff --git a/Storage/Tape/Formats/TZX.hpp b/Storage/Tape/Formats/TZX.hpp index d0a3eedfd..97c5916b4 100644 --- a/Storage/Tape/Formats/TZX.hpp +++ b/Storage/Tape/Formats/TZX.hpp @@ -42,6 +42,7 @@ class TZX: public PulseQueuedTape, public Storage::FileHolder { void get_pure_tone_data_block(); void get_pulse_sequence(); void get_generalised_data_block(); + void get_pause(); void get_generalised_segment(uint32_t output_symbols, uint8_t max_pulses_per_symbol, uint8_t number_of_symbols, bool is_data); From 2fd071e45de3ac12328bf0cbb174cc6251c496c2 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 14:24:34 -0400 Subject: [PATCH 65/70] Made an honest attempt at outputting turbo speed data block data. The CPC now at least starts to load. --- Storage/Tape/Formats/TZX.cpp | 43 +++++++++++++++++++++++++++--------- 1 file changed, 32 insertions(+), 11 deletions(-) diff --git a/Storage/Tape/Formats/TZX.cpp b/Storage/Tape/Formats/TZX.cpp index 44596f16f..dba86f98d 100644 --- a/Storage/Tape/Formats/TZX.cpp +++ b/Storage/Tape/Formats/TZX.cpp @@ -177,21 +177,42 @@ void TZX::get_standard_speed_data_block() { } void TZX::get_turbo_speed_data_block() { - __unused uint16_t length_of_pilot_pulse = fgetc16le(); - __unused uint16_t length_of_sync_first_pulse = fgetc16le(); - __unused uint16_t length_of_sync_second_pulse = fgetc16le(); - __unused uint16_t length_of_zero_bit_pulse = fgetc16le(); - __unused uint16_t length_of_one_bit_pulse = fgetc16le(); - __unused uint16_t length_of_pilot_tone = fgetc16le(); - __unused uint8_t number_of_bits_in_final_byte = (uint8_t)fgetc(file_); - __unused uint16_t pause_after_block = fgetc16le(); + uint16_t length_of_pilot_pulse = fgetc16le(); + uint16_t length_of_sync_first_pulse = fgetc16le(); + uint16_t length_of_sync_second_pulse = fgetc16le(); + uint16_t length_of_zero_bit_pulse = fgetc16le(); + uint16_t length_of_one_bit_pulse = fgetc16le(); + uint16_t length_of_pilot_tone = fgetc16le(); + uint8_t number_of_bits_in_final_byte = (uint8_t)fgetc(file_); + uint16_t pause_after_block = fgetc16le(); long data_length = fgetc16le(); data_length |= (long)(fgetc(file_) << 16); - printf("Skipping %lu bytes of turbo speed data\n", data_length); + // Output pilot tone. + while(length_of_pilot_tone--) { + post_pulse(length_of_pilot_pulse); + } - // TODO output as described - fseek(file_, data_length, SEEK_CUR); + // Output sync pulses. + post_pulse(length_of_sync_first_pulse); + post_pulse(length_of_sync_second_pulse); + + // Output data. + while(data_length--) { + uint8_t next_byte = (uint8_t)fgetc(file_); + + int c = data_length ? 8 : number_of_bits_in_final_byte; + while(c--) { + uint16_t pulse_length = (next_byte & 0x80) ? length_of_one_bit_pulse : length_of_zero_bit_pulse; + next_byte <<= 1; + + post_pulse(pulse_length); + post_pulse(pulse_length); + } + } + + // Output gap. + post_gap(pause_after_block); } void TZX::get_pure_tone_data_block() { From 2d2cefb0b09686268a6418c28acd942a25302a15 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 14:36:47 -0400 Subject: [PATCH 66/70] Adjusted factoring to introduce support for block 10. --- Storage/Tape/Formats/TZX.cpp | 64 +++++++++++++++++++++--------------- Storage/Tape/Formats/TZX.hpp | 13 ++++++++ 2 files changed, 50 insertions(+), 27 deletions(-) diff --git a/Storage/Tape/Formats/TZX.cpp b/Storage/Tape/Formats/TZX.cpp index dba86f98d..efa3b3073 100644 --- a/Storage/Tape/Formats/TZX.cpp +++ b/Storage/Tape/Formats/TZX.cpp @@ -162,48 +162,58 @@ void TZX::get_generalised_segment(uint32_t output_symbols, uint8_t max_pulses_pe } void TZX::get_standard_speed_data_block() { - __unused uint16_t pause_after_block = fgetc16le(); - uint16_t data_length = fgetc16le(); - if(!data_length) return; + DataBlock data_block; + data_block.length_of_pilot_pulse = 2168; + data_block.length_of_sync_first_pulse = 667; + data_block.length_of_sync_second_pulse = 735; + data_block.length_of_zero_bit_pulse = 855; + data_block.length_of_one_bit_pulse = 1710; + data_block.number_of_bits_in_final_byte = 8; + + data_block.pause_after_block = fgetc16le(); + data_block.data_length = fgetc16le(); + if(!data_block.data_length) return; uint8_t first_byte = (uint8_t)fgetc(file_); - __unused int pilot_tone_pulses = (first_byte < 128) ? 8063 : 3223; + data_block.length_of_pilot_tone = (first_byte < 128) ? 8063 : 3223; ungetc(first_byte, file_); - // TODO: output pilot_tone_pulses pulses - // TODO: output data_length bytes, in the Spectrum encoding - fseek(file_, data_length, SEEK_CUR); - // TODO: output a pause of length paused_after_block ms + get_data_block(data_block); } void TZX::get_turbo_speed_data_block() { - uint16_t length_of_pilot_pulse = fgetc16le(); - uint16_t length_of_sync_first_pulse = fgetc16le(); - uint16_t length_of_sync_second_pulse = fgetc16le(); - uint16_t length_of_zero_bit_pulse = fgetc16le(); - uint16_t length_of_one_bit_pulse = fgetc16le(); - uint16_t length_of_pilot_tone = fgetc16le(); - uint8_t number_of_bits_in_final_byte = (uint8_t)fgetc(file_); - uint16_t pause_after_block = fgetc16le(); - long data_length = fgetc16le(); - data_length |= (long)(fgetc(file_) << 16); + DataBlock data_block; + data_block.length_of_pilot_pulse = fgetc16le(); + data_block.length_of_sync_first_pulse = fgetc16le(); + data_block.length_of_sync_second_pulse = fgetc16le(); + data_block.length_of_zero_bit_pulse = fgetc16le(); + data_block.length_of_one_bit_pulse = fgetc16le(); + data_block.length_of_pilot_tone = fgetc16le(); + data_block.number_of_bits_in_final_byte = (uint8_t)fgetc(file_); + data_block.pause_after_block = fgetc16le(); + data_block.data_length = fgetc16le(); + data_block.data_length |= (long)(fgetc(file_) << 16); + get_data_block(data_block); +} + +void TZX::get_data_block(const DataBlock &data_block) { // Output pilot tone. - while(length_of_pilot_tone--) { - post_pulse(length_of_pilot_pulse); + for(unsigned int c = 0; c < data_block.length_of_pilot_tone; c++) { + post_pulse(data_block.length_of_pilot_pulse); } // Output sync pulses. - post_pulse(length_of_sync_first_pulse); - post_pulse(length_of_sync_second_pulse); + post_pulse(data_block.length_of_sync_first_pulse); + post_pulse(data_block.length_of_sync_second_pulse); // Output data. - while(data_length--) { + for(unsigned int c = 0; c < data_block.data_length; c++) { uint8_t next_byte = (uint8_t)fgetc(file_); - int c = data_length ? 8 : number_of_bits_in_final_byte; - while(c--) { - uint16_t pulse_length = (next_byte & 0x80) ? length_of_one_bit_pulse : length_of_zero_bit_pulse; + unsigned int bits = (c != data_block.data_length-1) ? 8 : data_block.number_of_bits_in_final_byte; + while(bits--) { + unsigned int pulse_length = (next_byte & 0x80) ? data_block.length_of_one_bit_pulse : data_block.length_of_zero_bit_pulse; next_byte <<= 1; post_pulse(pulse_length); @@ -212,7 +222,7 @@ void TZX::get_turbo_speed_data_block() { } // Output gap. - post_gap(pause_after_block); + post_gap(data_block.pause_after_block); } void TZX::get_pure_tone_data_block() { diff --git a/Storage/Tape/Formats/TZX.hpp b/Storage/Tape/Formats/TZX.hpp index 97c5916b4..aeedf8ce9 100644 --- a/Storage/Tape/Formats/TZX.hpp +++ b/Storage/Tape/Formats/TZX.hpp @@ -44,7 +44,20 @@ class TZX: public PulseQueuedTape, public Storage::FileHolder { void get_generalised_data_block(); void get_pause(); + struct DataBlock { + unsigned int length_of_pilot_pulse; + unsigned int length_of_sync_first_pulse; + unsigned int length_of_sync_second_pulse; + unsigned int length_of_zero_bit_pulse; + unsigned int length_of_one_bit_pulse; + unsigned int length_of_pilot_tone; + unsigned int number_of_bits_in_final_byte; + unsigned int pause_after_block; + long data_length; + }; + void get_generalised_segment(uint32_t output_symbols, uint8_t max_pulses_per_symbol, uint8_t number_of_symbols, bool is_data); + void get_data_block(const DataBlock &); void post_pulse(unsigned int length); void post_gap(unsigned int milliseconds); From 3ca9c38777ea81d269a6f6d00c7662b30301b1a9 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 19:45:58 -0400 Subject: [PATCH 67/70] =?UTF-8?q?Attempted=20to=20move=20to=20more=20accur?= =?UTF-8?q?ate=20bus=20reading=20=E2=80=94=20if=20control=20lines=20are=20?= =?UTF-8?q?set=20then=20all=20subsequent=20data=20inputs=20should=20act=20?= =?UTF-8?q?according=20to=20the=20current=20control=20lines;=20changes=20t?= =?UTF-8?q?o=20port=20input=20should=20be=20reflected=20live=20upon=20read?= =?UTF-8?q?ings,=20etc.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- Components/AY38910/AY38910.cpp | 36 +++++++++++++++++++++------------- Components/AY38910/AY38910.hpp | 2 ++ 2 files changed, 24 insertions(+), 14 deletions(-) diff --git a/Components/AY38910/AY38910.cpp b/Components/AY38910/AY38910.cpp index d3d7329c4..1a3f0b33e 100644 --- a/Components/AY38910/AY38910.cpp +++ b/Components/AY38910/AY38910.cpp @@ -166,6 +166,8 @@ void AY38910::evaluate_output_volume() { ); } +#pragma mark - Register manipulation + void AY38910::select_register(uint8_t r) { selected_register_ = r & 0xf; } @@ -227,16 +229,22 @@ uint8_t AY38910::get_register_value() { return registers_[selected_register_] | register_masks[selected_register_]; } +#pragma mark - Port handling + uint8_t AY38910::get_port_output(bool port_b) { return registers_[port_b ? 15 : 14]; } void AY38910::set_port_input(bool port_b, uint8_t value) { registers_[port_b ? 15 : 14] = value; + update_bus(); } +#pragma mark - Bus handling + void AY38910::set_data_input(uint8_t r) { data_input_ = r; + update_bus(); } uint8_t AY38910::get_data_output() { @@ -244,25 +252,25 @@ uint8_t AY38910::get_data_output() { } void AY38910::set_control_lines(ControlLines control_lines) { - ControlState new_state; switch((int)control_lines) { - default: new_state = Inactive; break; + default: control_state_ = Inactive; break; case (int)(BDIR | BC2 | BC1): case BDIR: - case BC1: new_state = LatchAddress; break; + case BC1: control_state_ = LatchAddress; break; - case (int)(BC2 | BC1): new_state = Read; break; - case (int)(BDIR | BC2): new_state = Write; break; + case (int)(BC2 | BC1): control_state_ = Read; break; + case (int)(BDIR | BC2): control_state_ = Write; break; } -// if(new_state != control_state_) { - control_state_ = new_state; - switch(new_state) { - default: break; - case LatchAddress: select_register(data_input_); break; - case Write: set_register_value(data_input_); break; - case Read: data_output_ = get_register_value(); break; - } -// } + update_bus(); +} + +void AY38910::update_bus() { + switch(control_state_) { + default: break; + case LatchAddress: select_register(data_input_); break; + case Write: set_register_value(data_input_); break; + case Read: data_output_ = get_register_value(); break; + } } diff --git a/Components/AY38910/AY38910.hpp b/Components/AY38910/AY38910.hpp index c079d7fce..8065990de 100644 --- a/Components/AY38910/AY38910.hpp +++ b/Components/AY38910/AY38910.hpp @@ -94,6 +94,8 @@ class AY38910: public ::Outputs::Filter { int16_t output_volume_; inline void evaluate_output_volume(); + + inline void update_bus(); }; }; From f5e2dd410ef39456984005effacb2f82756f2489 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 19:55:44 -0400 Subject: [PATCH 68/70] Constrained output to the centre 90%. --- Machines/AmstradCPC/AmstradCPC.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 6430828bb..3a32b6676 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -224,7 +224,7 @@ class CRTCBusHandler { "uint sample = texture(texID, coordinate).r;" "return vec3(float((sample >> 4) & 3u), float((sample >> 2) & 3u), float(sample & 3u)) / 2.0;" "}"); - // TODO: better vectorise the above. + crt_->set_visible_area(Outputs::CRT::Rect(0.05f, 0.05f, 0.9f, 0.9f)); } void close_output() { From d3bf8fa53bb099635d6faf198493b49b5ca1d892 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 20:37:26 -0400 Subject: [PATCH 69/70] Upped the documentation. --- Machines/AmstradCPC/AmstradCPC.cpp | 162 ++++++++++++++++++++++++----- Machines/AmstradCPC/AmstradCPC.hpp | 11 +- 2 files changed, 148 insertions(+), 25 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 3a32b6676..f8efa5c6a 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -18,17 +18,34 @@ using namespace AmstradCPC; +/*! + Models the CPC's interrupt timer. Inputs are vsync, hsync, interrupt acknowledge and reset, and its output + is simply yes or no on whether an interupt is currently requested. Internally it uses a counter with a period + of 52 and occasionally adjusts or makes decisions based on bit 4. + + Hsync and vsync signals are expected to come directly from the CRTC; they are not decoded from a composite stream. +*/ class InterruptTimer { public: InterruptTimer() : timer_(0), interrupt_request_(false) {} - inline void increment() { + /*! + Indicates that a new hsync pulse has been recognised. Per documentation + difficulties, it is not presently clear to me whether this should be + the leading or trailing edge of horizontal sync. + */ + inline void signal_hsync() { + // Increment the timer and if it has hit 52 then reset it and + // set the interrupt request line to true. timer_++; if(timer_ == 52) { timer_ = 0; interrupt_request_ = true; } + // If a vertical sync has previously been indicated then after two + // further horizontal syncs the timer should either (i) set the interrupt + // line, if bit 4 is clear; or (ii) reset the timer. if(reset_counter_) { reset_counter_--; if(!reset_counter_) { @@ -40,54 +57,68 @@ class InterruptTimer { } } + /// Indicates the leading edge of a new vertical sync. inline void signal_vsync() { reset_counter_ = 2; } - inline void reset_request() { + /// Indicates that an interrupt acknowledge has been received from the Z80. + inline void signal_interrupt_acknowledge() { interrupt_request_ = false; timer_ &= ~32; } + /// @returns @c true if an interrupt is currently requested; @c false otherwise. inline bool get_request() { return interrupt_request_; } + /// Resets the timer. inline void reset_count() { timer_ = 0; } private: - int reset_counter_; bool interrupt_request_; int timer_; }; +/*! + Provides a holder for an AY-3-8910 and its current cycles-since-updated count. + Therefore acts both to store an AY and to bookkeep this emulator's idiomatic + deferred clocking for this component. +*/ class AYDeferrer { public: + /// Constructs a new AY instance and sets its clock rate. inline void setup_output() { ay_.reset(new GI::AY38910); ay_->set_input_rate(1000000); } + /// Destructs the AY. inline void close_output() { ay_.reset(); } + /// Adds @c half_cycles half cycles to the amount of time that has passed. inline void run_for(HalfCycles half_cycles) { cycles_since_update_ += half_cycles; } + /// Issues a request to the AY to perform all processing up to the current time. inline void flush() { ay_->run_for(cycles_since_update_.divide_cycles(Cycles(4))); ay_->flush(); } + /// @returns the speaker the AY is using for output. std::shared_ptr get_speaker() { return ay_; } + /// @returns the AY itself. GI::AY38910 *ay() { return ay_.get(); } @@ -97,6 +128,11 @@ class AYDeferrer { HalfCycles cycles_since_update_; }; +/*! + Provides the mechanism of receipt for the CRTC outputs. In practice has the gate array's + video fetching and serialisation logic built in. So this is responsible for all video + generation and therefore owns details such as the current palette. +*/ class CRTCBusHandler { public: CRTCBusHandler(uint8_t *ram, InterruptTimer &interrupt_timer) : @@ -114,10 +150,16 @@ class CRTCBusHandler { build_mode_tables(); } + /*! + The CRTC entry function; takes the current bus state and determines what output + to produce based on the current palette and mode. + */ inline void perform_bus_cycle(const Motorola::CRTC::BusState &state) { + // Sync is taken to override pixels, and is combined as a simple OR. bool is_sync = state.hsync || state.vsync; - // if a transition between sync/border/pixels just occurred, announce it + // If a transition between sync/border/pixels just occurred, flush whatever was + // in progress to the CRT and reset counting. if(state.display_enable != was_enabled_ || is_sync != was_sync_) { if(was_sync_) { crt_->output_sync(cycles_ * 16); @@ -150,6 +192,7 @@ class CRTCBusHandler { if(pixel_pointer_) { // the CPC shuffles output lines as: // MA13 MA12 RA2 RA1 RA0 MA9 MA8 MA7 MA6 MA5 MA4 MA3 MA2 MA1 MA0 CCLK + // ... so form the real access address. uint16_t address = (uint16_t)( ((state.refresh_address & 0x3ff) << 1) | @@ -157,6 +200,7 @@ class CRTCBusHandler { ((state.refresh_address & 0x3000) << 2) ); + // fetch two bytes and translate into pixels switch(mode_) { case 0: ((uint16_t *)pixel_pointer_)[0] = mode0_output_[ram_[address]]; @@ -184,7 +228,9 @@ class CRTCBusHandler { } - // flush the current buffer if full + // flush the current buffer pixel if full; the CRTC allows many different display + // widths so it's not necessarily possible to predict the correct number in advance + // and using the upper bound could lead to inefficient behaviour if(pixel_pointer_ == pixel_data_ + 320) { crt_->output_data(cycles_ * 16, pixel_divider_); pixel_pointer_ = pixel_data_ = nullptr; @@ -193,7 +239,8 @@ class CRTCBusHandler { } } - // check for a trailing hsync + // check for a trailing hsync; if one occurred then that's the trigger potentially to change + // modes, and should also be sent on to the interrupt timer if(was_hsync_ && !state.hsync) { if(mode_ != next_mode_) { mode_ = next_mode_; @@ -205,17 +252,20 @@ class CRTCBusHandler { } } - interrupt_timer_.increment(); + interrupt_timer_.signal_hsync(); } + // check for a leading vsync; that also needs to be communicated to the interrupt timer if(!was_vsync_ && state.vsync) { interrupt_timer_.signal_vsync(); } + // update current state for edge detection next time around was_vsync_ = state.vsync; was_hsync_ = state.hsync; } + /// Constructs an appropriate CRT for video output. void setup_output(float aspect_ratio) { crt_.reset(new Outputs::CRT::CRT(1024, 16, Outputs::CRT::DisplayType::PAL50, 1)); crt_->set_rgb_sampling_function( @@ -227,26 +277,35 @@ class CRTCBusHandler { crt_->set_visible_area(Outputs::CRT::Rect(0.05f, 0.05f, 0.9f, 0.9f)); } + /// Destructs the CRT. void close_output() { crt_.reset(); } + /// @returns the CRT. std::shared_ptr get_crt() { return crt_; } + /*! + Sets the next video mode. Per the documentation, mode changes take effect only at the end of line, + not immediately. So next means "as of the end of this line". + */ void set_next_mode(int mode) { next_mode_ = mode; } + /// @returns the current value of the CRTC's vertical sync output. bool get_vsync() const { return was_vsync_; } + /// Palette management: selects a pen to modify. void select_pen(int pen) { pen_ = pen; } + /// Palette management: sets the colour of the selected pen. void set_colour(uint8_t colour) { if(pen_ & 16) { border_ = mapped_palette_value(colour); @@ -329,35 +388,58 @@ class CRTCBusHandler { InterruptTimer &interrupt_timer_; }; +/*! + Passively holds the current keyboard state. Keyboard state is modified in response + to external messages, so is handled by the machine, but is read by the i8255 port + handler, so is factored out. +*/ struct KeyboardState { KeyboardState() : rows{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff} {} uint8_t rows[10]; }; +/*! + Provides the mechanism of receipt for input and output of the 8255's various ports. +*/ class i8255PortHandler : public Intel::i8255::PortHandler { public: - i8255PortHandler(const KeyboardState &key_state, const CRTCBusHandler &crtc_bus_handler, AYDeferrer &ay, Storage::Tape::BinaryTapePlayer &tape_player) : - key_state_(key_state), crtc_bus_handler_(crtc_bus_handler), ay_(ay), tape_player_(tape_player) {} + i8255PortHandler( + const KeyboardState &key_state, + const CRTCBusHandler &crtc_bus_handler, + AYDeferrer &ay, + Storage::Tape::BinaryTapePlayer &tape_player) : + key_state_(key_state), + crtc_bus_handler_(crtc_bus_handler), + ay_(ay), + tape_player_(tape_player) {} + /// The i8255 will call this to set a new output value of @c value for @c port. void set_value(int port, uint8_t value) { switch(port) { case 0: - ay_.flush(); + // Port A is connected to the AY's data bus. ay_.ay()->set_data_input(value); break; case 1: -// printf("Vsync, etc: %02x\n", value); + // Port B is an input only. So output goes nowehere. break; case 2: { - // TODO: the AY really should allow port communications to be active. Work needed. + // The low four bits of the value sent to Port C select a keyboard line. + // At least for now, do a static push of the keyboard state here. So this + // is a capture. TODO: it should be a live connection. int key_row = value & 15; if(key_row < 10) { ay_.ay()->set_port_input(false, key_state_.rows[key_row]); } else { ay_.ay()->set_port_input(false, 0xff); } - tape_player_.set_motor_control(!!((value >> 4) & 1)); - tape_player_.set_tape_output(!!((value >> 5) & 1)); + + // Bit 4 sets the tape motor on or off. + tape_player_.set_motor_control((value & 0x10) ? true : false); + // Bit 5 sets the current tape output level + tape_player_.set_tape_output((value & 0x20) ? true : false); + + // Bits 6 and 7 set BDIR and BC1 for the AY. ay_.ay()->set_control_lines( (GI::AY38910::ControlLines)( ((value & 0x80) ? GI::AY38910::BDIR : 0) | @@ -368,18 +450,21 @@ class i8255PortHandler : public Intel::i8255::PortHandler { } } + /// The i8255 will call this to obtain a new input for @c port. uint8_t get_value(int port) { switch(port) { - case 0: return ay_.ay()->get_data_output(); + case 0: return ay_.ay()->get_data_output(); // Port A is wired to the AY case 1: return - (crtc_bus_handler_.get_vsync() ? 0x01 : 0x00) | - (tape_player_.get_input() ? 0x80 : 0x00) | - 0x7e; - case 2: -// printf("[In] Key row, etc\n"); - break; + (crtc_bus_handler_.get_vsync() ? 0x01 : 0x00) | // Bit 0 returns CRTC vsync. + (tape_player_.get_input() ? 0x80 : 0x00) | // Bit 7 returns cassette input. + 0x7e; // Bits unimplemented: + // + // Bit 6: printer ready (1 = not) + // Bit 5: the expansion port /EXP pin, so depends on connected hardware + // Bit 4: 50/60Hz switch (1 = 50Hz) + // Bits 1–3: distributor ID (111 = Amstrad) + default: return 0xff; } - return 0xff; } private: @@ -389,6 +474,9 @@ class i8255PortHandler : public Intel::i8255::PortHandler { Storage::Tape::BinaryTapePlayer &tape_player_; }; +/*! + The actual Amstrad CPC implementation; tying the 8255, 6845 and AY to the Z80. +*/ class ConcreteMachine: public CPU::Z80::Processor, public Machine { @@ -404,6 +492,7 @@ class ConcreteMachine: set_clock_rate(4000000); } + /// The entry point for performing a partial Z80 machine cycle. inline HalfCycles perform_machine_cycle(const CPU::Z80::PartialMachineCycle &cycle) { // Amstrad CPC timing scheme: assert WAIT for three out of four cycles clock_offset_ = (clock_offset_ + cycle.length) & HalfCycles(7); @@ -446,9 +535,14 @@ class ConcreteMachine: case 0: crtc_bus_handler_.select_pen(*cycle.value & 0x1f); break; case 1: crtc_bus_handler_.set_colour(*cycle.value & 0x1f); break; case 2: + // Perform ROM paging. read_pointers_[0] = (*cycle.value & 4) ? &ram_[0] : os_.data(); read_pointers_[3] = (*cycle.value & 8) ? &ram_[49152] : basic_.data(); + + // Reset the interrupt timer if requested. if(*cycle.value & 15) interrupt_timer_.reset_count(); + + // Post the next mode. crtc_bus_handler_.set_next_mode(*cycle.value & 3); break; case 3: printf("RAM paging?\n"); break; @@ -464,12 +558,13 @@ class ConcreteMachine: } } - // Check for a PIO access + // Check for an 8255 PIO access if(!(address & 0x800)) { i8255_.set_register((address >> 8) & 3, *cycle.value); } break; case CPU::Z80::PartialMachineCycle::Input: + // Default to nothing answering *cycle.value = 0xff; // Check for a CRTC access @@ -488,42 +583,56 @@ class ConcreteMachine: break; case CPU::Z80::PartialMachineCycle::Interrupt: + // Nothing is loaded onto the bus during an interrupt acknowledge, but + // the fact of the acknowledge needs to be posted on to the interrupt timer. *cycle.value = 0xff; - interrupt_timer_.reset_request(); + interrupt_timer_.signal_interrupt_acknowledge(); break; default: break; } + // This implementation doesn't use time-stuffing; once in-phase waits won't be longer + // than a single cycle so there's no real performance benefit to trying to find the + // next non-wait when a wait cycle comes in, and there'd be no benefit to reproducing + // the Z80's knowledge of where wait cycles occur here. return HalfCycles(0); } + /// Another Z80 entry point; indicates that a partcular run request has concluded. void flush() { + // Just flush the AY. ay_.flush(); } + /// A CRTMachine function; indicates that outputs should be created now. void setup_output(float aspect_ratio) { crtc_bus_handler_.setup_output(aspect_ratio); ay_.setup_output(); } + /// A CRTMachine function; indicates that outputs should be destroyed now. void close_output() { crtc_bus_handler_.close_output(); ay_.close_output(); } + /// @returns the CRT in use. std::shared_ptr get_crt() { return crtc_bus_handler_.get_crt(); } + /// @returns the speaker in use. std::shared_ptr get_speaker() { return ay_.get_speaker(); } + /// Wires virtual-dispatched CRTMachine run_for requests to the static Z80 method. void run_for(const Cycles cycles) { CPU::Z80::Processor::run_for(cycles); } + /// The ConfigurationTarget entry point; should configure this meachine as described by @c target. void configure_as_target(const StaticAnalyser::Target &target) { // Establish reset memory map as per machine model (or, for now, as a hard-wired 464) read_pointers_[0] = os_.data(); @@ -536,11 +645,13 @@ class ConcreteMachine: write_pointers_[2] = &ram_[32768]; write_pointers_[3] = &ram_[49152]; + // If there are any tapes supplied, use the first of them. if(!target.tapes.empty()) { tape_player_.set_tape(target.tapes.front()); } } + // See header; provides the system ROMs. void set_rom(ROMType type, std::vector data) { // Keep only the two ROMs that are currently of interest. switch(type) { @@ -550,12 +661,14 @@ class ConcreteMachine: } } + // See header; sets a key as either pressed or released. void set_key_state(uint16_t key, bool isPressed) { int line = key >> 4; uint8_t mask = (uint8_t)(1 << (key & 7)); if(isPressed) key_state_.rows[line] &= ~mask; else key_state_.rows[line] |= mask; } + // See header; sets all keys to released. void clear_all_keys() { memset(key_state_.rows, 0xff, 10); } @@ -584,6 +697,7 @@ class ConcreteMachine: KeyboardState key_state_; }; +// See header; constructs and returns an instance of the Amstrad CPC. Machine *Machine::AmstradCPC() { return new ConcreteMachine; } diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 0abe34dd7..964d6b22a 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -45,14 +45,23 @@ enum Key: uint16_t { #undef Line }; +/*! + Models an Amstrad CPC, a CRT-outputting machine that can accept configuration targets. +*/ class Machine: public CRTMachine::Machine, public ConfigurationTarget::Machine { public: + /// Creates an returns an Amstrad CPC on the heap. static Machine *AmstradCPC(); + /// Sets the contents of rom @c type to @c data. Assumed to be a setup step; has no effect once a machine is running. virtual void set_rom(ROMType type, std::vector data) = 0; - virtual void set_key_state(uint16_t key, bool isPressed) = 0; + + /// Indicates that @c key is either pressed or released, according to @c is_pressed. + virtual void set_key_state(uint16_t key, bool is_pressed) = 0; + + /// Indicates that all keys are now released. virtual void clear_all_keys() = 0; }; From 707821ca5573cae692dea574bf68dd73e048b162 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 2 Aug 2017 20:45:14 -0400 Subject: [PATCH 70/70] Added the normal readme to explain what's omitted here. --- ROMImages/AmstradCPC/readme.txt | 11 +++++++++++ 1 file changed, 11 insertions(+) create mode 100644 ROMImages/AmstradCPC/readme.txt diff --git a/ROMImages/AmstradCPC/readme.txt b/ROMImages/AmstradCPC/readme.txt new file mode 100644 index 000000000..ebe897eff --- /dev/null +++ b/ROMImages/AmstradCPC/readme.txt @@ -0,0 +1,11 @@ +ROM files would ordinarily go here; the copyright status of these is uncertain so they have not been included in this repository. Per http://www.worldofspectrum.org/permits/amstrad-roms.txt, Amstrad themselves allow redistribution but the status of Locomotive's input is unclear. + +Expected files: + +amsdos.rom +basic464.rom +basic664.rom +basic6128.rom +os464.rom +os664.rom +os6128.rom \ No newline at end of file