2017-11-05 12:49:28 -05:00
//
// MachineForTarget.cpp
// Clock Signal
//
// Created by Thomas Harte on 04/11/2017.
2018-05-13 15:19:52 -04:00
// Copyright 2017 Thomas Harte. All rights reserved.
2017-11-05 12:49:28 -05:00
//
# include "MachineForTarget.hpp"
2020-03-18 23:11:25 -04:00
# include <algorithm>
2020-03-12 20:56:02 -04:00
// Sources for runtime options.
2017-11-05 12:49:28 -05:00
# include "../AmstradCPC/AmstradCPC.hpp"
2019-05-03 18:14:10 -04:00
# include "../Apple/AppleII/AppleII.hpp"
2019-05-03 22:16:07 -04:00
# include "../Apple/Macintosh/Macintosh.hpp"
2019-11-09 15:31:41 -05:00
# include "../Atari/2600/Atari2600.hpp"
# include "../Atari/ST/AtariST.hpp"
2018-02-23 22:47:15 -05:00
# include "../ColecoVision/ColecoVision.hpp"
2017-11-05 12:49:28 -05:00
# include "../Commodore/Vic-20/Vic20.hpp"
# include "../Electron/Electron.hpp"
2018-09-21 22:13:07 -04:00
# include "../MasterSystem/MasterSystem.hpp"
2017-11-25 13:18:24 -05:00
# include "../MSX/MSX.hpp"
2017-11-05 12:49:28 -05:00
# include "../Oric/Oric.hpp"
# include "../ZX8081/ZX8081.hpp"
2020-03-12 20:56:02 -04:00
// Sources for construction options.
# include "../../Analyser/Static/Acorn/Target.hpp"
# include "../../Analyser/Static/AmstradCPC/Target.hpp"
# include "../../Analyser/Static/AppleII/Target.hpp"
2020-03-15 12:54:55 -04:00
# include "../../Analyser/Static/Atari2600/Target.hpp"
2020-03-12 20:56:02 -04:00
# 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"
2020-03-15 12:54:55 -04:00
# include "../../Analyser/Static/Sega/Target.hpp"
2020-03-12 20:56:02 -04:00
# include "../../Analyser/Static/ZX8081/Target.hpp"
2018-01-28 22:22:21 -05:00
# include "../../Analyser/Dynamic/MultiMachine/MultiMachine.hpp"
2017-11-24 21:36:22 -05:00
# include "TypedDynamicMachine.hpp"
2017-11-05 12:49:28 -05:00
2020-03-15 12:54:55 -04:00
Machine : : DynamicMachine * Machine : : MachineForTarget ( const Analyser : : Static : : Target * target , const ROMMachine : : ROMFetcher & rom_fetcher , Machine : : Error & error ) {
2018-01-28 22:22:21 -05:00
error = Machine : : Error : : None ;
2018-01-25 18:28:19 -05:00
2018-07-10 20:00:46 -04:00
Machine : : DynamicMachine * machine = nullptr ;
try {
# define BindD(name, m) case Analyser::Machine::m: machine = new Machine::TypedDynamicMachine<name::Machine>(name::Machine::m(target, rom_fetcher)); break;
# define Bind(m) BindD(m, m)
switch ( target - > machine ) {
Bind ( AmstradCPC )
2019-05-03 18:14:10 -04:00
BindD ( Apple : : II , AppleII )
2019-05-03 22:16:07 -04:00
BindD ( Apple : : Macintosh , Macintosh )
2018-07-10 20:00:46 -04:00
Bind ( Atari2600 )
2019-10-03 20:17:26 -04:00
BindD ( Atari : : ST , AtariST )
2018-07-10 20:00:46 -04:00
BindD ( Coleco : : Vision , ColecoVision )
2019-06-02 13:39:25 -04:00
BindD ( Commodore : : Vic20 , Vic20 )
2018-07-10 20:00:46 -04:00
Bind ( Electron )
Bind ( MSX )
Bind ( Oric )
2019-06-02 13:39:25 -04:00
BindD ( Sega : : MasterSystem , MasterSystem )
2018-07-10 20:00:46 -04:00
Bind ( ZX8081 )
2018-01-25 18:28:19 -05:00
2018-07-10 20:00:46 -04:00
default :
error = Machine : : Error : : UnknownMachine ;
2018-01-25 18:28:19 -05:00
return nullptr ;
}
2018-07-10 20:00:46 -04:00
# undef Bind
} catch ( ROMMachine : : Error construction_error ) {
switch ( construction_error ) {
case ROMMachine : : Error : : MissingROMs :
error = Machine : : Error : : MissingROM ;
break ;
default :
error = Machine : : Error : : UnknownError ;
break ;
}
2018-01-25 18:28:19 -05:00
}
return machine ;
2017-11-05 20:12:01 -05:00
}
2020-03-15 12:54:55 -04:00
Machine : : DynamicMachine * Machine : : MachineForTargets ( const Analyser : : Static : : TargetList & targets , const ROMMachine : : ROMFetcher & rom_fetcher , Error & error ) {
2018-01-28 22:22:21 -05:00
// Zero targets implies no machine.
if ( targets . empty ( ) ) {
error = Error : : NoTargets ;
return nullptr ;
}
// If there's more than one target, get all the machines and combine them into a multimachine.
if ( targets . size ( ) > 1 ) {
std : : vector < std : : unique_ptr < Machine : : DynamicMachine > > machines ;
for ( const auto & target : targets ) {
2018-03-09 15:36:11 -05:00
machines . emplace_back ( MachineForTarget ( target . get ( ) , rom_fetcher , error ) ) ;
2018-02-08 20:33:57 -05:00
// Exit early if any errors have occurred.
if ( error ! = Error : : None ) {
return nullptr ;
}
2018-01-28 22:22:21 -05:00
}
2018-03-06 19:08:02 -05:00
// If a multimachine would just instantly collapse the list to a single machine, do
// so without the ongoing baggage of a multimachine.
if ( Analyser : : Dynamic : : MultiMachine : : would_collapse ( machines ) ) {
return machines . front ( ) . release ( ) ;
} else {
return new Analyser : : Dynamic : : MultiMachine ( std : : move ( machines ) ) ;
}
2018-01-28 22:22:21 -05:00
}
// There's definitely exactly one target.
2018-03-09 15:36:11 -05:00
return MachineForTarget ( targets . front ( ) . get ( ) , rom_fetcher , error ) ;
2018-01-28 22:22:21 -05:00
}
2018-01-24 21:48:44 -05:00
std : : string Machine : : ShortNameForTargetMachine ( const Analyser : : Machine machine ) {
2017-11-20 21:55:32 -05:00
switch ( machine ) {
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : AmstradCPC : return " AmstradCPC " ;
2018-04-14 19:46:15 -04:00
case Analyser : : Machine : : AppleII : return " AppleII " ;
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : Atari2600 : return " Atari2600 " ;
2019-10-03 20:17:26 -04:00
case Analyser : : Machine : : AtariST : return " AtariST " ;
2018-02-23 22:47:15 -05:00
case Analyser : : Machine : : ColecoVision : return " ColecoVision " ;
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : Electron : return " Electron " ;
2019-05-03 22:16:07 -04:00
case Analyser : : Machine : : Macintosh : return " Macintosh " ;
2019-09-30 21:10:30 -04:00
case Analyser : : Machine : : MasterSystem : return " MasterSystem " ;
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : MSX : return " MSX " ;
case Analyser : : Machine : : Oric : return " Oric " ;
case Analyser : : Machine : : Vic20 : return " Vic20 " ;
case Analyser : : Machine : : ZX8081 : return " ZX8081 " ;
2017-11-12 16:41:09 -05:00
default : return " " ;
2017-11-05 12:49:28 -05:00
}
}
2017-11-20 21:55:32 -05:00
2018-01-24 21:48:44 -05:00
std : : string Machine : : LongNameForTargetMachine ( Analyser : : Machine machine ) {
2017-11-20 21:55:32 -05:00
switch ( machine ) {
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : AmstradCPC : return " Amstrad CPC " ;
2018-04-14 19:46:15 -04:00
case Analyser : : Machine : : AppleII : return " Apple II " ;
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : Atari2600 : return " Atari 2600 " ;
2019-10-03 20:17:26 -04:00
case Analyser : : Machine : : AtariST : return " Atari ST " ;
2018-02-23 22:47:15 -05:00
case Analyser : : Machine : : ColecoVision : return " ColecoVision " ;
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : Electron : return " Acorn Electron " ;
2019-05-03 22:16:07 -04:00
case Analyser : : Machine : : Macintosh : return " Apple Macintosh " ;
2019-09-30 21:10:30 -04:00
case Analyser : : Machine : : MasterSystem : return " Sega Master System " ;
2018-01-24 21:48:44 -05:00
case Analyser : : Machine : : MSX : return " MSX " ;
case Analyser : : Machine : : Oric : return " Oric " ;
case Analyser : : Machine : : Vic20 : return " Vic 20 " ;
case Analyser : : Machine : : ZX8081 : return " ZX80/81 " ;
2017-11-20 21:55:32 -05:00
default : return " " ;
}
}
2020-03-18 23:11:25 -04:00
std : : vector < std : : string > Machine : : AllMachines ( Type type , bool long_names ) {
2020-03-15 00:15:19 -04:00
std : : vector < std : : string > result ;
# define AddName(x) result.push_back(long_names ? LongNameForTargetMachine(Analyser::Machine::x) : ShortNameForTargetMachine(Analyser::Machine::x))
2020-03-18 23:11:25 -04:00
2020-03-19 19:41:50 -04:00
if ( type = = Type : : Any | | type = = Type : : RequiresMedia ) {
AddName ( Atari2600 ) ;
AddName ( ColecoVision ) ;
AddName ( MasterSystem ) ;
}
2020-03-18 23:11:25 -04:00
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 ) ;
}
2020-03-15 00:15:19 -04:00
# undef AddName
return result ;
}
2020-03-15 23:48:53 -04:00
std : : map < std : : string , std : : unique_ptr < Reflection : : Struct > > Machine : : AllOptionsByMachineName ( ) {
std : : map < std : : string , std : : unique_ptr < Reflection : : Struct > > options ;
2020-03-16 23:25:05 -04:00
# define Emplace(machine, class) \
options . emplace ( std : : make_pair ( LongNameForTargetMachine ( Analyser : : Machine : : machine ) , std : : make_unique < class : : Options > ( Configurable : : OptionsType : : UserFriendly ) ) ) ;
Emplace ( AmstradCPC , AmstradCPC : : Machine ) ;
2020-03-17 21:53:26 -04:00
Emplace ( AppleII , Apple : : II : : Machine ) ;
2020-03-17 23:52:55 -04:00
Emplace ( AtariST , Atari : : ST : : Machine ) ;
2020-03-18 00:06:52 -04:00
Emplace ( ColecoVision , Coleco : : Vision : : Machine ) ;
2020-03-17 22:06:20 -04:00
Emplace ( Electron , Electron : : Machine ) ;
2020-03-18 21:50:02 -04:00
Emplace ( Macintosh , Apple : : Macintosh : : Machine ) ;
2020-03-18 18:26:22 -04:00
Emplace ( MasterSystem , Sega : : MasterSystem : : Machine ) ;
Emplace ( MSX , MSX : : Machine ) ;
2020-03-18 18:31:31 -04:00
Emplace ( Oric , Oric : : Machine ) ;
2020-03-18 20:23:55 -04:00
Emplace ( Vic20 , Commodore : : Vic20 : : Machine ) ;
2020-03-16 23:25:05 -04:00
Emplace ( ZX8081 , ZX8081 : : Machine ) ;
# undef Emplace
2017-11-20 21:55:32 -05:00
return options ;
}
2020-03-12 20:56:02 -04:00
2020-03-15 12:54:55 -04:00
std : : map < std : : string , std : : unique_ptr < Analyser : : Static : : Target > > Machine : : TargetsByMachineName ( bool meaningful_without_media_only ) {
std : : map < std : : string , std : : unique_ptr < Analyser : : Static : : Target > > options ;
2020-03-12 20:56:02 -04:00
2020-03-15 00:15:19 -04:00
# define AddMapped(Name, TargetNamespace) \
options . emplace ( std : : make_pair ( LongNameForTargetMachine ( Analyser : : Machine : : Name ) , new Analyser : : Static : : TargetNamespace : : Target ) ) ;
# define Add(Name) AddMapped(Name, Name)
2020-03-12 20:56:02 -04:00
Add ( AmstradCPC ) ;
Add ( AppleII ) ;
2020-03-15 00:15:19 -04:00
Add ( AtariST ) ;
AddMapped ( Electron , Acorn ) ;
Add ( Macintosh ) ;
2020-03-12 20:56:02 -04:00
Add ( MSX ) ;
Add ( Oric ) ;
2020-03-15 00:15:19 -04:00
AddMapped ( Vic20 , Commodore ) ;
2020-03-12 20:56:02 -04:00
Add ( ZX8081 ) ;
2020-03-15 12:54:55 -04:00
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 ) ;
}
2020-03-15 00:15:19 -04:00
# undef Add
# undef AddTwo
2020-03-12 20:56:02 -04:00
return options ;
}