From 7bf04d5338d18e0a5397238c6470ecd9a09e9333 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 17 Feb 2020 18:08:46 -0500 Subject: [PATCH 01/48] Adds a prototype reflective enum. I need to make this scopeable before it is acceptable. --- Analyser/Static/Macintosh/Target.hpp | 14 ++-- .../Clock Signal.xcodeproj/project.pbxproj | 2 + Reflection/Enum.h | 80 +++++++++++++++++++ 3 files changed, 88 insertions(+), 8 deletions(-) create mode 100644 Reflection/Enum.h diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 5d70748f0..d1bb2d5be 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -9,19 +9,17 @@ #ifndef Analyser_Static_Macintosh_Target_h #define Analyser_Static_Macintosh_Target_h +#include "../../../Reflection/Enum.h" + +ReflectiveEnum(MacintoshModel, int, Mac128k, Mac512k, Mac128ke, MacPlus); + namespace Analyser { namespace Static { namespace Macintosh { -struct Target: public ::Analyser::Static::Target { - enum class Model { - Mac128k, - Mac512k, - Mac512ke, - MacPlus - }; - Model model = Model::MacPlus; +struct Target: public ::Analyser::Static::Target { + MacintoshModel model = MacintoshModel::MacPlus; }; } diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index ed74f9c91..3f246ad8d 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -1138,6 +1138,7 @@ 4B643F3C1D77AE5C00D431D6 /* CSMachine+Target.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = "CSMachine+Target.h"; sourceTree = ""; }; 4B643F3E1D77B88000D431D6 /* DocumentController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = DocumentController.swift; sourceTree = ""; }; 4B644ED023F0FB55006C0CC5 /* ScanSynchroniser.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; path = ScanSynchroniser.hpp; sourceTree = ""; }; + 4B644ED123FAF162006C0CC5 /* Enum.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = Enum.h; path = ../../Reflection/Enum.h; sourceTree = ""; }; 4B65085F22F4CF8D009C1100 /* Keyboard.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = Keyboard.cpp; sourceTree = ""; }; 4B670A832401CB8400D4E002 /* z80memptr.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80memptr.tap; sourceTree = ""; }; 4B670A852401CB8400D4E002 /* z80ccf.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80ccf.tap; sourceTree = ""; }; @@ -3246,6 +3247,7 @@ 4BB73E951B587A5100552FC2 = { isa = PBXGroup; children = ( + 4B644ED123FAF162006C0CC5 /* Enum.h */, 4BC76E6A1C98F43700E6EF73 /* Accelerate.framework */, 4B51F70820A521D700AFA2C1 /* Activity */, 4B8944E2201967B4007DE474 /* Analyser */, diff --git a/Reflection/Enum.h b/Reflection/Enum.h new file mode 100644 index 000000000..0b7c329c9 --- /dev/null +++ b/Reflection/Enum.h @@ -0,0 +1,80 @@ +// +// Enum.h +// Clock Signal +// +// Created by Thomas Harte on 17/02/2020. +// Copyright © 2020 Thomas Harte. All rights reserved. +// + +#ifndef Enum_h +#define Enum_h + +#include + +namespace Reflection { + +template struct Enum { + static size_t size() { + return members().size(); + } + + /*! + @returns A @c string_view name for the enum value @c e. + */ + static std::string_view toString(EnumType e) { + return members()[size_t(e)]; + } + + /*! + @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if + the string is not found. + */ + static EnumType fromString(const std::string_view &str) { + const auto member_list = members(); + auto position = std::find(member_list.begin(), member_list.end(), str); + if(position == member_list.end()) return EnumType(-1); + return EnumType(position - member_list.begin()); + } + + /*! + @returns A vector of string_views naming the members of this enum in value order. + */ + static std::vector members() { + Enum m; + const char *const declaration = m.declaration(); + const char *d_ptr = declaration; + + std::vector result; + while(true) { + // Skip non-alphas, and exit if the terminator is found. + while(*d_ptr && !isalpha(*d_ptr)) ++d_ptr; + if(!*d_ptr) break; + + // Note the current location and proceed for all alphas and digits. + const auto start = d_ptr; + while(isalpha(*d_ptr) || isdigit(*d_ptr)) ++d_ptr; + + // Add a string view. + result.emplace_back(start, d_ptr - start); + } + + return result; + } + + private: + constexpr const char *declaration(); + +}; + +} + +/*! + Provides a very limited subset of normal enums, with the addition of reflection. + + Enum members must take default values, and this enum must be in the global scope. +*/ +#define ReflectiveEnum(Name, Type, ...) \ + enum class Name: Type { __VA_ARGS__ }; \ + template <> constexpr const char *::Reflection::Enum::declaration() { return #__VA_ARGS__; }; + +#endif /* Enum_h */ From d6c6b9bdb89b6297eecb5e3c2c09fbf71ab2dd00 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 19 Feb 2020 22:26:31 -0500 Subject: [PATCH 02/48] Gives function overloading a try. --- Analyser/Static/Macintosh/Target.hpp | 16 ++++++++++++++-- .../xcschemes/Clock Signal Kiosk.xcscheme | 1 + Reflection/Enum.h | 10 +++------- 3 files changed, 18 insertions(+), 9 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index d1bb2d5be..eac46acb7 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -11,7 +11,6 @@ #include "../../../Reflection/Enum.h" -ReflectiveEnum(MacintoshModel, int, Mac128k, Mac512k, Mac128ke, MacPlus); namespace Analyser { namespace Static { @@ -19,11 +18,24 @@ namespace Macintosh { struct Target: public ::Analyser::Static::Target { - MacintoshModel model = MacintoshModel::MacPlus; + ReflectiveEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); + + Target() { +// Model m; +// printf("%s\n", __declaration(m)); + printf("%zu\n", Reflection::Enum::size()); +// for(size_t c = 0; c < Reflection::Enum::size(); ++c) { +// const auto name = Reflection::Enum::toString(Model(c)); +// printf("%.*s\n", int(name.size()), name.data()); +// } + } + + Model model = Model::MacPlus; }; } } } + #endif /* Analyser_Static_Macintosh_Target_h */ diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 6993d95b7..f84023923 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -34,6 +34,7 @@ buildConfiguration = "Release" selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB" selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB" + enableThreadSanitizer = "YES" disableMainThreadChecker = "YES" launchStyle = "0" useCustomWorkingDirectory = "NO" diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 0b7c329c9..49359c271 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -40,8 +40,8 @@ template struct Enum { @returns A vector of string_views naming the members of this enum in value order. */ static std::vector members() { - Enum m; - const char *const declaration = m.declaration(); + EnumType m; + const char *const declaration = __declaration(m); const char *d_ptr = declaration; std::vector result; @@ -60,10 +60,6 @@ template struct Enum { return result; } - - private: - constexpr const char *declaration(); - }; } @@ -75,6 +71,6 @@ template struct Enum { */ #define ReflectiveEnum(Name, Type, ...) \ enum class Name: Type { __VA_ARGS__ }; \ - template <> constexpr const char *::Reflection::Enum::declaration() { return #__VA_ARGS__; }; + constexpr const char *__declaration(Name) { return #__VA_ARGS__; } #endif /* Enum_h */ From 5248475e738c9ad4ed98e6cf650ff2d90f247c03 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 8 Mar 2020 11:26:17 -0400 Subject: [PATCH 03/48] Starts experimenting with declared reflection. --- Reflection/Struct.h | 52 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 52 insertions(+) create mode 100644 Reflection/Struct.h diff --git a/Reflection/Struct.h b/Reflection/Struct.h new file mode 100644 index 000000000..250396b76 --- /dev/null +++ b/Reflection/Struct.h @@ -0,0 +1,52 @@ +// +// Struct.h +// Clock Signal +// +// Created by Thomas Harte on 06/03/2020. +// Copyright © 2020 Thomas Harte. All rights reserved. +// + +#ifndef Struct_h +#define Struct_h + +#include +#include +#include +#include + +namespace Reflection { + +class Struct { + public: + template const Type &get(const std::string &name) { + return *std::any_cast(contents_[name]); + } + + template void set(const std::string &name, const Type &value) { + *std::any_cast(contents_[name]) = value; + } + + const std::type_info &type_of(const std::string &name) { + return contents_[name].type(); + } + + std::vector all_keys() { + std::vector keys; + for(const auto &pair: contents_) { + keys.push_back(pair.first); + } + return keys; + } + + protected: + template void declare(Type *t, const std::string &name) { + contents_[name] = t; + } + + private: + std::unordered_map contents_; +}; + +} + +#endif /* Struct_h */ From 6a8c6f5a0617a6d7d207d4aaddcfa3cc2f248b87 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 8 Mar 2020 21:23:29 -0400 Subject: [PATCH 04/48] Switches to class storage. --- Reflection/Struct.h | 53 +++++++++++++++++++++++++++++++++++++-------- 1 file changed, 44 insertions(+), 9 deletions(-) diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 250396b76..0f6e83f19 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -9,25 +9,32 @@ #ifndef Struct_h #define Struct_h -#include #include +#include +#include #include #include namespace Reflection { -class Struct { +template class Struct { public: - template const Type &get(const std::string &name) { - return *std::any_cast(contents_[name]); + template const Type *get(const std::string &name) { + const auto iterator = contents_.find(name); + if(iterator == contents_.end()) return nullptr; + return reinterpret_cast(reinterpret_cast(this) + iterator->second.offset); } template void set(const std::string &name, const Type &value) { - *std::any_cast(contents_[name]) = value; + const auto iterator = contents_.find(name); + if(iterator == contents_.end()) return; + *reinterpret_cast(reinterpret_cast(this) + iterator->second.offset) = value; } - const std::type_info &type_of(const std::string &name) { - return contents_[name].type(); + const std::type_info *type_of(const std::string &name) { + const auto iterator = contents_.find(name); + if(iterator == contents_.end()) return nullptr; + return iterator->second.type; } std::vector all_keys() { @@ -39,12 +46,40 @@ class Struct { } protected: + /* + This interface requires reflective structs to declare all fields; + specifically they should call: + + declare_field(&field1, "field1"); + declare_field(&field2, "field2"); + + Fields are registered in class storage. So callers can use needs_declare() + to determine whether a class of this type has already established the + reflective fields. + */ template void declare(Type *t, const std::string &name) { - contents_[name] = t; + contents_.emplace( + std::make_pair( + name, + Field(typeid(Type), reinterpret_cast(t) - reinterpret_cast(this)) + )); + } + + /*! + @returns @c true if this + */ + bool needs_declare() { + return !contents_.size(); } private: - std::unordered_map contents_; + struct Field { + const std::type_info *type; + ssize_t offset; + Field(const std::type_info &type, ssize_t offset) : + type(&type), offset(offset) {} + }; + static inline std::unordered_map contents_; }; } From 0502e6be67c92e66b0a23f583aeedcd03b0b8354 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 9 Mar 2020 22:05:31 -0400 Subject: [PATCH 05/48] Starts working towards a registration-based model of reflective enums. --- Analyser/Static/Macintosh/Target.hpp | 20 +++++++---------- Machines/Apple/Macintosh/Macintosh.cpp | 3 +++ .../Clock Signal.xcodeproj/project.pbxproj | 15 +++++++++++-- Reflection/Enum.h | 22 ++++++++++++++++--- Reflection/Struct.h | 12 ++++++++++ 5 files changed, 55 insertions(+), 17 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index eac46acb7..5e7d4913e 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -10,24 +10,21 @@ #define Analyser_Static_Macintosh_Target_h #include "../../../Reflection/Enum.h" - +#include "../../../Reflection/Struct.h" namespace Analyser { namespace Static { namespace Macintosh { - -struct Target: public ::Analyser::Static::Target { - ReflectiveEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { + ReflectableEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); Target() { -// Model m; -// printf("%s\n", __declaration(m)); - printf("%zu\n", Reflection::Enum::size()); -// for(size_t c = 0; c < Reflection::Enum::size(); ++c) { -// const auto name = Reflection::Enum::toString(Model(c)); -// printf("%.*s\n", int(name.size()), name.data()); -// } + // Boilerplate for declaring fields and potential values. + if(needs_declare()) { + declare(&model, "model"); + declare_enum(&model, EnumDeclaration(Model)); + } } Model model = Model::MacPlus; @@ -37,5 +34,4 @@ struct Target: public ::Analyser::Static::Target { } } - #endif /* Analyser_Static_Macintosh_Target_h */ diff --git a/Machines/Apple/Macintosh/Macintosh.cpp b/Machines/Apple/Macintosh/Macintosh.cpp index b9ceedf4c..f8b1306c4 100644 --- a/Machines/Apple/Macintosh/Macintosh.cpp +++ b/Machines/Apple/Macintosh/Macintosh.cpp @@ -51,6 +51,9 @@ namespace { constexpr int CLOCK_RATE = 7833600; + +Analyser::Static::Macintosh::Target nothing; + } namespace Apple { diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 3f246ad8d..c7f57a85b 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -1040,6 +1040,8 @@ 4B38F3471F2EC11D00D9235D /* AmstradCPC.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = AmstradCPC.hpp; path = AmstradCPC/AmstradCPC.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 = ""; }; + 4B3AF7D02413470E00873C0B /* Enum.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = Enum.h; sourceTree = ""; }; + 4B3AF7D12413472200873C0B /* Struct.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = Struct.h; sourceTree = ""; }; 4B3BA0C21D318AEB005DD7A7 /* C1540Tests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = C1540Tests.swift; sourceTree = ""; }; 4B3BA0C51D318B44005DD7A7 /* C1540Bridge.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = C1540Bridge.h; sourceTree = ""; }; 4B3BA0C61D318B44005DD7A7 /* C1540Bridge.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = C1540Bridge.mm; sourceTree = ""; }; @@ -1138,7 +1140,6 @@ 4B643F3C1D77AE5C00D431D6 /* CSMachine+Target.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = "CSMachine+Target.h"; sourceTree = ""; }; 4B643F3E1D77B88000D431D6 /* DocumentController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = DocumentController.swift; sourceTree = ""; }; 4B644ED023F0FB55006C0CC5 /* ScanSynchroniser.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; path = ScanSynchroniser.hpp; sourceTree = ""; }; - 4B644ED123FAF162006C0CC5 /* Enum.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = Enum.h; path = ../../Reflection/Enum.h; sourceTree = ""; }; 4B65085F22F4CF8D009C1100 /* Keyboard.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = Keyboard.cpp; sourceTree = ""; }; 4B670A832401CB8400D4E002 /* z80memptr.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80memptr.tap; sourceTree = ""; }; 4B670A852401CB8400D4E002 /* z80ccf.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80ccf.tap; sourceTree = ""; }; @@ -2195,6 +2196,16 @@ name = Concurrency; sourceTree = ""; }; + 4B3AF7CF2413470E00873C0B /* Reflection */ = { + isa = PBXGroup; + children = ( + 4B3AF7D02413470E00873C0B /* Enum.h */, + 4B3AF7D12413472200873C0B /* Struct.h */, + ); + name = Reflection; + path = ../../Reflection; + sourceTree = ""; + }; 4B3BA0C41D318B44005DD7A7 /* Bridges */ = { isa = PBXGroup; children = ( @@ -3247,7 +3258,6 @@ 4BB73E951B587A5100552FC2 = { isa = PBXGroup; children = ( - 4B644ED123FAF162006C0CC5 /* Enum.h */, 4BC76E6A1C98F43700E6EF73 /* Accelerate.framework */, 4B51F70820A521D700AFA2C1 /* Activity */, 4B8944E2201967B4007DE474 /* Analyser */, @@ -3265,6 +3275,7 @@ 4B366DFD1B5C165F0026627B /* Outputs */, 4BB73EDD1B587CA500552FC2 /* Processors */, 4BB73E9F1B587A5100552FC2 /* Products */, + 4B3AF7CF2413470E00873C0B /* Reflection */, 4B055A7B1FAE84A50060FFFF /* SDL */, 4B2409591C45DF85004DA684 /* SignalProcessing */, 4B69FB391C4D908A00B5F0AA /* Storage */, diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 49359c271..781c91468 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -10,9 +10,17 @@ #define Enum_h #include +#include namespace Reflection { +#define ReflectableEnum(Name, Type, ...) \ + enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; \ + constexpr static const char *__declaration##Name = #__VA_ARGS__; + +#define EnumDeclaration(Name) __declaration##Name + + template struct Enum { static size_t size() { return members().size(); @@ -69,8 +77,16 @@ template struct Enum { Enum members must take default values, and this enum must be in the global scope. */ -#define ReflectiveEnum(Name, Type, ...) \ - enum class Name: Type { __VA_ARGS__ }; \ - constexpr const char *__declaration(Name) { return #__VA_ARGS__; } +//#define DefX #define X + +//#define ForwardDeclareReflectiveEnum(Namespace, Name, ...) \ +// #define HAT + +// DeclName(m) m(__VA_ARGS__) + +// constexpr const char *__declaration(Namespace::Name) { return __VA_ARGS__; } + +//#define DefineReflectiveEnum(Name, Type) \ +// enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; #endif /* Enum_h */ diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 0f6e83f19..7e55d702a 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -57,6 +57,10 @@ template class Struct { to determine whether a class of this type has already established the reflective fields. */ + + /*! + Exposes the field pointed to by @c t for reflection as @c name. + */ template void declare(Type *t, const std::string &name) { contents_.emplace( std::make_pair( @@ -65,6 +69,14 @@ template class Struct { )); } + /*! + Provides the original declaration of an enum. + */ + template void declare_enum(Type *t, const char *declaration) { + // TODO: something. + printf("%s\n", declaration); + } + /*! @returns @c true if this */ From 238145f27f1e4c283d15af41d07e90980acad750 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 10 Mar 2020 23:36:52 -0400 Subject: [PATCH 06/48] Attempts to flesh out Reflection::Enum. --- Analyser/Static/Macintosh/Target.hpp | 2 +- Reflection/Enum.h | 99 ++++++++++++++-------------- 2 files changed, 50 insertions(+), 51 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 5e7d4913e..e9cbc82df 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -23,7 +23,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::Struct(EnumDeclaration(Model)); } } diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 781c91468..9db6bb1a7 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -11,6 +11,10 @@ #include #include +#include +#include +#include +#include namespace Reflection { @@ -20,39 +24,12 @@ namespace Reflection { #define EnumDeclaration(Name) __declaration##Name - -template struct Enum { - static size_t size() { - return members().size(); - } - - /*! - @returns A @c string_view name for the enum value @c e. - */ - static std::string_view toString(EnumType e) { - return members()[size_t(e)]; - } - - /*! - @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if - the string is not found. - */ - static EnumType fromString(const std::string_view &str) { - const auto member_list = members(); - auto position = std::find(member_list.begin(), member_list.end(), str); - if(position == member_list.end()) return EnumType(-1); - return EnumType(position - member_list.begin()); - } - - /*! - @returns A vector of string_views naming the members of this enum in value order. - */ - static std::vector members() { - EnumType m; - const char *const declaration = __declaration(m); +class Enum { + public: + template static void declare(const char *declaration) { const char *d_ptr = declaration; - std::vector result; + std::vector result; while(true) { // Skip non-alphas, and exit if the terminator is found. while(*d_ptr && !isalpha(*d_ptr)) ++d_ptr; @@ -63,30 +40,52 @@ template struct Enum { while(isalpha(*d_ptr) || isdigit(*d_ptr)) ++d_ptr; // Add a string view. - result.emplace_back(start, d_ptr - start); + result.emplace_back(std::string(start, size_t(d_ptr - start))); } - return result; + members_by_type_.emplace(std::make_pair(&typeid(Type), result)); } + + template size_t size() { + return size(typeid(Type)); + } + + size_t size(const std::type_info &type) { + const auto entry = members_by_type_.find(&type); + if(entry == members_by_type_.end()) return 0; + return entry->second.size(); + } + + /*! + @returns A @c string_view name for the enum value @c e. + */ + template static const std::string &toString(EnumType e) { + const auto entry = members_by_type_.find(&typeid(EnumType)); + if(entry == members_by_type_.end()) return empty_string_; + return entry->second[size_t(e)]; + } + + /*! + @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if + the string is not found. + */ + template Type fromString(const std::string &str) { + return Type(fromString(typeid(Type), str)); + } + + size_t fromString(const std::type_info &type, const std::string &str) { + const auto entry = members_by_type_.find(&type); + if(entry == members_by_type_.end()) return 0; + const auto iterator = std::find(entry->second.begin(), entry->second.end(), str); + if(iterator == entry->second.end()) return 0; + return size_t(iterator - entry->second.begin()); + } + + private: + static inline std::unordered_map> members_by_type_; + static inline const std::string empty_string_; }; } -/*! - Provides a very limited subset of normal enums, with the addition of reflection. - - Enum members must take default values, and this enum must be in the global scope. -*/ -//#define DefX #define X - -//#define ForwardDeclareReflectiveEnum(Namespace, Name, ...) \ -// #define HAT - -// DeclName(m) m(__VA_ARGS__) - -// constexpr const char *__declaration(Namespace::Name) { return __VA_ARGS__; } - -//#define DefineReflectiveEnum(Name, Type) \ -// enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; - #endif /* Enum_h */ From a546880a65c067e4672b93ed0a1b1a0739b4370b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 11 Mar 2020 22:06:16 -0400 Subject: [PATCH 07/48] Beefs up documentation on this miniature sort-of reflection. --- Analyser/Static/Macintosh/Target.hpp | 2 +- .../Clock Signal.xcodeproj/project.pbxproj | 2 - Reflection/Enum.h | 86 ++++++++++++++++--- Reflection/Struct.h | 25 ++++-- 4 files changed, 91 insertions(+), 24 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index e9cbc82df..86295e31a 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -23,7 +23,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::Struct(EnumDeclaration(Model)); + AnnounceEnum(Model); } } diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index c7f57a85b..ee4db8516 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -5117,7 +5117,6 @@ CLANG_WARN_DOCUMENTATION_COMMENTS = YES; CLANG_WARN_IMPLICIT_SIGN_CONVERSION = YES; CLANG_WARN_SUSPICIOUS_IMPLICIT_CONVERSION = YES; - CLANG_WARN__EXIT_TIME_DESTRUCTORS = YES; CODE_SIGN_ENTITLEMENTS = "Clock Signal/Clock Signal.entitlements"; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; @@ -5164,7 +5163,6 @@ CLANG_WARN_DOCUMENTATION_COMMENTS = YES; CLANG_WARN_IMPLICIT_SIGN_CONVERSION = YES; CLANG_WARN_SUSPICIOUS_IMPLICIT_CONVERSION = YES; - CLANG_WARN__EXIT_TIME_DESTRUCTORS = YES; CODE_SIGN_ENTITLEMENTS = "Clock Signal/Clock Signal.entitlements"; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 9db6bb1a7..1410d6a4c 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -22,11 +22,39 @@ namespace Reflection { enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; \ constexpr static const char *__declaration##Name = #__VA_ARGS__; -#define EnumDeclaration(Name) __declaration##Name +#define EnumDeclaration(Name) #Name, __declaration##Name +#define AnnounceEnum(Name) ::Reflection::Enum::declare(EnumDeclaration(Name)) + +/*! + This provides a very slight version of enum reflection; you can introspect only: + + * enums have been registered, along with the text of their declarations; + * provided that those enums do not declare specific values for their members. + + The macros above help avoid duplication of the declaration, making this just mildly less + terrible than it might have been. + + No guarantees of speed or any other kind of efficiency are offered. +*/ class Enum { public: - template static void declare(const char *declaration) { + /*! + Registers @c name and the entries within @c declaration for the enum type @c Type. + + Assuming the caller used the macros above, a standard pattern where both things can be placed in + the same namespace might look like: + + ReflectableEnum(MyEnum, int, A, B, C); + + ... + + AnnounceEnum(MyEnum) + + If AnnounceEnum cannot be placed into the same namespace as ReflectableEnum, see the + EnumDeclaration macro. + */ + template static void declare(const char *name, const char *declaration) { const char *d_ptr = declaration; std::vector result; @@ -46,43 +74,77 @@ class Enum { members_by_type_.emplace(std::make_pair(&typeid(Type), result)); } + /*! + @returns the declared name of the enum @c Type if it has been registered; the empty string otherwise. + */ + template const std::string &name() { + return name(typeid(Type)); + } + + /*! + @returns the declared name of the enum with type_info @c type if it has been registered; the empty string otherwise. + */ + const std::string &name(const std::type_info &type) { + const auto entry = names_by_type_.find(&type); + if(entry == names_by_type_.end()) return empty_string_; + return entry->second; + } + + /*! + @returns the number of members of the enum @c Type if it has been registered; 0 otherwise. + */ template size_t size() { return size(typeid(Type)); } + /*! + @returns the number of members of the enum with type_info @c type if it has been registered; @c std::string::npos otherwise. + */ size_t size(const std::type_info &type) { const auto entry = members_by_type_.find(&type); - if(entry == members_by_type_.end()) return 0; + if(entry == members_by_type_.end()) return std::string::npos; return entry->second.size(); } /*! - @returns A @c string_view name for the enum value @c e. + @returns A @c std::string name for the enum value @c e. */ - template static const std::string &toString(EnumType e) { - const auto entry = members_by_type_.find(&typeid(EnumType)); - if(entry == members_by_type_.end()) return empty_string_; - return entry->second[size_t(e)]; + template static const std::string &toString(Type e) { + return toString(typeid(Type), size_t(e)); } /*! - @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if - the string is not found. + @returns A @c std::string name for the enum value @c e from the enum with type_info @c type. + */ + static const std::string &toString(const std::type_info &type, size_t e) { + const auto entry = members_by_type_.find(&type); + if(entry == members_by_type_.end()) return empty_string_; + return entry->second[e]; + } + + /*! + @returns A value of @c Type for the name @c str, or @c EnumType(std::string::npos) if + the name is not found. */ template Type fromString(const std::string &str) { return Type(fromString(typeid(Type), str)); } + /*! + @returns A value for the name @c str in the enum with type_info @c type , or @c std::string::npos if + the name is not found. + */ size_t fromString(const std::type_info &type, const std::string &str) { const auto entry = members_by_type_.find(&type); - if(entry == members_by_type_.end()) return 0; + if(entry == members_by_type_.end()) return std::string::npos; const auto iterator = std::find(entry->second.begin(), entry->second.end(), str); - if(iterator == entry->second.end()) return 0; + if(iterator == entry->second.end()) return std::string::npos; return size_t(iterator - entry->second.begin()); } private: static inline std::unordered_map> members_by_type_; + static inline std::unordered_map names_by_type_; static inline const std::string empty_string_; }; diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 7e55d702a..27b7c28dd 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -19,24 +19,39 @@ namespace Reflection { template class Struct { public: + /*! + @returns the value of type @c Type that is loaded from the offset registered for the field @c name. + It is the caller's responsibility to provide an appropriate type of data. + */ template const Type *get(const std::string &name) { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; return reinterpret_cast(reinterpret_cast(this) + iterator->second.offset); } + /*! + Stores the @c value of type @c Type to the offset registered for the field @c name. + + It is the caller's responsibility to provide an appropriate type of data. + */ template void set(const std::string &name, const Type &value) { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return; *reinterpret_cast(reinterpret_cast(this) + iterator->second.offset) = value; } + /*! + @returns @c type_info for the field @c name. + */ const std::type_info *type_of(const std::string &name) { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; return iterator->second.type; } + /*! + @returns A vector of all declared fields for this struct. + */ std::vector all_keys() { std::vector keys; for(const auto &pair: contents_) { @@ -70,15 +85,7 @@ template class Struct { } /*! - Provides the original declaration of an enum. - */ - template void declare_enum(Type *t, const char *declaration) { - // TODO: something. - printf("%s\n", declaration); - } - - /*! - @returns @c true if this + @returns @c true if this subclass of @c Struct has not yet declared any fields. */ bool needs_declare() { return !contents_.size(); From 6a9b14f7d1456f355631a95ee0ca459158202b4e Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 17 Feb 2020 18:08:46 -0500 Subject: [PATCH 08/48] Adds a prototype reflective enum. I need to make this scopeable before it is acceptable. --- Analyser/Static/Macintosh/Target.hpp | 14 ++-- .../Clock Signal.xcodeproj/project.pbxproj | 2 + Reflection/Enum.h | 80 +++++++++++++++++++ 3 files changed, 88 insertions(+), 8 deletions(-) create mode 100644 Reflection/Enum.h diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 5d70748f0..d1bb2d5be 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -9,19 +9,17 @@ #ifndef Analyser_Static_Macintosh_Target_h #define Analyser_Static_Macintosh_Target_h +#include "../../../Reflection/Enum.h" + +ReflectiveEnum(MacintoshModel, int, Mac128k, Mac512k, Mac128ke, MacPlus); + namespace Analyser { namespace Static { namespace Macintosh { -struct Target: public ::Analyser::Static::Target { - enum class Model { - Mac128k, - Mac512k, - Mac512ke, - MacPlus - }; - Model model = Model::MacPlus; +struct Target: public ::Analyser::Static::Target { + MacintoshModel model = MacintoshModel::MacPlus; }; } diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index ed74f9c91..3f246ad8d 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -1138,6 +1138,7 @@ 4B643F3C1D77AE5C00D431D6 /* CSMachine+Target.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = "CSMachine+Target.h"; sourceTree = ""; }; 4B643F3E1D77B88000D431D6 /* DocumentController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = DocumentController.swift; sourceTree = ""; }; 4B644ED023F0FB55006C0CC5 /* ScanSynchroniser.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; path = ScanSynchroniser.hpp; sourceTree = ""; }; + 4B644ED123FAF162006C0CC5 /* Enum.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = Enum.h; path = ../../Reflection/Enum.h; sourceTree = ""; }; 4B65085F22F4CF8D009C1100 /* Keyboard.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = Keyboard.cpp; sourceTree = ""; }; 4B670A832401CB8400D4E002 /* z80memptr.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80memptr.tap; sourceTree = ""; }; 4B670A852401CB8400D4E002 /* z80ccf.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80ccf.tap; sourceTree = ""; }; @@ -3246,6 +3247,7 @@ 4BB73E951B587A5100552FC2 = { isa = PBXGroup; children = ( + 4B644ED123FAF162006C0CC5 /* Enum.h */, 4BC76E6A1C98F43700E6EF73 /* Accelerate.framework */, 4B51F70820A521D700AFA2C1 /* Activity */, 4B8944E2201967B4007DE474 /* Analyser */, diff --git a/Reflection/Enum.h b/Reflection/Enum.h new file mode 100644 index 000000000..0b7c329c9 --- /dev/null +++ b/Reflection/Enum.h @@ -0,0 +1,80 @@ +// +// Enum.h +// Clock Signal +// +// Created by Thomas Harte on 17/02/2020. +// Copyright © 2020 Thomas Harte. All rights reserved. +// + +#ifndef Enum_h +#define Enum_h + +#include + +namespace Reflection { + +template struct Enum { + static size_t size() { + return members().size(); + } + + /*! + @returns A @c string_view name for the enum value @c e. + */ + static std::string_view toString(EnumType e) { + return members()[size_t(e)]; + } + + /*! + @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if + the string is not found. + */ + static EnumType fromString(const std::string_view &str) { + const auto member_list = members(); + auto position = std::find(member_list.begin(), member_list.end(), str); + if(position == member_list.end()) return EnumType(-1); + return EnumType(position - member_list.begin()); + } + + /*! + @returns A vector of string_views naming the members of this enum in value order. + */ + static std::vector members() { + Enum m; + const char *const declaration = m.declaration(); + const char *d_ptr = declaration; + + std::vector result; + while(true) { + // Skip non-alphas, and exit if the terminator is found. + while(*d_ptr && !isalpha(*d_ptr)) ++d_ptr; + if(!*d_ptr) break; + + // Note the current location and proceed for all alphas and digits. + const auto start = d_ptr; + while(isalpha(*d_ptr) || isdigit(*d_ptr)) ++d_ptr; + + // Add a string view. + result.emplace_back(start, d_ptr - start); + } + + return result; + } + + private: + constexpr const char *declaration(); + +}; + +} + +/*! + Provides a very limited subset of normal enums, with the addition of reflection. + + Enum members must take default values, and this enum must be in the global scope. +*/ +#define ReflectiveEnum(Name, Type, ...) \ + enum class Name: Type { __VA_ARGS__ }; \ + template <> constexpr const char *::Reflection::Enum::declaration() { return #__VA_ARGS__; }; + +#endif /* Enum_h */ From 5a8fcac4dc0d0d3d559005e6a80ffada2093bcba Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 19 Feb 2020 22:26:31 -0500 Subject: [PATCH 09/48] Gives function overloading a try. --- Analyser/Static/Macintosh/Target.hpp | 16 ++++++++++++++-- .../xcschemes/Clock Signal Kiosk.xcscheme | 1 + Reflection/Enum.h | 10 +++------- 3 files changed, 18 insertions(+), 9 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index d1bb2d5be..eac46acb7 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -11,7 +11,6 @@ #include "../../../Reflection/Enum.h" -ReflectiveEnum(MacintoshModel, int, Mac128k, Mac512k, Mac128ke, MacPlus); namespace Analyser { namespace Static { @@ -19,11 +18,24 @@ namespace Macintosh { struct Target: public ::Analyser::Static::Target { - MacintoshModel model = MacintoshModel::MacPlus; + ReflectiveEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); + + Target() { +// Model m; +// printf("%s\n", __declaration(m)); + printf("%zu\n", Reflection::Enum::size()); +// for(size_t c = 0; c < Reflection::Enum::size(); ++c) { +// const auto name = Reflection::Enum::toString(Model(c)); +// printf("%.*s\n", int(name.size()), name.data()); +// } + } + + Model model = Model::MacPlus; }; } } } + #endif /* Analyser_Static_Macintosh_Target_h */ diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 6993d95b7..f84023923 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -34,6 +34,7 @@ buildConfiguration = "Release" selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB" selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB" + enableThreadSanitizer = "YES" disableMainThreadChecker = "YES" launchStyle = "0" useCustomWorkingDirectory = "NO" diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 0b7c329c9..49359c271 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -40,8 +40,8 @@ template struct Enum { @returns A vector of string_views naming the members of this enum in value order. */ static std::vector members() { - Enum m; - const char *const declaration = m.declaration(); + EnumType m; + const char *const declaration = __declaration(m); const char *d_ptr = declaration; std::vector result; @@ -60,10 +60,6 @@ template struct Enum { return result; } - - private: - constexpr const char *declaration(); - }; } @@ -75,6 +71,6 @@ template struct Enum { */ #define ReflectiveEnum(Name, Type, ...) \ enum class Name: Type { __VA_ARGS__ }; \ - template <> constexpr const char *::Reflection::Enum::declaration() { return #__VA_ARGS__; }; + constexpr const char *__declaration(Name) { return #__VA_ARGS__; } #endif /* Enum_h */ From 3428e9887d69e33e75fb118a3fc79d7a9647ed9f Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 8 Mar 2020 11:26:17 -0400 Subject: [PATCH 10/48] Starts experimenting with declared reflection. --- Reflection/Struct.h | 52 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 52 insertions(+) create mode 100644 Reflection/Struct.h diff --git a/Reflection/Struct.h b/Reflection/Struct.h new file mode 100644 index 000000000..250396b76 --- /dev/null +++ b/Reflection/Struct.h @@ -0,0 +1,52 @@ +// +// Struct.h +// Clock Signal +// +// Created by Thomas Harte on 06/03/2020. +// Copyright © 2020 Thomas Harte. All rights reserved. +// + +#ifndef Struct_h +#define Struct_h + +#include +#include +#include +#include + +namespace Reflection { + +class Struct { + public: + template const Type &get(const std::string &name) { + return *std::any_cast(contents_[name]); + } + + template void set(const std::string &name, const Type &value) { + *std::any_cast(contents_[name]) = value; + } + + const std::type_info &type_of(const std::string &name) { + return contents_[name].type(); + } + + std::vector all_keys() { + std::vector keys; + for(const auto &pair: contents_) { + keys.push_back(pair.first); + } + return keys; + } + + protected: + template void declare(Type *t, const std::string &name) { + contents_[name] = t; + } + + private: + std::unordered_map contents_; +}; + +} + +#endif /* Struct_h */ From a99bb3ba6d8b48401d4447d1f52e4b935ccd67d5 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 8 Mar 2020 21:23:29 -0400 Subject: [PATCH 11/48] Switches to class storage. --- Reflection/Struct.h | 53 +++++++++++++++++++++++++++++++++++++-------- 1 file changed, 44 insertions(+), 9 deletions(-) diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 250396b76..0f6e83f19 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -9,25 +9,32 @@ #ifndef Struct_h #define Struct_h -#include #include +#include +#include #include #include namespace Reflection { -class Struct { +template class Struct { public: - template const Type &get(const std::string &name) { - return *std::any_cast(contents_[name]); + template const Type *get(const std::string &name) { + const auto iterator = contents_.find(name); + if(iterator == contents_.end()) return nullptr; + return reinterpret_cast(reinterpret_cast(this) + iterator->second.offset); } template void set(const std::string &name, const Type &value) { - *std::any_cast(contents_[name]) = value; + const auto iterator = contents_.find(name); + if(iterator == contents_.end()) return; + *reinterpret_cast(reinterpret_cast(this) + iterator->second.offset) = value; } - const std::type_info &type_of(const std::string &name) { - return contents_[name].type(); + const std::type_info *type_of(const std::string &name) { + const auto iterator = contents_.find(name); + if(iterator == contents_.end()) return nullptr; + return iterator->second.type; } std::vector all_keys() { @@ -39,12 +46,40 @@ class Struct { } protected: + /* + This interface requires reflective structs to declare all fields; + specifically they should call: + + declare_field(&field1, "field1"); + declare_field(&field2, "field2"); + + Fields are registered in class storage. So callers can use needs_declare() + to determine whether a class of this type has already established the + reflective fields. + */ template void declare(Type *t, const std::string &name) { - contents_[name] = t; + contents_.emplace( + std::make_pair( + name, + Field(typeid(Type), reinterpret_cast(t) - reinterpret_cast(this)) + )); + } + + /*! + @returns @c true if this + */ + bool needs_declare() { + return !contents_.size(); } private: - std::unordered_map contents_; + struct Field { + const std::type_info *type; + ssize_t offset; + Field(const std::type_info &type, ssize_t offset) : + type(&type), offset(offset) {} + }; + static inline std::unordered_map contents_; }; } From ce80825abbb8b4aec0357441b89d2d7fd97f487a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 9 Mar 2020 22:05:31 -0400 Subject: [PATCH 12/48] Starts working towards a registration-based model of reflective enums. --- Analyser/Static/Macintosh/Target.hpp | 20 +++++++---------- Machines/Apple/Macintosh/Macintosh.cpp | 3 +++ .../Clock Signal.xcodeproj/project.pbxproj | 15 +++++++++++-- Reflection/Enum.h | 22 ++++++++++++++++--- Reflection/Struct.h | 12 ++++++++++ 5 files changed, 55 insertions(+), 17 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index eac46acb7..5e7d4913e 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -10,24 +10,21 @@ #define Analyser_Static_Macintosh_Target_h #include "../../../Reflection/Enum.h" - +#include "../../../Reflection/Struct.h" namespace Analyser { namespace Static { namespace Macintosh { - -struct Target: public ::Analyser::Static::Target { - ReflectiveEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { + ReflectableEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); Target() { -// Model m; -// printf("%s\n", __declaration(m)); - printf("%zu\n", Reflection::Enum::size()); -// for(size_t c = 0; c < Reflection::Enum::size(); ++c) { -// const auto name = Reflection::Enum::toString(Model(c)); -// printf("%.*s\n", int(name.size()), name.data()); -// } + // Boilerplate for declaring fields and potential values. + if(needs_declare()) { + declare(&model, "model"); + declare_enum(&model, EnumDeclaration(Model)); + } } Model model = Model::MacPlus; @@ -37,5 +34,4 @@ struct Target: public ::Analyser::Static::Target { } } - #endif /* Analyser_Static_Macintosh_Target_h */ diff --git a/Machines/Apple/Macintosh/Macintosh.cpp b/Machines/Apple/Macintosh/Macintosh.cpp index b9ceedf4c..f8b1306c4 100644 --- a/Machines/Apple/Macintosh/Macintosh.cpp +++ b/Machines/Apple/Macintosh/Macintosh.cpp @@ -51,6 +51,9 @@ namespace { constexpr int CLOCK_RATE = 7833600; + +Analyser::Static::Macintosh::Target nothing; + } namespace Apple { diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 3f246ad8d..c7f57a85b 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -1040,6 +1040,8 @@ 4B38F3471F2EC11D00D9235D /* AmstradCPC.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = AmstradCPC.hpp; path = AmstradCPC/AmstradCPC.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 = ""; }; + 4B3AF7D02413470E00873C0B /* Enum.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = Enum.h; sourceTree = ""; }; + 4B3AF7D12413472200873C0B /* Struct.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = Struct.h; sourceTree = ""; }; 4B3BA0C21D318AEB005DD7A7 /* C1540Tests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = C1540Tests.swift; sourceTree = ""; }; 4B3BA0C51D318B44005DD7A7 /* C1540Bridge.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = C1540Bridge.h; sourceTree = ""; }; 4B3BA0C61D318B44005DD7A7 /* C1540Bridge.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = C1540Bridge.mm; sourceTree = ""; }; @@ -1138,7 +1140,6 @@ 4B643F3C1D77AE5C00D431D6 /* CSMachine+Target.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = "CSMachine+Target.h"; sourceTree = ""; }; 4B643F3E1D77B88000D431D6 /* DocumentController.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = DocumentController.swift; sourceTree = ""; }; 4B644ED023F0FB55006C0CC5 /* ScanSynchroniser.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; path = ScanSynchroniser.hpp; sourceTree = ""; }; - 4B644ED123FAF162006C0CC5 /* Enum.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = Enum.h; path = ../../Reflection/Enum.h; sourceTree = ""; }; 4B65085F22F4CF8D009C1100 /* Keyboard.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = Keyboard.cpp; sourceTree = ""; }; 4B670A832401CB8400D4E002 /* z80memptr.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80memptr.tap; sourceTree = ""; }; 4B670A852401CB8400D4E002 /* z80ccf.tap */ = {isa = PBXFileReference; lastKnownFileType = file; path = z80ccf.tap; sourceTree = ""; }; @@ -2195,6 +2196,16 @@ name = Concurrency; sourceTree = ""; }; + 4B3AF7CF2413470E00873C0B /* Reflection */ = { + isa = PBXGroup; + children = ( + 4B3AF7D02413470E00873C0B /* Enum.h */, + 4B3AF7D12413472200873C0B /* Struct.h */, + ); + name = Reflection; + path = ../../Reflection; + sourceTree = ""; + }; 4B3BA0C41D318B44005DD7A7 /* Bridges */ = { isa = PBXGroup; children = ( @@ -3247,7 +3258,6 @@ 4BB73E951B587A5100552FC2 = { isa = PBXGroup; children = ( - 4B644ED123FAF162006C0CC5 /* Enum.h */, 4BC76E6A1C98F43700E6EF73 /* Accelerate.framework */, 4B51F70820A521D700AFA2C1 /* Activity */, 4B8944E2201967B4007DE474 /* Analyser */, @@ -3265,6 +3275,7 @@ 4B366DFD1B5C165F0026627B /* Outputs */, 4BB73EDD1B587CA500552FC2 /* Processors */, 4BB73E9F1B587A5100552FC2 /* Products */, + 4B3AF7CF2413470E00873C0B /* Reflection */, 4B055A7B1FAE84A50060FFFF /* SDL */, 4B2409591C45DF85004DA684 /* SignalProcessing */, 4B69FB391C4D908A00B5F0AA /* Storage */, diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 49359c271..781c91468 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -10,9 +10,17 @@ #define Enum_h #include +#include namespace Reflection { +#define ReflectableEnum(Name, Type, ...) \ + enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; \ + constexpr static const char *__declaration##Name = #__VA_ARGS__; + +#define EnumDeclaration(Name) __declaration##Name + + template struct Enum { static size_t size() { return members().size(); @@ -69,8 +77,16 @@ template struct Enum { Enum members must take default values, and this enum must be in the global scope. */ -#define ReflectiveEnum(Name, Type, ...) \ - enum class Name: Type { __VA_ARGS__ }; \ - constexpr const char *__declaration(Name) { return #__VA_ARGS__; } +//#define DefX #define X + +//#define ForwardDeclareReflectiveEnum(Namespace, Name, ...) \ +// #define HAT + +// DeclName(m) m(__VA_ARGS__) + +// constexpr const char *__declaration(Namespace::Name) { return __VA_ARGS__; } + +//#define DefineReflectiveEnum(Name, Type) \ +// enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; #endif /* Enum_h */ diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 0f6e83f19..7e55d702a 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -57,6 +57,10 @@ template class Struct { to determine whether a class of this type has already established the reflective fields. */ + + /*! + Exposes the field pointed to by @c t for reflection as @c name. + */ template void declare(Type *t, const std::string &name) { contents_.emplace( std::make_pair( @@ -65,6 +69,14 @@ template class Struct { )); } + /*! + Provides the original declaration of an enum. + */ + template void declare_enum(Type *t, const char *declaration) { + // TODO: something. + printf("%s\n", declaration); + } + /*! @returns @c true if this */ From 7e8b86e9bb8bb7fa634bf885c63b919ee7236ddf Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 10 Mar 2020 23:36:52 -0400 Subject: [PATCH 13/48] Attempts to flesh out Reflection::Enum. --- Analyser/Static/Macintosh/Target.hpp | 2 +- Reflection/Enum.h | 99 ++++++++++++++-------------- 2 files changed, 50 insertions(+), 51 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 5e7d4913e..e9cbc82df 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -23,7 +23,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::Struct(EnumDeclaration(Model)); } } diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 781c91468..9db6bb1a7 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -11,6 +11,10 @@ #include #include +#include +#include +#include +#include namespace Reflection { @@ -20,39 +24,12 @@ namespace Reflection { #define EnumDeclaration(Name) __declaration##Name - -template struct Enum { - static size_t size() { - return members().size(); - } - - /*! - @returns A @c string_view name for the enum value @c e. - */ - static std::string_view toString(EnumType e) { - return members()[size_t(e)]; - } - - /*! - @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if - the string is not found. - */ - static EnumType fromString(const std::string_view &str) { - const auto member_list = members(); - auto position = std::find(member_list.begin(), member_list.end(), str); - if(position == member_list.end()) return EnumType(-1); - return EnumType(position - member_list.begin()); - } - - /*! - @returns A vector of string_views naming the members of this enum in value order. - */ - static std::vector members() { - EnumType m; - const char *const declaration = __declaration(m); +class Enum { + public: + template static void declare(const char *declaration) { const char *d_ptr = declaration; - std::vector result; + std::vector result; while(true) { // Skip non-alphas, and exit if the terminator is found. while(*d_ptr && !isalpha(*d_ptr)) ++d_ptr; @@ -63,30 +40,52 @@ template struct Enum { while(isalpha(*d_ptr) || isdigit(*d_ptr)) ++d_ptr; // Add a string view. - result.emplace_back(start, d_ptr - start); + result.emplace_back(std::string(start, size_t(d_ptr - start))); } - return result; + members_by_type_.emplace(std::make_pair(&typeid(Type), result)); } + + template size_t size() { + return size(typeid(Type)); + } + + size_t size(const std::type_info &type) { + const auto entry = members_by_type_.find(&type); + if(entry == members_by_type_.end()) return 0; + return entry->second.size(); + } + + /*! + @returns A @c string_view name for the enum value @c e. + */ + template static const std::string &toString(EnumType e) { + const auto entry = members_by_type_.find(&typeid(EnumType)); + if(entry == members_by_type_.end()) return empty_string_; + return entry->second[size_t(e)]; + } + + /*! + @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if + the string is not found. + */ + template Type fromString(const std::string &str) { + return Type(fromString(typeid(Type), str)); + } + + size_t fromString(const std::type_info &type, const std::string &str) { + const auto entry = members_by_type_.find(&type); + if(entry == members_by_type_.end()) return 0; + const auto iterator = std::find(entry->second.begin(), entry->second.end(), str); + if(iterator == entry->second.end()) return 0; + return size_t(iterator - entry->second.begin()); + } + + private: + static inline std::unordered_map> members_by_type_; + static inline const std::string empty_string_; }; } -/*! - Provides a very limited subset of normal enums, with the addition of reflection. - - Enum members must take default values, and this enum must be in the global scope. -*/ -//#define DefX #define X - -//#define ForwardDeclareReflectiveEnum(Namespace, Name, ...) \ -// #define HAT - -// DeclName(m) m(__VA_ARGS__) - -// constexpr const char *__declaration(Namespace::Name) { return __VA_ARGS__; } - -//#define DefineReflectiveEnum(Name, Type) \ -// enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; - #endif /* Enum_h */ From 044a2b67e19e17ec97a55e055cff00d6f3dbb255 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 11 Mar 2020 22:06:16 -0400 Subject: [PATCH 14/48] Beefs up documentation on this miniature sort-of reflection. --- Analyser/Static/Macintosh/Target.hpp | 2 +- .../Clock Signal.xcodeproj/project.pbxproj | 2 - Reflection/Enum.h | 86 ++++++++++++++++--- Reflection/Struct.h | 25 ++++-- 4 files changed, 91 insertions(+), 24 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index e9cbc82df..86295e31a 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -23,7 +23,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::Struct(EnumDeclaration(Model)); + AnnounceEnum(Model); } } diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index c7f57a85b..ee4db8516 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -5117,7 +5117,6 @@ CLANG_WARN_DOCUMENTATION_COMMENTS = YES; CLANG_WARN_IMPLICIT_SIGN_CONVERSION = YES; CLANG_WARN_SUSPICIOUS_IMPLICIT_CONVERSION = YES; - CLANG_WARN__EXIT_TIME_DESTRUCTORS = YES; CODE_SIGN_ENTITLEMENTS = "Clock Signal/Clock Signal.entitlements"; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; @@ -5164,7 +5163,6 @@ CLANG_WARN_DOCUMENTATION_COMMENTS = YES; CLANG_WARN_IMPLICIT_SIGN_CONVERSION = YES; CLANG_WARN_SUSPICIOUS_IMPLICIT_CONVERSION = YES; - CLANG_WARN__EXIT_TIME_DESTRUCTORS = YES; CODE_SIGN_ENTITLEMENTS = "Clock Signal/Clock Signal.entitlements"; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 9db6bb1a7..1410d6a4c 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -22,11 +22,39 @@ namespace Reflection { enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; \ constexpr static const char *__declaration##Name = #__VA_ARGS__; -#define EnumDeclaration(Name) __declaration##Name +#define EnumDeclaration(Name) #Name, __declaration##Name +#define AnnounceEnum(Name) ::Reflection::Enum::declare(EnumDeclaration(Name)) + +/*! + This provides a very slight version of enum reflection; you can introspect only: + + * enums have been registered, along with the text of their declarations; + * provided that those enums do not declare specific values for their members. + + The macros above help avoid duplication of the declaration, making this just mildly less + terrible than it might have been. + + No guarantees of speed or any other kind of efficiency are offered. +*/ class Enum { public: - template static void declare(const char *declaration) { + /*! + Registers @c name and the entries within @c declaration for the enum type @c Type. + + Assuming the caller used the macros above, a standard pattern where both things can be placed in + the same namespace might look like: + + ReflectableEnum(MyEnum, int, A, B, C); + + ... + + AnnounceEnum(MyEnum) + + If AnnounceEnum cannot be placed into the same namespace as ReflectableEnum, see the + EnumDeclaration macro. + */ + template static void declare(const char *name, const char *declaration) { const char *d_ptr = declaration; std::vector result; @@ -46,43 +74,77 @@ class Enum { members_by_type_.emplace(std::make_pair(&typeid(Type), result)); } + /*! + @returns the declared name of the enum @c Type if it has been registered; the empty string otherwise. + */ + template const std::string &name() { + return name(typeid(Type)); + } + + /*! + @returns the declared name of the enum with type_info @c type if it has been registered; the empty string otherwise. + */ + const std::string &name(const std::type_info &type) { + const auto entry = names_by_type_.find(&type); + if(entry == names_by_type_.end()) return empty_string_; + return entry->second; + } + + /*! + @returns the number of members of the enum @c Type if it has been registered; 0 otherwise. + */ template size_t size() { return size(typeid(Type)); } + /*! + @returns the number of members of the enum with type_info @c type if it has been registered; @c std::string::npos otherwise. + */ size_t size(const std::type_info &type) { const auto entry = members_by_type_.find(&type); - if(entry == members_by_type_.end()) return 0; + if(entry == members_by_type_.end()) return std::string::npos; return entry->second.size(); } /*! - @returns A @c string_view name for the enum value @c e. + @returns A @c std::string name for the enum value @c e. */ - template static const std::string &toString(EnumType e) { - const auto entry = members_by_type_.find(&typeid(EnumType)); - if(entry == members_by_type_.end()) return empty_string_; - return entry->second[size_t(e)]; + template static const std::string &toString(Type e) { + return toString(typeid(Type), size_t(e)); } /*! - @returns A value of @c EnumType name for the name @c str, or @c EnumType(-1) if - the string is not found. + @returns A @c std::string name for the enum value @c e from the enum with type_info @c type. + */ + static const std::string &toString(const std::type_info &type, size_t e) { + const auto entry = members_by_type_.find(&type); + if(entry == members_by_type_.end()) return empty_string_; + return entry->second[e]; + } + + /*! + @returns A value of @c Type for the name @c str, or @c EnumType(std::string::npos) if + the name is not found. */ template Type fromString(const std::string &str) { return Type(fromString(typeid(Type), str)); } + /*! + @returns A value for the name @c str in the enum with type_info @c type , or @c std::string::npos if + the name is not found. + */ size_t fromString(const std::type_info &type, const std::string &str) { const auto entry = members_by_type_.find(&type); - if(entry == members_by_type_.end()) return 0; + if(entry == members_by_type_.end()) return std::string::npos; const auto iterator = std::find(entry->second.begin(), entry->second.end(), str); - if(iterator == entry->second.end()) return 0; + if(iterator == entry->second.end()) return std::string::npos; return size_t(iterator - entry->second.begin()); } private: static inline std::unordered_map> members_by_type_; + static inline std::unordered_map names_by_type_; static inline const std::string empty_string_; }; diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 7e55d702a..27b7c28dd 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -19,24 +19,39 @@ namespace Reflection { template class Struct { public: + /*! + @returns the value of type @c Type that is loaded from the offset registered for the field @c name. + It is the caller's responsibility to provide an appropriate type of data. + */ template const Type *get(const std::string &name) { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; return reinterpret_cast(reinterpret_cast(this) + iterator->second.offset); } + /*! + Stores the @c value of type @c Type to the offset registered for the field @c name. + + It is the caller's responsibility to provide an appropriate type of data. + */ template void set(const std::string &name, const Type &value) { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return; *reinterpret_cast(reinterpret_cast(this) + iterator->second.offset) = value; } + /*! + @returns @c type_info for the field @c name. + */ const std::type_info *type_of(const std::string &name) { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; return iterator->second.type; } + /*! + @returns A vector of all declared fields for this struct. + */ std::vector all_keys() { std::vector keys; for(const auto &pair: contents_) { @@ -70,15 +85,7 @@ template class Struct { } /*! - Provides the original declaration of an enum. - */ - template void declare_enum(Type *t, const char *declaration) { - // TODO: something. - printf("%s\n", declaration); - } - - /*! - @returns @c true if this + @returns @c true if this subclass of @c Struct has not yet declared any fields. */ bool needs_declare() { return !contents_.size(); From fd052189cada68eabbf63fae39c1dd31e4f1ee40 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 11 Mar 2020 23:25:29 -0400 Subject: [PATCH 15/48] Adds reflection to all of the other computer targets. --- Analyser/Static/Acorn/Target.hpp | 10 +++++++++- Analyser/Static/AmstradCPC/Target.hpp | 18 +++++++++++------- Analyser/Static/AppleII/Target.hpp | 21 ++++++++++++++++----- Analyser/Static/Commodore/Target.hpp | 21 ++++++++++++++++++--- Analyser/Static/MSX/Target.hpp | 17 ++++++++++++++--- Analyser/Static/Macintosh/Target.hpp | 5 ++--- Analyser/Static/Oric/Target.hpp | 25 ++++++++++++++++++------- Analyser/Static/ZX8081/Target.hpp | 17 ++++++++++++++--- Machines/Apple/Macintosh/Macintosh.cpp | 3 --- Reflection/Enum.h | 2 +- Reflection/Struct.h | 2 ++ 11 files changed, 105 insertions(+), 36 deletions(-) diff --git a/Analyser/Static/Acorn/Target.hpp b/Analyser/Static/Acorn/Target.hpp index e2bfcb779..31d3e93aa 100644 --- a/Analyser/Static/Acorn/Target.hpp +++ b/Analyser/Static/Acorn/Target.hpp @@ -9,6 +9,7 @@ #ifndef Analyser_Static_Acorn_Target_h #define Analyser_Static_Acorn_Target_h +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" #include @@ -16,11 +17,18 @@ namespace Analyser { namespace Static { namespace Acorn { -struct Target: public ::Analyser::Static::Target { +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { bool has_adfs = false; bool has_dfs = false; bool should_shift_restart = false; std::string loading_command; + + Target() { + if(needs_declare()) { + DeclareField(has_adfs); + DeclareField(has_dfs); + } + } }; } diff --git a/Analyser/Static/AmstradCPC/Target.hpp b/Analyser/Static/AmstradCPC/Target.hpp index e30600d24..451389661 100644 --- a/Analyser/Static/AmstradCPC/Target.hpp +++ b/Analyser/Static/AmstradCPC/Target.hpp @@ -9,6 +9,8 @@ #ifndef Analyser_Static_AmstradCPC_Target_h #define Analyser_Static_AmstradCPC_Target_h +#include "../../../Reflection/Enum.h" +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" #include @@ -16,15 +18,17 @@ namespace Analyser { namespace Static { namespace AmstradCPC { -struct Target: public ::Analyser::Static::Target { - enum class Model { - CPC464, - CPC664, - CPC6128 - }; - +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { + ReflectableEnum(Model, int, CPC464, CPC664, CPC6128); Model model = Model::CPC464; std::string loading_command; + + Target() { + if(needs_declare()) { + DeclareField(model); + AnnounceEnum(Model); + } + } }; } diff --git a/Analyser/Static/AppleII/Target.hpp b/Analyser/Static/AppleII/Target.hpp index 3c97abfcb..04ee6caee 100644 --- a/Analyser/Static/AppleII/Target.hpp +++ b/Analyser/Static/AppleII/Target.hpp @@ -9,27 +9,38 @@ #ifndef Target_h #define Target_h +#include "../../../Reflection/Enum.h" +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" namespace Analyser { namespace Static { namespace AppleII { -struct Target: public ::Analyser::Static::Target { - enum class Model { +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { + ReflectableEnum(Model, int, II, IIplus, IIe, EnhancedIIe - }; - enum class DiskController { + ); + ReflectableEnum(DiskController, int, None, SixteenSector, ThirteenSector - }; + ); Model model = Model::IIe; DiskController disk_controller = DiskController::None; + + Target() { + if(needs_declare()) { + DeclareField(model); + DeclareField(disk_controller); + AnnounceEnum(Model); + AnnounceEnum(DiskController); + } + } }; } diff --git a/Analyser/Static/Commodore/Target.hpp b/Analyser/Static/Commodore/Target.hpp index 89ad4538e..59ec73d57 100644 --- a/Analyser/Static/Commodore/Target.hpp +++ b/Analyser/Static/Commodore/Target.hpp @@ -9,6 +9,8 @@ #ifndef Analyser_Static_Commodore_Target_h #define Analyser_Static_Commodore_Target_h +#include "../../../Reflection/Enum.h" +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" #include @@ -16,20 +18,20 @@ namespace Analyser { namespace Static { namespace Commodore { -struct Target: public ::Analyser::Static::Target { +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { enum class MemoryModel { Unexpanded, EightKB, ThirtyTwoKB }; - enum class Region { + ReflectableEnum(Region, int, American, Danish, Japanese, European, Swedish - }; + ); /// Maps from a named memory model to a bank enabled/disabled set. void set_memory_model(MemoryModel memory_model) { @@ -54,6 +56,19 @@ struct Target: public ::Analyser::Static::Target { Region region = Region::European; bool has_c1540 = false; std::string loading_command; + + Target() { + if(needs_declare()) { + DeclareField(enabled_ram.bank0); + DeclareField(enabled_ram.bank1); + DeclareField(enabled_ram.bank2); + DeclareField(enabled_ram.bank3); + DeclareField(enabled_ram.bank5); + DeclareField(region); + DeclareField(has_c1540); + AnnounceEnum(Region); + } + } }; } diff --git a/Analyser/Static/MSX/Target.hpp b/Analyser/Static/MSX/Target.hpp index fa4e379bc..14cd7b4ad 100644 --- a/Analyser/Static/MSX/Target.hpp +++ b/Analyser/Static/MSX/Target.hpp @@ -9,6 +9,8 @@ #ifndef Analyser_Static_MSX_Target_h #define Analyser_Static_MSX_Target_h +#include "../../../Reflection/Enum.h" +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" #include @@ -16,15 +18,24 @@ namespace Analyser { namespace Static { namespace MSX { -struct Target: public ::Analyser::Static::Target { +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { bool has_disk_drive = false; std::string loading_command; - enum class Region { + ReflectableEnum(Region, int, Japan, USA, Europe - } region = Region::USA; + ); + Region region = Region::USA; + + Target() { + if(needs_declare()) { + DeclareField(has_disk_drive); + DeclareField(region); + AnnounceEnum(Region); + } + } }; } diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 86295e31a..9170202d4 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -18,16 +18,15 @@ namespace Macintosh { struct Target: public ::Analyser::Static::Target, public Reflection::Struct { ReflectableEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); + Model model = Model::MacPlus; Target() { // Boilerplate for declaring fields and potential values. if(needs_declare()) { - declare(&model, "model"); + DeclareField(model); AnnounceEnum(Model); } } - - Model model = Model::MacPlus; }; } diff --git a/Analyser/Static/Oric/Target.hpp b/Analyser/Static/Oric/Target.hpp index 24a5d1371..b18525bda 100644 --- a/Analyser/Static/Oric/Target.hpp +++ b/Analyser/Static/Oric/Target.hpp @@ -9,6 +9,8 @@ #ifndef Analyser_Static_Oric_Target_h #define Analyser_Static_Oric_Target_h +#include "../../../Reflection/Enum.h" +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" #include @@ -16,25 +18,34 @@ namespace Analyser { namespace Static { namespace Oric { -struct Target: public ::Analyser::Static::Target { - enum class ROM { +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { + ReflectableEnum(ROM, int, BASIC10, BASIC11, Pravetz - }; + ); - enum class DiskInterface { + ReflectableEnum(DiskInterface, int, + None, Microdisc, Pravetz, Jasmin, - BD500, - None - }; + BD500 + ); ROM rom = ROM::BASIC11; DiskInterface disk_interface = DiskInterface::None; std::string loading_command; bool should_start_jasmin = false; + + Target() { + if(needs_declare()) { + DeclareField(rom); + DeclareField(disk_interface); + AnnounceEnum(ROM); + AnnounceEnum(DiskInterface); + } + } }; } diff --git a/Analyser/Static/ZX8081/Target.hpp b/Analyser/Static/ZX8081/Target.hpp index 5671ab1d6..2551a0f5d 100644 --- a/Analyser/Static/ZX8081/Target.hpp +++ b/Analyser/Static/ZX8081/Target.hpp @@ -9,6 +9,8 @@ #ifndef Analyser_Static_ZX8081_Target_h #define Analyser_Static_ZX8081_Target_h +#include "../../../Reflection/Enum.h" +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" #include @@ -16,17 +18,26 @@ namespace Analyser { namespace Static { namespace ZX8081 { -struct Target: public ::Analyser::Static::Target { - enum class MemoryModel { +struct Target: public ::Analyser::Static::Target, public Reflection::Struct { + ReflectableEnum(MemoryModel, int, Unexpanded, SixteenKB, SixtyFourKB - }; + ); MemoryModel memory_model = MemoryModel::Unexpanded; bool is_ZX81 = false; bool ZX80_uses_ZX81_ROM = false; std::string loading_command; + + Target() { + if(needs_declare()) { + DeclareField(memory_model); + DeclareField(is_ZX81); + DeclareField(ZX80_uses_ZX81_ROM); + AnnounceEnum(MemoryModel); + } + } }; } diff --git a/Machines/Apple/Macintosh/Macintosh.cpp b/Machines/Apple/Macintosh/Macintosh.cpp index f8b1306c4..b9ceedf4c 100644 --- a/Machines/Apple/Macintosh/Macintosh.cpp +++ b/Machines/Apple/Macintosh/Macintosh.cpp @@ -51,9 +51,6 @@ namespace { constexpr int CLOCK_RATE = 7833600; - -Analyser::Static::Macintosh::Target nothing; - } namespace Apple { diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 1410d6a4c..e47d1b5b6 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -19,7 +19,7 @@ namespace Reflection { #define ReflectableEnum(Name, Type, ...) \ - enum class Name: Type { Mac128k, Mac512k, Mac512ke, MacPlus }; \ + enum class Name: Type { __VA_ARGS__ }; \ constexpr static const char *__declaration##Name = #__VA_ARGS__; #define EnumDeclaration(Name) #Name, __declaration##Name diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 27b7c28dd..1d434781c 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -17,6 +17,8 @@ namespace Reflection { +#define DeclareField(Name) declare(&Name, #Name) + template class Struct { public: /*! From 724e2e6d272e4420978a72619e50e4beb507c9e9 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 11 Mar 2020 23:28:38 -0400 Subject: [PATCH 16/48] Withdraws ability to select an integer size for ReflectableEnums. It isn't that useful, and this'll help if/when I get to serialisation. --- Analyser/Static/AmstradCPC/Target.hpp | 2 +- Analyser/Static/AppleII/Target.hpp | 4 ++-- Analyser/Static/Commodore/Target.hpp | 2 +- Analyser/Static/MSX/Target.hpp | 2 +- Analyser/Static/Macintosh/Target.hpp | 2 +- Analyser/Static/Oric/Target.hpp | 4 ++-- Analyser/Static/ZX8081/Target.hpp | 2 +- Reflection/Enum.h | 4 ++-- 8 files changed, 11 insertions(+), 11 deletions(-) diff --git a/Analyser/Static/AmstradCPC/Target.hpp b/Analyser/Static/AmstradCPC/Target.hpp index 451389661..bb35a034d 100644 --- a/Analyser/Static/AmstradCPC/Target.hpp +++ b/Analyser/Static/AmstradCPC/Target.hpp @@ -19,7 +19,7 @@ namespace Static { namespace AmstradCPC { struct Target: public ::Analyser::Static::Target, public Reflection::Struct { - ReflectableEnum(Model, int, CPC464, CPC664, CPC6128); + ReflectableEnum(Model, CPC464, CPC664, CPC6128); Model model = Model::CPC464; std::string loading_command; diff --git a/Analyser/Static/AppleII/Target.hpp b/Analyser/Static/AppleII/Target.hpp index 04ee6caee..fb4409a47 100644 --- a/Analyser/Static/AppleII/Target.hpp +++ b/Analyser/Static/AppleII/Target.hpp @@ -18,13 +18,13 @@ namespace Static { namespace AppleII { struct Target: public ::Analyser::Static::Target, public Reflection::Struct { - ReflectableEnum(Model, int, + ReflectableEnum(Model, II, IIplus, IIe, EnhancedIIe ); - ReflectableEnum(DiskController, int, + ReflectableEnum(DiskController, None, SixteenSector, ThirteenSector diff --git a/Analyser/Static/Commodore/Target.hpp b/Analyser/Static/Commodore/Target.hpp index 59ec73d57..8655da9cc 100644 --- a/Analyser/Static/Commodore/Target.hpp +++ b/Analyser/Static/Commodore/Target.hpp @@ -25,7 +25,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::Struct { - ReflectableEnum(Model, int, Mac128k, Mac512k, Mac512ke, MacPlus); + ReflectableEnum(Model, Mac128k, Mac512k, Mac512ke, MacPlus); Model model = Model::MacPlus; Target() { diff --git a/Analyser/Static/Oric/Target.hpp b/Analyser/Static/Oric/Target.hpp index b18525bda..281c3b7f8 100644 --- a/Analyser/Static/Oric/Target.hpp +++ b/Analyser/Static/Oric/Target.hpp @@ -19,13 +19,13 @@ namespace Static { namespace Oric { struct Target: public ::Analyser::Static::Target, public Reflection::Struct { - ReflectableEnum(ROM, int, + ReflectableEnum(ROM, BASIC10, BASIC11, Pravetz ); - ReflectableEnum(DiskInterface, int, + ReflectableEnum(DiskInterface, None, Microdisc, Pravetz, diff --git a/Analyser/Static/ZX8081/Target.hpp b/Analyser/Static/ZX8081/Target.hpp index 2551a0f5d..e3c100d61 100644 --- a/Analyser/Static/ZX8081/Target.hpp +++ b/Analyser/Static/ZX8081/Target.hpp @@ -19,7 +19,7 @@ namespace Static { namespace ZX8081 { struct Target: public ::Analyser::Static::Target, public Reflection::Struct { - ReflectableEnum(MemoryModel, int, + ReflectableEnum(MemoryModel, Unexpanded, SixteenKB, SixtyFourKB diff --git a/Reflection/Enum.h b/Reflection/Enum.h index e47d1b5b6..0a9e86a9c 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -18,8 +18,8 @@ namespace Reflection { -#define ReflectableEnum(Name, Type, ...) \ - enum class Name: Type { __VA_ARGS__ }; \ +#define ReflectableEnum(Name, ...) \ + enum class Name { __VA_ARGS__ }; \ constexpr static const char *__declaration##Name = #__VA_ARGS__; #define EnumDeclaration(Name) #Name, __declaration##Name From 52f644c4f1950e1fb9989abb4137eb18f77601a1 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 12 Mar 2020 20:56:02 -0400 Subject: [PATCH 17/48] Ensures that reflection is completely blind; starts adding SDL instantiation logic. --- Analyser/Static/Acorn/Target.hpp | 2 +- Analyser/Static/AmstradCPC/Target.hpp | 2 +- Analyser/Static/AppleII/Target.hpp | 2 +- Analyser/Static/Commodore/Target.hpp | 2 +- Analyser/Static/MSX/Target.hpp | 2 +- Analyser/Static/Macintosh/Target.hpp | 2 +- Analyser/Static/Oric/Target.hpp | 2 +- Analyser/Static/ZX8081/Target.hpp | 2 +- Machines/Utility/MachineForTarget.cpp | 32 ++++++++++++++++ Machines/Utility/MachineForTarget.hpp | 4 ++ .../xcschemes/Clock Signal Kiosk.xcscheme | 8 +++- OSBindings/SDL/main.cpp | 31 ++++++++++++++- Reflection/Enum.h | 38 ++++++++++++++----- Reflection/Struct.h | 30 +++++++++------ 14 files changed, 127 insertions(+), 32 deletions(-) diff --git a/Analyser/Static/Acorn/Target.hpp b/Analyser/Static/Acorn/Target.hpp index 31d3e93aa..7c312e1fd 100644 --- a/Analyser/Static/Acorn/Target.hpp +++ b/Analyser/Static/Acorn/Target.hpp @@ -17,7 +17,7 @@ namespace Analyser { namespace Static { namespace Acorn { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { bool has_adfs = false; bool has_dfs = false; bool should_shift_restart = false; diff --git a/Analyser/Static/AmstradCPC/Target.hpp b/Analyser/Static/AmstradCPC/Target.hpp index bb35a034d..765b2f04f 100644 --- a/Analyser/Static/AmstradCPC/Target.hpp +++ b/Analyser/Static/AmstradCPC/Target.hpp @@ -18,7 +18,7 @@ namespace Analyser { namespace Static { namespace AmstradCPC { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(Model, CPC464, CPC664, CPC6128); Model model = Model::CPC464; std::string loading_command; diff --git a/Analyser/Static/AppleII/Target.hpp b/Analyser/Static/AppleII/Target.hpp index fb4409a47..e1a6141cc 100644 --- a/Analyser/Static/AppleII/Target.hpp +++ b/Analyser/Static/AppleII/Target.hpp @@ -17,7 +17,7 @@ namespace Analyser { namespace Static { namespace AppleII { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(Model, II, IIplus, diff --git a/Analyser/Static/Commodore/Target.hpp b/Analyser/Static/Commodore/Target.hpp index 8655da9cc..d207cf0bd 100644 --- a/Analyser/Static/Commodore/Target.hpp +++ b/Analyser/Static/Commodore/Target.hpp @@ -18,7 +18,7 @@ namespace Analyser { namespace Static { namespace Commodore { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { enum class MemoryModel { Unexpanded, EightKB, diff --git a/Analyser/Static/MSX/Target.hpp b/Analyser/Static/MSX/Target.hpp index 685a3dc7a..f7c7ddbac 100644 --- a/Analyser/Static/MSX/Target.hpp +++ b/Analyser/Static/MSX/Target.hpp @@ -18,7 +18,7 @@ namespace Analyser { namespace Static { namespace MSX { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { bool has_disk_drive = false; std::string loading_command; diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index e4a588d2e..9b8aab7e7 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -16,7 +16,7 @@ namespace Analyser { namespace Static { namespace Macintosh { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(Model, Mac128k, Mac512k, Mac512ke, MacPlus); Model model = Model::MacPlus; diff --git a/Analyser/Static/Oric/Target.hpp b/Analyser/Static/Oric/Target.hpp index 281c3b7f8..114b6a4a3 100644 --- a/Analyser/Static/Oric/Target.hpp +++ b/Analyser/Static/Oric/Target.hpp @@ -18,7 +18,7 @@ namespace Analyser { namespace Static { namespace Oric { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(ROM, BASIC10, BASIC11, diff --git a/Analyser/Static/ZX8081/Target.hpp b/Analyser/Static/ZX8081/Target.hpp index e3c100d61..b85e2cce8 100644 --- a/Analyser/Static/ZX8081/Target.hpp +++ b/Analyser/Static/ZX8081/Target.hpp @@ -18,7 +18,7 @@ namespace Analyser { namespace Static { namespace ZX8081 { -struct Target: public ::Analyser::Static::Target, public Reflection::Struct { +struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(MemoryModel, Unexpanded, SixteenKB, diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index ece5a9f02..721ea68e8 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -8,6 +8,7 @@ #include "MachineForTarget.hpp" +// Sources for runtime options. #include "../AmstradCPC/AmstradCPC.hpp" #include "../Apple/AppleII/AppleII.hpp" #include "../Apple/Macintosh/Macintosh.hpp" @@ -21,6 +22,17 @@ #include "../Oric/Oric.hpp" #include "../ZX8081/ZX8081.hpp" +// Sources for construction options. +#include "../../Analyser/Static/Acorn/Target.hpp" +#include "../../Analyser/Static/AmstradCPC/Target.hpp" +#include "../../Analyser/Static/AppleII/Target.hpp" +#include "../../Analyser/Static/AtariST/Target.hpp" +#include "../../Analyser/Static/Commodore/Target.hpp" +#include "../../Analyser/Static/Macintosh/Target.hpp" +#include "../../Analyser/Static/MSX/Target.hpp" +#include "../../Analyser/Static/Oric/Target.hpp" +#include "../../Analyser/Static/ZX8081/Target.hpp" + #include "../../Analyser/Dynamic/MultiMachine/MultiMachine.hpp" #include "TypedDynamicMachine.hpp" @@ -155,3 +167,23 @@ std::map>> Machin return options; } + +std::map> Machine::ConstructionOptionsByMachineName() { + std::map> options; + +#define Add(Name) \ + options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Name), new Analyser::Static::Name::Target)); + + Add(AmstradCPC); + Add(AppleII); +// Add(AtariST); +// Add(Electron); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Electron), Electron::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); + Add(MSX); + Add(Oric); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); + Add(ZX8081); + + return options; +} diff --git a/Machines/Utility/MachineForTarget.hpp b/Machines/Utility/MachineForTarget.hpp index 0b156c7c7..6cc29bcae 100644 --- a/Machines/Utility/MachineForTarget.hpp +++ b/Machines/Utility/MachineForTarget.hpp @@ -10,10 +10,12 @@ #define MachineForTarget_hpp #include "../../Analyser/Static/StaticAnalyser.hpp" +#include "../../Reflection/Struct.h" #include "../DynamicMachine.hpp" #include +#include #include #include @@ -53,6 +55,8 @@ std::string LongNameForTargetMachine(const Analyser::Machine target); */ std::map>> AllOptionsByMachineName(); +std::map> ConstructionOptionsByMachineName(); + } #endif /* MachineForTarget_hpp */ diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index f84023923..5a6721911 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -31,7 +31,7 @@ + isEnabled = "NO"> + + diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index 13a566feb..04a898a83 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -34,6 +34,9 @@ #include "../../Outputs/OpenGL/ScanTarget.hpp" #include "../../Outputs/OpenGL/Screenshot.hpp" +#include "../../Reflection/Enum.h" +#include "../../Reflection/Struct.h" + namespace { struct MachineRunner { @@ -478,7 +481,7 @@ int main(int argc, char *argv[]) { std::cout << "Use alt+enter to toggle full screen display. Use control+shift+V to paste text." << std::endl; std::cout << "Required machine type and configuration is determined from the file. Machines with further options:" << std::endl << std::endl; - auto all_options = Machine::AllOptionsByMachineName(); + const auto all_options = Machine::AllOptionsByMachineName(); for(const auto &machine_options: all_options) { std::cout << machine_options.first << ":" << std::endl; for(const auto &option: machine_options.second) { @@ -499,6 +502,32 @@ int main(int argc, char *argv[]) { } std::cout << std::endl; } + + const auto construction_options = Machine::ConstructionOptionsByMachineName(); + for(const auto &options: construction_options) { + std::cout << options.first << ":" << std::endl; + for(const auto &option: options.second->all_keys()) { + std::cout << '\t' << "--" << option; + const auto type = options.second->type_of(option); + + // Is this a registered enum? If so, list options. + if(!Reflection::Enum::name(*type).empty()) { + std::cout << "={"; + bool is_first = true; + for(const auto &value: Reflection::Enum::all_values(*type)) { + if(!is_first) std::cout << '|'; + is_first = false; + std::cout << value; + } + std::cout << "}"; + } + + // TODO: if not a registered enum... then assume it was a Boolean? + std::cout << std::endl; + } + + std::cout << std::endl; + } return EXIT_SUCCESS; } diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 0a9e86a9c..85e38b7c9 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -72,19 +72,20 @@ class Enum { } members_by_type_.emplace(std::make_pair(&typeid(Type), result)); + names_by_type_.emplace(std::make_pair(&typeid(Type), std::string(name))); } /*! @returns the declared name of the enum @c Type if it has been registered; the empty string otherwise. */ - template const std::string &name() { + template static const std::string &name() { return name(typeid(Type)); } /*! @returns the declared name of the enum with type_info @c type if it has been registered; the empty string otherwise. */ - const std::string &name(const std::type_info &type) { + static const std::string &name(const std::type_info &type) { const auto entry = names_by_type_.find(&type); if(entry == names_by_type_.end()) return empty_string_; return entry->second; @@ -93,14 +94,14 @@ class Enum { /*! @returns the number of members of the enum @c Type if it has been registered; 0 otherwise. */ - template size_t size() { + template static size_t size() { return size(typeid(Type)); } /*! @returns the number of members of the enum with type_info @c type if it has been registered; @c std::string::npos otherwise. */ - size_t size(const std::type_info &type) { + static size_t size(const std::type_info &type) { const auto entry = members_by_type_.find(&type); if(entry == members_by_type_.end()) return std::string::npos; return entry->second.size(); @@ -109,32 +110,48 @@ class Enum { /*! @returns A @c std::string name for the enum value @c e. */ - template static const std::string &toString(Type e) { - return toString(typeid(Type), size_t(e)); + template static const std::string &to_string(Type e) { + return to_string(typeid(Type), size_t(e)); } /*! @returns A @c std::string name for the enum value @c e from the enum with type_info @c type. */ - static const std::string &toString(const std::type_info &type, size_t e) { + static const std::string &to_string(const std::type_info &type, size_t e) { const auto entry = members_by_type_.find(&type); if(entry == members_by_type_.end()) return empty_string_; return entry->second[e]; } + /*! + @returns a vector naming the members of the enum with type_info @c type if it has been registered; an empty vector otherwise. + */ + static const std::vector &all_values(const std::type_info &type) { + const auto entry = members_by_type_.find(&type); + if(entry == members_by_type_.end()) return empty_vector_; + return entry->second; + } + + /*! + @returns a vector naming the members of the enum @c Type type if it has been registered; an empty vector otherwise. + */ + template static const std::vector &all_values() { + return all_values(typeid(Type)); + } + /*! @returns A value of @c Type for the name @c str, or @c EnumType(std::string::npos) if the name is not found. */ - template Type fromString(const std::string &str) { - return Type(fromString(typeid(Type), str)); + template static Type from_string(const std::string &str) { + return Type(from_string(typeid(Type), str)); } /*! @returns A value for the name @c str in the enum with type_info @c type , or @c std::string::npos if the name is not found. */ - size_t fromString(const std::type_info &type, const std::string &str) { + static size_t from_string(const std::type_info &type, const std::string &str) { const auto entry = members_by_type_.find(&type); if(entry == members_by_type_.end()) return std::string::npos; const auto iterator = std::find(entry->second.begin(), entry->second.end(), str); @@ -146,6 +163,7 @@ class Enum { static inline std::unordered_map> members_by_type_; static inline std::unordered_map names_by_type_; static inline const std::string empty_string_; + static inline const std::vector empty_vector_; }; } diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 1d434781c..86772b010 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -19,16 +19,24 @@ namespace Reflection { #define DeclareField(Name) declare(&Name, #Name) -template class Struct { +struct Struct { + virtual std::vector all_keys() = 0; + virtual const std::type_info *type_of(const std::string &name) = 0; + virtual void set(const std::string &name, const void *value) = 0; + virtual const void *get(const std::string &name) = 0; + virtual ~Struct() {} +}; + +template class StructImpl: public Struct { public: /*! @returns the value of type @c Type that is loaded from the offset registered for the field @c name. It is the caller's responsibility to provide an appropriate type of data. */ - template const Type *get(const std::string &name) { + const void *get(const std::string &name) final { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; - return reinterpret_cast(reinterpret_cast(this) + iterator->second.offset); + return reinterpret_cast(this) + iterator->second.offset; } /*! @@ -36,16 +44,16 @@ template class Struct { It is the caller's responsibility to provide an appropriate type of data. */ - template void set(const std::string &name, const Type &value) { + void set(const std::string &name, const void *value) final { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return; - *reinterpret_cast(reinterpret_cast(this) + iterator->second.offset) = value; + memcpy(reinterpret_cast(this) + iterator->second.offset, value, iterator->second.size); } /*! @returns @c type_info for the field @c name. */ - const std::type_info *type_of(const std::string &name) { + const std::type_info *type_of(const std::string &name) final { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; return iterator->second.type; @@ -54,7 +62,7 @@ template class Struct { /*! @returns A vector of all declared fields for this struct. */ - std::vector all_keys() { + std::vector all_keys() final { std::vector keys; for(const auto &pair: contents_) { keys.push_back(pair.first); @@ -82,7 +90,7 @@ template class Struct { contents_.emplace( std::make_pair( name, - Field(typeid(Type), reinterpret_cast(t) - reinterpret_cast(this)) + Field(typeid(Type), reinterpret_cast(t) - reinterpret_cast(this), sizeof(Type)) )); } @@ -96,9 +104,9 @@ template class Struct { private: struct Field { const std::type_info *type; - ssize_t offset; - Field(const std::type_info &type, ssize_t offset) : - type(&type), offset(offset) {} + ssize_t offset, size; + Field(const std::type_info &type, ssize_t offset, size_t size) : + type(&type), offset(offset), size(size) {} }; static inline std::unordered_map contents_; }; From f27e0a141dbe07a8bcc8a13b558685b50b56de4c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 13 Mar 2020 20:16:36 -0400 Subject: [PATCH 18/48] Sketches but doesn't implement an interface for serialisation. --- Reflection/Struct.h | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 86772b010..a040761a7 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -27,7 +27,15 @@ struct Struct { virtual ~Struct() {} }; -template class StructImpl: public Struct { +struct Serialisable { + /// Serialises this object, appending it to @c target. + virtual void serialise(std::vector &target) = 0; + /// Deserialises this object from @c source. + /// @returns @c true if the deserialisation was successful; @c false otherwise. + virtual bool deserialise(const std::vector &source) = 0; +}; + +template class StructImpl: public Struct, public Serialisable { public: /*! @returns the value of type @c Type that is loaded from the offset registered for the field @c name. From 1a2872c815f0c7ed1f702b004b69ec2b4fa545fc Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 13 Mar 2020 22:42:37 -0400 Subject: [PATCH 19/48] Starts to build an easy `set` interface. --- Machines/Apple/Macintosh/Macintosh.cpp | 3 - OSBindings/SDL/main.cpp | 3 + Reflection/Struct.h | 78 ++++++++++++++++++++++++-- 3 files changed, 76 insertions(+), 8 deletions(-) diff --git a/Machines/Apple/Macintosh/Macintosh.cpp b/Machines/Apple/Macintosh/Macintosh.cpp index f8b1306c4..b9ceedf4c 100644 --- a/Machines/Apple/Macintosh/Macintosh.cpp +++ b/Machines/Apple/Macintosh/Macintosh.cpp @@ -51,9 +51,6 @@ namespace { constexpr int CLOCK_RATE = 7833600; - -Analyser::Static::Macintosh::Target nothing; - } namespace Apple { diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index 04a898a83..f7eaa114a 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -39,6 +39,9 @@ namespace { +/// Takes an enum-style camel-case string (e.g. Mac128k) and +//std::string + struct MachineRunner { MachineRunner() { frame_lock_.clear(); diff --git a/Reflection/Struct.h b/Reflection/Struct.h index a040761a7..058de8b13 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -15,6 +15,8 @@ #include #include +#include "Enum.h" + namespace Reflection { #define DeclareField(Name) declare(&Name, #Name) @@ -22,9 +24,75 @@ namespace Reflection { struct Struct { virtual std::vector all_keys() = 0; virtual const std::type_info *type_of(const std::string &name) = 0; - virtual void set(const std::string &name, const void *value) = 0; - virtual const void *get(const std::string &name) = 0; + virtual void set_raw(const std::string &name, const void *value) = 0; + virtual const void *get_raw(const std::string &name) = 0; virtual ~Struct() {} + + /*! + Attempts to set the property @c name to @c value ; will perform limited type conversions. + + @returns @c true if t + */ + template bool set(const std::string &name, Type value) { + return false; + } + + /*! + Setting an int: + + * to an int copies the int; + * to an int64_t promotes the int; and + * to a registered enum, copies the int. + */ + template <> bool set(const std::string &name, int value) { + const auto target_type = type_of(name); + if(!target_type) return false; + + // No need to convert an int or a registered enum. + if(*target_type == typeid(int) || !Reflection::Enum::name(*target_type).empty()) { + set_raw(name, &value); + return true; + } + + // Promote to an int64_t. + if(*target_type == typeid(int64_t)) { + const auto int64 = int64_t(value); + set_raw(name, &int64); + return true; + } + + return false; + } + + /*! + Setting a string: + + * to an enum, if the string names a member of the enum, sets the value. + */ + template <> bool set(const std::string &name, const std::string &value) { + const auto target_type = type_of(name); + if(!target_type) return false; + + if(Reflection::Enum::name(*target_type).empty()) { + return false; + } + + const auto enum_value = Reflection::Enum::from_string(*target_type, value); + if(enum_value == std::string::npos) { + return false; + } + + int int_value = int(enum_value); + set_raw(name, &int_value); + + return true; + } + + template <> bool set(const std::string &name, const char *value) { + const std::string string(value); + return set(name, string); + } + }; struct Serialisable { @@ -35,13 +103,13 @@ struct Serialisable { virtual bool deserialise(const std::vector &source) = 0; }; -template class StructImpl: public Struct, public Serialisable { +template class StructImpl: public Struct { public: /*! @returns the value of type @c Type that is loaded from the offset registered for the field @c name. It is the caller's responsibility to provide an appropriate type of data. */ - const void *get(const std::string &name) final { + const void *get_raw(const std::string &name) final { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; return reinterpret_cast(this) + iterator->second.offset; @@ -52,7 +120,7 @@ template class StructImpl: public Struct, public Serialisable { It is the caller's responsibility to provide an appropriate type of data. */ - void set(const std::string &name, const void *value) final { + void set_raw(const std::string &name, const void *value) final { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return; memcpy(reinterpret_cast(this) + iterator->second.offset, value, iterator->second.size); From cab4bead726186427c8faef79a4c0642b99f916a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Fri, 13 Mar 2020 23:38:29 -0400 Subject: [PATCH 20/48] Promotes explicit specialisations to namespace scope. --- .../Clock Signal.xcodeproj/project.pbxproj | 6 ++ OSBindings/SDL/SConstruct | 2 + Reflection/Enum.h | 28 +++--- Reflection/Struct.cpp | 53 ++++++++++ Reflection/Struct.h | 98 ++++++------------- 5 files changed, 103 insertions(+), 84 deletions(-) create mode 100644 Reflection/Struct.cpp diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index ee4db8516..377c50181 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -198,6 +198,8 @@ 4B4518A31F75FD1C00926311 /* HFE.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B4518951F75FD1B00926311 /* HFE.cpp */; }; 4B4518A41F75FD1C00926311 /* OricMFMDSK.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B4518971F75FD1B00926311 /* OricMFMDSK.cpp */; }; 4B4518A51F75FD1C00926311 /* SSD.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B4518991F75FD1B00926311 /* SSD.cpp */; }; + 4B47F6C5241C87A100ED06F7 /* Struct.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B47F6C4241C87A100ED06F7 /* Struct.cpp */; }; + 4B47F6C6241C87A100ED06F7 /* Struct.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B47F6C4241C87A100ED06F7 /* Struct.cpp */; }; 4B49F0A923346F7A0045E6A6 /* MacintoshOptions.xib in Resources */ = {isa = PBXBuildFile; fileRef = 4B49F0A723346F7A0045E6A6 /* MacintoshOptions.xib */; }; 4B4A76301DB1A3FA007AAE2E /* AY38910.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B4A762E1DB1A3FA007AAE2E /* AY38910.cpp */; }; 4B4B1A3C200198CA00A0F866 /* KonamiSCC.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B4B1A3A200198C900A0F866 /* KonamiSCC.cpp */; }; @@ -1095,6 +1097,7 @@ 4B45189A1F75FD1B00926311 /* SSD.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = SSD.hpp; sourceTree = ""; }; 4B4518A71F76004200926311 /* TapeParser.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = TapeParser.hpp; path = Parsers/TapeParser.hpp; sourceTree = ""; }; 4B4518A81F76022000926311 /* DiskImageImplementation.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; path = DiskImageImplementation.hpp; sourceTree = ""; }; + 4B47F6C4241C87A100ED06F7 /* Struct.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = Struct.cpp; sourceTree = ""; }; 4B49F0A823346F7A0045E6A6 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.xib; name = Base; path = "Clock Signal/Base.lproj/MacintoshOptions.xib"; sourceTree = SOURCE_ROOT; }; 4B4A762E1DB1A3FA007AAE2E /* AY38910.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; name = AY38910.cpp; path = AY38910/AY38910.cpp; sourceTree = ""; }; 4B4A762F1DB1A3FA007AAE2E /* AY38910.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = AY38910.hpp; path = AY38910/AY38910.hpp; sourceTree = ""; }; @@ -2201,6 +2204,7 @@ children = ( 4B3AF7D02413470E00873C0B /* Enum.h */, 4B3AF7D12413472200873C0B /* Struct.h */, + 4B47F6C4241C87A100ED06F7 /* Struct.cpp */, ); name = Reflection; path = ../../Reflection; @@ -4421,6 +4425,7 @@ 4B0ACC3323775819008902D0 /* Atari2600.cpp in Sources */, 4BD424E02193B5340097291A /* TextureTarget.cpp in Sources */, 4B055AB51FAE860F0060FFFF /* TapePRG.cpp in Sources */, + 4B47F6C6241C87A100ED06F7 /* Struct.cpp in Sources */, 4B055AE01FAE9B660060FFFF /* CRT.cpp in Sources */, 4B894527201967B4007DE474 /* StaticAnalyser.cpp in Sources */, 4BB244D622AABAF600BE20E5 /* z8530.cpp in Sources */, @@ -4580,6 +4585,7 @@ 4BA0F68E1EEA0E8400E9489E /* ZX8081.cpp in Sources */, 4BD468F71D8DF41D0084958B /* 1770.cpp in Sources */, 4B7F1897215486A200388727 /* StaticAnalyser.cpp in Sources */, + 4B47F6C5241C87A100ED06F7 /* Struct.cpp in Sources */, 4BD3A30B1EE755C800B5B501 /* Video.cpp in Sources */, 4B5FADBA1DE3151600AEC565 /* FileHolder.cpp in Sources */, 4B643F3A1D77AD1900D431D6 /* CSStaticAnalyser.mm in Sources */, diff --git a/OSBindings/SDL/SConstruct b/OSBindings/SDL/SConstruct index 93d84ede7..3889c723d 100644 --- a/OSBindings/SDL/SConstruct +++ b/OSBindings/SDL/SConstruct @@ -85,6 +85,8 @@ SOURCES += glob.glob('../../Processors/6502/Implementation/*.cpp') SOURCES += glob.glob('../../Processors/68000/Implementation/*.cpp') SOURCES += glob.glob('../../Processors/Z80/Implementation/*.cpp') +SOURCES += glob.glob('../../Reflection/*.cpp') + SOURCES += glob.glob('../../SignalProcessing/*.cpp') SOURCES += glob.glob('../../Storage/*.cpp') diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 85e38b7c9..f91106c81 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -71,8 +71,8 @@ class Enum { result.emplace_back(std::string(start, size_t(d_ptr - start))); } - members_by_type_.emplace(std::make_pair(&typeid(Type), result)); - names_by_type_.emplace(std::make_pair(&typeid(Type), std::string(name))); + members_by_type_.emplace(std::make_pair(std::type_index(typeid(Type)), result)); + names_by_type_.emplace(std::make_pair(std::type_index(typeid(Type)), std::string(name))); } /*! @@ -85,8 +85,8 @@ class Enum { /*! @returns the declared name of the enum with type_info @c type if it has been registered; the empty string otherwise. */ - static const std::string &name(const std::type_info &type) { - const auto entry = names_by_type_.find(&type); + static const std::string &name(std::type_index type) { + const auto entry = names_by_type_.find(type); if(entry == names_by_type_.end()) return empty_string_; return entry->second; } @@ -101,8 +101,8 @@ class Enum { /*! @returns the number of members of the enum with type_info @c type if it has been registered; @c std::string::npos otherwise. */ - static size_t size(const std::type_info &type) { - const auto entry = members_by_type_.find(&type); + static size_t size(std::type_index type) { + const auto entry = members_by_type_.find(type); if(entry == members_by_type_.end()) return std::string::npos; return entry->second.size(); } @@ -117,8 +117,8 @@ class Enum { /*! @returns A @c std::string name for the enum value @c e from the enum with type_info @c type. */ - static const std::string &to_string(const std::type_info &type, size_t e) { - const auto entry = members_by_type_.find(&type); + static const std::string &to_string(std::type_index type, size_t e) { + const auto entry = members_by_type_.find(type); if(entry == members_by_type_.end()) return empty_string_; return entry->second[e]; } @@ -126,8 +126,8 @@ class Enum { /*! @returns a vector naming the members of the enum with type_info @c type if it has been registered; an empty vector otherwise. */ - static const std::vector &all_values(const std::type_info &type) { - const auto entry = members_by_type_.find(&type); + static const std::vector &all_values(std::type_index type) { + const auto entry = members_by_type_.find(type); if(entry == members_by_type_.end()) return empty_vector_; return entry->second; } @@ -151,8 +151,8 @@ class Enum { @returns A value for the name @c str in the enum with type_info @c type , or @c std::string::npos if the name is not found. */ - static size_t from_string(const std::type_info &type, const std::string &str) { - const auto entry = members_by_type_.find(&type); + static size_t from_string(std::type_index type, const std::string &str) { + const auto entry = members_by_type_.find(type); if(entry == members_by_type_.end()) return std::string::npos; const auto iterator = std::find(entry->second.begin(), entry->second.end(), str); if(iterator == entry->second.end()) return std::string::npos; @@ -160,8 +160,8 @@ class Enum { } private: - static inline std::unordered_map> members_by_type_; - static inline std::unordered_map names_by_type_; + static inline std::unordered_map> members_by_type_; + static inline std::unordered_map names_by_type_; static inline const std::string empty_string_; static inline const std::vector empty_vector_; }; diff --git a/Reflection/Struct.cpp b/Reflection/Struct.cpp new file mode 100644 index 000000000..fbf92ab09 --- /dev/null +++ b/Reflection/Struct.cpp @@ -0,0 +1,53 @@ +// +// Struct.cpp +// Clock Signal +// +// Created by Thomas Harte on 13/03/2020. +// Copyright © 2020 Thomas Harte. All rights reserved. +// + +#include "Struct.h" + +template <> bool Reflection::set(Struct &target, const std::string &name, int value) { + const auto target_type = target.type_of(name); + if(!target_type) return false; + + // No need to convert an int or a registered enum. + if(*target_type == typeid(int) || !Reflection::Enum::name(*target_type).empty()) { + target.set(name, &value); + return true; + } + + // Promote to an int64_t. + if(*target_type == typeid(int64_t)) { + const auto int64 = int64_t(value); + target.set(name, &int64); + return true; + } + + return false; +} + +template <> bool Reflection::set(Struct &target, const std::string &name, const std::string &value) { + const auto target_type = target.type_of(name); + if(!target_type) return false; + + if(Reflection::Enum::name(*target_type).empty()) { + return false; + } + + const auto enum_value = Reflection::Enum::from_string(*target_type, value); + if(enum_value == std::string::npos) { + return false; + } + + int int_value = int(enum_value); + target.set(name, &int_value); + + return true; +} + +template <> bool Reflection::set(Struct &target, const std::string &name, const char *value) { + const std::string string(value); + return set(target, name, string); +} diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 058de8b13..3ceae6155 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -24,77 +24,35 @@ namespace Reflection { struct Struct { virtual std::vector all_keys() = 0; virtual const std::type_info *type_of(const std::string &name) = 0; - virtual void set_raw(const std::string &name, const void *value) = 0; - virtual const void *get_raw(const std::string &name) = 0; + virtual void set(const std::string &name, const void *value) = 0; + virtual const void *get(const std::string &name) = 0; virtual ~Struct() {} - - /*! - Attempts to set the property @c name to @c value ; will perform limited type conversions. - - @returns @c true if t - */ - template bool set(const std::string &name, Type value) { - return false; - } - - /*! - Setting an int: - - * to an int copies the int; - * to an int64_t promotes the int; and - * to a registered enum, copies the int. - */ - template <> bool set(const std::string &name, int value) { - const auto target_type = type_of(name); - if(!target_type) return false; - - // No need to convert an int or a registered enum. - if(*target_type == typeid(int) || !Reflection::Enum::name(*target_type).empty()) { - set_raw(name, &value); - return true; - } - - // Promote to an int64_t. - if(*target_type == typeid(int64_t)) { - const auto int64 = int64_t(value); - set_raw(name, &int64); - return true; - } - - return false; - } - - /*! - Setting a string: - - * to an enum, if the string names a member of the enum, sets the value. - */ - template <> bool set(const std::string &name, const std::string &value) { - const auto target_type = type_of(name); - if(!target_type) return false; - - if(Reflection::Enum::name(*target_type).empty()) { - return false; - } - - const auto enum_value = Reflection::Enum::from_string(*target_type, value); - if(enum_value == std::string::npos) { - return false; - } - - int int_value = int(enum_value); - set_raw(name, &int_value); - - return true; - } - - template <> bool set(const std::string &name, const char *value) { - const std::string string(value); - return set(name, string); - } - }; +/*! + Attempts to set the property @c name to @c value ; will perform limited type conversions. + + @returns @c true if t +*/ +template bool set(Struct &target, const std::string &name, Type value); + +/*! + Setting an int: + + * to an int copies the int; + * to an int64_t promotes the int; and + * to a registered enum, copies the int. +*/ +template <> bool set(Struct &target, const std::string &name, int value); + +/*! + Setting a string: + + * to an enum, if the string names a member of the enum, sets the value. +*/ +template <> bool set(Struct &target, const std::string &name, const std::string &value); +template <> bool set(Struct &target, const std::string &name, const char *value); + struct Serialisable { /// Serialises this object, appending it to @c target. virtual void serialise(std::vector &target) = 0; @@ -109,7 +67,7 @@ template class StructImpl: public Struct { @returns the value of type @c Type that is loaded from the offset registered for the field @c name. It is the caller's responsibility to provide an appropriate type of data. */ - const void *get_raw(const std::string &name) final { + const void *get(const std::string &name) final { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return nullptr; return reinterpret_cast(this) + iterator->second.offset; @@ -120,7 +78,7 @@ template class StructImpl: public Struct { It is the caller's responsibility to provide an appropriate type of data. */ - void set_raw(const std::string &name, const void *value) final { + void set(const std::string &name, const void *value) final { const auto iterator = contents_.find(name); if(iterator == contents_.end()) return; memcpy(reinterpret_cast(this) + iterator->second.offset, value, iterator->second.size); From a59963b6a09503701c6f9a9de90ce9f6df762053 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 14 Mar 2020 00:17:58 -0400 Subject: [PATCH 21/48] Adds necessary header for memcpy. --- Reflection/Struct.h | 1 + 1 file changed, 1 insertion(+) diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 3ceae6155..65adec45a 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -9,6 +9,7 @@ #ifndef Struct_h #define Struct_h +#include #include #include #include From 36acc2ddddb80e28fd78f486519d61353825fd80 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sat, 14 Mar 2020 00:22:23 -0400 Subject: [PATCH 22/48] Add necessary include for std::find. --- Reflection/Enum.h | 1 + 1 file changed, 1 insertion(+) diff --git a/Reflection/Enum.h b/Reflection/Enum.h index f91106c81..890032062 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -9,6 +9,7 @@ #ifndef Enum_h #define Enum_h +#include #include #include #include From f9c8470b20bea47ce6d6681b86be675b96419c02 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 15 Mar 2020 00:13:38 -0400 Subject: [PATCH 23/48] Ensure targets always nominate a machine. --- Analyser/Static/Acorn/StaticAnalyser.cpp | 1 - Analyser/Static/Acorn/Target.hpp | 2 +- Analyser/Static/AmstradCPC/StaticAnalyser.cpp | 1 - Analyser/Static/AmstradCPC/Target.hpp | 4 ++-- Analyser/Static/AppleII/StaticAnalyser.cpp | 1 - Analyser/Static/AppleII/Target.hpp | 4 ++-- Analyser/Static/Atari2600/StaticAnalyser.cpp | 1 - Analyser/Static/Atari2600/Target.hpp | 2 ++ Analyser/Static/AtariST/StaticAnalyser.cpp | 5 ++--- Analyser/Static/AtariST/Target.hpp | 6 +++++- Analyser/Static/Coleco/StaticAnalyser.cpp | 3 +-- Analyser/Static/Commodore/Target.hpp | 4 ++-- Analyser/Static/DiskII/StaticAnalyser.cpp | 2 -- Analyser/Static/MSX/StaticAnalyser.cpp | 2 -- Analyser/Static/MSX/Target.hpp | 2 +- Analyser/Static/Macintosh/StaticAnalyser.cpp | 1 - Analyser/Static/Macintosh/Target.hpp | 4 ++-- Analyser/Static/Oric/StaticAnalyser.cpp | 1 - Analyser/Static/Oric/Target.hpp | 4 ++-- Analyser/Static/Sega/StaticAnalyser.cpp | 2 -- Analyser/Static/Sega/Target.hpp | 6 +++++- Analyser/Static/StaticAnalyser.hpp | 3 ++- Analyser/Static/ZX8081/Target.hpp | 2 +- .../xcschemes/Clock Signal Kiosk.xcscheme | 4 ++-- .../Machine/StaticAnalyser/CSStaticAnalyser.mm | 13 ++----------- Reflection/Struct.h | 3 ++- 26 files changed, 36 insertions(+), 47 deletions(-) diff --git a/Analyser/Static/Acorn/StaticAnalyser.cpp b/Analyser/Static/Acorn/StaticAnalyser.cpp index 252176f56..75cafb509 100644 --- a/Analyser/Static/Acorn/StaticAnalyser.cpp +++ b/Analyser/Static/Acorn/StaticAnalyser.cpp @@ -59,7 +59,6 @@ static std::vector> Analyser::Static::TargetList Analyser::Static::Acorn::GetTargets(const Media &media, const std::string &file_name, TargetPlatform::IntType potential_platforms) { auto target = std::make_unique(); - target->machine = Machine::Electron; target->confidence = 0.5; // TODO: a proper estimation target->has_dfs = false; target->has_adfs = false; diff --git a/Analyser/Static/Acorn/Target.hpp b/Analyser/Static/Acorn/Target.hpp index 7c312e1fd..f049f3e8f 100644 --- a/Analyser/Static/Acorn/Target.hpp +++ b/Analyser/Static/Acorn/Target.hpp @@ -23,7 +23,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl< bool should_shift_restart = false; std::string loading_command; - Target() { + Target() : Analyser::Static::Target(Machine::Electron) { if(needs_declare()) { DeclareField(has_adfs); DeclareField(has_dfs); diff --git a/Analyser/Static/AmstradCPC/StaticAnalyser.cpp b/Analyser/Static/AmstradCPC/StaticAnalyser.cpp index 7efadf16d..c45adf2b2 100644 --- a/Analyser/Static/AmstradCPC/StaticAnalyser.cpp +++ b/Analyser/Static/AmstradCPC/StaticAnalyser.cpp @@ -182,7 +182,6 @@ static bool CheckBootSector(const std::shared_ptr &disk, co Analyser::Static::TargetList Analyser::Static::AmstradCPC::GetTargets(const Media &media, const std::string &file_name, TargetPlatform::IntType potential_platforms) { TargetList destination; auto target = std::make_unique(); - target->machine = Machine::AmstradCPC; target->confidence = 0.5; target->model = Target::Model::CPC6128; diff --git a/Analyser/Static/AmstradCPC/Target.hpp b/Analyser/Static/AmstradCPC/Target.hpp index 765b2f04f..586398e6e 100644 --- a/Analyser/Static/AmstradCPC/Target.hpp +++ b/Analyser/Static/AmstradCPC/Target.hpp @@ -18,12 +18,12 @@ namespace Analyser { namespace Static { namespace AmstradCPC { -struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { +struct Target: public Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(Model, CPC464, CPC664, CPC6128); Model model = Model::CPC464; std::string loading_command; - Target() { + Target() : Analyser::Static::Target(Machine::AmstradCPC) { if(needs_declare()) { DeclareField(model); AnnounceEnum(Model); diff --git a/Analyser/Static/AppleII/StaticAnalyser.cpp b/Analyser/Static/AppleII/StaticAnalyser.cpp index b0b280fb1..71791a743 100644 --- a/Analyser/Static/AppleII/StaticAnalyser.cpp +++ b/Analyser/Static/AppleII/StaticAnalyser.cpp @@ -11,7 +11,6 @@ Analyser::Static::TargetList Analyser::Static::AppleII::GetTargets(const Media &media, const std::string &file_name, TargetPlatform::IntType potential_platforms) { auto target = std::make_unique(); - target->machine = Machine::AppleII; target->media = media; if(!target->media.disks.empty()) diff --git a/Analyser/Static/AppleII/Target.hpp b/Analyser/Static/AppleII/Target.hpp index e1a6141cc..eb67b2af2 100644 --- a/Analyser/Static/AppleII/Target.hpp +++ b/Analyser/Static/AppleII/Target.hpp @@ -17,7 +17,7 @@ namespace Analyser { namespace Static { namespace AppleII { -struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { +struct Target: public Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(Model, II, IIplus, @@ -33,7 +33,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl< Model model = Model::IIe; DiskController disk_controller = DiskController::None; - Target() { + Target() : Analyser::Static::Target(Machine::AppleII) { if(needs_declare()) { DeclareField(model); DeclareField(disk_controller); diff --git a/Analyser/Static/Atari2600/StaticAnalyser.cpp b/Analyser/Static/Atari2600/StaticAnalyser.cpp index cd756e59e..9b627de38 100644 --- a/Analyser/Static/Atari2600/StaticAnalyser.cpp +++ b/Analyser/Static/Atari2600/StaticAnalyser.cpp @@ -184,7 +184,6 @@ static void DeterminePagingForCartridge(Target &target, const Storage::Cartridge Analyser::Static::TargetList Analyser::Static::Atari2600::GetTargets(const Media &media, const std::string &file_name, TargetPlatform::IntType potential_platforms) { // TODO: sanity checking; is this image really for an Atari 2600? auto target = std::make_unique(); - target->machine = Machine::Atari2600; target->confidence = 0.5; target->media.cartridges = media.cartridges; target->paging_model = Target::PagingModel::None; diff --git a/Analyser/Static/Atari2600/Target.hpp b/Analyser/Static/Atari2600/Target.hpp index 4bd4c5384..c372656ff 100644 --- a/Analyser/Static/Atari2600/Target.hpp +++ b/Analyser/Static/Atari2600/Target.hpp @@ -34,6 +34,8 @@ struct Target: public ::Analyser::Static::Target { // TODO: shouldn't these be properties of the cartridge? PagingModel paging_model = PagingModel::None; bool uses_superchip = false; + + Target() : Analyser::Static::Target(Machine::Atari2600) {} }; } diff --git a/Analyser/Static/AtariST/StaticAnalyser.cpp b/Analyser/Static/AtariST/StaticAnalyser.cpp index b46d0e17d..e2896aa6d 100644 --- a/Analyser/Static/AtariST/StaticAnalyser.cpp +++ b/Analyser/Static/AtariST/StaticAnalyser.cpp @@ -16,9 +16,8 @@ Analyser::Static::TargetList Analyser::Static::AtariST::GetTargets(const Media & // As there is at least one usable media image, wave it through. Analyser::Static::TargetList targets; - using Target = Analyser::Static::Target; - auto *target = new Target; - target->machine = Analyser::Machine::AtariST; + using Target = Analyser::Static::AtariST::Target; + auto *target = new Target(); target->media = media; targets.push_back(std::unique_ptr(target)); diff --git a/Analyser/Static/AtariST/Target.hpp b/Analyser/Static/AtariST/Target.hpp index f9c6f2828..4933c8776 100644 --- a/Analyser/Static/AtariST/Target.hpp +++ b/Analyser/Static/AtariST/Target.hpp @@ -9,11 +9,15 @@ #ifndef Analyser_Static_AtariST_Target_h #define Analyser_Static_AtariST_Target_h +#include "../../../Reflection/Struct.h" +#include "../StaticAnalyser.hpp" + namespace Analyser { namespace Static { namespace AtariST { -struct Target: public ::Analyser::Static::Target { +struct Target: public Analyser::Static::Target, public Reflection::StructImpl { + Target() : Analyser::Static::Target(Machine::AtariST) {} }; } diff --git a/Analyser/Static/Coleco/StaticAnalyser.cpp b/Analyser/Static/Coleco/StaticAnalyser.cpp index 25f69e926..34abed780 100644 --- a/Analyser/Static/Coleco/StaticAnalyser.cpp +++ b/Analyser/Static/Coleco/StaticAnalyser.cpp @@ -54,8 +54,7 @@ static std::vector> Analyser::Static::TargetList Analyser::Static::Coleco::GetTargets(const Media &media, const std::string &file_name, TargetPlatform::IntType potential_platforms) { TargetList targets; - auto target = std::make_unique(); - target->machine = Machine::ColecoVision; + auto target = std::make_unique(Machine::ColecoVision); target->confidence = 1.0f - 1.0f / 32768.0f; target->media.cartridges = ColecoCartridgesFrom(media.cartridges); if(!target->media.empty()) diff --git a/Analyser/Static/Commodore/Target.hpp b/Analyser/Static/Commodore/Target.hpp index d207cf0bd..dd08b87d9 100644 --- a/Analyser/Static/Commodore/Target.hpp +++ b/Analyser/Static/Commodore/Target.hpp @@ -18,7 +18,7 @@ namespace Analyser { namespace Static { namespace Commodore { -struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { +struct Target: public Analyser::Static::Target, public Reflection::StructImpl { enum class MemoryModel { Unexpanded, EightKB, @@ -57,7 +57,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl< bool has_c1540 = false; std::string loading_command; - Target() { + Target() : Analyser::Static::Target(Machine::Vic20) { if(needs_declare()) { DeclareField(enabled_ram.bank0); DeclareField(enabled_ram.bank1); diff --git a/Analyser/Static/DiskII/StaticAnalyser.cpp b/Analyser/Static/DiskII/StaticAnalyser.cpp index 5e8824ac2..56c7c5d3f 100644 --- a/Analyser/Static/DiskII/StaticAnalyser.cpp +++ b/Analyser/Static/DiskII/StaticAnalyser.cpp @@ -21,7 +21,6 @@ namespace { Analyser::Static::Target *AppleTarget(const Storage::Encodings::AppleGCR::Sector *sector_zero) { using Target = Analyser::Static::AppleII::Target; auto *target = new Target; - target->machine = Analyser::Machine::AppleII; if(sector_zero && sector_zero->encoding == Storage::Encodings::AppleGCR::Sector::Encoding::FiveAndThree) { target->disk_controller = Target::DiskController::ThirteenSector; @@ -35,7 +34,6 @@ Analyser::Static::Target *AppleTarget(const Storage::Encodings::AppleGCR::Sector Analyser::Static::Target *OricTarget(const Storage::Encodings::AppleGCR::Sector *sector_zero) { using Target = Analyser::Static::Oric::Target; auto *target = new Target; - target->machine = Analyser::Machine::Oric; target->rom = Target::ROM::Pravetz; target->disk_interface = Target::DiskInterface::Pravetz; target->loading_command = "CALL 800\n"; diff --git a/Analyser/Static/MSX/StaticAnalyser.cpp b/Analyser/Static/MSX/StaticAnalyser.cpp index a20350ca9..7342a793c 100644 --- a/Analyser/Static/MSX/StaticAnalyser.cpp +++ b/Analyser/Static/MSX/StaticAnalyser.cpp @@ -35,7 +35,6 @@ static std::unique_ptr CartridgeTarget( } auto target = std::make_unique(); - target->machine = Analyser::Machine::MSX; target->confidence = confidence; if(type == Analyser::Static::MSX::Cartridge::Type::None) { @@ -295,7 +294,6 @@ Analyser::Static::TargetList Analyser::Static::MSX::GetTargets(const Media &medi target->has_disk_drive = !media.disks.empty(); if(!target->media.empty()) { - target->machine = Machine::MSX; target->confidence = 0.5; destination.push_back(std::move(target)); } diff --git a/Analyser/Static/MSX/Target.hpp b/Analyser/Static/MSX/Target.hpp index f7c7ddbac..7e16ae25a 100644 --- a/Analyser/Static/MSX/Target.hpp +++ b/Analyser/Static/MSX/Target.hpp @@ -29,7 +29,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl< ); Region region = Region::USA; - Target() { + Target(): Analyser::Static::Target(Machine::MSX) { if(needs_declare()) { DeclareField(has_disk_drive); DeclareField(region); diff --git a/Analyser/Static/Macintosh/StaticAnalyser.cpp b/Analyser/Static/Macintosh/StaticAnalyser.cpp index 1dc76db84..d14a69554 100644 --- a/Analyser/Static/Macintosh/StaticAnalyser.cpp +++ b/Analyser/Static/Macintosh/StaticAnalyser.cpp @@ -18,7 +18,6 @@ Analyser::Static::TargetList Analyser::Static::Macintosh::GetTargets(const Media using Target = Analyser::Static::Macintosh::Target; auto *target = new Target; - target->machine = Analyser::Machine::Macintosh; target->media = media; targets.push_back(std::unique_ptr(target)); diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 9b8aab7e7..905739429 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -16,11 +16,11 @@ namespace Analyser { namespace Static { namespace Macintosh { -struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { +struct Target: public Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(Model, Mac128k, Mac512k, Mac512ke, MacPlus); Model model = Model::MacPlus; - Target() { + Target() : Analyser::Static::Target(Machine::Macintosh) { // Boilerplate for declaring fields and potential values. if(needs_declare()) { DeclareField(model); diff --git a/Analyser/Static/Oric/StaticAnalyser.cpp b/Analyser/Static/Oric/StaticAnalyser.cpp index 3a39c4eb4..22d607f29 100644 --- a/Analyser/Static/Oric/StaticAnalyser.cpp +++ b/Analyser/Static/Oric/StaticAnalyser.cpp @@ -147,7 +147,6 @@ bool is_bd500(Storage::Encodings::MFM::Parser &parser) { Analyser::Static::TargetList Analyser::Static::Oric::GetTargets(const Media &media, const std::string &file_name, TargetPlatform::IntType potential_platforms) { auto target = std::make_unique(); - target->machine = Machine::Oric; target->confidence = 0.5; int basic10_votes = 0; diff --git a/Analyser/Static/Oric/Target.hpp b/Analyser/Static/Oric/Target.hpp index 114b6a4a3..e0f81f6d7 100644 --- a/Analyser/Static/Oric/Target.hpp +++ b/Analyser/Static/Oric/Target.hpp @@ -18,7 +18,7 @@ namespace Analyser { namespace Static { namespace Oric { -struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl { +struct Target: public Analyser::Static::Target, public Reflection::StructImpl { ReflectableEnum(ROM, BASIC10, BASIC11, @@ -38,7 +38,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl< std::string loading_command; bool should_start_jasmin = false; - Target() { + Target(): Analyser::Static::Target(Machine::Oric) { if(needs_declare()) { DeclareField(rom); DeclareField(disk_interface); diff --git a/Analyser/Static/Sega/StaticAnalyser.cpp b/Analyser/Static/Sega/StaticAnalyser.cpp index 37b9d54d4..7e0db0c84 100644 --- a/Analyser/Static/Sega/StaticAnalyser.cpp +++ b/Analyser/Static/Sega/StaticAnalyser.cpp @@ -20,8 +20,6 @@ Analyser::Static::TargetList Analyser::Static::Sega::GetTargets(const Media &med TargetList targets; auto target = std::make_unique(); - target->machine = Machine::MasterSystem; - // Files named .sg are treated as for the SG1000; otherwise assume a Master System. if(file_name.size() >= 2 && *(file_name.end() - 2) == 's' && *(file_name.end() - 1) == 'g') { target->model = Target::Model::SG1000; diff --git a/Analyser/Static/Sega/Target.hpp b/Analyser/Static/Sega/Target.hpp index f95766ab6..1735704ed 100644 --- a/Analyser/Static/Sega/Target.hpp +++ b/Analyser/Static/Sega/Target.hpp @@ -9,11 +9,13 @@ #ifndef Analyser_Static_Sega_Target_h #define Analyser_Static_Sega_Target_h +#include "../StaticAnalyser.hpp" + namespace Analyser { namespace Static { namespace Sega { -struct Target: public ::Analyser::Static::Target { +struct Target: public Analyser::Static::Target { enum class Model { SG1000, MasterSystem, @@ -35,6 +37,8 @@ struct Target: public ::Analyser::Static::Target { Model model = Model::MasterSystem; Region region = Region::Japan; PagingScheme paging_scheme = PagingScheme::Sega; + + Target() : Analyser::Static::Target(Machine::MasterSystem) {} }; #define is_master_system(v) v >= Analyser::Static::Sega::Target::Model::MasterSystem diff --git a/Analyser/Static/StaticAnalyser.hpp b/Analyser/Static/StaticAnalyser.hpp index 5981fc435..b9a62770d 100644 --- a/Analyser/Static/StaticAnalyser.hpp +++ b/Analyser/Static/StaticAnalyser.hpp @@ -42,11 +42,12 @@ struct Media { and instructions on how to launch the software attached, plus a measure of confidence in this target's correctness. */ struct Target { + Target(Machine machine) : machine(machine) {} virtual ~Target() {} Machine machine; Media media; - float confidence; + float confidence = 0.0f; }; typedef std::vector> TargetList; diff --git a/Analyser/Static/ZX8081/Target.hpp b/Analyser/Static/ZX8081/Target.hpp index b85e2cce8..3461974ed 100644 --- a/Analyser/Static/ZX8081/Target.hpp +++ b/Analyser/Static/ZX8081/Target.hpp @@ -30,7 +30,7 @@ struct Target: public ::Analyser::Static::Target, public Reflection::StructImpl< bool ZX80_uses_ZX81_ROM = false; std::string loading_command; - Target() { + Target(): Analyser::Static::Target(Machine::ZX8081) { if(needs_declare()) { DeclareField(memory_model); DeclareField(is_ZX81); diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 5a6721911..6f2dd2302 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -79,11 +79,11 @@ + isEnabled = "YES"> + isEnabled = "YES"> (); - target->machine = Analyser::Machine::Electron; target->has_dfs = !!dfs; target->has_adfs = !!adfs; _targets.push_back(std::move(target)); @@ -59,7 +59,6 @@ if(self) { using Target = Analyser::Static::AmstradCPC::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::AmstradCPC; switch(model) { case CSMachineCPCModel464: target->model = Analyser::Static::AmstradCPC::Target::Model::CPC464; break; case CSMachineCPCModel664: target->model = Analyser::Static::AmstradCPC::Target::Model::CPC664; break; @@ -75,7 +74,6 @@ if(self) { using Target = Analyser::Static::MSX::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::MSX; target->has_disk_drive = !!hasDiskDrive; switch(region) { case CSMachineMSXRegionAmerican: target->region = Analyser::Static::MSX::Target::Region::USA; break; @@ -92,7 +90,6 @@ if(self) { using Target = Analyser::Static::Oric::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::Oric; switch(model) { case CSMachineOricModelOric1: target->rom = Target::ROM::BASIC10; break; case CSMachineOricModelOricAtmos: target->rom = Target::ROM::BASIC11; break; @@ -115,7 +112,6 @@ if(self) { using Target = Analyser::Static::Commodore::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::Vic20; switch(region) { case CSMachineVic20RegionDanish: target->region = Target::Region::Danish; break; case CSMachineVic20RegionSwedish: target->region = Target::Region::Swedish; break; @@ -150,7 +146,6 @@ static Analyser::Static::ZX8081::Target::MemoryModel ZX8081MemoryModelFromSize(K if(self) { using Target = Analyser::Static::ZX8081::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::ZX8081; target->is_ZX81 = false; target->ZX80_uses_ZX81_ROM = !!useZX81ROM; target->memory_model = ZX8081MemoryModelFromSize(memorySize); @@ -164,7 +159,6 @@ static Analyser::Static::ZX8081::Target::MemoryModel ZX8081MemoryModelFromSize(K if(self) { using Target = Analyser::Static::ZX8081::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::ZX8081; target->is_ZX81 = true; target->memory_model = ZX8081MemoryModelFromSize(memorySize); _targets.push_back(std::move(target)); @@ -177,7 +171,6 @@ static Analyser::Static::ZX8081::Target::MemoryModel ZX8081MemoryModelFromSize(K if(self) { using Target = Analyser::Static::AppleII::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::AppleII; switch(model) { default: target->model = Target::Model::II; break; case CSMachineAppleIIModelAppleIIPlus: target->model = Target::Model::IIplus; break; @@ -200,7 +193,6 @@ static Analyser::Static::ZX8081::Target::MemoryModel ZX8081MemoryModelFromSize(K if(self) { using Target = Analyser::Static::Macintosh::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::Macintosh; using Model = Target::Model; switch(model) { @@ -219,9 +211,8 @@ static Analyser::Static::ZX8081::Target::MemoryModel ZX8081MemoryModelFromSize(K - (instancetype)initWithAtariSTModel:(CSMachineAtariSTModel)model { self = [super init]; if(self) { - using Target = Analyser::Static::Macintosh::Target; + using Target = Analyser::Static::AtariST::Target; auto target = std::make_unique(); - target->machine = Analyser::Machine::AtariST; _targets.push_back(std::move(target)); } return self; diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 65adec45a..9dcf0ac2a 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -139,7 +139,8 @@ template class StructImpl: public Struct { private: struct Field { const std::type_info *type; - ssize_t offset, size; + ssize_t offset; + size_t size; Field(const std::type_info &type, ssize_t offset, size_t size) : type(&type), offset(offset), size(size) {} }; From 880bed04f5518087cc710995eccbb4b20507ffbb Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 15 Mar 2020 00:15:19 -0400 Subject: [PATCH 24/48] Adds `AllMachines`, rounds out ConstructionOptionsByMachineName. --- Machines/Utility/MachineForTarget.cpp | 42 ++++++++++++++++++++++----- Machines/Utility/MachineForTarget.hpp | 5 ++++ 2 files changed, 40 insertions(+), 7 deletions(-) diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 721ea68e8..73db40743 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -150,6 +150,31 @@ std::string Machine::LongNameForTargetMachine(Analyser::Machine machine) { } } +std::vector Machine::AllMachines(bool meaningful_without_media_only, bool long_names) { + std::vector result; + +#define AddName(x) result.push_back(long_names ? LongNameForTargetMachine(Analyser::Machine::x) : ShortNameForTargetMachine(Analyser::Machine::x)) +#define AddConditionalName(x) if(!meaningful_without_media_only) result.push_back(long_names ? LongNameForTargetMachine(Analyser::Machine::x) : ShortNameForTargetMachine(Analyser::Machine::x)) + + AddName(AmstradCPC); + AddName(AppleII); + AddConditionalName(Atari2600); + AddName(AtariST); + AddConditionalName(ColecoVision); + AddName(Electron); + AddName(Macintosh); + AddConditionalName(MasterSystem); + AddName(MSX); + AddName(Oric); + AddName(Vic20); + AddName(ZX8081); + +#undef AddConditionalName +#undef AddName + + return result; +} + std::map>> Machine::AllOptionsByMachineName() { std::map>> options; @@ -171,19 +196,22 @@ std::map>> Machin std::map> Machine::ConstructionOptionsByMachineName() { std::map> options; -#define Add(Name) \ - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Name), new Analyser::Static::Name::Target)); +#define AddMapped(Name, TargetNamespace) \ + options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Name), new Analyser::Static::TargetNamespace::Target)); +#define Add(Name) AddMapped(Name, Name) Add(AmstradCPC); Add(AppleII); -// Add(AtariST); -// Add(Electron); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Electron), Electron::get_options())); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); + Add(AtariST); + AddMapped(Electron, Acorn); + Add(Macintosh); Add(MSX); Add(Oric); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); + AddMapped(Vic20, Commodore); Add(ZX8081); +#undef Add +#undef AddTwo + return options; } diff --git a/Machines/Utility/MachineForTarget.hpp b/Machines/Utility/MachineForTarget.hpp index 6cc29bcae..15fd27bf9 100644 --- a/Machines/Utility/MachineForTarget.hpp +++ b/Machines/Utility/MachineForTarget.hpp @@ -49,6 +49,11 @@ std::string ShortNameForTargetMachine(const Analyser::Machine target); */ std::string LongNameForTargetMachine(const Analyser::Machine target); +/*! + @returns A list of all available machines. +*/ +std::vector AllMachines(bool meaningful_without_media_only, bool long_names); + /*! Returns a map from machine name to the list of options that machine exposes, for all machines. From fc3d3c76f81b6b348b01f5335cf83a3328a39bcb Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 15 Mar 2020 12:54:55 -0400 Subject: [PATCH 25/48] Edges further towards providing enough information for dynamic user-provided machine creation. --- Machines/Utility/MachineForTarget.cpp | 20 +++++---- Machines/Utility/MachineForTarget.hpp | 17 +++++++- OSBindings/SDL/main.cpp | 63 ++++++++++++++------------- 3 files changed, 60 insertions(+), 40 deletions(-) diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 73db40743..638cfbf85 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -26,19 +26,19 @@ #include "../../Analyser/Static/Acorn/Target.hpp" #include "../../Analyser/Static/AmstradCPC/Target.hpp" #include "../../Analyser/Static/AppleII/Target.hpp" +#include "../../Analyser/Static/Atari2600/Target.hpp" #include "../../Analyser/Static/AtariST/Target.hpp" #include "../../Analyser/Static/Commodore/Target.hpp" #include "../../Analyser/Static/Macintosh/Target.hpp" #include "../../Analyser/Static/MSX/Target.hpp" #include "../../Analyser/Static/Oric/Target.hpp" +#include "../../Analyser/Static/Sega/Target.hpp" #include "../../Analyser/Static/ZX8081/Target.hpp" #include "../../Analyser/Dynamic/MultiMachine/MultiMachine.hpp" #include "TypedDynamicMachine.hpp" -namespace { - -::Machine::DynamicMachine *MachineForTarget(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher, Machine::Error &error) { +Machine::DynamicMachine *Machine::MachineForTarget(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher, Machine::Error &error) { error = Machine::Error::None; Machine::DynamicMachine *machine = nullptr; @@ -78,9 +78,7 @@ namespace { return machine; } -} - -::Machine::DynamicMachine *::Machine::MachineForTargets(const Analyser::Static::TargetList &targets, const ROMMachine::ROMFetcher &rom_fetcher, Error &error) { +Machine::DynamicMachine *Machine::MachineForTargets(const Analyser::Static::TargetList &targets, const ROMMachine::ROMFetcher &rom_fetcher, Error &error) { // Zero targets implies no machine. if(targets.empty()) { error = Error::NoTargets; @@ -193,8 +191,8 @@ std::map>> Machin return options; } -std::map> Machine::ConstructionOptionsByMachineName() { - std::map> options; +std::map> Machine::TargetsByMachineName(bool meaningful_without_media_only) { + std::map> options; #define AddMapped(Name, TargetNamespace) \ options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Name), new Analyser::Static::TargetNamespace::Target)); @@ -210,6 +208,12 @@ std::map> Machine::Construction AddMapped(Vic20, Commodore); Add(ZX8081); + if(!meaningful_without_media_only) { + Add(Atari2600); + options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), new Analyser::Static::Target(Analyser::Machine::ColecoVision))); + AddMapped(MasterSystem, Sega); + } + #undef Add #undef AddTwo diff --git a/Machines/Utility/MachineForTarget.hpp b/Machines/Utility/MachineForTarget.hpp index 15fd27bf9..e5891b5b4 100644 --- a/Machines/Utility/MachineForTarget.hpp +++ b/Machines/Utility/MachineForTarget.hpp @@ -19,6 +19,8 @@ #include #include +/*! +*/ namespace Machine { enum class Error { @@ -36,6 +38,12 @@ enum class Error { */ DynamicMachine *MachineForTargets(const Analyser::Static::TargetList &targets, const ::ROMMachine::ROMFetcher &rom_fetcher, Error &error); +/*! + Allocates an instance of DynamicMaachine holding the machine described + by @c target. It is the caller's responsibility to delete the class when finished. +*/ +DynamicMachine *MachineForTarget(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher, Machine::Error &error); + /*! Returns a short string name for the machine identified by the target, which is guaranteed not to have any spaces or other potentially @@ -55,12 +63,17 @@ std::string LongNameForTargetMachine(const Analyser::Machine target); std::vector AllMachines(bool meaningful_without_media_only, bool long_names); /*! - Returns a map from machine name to the list of options that machine + Returns a map from long machine name to the list of options that machine exposes, for all machines. */ std::map>> AllOptionsByMachineName(); -std::map> ConstructionOptionsByMachineName(); +/*! + Returns a map from long machine name to appropriate instances of Target for the machine. + + NB: Usually the instances of Target can be dynamic_casted to Reflection::Struct in order to determine available properties. +*/ +std::map> TargetsByMachineName(bool meaningful_without_media_only); } diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index f7eaa114a..927ca35be 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -13,10 +13,11 @@ #include #include #include +#include #include #include #include -#include +#include #include @@ -363,7 +364,7 @@ bool KeyboardKeyForSDLScancode(SDL_Scancode scancode, Inputs::Keyboard::Key &key struct ParsedArguments { std::string file_name; - Configurable::SelectionSet selections; + std::map> selections; }; /*! Parses an argc/argv pair to discern program arguments. */ @@ -388,11 +389,11 @@ ParsedArguments parse_arguments(int argc, char *argv[]) { std::size_t split_index = argument.find("="); if(split_index == std::string::npos) { - arguments.selections[argument] = std::make_unique(true); + arguments.selections[argument] = true; } else { std::string name = argument.substr(0, split_index); std::string value = argument.substr(split_index+1, std::string::npos); - arguments.selections[name] = std::make_unique(value); + arguments.selections[name] = value; } } else { arguments.file_name = arg; @@ -506,12 +507,14 @@ int main(int argc, char *argv[]) { std::cout << std::endl; } - const auto construction_options = Machine::ConstructionOptionsByMachineName(); - for(const auto &options: construction_options) { - std::cout << options.first << ":" << std::endl; - for(const auto &option: options.second->all_keys()) { + const auto targets = Machine::TargetsByMachineName(false); + for(const auto &target: targets) { + const auto reflectable = dynamic_cast(target.second.get()); + + std::cout << target.first << ":" << std::endl; + for(const auto &option: reflectable->all_keys()) { std::cout << '\t' << "--" << option; - const auto type = options.second->type_of(option); + const auto type = reflectable->type_of(option); // Is this a registered enum? If so, list options. if(!Reflection::Enum::name(*type).empty()) { @@ -565,14 +568,14 @@ int main(int argc, char *argv[]) { "/usr/local/share/CLK/", "/usr/share/CLK/" }; - if(arguments.selections.find("rompath") != arguments.selections.end()) { - const std::string user_path = arguments.selections["rompath"]->list_selection()->value; - if(user_path.back() != '/') { - paths.push_back(user_path + "/"); - } else { - paths.push_back(user_path); - } - } +// if(arguments.selections.find("rompath") != arguments.selections.end()) { +// const std::string user_path = arguments.selections["rompath"]->list_selection()->value; +// if(user_path.back() != '/') { +// paths.push_back(user_path + "/"); +// } else { +// paths.push_back(user_path); +// } +// } std::vector>> results; for(const auto &rom: roms) { @@ -630,17 +633,17 @@ int main(int argc, char *argv[]) { // Apply the speed multiplier, if one was requested. if(arguments.selections.find("speed") != arguments.selections.end()) { - const char *speed_string = arguments.selections["speed"]->list_selection()->value.c_str(); - char *end; - double speed = strtod(speed_string, &end); - - if(size_t(end - speed_string) != strlen(speed_string)) { - std::cerr << "Unable to parse speed: " << speed_string << std::endl; - } else if(speed <= 0.0) { - std::cerr << "Cannot run at speed " << speed_string << "; speeds must be positive." << std::endl; - } else { - machine_runner.set_speed_multiplier(speed); - } +// const char *speed_string = arguments.selections["speed"]->list_selection()->value.c_str(); +// char *end; +// double speed = strtod(speed_string, &end); +// +// if(size_t(end - speed_string) != strlen(speed_string)) { +// std::cerr << "Unable to parse speed: " << speed_string << std::endl; +// } else if(speed <= 0.0) { +// std::cerr << "Cannot run at speed " << speed_string << "; speeds must be positive." << std::endl; +// } else { +// machine_runner.set_speed_multiplier(speed); +// } } // Check whether a 'logical' keyboard has been requested. @@ -718,7 +721,7 @@ int main(int argc, char *argv[]) { int window_width, window_height; SDL_GetWindowSize(window, &window_width, &window_height); - Configurable::Device *const configurable_device = machine->configurable_device(); +/* Configurable::Device *const configurable_device = machine->configurable_device(); if(configurable_device) { // Establish user-friendly options by default. configurable_device->set_selections(configurable_device->get_user_friendly_selections()); @@ -741,7 +744,7 @@ int main(int argc, char *argv[]) { // Apply the user's actual selections to final the defaults. configurable_device->set_selections(arguments.selections); - } + }*/ // If this is a joystick machine, check for and open attached joysticks. /*! From 2031a33edf1a910e61248135009dd3fdddf193b9 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 15 Mar 2020 21:50:43 -0400 Subject: [PATCH 26/48] Technically SDL users can now start a new machine. Missing though: all the old per-machine command-line options, and any control over the new one. --- Machines/Utility/MachineForTarget.hpp | 17 ++- .../Clock Signal.xcodeproj/project.pbxproj | 4 +- .../xcschemes/Clock Signal Kiosk.xcscheme | 7 +- OSBindings/SDL/main.cpp | 134 ++++++++++++------ Reflection/Struct.cpp | 4 + Reflection/Struct.h | 30 +++- 6 files changed, 140 insertions(+), 56 deletions(-) diff --git a/Machines/Utility/MachineForTarget.hpp b/Machines/Utility/MachineForTarget.hpp index e5891b5b4..a51508f47 100644 --- a/Machines/Utility/MachineForTarget.hpp +++ b/Machines/Utility/MachineForTarget.hpp @@ -20,6 +20,14 @@ #include /*! + This namespace acts as a grab-bag of functions that allow a client to: + + (i) discover the total list of implemented machines; + (ii) discover the construction and runtime options available for controlling them; and + (iii) create any implemented machine via its construction options. + + See Reflection::Struct and Reflection::Enum for getting dynamic information from the + Targets that this namespace deals in. */ namespace Machine { @@ -58,13 +66,16 @@ std::string ShortNameForTargetMachine(const Analyser::Machine target); std::string LongNameForTargetMachine(const Analyser::Machine target); /*! - @returns A list of all available machines. + @param meaningful_without_media_only If this is @c true then only machines that it is meaningful to start without a piece of media will be listed; + otherwise all supported machines will be listed. + @param long_names If this is @c true then long names will be returned; otherwise short names will be returned. + + @returns A list of all available machines. Names are always guaranteed to be in the same order. */ std::vector AllMachines(bool meaningful_without_media_only, bool long_names); /*! - Returns a map from long machine name to the list of options that machine - exposes, for all machines. + Returns a map from long machine name to the list of options that machine exposes, for all machines. */ std::map>> AllOptionsByMachineName(); diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 377c50181..407c7943d 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -4981,7 +4981,7 @@ "$(USER_LIBRARY_DIR)/Frameworks", ); GCC_C_LANGUAGE_STANDARD = gnu11; - MACOSX_DEPLOYMENT_TARGET = 10.12; + MACOSX_DEPLOYMENT_TARGET = 10.14; PRODUCT_NAME = "$(TARGET_NAME)"; }; name = Debug; @@ -5003,7 +5003,7 @@ ); GCC_C_LANGUAGE_STANDARD = gnu11; GCC_OPTIMIZATION_LEVEL = 2; - MACOSX_DEPLOYMENT_TARGET = 10.12; + MACOSX_DEPLOYMENT_TARGET = 10.14; PRODUCT_NAME = "$(TARGET_NAME)"; }; name = Release; diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 6f2dd2302..aac0802ab 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -34,7 +34,6 @@ buildConfiguration = "Debug" selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB" selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB" - enableThreadSanitizer = "YES" disableMainThreadChecker = "YES" launchStyle = "0" useCustomWorkingDirectory = "NO" @@ -57,6 +56,10 @@ argument = "/Users/thomasharte/Downloads/test-dsk-for-rw-and-50-60-hz/TEST-RW-60Hz.DSK" isEnabled = "NO"> + + @@ -87,7 +90,7 @@ + isEnabled = "NO"> diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index 927ca35be..d7cf2c4b0 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -17,7 +17,6 @@ #include #include #include -#include #include @@ -364,7 +363,7 @@ bool KeyboardKeyForSDLScancode(SDL_Scancode scancode, Inputs::Keyboard::Key &key struct ParsedArguments { std::string file_name; - std::map> selections; + std::map selections; // The empty string will be inserted for arguments without an = suffix. }; /*! Parses an argc/argv pair to discern program arguments. */ @@ -389,7 +388,7 @@ ParsedArguments parse_arguments(int argc, char *argv[]) { std::size_t split_index = argument.find("="); if(split_index == std::string::npos) { - arguments.selections[argument] = true; + arguments.selections[argument]; // To create an entry with the default empty string. } else { std::string name = argument.substr(0, split_index); std::string value = argument.substr(split_index+1, std::string::npos); @@ -477,39 +476,55 @@ int main(int argc, char *argv[]) { ParsedArguments arguments = parse_arguments(argc, argv); // This may be printed either as - const std::string usage_suffix = " [file] [OPTIONS] [--rompath={path to ROMs}] [--speed={speed multiplier, e.g. 1.5}]"; /* [--logical-keyboard] */ + const std::string usage_suffix = " [file or --new={machine}] [OPTIONS] [--rompath={path to ROMs}] [--speed={speed multiplier, e.g. 1.5}] [--logical-keyboard]"; // Print a help message if requested. if(arguments.selections.find("help") != arguments.selections.end() || arguments.selections.find("h") != arguments.selections.end()) { + const auto all_machines = Machine::AllMachines(false, false); + std::cout << "Usage: " << final_path_component(argv[0]) << usage_suffix << std::endl; std::cout << "Use alt+enter to toggle full screen display. Use control+shift+V to paste text." << std::endl; - std::cout << "Required machine type and configuration is determined from the file. Machines with further options:" << std::endl << std::endl; - - const auto all_options = Machine::AllOptionsByMachineName(); - for(const auto &machine_options: all_options) { - std::cout << machine_options.first << ":" << std::endl; - for(const auto &option: machine_options.second) { - std::cout << '\t' << "--" << option->short_name; - - Configurable::ListOption *list_option = dynamic_cast(option.get()); - if(list_option) { - std::cout << "={"; - bool is_first = true; - for(const auto &option: list_option->options) { - if(!is_first) std::cout << '|'; - is_first = false; - std::cout << option; - } - std::cout << "}"; - } - std::cout << std::endl; - } - std::cout << std::endl; + std::cout << "Required machine type and configuration is determined from the file if specified; otherwise use:" << std::endl << std::endl; + std::cout << "\t--new={"; + bool is_first = true; + for(const auto &name: all_machines) { + if(!is_first) std::cout << "|"; + std::cout << name; + is_first = false; } + std::cout << "}" << std::endl << std::endl; + + std::cout << "Further machine options:" << std::endl; + std::cout << "(* means: a selection will be made automatically based on the file selected, if any)" << std::endl << std::endl; + +// const auto all_options = Machine::AllOptionsByMachineName(); +// for(const auto &machine_options: all_options) { +// std::cout << machine_options.first << ":" << std::endl; +// for(const auto &option: machine_options.second) { +// std::cout << '\t' << "--" << option->short_name; +// +// Configurable::ListOption *list_option = dynamic_cast(option.get()); +// if(list_option) { +// std::cout << "={"; +// bool is_first = true; +// for(const auto &option: list_option->options) { +// if(!is_first) std::cout << '|'; +// is_first = false; +// std::cout << option; +// } +// std::cout << "}"; +// } +// std::cout << std::endl; +// } +// std::cout << std::endl; +// } const auto targets = Machine::TargetsByMachineName(false); for(const auto &target: targets) { const auto reflectable = dynamic_cast(target.second.get()); + if(!reflectable) continue; + const auto all_keys = reflectable->all_keys(); + if(all_keys.empty()) continue; std::cout << target.first << ":" << std::endl; for(const auto &option: reflectable->all_keys()) { @@ -529,7 +544,7 @@ int main(int argc, char *argv[]) { } // TODO: if not a registered enum... then assume it was a Boolean? - std::cout << std::endl; + std::cout << "\t*" << std::endl; } std::cout << std::endl; @@ -537,17 +552,50 @@ int main(int argc, char *argv[]) { return EXIT_SUCCESS; } - // Perform a sanity check on arguments. - if(arguments.file_name.empty()) { - std::cerr << "Usage: " << final_path_component(argv[0]) << usage_suffix << std::endl; - std::cerr << "Use --help to learn more about available options." << std::endl; - return EXIT_FAILURE; + // Determine the machine for the supplied file, if any, or from --new. + Analyser::Static::TargetList targets; + if(!arguments.file_name.empty()) { + targets = Analyser::Static::GetTargets(arguments.file_name); + } + + const auto new_argument = arguments.selections.find("new"); + if(new_argument != arguments.selections.end() && !new_argument->second.empty()) { + // Perform for a case-insensitive search against short names. + const auto short_names = Machine::AllMachines(false, false); + auto short_name = short_names.begin(); + while(short_name != short_names.end()) { + if(std::equal( + short_name->begin(), short_name->end(), + new_argument->second.begin(), new_argument->second.end(), + [](char a, char b) { return tolower(b) == tolower(a); })) { + break; + } + ++short_name; + } + + // If a match was found, use the corresponding long name to look up a suitable + // Analyser::Statuc::Target and move that to the targets list. + if(short_name != short_names.end()) { + const auto long_name = Machine::AllMachines(false, true)[short_name - short_names.begin()]; + auto targets_by_machine = Machine::TargetsByMachineName(false); + std::unique_ptr tgt = std::move(targets_by_machine[long_name]); + targets.push_back(std::move(tgt)); + } } - // Determine the machine for the supplied file. - const auto targets = Analyser::Static::GetTargets(arguments.file_name); if(targets.empty()) { - std::cerr << "Cannot open " << arguments.file_name << "; no target machine found" << std::endl; + if(!arguments.file_name.empty()) { + std::cerr << "Cannot open " << arguments.file_name << "; no target machine found" << std::endl; + return EXIT_FAILURE; + } + + if(!new_argument->second.empty()) { + std::cerr << "Unknown machine: " << new_argument->second << std::endl; + return EXIT_FAILURE; + } + + std::cerr << "Usage: " << final_path_component(argv[0]) << usage_suffix << std::endl; + std::cerr << "Use --help to learn more about available options." << std::endl; return EXIT_FAILURE; } @@ -568,14 +616,14 @@ int main(int argc, char *argv[]) { "/usr/local/share/CLK/", "/usr/share/CLK/" }; -// if(arguments.selections.find("rompath") != arguments.selections.end()) { -// const std::string user_path = arguments.selections["rompath"]->list_selection()->value; -// if(user_path.back() != '/') { -// paths.push_back(user_path + "/"); -// } else { -// paths.push_back(user_path); -// } -// } + if(arguments.selections.find("rompath") != arguments.selections.end()) { + const std::string user_path = arguments.selections["rompath"]; + if(user_path.back() != '/') { + paths.push_back(user_path + "/"); + } else { + paths.push_back(user_path); + } + } std::vector>> results; for(const auto &rom: roms) { diff --git a/Reflection/Struct.cpp b/Reflection/Struct.cpp index fbf92ab09..cbe533987 100644 --- a/Reflection/Struct.cpp +++ b/Reflection/Struct.cpp @@ -51,3 +51,7 @@ template <> bool Reflection::set(Struct &target, const std::string &name, const const std::string string(value); return set(target, name, string); } + +bool Reflection::fuzzy_set(Struct &target, const std::string &name, const std::string &value) { + return false; +} diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 9dcf0ac2a..256d53876 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -33,7 +33,7 @@ struct Struct { /*! Attempts to set the property @c name to @c value ; will perform limited type conversions. - @returns @c true if t + @returns @c true if the property was successfully set; @c false otherwise. */ template bool set(Struct &target, const std::string &name, Type value); @@ -54,12 +54,30 @@ template <> bool set(Struct &target, const std::string &name, int value); template <> bool set(Struct &target, const std::string &name, const std::string &value); template <> bool set(Struct &target, const std::string &name, const char *value); + +/*! + Fuzzy-set attempts to set any property based on a string value. This is intended to allow input provided by the user. + + Amongst other steps, it might: + * if the target is a bool, map true, false, yes, no, y, n, etc; + * if the target is an integer, parse like strtrol; + * if the target is a float, parse like strtod; or + * if the target is a reflective enum, attempt to match to enum members (possibly doing so in a case insensitive fashion). + + This method reserves the right to perform more or fewer attempted mappings, using any other logic it + decides is appropriate. + +@returns @c true if the property was successfully set; @c false otherwise. +*/ +bool fuzzy_set(Struct &target, const std::string &name, const std::string &value); + +// TODO: move this elsewhere. It's just a sketch anyway. struct Serialisable { - /// Serialises this object, appending it to @c target. - virtual void serialise(std::vector &target) = 0; - /// Deserialises this object from @c source. - /// @returns @c true if the deserialisation was successful; @c false otherwise. - virtual bool deserialise(const std::vector &source) = 0; + /// Serialises this object, appending it to @c target. + virtual void serialise(std::vector &target) = 0; + /// Deserialises this object from @c source. + /// @returns @c true if the deserialisation was successful; @c false otherwise. + virtual bool deserialise(const std::vector &source) = 0; }; template class StructImpl: public Struct { From 8e3bf0dbcab196ec2f1390e4a8984ef38dd4d307 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 15 Mar 2020 23:48:53 -0400 Subject: [PATCH 27/48] Starts moving towards a Deflectable-based system of runtime options. --- .../Implementation/MultiConfigurable.cpp | 11 ++- .../Implementation/MultiConfigurable.hpp | 11 ++- Configurable/Configurable.cpp | 27 ------ Configurable/Configurable.hpp | 93 ++++--------------- Configurable/StandardOptions.cpp | 24 +---- Configurable/StandardOptions.hpp | 76 +-------------- Machines/AmstradCPC/AmstradCPC.cpp | 56 ++++++----- Machines/AmstradCPC/AmstradCPC.hpp | 5 +- Machines/Apple/AppleII/AppleII.cpp | 51 +++++----- Machines/Apple/AppleII/AppleII.hpp | 5 +- Machines/Apple/Macintosh/Macintosh.cpp | 71 ++++++++------ Machines/Apple/Macintosh/Macintosh.hpp | 6 +- Machines/Atari/ST/AtariST.cpp | 52 ++++++----- Machines/Atari/ST/AtariST.hpp | 6 +- Machines/ColecoVision/ColecoVision.cpp | 57 +++++++----- Machines/ColecoVision/ColecoVision.hpp | 6 +- Machines/Commodore/Vic-20/Vic20.cpp | 71 +++++++------- Machines/Commodore/Vic-20/Vic20.hpp | 5 +- Machines/Electron/Electron.cpp | 74 ++++++++------- Machines/Electron/Electron.hpp | 6 +- Machines/MSX/MSX.cpp | 73 ++++++++------- Machines/MSX/MSX.hpp | 5 +- Machines/MasterSystem/MasterSystem.cpp | 57 +++++++----- Machines/MasterSystem/MasterSystem.hpp | 6 +- Machines/Oric/Oric.cpp | 78 +++++++++------- Machines/Oric/Oric.hpp | 6 +- Machines/Utility/MachineForTarget.cpp | 24 ++--- Machines/Utility/MachineForTarget.hpp | 2 +- Machines/ZX8081/ZX8081.cpp | 57 ++++++------ Machines/ZX8081/ZX8081.hpp | 6 +- .../Clock Signal.xcodeproj/project.pbxproj | 12 +-- Reflection/Struct.cpp | 22 +++++ Reflection/Struct.h | 11 +++ 33 files changed, 524 insertions(+), 548 deletions(-) delete mode 100644 Configurable/Configurable.cpp diff --git a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp index 8e85c9410..c77ff6de0 100644 --- a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp +++ b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp @@ -19,7 +19,15 @@ MultiConfigurable::MultiConfigurable(const std::vector> MultiConfigurable::get_options() { +void MultiConfigurable::set_options(const std::unique_ptr &options) { +} + +std::unique_ptr MultiConfigurable::get_options(OptionsType type) { + // TODO: this'll need to mash options together, maybe? Or just take the front? + return nullptr; +} + +/*std::vector> MultiConfigurable::get_options() { std::vector> options; // Produce the list of unique options. @@ -62,3 +70,4 @@ Configurable::SelectionSet MultiConfigurable::get_user_friendly_selections() { } return set; } +*/ diff --git a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp index ac4aecae9..7d4672987 100644 --- a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp +++ b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp @@ -10,6 +10,7 @@ #define MultiConfigurable_hpp #include "../../../../Machines/DynamicMachine.hpp" +#include "../../../../Configurable/Configurable.hpp" #include #include @@ -28,10 +29,12 @@ class MultiConfigurable: public Configurable::Device { MultiConfigurable(const std::vector> &machines); // Below is the standard Configurable::Device interface; see there for documentation. - std::vector> get_options() final; - void set_selections(const Configurable::SelectionSet &selection_by_option) final; - Configurable::SelectionSet get_accurate_selections() final; - Configurable::SelectionSet get_user_friendly_selections() final; + void set_options(const std::unique_ptr &options) final; + std::unique_ptr get_options(OptionsType type) final; +// std::vector> get_options() final; +// void set_selections(const Configurable::SelectionSet &selection_by_option) final; +// Configurable::SelectionSet get_accurate_selections() final; +// Configurable::SelectionSet get_user_friendly_selections() final; private: std::vector devices_; diff --git a/Configurable/Configurable.cpp b/Configurable/Configurable.cpp deleted file mode 100644 index 1834d3eb4..000000000 --- a/Configurable/Configurable.cpp +++ /dev/null @@ -1,27 +0,0 @@ -// -// Configurable.cpp -// Clock Signal -// -// Created by Thomas Harte on 18/11/2017. -// Copyright 2017 Thomas Harte. All rights reserved. -// - -#include "Configurable.hpp" - -using namespace Configurable; - -ListSelection *BooleanSelection::list_selection() { - return new ListSelection(value ? "yes" : "no"); -} - -ListSelection *ListSelection::list_selection() { - return new ListSelection(value); -} - -BooleanSelection *ListSelection::boolean_selection() { - return new BooleanSelection(value != "no" && value != "n" && value != "false" && value != "f"); -} - -BooleanSelection *BooleanSelection::boolean_selection() { - return new BooleanSelection(value); -} diff --git a/Configurable/Configurable.hpp b/Configurable/Configurable.hpp index 833989951..9876e02d4 100644 --- a/Configurable/Configurable.hpp +++ b/Configurable/Configurable.hpp @@ -9,89 +9,34 @@ #ifndef Configurable_h #define Configurable_h -#include -#include -#include -#include +#include "../Reflection/Struct.h" namespace Configurable { /*! - The Option class hierarchy provides a way for components, machines, etc, to provide a named - list of typed options to which they can respond. -*/ -struct Option { - std::string long_name; - std::string short_name; - virtual ~Option() {} + A Configurable::Device provides a reflective struct listing the available runtime options for this machine. + It can provide it populated with 'accurate' options, 'user-friendly' options or just whatever the user + currently has selected. - Option(const std::string &long_name, const std::string &short_name) : long_name(long_name), short_name(short_name) {} + 'Accurate' options should correspond to the way that this device was usually used during its lifespan. + E.g. a ColecoVision might accurately be given composite output. - virtual bool operator==(const Option &rhs) { - return long_name == rhs.long_name && short_name == rhs.short_name; - } -}; - -struct BooleanOption: public Option { - BooleanOption(const std::string &long_name, const std::string &short_name) : Option(long_name, short_name) {} -}; - -struct ListOption: public Option { - std::vector options; - ListOption(const std::string &long_name, const std::string &short_name, const std::vector &options) : Option(long_name, short_name), options(options) {} - - virtual bool operator==(const Option &rhs) { - const ListOption *list_rhs = dynamic_cast(&rhs); - if(!list_rhs) return false; - return long_name == rhs.long_name && short_name == rhs.short_name && options == list_rhs->options; - } -}; - -struct BooleanSelection; -struct ListSelection; - -/*! - Selections are responses to Options. -*/ -struct Selection { - virtual ~Selection() {} - virtual ListSelection *list_selection() = 0; - virtual BooleanSelection *boolean_selection() = 0; -}; - -struct BooleanSelection: public Selection { - bool value; - - ListSelection *list_selection(); - BooleanSelection *boolean_selection(); - BooleanSelection(bool value) : value(value) {} -}; - -struct ListSelection: public Selection { - std::string value; - - ListSelection *list_selection(); - BooleanSelection *boolean_selection(); - ListSelection(const std::string value) : value(value) {} -}; - -using SelectionSet = std::map>; - -/*! - A Configuratble provides the options that it responds to and allows selections to be set. + 'User-friendly' options should be more like those that a user today might most expect from an emulator. + E.g. the ColecoVision might bump itself up to S-Video output. */ struct Device { - virtual std::vector> get_options() = 0; - virtual void set_selections(const SelectionSet &selection_by_option) = 0; - virtual SelectionSet get_accurate_selections() = 0; - virtual SelectionSet get_user_friendly_selections() = 0; -}; + /// Sets the current options. + virtual void set_options(const std::unique_ptr &options) = 0; -template T *selection(const Configurable::SelectionSet &selections_by_option, const std::string &name) { - auto selection = selections_by_option.find(name); - if(selection == selections_by_option.end()) return nullptr; - return dynamic_cast(selection->second.get()); -} + enum class OptionsType { + Current, + Accurate, + UserFriendly + }; + + /// @returns Options of type @c type. + virtual std::unique_ptr get_options(OptionsType type) = 0; +}; } diff --git a/Configurable/StandardOptions.cpp b/Configurable/StandardOptions.cpp index 1cb0badf2..1dce510b7 100644 --- a/Configurable/StandardOptions.cpp +++ b/Configurable/StandardOptions.cpp @@ -8,27 +8,7 @@ #include "StandardOptions.hpp" -namespace { - -/*! - Appends a Boolean selection of @c selection for option @c name to @c selection_set. -*/ -void append_bool(Configurable::SelectionSet &selection_set, const std::string &name, bool selection) { - selection_set[name] = std::make_unique(selection); -} - -/*! - Enquires for a Boolean selection for option @c name from @c selections_by_option, storing it to @c result if found. -*/ -bool get_bool(const Configurable::SelectionSet &selections_by_option, const std::string &name, bool &result) { - auto selection = Configurable::selection(selections_by_option, name); - if(!selection) return false; - result = selection->value; - return true; -} - -} - +/* // MARK: - Standard option list builder std::vector> Configurable::standard_options(Configurable::StandardOptions mask) { std::vector> options; @@ -105,4 +85,4 @@ bool Configurable::get_display(const Configurable::SelectionSet &selections_by_o bool Configurable::get_quick_boot(const Configurable::SelectionSet &selections_by_option, bool &result) { return get_bool(selections_by_option, "quickboot", result); -} +}*/ diff --git a/Configurable/StandardOptions.hpp b/Configurable/StandardOptions.hpp index fa042ee65..53bc0f600 100644 --- a/Configurable/StandardOptions.hpp +++ b/Configurable/StandardOptions.hpp @@ -9,87 +9,17 @@ #ifndef StandardOptions_hpp #define StandardOptions_hpp -#include "Configurable.hpp" +#include "../Reflection/Enum.h" namespace Configurable { -enum StandardOptions { - DisplayRGB = (1 << 0), - DisplaySVideo = (1 << 1), - DisplayCompositeColour = (1 << 2), - DisplayCompositeMonochrome = (1 << 3), - QuickLoadTape = (1 << 4), - AutomaticTapeMotorControl = (1 << 5), - QuickBoot = (1 << 6), -}; - -enum class Display { +ReflectableEnum(Display, RGB, SVideo, CompositeColour, CompositeMonochrome -}; +); -/*! - @returns An option list comprised of the standard names for all the options indicated by @c mask. -*/ -std::vector> standard_options(StandardOptions mask); - -/*! - Appends to @c selection_set a selection of @c selection for QuickLoadTape. -*/ -void append_quick_load_tape_selection(SelectionSet &selection_set, bool selection); - -/*! - Appends to @c selection_set a selection of @c selection for AutomaticTapeMotorControl. -*/ -void append_automatic_tape_motor_control_selection(SelectionSet &selection_set, bool selection); - -/*! - Appends to @c selection_set a selection of @c selection for DisplayRGBComposite. -*/ -void append_display_selection(SelectionSet &selection_set, Display selection); - -/*! - Appends to @c selection_set a selection of @c selection for QuickBoot. -*/ -void append_quick_boot_selection(SelectionSet &selection_set, bool selection); - -/*! - Attempts to discern a QuickLoadTape selection from @c selections_by_option. - - @param selections_by_option The user selections. - @param result The location to which the selection will be stored if found. - @returns @c true if a selection is found; @c false otherwise. -*/ -bool get_quick_load_tape(const SelectionSet &selections_by_option, bool &result); - -/*! - Attempts to discern an AutomaticTapeMotorControl selection from @c selections_by_option. - - @param selections_by_option The user selections. - @param result The location to which the selection will be stored if found. - @returns @c true if a selection is found; @c false otherwise. -*/ -bool get_automatic_tape_motor_control_selection(const SelectionSet &selections_by_option, bool &result); - -/*! - Attempts to discern a display RGB/composite selection from @c selections_by_option. - - @param selections_by_option The user selections. - @param result The location to which the selection will be stored if found. - @returns @c true if a selection is found; @c false otherwise. -*/ -bool get_display(const SelectionSet &selections_by_option, Display &result); - -/*! - Attempts to QuickBoot a QuickLoadTape selection from @c selections_by_option. - - @param selections_by_option The user selections. - @param result The location to which the selection will be stored if found. - @returns @c true if a selection is found; @c false otherwise. -*/ -bool get_quick_boot(const SelectionSet &selections_by_option, bool &result); } diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index ec6565d74..956dc62de 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -40,10 +40,14 @@ namespace AmstradCPC { -std::vector> get_options() { - return Configurable::standard_options( - Configurable::StandardOptions(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// Configurable::StandardOptions(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } /*! @@ -1114,28 +1118,34 @@ template class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return AmstradCPC::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::RGB); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::RGB); - return selection_set; + void set_options(const std::unique_ptr &options) { } +// std::vector> get_options() final { +// return AmstradCPC::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); +// return selection_set; +// } // MARK: - Joysticks const std::vector> &get_joysticks() final { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index e523a69c0..98eece1f5 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -9,17 +9,16 @@ #ifndef AmstradCPC_hpp #define AmstradCPC_hpp -#include "../../Configurable/Configurable.hpp" +#include "../../Reflection/Struct.h" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" #include -#include namespace AmstradCPC { /// @returns The options available for an Amstrad CPC. -std::vector> get_options(); +std::unique_ptr get_options(); /*! Models an Amstrad CPC. diff --git a/Machines/Apple/AppleII/AppleII.cpp b/Machines/Apple/AppleII/AppleII.cpp index 3c6c25c44..3c7fd9d3d 100644 --- a/Machines/Apple/AppleII/AppleII.cpp +++ b/Machines/Apple/AppleII/AppleII.cpp @@ -37,11 +37,14 @@ namespace Apple { namespace II { -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::DisplayCompositeMonochrome | Configurable::DisplayCompositeColour) - ); +std::unique_ptr get_options() { + return nullptr; } +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::DisplayCompositeMonochrome | Configurable::DisplayCompositeColour) +// ); +//} #define is_iie() ((model == Analyser::Static::AppleII::Target::Model::IIe) || (model == Analyser::Static::AppleII::Target::Model::EnhancedIIe)) @@ -866,26 +869,32 @@ template class ConcreteMachine: } // MARK:: Configuration options. - std::vector> get_options() final { - return Apple::II::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - return get_accurate_selections(); + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return Apple::II::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// return get_accurate_selections(); +// } // MARK: MediaTarget bool insert_media(const Analyser::Static::Media &media) final { diff --git a/Machines/Apple/AppleII/AppleII.hpp b/Machines/Apple/AppleII/AppleII.hpp index ad4fededd..647587c21 100644 --- a/Machines/Apple/AppleII/AppleII.hpp +++ b/Machines/Apple/AppleII/AppleII.hpp @@ -9,18 +9,17 @@ #ifndef AppleII_hpp #define AppleII_hpp -#include "../../../Configurable/Configurable.hpp" +#include "../../../Reflection/Struct.h" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" #include -#include namespace Apple { namespace II { /// @returns The options available for an Apple II. -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/Machines/Apple/Macintosh/Macintosh.cpp b/Machines/Apple/Macintosh/Macintosh.cpp index b9ceedf4c..59fb68239 100644 --- a/Machines/Apple/Macintosh/Macintosh.cpp +++ b/Machines/Apple/Macintosh/Macintosh.cpp @@ -21,6 +21,7 @@ #include "../../KeyboardMachine.hpp" #include "../../MediaTarget.hpp" #include "../../MouseMachine.hpp" +#include "../../../Configurable/Configurable.hpp" #include "../../../Inputs/QuadratureMouse/QuadratureMouse.hpp" #include "../../../Outputs/Log.hpp" @@ -56,10 +57,14 @@ constexpr int CLOCK_RATE = 7833600; namespace Apple { namespace Macintosh { -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::QuickBoot) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::QuickBoot) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } template class ConcreteMachine: @@ -522,35 +527,41 @@ template class ConcreteMachin } // MARK: - Configuration options. - std::vector> get_options() final { - return Apple::Macintosh::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - bool quick_boot; - if(Configurable::get_quick_boot(selections_by_option, quick_boot)) { - if(quick_boot) { - // Cf. Big Mess o' Wires' disassembly of the Mac Plus ROM, and the - // test at $E00. TODO: adapt as(/if?) necessary for other Macs. - ram_[0x02ae] = 0x40; - ram_[0x02af] = 0x00; - ram_[0x02b0] = 0x00; - ram_[0x02b1] = 0x00; - } - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_boot_selection(selection_set, false); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_boot_selection(selection_set, true); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return Apple::Macintosh::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// bool quick_boot; +// if(Configurable::get_quick_boot(selections_by_option, quick_boot)) { +// if(quick_boot) { +// // Cf. Big Mess o' Wires' disassembly of the Mac Plus ROM, and the +// // test at $E00. TODO: adapt as(/if?) necessary for other Macs. +// ram_[0x02ae] = 0x40; +// ram_[0x02af] = 0x00; +// ram_[0x02b0] = 0x00; +// ram_[0x02b1] = 0x00; +// } +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_boot_selection(selection_set, false); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_boot_selection(selection_set, true); +// return selection_set; +// } private: void set_component_prefers_clocking(ClockingHint::Source *component, ClockingHint::Preference clocking) final { diff --git a/Machines/Apple/Macintosh/Macintosh.hpp b/Machines/Apple/Macintosh/Macintosh.hpp index 58623ea7c..e29665d10 100644 --- a/Machines/Apple/Macintosh/Macintosh.hpp +++ b/Machines/Apple/Macintosh/Macintosh.hpp @@ -9,14 +9,16 @@ #ifndef Macintosh_hpp #define Macintosh_hpp -#include "../../../Configurable/Configurable.hpp" +#include "../../../Reflection/Struct.h" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" +#include + namespace Apple { namespace Macintosh { -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/Machines/Atari/ST/AtariST.cpp b/Machines/Atari/ST/AtariST.cpp index 7be6a7747..afcc3182c 100644 --- a/Machines/Atari/ST/AtariST.cpp +++ b/Machines/Atari/ST/AtariST.cpp @@ -42,10 +42,14 @@ namespace Atari { namespace ST { -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } constexpr int CLOCK_RATE = 8021247; @@ -678,28 +682,30 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return Atari::ST::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::RGB); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); +// return selection_set; +// } }; } diff --git a/Machines/Atari/ST/AtariST.hpp b/Machines/Atari/ST/AtariST.hpp index 6345620cd..abd71e3ed 100644 --- a/Machines/Atari/ST/AtariST.hpp +++ b/Machines/Atari/ST/AtariST.hpp @@ -9,15 +9,17 @@ #ifndef AtariST_hpp #define AtariST_hpp -#include "../../../Configurable/Configurable.hpp" +#include "../../../Reflection/Struct.h" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" +#include + namespace Atari { namespace ST { /// @returns The options available for an Atari ST. -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/Machines/ColecoVision/ColecoVision.cpp b/Machines/ColecoVision/ColecoVision.cpp index 02aacc6df..0d5c5847b 100644 --- a/Machines/ColecoVision/ColecoVision.cpp +++ b/Machines/ColecoVision/ColecoVision.cpp @@ -16,6 +16,7 @@ #include "../CRTMachine.hpp" #include "../JoystickMachine.hpp" +#include "../../Configurable/Configurable.hpp" #include "../../Configurable/StandardOptions.hpp" #include "../../ClockReceiver/ForceInline.hpp" @@ -33,10 +34,14 @@ constexpr int sn76489_divider = 2; namespace Coleco { namespace Vision { -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::DisplaySVideo | Configurable::DisplayCompositeColour) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::DisplaySVideo | Configurable::DisplayCompositeColour) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } class Joystick: public Inputs::ConcreteJoystick { @@ -368,28 +373,34 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return Coleco::Vision::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::SVideo); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return Coleco::Vision::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::SVideo); +// return selection_set; +// } private: inline void page_megacart(uint16_t address) { diff --git a/Machines/ColecoVision/ColecoVision.hpp b/Machines/ColecoVision/ColecoVision.hpp index 750e5bcbe..c245981a0 100644 --- a/Machines/ColecoVision/ColecoVision.hpp +++ b/Machines/ColecoVision/ColecoVision.hpp @@ -9,14 +9,16 @@ #ifndef ColecoVision_hpp #define ColecoVision_hpp -#include "../../Configurable/Configurable.hpp" +#include "../../Reflection/Struct.h" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" +#include + namespace Coleco { namespace Vision { -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/Machines/Commodore/Vic-20/Vic20.cpp b/Machines/Commodore/Vic-20/Vic20.cpp index 94e328c00..b07c522b5 100644 --- a/Machines/Commodore/Vic-20/Vic20.cpp +++ b/Machines/Commodore/Vic-20/Vic20.cpp @@ -49,11 +49,14 @@ enum ROMSlot { Drive }; -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::DisplaySVideo | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) - ); +std::unique_ptr get_options() { + return nullptr; } +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::DisplaySVideo | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) +// ); +//} enum JoystickInput { Up = 0x04, @@ -679,36 +682,42 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return Commodore::Vic20::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - bool quickload; - if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { - allow_fast_tape_hack_ = quickload; - set_use_fast_tape(); - } - - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, false); - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, true); - Configurable::append_display_selection(selection_set, Configurable::Display::SVideo); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return Commodore::Vic20::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// bool quickload; +// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { +// allow_fast_tape_hack_ = quickload; +// set_use_fast_tape(); +// } +// +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, false); +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, true); +// Configurable::append_display_selection(selection_set, Configurable::Display::SVideo); +// return selection_set; +// } void set_component_prefers_clocking(ClockingHint::Source *component, ClockingHint::Preference clocking) final { tape_is_sleeping_ = clocking == ClockingHint::Preference::None; diff --git a/Machines/Commodore/Vic-20/Vic20.hpp b/Machines/Commodore/Vic-20/Vic20.hpp index ea5e7d0b7..8c442809f 100644 --- a/Machines/Commodore/Vic-20/Vic20.hpp +++ b/Machines/Commodore/Vic-20/Vic20.hpp @@ -9,18 +9,17 @@ #ifndef Vic20_hpp #define Vic20_hpp -#include "../../../Configurable/Configurable.hpp" +#include "../../../Reflection/Struct.h" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" #include -#include namespace Commodore { namespace Vic20 { /// @returns The options available for a Vic-20. -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/Machines/Electron/Electron.cpp b/Machines/Electron/Electron.cpp index 2c1fc3ac5..7546d98ce 100644 --- a/Machines/Electron/Electron.cpp +++ b/Machines/Electron/Electron.cpp @@ -12,6 +12,7 @@ #include "../MediaTarget.hpp" #include "../CRTMachine.hpp" #include "../KeyboardMachine.hpp" +#include "../../Configurable/Configurable.hpp" #include "../../ClockReceiver/ClockReceiver.hpp" #include "../../ClockReceiver/ForceInline.hpp" @@ -32,12 +33,17 @@ namespace Electron { -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } + class ConcreteMachine: public Machine, public CRTMachine::Machine, @@ -448,36 +454,42 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return Electron::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - bool quickload; - if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { - allow_fast_tape_hack_ = quickload; - set_use_fast_tape_hack(); - } - - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, false); - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, true); - Configurable::append_display_selection(selection_set, Configurable::Display::RGB); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return Electron::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// bool quickload; +// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { +// allow_fast_tape_hack_ = quickload; +// set_use_fast_tape_hack(); +// } +// +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, false); +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, true); +// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); +// return selection_set; +// } // MARK: - Activity Source void set_activity_observer(Activity::Observer *observer) final { diff --git a/Machines/Electron/Electron.hpp b/Machines/Electron/Electron.hpp index 90cf12993..b6c712c5a 100644 --- a/Machines/Electron/Electron.hpp +++ b/Machines/Electron/Electron.hpp @@ -9,18 +9,16 @@ #ifndef Electron_hpp #define Electron_hpp -#include "../../Configurable/Configurable.hpp" +#include "../../Reflection/Struct.h" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" -#include #include -#include namespace Electron { /// @returns The options available for an Electron. -std::vector> get_options(); +std::unique_ptr get_options(); /*! @abstract Represents an Acorn Electron. diff --git a/Machines/MSX/MSX.cpp b/Machines/MSX/MSX.cpp index 9f8fccb0d..38313ea3e 100644 --- a/Machines/MSX/MSX.cpp +++ b/Machines/MSX/MSX.cpp @@ -37,6 +37,7 @@ #include "../JoystickMachine.hpp" #include "../MediaTarget.hpp" #include "../KeyboardMachine.hpp" +#include "../../Configurable/Configurable.hpp" #include "../../Outputs/Log.hpp" #include "../../Outputs/Speaker/Implementation/CompoundSource.hpp" @@ -51,10 +52,14 @@ namespace MSX { -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::DisplayRGB | Configurable::DisplaySVideo | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::DisplayRGB | Configurable::DisplaySVideo | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } class AYPortHandler: public GI::AY38910::PortHandler { @@ -648,36 +653,42 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return MSX::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - bool quickload; - if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { - allow_fast_tape_ = quickload; - set_use_fast_tape(); - } - - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, false); - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, true); - Configurable::append_display_selection(selection_set, Configurable::Display::RGB); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return MSX::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// bool quickload; +// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { +// allow_fast_tape_ = quickload; +// set_use_fast_tape(); +// } +// +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, false); +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, true); +// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); +// return selection_set; +// } // MARK: - Sleeper void set_component_prefers_clocking(ClockingHint::Source *component, ClockingHint::Preference clocking) final { diff --git a/Machines/MSX/MSX.hpp b/Machines/MSX/MSX.hpp index 1088786e2..9f68a3645 100644 --- a/Machines/MSX/MSX.hpp +++ b/Machines/MSX/MSX.hpp @@ -9,16 +9,15 @@ #ifndef MSX_hpp #define MSX_hpp -#include "../../Configurable/Configurable.hpp" +#include "../../Reflection/Struct.h" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" #include -#include namespace MSX { -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/Machines/MasterSystem/MasterSystem.cpp b/Machines/MasterSystem/MasterSystem.cpp index ea4d46cfb..5fb0cb70b 100644 --- a/Machines/MasterSystem/MasterSystem.cpp +++ b/Machines/MasterSystem/MasterSystem.cpp @@ -16,6 +16,7 @@ #include "../CRTMachine.hpp" #include "../JoystickMachine.hpp" #include "../KeyboardMachine.hpp" +#include "../../Configurable/Configurable.hpp" #include "../../ClockReceiver/ForceInline.hpp" #include "../../ClockReceiver/JustInTime.hpp" @@ -37,10 +38,14 @@ constexpr int sn76489_divider = 2; namespace Sega { namespace MasterSystem { -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } class Joystick: public Inputs::ConcreteJoystick { @@ -373,28 +378,34 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return Sega::MasterSystem::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_display_selection(selection_set, Configurable::Display::RGB); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return Sega::MasterSystem::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); +// return selection_set; +// } private: static TI::TMS::Personality tms_personality_for_model(Analyser::Static::Sega::Target::Model model) { diff --git a/Machines/MasterSystem/MasterSystem.hpp b/Machines/MasterSystem/MasterSystem.hpp index b192dfae7..c63228490 100644 --- a/Machines/MasterSystem/MasterSystem.hpp +++ b/Machines/MasterSystem/MasterSystem.hpp @@ -9,14 +9,16 @@ #ifndef MasterSystem_hpp #define MasterSystem_hpp -#include "../../Configurable/Configurable.hpp" +#include "../../Reflection/Struct.h" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" +#include + namespace Sega { namespace MasterSystem { -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/Machines/Oric/Oric.cpp b/Machines/Oric/Oric.cpp index 21d54f189..7417e6ef2 100644 --- a/Machines/Oric/Oric.cpp +++ b/Machines/Oric/Oric.cpp @@ -50,14 +50,18 @@ enum ROM { BASIC10 = 0, BASIC11, Microdisc, Colour }; -std::vector> get_options() { - return Configurable::standard_options( - static_cast( - Configurable::DisplayRGB | - Configurable::DisplayCompositeColour | - Configurable::DisplayCompositeMonochrome | - Configurable::QuickLoadTape) - ); +//std::vector> get_options() { +// return Configurable::standard_options( +// static_cast( +// Configurable::DisplayRGB | +// Configurable::DisplayCompositeColour | +// Configurable::DisplayCompositeMonochrome | +// Configurable::QuickLoadTape) +// ); +//} + +std::unique_ptr get_options() { + return nullptr; } /*! @@ -623,35 +627,41 @@ template class Co } // MARK: - Configuration options. - std::vector> get_options() final { - return Oric::get_options(); + std::unique_ptr get_options(OptionsType type) final { + return nullptr; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - bool quickload; - if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { - set_use_fast_tape_hack(quickload); - } - - Configurable::Display display; - if(Configurable::get_display(selections_by_option, display)) { - set_video_signal_configurable(display); - } - } - - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, false); - Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, true); - Configurable::append_display_selection(selection_set, Configurable::Display::RGB); - return selection_set; + void set_options(const std::unique_ptr &options) final { } +// std::vector> get_options() final { +// return Oric::get_options(); +// } +// +// void set_selections(const Configurable::SelectionSet &selections_by_option) final { +// bool quickload; +// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { +// set_use_fast_tape_hack(quickload); +// } +// +// Configurable::Display display; +// if(Configurable::get_display(selections_by_option, display)) { +// set_video_signal_configurable(display); +// } +// } +// +// Configurable::SelectionSet get_accurate_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, false); +// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); +// return selection_set; +// } +// +// Configurable::SelectionSet get_user_friendly_selections() final { +// Configurable::SelectionSet selection_set; +// Configurable::append_quick_load_tape_selection(selection_set, true); +// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); +// return selection_set; +// } void set_activity_observer(Activity::Observer *observer) final { switch(disk_interface) { diff --git a/Machines/Oric/Oric.hpp b/Machines/Oric/Oric.hpp index 8e3977bdf..57c6fdba3 100644 --- a/Machines/Oric/Oric.hpp +++ b/Machines/Oric/Oric.hpp @@ -9,14 +9,16 @@ #ifndef Oric_hpp #define Oric_hpp -#include "../../Configurable/Configurable.hpp" +#include "../../Reflection/Struct.h" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" +#include + namespace Oric { /// @returns The options available for an Oric. -std::vector> get_options(); +std::unique_ptr get_options(); /*! Models an Oric 1/Atmos with or without a Microdisc. diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 638cfbf85..baa9e515b 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -173,19 +173,19 @@ std::vector Machine::AllMachines(bool meaningful_without_media_only return result; } -std::map>> Machine::AllOptionsByMachineName() { - std::map>> options; +std::map> Machine::AllOptionsByMachineName() { + std::map> options; - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AmstradCPC), AmstradCPC::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AppleII), Apple::II::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AtariST), Atari::ST::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), Coleco::Vision::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Electron), Electron::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MasterSystem), Sega::MasterSystem::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MSX), MSX::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Oric), Oric::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AmstradCPC), AmstradCPC::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AppleII), Apple::II::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AtariST), Atari::ST::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), Coleco::Vision::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Electron), Electron::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MasterSystem), Sega::MasterSystem::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MSX), MSX::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Oric), Oric::get_options())); +// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ZX8081), ZX8081::get_options())); return options; diff --git a/Machines/Utility/MachineForTarget.hpp b/Machines/Utility/MachineForTarget.hpp index a51508f47..c1228d610 100644 --- a/Machines/Utility/MachineForTarget.hpp +++ b/Machines/Utility/MachineForTarget.hpp @@ -77,7 +77,7 @@ std::vector AllMachines(bool meaningful_without_media_only, bool lo /*! Returns a map from long machine name to the list of options that machine exposes, for all machines. */ -std::map>> AllOptionsByMachineName(); +std::map> AllOptionsByMachineName(); /*! Returns a map from long machine name to appropriate instances of Target for the machine. diff --git a/Machines/ZX8081/ZX8081.cpp b/Machines/ZX8081/ZX8081.cpp index 9a5be9e08..4a47e40b6 100644 --- a/Machines/ZX8081/ZX8081.cpp +++ b/Machines/ZX8081/ZX8081.cpp @@ -18,7 +18,6 @@ #include "../../Storage/Tape/Parsers/ZX8081.hpp" #include "../../ClockReceiver/ForceInline.hpp" -#include "../../Configurable/StandardOptions.hpp" #include "../Utility/MemoryFuzzer.hpp" #include "../Utility/Typer.hpp" @@ -51,10 +50,20 @@ enum ROMType: uint8_t { ZX80 = 0, ZX81 }; -std::vector> get_options() { - return Configurable::standard_options( - static_cast(Configurable::AutomaticTapeMotorControl | Configurable::QuickLoadTape) - ); +struct Options: public Reflection::StructImpl { + bool automatic_tape_motor_control = true; + bool quickload = true; + + Options() { + if(needs_declare()) { + DeclareField(automatic_tape_motor_control); + DeclareField(quickload); + } + } +}; + +std::unique_ptr get_options() { + return std::make_unique(); } template class ConcreteMachine: @@ -413,37 +422,33 @@ template class ConcreteMachine: } // MARK: - Configuration options. - std::vector> get_options() final { - return ZX8081::get_options(); + std::unique_ptr get_options(OptionsType type) final { + auto options = std::make_unique(); + switch(type) { + case OptionsType::Current: + options->automatic_tape_motor_control = use_automatic_tape_motor_control_; + options->quickload = allow_fast_tape_hack_; + break; + case OptionsType::Accurate: + case OptionsType::UserFriendly: + options->automatic_tape_motor_control = + options->quickload = type == OptionsType::UserFriendly; + break; + } + return options; } - void set_selections(const Configurable::SelectionSet &selections_by_option) final { - bool quickload; - if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { - allow_fast_tape_hack_ = quickload; + void set_options(const std::unique_ptr &options) { + if(Reflection::get(*options, "quickload", allow_fast_tape_hack_)) { set_use_fast_tape(); } bool autotapemotor; - if(Configurable::get_automatic_tape_motor_control_selection(selections_by_option, autotapemotor)) { + if(Reflection::get(*options, "automatic_tape_motor_control", autotapemotor)) { set_use_automatic_tape_motor_control(autotapemotor); } } - Configurable::SelectionSet get_accurate_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, false); - Configurable::append_automatic_tape_motor_control_selection(selection_set, false); - return selection_set; - } - - Configurable::SelectionSet get_user_friendly_selections() final { - Configurable::SelectionSet selection_set; - Configurable::append_quick_load_tape_selection(selection_set, true); - Configurable::append_automatic_tape_motor_control_selection(selection_set, true); - return selection_set; - } - private: CPU::Z80::Processor z80_; Video video_; diff --git a/Machines/ZX8081/ZX8081.hpp b/Machines/ZX8081/ZX8081.hpp index 301d8529a..f35d2b9c7 100644 --- a/Machines/ZX8081/ZX8081.hpp +++ b/Machines/ZX8081/ZX8081.hpp @@ -9,14 +9,16 @@ #ifndef ZX8081_hpp #define ZX8081_hpp -#include "../../Configurable/Configurable.hpp" +#include "../../Reflection/Struct.h" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" +#include + namespace ZX8081 { /// @returns The options available for a ZX80 or ZX81. -std::vector> get_options(); +std::unique_ptr get_options(); class Machine { public: diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 407c7943d..2e5b8d107 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -101,8 +101,6 @@ 4B055AEF1FAE9BF00060FFFF /* Typer.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B2B3A471F9B8FA70062DABF /* Typer.cpp */; }; 4B055AF11FAE9C160060FFFF /* Accelerate.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4BC76E6A1C98F43700E6EF73 /* Accelerate.framework */; }; 4B055AF21FAE9C1C0060FFFF /* OpenGL.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4B055AF01FAE9C080060FFFF /* OpenGL.framework */; }; - 4B07835A1FC11D10001D12BB /* Configurable.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B0783591FC11D10001D12BB /* Configurable.cpp */; }; - 4B07835B1FC11D42001D12BB /* Configurable.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B0783591FC11D10001D12BB /* Configurable.cpp */; }; 4B08A2751EE35D56008B7065 /* Z80InterruptTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4B08A2741EE35D56008B7065 /* Z80InterruptTests.swift */; }; 4B08A2781EE39306008B7065 /* TestMachine.mm in Sources */ = {isa = PBXBuildFile; fileRef = 4B08A2771EE39306008B7065 /* TestMachine.mm */; }; 4B08A56920D72BEF0016CE5A /* Activity.xib in Resources */ = {isa = PBXBuildFile; fileRef = 4B08A56720D72BEF0016CE5A /* Activity.xib */; }; @@ -364,7 +362,6 @@ 4B778F5C23A5F3070000D260 /* MSX.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B0E61051FF34737002A9DBD /* MSX.cpp */; }; 4B778F5D23A5F3230000D260 /* Commodore.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B8805F21DCFD22A003085B1 /* Commodore.cpp */; }; 4B778F5E23A5F3230000D260 /* Oric.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B8805F91DCFF807003085B1 /* Oric.cpp */; }; - 4B778F5F23A5F3300000D260 /* Configurable.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B0783591FC11D10001D12BB /* Configurable.cpp */; }; 4B778F6023A5F3460000D260 /* Disk.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B8944EC201967B4007DE474 /* Disk.cpp */; }; 4B778F6123A5F3560000D260 /* Disk.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B8944FC201967B4007DE474 /* Disk.cpp */; }; 4B778F6223A5F35F0000D260 /* File.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 4B894500201967B4007DE474 /* File.cpp */; }; @@ -916,7 +913,6 @@ 4B055ABE1FAE98000060FFFF /* MachineForTarget.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = MachineForTarget.cpp; sourceTree = ""; }; 4B055ABF1FAE98000060FFFF /* MachineForTarget.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; path = MachineForTarget.hpp; sourceTree = ""; }; 4B055AF01FAE9C080060FFFF /* OpenGL.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = OpenGL.framework; path = System/Library/Frameworks/OpenGL.framework; sourceTree = SDKROOT; }; - 4B0783591FC11D10001D12BB /* Configurable.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; path = Configurable.cpp; sourceTree = ""; }; 4B08A2741EE35D56008B7065 /* Z80InterruptTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Z80InterruptTests.swift; sourceTree = ""; }; 4B08A2761EE39306008B7065 /* TestMachine.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = TestMachine.h; sourceTree = ""; }; 4B08A2771EE39306008B7065 /* TestMachine.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = TestMachine.mm; sourceTree = ""; }; @@ -2131,7 +2127,6 @@ isa = PBXGroup; children = ( 4B31B88F1FBFBCD800C140D5 /* Configurable.hpp */, - 4B0783591FC11D10001D12BB /* Configurable.cpp */, 4BFE7B851FC39BF100160B38 /* StandardOptions.cpp */, 4BFE7B861FC39BF100160B38 /* StandardOptions.hpp */, ); @@ -4443,7 +4438,6 @@ 4B055A7E1FAE84AA0060FFFF /* main.cpp in Sources */, 4B894537201967B4007DE474 /* Z80.cpp in Sources */, 4B055A9F1FAE85DA0060FFFF /* HFE.cpp in Sources */, - 4B07835B1FC11D42001D12BB /* Configurable.cpp in Sources */, 4BD191F52191180E0042E144 /* ScanTarget.cpp in Sources */, 4B055AEC1FAE9BA20060FFFF /* Z80Base.cpp in Sources */, 4B0F94FF208C1A1600FE41D9 /* NIB.cpp in Sources */, @@ -4502,7 +4496,6 @@ 4B894538201967B4007DE474 /* Tape.cpp in Sources */, 4B54C0CB1F8D92590050900F /* Keyboard.cpp in Sources */, 4BEA525E1DF33323007E74F2 /* Tape.cpp in Sources */, - 4B07835A1FC11D10001D12BB /* Configurable.cpp in Sources */, 4B2BF19623E10F0100C3AD60 /* CSHighPrecisionTimer.m in Sources */, 4B8334951F5E25B60097E338 /* C1540.cpp in Sources */, 4B89453C201967B4007DE474 /* StaticAnalyser.cpp in Sources */, @@ -4777,7 +4770,6 @@ 4B778F5623A5F2AF0000D260 /* CPM.cpp in Sources */, 4B778F1C23A5ED3F0000D260 /* TimedEventLoop.cpp in Sources */, 4B3BA0D01D318B44005DD7A7 /* MOS6532Bridge.mm in Sources */, - 4B778F5F23A5F3300000D260 /* Configurable.cpp in Sources */, 4B778F3823A5F11C0000D260 /* SegmentParser.cpp in Sources */, 4B778F0723A5EC150000D260 /* CommodoreTAP.cpp in Sources */, 4B778F4123A5F19A0000D260 /* MemoryPacker.cpp in Sources */, @@ -4981,7 +4973,7 @@ "$(USER_LIBRARY_DIR)/Frameworks", ); GCC_C_LANGUAGE_STANDARD = gnu11; - MACOSX_DEPLOYMENT_TARGET = 10.14; + MACOSX_DEPLOYMENT_TARGET = 10.10; PRODUCT_NAME = "$(TARGET_NAME)"; }; name = Debug; @@ -5003,7 +4995,7 @@ ); GCC_C_LANGUAGE_STANDARD = gnu11; GCC_OPTIMIZATION_LEVEL = 2; - MACOSX_DEPLOYMENT_TARGET = 10.14; + MACOSX_DEPLOYMENT_TARGET = 10.10; PRODUCT_NAME = "$(TARGET_NAME)"; }; name = Release; diff --git a/Reflection/Struct.cpp b/Reflection/Struct.cpp index cbe533987..b3fb8bdfd 100644 --- a/Reflection/Struct.cpp +++ b/Reflection/Struct.cpp @@ -8,6 +8,8 @@ #include "Struct.h" +// MARK: - Setters + template <> bool Reflection::set(Struct &target, const std::string &name, int value) { const auto target_type = target.type_of(name); if(!target_type) return false; @@ -52,6 +54,26 @@ template <> bool Reflection::set(Struct &target, const std::string &name, const return set(target, name, string); } +// MARK: - Fuzzy setter + bool Reflection::fuzzy_set(Struct &target, const std::string &name, const std::string &value) { return false; } + +// MARK: - Getters + +template bool Reflection::get(Struct &target, const std::string &name, Type &value) { + return false; +} + +template <> bool Reflection::get(Struct &target, const std::string &name, bool &value) { + const auto target_type = target.type_of(name); + if(!target_type) return false; + + if(*target_type == typeid(bool)) { + value = *reinterpret_cast(target.get(name)); + return true; + } + + return false; +} diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 256d53876..4b1af1ea4 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -71,6 +71,17 @@ template <> bool set(Struct &target, const std::string &name, const char *value) */ bool fuzzy_set(Struct &target, const std::string &name, const std::string &value); + +/*! + Attempts to get the property @c name to @c value ; will perform limited type conversions. + + @returns @c true if the property was successfully read; @c false otherwise. +*/ +template bool get(Struct &target, const std::string &name, Type &value); + +template <> bool get(Struct &target, const std::string &name, bool &value); + + // TODO: move this elsewhere. It's just a sketch anyway. struct Serialisable { /// Serialises this object, appending it to @c target. From 1d40aa687e98449b7cbed2fa8ec5cd48848da072 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Sun, 15 Mar 2020 23:52:24 -0400 Subject: [PATCH 28/48] Adds necessary include for unique_ptr. --- Configurable/Configurable.hpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/Configurable/Configurable.hpp b/Configurable/Configurable.hpp index 9876e02d4..e2ed66628 100644 --- a/Configurable/Configurable.hpp +++ b/Configurable/Configurable.hpp @@ -11,6 +11,8 @@ #include "../Reflection/Struct.h" +#include + namespace Configurable { /*! From 394ee61c78d576831c73ed837df6854d3331aa8a Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Mon, 16 Mar 2020 23:25:05 -0400 Subject: [PATCH 29/48] Starts a switch to reflectable-style runtime options. The Amstrad CPC and ZX80/81 have made the jump so far, subject to caveats. The macOS build is unlikely currently to work properly. --- .../Implementation/MultiConfigurable.cpp | 2 +- .../Implementation/MultiConfigurable.hpp | 2 +- Components/6560/6560.hpp | 1 + Configurable/Configurable.hpp | 29 ++++---- Machines/AmstradCPC/AmstradCPC.cpp | 52 +++++--------- Machines/AmstradCPC/AmstradCPC.hpp | 22 ++++-- Machines/Apple/AppleII/AppleII.cpp | 2 +- Machines/Apple/Macintosh/Macintosh.cpp | 2 +- Machines/Atari/ST/AtariST.cpp | 2 +- Machines/CRTMachine.hpp | 20 +++++- Machines/ColecoVision/ColecoVision.cpp | 2 +- Machines/Commodore/Vic-20/Vic20.cpp | 2 +- Machines/Electron/Electron.cpp | 2 +- Machines/MSX/MSX.cpp | 2 +- Machines/MasterSystem/MasterSystem.cpp | 2 +- Machines/Oric/Oric.cpp | 2 +- Machines/Utility/MachineForTarget.cpp | 10 ++- Machines/ZX8081/ZX8081.cpp | 47 +++---------- Machines/ZX8081/ZX8081.hpp | 24 +++++-- .../xcschemes/Clock Signal Kiosk.xcscheme | 2 +- OSBindings/SDL/main.cpp | 67 ++++++++++--------- Outputs/CRT/CRT.cpp | 4 ++ Outputs/CRT/CRT.hpp | 3 + Reflection/Enum.h | 1 + 24 files changed, 163 insertions(+), 141 deletions(-) diff --git a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp index c77ff6de0..c151d52a7 100644 --- a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp +++ b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp @@ -22,7 +22,7 @@ MultiConfigurable::MultiConfigurable(const std::vector &options) { } -std::unique_ptr MultiConfigurable::get_options(OptionsType type) { +std::unique_ptr MultiConfigurable::get_options() { // TODO: this'll need to mash options together, maybe? Or just take the front? return nullptr; } diff --git a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp index 7d4672987..88c264e52 100644 --- a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp +++ b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp @@ -30,7 +30,7 @@ class MultiConfigurable: public Configurable::Device { // Below is the standard Configurable::Device interface; see there for documentation. void set_options(const std::unique_ptr &options) final; - std::unique_ptr get_options(OptionsType type) final; + std::unique_ptr get_options() final; // std::vector> get_options() final; // void set_selections(const Configurable::SelectionSet &selection_by_option) final; // Configurable::SelectionSet get_accurate_selections() final; diff --git a/Components/6560/6560.hpp b/Components/6560/6560.hpp index 05e1fb2e0..0f7013dc9 100644 --- a/Components/6560/6560.hpp +++ b/Components/6560/6560.hpp @@ -87,6 +87,7 @@ template class MOS6560 { void set_scan_target(Outputs::Display::ScanTarget *scan_target) { crt_.set_scan_target(scan_target); } Outputs::Display::ScanStatus get_scaled_scan_status() const { return crt_.get_scaled_scan_status() / 4.0f; } void set_display_type(Outputs::Display::DisplayType display_type) { crt_.set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() { return crt_.get_display_type(); } Outputs::Speaker::Speaker *get_speaker() { return &speaker_; } void set_high_frequency_cutoff(float cutoff) { diff --git a/Configurable/Configurable.hpp b/Configurable/Configurable.hpp index e2ed66628..cbb037ff3 100644 --- a/Configurable/Configurable.hpp +++ b/Configurable/Configurable.hpp @@ -17,27 +17,28 @@ namespace Configurable { /*! A Configurable::Device provides a reflective struct listing the available runtime options for this machine. - It can provide it populated with 'accurate' options, 'user-friendly' options or just whatever the user - currently has selected. + You can ordinarily either get or set a machine's current options, or else construct a new instance of + its options with one of the OptionsTypes defined below. +*/ +struct Device { + /// Sets the current options. The caller must ensure that the object passed in is either an instance of the machine's + /// Options struct, or else was previously returned by get_options. + virtual void set_options(const std::unique_ptr &options) = 0; + /// @returns An options object + virtual std::unique_ptr get_options() = 0; +}; + +/*! 'Accurate' options should correspond to the way that this device was usually used during its lifespan. E.g. a ColecoVision might accurately be given composite output. 'User-friendly' options should be more like those that a user today might most expect from an emulator. E.g. the ColecoVision might bump itself up to S-Video output. */ -struct Device { - /// Sets the current options. - virtual void set_options(const std::unique_ptr &options) = 0; - - enum class OptionsType { - Current, - Accurate, - UserFriendly - }; - - /// @returns Options of type @c type. - virtual std::unique_ptr get_options(OptionsType type) = 0; +enum class OptionsType { + Accurate, + UserFriendly }; } diff --git a/Machines/AmstradCPC/AmstradCPC.cpp b/Machines/AmstradCPC/AmstradCPC.cpp index 956dc62de..85910387b 100644 --- a/Machines/AmstradCPC/AmstradCPC.cpp +++ b/Machines/AmstradCPC/AmstradCPC.cpp @@ -40,16 +40,6 @@ namespace AmstradCPC { -//std::vector> get_options() { -// return Configurable::standard_options( -// Configurable::StandardOptions(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) -// ); -//} - -std::unique_ptr get_options() { - return nullptr; -} - /*! 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 @@ -358,6 +348,11 @@ class CRTCBusHandler { crt_.set_display_type(display_type); } + /// Gets the type of display. + Outputs::Display::DisplayType get_display_type() { + return crt_.get_display_type(); + } + /*! 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". @@ -1049,6 +1044,11 @@ template class ConcreteMachine: crtc_bus_handler_.set_display_type(display_type); } + /// A CRTMachine function; gets the output display type. + Outputs::Display::DisplayType get_display_type() { + return crtc_bus_handler_.get_display_type(); + } + /// @returns the speaker in use. Outputs::Speaker::Speaker *get_speaker() final { return ay_.get_speaker(); @@ -1118,34 +1118,16 @@ template class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { - return nullptr; + std::unique_ptr get_options() final { + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + return options; } - void set_options(const std::unique_ptr &options) { + void set_options(const std::unique_ptr &str) { + const auto options = dynamic_cast(str.get()); + set_video_signal_configurable(options->output); } -// std::vector> get_options() final { -// return AmstradCPC::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); -// return selection_set; -// } // MARK: - Joysticks const std::vector> &get_joysticks() final { diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 98eece1f5..8e8e843c7 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -9,7 +9,8 @@ #ifndef AmstradCPC_hpp #define AmstradCPC_hpp -#include "../../Reflection/Struct.h" +#include "../../Configurable/Configurable.hpp" +#include "../../Configurable/StandardOptions.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -17,9 +18,6 @@ namespace AmstradCPC { -/// @returns The options available for an Amstrad CPC. -std::unique_ptr get_options(); - /*! Models an Amstrad CPC. */ @@ -29,6 +27,22 @@ class Machine { /// Creates and returns an Amstrad CPC. static Machine *AmstradCPC(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + /// Defines the runtime options available for a ZX80/81. + class Options: public Reflection::StructImpl { + public: + Configurable::Display output = Configurable::Display::RGB; + + Options(Configurable::OptionsType type) { + // Declare fields if necessary. + if(needs_declare()) { + DeclareField(output); + AnnounceEnumNS(Configurable, Display); + // TODO: some way to set limited enum support for a struct? + // In this case, to support only RGB and CompositeColour. + } + } + }; }; } diff --git a/Machines/Apple/AppleII/AppleII.cpp b/Machines/Apple/AppleII/AppleII.cpp index 3c7fd9d3d..f1a6db001 100644 --- a/Machines/Apple/AppleII/AppleII.cpp +++ b/Machines/Apple/AppleII/AppleII.cpp @@ -869,7 +869,7 @@ template class ConcreteMachine: } // MARK:: Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/Apple/Macintosh/Macintosh.cpp b/Machines/Apple/Macintosh/Macintosh.cpp index 59fb68239..c71fcf103 100644 --- a/Machines/Apple/Macintosh/Macintosh.cpp +++ b/Machines/Apple/Macintosh/Macintosh.cpp @@ -527,7 +527,7 @@ template class ConcreteMachin } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/Atari/ST/AtariST.cpp b/Machines/Atari/ST/AtariST.cpp index afcc3182c..6008ad075 100644 --- a/Machines/Atari/ST/AtariST.cpp +++ b/Machines/Atari/ST/AtariST.cpp @@ -682,7 +682,7 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/CRTMachine.hpp b/Machines/CRTMachine.hpp index 1e57575c1..fc86a7379 100644 --- a/Machines/CRTMachine.hpp +++ b/Machines/CRTMachine.hpp @@ -181,10 +181,28 @@ class Machine { } /*! - Forwards the video signal to the target returned by get_crt(). + Maps back from Outputs::Display::VideoSignal to Configurable::Display, + calling @c get_display_type for the input. + */ + Configurable::Display get_video_signal_configurable() { + switch(get_display_type()) { + default: + case Outputs::Display::DisplayType::RGB: return Configurable::Display::RGB; + case Outputs::Display::DisplayType::SVideo: return Configurable::Display::SVideo; + case Outputs::Display::DisplayType::CompositeColour: return Configurable::Display::CompositeColour; + case Outputs::Display::DisplayType::CompositeMonochrome: return Configurable::Display::CompositeMonochrome; + } + } + + /*! + Sets the display type. */ virtual void set_display_type(Outputs::Display::DisplayType display_type) {} + /*! + Gets the display type. + */ + virtual Outputs::Display::DisplayType get_display_type() { return Outputs::Display::DisplayType::RGB; } private: double clock_rate_ = 1.0; diff --git a/Machines/ColecoVision/ColecoVision.cpp b/Machines/ColecoVision/ColecoVision.cpp index 0d5c5847b..45b875ffc 100644 --- a/Machines/ColecoVision/ColecoVision.cpp +++ b/Machines/ColecoVision/ColecoVision.cpp @@ -373,7 +373,7 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/Commodore/Vic-20/Vic20.cpp b/Machines/Commodore/Vic-20/Vic20.cpp index b07c522b5..2984544b6 100644 --- a/Machines/Commodore/Vic-20/Vic20.cpp +++ b/Machines/Commodore/Vic-20/Vic20.cpp @@ -682,7 +682,7 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/Electron/Electron.cpp b/Machines/Electron/Electron.cpp index 7546d98ce..2f67b3c12 100644 --- a/Machines/Electron/Electron.cpp +++ b/Machines/Electron/Electron.cpp @@ -454,7 +454,7 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/MSX/MSX.cpp b/Machines/MSX/MSX.cpp index 38313ea3e..0ca91006e 100644 --- a/Machines/MSX/MSX.cpp +++ b/Machines/MSX/MSX.cpp @@ -653,7 +653,7 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/MasterSystem/MasterSystem.cpp b/Machines/MasterSystem/MasterSystem.cpp index 5fb0cb70b..614dc7d0f 100644 --- a/Machines/MasterSystem/MasterSystem.cpp +++ b/Machines/MasterSystem/MasterSystem.cpp @@ -378,7 +378,7 @@ class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/Oric/Oric.cpp b/Machines/Oric/Oric.cpp index 7417e6ef2..8931be836 100644 --- a/Machines/Oric/Oric.cpp +++ b/Machines/Oric/Oric.cpp @@ -627,7 +627,7 @@ template class Co } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { + std::unique_ptr get_options() final { return nullptr; } diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index baa9e515b..7a50e60f8 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -176,7 +176,6 @@ std::vector Machine::AllMachines(bool meaningful_without_media_only std::map> Machine::AllOptionsByMachineName() { std::map> options; -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AmstradCPC), AmstradCPC::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AppleII), Apple::II::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AtariST), Atari::ST::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), Coleco::Vision::get_options())); @@ -186,7 +185,14 @@ std::map> Machine::AllOptionsBy // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MSX), MSX::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Oric), Oric::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); - options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ZX8081), ZX8081::get_options())); + +#define Emplace(machine, class) \ + options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::machine), std::make_unique(Configurable::OptionsType::UserFriendly))); + + Emplace(AmstradCPC, AmstradCPC::Machine); + Emplace(ZX8081, ZX8081::Machine); + +#undef Emplace return options; } diff --git a/Machines/ZX8081/ZX8081.cpp b/Machines/ZX8081/ZX8081.cpp index 4a47e40b6..6030b2a08 100644 --- a/Machines/ZX8081/ZX8081.cpp +++ b/Machines/ZX8081/ZX8081.cpp @@ -50,22 +50,6 @@ enum ROMType: uint8_t { ZX80 = 0, ZX81 }; -struct Options: public Reflection::StructImpl { - bool automatic_tape_motor_control = true; - bool quickload = true; - - Options() { - if(needs_declare()) { - DeclareField(automatic_tape_motor_control); - DeclareField(quickload); - } - } -}; - -std::unique_ptr get_options() { - return std::make_unique(); -} - template class ConcreteMachine: public CRTMachine::Machine, public MediaTarget::Machine, @@ -422,31 +406,18 @@ template class ConcreteMachine: } // MARK: - Configuration options. - std::unique_ptr get_options(OptionsType type) final { - auto options = std::make_unique(); - switch(type) { - case OptionsType::Current: - options->automatic_tape_motor_control = use_automatic_tape_motor_control_; - options->quickload = allow_fast_tape_hack_; - break; - case OptionsType::Accurate: - case OptionsType::UserFriendly: - options->automatic_tape_motor_control = - options->quickload = type == OptionsType::UserFriendly; - break; - } + std::unique_ptr get_options() final { + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); // OptionsType is arbitrary, but not optional. + options->automatic_tape_motor_control = use_automatic_tape_motor_control_; + options->quickload = allow_fast_tape_hack_; return options; } - void set_options(const std::unique_ptr &options) { - if(Reflection::get(*options, "quickload", allow_fast_tape_hack_)) { - set_use_fast_tape(); - } - - bool autotapemotor; - if(Reflection::get(*options, "automatic_tape_motor_control", autotapemotor)) { - set_use_automatic_tape_motor_control(autotapemotor); - } + void set_options(const std::unique_ptr &str) { + const auto options = dynamic_cast(str.get()); + set_use_automatic_tape_motor_control(options->automatic_tape_motor_control); + allow_fast_tape_hack_ = options->quickload; + set_use_fast_tape(); } private: diff --git a/Machines/ZX8081/ZX8081.hpp b/Machines/ZX8081/ZX8081.hpp index f35d2b9c7..1bf00b34b 100644 --- a/Machines/ZX8081/ZX8081.hpp +++ b/Machines/ZX8081/ZX8081.hpp @@ -9,7 +9,7 @@ #ifndef ZX8081_hpp #define ZX8081_hpp -#include "../../Reflection/Struct.h" +#include "../../Configurable/Configurable.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -17,9 +17,7 @@ namespace ZX8081 { -/// @returns The options available for a ZX80 or ZX81. -std::unique_ptr get_options(); - +/// The ZX80/81 machine. class Machine { public: virtual ~Machine(); @@ -28,6 +26,24 @@ class Machine { virtual void set_tape_is_playing(bool is_playing) = 0; virtual bool get_tape_is_playing() = 0; + + /// Defines the runtime options available for a ZX80/81. + class Options: public Reflection::StructImpl { + public: + bool automatic_tape_motor_control; + bool quickload; + + Options(Configurable::OptionsType type): + automatic_tape_motor_control(type == Configurable::OptionsType::UserFriendly), + quickload(automatic_tape_motor_control) { + + // Declare fields if necessary. + if(needs_declare()) { + DeclareField(automatic_tape_motor_control); + DeclareField(quickload); + } + } + }; }; } diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index aac0802ab..b1163abdb 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -90,7 +90,7 @@ + isEnabled = "YES"> diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index d7cf2c4b0..b387b0c00 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -497,39 +497,42 @@ int main(int argc, char *argv[]) { std::cout << "Further machine options:" << std::endl; std::cout << "(* means: a selection will be made automatically based on the file selected, if any)" << std::endl << std::endl; -// const auto all_options = Machine::AllOptionsByMachineName(); -// for(const auto &machine_options: all_options) { -// std::cout << machine_options.first << ":" << std::endl; -// for(const auto &option: machine_options.second) { -// std::cout << '\t' << "--" << option->short_name; -// -// Configurable::ListOption *list_option = dynamic_cast(option.get()); -// if(list_option) { -// std::cout << "={"; -// bool is_first = true; -// for(const auto &option: list_option->options) { -// if(!is_first) std::cout << '|'; -// is_first = false; -// std::cout << option; -// } -// std::cout << "}"; -// } -// std::cout << std::endl; -// } -// std::cout << std::endl; -// } - const auto targets = Machine::TargetsByMachineName(false); - for(const auto &target: targets) { - const auto reflectable = dynamic_cast(target.second.get()); - if(!reflectable) continue; - const auto all_keys = reflectable->all_keys(); - if(all_keys.empty()) continue; + const auto runtime_options = Machine::AllOptionsByMachineName(); + const auto machine_names = Machine::AllMachines(false, true); + for(const auto &machine: machine_names) { + const auto target = targets.find(machine); + const auto options = runtime_options.find(machine); - std::cout << target.first << ":" << std::endl; - for(const auto &option: reflectable->all_keys()) { + const auto target_reflectable = dynamic_cast(target != targets.end() ? target->second.get() : nullptr); + const auto options_reflectable = dynamic_cast(options != runtime_options.end() ? options->second.get() : nullptr); + + // Don't print a section for this machine if it has no construction and no runtime options objects. + if(!target_reflectable && !options_reflectable) continue; + + const auto target_keys = target_reflectable ? target_reflectable->all_keys() : std::vector(); + const auto options_keys = options_reflectable ? options_reflectable->all_keys() : std::vector(); + + // Don't print a section for this machine if it doesn't actually have any options. + if(target_keys.empty() && options_keys.empty()) { + continue; + } + + std::cout << machine << ":" << std::endl; + + // Join the two lists of properties. + std::vector all_options = options_keys; + all_options.insert(all_options.end(), target_keys.begin(), target_keys.end()); + + for(const auto &option: all_options) { std::cout << '\t' << "--" << option; - const auto type = reflectable->type_of(option); + + bool is_construction_option = true; + auto type = target_reflectable->type_of(option); + if(!type) { + is_construction_option = false; + type = options_reflectable->type_of(option); + } // Is this a registered enum? If so, list options. if(!Reflection::Enum::name(*type).empty()) { @@ -544,7 +547,9 @@ int main(int argc, char *argv[]) { } // TODO: if not a registered enum... then assume it was a Boolean? - std::cout << "\t*" << std::endl; + + if(is_construction_option) std::cout << "\t*"; + std::cout << std::endl; } std::cout << std::endl; diff --git a/Outputs/CRT/CRT.cpp b/Outputs/CRT/CRT.cpp index b624fa6cf..8f5b0321f 100644 --- a/Outputs/CRT/CRT.cpp +++ b/Outputs/CRT/CRT.cpp @@ -93,6 +93,10 @@ void CRT::set_display_type(Outputs::Display::DisplayType display_type) { scan_target_->set_modals(scan_target_modals_); } +Outputs::Display::DisplayType CRT::get_display_type() { + return scan_target_modals_.display_type; +} + void CRT::set_phase_linked_luminance_offset(float offset) { scan_target_modals_.input_data_tweaks.phase_linked_luminance_offset = offset; scan_target_->set_modals(scan_target_modals_); diff --git a/Outputs/CRT/CRT.hpp b/Outputs/CRT/CRT.hpp index be07a50ba..9988968f5 100644 --- a/Outputs/CRT/CRT.hpp +++ b/Outputs/CRT/CRT.hpp @@ -298,6 +298,9 @@ class CRT { /*! Sets the display type that will be nominated to the scan target. */ void set_display_type(Outputs::Display::DisplayType); + /*! Gets the last display type provided to set_display_type. */ + Outputs::Display::DisplayType get_display_type(); + /*! Sets the offset to apply to phase when using the PhaseLinkedLuminance8 input data type. */ void set_phase_linked_luminance_offset(float); diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 890032062..412fb955a 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -26,6 +26,7 @@ namespace Reflection { #define EnumDeclaration(Name) #Name, __declaration##Name #define AnnounceEnum(Name) ::Reflection::Enum::declare(EnumDeclaration(Name)) +#define AnnounceEnumNS(Namespace, Name) ::Reflection::Enum::declare(#Name, Namespace::__declaration##Name) /*! This provides a very slight version of enum reflection; you can introspect only: From f9ca443667c4f8a4f8b05eec62faafa4a06cade2 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 17 Mar 2020 21:44:04 -0400 Subject: [PATCH 30/48] Adds the ability for reflective structs to limit the permitted values to enumerated properties. --- Machines/AmstradCPC/AmstradCPC.hpp | 3 +- OSBindings/SDL/main.cpp | 15 +++--- Reflection/Struct.h | 84 ++++++++++++++++++++++++++++++ 3 files changed, 93 insertions(+), 9 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index 8e8e843c7..d122b2bb8 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -38,8 +38,7 @@ class Machine { if(needs_declare()) { DeclareField(output); AnnounceEnumNS(Configurable, Display); - // TODO: some way to set limited enum support for a struct? - // In this case, to support only RGB and CompositeColour. + limit_enum(&output, Configurable::Display::RGB, Configurable::Display::CompositeColour, -1); } } }; diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index b387b0c00..33741b15c 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -495,7 +495,6 @@ int main(int argc, char *argv[]) { std::cout << "}" << std::endl << std::endl; std::cout << "Further machine options:" << std::endl; - std::cout << "(* means: a selection will be made automatically based on the file selected, if any)" << std::endl << std::endl; const auto targets = Machine::TargetsByMachineName(false); const auto runtime_options = Machine::AllOptionsByMachineName(); @@ -520,17 +519,18 @@ int main(int argc, char *argv[]) { std::cout << machine << ":" << std::endl; - // Join the two lists of properties. + // Join the two lists of properties and sort the result. std::vector all_options = options_keys; all_options.insert(all_options.end(), target_keys.begin(), target_keys.end()); + std::sort(all_options.begin(), all_options.end()); for(const auto &option: all_options) { std::cout << '\t' << "--" << option; - bool is_construction_option = true; + auto source = target_reflectable; auto type = target_reflectable->type_of(option); if(!type) { - is_construction_option = false; + source = options_reflectable; type = options_reflectable->type_of(option); } @@ -538,7 +538,7 @@ int main(int argc, char *argv[]) { if(!Reflection::Enum::name(*type).empty()) { std::cout << "={"; bool is_first = true; - for(const auto &value: Reflection::Enum::all_values(*type)) { + for(const auto &value: source->values_for(option)) { if(!is_first) std::cout << '|'; is_first = false; std::cout << value; @@ -546,9 +546,10 @@ int main(int argc, char *argv[]) { std::cout << "}"; } - // TODO: if not a registered enum... then assume it was a Boolean? + // The above effectively assumes that every field is either a + // Boolean or an enum. This may need to be revisted. It also + // assumes no name collisions, but that's kind of unavoidable. - if(is_construction_option) std::cout << "\t*"; std::cout << std::endl; } diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 4b1af1ea4..ee94ba6ce 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -9,6 +9,7 @@ #ifndef Struct_h #define Struct_h +#include #include #include #include @@ -27,6 +28,7 @@ struct Struct { virtual const std::type_info *type_of(const std::string &name) = 0; virtual void set(const std::string &name, const void *value) = 0; virtual const void *get(const std::string &name) = 0; + virtual std::vector values_for(const std::string &name) = 0; virtual ~Struct() {} }; @@ -123,6 +125,39 @@ template class StructImpl: public Struct { return iterator->second.type; } + /*! + @returns a list of the valid enum value names for field @c name if it is a declared enum field of this struct; + the empty list otherwise. + */ + std::vector values_for(const std::string &name) final { + std::vector result; + + // Return an empty vector if this field isn't declared. + const auto type = type_of(name); + if(!type) return result; + + // Also return an empty vector if this field isn't a registered enum. + const auto all_values = Enum::all_values(*type); + if(all_values.empty()) return result; + + // If no restriction is stored, return all values. + const auto permitted_values = permitted_enum_values_.find(name); + if(permitted_values == permitted_enum_values_.end()) return all_values; + + // Compile a vector of only those values the stored set indicates. + auto value = all_values.begin(); + auto flag = permitted_values->second.begin(); + while(value != all_values.end() && flag != permitted_values->second.end()) { + if(*flag) { + result.push_back(*value); + } + ++flag; + ++value; + } + + return result; + } + /*! @returns A vector of all declared fields for this struct. */ @@ -158,6 +193,35 @@ template class StructImpl: public Struct { )); } + /*! + If @c t is a previously-declared field that links to a declared enum then the variable + arguments provide a list of the acceptable values for that field. The list should be terminated + with a value of -1. + */ + template void limit_enum(Type *t, ...) { + const auto name = name_of(t); + if(name.empty()) return; + + // The default vector size of '8' isn't especially scientific, + // but I feel like it's a good choice. + std::vector permitted_values(8); + + va_list list; + va_start(list, t); + while(true) { + const int next = va_arg(list, int); + if(next < 0) break; + + if(permitted_values.size() <= next) { + permitted_values.resize(permitted_values.size() << 1); + } + permitted_values[next] = true; + } + va_end(list); + + permitted_enum_values_.emplace(std::make_pair(name, permitted_values)); + } + /*! @returns @c true if this subclass of @c Struct has not yet declared any fields. */ @@ -165,6 +229,25 @@ template class StructImpl: public Struct { return !contents_.size(); } + /*! + Performs a reverse lookup from field to name. + */ + std::string name_of(void *field) { + const ssize_t offset = reinterpret_cast(field) - reinterpret_cast(this); + + auto iterator = contents_.begin(); + while(iterator != contents_.end()) { + if(iterator->second.offset == offset) break; + ++iterator; + } + + if(iterator != contents_.end()) { + return iterator->first; + } else { + return ""; + } + } + private: struct Field { const std::type_info *type; @@ -174,6 +257,7 @@ template class StructImpl: public Struct { type(&type), offset(offset), size(size) {} }; static inline std::unordered_map contents_; + static inline std::unordered_map> permitted_enum_values_; }; } From b6e81242e7c960a44fd63a54805d7e1543d48e70 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 17 Mar 2020 21:53:26 -0400 Subject: [PATCH 31/48] Reintroduces Apple II runtime options. --- Machines/AmstradCPC/AmstradCPC.hpp | 2 +- Machines/Apple/AppleII/AppleII.cpp | 41 +++++++-------------------- Machines/Apple/AppleII/AppleII.hpp | 21 +++++++++++--- Machines/Apple/AppleII/Video.cpp | 4 +++ Machines/Apple/AppleII/Video.hpp | 3 ++ Machines/Utility/MachineForTarget.cpp | 2 +- 6 files changed, 36 insertions(+), 37 deletions(-) diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index d122b2bb8..bb5ff1052 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -28,7 +28,7 @@ class Machine { /// Creates and returns an Amstrad CPC. static Machine *AmstradCPC(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); - /// Defines the runtime options available for a ZX80/81. + /// Defines the runtime options available for an Amstrad CPC. class Options: public Reflection::StructImpl { public: Configurable::Display output = Configurable::Display::RGB; diff --git a/Machines/Apple/AppleII/AppleII.cpp b/Machines/Apple/AppleII/AppleII.cpp index f1a6db001..7cff092cc 100644 --- a/Machines/Apple/AppleII/AppleII.cpp +++ b/Machines/Apple/AppleII/AppleII.cpp @@ -37,15 +37,6 @@ namespace Apple { namespace II { -std::unique_ptr get_options() { - return nullptr; -} -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::DisplayCompositeMonochrome | Configurable::DisplayCompositeColour) -// ); -//} - #define is_iie() ((model == Analyser::Static::AppleII::Target::Model::IIe) || (model == Analyser::Static::AppleII::Target::Model::EnhancedIIe)) template class ConcreteMachine: @@ -433,6 +424,10 @@ template class ConcreteMachine: video_.set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return video_.get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return &speaker_; } @@ -870,31 +865,15 @@ template class ConcreteMachine: // MARK:: Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) { + const auto options = dynamic_cast(str.get()); + set_video_signal_configurable(options->output); } -// std::vector> get_options() final { -// return Apple::II::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// return get_accurate_selections(); -// } // MARK: MediaTarget bool insert_media(const Analyser::Static::Media &media) final { diff --git a/Machines/Apple/AppleII/AppleII.hpp b/Machines/Apple/AppleII/AppleII.hpp index 647587c21..73b168b6e 100644 --- a/Machines/Apple/AppleII/AppleII.hpp +++ b/Machines/Apple/AppleII/AppleII.hpp @@ -9,7 +9,8 @@ #ifndef AppleII_hpp #define AppleII_hpp -#include "../../../Reflection/Struct.h" +#include "../../../Configurable/Configurable.hpp" +#include "../../../Configurable/StandardOptions.hpp" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" @@ -18,15 +19,27 @@ namespace Apple { namespace II { -/// @returns The options available for an Apple II. -std::unique_ptr get_options(); - class Machine { public: virtual ~Machine(); /// Creates and returns an AppleII. static Machine *AppleII(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + /// Defines the runtime options available for an Apple II. + class Options: public Reflection::StructImpl { + public: + Configurable::Display output = Configurable::Display::CompositeColour; + + Options(Configurable::OptionsType type) { + // Declare fields if necessary. + if(needs_declare()) { + DeclareField(output); + AnnounceEnumNS(Configurable, Display); + limit_enum(&output, Configurable::Display::CompositeMonochrome, Configurable::Display::CompositeColour, -1); + } + } + }; }; } diff --git a/Machines/Apple/AppleII/Video.cpp b/Machines/Apple/AppleII/Video.cpp index a5b0b38b2..398eb91c9 100644 --- a/Machines/Apple/AppleII/Video.cpp +++ b/Machines/Apple/AppleII/Video.cpp @@ -55,6 +55,10 @@ void VideoBase::set_display_type(Outputs::Display::DisplayType display_type) { crt_.set_display_type(display_type); } +Outputs::Display::DisplayType VideoBase::get_display_type() { + return crt_.get_display_type(); +} + /* Rote setters and getters. */ diff --git a/Machines/Apple/AppleII/Video.hpp b/Machines/Apple/AppleII/Video.hpp index e227bc656..cd005742f 100644 --- a/Machines/Apple/AppleII/Video.hpp +++ b/Machines/Apple/AppleII/Video.hpp @@ -46,6 +46,9 @@ class VideoBase { /// Sets the type of output. void set_display_type(Outputs::Display::DisplayType); + /// Gets the type of output. + Outputs::Display::DisplayType get_display_type(); + /* Descriptions for the setters below are taken verbatim from the Apple IIe Technical Reference. Addresses are the conventional diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 7a50e60f8..44bbb4056 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -176,7 +176,6 @@ std::vector Machine::AllMachines(bool meaningful_without_media_only std::map> Machine::AllOptionsByMachineName() { std::map> options; -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AppleII), Apple::II::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AtariST), Atari::ST::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), Coleco::Vision::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Electron), Electron::get_options())); @@ -190,6 +189,7 @@ std::map> Machine::AllOptionsBy options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::machine), std::make_unique(Configurable::OptionsType::UserFriendly))); Emplace(AmstradCPC, AmstradCPC::Machine); + Emplace(AppleII, Apple::II::Machine); Emplace(ZX8081, ZX8081::Machine); #undef Emplace From 8c6ca89da252fbbb457473422c25e24217a3c143 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 17 Mar 2020 22:06:20 -0400 Subject: [PATCH 32/48] Restores runtime options for the Acorn Electron. --- Configurable/StandardOptions.hpp | 1 - Machines/Electron/Electron.cpp | 46 ++++++++------------------- Machines/Electron/Electron.hpp | 25 ++++++++++++--- Machines/Electron/Video.cpp | 4 +++ Machines/Electron/Video.hpp | 3 ++ Machines/Utility/MachineForTarget.cpp | 2 +- 6 files changed, 43 insertions(+), 38 deletions(-) diff --git a/Configurable/StandardOptions.hpp b/Configurable/StandardOptions.hpp index 53bc0f600..9b1ec7653 100644 --- a/Configurable/StandardOptions.hpp +++ b/Configurable/StandardOptions.hpp @@ -20,7 +20,6 @@ ReflectableEnum(Display, CompositeMonochrome ); - } #endif /* StandardOptions_hpp */ diff --git a/Machines/Electron/Electron.cpp b/Machines/Electron/Electron.cpp index 2f67b3c12..defd1abd5 100644 --- a/Machines/Electron/Electron.cpp +++ b/Machines/Electron/Electron.cpp @@ -420,6 +420,10 @@ class ConcreteMachine: video_output_.set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return video_output_.get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return &speaker_; } @@ -455,41 +459,19 @@ class ConcreteMachine: // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + options->quickload = allow_fast_tape_hack_; + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + const auto options = dynamic_cast(str.get()); + + set_video_signal_configurable(options->output); + allow_fast_tape_hack_ = options->quickload; + set_use_fast_tape_hack(); } -// std::vector> get_options() final { -// return Electron::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// bool quickload; -// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { -// allow_fast_tape_hack_ = quickload; -// set_use_fast_tape_hack(); -// } -// -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, false); -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, true); -// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); -// return selection_set; -// } // MARK: - Activity Source void set_activity_observer(Activity::Observer *observer) final { diff --git a/Machines/Electron/Electron.hpp b/Machines/Electron/Electron.hpp index b6c712c5a..79697a0a0 100644 --- a/Machines/Electron/Electron.hpp +++ b/Machines/Electron/Electron.hpp @@ -9,7 +9,8 @@ #ifndef Electron_hpp #define Electron_hpp -#include "../../Reflection/Struct.h" +#include "../../Configurable/Configurable.hpp" +#include "../../Configurable/StandardOptions.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -17,9 +18,6 @@ namespace Electron { -/// @returns The options available for an Electron. -std::unique_ptr get_options(); - /*! @abstract Represents an Acorn Electron. @@ -32,6 +30,25 @@ class Machine { /// Creates and returns an Electron. static Machine *Electron(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + /// Defines the runtime options available for an Electron. + class Options: public Reflection::StructImpl { + public: + Configurable::Display output; + bool quickload; + + Options(Configurable::OptionsType type) : + output(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::RGB : Configurable::Display::CompositeColour), + quickload(type == Configurable::OptionsType::UserFriendly) { + + if(needs_declare()) { + DeclareField(output); + DeclareField(quickload); + AnnounceEnumNS(Configurable, Display); + limit_enum(&output, Configurable::Display::RGB, Configurable::Display::CompositeColour, -1); + } + } + }; }; } diff --git a/Machines/Electron/Video.cpp b/Machines/Electron/Video.cpp index 88de46806..292e799f1 100644 --- a/Machines/Electron/Video.cpp +++ b/Machines/Electron/Video.cpp @@ -64,6 +64,10 @@ void VideoOutput::set_display_type(Outputs::Display::DisplayType display_type) { crt_.set_display_type(display_type); } +Outputs::Display::DisplayType VideoOutput::get_display_type() { + return crt_.get_display_type(); +} + // MARK: - Display update methods void VideoOutput::start_pixel_line() { diff --git a/Machines/Electron/Video.hpp b/Machines/Electron/Video.hpp index af8e57111..aa2b28dec 100644 --- a/Machines/Electron/Video.hpp +++ b/Machines/Electron/Video.hpp @@ -45,6 +45,9 @@ class VideoOutput { /// Sets the type of output. void set_display_type(Outputs::Display::DisplayType); + /// Gets the type of output. + Outputs::Display::DisplayType get_display_type(); + /*! Writes @c value to the register at @c address. May mutate the results of @c get_next_interrupt, @c get_cycles_until_next_ram_availability and @c get_memory_access_range. diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 44bbb4056..e1a369ff1 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -178,7 +178,6 @@ std::map> Machine::AllOptionsBy // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AtariST), Atari::ST::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), Coleco::Vision::get_options())); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Electron), Electron::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MasterSystem), Sega::MasterSystem::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MSX), MSX::get_options())); @@ -190,6 +189,7 @@ std::map> Machine::AllOptionsBy Emplace(AmstradCPC, AmstradCPC::Machine); Emplace(AppleII, Apple::II::Machine); + Emplace(Electron, Electron::Machine); Emplace(ZX8081, ZX8081::Machine); #undef Emplace From ec6664f5903c4bd3b012782bf08d7855df6a90cc Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Tue, 17 Mar 2020 23:52:55 -0400 Subject: [PATCH 33/48] Takes steps to guarantee property naming; reintroduces Electron runtime options. --- Configurable/StandardOptions.hpp | 29 +++++++++++++++++++ Machines/AmstradCPC/AmstradCPC.hpp | 11 +++----- Machines/Apple/AppleII/AppleII.hpp | 11 +++----- Machines/Atari/ST/AtariST.cpp | 40 +++++++-------------------- Machines/Atari/ST/AtariST.hpp | 18 +++++++++--- Machines/Atari/ST/Video.cpp | 4 +++ Machines/Atari/ST/Video.hpp | 5 ++++ Machines/Electron/Electron.hpp | 17 +++++------- Machines/Utility/MachineForTarget.cpp | 2 +- Machines/ZX8081/ZX8081.hpp | 9 +++--- 10 files changed, 83 insertions(+), 63 deletions(-) diff --git a/Configurable/StandardOptions.hpp b/Configurable/StandardOptions.hpp index 9b1ec7653..ef6d3df38 100644 --- a/Configurable/StandardOptions.hpp +++ b/Configurable/StandardOptions.hpp @@ -20,6 +20,35 @@ ReflectableEnum(Display, CompositeMonochrome ); +//=== +// From here downward are a bunch of templates for individual option flags. +// Using them saves you marginally in syntax, but the primary gain is to +// ensure unified property naming. +//=== + +template class DisplayOption { + public: + Configurable::Display output; + DisplayOption(Configurable::Display output) : output(output) {} + + protected: + void declare_display_option() { + static_cast(this)->declare(&output, "output"); + AnnounceEnumNS(Configurable, Display); + } +}; + +template class QuickloadOption { + public: + bool quickload; + QuickloadOption(bool quickload) : quickload(quickload) {} + + protected: + void declare_quickload_option() { + static_cast(this)->declare(&quickload, "quickload"); + } +}; + } #endif /* StandardOptions_hpp */ diff --git a/Machines/AmstradCPC/AmstradCPC.hpp b/Machines/AmstradCPC/AmstradCPC.hpp index bb5ff1052..56c1186a2 100644 --- a/Machines/AmstradCPC/AmstradCPC.hpp +++ b/Machines/AmstradCPC/AmstradCPC.hpp @@ -29,15 +29,12 @@ class Machine { static Machine *AmstradCPC(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); /// Defines the runtime options available for an Amstrad CPC. - class Options: public Reflection::StructImpl { + class Options: public Reflection::StructImpl, public Configurable::DisplayOption { + friend Configurable::DisplayOption; public: - Configurable::Display output = Configurable::Display::RGB; - - Options(Configurable::OptionsType type) { - // Declare fields if necessary. + Options(Configurable::OptionsType type) : Configurable::DisplayOption(Configurable::Display::RGB) { if(needs_declare()) { - DeclareField(output); - AnnounceEnumNS(Configurable, Display); + declare_display_option(); limit_enum(&output, Configurable::Display::RGB, Configurable::Display::CompositeColour, -1); } } diff --git a/Machines/Apple/AppleII/AppleII.hpp b/Machines/Apple/AppleII/AppleII.hpp index 73b168b6e..1e1434f63 100644 --- a/Machines/Apple/AppleII/AppleII.hpp +++ b/Machines/Apple/AppleII/AppleII.hpp @@ -27,15 +27,12 @@ class Machine { static Machine *AppleII(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); /// Defines the runtime options available for an Apple II. - class Options: public Reflection::StructImpl { + class Options: public Reflection::StructImpl, public Configurable::DisplayOption { + friend Configurable::DisplayOption; public: - Configurable::Display output = Configurable::Display::CompositeColour; - - Options(Configurable::OptionsType type) { - // Declare fields if necessary. + Options(Configurable::OptionsType type) : Configurable::DisplayOption(Configurable::Display::CompositeColour) { if(needs_declare()) { - DeclareField(output); - AnnounceEnumNS(Configurable, Display); + declare_display_option(); limit_enum(&output, Configurable::Display::CompositeMonochrome, Configurable::Display::CompositeColour, -1); } } diff --git a/Machines/Atari/ST/AtariST.cpp b/Machines/Atari/ST/AtariST.cpp index 6008ad075..504e43400 100644 --- a/Machines/Atari/ST/AtariST.cpp +++ b/Machines/Atari/ST/AtariST.cpp @@ -42,16 +42,6 @@ namespace Atari { namespace ST { -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) -// ); -//} - -std::unique_ptr get_options() { - return nullptr; -} - constexpr int CLOCK_RATE = 8021247; using Target = Analyser::Static::Target; @@ -153,6 +143,10 @@ class ConcreteMachine: video_->set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return video_->get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return &speaker_; } @@ -683,29 +677,15 @@ class ConcreteMachine: // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + const auto options = dynamic_cast(str.get()); + set_video_signal_configurable(options->output); } -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); -// return selection_set; -// } }; } diff --git a/Machines/Atari/ST/AtariST.hpp b/Machines/Atari/ST/AtariST.hpp index abd71e3ed..8f3274780 100644 --- a/Machines/Atari/ST/AtariST.hpp +++ b/Machines/Atari/ST/AtariST.hpp @@ -9,7 +9,8 @@ #ifndef AtariST_hpp #define AtariST_hpp -#include "../../../Reflection/Struct.h" +#include "../../../Configurable/Configurable.hpp" +#include "../../../Configurable/StandardOptions.hpp" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" @@ -18,14 +19,23 @@ namespace Atari { namespace ST { -/// @returns The options available for an Atari ST. -std::unique_ptr get_options(); - class Machine { public: virtual ~Machine(); static Machine *AtariST(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + class Options: public Reflection::StructImpl, public Configurable::DisplayOption { + friend Configurable::DisplayOption; + public: + Options(Configurable::OptionsType type) : Configurable::DisplayOption( + type == Configurable::OptionsType::UserFriendly ? Configurable::Display::RGB : Configurable::Display::CompositeColour) { + if(needs_declare()) { + declare_display_option(); + limit_enum(&output, Configurable::Display::RGB, Configurable::Display::CompositeColour, -1); + } + } + }; }; } diff --git a/Machines/Atari/ST/Video.cpp b/Machines/Atari/ST/Video.cpp index 134f5ed87..5c3ba851f 100644 --- a/Machines/Atari/ST/Video.cpp +++ b/Machines/Atari/ST/Video.cpp @@ -146,6 +146,10 @@ void Video::set_display_type(Outputs::Display::DisplayType display_type) { crt_.set_display_type(display_type); } +Outputs::Display::DisplayType Video::get_display_type() { + return crt_.get_display_type(); +} + void Video::run_for(HalfCycles duration) { int integer_duration = int(duration.as_integral()); assert(integer_duration >= 0); diff --git a/Machines/Atari/ST/Video.hpp b/Machines/Atari/ST/Video.hpp index 34a0f94c0..851d8674c 100644 --- a/Machines/Atari/ST/Video.hpp +++ b/Machines/Atari/ST/Video.hpp @@ -54,6 +54,11 @@ class Video { */ void set_display_type(Outputs::Display::DisplayType); + /*! + Gets the type of output. + */ + Outputs::Display::DisplayType get_display_type(); + /*! Produces the next @c duration period of pixels. */ diff --git a/Machines/Electron/Electron.hpp b/Machines/Electron/Electron.hpp index 79697a0a0..e1b047deb 100644 --- a/Machines/Electron/Electron.hpp +++ b/Machines/Electron/Electron.hpp @@ -32,19 +32,16 @@ class Machine { static Machine *Electron(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); /// Defines the runtime options available for an Electron. - class Options: public Reflection::StructImpl { + class Options: public Reflection::StructImpl, public Configurable::DisplayOption, public Configurable::QuickloadOption { + friend Configurable::DisplayOption; + friend Configurable::QuickloadOption; public: - Configurable::Display output; - bool quickload; - Options(Configurable::OptionsType type) : - output(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::RGB : Configurable::Display::CompositeColour), - quickload(type == Configurable::OptionsType::UserFriendly) { - + Configurable::DisplayOption(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::RGB : Configurable::Display::CompositeColour), + Configurable::QuickloadOption(type == Configurable::OptionsType::UserFriendly) { if(needs_declare()) { - DeclareField(output); - DeclareField(quickload); - AnnounceEnumNS(Configurable, Display); + declare_display_option(); + declare_quickload_option(); limit_enum(&output, Configurable::Display::RGB, Configurable::Display::CompositeColour, -1); } } diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index e1a369ff1..3adea7e36 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -176,7 +176,6 @@ std::vector Machine::AllMachines(bool meaningful_without_media_only std::map> Machine::AllOptionsByMachineName() { std::map> options; -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::AtariST), Atari::ST::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), Coleco::Vision::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MasterSystem), Sega::MasterSystem::get_options())); @@ -189,6 +188,7 @@ std::map> Machine::AllOptionsBy Emplace(AmstradCPC, AmstradCPC::Machine); Emplace(AppleII, Apple::II::Machine); + Emplace(AtariST, Atari::ST::Machine); Emplace(Electron, Electron::Machine); Emplace(ZX8081, ZX8081::Machine); diff --git a/Machines/ZX8081/ZX8081.hpp b/Machines/ZX8081/ZX8081.hpp index 1bf00b34b..e6933afef 100644 --- a/Machines/ZX8081/ZX8081.hpp +++ b/Machines/ZX8081/ZX8081.hpp @@ -10,6 +10,7 @@ #define ZX8081_hpp #include "../../Configurable/Configurable.hpp" +#include "../../Configurable/StandardOptions.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -28,19 +29,19 @@ class Machine { virtual bool get_tape_is_playing() = 0; /// Defines the runtime options available for a ZX80/81. - class Options: public Reflection::StructImpl { + class Options: public Reflection::StructImpl, public Configurable::QuickloadOption { + friend Configurable::QuickloadOption; public: bool automatic_tape_motor_control; - bool quickload; Options(Configurable::OptionsType type): automatic_tape_motor_control(type == Configurable::OptionsType::UserFriendly), - quickload(automatic_tape_motor_control) { + Configurable::QuickloadOption(type == Configurable::OptionsType::UserFriendly) { // Declare fields if necessary. if(needs_declare()) { DeclareField(automatic_tape_motor_control); - DeclareField(quickload); + declare_quickload_option(); } } }; From a7e192059747d54416f70f4efde42b202cb0c4b1 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 00:06:52 -0400 Subject: [PATCH 34/48] Restores ColecoVision runtime options. --- Components/9918/9918.cpp | 4 +++ Components/9918/9918.hpp | 3 ++ Machines/ColecoVision/ColecoVision.cpp | 40 +++++++------------------- Machines/ColecoVision/ColecoVision.hpp | 17 +++++++++-- Machines/Utility/MachineForTarget.cpp | 2 +- OSBindings/SDL/main.cpp | 2 +- 6 files changed, 33 insertions(+), 35 deletions(-) diff --git a/Components/9918/9918.cpp b/Components/9918/9918.cpp index b6d672fc0..9ed2fbfb5 100644 --- a/Components/9918/9918.cpp +++ b/Components/9918/9918.cpp @@ -129,6 +129,10 @@ void TMS9918::set_display_type(Outputs::Display::DisplayType display_type) { crt_.set_display_type(display_type); } +Outputs::Display::DisplayType TMS9918::get_display_type() { + return crt_.get_display_type(); +} + void Base::LineBuffer::reset_sprite_collection() { sprites_stopped = false; active_sprite_slot = 0; diff --git a/Components/9918/9918.hpp b/Components/9918/9918.hpp index 2cca01693..4f2530bf3 100644 --- a/Components/9918/9918.hpp +++ b/Components/9918/9918.hpp @@ -50,6 +50,9 @@ class TMS9918: public Base { /*! Sets the type of display the CRT will request. */ void set_display_type(Outputs::Display::DisplayType); + /*! Gets the type of display the CRT will request. */ + Outputs::Display::DisplayType get_display_type(); + /*! Runs the VCP for the number of cycles indicate; it is an implicit assumption of the code that the input clock rate is 3579545 Hz, the NTSC colour clock rate. diff --git a/Machines/ColecoVision/ColecoVision.cpp b/Machines/ColecoVision/ColecoVision.cpp index 45b875ffc..142c82c47 100644 --- a/Machines/ColecoVision/ColecoVision.cpp +++ b/Machines/ColecoVision/ColecoVision.cpp @@ -34,12 +34,6 @@ constexpr int sn76489_divider = 2; namespace Coleco { namespace Vision { -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::DisplaySVideo | Configurable::DisplayCompositeColour) -// ); -//} - std::unique_ptr get_options() { return nullptr; } @@ -198,6 +192,10 @@ class ConcreteMachine: vdp_->set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return vdp_->get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return &speaker_; } @@ -374,33 +372,15 @@ class ConcreteMachine: // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + const auto options = dynamic_cast(str.get()); + set_video_signal_configurable(options->output); } -// std::vector> get_options() final { -// return Coleco::Vision::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::SVideo); -// return selection_set; -// } private: inline void page_megacart(uint16_t address) { diff --git a/Machines/ColecoVision/ColecoVision.hpp b/Machines/ColecoVision/ColecoVision.hpp index c245981a0..0e0dc8de5 100644 --- a/Machines/ColecoVision/ColecoVision.hpp +++ b/Machines/ColecoVision/ColecoVision.hpp @@ -9,7 +9,8 @@ #ifndef ColecoVision_hpp #define ColecoVision_hpp -#include "../../Reflection/Struct.h" +#include "../../Configurable/Configurable.hpp" +#include "../../Configurable/StandardOptions.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -18,12 +19,22 @@ namespace Coleco { namespace Vision { -std::unique_ptr get_options(); - class Machine { public: virtual ~Machine(); static Machine *ColecoVision(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + class Options: public Reflection::StructImpl, public Configurable::DisplayOption { + friend Configurable::DisplayOption; + public: + Options(Configurable::OptionsType type) : + Configurable::DisplayOption(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::SVideo : Configurable::Display::CompositeColour) { + if(needs_declare()) { + declare_display_option(); + limit_enum(&output, Configurable::Display::SVideo, Configurable::Display::CompositeColour, -1); + } + } + }; }; } diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 3adea7e36..0a56da7cf 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -176,7 +176,6 @@ std::vector Machine::AllMachines(bool meaningful_without_media_only std::map> Machine::AllOptionsByMachineName() { std::map> options; -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::ColecoVision), Coleco::Vision::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MasterSystem), Sega::MasterSystem::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MSX), MSX::get_options())); @@ -189,6 +188,7 @@ std::map> Machine::AllOptionsBy Emplace(AmstradCPC, AmstradCPC::Machine); Emplace(AppleII, Apple::II::Machine); Emplace(AtariST, Atari::ST::Machine); + Emplace(ColecoVision, Coleco::Vision::Machine); Emplace(Electron, Electron::Machine); Emplace(ZX8081, ZX8081::Machine); diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index 33741b15c..001718065 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -528,7 +528,7 @@ int main(int argc, char *argv[]) { std::cout << '\t' << "--" << option; auto source = target_reflectable; - auto type = target_reflectable->type_of(option); + auto type = target_reflectable ? target_reflectable->type_of(option) : nullptr; if(!type) { source = options_reflectable; type = options_reflectable->type_of(option); From ead282332224e47187fd8fc97d39fd2e98bf3cb4 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 18:26:22 -0400 Subject: [PATCH 35/48] Reintroduces MSX and Master System runtime options. --- Machines/MSX/MSX.cpp | 56 +++++++------------------- Machines/MSX/MSX.hpp | 19 +++++++-- Machines/MasterSystem/MasterSystem.cpp | 44 +++++--------------- Machines/MasterSystem/MasterSystem.hpp | 16 ++++++-- Machines/Utility/MachineForTarget.cpp | 4 +- 5 files changed, 55 insertions(+), 84 deletions(-) diff --git a/Machines/MSX/MSX.cpp b/Machines/MSX/MSX.cpp index 0ca91006e..7fd483d8a 100644 --- a/Machines/MSX/MSX.cpp +++ b/Machines/MSX/MSX.cpp @@ -52,16 +52,6 @@ namespace MSX { -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::DisplayRGB | Configurable::DisplaySVideo | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) -// ); -//} - -std::unique_ptr get_options() { - return nullptr; -} - class AYPortHandler: public GI::AY38910::PortHandler { public: AYPortHandler(Storage::Tape::BinaryTapePlayer &tape_player) : tape_player_(tape_player) { @@ -295,6 +285,10 @@ class ConcreteMachine: vdp_->set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return vdp_->get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return &speaker_; } @@ -654,41 +648,19 @@ class ConcreteMachine: // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + options->quickload = allow_fast_tape_; + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + const auto options = dynamic_cast(str.get()); + + set_video_signal_configurable(options->output); + allow_fast_tape_ = options->quickload; + set_use_fast_tape(); } -// std::vector> get_options() final { -// return MSX::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// bool quickload; -// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { -// allow_fast_tape_ = quickload; -// set_use_fast_tape(); -// } -// -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, false); -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, true); -// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); -// return selection_set; -// } // MARK: - Sleeper void set_component_prefers_clocking(ClockingHint::Source *component, ClockingHint::Preference clocking) final { diff --git a/Machines/MSX/MSX.hpp b/Machines/MSX/MSX.hpp index 9f68a3645..902ad5a78 100644 --- a/Machines/MSX/MSX.hpp +++ b/Machines/MSX/MSX.hpp @@ -9,7 +9,8 @@ #ifndef MSX_hpp #define MSX_hpp -#include "../../Reflection/Struct.h" +#include "../../Configurable/Configurable.hpp" +#include "../../Configurable/StandardOptions.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -17,12 +18,24 @@ namespace MSX { -std::unique_ptr get_options(); - class Machine { public: virtual ~Machine(); static Machine *MSX(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + class Options: public Reflection::StructImpl, public Configurable::DisplayOption, public Configurable::QuickloadOption { + friend Configurable::DisplayOption; + friend Configurable::QuickloadOption; + public: + Options(Configurable::OptionsType type) : + Configurable::DisplayOption(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::RGB : Configurable::Display::CompositeColour), + Configurable::QuickloadOption(type == Configurable::OptionsType::UserFriendly) { + if(needs_declare()) { + declare_display_option(); + declare_quickload_option(); + } + } + }; }; } diff --git a/Machines/MasterSystem/MasterSystem.cpp b/Machines/MasterSystem/MasterSystem.cpp index 614dc7d0f..06859d6bd 100644 --- a/Machines/MasterSystem/MasterSystem.cpp +++ b/Machines/MasterSystem/MasterSystem.cpp @@ -38,16 +38,6 @@ constexpr int sn76489_divider = 2; namespace Sega { namespace MasterSystem { -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour) -// ); -//} - -std::unique_ptr get_options() { - return nullptr; -} - class Joystick: public Inputs::ConcreteJoystick { public: Joystick() : @@ -192,6 +182,10 @@ class ConcreteMachine: vdp_->set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return vdp_->get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return &speaker_; } @@ -379,33 +373,15 @@ class ConcreteMachine: // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + const auto options = dynamic_cast(str.get()); + set_video_signal_configurable(options->output); } -// std::vector> get_options() final { -// return Sega::MasterSystem::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); -// return selection_set; -// } private: static TI::TMS::Personality tms_personality_for_model(Analyser::Static::Sega::Target::Model model) { diff --git a/Machines/MasterSystem/MasterSystem.hpp b/Machines/MasterSystem/MasterSystem.hpp index c63228490..348842be9 100644 --- a/Machines/MasterSystem/MasterSystem.hpp +++ b/Machines/MasterSystem/MasterSystem.hpp @@ -9,7 +9,8 @@ #ifndef MasterSystem_hpp #define MasterSystem_hpp -#include "../../Reflection/Struct.h" +#include "../../Configurable/Configurable.hpp" +#include "../../Configurable/StandardOptions.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -18,12 +19,21 @@ namespace Sega { namespace MasterSystem { -std::unique_ptr get_options(); - class Machine { public: virtual ~Machine(); static Machine *MasterSystem(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + class Options: public Reflection::StructImpl, public Configurable::DisplayOption { + friend Configurable::DisplayOption; + public: + Options(Configurable::OptionsType type) : + Configurable::DisplayOption(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::RGB : Configurable::Display::CompositeColour) { + if(needs_declare()) { + declare_display_option(); + } + } + }; }; } diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 0a56da7cf..bc45e42ce 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -177,8 +177,6 @@ std::map> Machine::AllOptionsBy std::map> options; // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MasterSystem), Sega::MasterSystem::get_options())); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::MSX), MSX::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Oric), Oric::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); @@ -190,6 +188,8 @@ std::map> Machine::AllOptionsBy Emplace(AtariST, Atari::ST::Machine); Emplace(ColecoVision, Coleco::Vision::Machine); Emplace(Electron, Electron::Machine); + Emplace(MasterSystem, Sega::MasterSystem::Machine); + Emplace(MSX, MSX::Machine); Emplace(ZX8081, ZX8081::Machine); #undef Emplace From ffc1b0ff29566d0894378b1a4ff2bc7f893acf96 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 18:31:31 -0400 Subject: [PATCH 36/48] Reintroduces Oric runtime options. --- Machines/Oric/Oric.cpp | 57 ++++++--------------------- Machines/Oric/Oric.hpp | 17 +++++++- Machines/Oric/Video.cpp | 4 ++ Machines/Oric/Video.hpp | 1 + Machines/Utility/MachineForTarget.cpp | 2 +- 5 files changed, 34 insertions(+), 47 deletions(-) diff --git a/Machines/Oric/Oric.cpp b/Machines/Oric/Oric.cpp index 8931be836..74d92136f 100644 --- a/Machines/Oric/Oric.cpp +++ b/Machines/Oric/Oric.cpp @@ -50,20 +50,6 @@ enum ROM { BASIC10 = 0, BASIC11, Microdisc, Colour }; -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast( -// Configurable::DisplayRGB | -// Configurable::DisplayCompositeColour | -// Configurable::DisplayCompositeMonochrome | -// Configurable::QuickLoadTape) -// ); -//} - -std::unique_ptr get_options() { - return nullptr; -} - /*! Models the Oric's keyboard: eight key rows, containing a bitfield of keys set. @@ -569,6 +555,10 @@ template class Co video_output_.set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return video_output_.get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return &speaker_; } @@ -628,40 +618,17 @@ template class Co // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + options->quickload = use_fast_tape_hack_; + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + const auto options = dynamic_cast(str.get()); + set_video_signal_configurable(options->output); + set_use_fast_tape_hack(options->quickload); } -// std::vector> get_options() final { -// return Oric::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// bool quickload; -// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { -// set_use_fast_tape_hack(quickload); -// } -// -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, false); -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, true); -// Configurable::append_display_selection(selection_set, Configurable::Display::RGB); -// return selection_set; -// } void set_activity_observer(Activity::Observer *observer) final { switch(disk_interface) { diff --git a/Machines/Oric/Oric.hpp b/Machines/Oric/Oric.hpp index 57c6fdba3..d9f8540b1 100644 --- a/Machines/Oric/Oric.hpp +++ b/Machines/Oric/Oric.hpp @@ -9,7 +9,8 @@ #ifndef Oric_hpp #define Oric_hpp -#include "../../Reflection/Struct.h" +#include "../../Configurable/Configurable.hpp" +#include "../../Configurable/StandardOptions.hpp" #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" @@ -29,6 +30,20 @@ class Machine { /// Creates and returns an Oric. static Machine *Oric(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + class Options: public Reflection::StructImpl, public Configurable::DisplayOption, public Configurable::QuickloadOption { + friend Configurable::DisplayOption; + friend Configurable::QuickloadOption; + public: + Options(Configurable::OptionsType type) : + Configurable::DisplayOption(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::RGB : Configurable::Display::CompositeColour), + Configurable::QuickloadOption(type == Configurable::OptionsType::UserFriendly) { + if(needs_declare()) { + declare_display_option(); + declare_quickload_option(); + } + } + }; }; } diff --git a/Machines/Oric/Video.cpp b/Machines/Oric/Video.cpp index 7e228c38e..8fb8926a0 100644 --- a/Machines/Oric/Video.cpp +++ b/Machines/Oric/Video.cpp @@ -67,6 +67,10 @@ void VideoOutput::set_display_type(Outputs::Display::DisplayType display_type) { } } +Outputs::Display::DisplayType VideoOutput::get_display_type() { + return crt_.get_display_type(); +} + void VideoOutput::set_scan_target(Outputs::Display::ScanTarget *scan_target) { crt_.set_scan_target(scan_target); } diff --git a/Machines/Oric/Video.hpp b/Machines/Oric/Video.hpp index 9d066a808..fe3001efe 100644 --- a/Machines/Oric/Video.hpp +++ b/Machines/Oric/Video.hpp @@ -27,6 +27,7 @@ class VideoOutput { void set_scan_target(Outputs::Display::ScanTarget *scan_target); void set_display_type(Outputs::Display::DisplayType display_type); + Outputs::Display::DisplayType get_display_type(); Outputs::Display::ScanStatus get_scaled_scan_status() const; void register_crt_frequency_mismatch(); diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index bc45e42ce..316ef8211 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -177,7 +177,6 @@ std::map> Machine::AllOptionsBy std::map> options; // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Oric), Oric::get_options())); // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); #define Emplace(machine, class) \ @@ -190,6 +189,7 @@ std::map> Machine::AllOptionsBy Emplace(Electron, Electron::Machine); Emplace(MasterSystem, Sega::MasterSystem::Machine); Emplace(MSX, MSX::Machine); + Emplace(Oric, Oric::Machine); Emplace(ZX8081, ZX8081::Machine); #undef Emplace From b2a381d4010c9df8b32016bb17576225eb8348da Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 20:23:55 -0400 Subject: [PATCH 37/48] Restores Vic-20 runtime options. --- Machines/Commodore/Vic-20/Vic20.cpp | 55 +++++++-------------------- Machines/Commodore/Vic-20/Vic20.hpp | 18 ++++++++- Machines/Electron/Electron.hpp | 2 +- Machines/Utility/MachineForTarget.cpp | 2 +- 4 files changed, 33 insertions(+), 44 deletions(-) diff --git a/Machines/Commodore/Vic-20/Vic20.cpp b/Machines/Commodore/Vic-20/Vic20.cpp index 2984544b6..4595f3ef3 100644 --- a/Machines/Commodore/Vic-20/Vic20.cpp +++ b/Machines/Commodore/Vic-20/Vic20.cpp @@ -49,15 +49,6 @@ enum ROMSlot { Drive }; -std::unique_ptr get_options() { - return nullptr; -} -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::DisplaySVideo | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) -// ); -//} - enum JoystickInput { Up = 0x04, Down = 0x08, @@ -656,6 +647,10 @@ class ConcreteMachine: mos6560_.set_display_type(display_type); } + Outputs::Display::DisplayType get_display_type() final { + return mos6560_.get_display_type(); + } + Outputs::Speaker::Speaker *get_speaker() final { return mos6560_.get_speaker(); } @@ -683,41 +678,19 @@ class ConcreteMachine: // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->output = get_video_signal_configurable(); + options->quickload = allow_fast_tape_hack_; + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + const auto options = dynamic_cast(str.get()); + + set_video_signal_configurable(options->output); + allow_fast_tape_hack_ = options->quickload; + set_use_fast_tape(); } -// std::vector> get_options() final { -// return Commodore::Vic20::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// bool quickload; -// if(Configurable::get_quick_load_tape(selections_by_option, quickload)) { -// allow_fast_tape_hack_ = quickload; -// set_use_fast_tape(); -// } -// -// Configurable::Display display; -// if(Configurable::get_display(selections_by_option, display)) { -// set_video_signal_configurable(display); -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, false); -// Configurable::append_display_selection(selection_set, Configurable::Display::CompositeColour); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_load_tape_selection(selection_set, true); -// Configurable::append_display_selection(selection_set, Configurable::Display::SVideo); -// return selection_set; -// } void set_component_prefers_clocking(ClockingHint::Source *component, ClockingHint::Preference clocking) final { tape_is_sleeping_ = clocking == ClockingHint::Preference::None; diff --git a/Machines/Commodore/Vic-20/Vic20.hpp b/Machines/Commodore/Vic-20/Vic20.hpp index 8c442809f..c9583bfa4 100644 --- a/Machines/Commodore/Vic-20/Vic20.hpp +++ b/Machines/Commodore/Vic-20/Vic20.hpp @@ -9,7 +9,8 @@ #ifndef Vic20_hpp #define Vic20_hpp -#include "../../../Reflection/Struct.h" +#include "../../../Configurable/Configurable.hpp" +#include "../../../Configurable/StandardOptions.hpp" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" @@ -27,6 +28,21 @@ class Machine { /// Creates and returns a Vic-20. static Machine *Vic20(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + class Options: public Reflection::StructImpl, public Configurable::DisplayOption, public Configurable::QuickloadOption { + friend Configurable::DisplayOption; + friend Configurable::QuickloadOption; + public: + Options(Configurable::OptionsType type) : + Configurable::DisplayOption(type == Configurable::OptionsType::UserFriendly ? Configurable::Display::SVideo : Configurable::Display::CompositeColour), + Configurable::QuickloadOption(type == Configurable::OptionsType::UserFriendly) { + if(needs_declare()) { + declare_display_option(); + declare_quickload_option(); + limit_enum(&output, Configurable::Display::SVideo, Configurable::Display::CompositeColour, -1); + } + } + }; }; } diff --git a/Machines/Electron/Electron.hpp b/Machines/Electron/Electron.hpp index e1b047deb..5fd5daaa6 100644 --- a/Machines/Electron/Electron.hpp +++ b/Machines/Electron/Electron.hpp @@ -42,7 +42,7 @@ class Machine { if(needs_declare()) { declare_display_option(); declare_quickload_option(); - limit_enum(&output, Configurable::Display::RGB, Configurable::Display::CompositeColour, -1); + limit_enum(&output, Configurable::Display::RGB, Configurable::Display::CompositeColour, Configurable::Display::CompositeMonochrome, -1); } } }; diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 316ef8211..1681ea1c7 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -177,7 +177,6 @@ std::map> Machine::AllOptionsBy std::map> options; // options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Vic20), Commodore::Vic20::get_options())); #define Emplace(machine, class) \ options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::machine), std::make_unique(Configurable::OptionsType::UserFriendly))); @@ -190,6 +189,7 @@ std::map> Machine::AllOptionsBy Emplace(MasterSystem, Sega::MasterSystem::Machine); Emplace(MSX, MSX::Machine); Emplace(Oric, Oric::Machine); + Emplace(Vic20, Commodore::Vic20::Machine); Emplace(ZX8081, ZX8081::Machine); #undef Emplace From 311458f41facc03c968cfeb2b03e30ef983773b9 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 21:50:02 -0400 Subject: [PATCH 38/48] Restores Macintosh 'runtime' options. Also cleans up some leftover parts elsewhere. --- Configurable/StandardOptions.hpp | 11 +++++ Machines/Apple/Macintosh/Macintosh.cpp | 60 ++++++++------------------ Machines/Apple/Macintosh/Macintosh.hpp | 18 +++++--- Machines/ColecoVision/ColecoVision.cpp | 4 -- Machines/ColecoVision/ColecoVision.hpp | 2 - Machines/Electron/Electron.cpp | 11 ----- Machines/Oric/Oric.hpp | 3 -- Machines/Utility/MachineForTarget.cpp | 3 +- 8 files changed, 44 insertions(+), 68 deletions(-) diff --git a/Configurable/StandardOptions.hpp b/Configurable/StandardOptions.hpp index ef6d3df38..4980b3315 100644 --- a/Configurable/StandardOptions.hpp +++ b/Configurable/StandardOptions.hpp @@ -49,6 +49,17 @@ template class QuickloadOption { } }; +template class QuickbootOption { + public: + bool quickboot; + QuickbootOption(bool quickboot) : quickboot(quickboot) {} + + protected: + void declare_quickboot_option() { + static_cast(this)->declare(&quickboot, "quickboot"); + } +}; + } #endif /* StandardOptions_hpp */ diff --git a/Machines/Apple/Macintosh/Macintosh.cpp b/Machines/Apple/Macintosh/Macintosh.cpp index c71fcf103..890db6619 100644 --- a/Machines/Apple/Macintosh/Macintosh.cpp +++ b/Machines/Apple/Macintosh/Macintosh.cpp @@ -57,16 +57,6 @@ constexpr int CLOCK_RATE = 7833600; namespace Apple { namespace Macintosh { -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::QuickBoot) -// ); -//} - -std::unique_ptr get_options() { - return nullptr; -} - template class ConcreteMachine: public Machine, public CRTMachine::Machine, @@ -528,42 +518,30 @@ template class ConcreteMachin // MARK: - Configuration options. std::unique_ptr get_options() final { - return nullptr; + auto options = std::make_unique(Configurable::OptionsType::UserFriendly); + options->quickboot = quickboot_; + return options; } - void set_options(const std::unique_ptr &options) final { + void set_options(const std::unique_ptr &str) final { + // TODO: should this really be a runtime option? + // It should probably be a construction option. + + const auto options = dynamic_cast(str.get()); + quickboot_ = options->quickboot; + if(quickboot_) { + // Cf. Big Mess o' Wires' disassembly of the Mac Plus ROM, and the + // test at $E00. TODO: adapt as(/if?) necessary for other Macs. + ram_[0x02ae] = 0x40; + ram_[0x02af] = 0x00; + ram_[0x02b0] = 0x00; + ram_[0x02b1] = 0x00; + } } -// std::vector> get_options() final { -// return Apple::Macintosh::get_options(); -// } -// -// void set_selections(const Configurable::SelectionSet &selections_by_option) final { -// bool quick_boot; -// if(Configurable::get_quick_boot(selections_by_option, quick_boot)) { -// if(quick_boot) { -// // Cf. Big Mess o' Wires' disassembly of the Mac Plus ROM, and the -// // test at $E00. TODO: adapt as(/if?) necessary for other Macs. -// ram_[0x02ae] = 0x40; -// ram_[0x02af] = 0x00; -// ram_[0x02b0] = 0x00; -// ram_[0x02b1] = 0x00; -// } -// } -// } -// -// Configurable::SelectionSet get_accurate_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_boot_selection(selection_set, false); -// return selection_set; -// } -// -// Configurable::SelectionSet get_user_friendly_selections() final { -// Configurable::SelectionSet selection_set; -// Configurable::append_quick_boot_selection(selection_set, true); -// return selection_set; -// } private: + bool quickboot_ = false; + void set_component_prefers_clocking(ClockingHint::Source *component, ClockingHint::Preference clocking) final { scsi_bus_is_clocked_ = scsi_bus_.preferred_clocking() != ClockingHint::Preference::None; } diff --git a/Machines/Apple/Macintosh/Macintosh.hpp b/Machines/Apple/Macintosh/Macintosh.hpp index e29665d10..5c42b555f 100644 --- a/Machines/Apple/Macintosh/Macintosh.hpp +++ b/Machines/Apple/Macintosh/Macintosh.hpp @@ -9,23 +9,31 @@ #ifndef Macintosh_hpp #define Macintosh_hpp -#include "../../../Reflection/Struct.h" +#include "../../../Configurable/Configurable.hpp" +#include "../../../Configurable/StandardOptions.hpp" #include "../../../Analyser/Static/StaticAnalyser.hpp" #include "../../ROMMachine.hpp" -#include - namespace Apple { namespace Macintosh { -std::unique_ptr get_options(); - class Machine { public: virtual ~Machine(); /// Creates and returns a Macintosh. static Machine *Macintosh(const Analyser::Static::Target *target, const ROMMachine::ROMFetcher &rom_fetcher); + + class Options: public Reflection::StructImpl, public Configurable::QuickbootOption { + friend Configurable::QuickbootOption; + public: + Options(Configurable::OptionsType type) : + Configurable::QuickbootOption(type == Configurable::OptionsType::UserFriendly) { + if(needs_declare()) { + declare_quickboot_option(); + } + } + }; }; diff --git a/Machines/ColecoVision/ColecoVision.cpp b/Machines/ColecoVision/ColecoVision.cpp index 142c82c47..5a773f883 100644 --- a/Machines/ColecoVision/ColecoVision.cpp +++ b/Machines/ColecoVision/ColecoVision.cpp @@ -34,10 +34,6 @@ constexpr int sn76489_divider = 2; namespace Coleco { namespace Vision { -std::unique_ptr get_options() { - return nullptr; -} - class Joystick: public Inputs::ConcreteJoystick { public: Joystick() : diff --git a/Machines/ColecoVision/ColecoVision.hpp b/Machines/ColecoVision/ColecoVision.hpp index 0e0dc8de5..4e5f0ce2c 100644 --- a/Machines/ColecoVision/ColecoVision.hpp +++ b/Machines/ColecoVision/ColecoVision.hpp @@ -14,8 +14,6 @@ #include "../../Analyser/Static/StaticAnalyser.hpp" #include "../ROMMachine.hpp" -#include - namespace Coleco { namespace Vision { diff --git a/Machines/Electron/Electron.cpp b/Machines/Electron/Electron.cpp index defd1abd5..b65ec7da9 100644 --- a/Machines/Electron/Electron.cpp +++ b/Machines/Electron/Electron.cpp @@ -33,17 +33,6 @@ namespace Electron { -//std::vector> get_options() { -// return Configurable::standard_options( -// static_cast(Configurable::DisplayRGB | Configurable::DisplayCompositeColour | Configurable::QuickLoadTape) -// ); -//} - -std::unique_ptr get_options() { - return nullptr; -} - - class ConcreteMachine: public Machine, public CRTMachine::Machine, diff --git a/Machines/Oric/Oric.hpp b/Machines/Oric/Oric.hpp index d9f8540b1..44167d77d 100644 --- a/Machines/Oric/Oric.hpp +++ b/Machines/Oric/Oric.hpp @@ -18,9 +18,6 @@ namespace Oric { -/// @returns The options available for an Oric. -std::unique_ptr get_options(); - /*! Models an Oric 1/Atmos with or without a Microdisc. */ diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 1681ea1c7..ca306a2c1 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -176,8 +176,6 @@ std::vector Machine::AllMachines(bool meaningful_without_media_only std::map> Machine::AllOptionsByMachineName() { std::map> options; -// options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::Macintosh), Apple::Macintosh::get_options())); - #define Emplace(machine, class) \ options.emplace(std::make_pair(LongNameForTargetMachine(Analyser::Machine::machine), std::make_unique(Configurable::OptionsType::UserFriendly))); @@ -186,6 +184,7 @@ std::map> Machine::AllOptionsBy Emplace(AtariST, Atari::ST::Machine); Emplace(ColecoVision, Coleco::Vision::Machine); Emplace(Electron, Electron::Machine); + Emplace(Macintosh, Apple::Macintosh::Machine); Emplace(MasterSystem, Sega::MasterSystem::Machine); Emplace(MSX, MSX::Machine); Emplace(Oric, Oric::Machine); From 615ea2f5730046da07aeead5ffe38706baecb738 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 22:31:32 -0400 Subject: [PATCH 39/48] Applies parsed arguments. --- .../xcschemes/Clock Signal Kiosk.xcscheme | 6 +- OSBindings/SDL/main.cpp | 96 ++++++++++--------- Reflection/Enum.h | 14 +-- Reflection/Struct.cpp | 55 ++++++++++- Reflection/Struct.h | 7 ++ 5 files changed, 119 insertions(+), 59 deletions(-) diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index b1163abdb..583cb9d88 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -82,7 +82,7 @@ + isEnabled = "NO"> + + diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index 001718065..575047d60 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -39,9 +39,6 @@ namespace { -/// Takes an enum-style camel-case string (e.g. Mac128k) and -//std::string - struct MachineRunner { MachineRunner() { frame_lock_.clear(); @@ -473,7 +470,7 @@ int main(int argc, char *argv[]) { SDL_Window *window = nullptr; // Attempt to parse arguments. - ParsedArguments arguments = parse_arguments(argc, argv); + const ParsedArguments arguments = parse_arguments(argc, argv); // This may be printed either as const std::string usage_suffix = " [file or --new={machine}] [OPTIONS] [--rompath={path to ROMs}] [--speed={speed multiplier, e.g. 1.5}] [--logical-keyboard]"; @@ -622,12 +619,13 @@ int main(int argc, char *argv[]) { "/usr/local/share/CLK/", "/usr/share/CLK/" }; - if(arguments.selections.find("rompath") != arguments.selections.end()) { - const std::string user_path = arguments.selections["rompath"]; - if(user_path.back() != '/') { - paths.push_back(user_path + "/"); + + const auto rompath = arguments.selections.find("rompath"); + if(rompath != arguments.selections.end()) { + if(rompath->second.back() != '/') { + paths.push_back(rompath->second + "/"); } else { - paths.push_back(user_path); + paths.push_back(rompath->second); } } @@ -662,6 +660,20 @@ int main(int argc, char *argv[]) { return results; }; + // Apply all command-line options to the targets. + for(auto &target: targets) { + auto reflectable_target = dynamic_cast(target.get()); + if(!reflectable_target) continue; + + for(const auto &argument: arguments.selections) { + if(argument.second.empty()) { + Reflection::set(*reflectable_target, argument.first, true); + } else { + Reflection::fuzzy_set(*reflectable_target, argument.first, argument.second); + } + } + } + // Create and configure a machine. ::Machine::Error error; std::mutex machine_mutex; @@ -685,19 +697,36 @@ int main(int argc, char *argv[]) { return EXIT_FAILURE; } + // Apply all command-line options to the machines. + auto configurable = machine->configurable_device(); + if(configurable) { + const auto options = configurable->get_options(); + + for(const auto &argument: arguments.selections) { + if(argument.second.empty()) { + Reflection::set(*options, argument.first, true); + } else { + Reflection::fuzzy_set(*options, argument.first, argument.second); + } + } + + configurable->set_options(options); + } + // Apply the speed multiplier, if one was requested. - if(arguments.selections.find("speed") != arguments.selections.end()) { -// const char *speed_string = arguments.selections["speed"]->list_selection()->value.c_str(); -// char *end; -// double speed = strtod(speed_string, &end); -// -// if(size_t(end - speed_string) != strlen(speed_string)) { -// std::cerr << "Unable to parse speed: " << speed_string << std::endl; -// } else if(speed <= 0.0) { -// std::cerr << "Cannot run at speed " << speed_string << "; speeds must be positive." << std::endl; -// } else { -// machine_runner.set_speed_multiplier(speed); -// } + const auto speed_argument = arguments.selections.find("speed"); + if(speed_argument != arguments.selections.end()) { + const char *speed_string = speed_argument->second.c_str(); + char *end; + double speed = strtod(speed_string, &end); + + if(size_t(end - speed_string) != strlen(speed_string)) { + std::cerr << "Unable to parse speed: " << speed_string << std::endl; + } else if(speed <= 0.0) { + std::cerr << "Cannot run at speed " << speed_string << "; speeds must be positive." << std::endl; + } else { + machine_runner.set_speed_multiplier(speed); + } } // Check whether a 'logical' keyboard has been requested. @@ -775,31 +804,6 @@ int main(int argc, char *argv[]) { int window_width, window_height; SDL_GetWindowSize(window, &window_width, &window_height); -/* Configurable::Device *const configurable_device = machine->configurable_device(); - if(configurable_device) { - // Establish user-friendly options by default. - configurable_device->set_selections(configurable_device->get_user_friendly_selections()); - - // Consider transcoding any list selections that map to Boolean options. - for(const auto &option: configurable_device->get_options()) { - // Check for a corresponding selection. - auto selection = arguments.selections.find(option->short_name); - if(selection != arguments.selections.end()) { - // Transcode selection if necessary. - if(dynamic_cast(option.get())) { - arguments.selections[selection->first] = std::unique_ptr(selection->second->boolean_selection()); - } - - if(dynamic_cast(option.get())) { - arguments.selections[selection->first] = std::unique_ptr(selection->second->list_selection()); - } - } - } - - // Apply the user's actual selections to final the defaults. - configurable_device->set_selections(arguments.selections); - }*/ - // If this is a joystick machine, check for and open attached joysticks. /*! Provides a wrapper for SDL_Joystick pointers that can keep track diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 412fb955a..95c808b2c 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -113,13 +113,13 @@ class Enum { @returns A @c std::string name for the enum value @c e. */ template static const std::string &to_string(Type e) { - return to_string(typeid(Type), size_t(e)); + return to_string(typeid(Type), int(e)); } /*! @returns A @c std::string name for the enum value @c e from the enum with type_info @c type. */ - static const std::string &to_string(std::type_index type, size_t e) { + static const std::string &to_string(std::type_index type, int e) { const auto entry = members_by_type_.find(type); if(entry == members_by_type_.end()) return empty_string_; return entry->second[e]; @@ -150,15 +150,15 @@ class Enum { } /*! - @returns A value for the name @c str in the enum with type_info @c type , or @c std::string::npos if + @returns A value for the name @c str in the enum with type_info @c type , or @c -1 if the name is not found. */ - static size_t from_string(std::type_index type, const std::string &str) { + static int from_string(std::type_index type, const std::string &str) { const auto entry = members_by_type_.find(type); - if(entry == members_by_type_.end()) return std::string::npos; + if(entry == members_by_type_.end()) return -1; const auto iterator = std::find(entry->second.begin(), entry->second.end(), str); - if(iterator == entry->second.end()) return std::string::npos; - return size_t(iterator - entry->second.begin()); + if(iterator == entry->second.end()) return -1; + return int(iterator - entry->second.begin()); } private: diff --git a/Reflection/Struct.cpp b/Reflection/Struct.cpp index b3fb8bdfd..dadad2fd8 100644 --- a/Reflection/Struct.cpp +++ b/Reflection/Struct.cpp @@ -8,6 +8,8 @@ #include "Struct.h" +#include + // MARK: - Setters template <> bool Reflection::set(Struct &target, const std::string &name, int value) { @@ -38,13 +40,11 @@ template <> bool Reflection::set(Struct &target, const std::string &name, const return false; } - const auto enum_value = Reflection::Enum::from_string(*target_type, value); - if(enum_value == std::string::npos) { + const int enum_value = Reflection::Enum::from_string(*target_type, value); + if(enum_value < 0) { return false; } - - int int_value = int(enum_value); - target.set(name, &int_value); + target.set(name, &enum_value); return true; } @@ -54,9 +54,54 @@ template <> bool Reflection::set(Struct &target, const std::string &name, const return set(target, name, string); } +template <> bool Reflection::set(Struct &target, const std::string &name, bool value) { + const auto target_type = target.type_of(name); + if(!target_type) return false; + + if(*target_type == typeid(bool)) { + target.set(name, &value);; + } + + return false; +} + // MARK: - Fuzzy setter bool Reflection::fuzzy_set(Struct &target, const std::string &name, const std::string &value) { + const auto target_type = target.type_of(name); + if(!target_type) return false; + + // If the target is a registered enum, ttry to convert the value. Failing that, + // try to match without case sensitivity. + if(Reflection::Enum::size(*target_type)) { + const int from_string = Reflection::Enum::from_string(*target_type, value); + if(from_string >= 0) { + target.set(name, &from_string); + return true; + } + + const auto all_values = Reflection::Enum::all_values(*target_type); + const auto value_location = std::find_if(all_values.begin(), all_values.end(), + [&value] (const auto &entry) { + if(value.size() != entry.size()) return false; + const char *v = value.c_str(); + const char *e = entry.c_str(); + while(*v) { + if(tolower(*v) != tolower(*e)) return false; + ++v; + ++e; + } + return true; + }); + if(value_location != all_values.end()) { + const int offset = int(value_location - all_values.begin()); + target.set(name, &offset); + return true; + } + + return false; + } + return false; } diff --git a/Reflection/Struct.h b/Reflection/Struct.h index ee94ba6ce..7c5cba833 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -56,6 +56,13 @@ template <> bool set(Struct &target, const std::string &name, int value); template <> bool set(Struct &target, const std::string &name, const std::string &value); template <> bool set(Struct &target, const std::string &name, const char *value); +/*! + Setting a bool: + + * to a bool, copies the value. +*/ +template <> bool set(Struct &target, const std::string &name, bool value); + /*! Fuzzy-set attempts to set any property based on a string value. This is intended to allow input provided by the user. From c6f35c9aacfd0e0cc9e28cef676c679210d8471c Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 23:11:25 -0400 Subject: [PATCH 40/48] Rejigs help output. --- Machines/Utility/MachineForTarget.cpp | 37 ++++++----- Machines/Utility/MachineForTarget.hpp | 11 +++- .../xcschemes/Clock Signal Kiosk.xcscheme | 2 +- OSBindings/SDL/main.cpp | 63 +++++++++++-------- 4 files changed, 69 insertions(+), 44 deletions(-) diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index ca306a2c1..253dd4d2a 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -8,6 +8,8 @@ #include "MachineForTarget.hpp" +#include + // Sources for runtime options. #include "../AmstradCPC/AmstradCPC.hpp" #include "../Apple/AppleII/AppleII.hpp" @@ -148,28 +150,33 @@ std::string Machine::LongNameForTargetMachine(Analyser::Machine machine) { } } -std::vector Machine::AllMachines(bool meaningful_without_media_only, bool long_names) { +std::vector Machine::AllMachines(Type type, bool long_names) { std::vector result; #define AddName(x) result.push_back(long_names ? LongNameForTargetMachine(Analyser::Machine::x) : ShortNameForTargetMachine(Analyser::Machine::x)) -#define AddConditionalName(x) if(!meaningful_without_media_only) result.push_back(long_names ? LongNameForTargetMachine(Analyser::Machine::x) : ShortNameForTargetMachine(Analyser::Machine::x)) - AddName(AmstradCPC); - AddName(AppleII); - AddConditionalName(Atari2600); - AddName(AtariST); - AddConditionalName(ColecoVision); - AddName(Electron); - AddName(Macintosh); - AddConditionalName(MasterSystem); - AddName(MSX); - AddName(Oric); - AddName(Vic20); - AddName(ZX8081); + if(type == Type::Any || type == Type::DoesntRequireMedia) { + AddName(AmstradCPC); + AddName(AppleII); + AddName(AtariST); + AddName(Electron); + AddName(Macintosh); + AddName(MSX); + AddName(Oric); + AddName(Vic20); + AddName(ZX8081); + } + + if(type == Type::Any || type == Type::RequiresMedia) { + AddName(Atari2600); + AddName(ColecoVision); + AddName(MasterSystem); + } -#undef AddConditionalName #undef AddName + std::sort(result.begin(), result.end()); + return result; } diff --git a/Machines/Utility/MachineForTarget.hpp b/Machines/Utility/MachineForTarget.hpp index c1228d610..5d481e0d4 100644 --- a/Machines/Utility/MachineForTarget.hpp +++ b/Machines/Utility/MachineForTarget.hpp @@ -65,14 +65,19 @@ std::string ShortNameForTargetMachine(const Analyser::Machine target); */ std::string LongNameForTargetMachine(const Analyser::Machine target); +enum class Type { + RequiresMedia, + DoesntRequireMedia, + Any +}; + /*! - @param meaningful_without_media_only If this is @c true then only machines that it is meaningful to start without a piece of media will be listed; - otherwise all supported machines will be listed. + @param type the type of machines to include. @param long_names If this is @c true then long names will be returned; otherwise short names will be returned. @returns A list of all available machines. Names are always guaranteed to be in the same order. */ -std::vector AllMachines(bool meaningful_without_media_only, bool long_names); +std::vector AllMachines(Type type, bool long_names); /*! Returns a map from long machine name to the list of options that machine exposes, for all machines. diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 583cb9d88..e58d5fee5 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -90,7 +90,7 @@ + isEnabled = "YES"> selections; // The empty string will be inserted for arguments without an = suffix. + + void apply(Reflection::Struct *reflectable) const { + for(const auto &argument: selections) { + // Replace any dashes with underscores in the argument name. + std::string property; + std::transform(argument.first.begin(), argument.first.end(), std::back_inserter(property), [](char c) { return c == '-' ? '_' : c; }); + + if(argument.second.empty()) { + Reflection::set(*reflectable, property, true); + } else { + Reflection::fuzzy_set(*reflectable, property, argument.second); + } + } + } }; /*! Parses an argc/argv pair to discern program arguments. */ @@ -387,7 +401,7 @@ ParsedArguments parse_arguments(int argc, char *argv[]) { if(split_index == std::string::npos) { arguments.selections[argument]; // To create an entry with the default empty string. } else { - std::string name = argument.substr(0, split_index); + const std::string name = argument.substr(0, split_index); std::string value = argument.substr(split_index+1, std::string::npos); arguments.selections[name] = value; } @@ -477,11 +491,11 @@ int main(int argc, char *argv[]) { // Print a help message if requested. if(arguments.selections.find("help") != arguments.selections.end() || arguments.selections.find("h") != arguments.selections.end()) { - const auto all_machines = Machine::AllMachines(false, false); + const auto all_machines = Machine::AllMachines(Machine::Type::DoesntRequireMedia, false); std::cout << "Usage: " << final_path_component(argv[0]) << usage_suffix << std::endl; std::cout << "Use alt+enter to toggle full screen display. Use control+shift+V to paste text." << std::endl; - std::cout << "Required machine type and configuration is determined from the file if specified; otherwise use:" << std::endl << std::endl; + std::cout << "Required machine type **and all options** are determined from the file if specified; otherwise use:" << std::endl << std::endl; std::cout << "\t--new={"; bool is_first = true; for(const auto &name: all_machines) { @@ -491,11 +505,21 @@ int main(int argc, char *argv[]) { } std::cout << "}" << std::endl << std::endl; - std::cout << "Further machine options:" << std::endl; + std::cout << "Media is required to start the: "; + const auto other_machines = Machine::AllMachines(Machine::Type::RequiresMedia, true); + is_first = true; + for(const auto &name: other_machines) { + if(!is_first) std::cout << ", "; + std::cout << name; + is_first = false; + } + std::cout << "." << std::endl << std::endl; + + std::cout << "Further machine options:" << std::endl << std::endl;; const auto targets = Machine::TargetsByMachineName(false); const auto runtime_options = Machine::AllOptionsByMachineName(); - const auto machine_names = Machine::AllMachines(false, true); + const auto machine_names = Machine::AllMachines(Machine::Type::Any, true); for(const auto &machine: machine_names) { const auto target = targets.find(machine); const auto options = runtime_options.find(machine); @@ -522,7 +546,10 @@ int main(int argc, char *argv[]) { std::sort(all_options.begin(), all_options.end()); for(const auto &option: all_options) { - std::cout << '\t' << "--" << option; + // Replace any underscores with hyphens, better to conform to command-line norms. + std::string mapped_option; + std::transform(option.begin(), option.end(), std::back_inserter(mapped_option), [](char c) { return c == '_' ? '-' : c; }); + std::cout << '\t' << "--" << mapped_option; auto source = target_reflectable; auto type = target_reflectable ? target_reflectable->type_of(option) : nullptr; @@ -538,6 +565,7 @@ int main(int argc, char *argv[]) { for(const auto &value: source->values_for(option)) { if(!is_first) std::cout << '|'; is_first = false; + std::cout << value; } std::cout << "}"; @@ -564,7 +592,7 @@ int main(int argc, char *argv[]) { const auto new_argument = arguments.selections.find("new"); if(new_argument != arguments.selections.end() && !new_argument->second.empty()) { // Perform for a case-insensitive search against short names. - const auto short_names = Machine::AllMachines(false, false); + const auto short_names = Machine::AllMachines(Machine::Type::DoesntRequireMedia, false); auto short_name = short_names.begin(); while(short_name != short_names.end()) { if(std::equal( @@ -579,7 +607,7 @@ int main(int argc, char *argv[]) { // If a match was found, use the corresponding long name to look up a suitable // Analyser::Statuc::Target and move that to the targets list. if(short_name != short_names.end()) { - const auto long_name = Machine::AllMachines(false, true)[short_name - short_names.begin()]; + const auto long_name = Machine::AllMachines(Machine::Type::DoesntRequireMedia, true)[short_name - short_names.begin()]; auto targets_by_machine = Machine::TargetsByMachineName(false); std::unique_ptr tgt = std::move(targets_by_machine[long_name]); targets.push_back(std::move(tgt)); @@ -664,14 +692,7 @@ int main(int argc, char *argv[]) { for(auto &target: targets) { auto reflectable_target = dynamic_cast(target.get()); if(!reflectable_target) continue; - - for(const auto &argument: arguments.selections) { - if(argument.second.empty()) { - Reflection::set(*reflectable_target, argument.first, true); - } else { - Reflection::fuzzy_set(*reflectable_target, argument.first, argument.second); - } - } + arguments.apply(reflectable_target); } // Create and configure a machine. @@ -701,15 +722,7 @@ int main(int argc, char *argv[]) { auto configurable = machine->configurable_device(); if(configurable) { const auto options = configurable->get_options(); - - for(const auto &argument: arguments.selections) { - if(argument.second.empty()) { - Reflection::set(*options, argument.first, true); - } else { - Reflection::fuzzy_set(*options, argument.first, argument.second); - } - } - + arguments.apply(options.get()); configurable->set_options(options); } From 9995d776de4b38f2d4c6096e5ae964d2b57accee Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Wed, 18 Mar 2020 23:29:09 -0400 Subject: [PATCH 41/48] Attempts to fix the macOS version, plus some implicit type conversions. --- .../Mac/Clock Signal/Machine/CSMachine.mm | 63 ++++++++----------- Reflection/Enum.h | 2 +- Reflection/Struct.h | 4 +- 3 files changed, 29 insertions(+), 40 deletions(-) diff --git a/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm b/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm index c61e9e699..a3f3f27ae 100644 --- a/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm +++ b/OSBindings/Mac/Clock Signal/Machine/CSMachine.mm @@ -586,9 +586,9 @@ struct ActivityObserver: public Activity::Observer { @synchronized(self) { _useFastLoadingHack = useFastLoadingHack; - Configurable::SelectionSet selection_set; - append_quick_load_tape_selection(selection_set, useFastLoadingHack ? true : false); - configurable_device->set_selections(selection_set); + auto options = configurable_device->get_options(); + Reflection::set(*options, "quickload", useFastLoadingHack ? true : false); + configurable_device->set_options(options); } } @@ -599,7 +599,6 @@ struct ActivityObserver: public Activity::Observer { @synchronized(self) { _videoSignal = videoSignal; - Configurable::SelectionSet selection_set; Configurable::Display display; switch(videoSignal) { case CSMachineVideoSignalRGB: display = Configurable::Display::RGB; break; @@ -607,8 +606,10 @@ struct ActivityObserver: public Activity::Observer { case CSMachineVideoSignalComposite: display = Configurable::Display::CompositeColour; break; case CSMachineVideoSignalMonochromeComposite: display = Configurable::Display::CompositeMonochrome; break; } - append_display_selection(selection_set, display); - configurable_device->set_selections(selection_set); + + auto options = configurable_device->get_options(); + Reflection::set(*options, "output", int(display)); + configurable_device->set_options(options); } } @@ -617,35 +618,23 @@ struct ActivityObserver: public Activity::Observer { if(!configurable_device) return NO; // Get the options this machine provides. - std::vector> options; @synchronized(self) { - options = configurable_device->get_options(); - } + auto options = configurable_device->get_options(); - // Get the standard option for this video signal. - Configurable::StandardOptions option; - switch(videoSignal) { - case CSMachineVideoSignalRGB: option = Configurable::DisplayRGB; break; - case CSMachineVideoSignalSVideo: option = Configurable::DisplaySVideo; break; - case CSMachineVideoSignalComposite: option = Configurable::DisplayCompositeColour; break; - case CSMachineVideoSignalMonochromeComposite: option = Configurable::DisplayCompositeMonochrome; break; - } - std::unique_ptr display_option = std::move(standard_options(option).front()); - Configurable::ListOption *display_list_option = dynamic_cast(display_option.get()); - NSAssert(display_list_option, @"Expected display option to be a list"); + // Get the standard option for this video signal. + Configurable::Display option; + switch(videoSignal) { + case CSMachineVideoSignalRGB: option = Configurable::Display::RGB; break; + case CSMachineVideoSignalSVideo: option = Configurable::Display::SVideo; break; + case CSMachineVideoSignalComposite: option = Configurable::Display::CompositeColour; break; + case CSMachineVideoSignalMonochromeComposite: option = Configurable::Display::CompositeMonochrome; break; + } - // See whether the video signal is included in the machine options. - for(auto &candidate: options) { - Configurable::ListOption *list_option = dynamic_cast(candidate.get()); + // Map to a string and check against returned options for the 'output' field. + const auto string_option = Reflection::Enum::to_string(option); + const auto all_values = options->values_for("output"); - // Both should be list options - if(!list_option) continue; - - // Check for same name of option. - if(candidate->short_name != display_option->short_name) continue; - - // Check that the video signal option is included. - return std::find(list_option->options.begin(), list_option->options.end(), display_list_option->options.front()) != list_option->options.end(); + return std::find(all_values.begin(), all_values.end(), string_option) != all_values.end(); } return NO; @@ -658,9 +647,9 @@ struct ActivityObserver: public Activity::Observer { @synchronized(self) { _useAutomaticTapeMotorControl = useAutomaticTapeMotorControl; - Configurable::SelectionSet selection_set; - append_automatic_tape_motor_control_selection(selection_set, useAutomaticTapeMotorControl ? true : false); - configurable_device->set_selections(selection_set); + auto options = configurable_device->get_options(); + Reflection::set(*options, "automatic_tape_motor_control", useAutomaticTapeMotorControl ? true : false); + configurable_device->set_options(options); } } @@ -671,9 +660,9 @@ struct ActivityObserver: public Activity::Observer { @synchronized(self) { _useQuickBootingHack = useQuickBootingHack; - Configurable::SelectionSet selection_set; - append_quick_boot_selection(selection_set, useQuickBootingHack ? true : false); - configurable_device->set_selections(selection_set); + auto options = configurable_device->get_options(); + Reflection::set(*options, "quickboot", useQuickBootingHack ? true : false); + configurable_device->set_options(options); } } diff --git a/Reflection/Enum.h b/Reflection/Enum.h index 95c808b2c..ede607246 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.h @@ -122,7 +122,7 @@ class Enum { static const std::string &to_string(std::type_index type, int e) { const auto entry = members_by_type_.find(type); if(entry == members_by_type_.end()) return empty_string_; - return entry->second[e]; + return entry->second[size_t(e)]; } /*! diff --git a/Reflection/Struct.h b/Reflection/Struct.h index 7c5cba833..a1a2d99d4 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.h @@ -219,10 +219,10 @@ template class StructImpl: public Struct { const int next = va_arg(list, int); if(next < 0) break; - if(permitted_values.size() <= next) { + if(permitted_values.size() <= size_t(next)) { permitted_values.resize(permitted_values.size() << 1); } - permitted_values[next] = true; + permitted_values[size_t(next)] = true; } va_end(list); From b8ebdc012f17803a183898b3635852ac9d80421b Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 19 Mar 2020 18:58:36 -0400 Subject: [PATCH 42/48] Ensure normative construction declaration ordering. --- Machines/ZX8081/ZX8081.hpp | 4 ++-- .../xcshareddata/xcschemes/Clock Signal.xcscheme | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/Machines/ZX8081/ZX8081.hpp b/Machines/ZX8081/ZX8081.hpp index e6933afef..5fff0ffe8 100644 --- a/Machines/ZX8081/ZX8081.hpp +++ b/Machines/ZX8081/ZX8081.hpp @@ -35,8 +35,8 @@ class Machine { bool automatic_tape_motor_control; Options(Configurable::OptionsType type): - automatic_tape_motor_control(type == Configurable::OptionsType::UserFriendly), - Configurable::QuickloadOption(type == Configurable::OptionsType::UserFriendly) { + Configurable::QuickloadOption(type == Configurable::OptionsType::UserFriendly), + automatic_tape_motor_control(type == Configurable::OptionsType::UserFriendly) { // Declare fields if necessary. if(needs_declare()) { 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 f049730f4..7f370216e 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal.xcscheme @@ -67,7 +67,7 @@ Date: Thu, 19 Mar 2020 19:41:50 -0400 Subject: [PATCH 43/48] Ensures consistent ordering. --- Machines/Utility/MachineForTarget.cpp | 14 ++++++-------- 1 file changed, 6 insertions(+), 8 deletions(-) diff --git a/Machines/Utility/MachineForTarget.cpp b/Machines/Utility/MachineForTarget.cpp index 253dd4d2a..2d721a4dc 100644 --- a/Machines/Utility/MachineForTarget.cpp +++ b/Machines/Utility/MachineForTarget.cpp @@ -155,6 +155,12 @@ std::vector Machine::AllMachines(Type type, bool long_names) { #define AddName(x) result.push_back(long_names ? LongNameForTargetMachine(Analyser::Machine::x) : ShortNameForTargetMachine(Analyser::Machine::x)) + if(type == Type::Any || type == Type::RequiresMedia) { + AddName(Atari2600); + AddName(ColecoVision); + AddName(MasterSystem); + } + if(type == Type::Any || type == Type::DoesntRequireMedia) { AddName(AmstradCPC); AddName(AppleII); @@ -167,16 +173,8 @@ std::vector Machine::AllMachines(Type type, bool long_names) { AddName(ZX8081); } - if(type == Type::Any || type == Type::RequiresMedia) { - AddName(Atari2600); - AddName(ColecoVision); - AddName(MasterSystem); - } - #undef AddName - std::sort(result.begin(), result.end()); - return result; } From c1b69fd09103e9775391d82192d702f251a5d0a2 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 19 Mar 2020 20:40:43 -0400 Subject: [PATCH 44/48] Attempts to support multiple pieces of media on the SDL command line, ensures proper window titling. --- Analyser/Static/StaticAnalyser.hpp | 10 +++++ .../xcschemes/Clock Signal Kiosk.xcscheme | 2 +- OSBindings/SDL/main.cpp | 38 ++++++++++++++----- 3 files changed, 40 insertions(+), 10 deletions(-) diff --git a/Analyser/Static/StaticAnalyser.hpp b/Analyser/Static/StaticAnalyser.hpp index b9a62770d..e806ae042 100644 --- a/Analyser/Static/StaticAnalyser.hpp +++ b/Analyser/Static/StaticAnalyser.hpp @@ -35,6 +35,16 @@ struct Media { bool empty() const { return disks.empty() && tapes.empty() && cartridges.empty() && mass_storage_devices.empty(); } + + Media &operator +=(const Media &rhs) { +#define append(name) name.insert(name.end(), rhs.name.begin(), rhs.name.end()); + append(disks); + append(tapes); + append(cartridges); + append(mass_storage_devices); +#undef append + return *this; + } }; /*! diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index e58d5fee5..583cb9d88 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -90,7 +90,7 @@ + isEnabled = "NO"> file_names; std::map selections; // The empty string will be inserted for arguments without an = suffix. void apply(Reflection::Struct *reflectable) const { @@ -406,7 +406,7 @@ ParsedArguments parse_arguments(int argc, char *argv[]) { arguments.selections[name] = value; } } else { - arguments.file_name = arg; + arguments.file_names.push_back(arg); } } @@ -585,11 +585,17 @@ int main(int argc, char *argv[]) { // Determine the machine for the supplied file, if any, or from --new. Analyser::Static::TargetList targets; - if(!arguments.file_name.empty()) { - targets = Analyser::Static::GetTargets(arguments.file_name); + if(!arguments.file_names.empty()) { + // Take the first file name that actually implies a machine. + auto file_name = arguments.file_names.begin(); + while(file_name != arguments.file_names.end() && targets.empty()) { + targets = Analyser::Static::GetTargets(*file_name); + ++file_name; + } } const auto new_argument = arguments.selections.find("new"); + std::string long_machine_name; if(new_argument != arguments.selections.end() && !new_argument->second.empty()) { // Perform for a case-insensitive search against short names. const auto short_names = Machine::AllMachines(Machine::Type::DoesntRequireMedia, false); @@ -607,16 +613,23 @@ int main(int argc, char *argv[]) { // If a match was found, use the corresponding long name to look up a suitable // Analyser::Statuc::Target and move that to the targets list. if(short_name != short_names.end()) { - const auto long_name = Machine::AllMachines(Machine::Type::DoesntRequireMedia, true)[short_name - short_names.begin()]; + long_machine_name = Machine::AllMachines(Machine::Type::DoesntRequireMedia, true)[short_name - short_names.begin()]; auto targets_by_machine = Machine::TargetsByMachineName(false); - std::unique_ptr tgt = std::move(targets_by_machine[long_name]); + std::unique_ptr tgt = std::move(targets_by_machine[long_machine_name]); targets.push_back(std::move(tgt)); } } if(targets.empty()) { - if(!arguments.file_name.empty()) { - std::cerr << "Cannot open " << arguments.file_name << "; no target machine found" << std::endl; + if(!arguments.file_names.empty()) { + std::cerr << "Cannot open "; + bool is_first = true; + for(const auto &name: arguments.file_names) { + if(!is_first) std::cerr << ", "; + is_first = false; + std::cerr << name; + } + std::cerr << "; no target machine found" << std::endl; return EXIT_FAILURE; } @@ -752,6 +765,13 @@ int main(int argc, char *argv[]) { machine_runner.machine = machine.get(); machine_runner.machine_mutex = &machine_mutex; + // Ensure all media is inserted. + Analyser::Static::Media media; + for(const auto &file_name: arguments.file_names) { + media += Analyser::Static::GetMedia(file_name); + } + machine->media_target()->insert_media(media); + // Attempt to set up video and audio. if(SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO) < 0) { std::cerr << "SDL could not initialize! SDL_Error: " << SDL_GetError() << std::endl; @@ -765,7 +785,7 @@ int main(int argc, char *argv[]) { SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 2); SDL_GL_SetSwapInterval(1); - window = SDL_CreateWindow( final_path_component(arguments.file_name).c_str(), + window = SDL_CreateWindow( long_machine_name.empty() ? final_path_component(arguments.file_names.front()).c_str() : long_machine_name.c_str(), SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, 400, 300, SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE); From 23aa7ea85f95668972d4e6cb16b48e78d6b29c53 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 19 Mar 2020 21:02:14 -0400 Subject: [PATCH 45/48] Revives MultiConfigurable. --- .../Implementation/MultiConfigurable.cpp | 127 +++++++++++------- .../Implementation/MultiConfigurable.hpp | 4 - 2 files changed, 79 insertions(+), 52 deletions(-) diff --git a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp index c151d52a7..6688debba 100644 --- a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp +++ b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.cpp @@ -12,6 +12,81 @@ using namespace Analyser::Dynamic; +namespace { + +class MultiStruct: public Reflection::Struct { + public: + MultiStruct(const std::vector &devices) : devices_(devices) { + for(auto device: devices) { + options_.emplace_back(device->get_options()); + } + } + + void apply() { + auto options = options_.begin(); + for(auto device: devices_) { + device->set_options(*options); + ++options; + } + } + + std::vector all_keys() final { + std::set keys; + for(auto &options: options_) { + const auto new_keys = options->all_keys(); + keys.insert(new_keys.begin(), new_keys.end()); + } + return std::vector(keys.begin(), keys.end()); + } + + std::vector values_for(const std::string &name) final { + std::set values; + for(auto &options: options_) { + const auto new_values = options->values_for(name); + values.insert(new_values.begin(), new_values.end()); + } + return std::vector(values.begin(), values.end()); + } + + const std::type_info *type_of(const std::string &name) final { + for(auto &options: options_) { + auto info = options->type_of(name); + if(info) return info; + } + return nullptr; + } + + const void *get(const std::string &name) final { + for(auto &options: options_) { + auto value = options->get(name); + if(value) return value; + } + return nullptr; + } + + void set(const std::string &name, const void *value) final { + const auto safe_type = type_of(name); + if(!safe_type) return; + + // Set this property only where the child's type is the same as that + // which was returned from here for type_of. + for(auto &options: options_) { + const auto type = options->type_of(name); + if(!type) continue; + + if(*type == *safe_type) { + options->set(name, value); + } + } + } + + private: + const std::vector &devices_; + std::vector> options_; +}; + +} + MultiConfigurable::MultiConfigurable(const std::vector> &machines) { for(const auto &machine: machines) { Configurable::Device *device = machine->configurable_device(); @@ -19,55 +94,11 @@ MultiConfigurable::MultiConfigurable(const std::vector &options) { +void MultiConfigurable::set_options(const std::unique_ptr &str) { + const auto options = dynamic_cast(str.get()); + options->apply(); } std::unique_ptr MultiConfigurable::get_options() { - // TODO: this'll need to mash options together, maybe? Or just take the front? - return nullptr; + return std::make_unique(devices_); } - -/*std::vector> MultiConfigurable::get_options() { - std::vector> options; - - // Produce the list of unique options. - for(const auto &device : devices_) { - std::vector> device_options = device->get_options(); - for(auto &option : device_options) { - if(std::find(options.begin(), options.end(), option) == options.end()) { - options.push_back(std::move(option)); - } - } - } - - return options; -} - -void MultiConfigurable::set_selections(const Configurable::SelectionSet &selection_by_option) { - for(const auto &device : devices_) { - device->set_selections(selection_by_option); - } -} - -Configurable::SelectionSet MultiConfigurable::get_accurate_selections() { - Configurable::SelectionSet set; - for(const auto &device : devices_) { - Configurable::SelectionSet device_set = device->get_accurate_selections(); - for(auto &selection : device_set) { - set.insert(std::move(selection)); - } - } - return set; -} - -Configurable::SelectionSet MultiConfigurable::get_user_friendly_selections() { - Configurable::SelectionSet set; - for(const auto &device : devices_) { - Configurable::SelectionSet device_set = device->get_user_friendly_selections(); - for(auto &selection : device_set) { - set.insert(std::move(selection)); - } - } - return set; -} -*/ diff --git a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp index 88c264e52..2595fc69c 100644 --- a/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp +++ b/Analyser/Dynamic/MultiMachine/Implementation/MultiConfigurable.hpp @@ -31,10 +31,6 @@ class MultiConfigurable: public Configurable::Device { // Below is the standard Configurable::Device interface; see there for documentation. void set_options(const std::unique_ptr &options) final; std::unique_ptr get_options() final; -// std::vector> get_options() final; -// void set_selections(const Configurable::SelectionSet &selection_by_option) final; -// Configurable::SelectionSet get_accurate_selections() final; -// Configurable::SelectionSet get_user_friendly_selections() final; private: std::vector devices_; From eef78681994989607a5499c588bb487cde2881c3 Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 19 Mar 2020 21:15:38 -0400 Subject: [PATCH 46/48] Ensures 'new' overrides default selection; doesn't try to propagate multiple files if machines won't take them. --- .../xcschemes/Clock Signal Kiosk.xcscheme | 6 ++-- OSBindings/SDL/main.cpp | 30 +++++++++++-------- 2 files changed, 20 insertions(+), 16 deletions(-) diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 583cb9d88..1e872a546 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -58,7 +58,7 @@ + isEnabled = "NO"> + isEnabled = "YES"> + isEnabled = "YES"> tgt = std::move(targets_by_machine[long_machine_name]); targets.push_back(std::move(tgt)); } + } else if(!arguments.file_names.empty()) { + // Take the first file name that actually implies a machine. + auto file_name = arguments.file_names.begin(); + while(file_name != arguments.file_names.end() && targets.empty()) { + targets = Analyser::Static::GetTargets(*file_name); + ++file_name; + } } if(targets.empty()) { @@ -765,12 +764,17 @@ int main(int argc, char *argv[]) { machine_runner.machine = machine.get(); machine_runner.machine_mutex = &machine_mutex; - // Ensure all media is inserted. - Analyser::Static::Media media; - for(const auto &file_name: arguments.file_names) { - media += Analyser::Static::GetMedia(file_name); + // Ensure all media is inserted, if this machine accepts it. + { + auto media_target = machine->media_target(); + if(media_target) { + Analyser::Static::Media media; + for(const auto &file_name: arguments.file_names) { + media += Analyser::Static::GetMedia(file_name); + } + media_target->insert_media(media); + } } - machine->media_target()->insert_media(media); // Attempt to set up video and audio. if(SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO) < 0) { From b8b335f67d7e2b848fee2cc3911ff7a2430c042d Mon Sep 17 00:00:00 2001 From: Thomas Harte Date: Thu, 19 Mar 2020 21:46:42 -0400 Subject: [PATCH 47/48] Exposes the Master System's region for SDL selection. --- Analyser/Static/Macintosh/Target.hpp | 1 + Analyser/Static/Sega/Target.hpp | 15 +++++++++++---- .../xcschemes/Clock Signal Kiosk.xcscheme | 12 ++++++++---- 3 files changed, 20 insertions(+), 8 deletions(-) diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 905739429..243ccf36e 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -11,6 +11,7 @@ #include "../../../Reflection/Enum.h" #include "../../../Reflection/Struct.h" +#include "../StaticAnalyser.hpp" namespace Analyser { namespace Static { diff --git a/Analyser/Static/Sega/Target.hpp b/Analyser/Static/Sega/Target.hpp index 1735704ed..f0926d689 100644 --- a/Analyser/Static/Sega/Target.hpp +++ b/Analyser/Static/Sega/Target.hpp @@ -9,25 +9,27 @@ #ifndef Analyser_Static_Sega_Target_h #define Analyser_Static_Sega_Target_h +#include "../../../Reflection/Enum.h" +#include "../../../Reflection/Struct.h" #include "../StaticAnalyser.hpp" namespace Analyser { namespace Static { namespace Sega { -struct Target: public Analyser::Static::Target { +struct Target: public Analyser::Static::Target, public Reflection::StructImpl { enum class Model { SG1000, MasterSystem, MasterSystem2, }; - enum class Region { + ReflectableEnum(Region, Japan, USA, Europe, Brazil - }; + ); enum class PagingScheme { Sega, @@ -38,7 +40,12 @@ struct Target: public Analyser::Static::Target { Region region = Region::Japan; PagingScheme paging_scheme = PagingScheme::Sega; - Target() : Analyser::Static::Target(Machine::MasterSystem) {} + Target() : Analyser::Static::Target(Machine::MasterSystem) { + if(needs_declare()) { + DeclareField(region); + AnnounceEnum(Region); + } + } }; #define is_master_system(v) v >= Analyser::Static::Sega::Target::Model::MasterSystem diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme index 1e872a546..66ee95aed 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme +++ b/OSBindings/Mac/Clock Signal.xcodeproj/xcshareddata/xcschemes/Clock Signal Kiosk.xcscheme @@ -66,7 +66,11 @@ + isEnabled = "YES"> + + + isEnabled = "NO"> + isEnabled = "NO"> + isEnabled = "YES"> Date: Thu, 19 Mar 2020 23:24:06 -0400 Subject: [PATCH 48/48] Renames files to match project convention. --- Analyser/Static/Acorn/Target.hpp | 2 +- Analyser/Static/AmstradCPC/Target.hpp | 4 ++-- Analyser/Static/AppleII/Target.hpp | 4 ++-- Analyser/Static/AtariST/Target.hpp | 2 +- Analyser/Static/Commodore/Target.hpp | 4 ++-- Analyser/Static/MSX/Target.hpp | 4 ++-- Analyser/Static/Macintosh/Target.hpp | 4 ++-- Analyser/Static/Oric/Target.hpp | 4 ++-- Analyser/Static/Sega/Target.hpp | 4 ++-- Analyser/Static/ZX8081/Target.hpp | 4 ++-- Configurable/Configurable.hpp | 2 +- Configurable/StandardOptions.hpp | 2 +- Machines/Utility/MachineForTarget.hpp | 2 +- OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj | 8 ++++---- OSBindings/SDL/main.cpp | 4 ++-- Reflection/{Enum.h => Enum.hpp} | 8 ++++---- Reflection/Struct.cpp | 2 +- Reflection/{Struct.h => Struct.hpp} | 10 +++++----- 18 files changed, 37 insertions(+), 37 deletions(-) rename Reflection/{Enum.h => Enum.hpp} (98%) rename Reflection/{Struct.h => Struct.hpp} (98%) diff --git a/Analyser/Static/Acorn/Target.hpp b/Analyser/Static/Acorn/Target.hpp index f049f3e8f..1ef46c291 100644 --- a/Analyser/Static/Acorn/Target.hpp +++ b/Analyser/Static/Acorn/Target.hpp @@ -9,7 +9,7 @@ #ifndef Analyser_Static_Acorn_Target_h #define Analyser_Static_Acorn_Target_h -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" #include diff --git a/Analyser/Static/AmstradCPC/Target.hpp b/Analyser/Static/AmstradCPC/Target.hpp index 586398e6e..6f708d4b0 100644 --- a/Analyser/Static/AmstradCPC/Target.hpp +++ b/Analyser/Static/AmstradCPC/Target.hpp @@ -9,8 +9,8 @@ #ifndef Analyser_Static_AmstradCPC_Target_h #define Analyser_Static_AmstradCPC_Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" #include diff --git a/Analyser/Static/AppleII/Target.hpp b/Analyser/Static/AppleII/Target.hpp index eb67b2af2..988e7d4ee 100644 --- a/Analyser/Static/AppleII/Target.hpp +++ b/Analyser/Static/AppleII/Target.hpp @@ -9,8 +9,8 @@ #ifndef Target_h #define Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" namespace Analyser { diff --git a/Analyser/Static/AtariST/Target.hpp b/Analyser/Static/AtariST/Target.hpp index 4933c8776..fd025baf8 100644 --- a/Analyser/Static/AtariST/Target.hpp +++ b/Analyser/Static/AtariST/Target.hpp @@ -9,7 +9,7 @@ #ifndef Analyser_Static_AtariST_Target_h #define Analyser_Static_AtariST_Target_h -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" namespace Analyser { diff --git a/Analyser/Static/Commodore/Target.hpp b/Analyser/Static/Commodore/Target.hpp index dd08b87d9..9da8506ec 100644 --- a/Analyser/Static/Commodore/Target.hpp +++ b/Analyser/Static/Commodore/Target.hpp @@ -9,8 +9,8 @@ #ifndef Analyser_Static_Commodore_Target_h #define Analyser_Static_Commodore_Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" #include diff --git a/Analyser/Static/MSX/Target.hpp b/Analyser/Static/MSX/Target.hpp index 7e16ae25a..f0d7a9f6a 100644 --- a/Analyser/Static/MSX/Target.hpp +++ b/Analyser/Static/MSX/Target.hpp @@ -9,8 +9,8 @@ #ifndef Analyser_Static_MSX_Target_h #define Analyser_Static_MSX_Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" #include diff --git a/Analyser/Static/Macintosh/Target.hpp b/Analyser/Static/Macintosh/Target.hpp index 243ccf36e..f00cb4a24 100644 --- a/Analyser/Static/Macintosh/Target.hpp +++ b/Analyser/Static/Macintosh/Target.hpp @@ -9,8 +9,8 @@ #ifndef Analyser_Static_Macintosh_Target_h #define Analyser_Static_Macintosh_Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" namespace Analyser { diff --git a/Analyser/Static/Oric/Target.hpp b/Analyser/Static/Oric/Target.hpp index e0f81f6d7..66aead3dd 100644 --- a/Analyser/Static/Oric/Target.hpp +++ b/Analyser/Static/Oric/Target.hpp @@ -9,8 +9,8 @@ #ifndef Analyser_Static_Oric_Target_h #define Analyser_Static_Oric_Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" #include diff --git a/Analyser/Static/Sega/Target.hpp b/Analyser/Static/Sega/Target.hpp index f0926d689..3eceb774d 100644 --- a/Analyser/Static/Sega/Target.hpp +++ b/Analyser/Static/Sega/Target.hpp @@ -9,8 +9,8 @@ #ifndef Analyser_Static_Sega_Target_h #define Analyser_Static_Sega_Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" namespace Analyser { diff --git a/Analyser/Static/ZX8081/Target.hpp b/Analyser/Static/ZX8081/Target.hpp index 3461974ed..c7dd7e60d 100644 --- a/Analyser/Static/ZX8081/Target.hpp +++ b/Analyser/Static/ZX8081/Target.hpp @@ -9,8 +9,8 @@ #ifndef Analyser_Static_ZX8081_Target_h #define Analyser_Static_ZX8081_Target_h -#include "../../../Reflection/Enum.h" -#include "../../../Reflection/Struct.h" +#include "../../../Reflection/Enum.hpp" +#include "../../../Reflection/Struct.hpp" #include "../StaticAnalyser.hpp" #include diff --git a/Configurable/Configurable.hpp b/Configurable/Configurable.hpp index cbb037ff3..d1aebbdfa 100644 --- a/Configurable/Configurable.hpp +++ b/Configurable/Configurable.hpp @@ -9,7 +9,7 @@ #ifndef Configurable_h #define Configurable_h -#include "../Reflection/Struct.h" +#include "../Reflection/Struct.hpp" #include diff --git a/Configurable/StandardOptions.hpp b/Configurable/StandardOptions.hpp index 4980b3315..abc7c284c 100644 --- a/Configurable/StandardOptions.hpp +++ b/Configurable/StandardOptions.hpp @@ -9,7 +9,7 @@ #ifndef StandardOptions_hpp #define StandardOptions_hpp -#include "../Reflection/Enum.h" +#include "../Reflection/Enum.hpp" namespace Configurable { diff --git a/Machines/Utility/MachineForTarget.hpp b/Machines/Utility/MachineForTarget.hpp index 5d481e0d4..066ef51b7 100644 --- a/Machines/Utility/MachineForTarget.hpp +++ b/Machines/Utility/MachineForTarget.hpp @@ -10,7 +10,7 @@ #define MachineForTarget_hpp #include "../../Analyser/Static/StaticAnalyser.hpp" -#include "../../Reflection/Struct.h" +#include "../../Reflection/Struct.hpp" #include "../DynamicMachine.hpp" diff --git a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj index 2e5b8d107..881f5f442 100644 --- a/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj +++ b/OSBindings/Mac/Clock Signal.xcodeproj/project.pbxproj @@ -1038,8 +1038,8 @@ 4B38F3471F2EC11D00D9235D /* AmstradCPC.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = AmstradCPC.hpp; path = AmstradCPC/AmstradCPC.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 = ""; }; - 4B3AF7D02413470E00873C0B /* Enum.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = Enum.h; sourceTree = ""; }; - 4B3AF7D12413472200873C0B /* Struct.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = Struct.h; sourceTree = ""; }; + 4B3AF7D02413470E00873C0B /* Enum.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = Enum.hpp; sourceTree = ""; }; + 4B3AF7D12413472200873C0B /* Struct.hpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.h; path = Struct.hpp; sourceTree = ""; }; 4B3BA0C21D318AEB005DD7A7 /* C1540Tests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = C1540Tests.swift; sourceTree = ""; }; 4B3BA0C51D318B44005DD7A7 /* C1540Bridge.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = C1540Bridge.h; sourceTree = ""; }; 4B3BA0C61D318B44005DD7A7 /* C1540Bridge.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = C1540Bridge.mm; sourceTree = ""; }; @@ -2197,8 +2197,8 @@ 4B3AF7CF2413470E00873C0B /* Reflection */ = { isa = PBXGroup; children = ( - 4B3AF7D02413470E00873C0B /* Enum.h */, - 4B3AF7D12413472200873C0B /* Struct.h */, + 4B3AF7D02413470E00873C0B /* Enum.hpp */, + 4B3AF7D12413472200873C0B /* Struct.hpp */, 4B47F6C4241C87A100ED06F7 /* Struct.cpp */, ); name = Reflection; diff --git a/OSBindings/SDL/main.cpp b/OSBindings/SDL/main.cpp index 07692ed41..451f3c8d9 100644 --- a/OSBindings/SDL/main.cpp +++ b/OSBindings/SDL/main.cpp @@ -34,8 +34,8 @@ #include "../../Outputs/OpenGL/ScanTarget.hpp" #include "../../Outputs/OpenGL/Screenshot.hpp" -#include "../../Reflection/Enum.h" -#include "../../Reflection/Struct.h" +#include "../../Reflection/Enum.hpp" +#include "../../Reflection/Struct.hpp" namespace { diff --git a/Reflection/Enum.h b/Reflection/Enum.hpp similarity index 98% rename from Reflection/Enum.h rename to Reflection/Enum.hpp index ede607246..09fd48719 100644 --- a/Reflection/Enum.h +++ b/Reflection/Enum.hpp @@ -1,13 +1,13 @@ // -// Enum.h +// Enum.hpp // Clock Signal // // Created by Thomas Harte on 17/02/2020. // Copyright © 2020 Thomas Harte. All rights reserved. // -#ifndef Enum_h -#define Enum_h +#ifndef Enum_hpp +#define Enum_hpp #include #include @@ -170,4 +170,4 @@ class Enum { } -#endif /* Enum_h */ +#endif /* Enum_hpp */ diff --git a/Reflection/Struct.cpp b/Reflection/Struct.cpp index dadad2fd8..19096e0ad 100644 --- a/Reflection/Struct.cpp +++ b/Reflection/Struct.cpp @@ -6,7 +6,7 @@ // Copyright © 2020 Thomas Harte. All rights reserved. // -#include "Struct.h" +#include "Struct.hpp" #include diff --git a/Reflection/Struct.h b/Reflection/Struct.hpp similarity index 98% rename from Reflection/Struct.h rename to Reflection/Struct.hpp index a1a2d99d4..e6423deae 100644 --- a/Reflection/Struct.h +++ b/Reflection/Struct.hpp @@ -1,13 +1,13 @@ // -// Struct.h +// Struct.hpp // Clock Signal // // Created by Thomas Harte on 06/03/2020. // Copyright © 2020 Thomas Harte. All rights reserved. // -#ifndef Struct_h -#define Struct_h +#ifndef Struct_hpp +#define Struct_hpp #include #include @@ -17,7 +17,7 @@ #include #include -#include "Enum.h" +#include "Enum.hpp" namespace Reflection { @@ -269,4 +269,4 @@ template class StructImpl: public Struct { } -#endif /* Struct_h */ +#endif /* Struct_hpp */