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();
+ }
+ }
+}