diff --git a/BizHawk.Common/Util.cs b/BizHawk.Common/Util.cs index 1fef4d6827..40e82da820 100644 --- a/BizHawk.Common/Util.cs +++ b/BizHawk.Common/Util.cs @@ -16,6 +16,18 @@ namespace BizHawk.Common HexConvPtr = (char*)HexConvHandle.AddrOfPinnedObject().ToPointer(); } + public static bool FindBytes(byte[] array, byte[] pattern) + { + int fidx = 0; + int result = Array.FindIndex(array, 0, array.Length, (byte b) => + { + fidx = (b == pattern[fidx]) ? fidx + 1 : 0; + return (fidx == pattern.Length); + }); + + return (result >= pattern.Length - 1); + } + public static char* HexConvPtr { get; set; } public static string Hash_MD5(byte[] data, int offset, int len) diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index 9321d436c2..79d314cdfa 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -176,25 +176,35 @@ + + + + + + + + + + diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.Core.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.Core.cs index 724ab811e9..5b4cd76ba8 100644 --- a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.Core.cs +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.Core.cs @@ -142,30 +142,96 @@ namespace BizHawk.Emulation.Cores.Atari.Atari2600 //regenerate mapper here to make sure its state is entirely clean switch (game.GetOptionsDict()["m"]) { - case "4K": mapper = new m4K(); break; - case "2K": mapper = new m2K(); break; - case "CV": mapper = new mCV(); break; - case "F8": mapper = new mF8(); break; - case "F6": + case "2IN1": case "4IN1": - case "F6SC": mapper = new mF6(); break; + case "8IN1": + case "16IN1": + case "32IN1": + mapper = new Multicart(); + break; + case "AR": + mapper = new mAR(); + break; + case "4K": + mapper = new m4K(); + break; + case "2K": + mapper = new m2K(); + break; + case "CV": + mapper = new mCV(); + break; + case "DPC": + mapper = new mDPC(); + break; + case "DPC+": + mapper = new mDPCPlus(); + break; + case "F8": + mapper = new mF8(); + break; + case "F6": + mapper = new mF6(); + break; + case "F6SC": + mapper = new mF6SC(); + break; case "F4": - case "F4SC": mapper = new mF4(); break; - case "FE": mapper = new mFE(); break; - case "E0": mapper = new mE0(); break; - case "3F": mapper = new m3F(); break; - case "FA": mapper = new mFA(); break; - case "E7": mapper = new mE7(); break; - case "F0": mapper = new mF0(); break; - case "UA": mapper = new mUA(); break; + mapper = new mF4(); + break; + case "F4SC": + mapper = new mF4SC(); + break; + case "FE": + mapper = new mFE(); + break; + case "E0": + mapper = new mE0(); + break; + case "3F": + mapper = new m3F(); + break; + case "FA": + mapper = new mFA(); + break; + case "FA2": + mapper = new mFA2(); + break; + case "E7": + mapper = new mE7(); + break; + case "F0": + mapper = new mF0(); + break; + case "UA": + mapper = new mUA(); + break; + //Homebrew mappers - case "3E": mapper = new m3E(); break; - case "0840": mapper = new m0840(); break; - case "MC": mapper = new mMC(); break; - case "EF": mapper = new mEF(); break; - case "X07": mapper = new mX07(); break; - case "4A50": mapper = new m4A50(); break; - case "DPC": mapper = new mDPC(); break; + case "3E": + mapper = new m3E(); + break; + case "0840": + mapper = new m0840(); + break; + case "MC": + mapper = new mMC(); + break; + case "EF": + mapper = new mEF(); + break; + case "EFSC": + mapper = new mEFSC(); + break; + case "X07": + mapper = new mX07(); + break; + case "4A50": + mapper = new m4A50(); + break; + case "SB": + mapper = new mSB(); + break; default: throw new InvalidOperationException("mapper not supported: " + game.GetOptionsDict()["m"]); } diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.RomHeuristics.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.RomHeuristics.cs new file mode 100644 index 0000000000..844868a683 --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.RomHeuristics.cs @@ -0,0 +1,460 @@ +using System.Collections.Generic; +using System.Linq; + +using BizHawk.Common; + +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + public partial class Atari2600 + { + // Heuristics logic based on Stella's logic + private static string DetectMapper(byte[] rom) + { + if (rom.Length % 8448 == 0 || rom.Length == 6114) // Only AR could be these odd numbers + { + return "AR"; + } + + if (rom.Length < 2048) // Less than 2k, then no bank switching needed + { + return "2K"; + } + + if (rom.Length == 2048 || // If 2k or the same 2k twice...Why would a rom be that way? Overdump? + (rom.Length == 4096 + && rom.Take(2048).SequenceEqual(rom.Skip(2048).Take(2048)))) + { + return IsProablyCV(rom) ? "CV" : "2K"; + } + + if (rom.Length == 4096) + { + return IsProablyCV(rom) ? "CV" : "4K"; + } + + if (rom.Length == 8192) // Several 8K Options + { + if (IsProbablySC(rom)) + { + return "F8SC"; + } + + if (rom.Take(4096).SequenceEqual(rom.Skip(4096).Take(4096))) + { + return "4K"; // Again if it is simply the same 4k twice. Got this scenario from Stella logic. Will assume a good reason for it + } + + if (IsProbablyE0(rom)) + { + return "E0"; + } + + if (IsProbably3E(rom)) + { + return "3E"; + } + + if (IsProbably3F(rom)) + { + return "3F"; + } + + if (IsProbablyUA(rom)) + { + return "UA"; + } + + if (IsProbablyFE(rom)) + { + return "FE"; + } + + if (IsProbably0840(rom)) + { + return "0840"; + } + + return "F8"; + } + + if (rom.Length >= 10240 && rom.Length <= 10496) // ~10K - Pitfall2 + { + return "DPC"; + } + + if (rom.Length == 12 * 1024) // 12K + { + return "FA"; + } + + if (rom.Length == 16 * 1024) // 16K + { + if (IsProbablySC(rom)) + { + return "F6SC"; + } + + if (IsProbablyE7(rom)) + { + return "E7"; + } + + if (IsProbably3E(rom)) + { + return "3E"; + } + + if (IsProbably3F(rom)) + { + return "3F"; + } + + return "F6"; + } + + if (rom.Length == 24 * 1024 || rom.Length == 28 * 1024) // 24K & 28K + { + return "FA2"; + } + + if (rom.Length == 29 * 1024) // 29K + { + return "DPC+"; + } + + if (rom.Length == 32 * 1024) // 32K + { + if (IsProbablySC(rom)) + { + return "F4SC"; + } + + if (IsProbably3E(rom)) + { + return "3E"; + } + + if (IsProbably3F(rom)) + { + return "3F"; + } + + if (IsProbablyDpcPlus(rom)) + { + return "DPC+"; + } + + return "F4"; + } + + if (rom.Length == 64 * 1064) // 64K + { + if (IsProbably3E(rom)) + { + return "3E"; + } + + if (IsProbably3F(rom)) + { + return "3F"; + } + + if (IsProbably4A50(rom)) + { + return "4A50"; + } + + if (IsProbablyEF(rom)) + { + return IsProbablySC(rom) ? "EFSC" : "EF"; + } + + if (IsProbablyX07(rom)) + { + return "X07"; + } + + return "F0"; + } + + if (rom.Length == 128 * 1024) // 128K + { + if (IsProbably3E(rom)) + { + return "3E"; + } + + if (IsProbably3F(rom)) + { + return "3F"; + } + + if (IsProbably4A50(rom)) + { + return "4A50"; + } + + if (IsProbablySB(rom)) + { + return "SB"; + } + + return "MC"; + } + + if (rom.Length == 256 * 1024) // 256K + { + if (IsProbably3E(rom)) + { + return "3E"; + } + + if (IsProbably3F(rom)) + { + return "3F"; + } + + return "SB"; + } + + // What else could it be? Try 3E or 3F + if (IsProbably3E(rom)) + { + return "3E"; + } + + if (IsProbably3F(rom)) + { + return "3F"; + } + + return "UNKNOWN"; + } + + private static bool IsProbablySC(IList rom) + { + // We assume a Superchip cart contains the same bytes for its entire + // RAM area; obviously this test will fail if it doesn't + // The RAM area will be the first 256 bytes of each 4K bank + var numBanks = rom.Count / 4096; + for (var i = 0; i < numBanks; i++) + { + var first = rom[i * 4096]; + for (var j = 0; j < 256; j++) + { + if (rom[(i * 4096) + j] != first) + { + return false; + } + } + } + + return false; + } + + private static bool IsProbably3E(byte[] rom) + { + // 3E cart bankswitching is triggered by storing the bank number + // in address 3E using 'STA $3E', commonly followed by an + // immediate mode LDA + return Util.FindBytes(rom, new byte[] { 0x85, 0x3E, 0xA9, 0x00 }); // STA $3E; LDA #$00 + } + + private static bool IsProbably3F(byte[] rom) + { + // 3F cart bankswitching is triggered by storing the bank number + // in address 3F using 'STA $3F' + // We expect it will be present at least 2 times, since there are at least two banks + return ContainsAll(rom, new List + { + new byte[] { 0x85, 0x3F }, + new byte[] { 0x85, 0x3F } + }); + } + + private static bool IsProbably4A50(IList rom) + { + // 4A50 carts store address $4A50 at the NMI vector, which + // in this scheme is always in the last page of ROM at + // $1FFA - $1FFB (at least this is true in rev 1 of the format) + if (rom[rom.Count - 6] == 0x50 && rom[rom.Count - 5] == 0x4A) + { + return true; + } + + // Program starts at $1Fxx with NOP $6Exx or NOP $6Fxx? + if (((rom[0xFFFD] & 0x1F) == 0x1F) && + (rom[rom[0xFFFD] * 256 + rom[0xFFFC]] == 0x0C) && + ((rom[rom[0xFFFD] * 256 + rom[0xFFFC] + 2] & 0xFE) == 0x6E)) + { + return true; + } + + return false; + } + + private static bool IsProbablyUA(byte[] rom) + { + // UA cart bankswitching switches to bank 1 by accessing address 0x240 + // using 'STA $240' or 'LDA $240' + return ContainsAny(rom, new List + { + new byte[] { 0x8D, 0x40, 0x02 }, // STA $240 + new byte[] { 0xAD, 0x40, 0x02 }, // LDA $240 + new byte[] { 0xBD, 0x1F, 0x02 } // LDA $21F,X + }); + } + + private static bool IsProbablyE0(byte[] rom) + { + // E0 cart bankswitching is triggered by accessing addresses + // $FE0 to $FF9 using absolute non-indexed addressing + // To eliminate false positives (and speed up processing), we + // search for only certain known signatures + // Thanks to "stella@casperkitty.com" for this advice + // These signatures are attributed to the MESS project + return ContainsAny(rom, new List + { + new byte[] { 0x8D, 0xE0, 0x1F }, // STA $1FE0 + new byte[] { 0x8D, 0xE0, 0x5F }, // STA $5FE0 + new byte[] { 0x8D, 0xE9, 0xFF }, // STA $FFE9 + new byte[] { 0x0C, 0xE0, 0x1F }, // NOP $1FE0 + new byte[] { 0xAD, 0xE0, 0x1F }, // LDA $1FE0 + new byte[] { 0xAD, 0xE9, 0xFF }, // LDA $FFE9 + new byte[] { 0xAD, 0xED, 0xFF }, // LDA $FFED + new byte[] { 0xAD, 0xF3, 0xBF } // LDA $BFF3 + }); + } + + private static bool IsProablyCV(byte[] rom) + { + // According to Stella: CV RAM access occurs at addresses $f3ff and $f400 + // These signatures are attributed to the MESS project + return ContainsAny(rom, new List + { + new byte[] { 0x9D, 0xFF, 0xF3 }, + new byte[] { 0x99, 0x00, 0xF4 } + }); + } + + private static bool IsProbablyFE(byte[] rom) + { + // FE bankswitching is very weird, but always seems to include a + // 'JSR $xxxx' + // These signatures are attributed to the MESS project + return ContainsAny(rom, new List + { + new byte[] { 0x20, 0x00, 0xD0, 0xC6, 0xC5 }, // JSR $D000; DEC $C5 + new byte[] { 0x20, 0xC3, 0xF8, 0xA5, 0x82 }, // JSR $F8C3; LDA $82 + new byte[] { 0xD0, 0xFB, 0x20, 0x73, 0xFE }, // BNE $FB; JSR $FE73 + new byte[] { 0x20, 0x00, 0xF0, 0x84, 0xD6 } // JSR $F000; STY $D6 + }); + } + + private static bool IsProbably0840(byte[] rom) + { + // 0840 cart bankswitching is triggered by accessing addresses 0x0800 + // or 0x0840 + if (ContainsAny(rom, new List + { + new byte[] { 0xAD, 0x00, 0x08 }, // LDA $0800 + new byte[] { 0xAD, 0x40, 0x08 } // LDA $0840 + })) + { + return true; + } + + return ContainsAny(rom, new List + { + new byte[] { 0x0C, 0x00, 0x08, 0x4C }, // NOP $0800; JMP ... + new byte[] { 0x0C, 0xFF, 0x0F, 0x4C } // NOP $0FFF; JMP ... + }); + } + + private static bool IsProbablyE7(byte[] rom) + { + // E7 cart bankswitching is triggered by accessing addresses + // $FE0 to $FE6 using absolute non-indexed addressing + // To eliminate false positives (and speed up processing), we + // search for only certain known signatures + // Thanks to "stella@casperkitty.com" for this advice + // These signatures are attributed to the MESS project + return ContainsAny(rom, new List + { + new byte[] { 0xAD, 0xE2, 0xFF }, // LDA $FFE2 + new byte[] { 0xAD, 0xE5, 0xFF }, // LDA $FFE5 + new byte[] { 0xAD, 0xE5, 0x1F }, // LDA $1FE5 + new byte[] { 0xAD, 0xE7, 0x1F }, // LDA $1FE7 + new byte[] { 0x0C, 0xE7, 0x1F }, // NOP $1FE7 + new byte[] { 0x8D, 0xE7, 0xFF }, // STA $FFE7 + new byte[] { 0x8D, 0xE7, 0x1F } // STA $1FE7 + }); + } + + private static bool IsProbablyDpcPlus(byte[] rom) + { + // E0 cart bankswitching is triggered by accessing addresses + // $FE0 to $FF9 using absolute non-indexed addressing + // To eliminate false positives (and speed up processing), we + // search for only certain known signatures + // Thanks to "stella@casperkitty.com" for this advice + // These signatures are attributed to the MESS project + return ContainsAny(rom, new List + { + new byte[] { 0x8D, 0xE0, 0x1F }, // STA $1FE0 + new byte[] { 0x8D, 0xE0, 0x5F }, // STA $5FE0 + new byte[] { 0x8D, 0xE9, 0xFF }, // STA $FFE9 + new byte[] { 0x0C, 0xE0, 0x1F }, // NOP $1FE0 + new byte[] { 0xAD, 0xE0, 0x1F }, // LDA $1FE0 + new byte[] { 0xAD, 0xE9, 0xFF }, // LDA $FFE9 + new byte[] { 0xAD, 0xED, 0xFF }, // LDA $FFED + new byte[] { 0xAD, 0xF3, 0xBF } // LDA $BFF3 + }); + } + + private static bool IsProbablyEF(byte[] rom) + { + // EF cart bankswitching switches banks by accessing addresses 0xFE0 + // to 0xFEF, usually with either a NOP or LDA + // It's likely that the code will switch to bank 0, so that's what is tested + return ContainsAny(rom, new List + { + new byte[] { 0x0C, 0xE0, 0xFF }, // NOP $FFE0 + new byte[] { 0xAD, 0xE0, 0xFF }, // LDA $FFE0 + new byte[] { 0x0C, 0xE0, 0x1F }, // NOP $1FE0 + new byte[] { 0xAD, 0xE0, 0x1F } // LDA $1FE0 + }); + } + + private static bool IsProbablyX07(byte[] rom) + { + // X07 bankswitching switches to bank 0, 1, 2, etc by accessing address 0x08xd + return ContainsAny(rom, new List + { + new byte[] { 0xAD, 0x0D, 0x08 }, // LDA $080D + new byte[] { 0xAD, 0x1D, 0x08 }, // LDA $081D + new byte[] { 0xAD, 0x2D, 0x08 } // LDA $082D + }); + } + + private static bool IsProbablySB(byte[] rom) + { + // SB cart bankswitching switches banks by accessing address 0x0800 + return ContainsAny(rom, new List + { + new byte[] { 0xBD, 0x00, 0x08 }, // LDA $0800,x + new byte[] { 0xAD, 0x00, 0x08 } // LDA $0800 + }); + } + + private static bool ContainsAny(byte[] rom, IEnumerable signatures) + { + return signatures.Any(signature => Util.FindBytes(rom, signature)); + } + + private static bool ContainsAll(byte[] rom, IEnumerable signatures) + { + return signatures.All(signature => Util.FindBytes(rom, signature)); + } + } +} diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs index 444fc820a8..73ef2dd2e8 100644 --- a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs @@ -44,33 +44,13 @@ namespace BizHawk.Emulation.Cores.Atari.Atari2600 if (!game.GetOptionsDict().ContainsKey("m")) { - DetectMapper(); + game.AddOption("m", DetectMapper(rom)); } Console.WriteLine("Game uses mapper " + game.GetOptionsDict()["m"]); HardReset(); } - void DetectMapper() - { - string m = "UNKNOWN"; - switch (rom.Length) - { - case 2048: m = "2K"; break; - case 4096: m = "4K"; break; - case 8192: m = "F8"; break; - case 16384: m = "F6"; break; - case 12288: m = "FA"; break; - case 32768: m = "F4"; break; - case 65536: m = "EF"; break; - case 131072: m = "MC"; break; - case 262144: m = "3F"; break; - case 524288: m = "3F"; break; - } - game.AddOption("m", m); - } - - public List> GetCpuFlagsAndRegisters() { return new List> diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mAR.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mAR.cs new file mode 100644 index 0000000000..ce98845d24 --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mAR.cs @@ -0,0 +1,68 @@ +using System; +using BizHawk.Common; +/** + This is the cartridge class for Arcadia (aka Starpath) Supercharger + games. Christopher Salomon provided most of the technical details + used in creating this class. A good description of the Supercharger + is provided in the Cuttle Cart's manual. + + The Supercharger has four 2K banks. There are three banks of RAM + and one bank of ROM. All 6K of the RAM can be read and written. +*/ +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + internal class mAR : MapperBase + { + private int _bank4k; + private ByteBuffer _ram = new ByteBuffer(6144); + + public mAR() + { + throw new NotImplementedException(); + } + + private byte ReadMem(ushort addr, bool peek) + { + if (!peek) + { + Address(addr); + } + + if (addr < 0x1000) + { + return base.ReadMemory(addr); + } + + return core.rom[(_bank4k << 12) + (addr & 0xFFF)]; + } + + public override byte ReadMemory(ushort addr) + { + return ReadMem(addr, false); + } + + public override byte PeekMemory(ushort addr) + { + return ReadMem(addr, true); + } + + public override void SyncState(Serializer ser) + { + base.SyncState(ser); + ser.Sync("bank4k", ref _bank4k); + ser.Sync("ram", ref _ram); + } + + private void Address(ushort addr) + { + if (addr == 0x1FF8) + { + _bank4k = 0; + } + else if (addr == 0x1FF9) + { + _bank4k = 1; + } + } + } +} diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mDPCPlus.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mDPCPlus.cs new file mode 100644 index 0000000000..98ca133217 --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mDPCPlus.cs @@ -0,0 +1,17 @@ +using System; + +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + /** + Cartridge class used for DPC+. There are six 4K program banks, a 4K + display bank, 1K frequency table and the DPC chip. For complete details on + the DPC chip see David P. Crane's United States Patent Number 4,644,495. + */ + internal class mDPCPlus : MapperBase + { + public mDPCPlus() + { + throw new NotImplementedException(); + } + } +} diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mEFSC.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mEFSC.cs new file mode 100644 index 0000000000..1d4cfbbfb6 --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mEFSC.cs @@ -0,0 +1,17 @@ +using System; + +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + /** + Cartridge class used for Homestar Runner by Paul Slocum. + There are 16 4K banks (total of 64K ROM) with 128 bytes of RAM. + Accessing $1FE0 - $1FEF switches to each bank. + */ + internal class mEFSC : MapperBase + { + public mEFSC() + { + throw new NotImplementedException(); + } + } +} diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF4SC.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF4SC.cs new file mode 100644 index 0000000000..837c0bfd41 --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF4SC.cs @@ -0,0 +1,16 @@ +using System; + +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + /** + Cartridge class used for Atari's 16K bankswitched games with + 128 bytes of RAM. There are four 4K banks. + */ + internal class mF4SC : MapperBase + { + public mF4SC() + { + throw new NotImplementedException(); + } + } +} \ No newline at end of file diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF6SC.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF6SC.cs new file mode 100644 index 0000000000..de61e9a55f --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF6SC.cs @@ -0,0 +1,16 @@ +using System; + +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + /** + Cartridge class used for Atari's 32K bankswitched games with + 128 bytes of RAM. There are eight 4K banks. + */ + internal class mF6SC : MapperBase + { + public mF6SC() + { + throw new NotImplementedException(); + } + } +} \ No newline at end of file diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF8SC.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF8SC.cs new file mode 100644 index 0000000000..fc39b315af --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mF8SC.cs @@ -0,0 +1,16 @@ +using System; + +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + /** + Cartridge class used for Atari's 8K bankswitched games with + 128 bytes of RAM. There are two 4K banks. + */ + internal class mF8SC : MapperBase + { + public mF8SC() + { + throw new NotImplementedException(); + } + } +} diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mFA2.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mFA2.cs new file mode 100644 index 0000000000..e9fd07798a --- /dev/null +++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Mappers/mFA2.cs @@ -0,0 +1,18 @@ +using System; + +namespace BizHawk.Emulation.Cores.Atari.Atari2600 +{ + /** + This is an extended version of the CBS RAM Plus bankswitching scheme + supported by the Harmony cartridge. + + There are six (or seven) 4K banks and 256 bytes of RAM. + */ + internal class mFA2 : MapperBase + { + public mFA2() + { + throw new NotImplementedException(); + } + } +}