diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NESControllers.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NESControllers.cs index b56299a2d7..f9b6ad8fb5 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NESControllers.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NESControllers.cs @@ -9,19 +9,50 @@ using Newtonsoft.Json; namespace BizHawk.Emulation.Cores.Nintendo.NES { - // we don't handle some possible connections of the expansion port that were never used + /* + * This file covers all NES and Famicom controller related stuff. + * It supports (or could be easily made to support by adding a new class) every existing + * controller device I know about. It does not support some things that were theoretically + * possible with the electronic interface available, but never used. + */ #region interfaces and such + /// + /// stores information about the strobe lines controlled by $4016 + /// public struct StrobeInfo { - public int OUT0; - public int OUT1; - public int OUT2; - public int OUT0old; - public int OUT1old; - public int OUT2old; + /// + /// the current value of $4016.0; strobes regular controller ports + /// + public readonly int OUT0; + /// + /// the current value of $4016.1; strobes expansion port + /// + public readonly int OUT1; + /// + /// the current value of $4016.2; strobes expansion port + /// + public readonly int OUT2; + /// + /// the previous value or $4016.0 (for edge sensitive equipment) + /// + public readonly int OUT0old; + /// + /// the previous value or $4016.1 (for edge sensitive equipment) + /// + public readonly int OUT1old; + /// + /// the previous value or $4016.2 (for edge sensitive equipment) + /// + public readonly int OUT2old; + /// + /// + /// + /// the old latched $4016 byte + /// the new latched $4016 byte public StrobeInfo(byte oldvalue, byte newvalue) { OUT0old = oldvalue & 1; @@ -33,24 +64,58 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } } + /// + /// the main system deck, handling all $4016 writes and $4016/$4017 reads + /// public interface IControllerDeck { + /// + /// call whenever $4016 is written + /// + /// + /// void Strobe(StrobeInfo s, IController c); + /// + /// call whenever $4016 is read + /// + /// + /// bits 0-4 are valid byte ReadA(IController c); // D0:D4 + /// + /// call whenever $4017 is read + /// + /// + /// bits 0-4 are valid byte ReadB(IController c); // D0:D4 ControllerDefinition GetDefinition(); void SyncState(Serializer ser); } + /// + /// a peripheral that plugs into the famicom expansion port + /// public interface IFamicomExpansion { void Strobe(StrobeInfo s, IController c); - byte ReadA(IController c); // only uses D1 - byte ReadB(IController c); // only uses D1:D4 + /// + /// read data from $4016 + /// + /// + /// only bit 1 is valid + byte ReadA(IController c); + /// + /// read data from $4017 + /// + /// + /// bits 1-4 are valid + byte ReadB(IController c); ControllerDefinition GetDefinition(); void SyncState(Serializer ser); } + /// + /// a peripheral that plugs into either of the two NES controller ports + /// public interface INesPort { void Strobe(StrobeInfo s, IController c); // only uses OUT0 @@ -61,6 +126,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES #endregion + /// + /// a NES or AV famicom, with two attached devices + /// public class NesDeck : IControllerDeck { INesPort Left; @@ -144,6 +212,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } } + /// + /// a NES controller; also used internally to represent the two famicom controllers + /// public class ControllerNES : INesPort { bool resetting = false; @@ -215,6 +286,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } } + /// + /// vaus paddle, the NES (not famicom) version + /// public class ArkanoidNES : INesPort { int shiftidx = 0; @@ -375,6 +449,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public class Zapper : INesPort, IFamicomExpansion { + /// + /// returns true if light was detected at the ppu coordinates specified + /// public Func PPUCallback; static ControllerDefinition Definition = new ControllerDefinition @@ -389,6 +466,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES { } + // NES controller port interface public byte Read(IController c) { byte ret = 0; @@ -493,6 +571,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } } + /// + /// vaus controller that plugs into a famicom's expansion port + /// public class ArkanoidFam : IFamicomExpansion { int shiftidx = 0; @@ -778,6 +859,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES int x = (int)c.GetFloat("0Pen X"); int y = (int)c.GetFloat("0Pen Y"); // http://forums.nesdev.com/viewtopic.php?p=19454#19454 + // it almost feels like the hardware guys got the request for + // a tablet that returned x in [0, 255] and y in [0, 239] and then + // accidentally flipped the whole thing sideways x = (x + 8) * 240 / 256; y = (y - 14) * 256 / 240; x &= 255; @@ -876,6 +960,10 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES #region control definition adapters + // the idea here is that various connected peripherals have their controls all merged + // into one definition, including logic to unmerge the data back so each one can work + // with it without knowing what else is connected + public class ControlDefUnMerger { Dictionary Remaps;