From 9d3f1ed6a820550ac1b8d5e90741d9ba70e2964d Mon Sep 17 00:00:00 2001 From: ls9512 <598914653@qq.com> Date: Mon, 29 Mar 2021 17:47:12 +0800 Subject: [PATCH] Init repo --- .gitignore | 35 + Aya.UNes.asmdef | 13 + Aya.UNes.asmdef.meta | 7 + README.md | 3 + Runtime.meta | 8 + Runtime/Addressable.cs | 62 ++ Runtime/Addressable.cs.meta | 11 + Runtime/CPU.Core.cs | 81 ++ Runtime/CPU.Core.cs.meta | 11 + Runtime/CPU.IORegisters.cs | 44 + Runtime/CPU.IORegisters.cs.meta | 11 + Runtime/CPU.Instructions.cs | 463 ++++++++++ Runtime/CPU.Instructions.cs.meta | 11 + Runtime/CPU.Memory.cs | 165 ++++ Runtime/CPU.Memory.cs.meta | 11 + Runtime/CPU.Registers.cs | 89 ++ Runtime/CPU.Registers.cs.meta | 11 + Runtime/CPU.cs | 65 ++ Runtime/CPU.cs.meta | 11 + Runtime/Cartridge.cs | 66 ++ Runtime/Cartridge.cs.meta | 11 + Runtime/Controller.meta | 8 + Runtime/Controller/IController.cs | 15 + Runtime/Controller/IController.cs.meta | 11 + Runtime/Controller/NesController.cs | 71 ++ Runtime/Controller/NesController.cs.meta | 11 + Runtime/Emulator.cs | 66 ++ Runtime/Emulator.cs.meta | 11 + Runtime/Mapper.meta | 8 + Runtime/Mapper/AxROM.cs | 26 + Runtime/Mapper/AxROM.cs.meta | 11 + Runtime/Mapper/BaseMapper.cs | 71 ++ Runtime/Mapper/BaseMapper.cs.meta | 11 + Runtime/Mapper/CNROM.cs | 32 + Runtime/Mapper/CNROM.cs.meta | 11 + Runtime/Mapper/Camerica.cs | 27 + Runtime/Mapper/Camerica.cs.meta | 11 + Runtime/Mapper/ColorDreams.cs | 29 + Runtime/Mapper/ColorDreams.cs.meta | 11 + Runtime/Mapper/DxROM.cs | 39 + Runtime/Mapper/DxROM.cs.meta | 11 + Runtime/Mapper/GxROM.cs | 29 + Runtime/Mapper/GxROM.cs.meta | 11 + Runtime/Mapper/Jaleco.cs | 29 + Runtime/Mapper/Jaleco.cs.meta | 11 + Runtime/Mapper/MMC1.cs | 170 ++++ Runtime/Mapper/MMC1.cs.meta | 11 + Runtime/Mapper/MMC2.cs | 33 + Runtime/Mapper/MMC2.cs.meta | 11 + Runtime/Mapper/MMC3.cs | 182 ++++ Runtime/Mapper/MMC3.cs.meta | 11 + Runtime/Mapper/MMC4.cs | 73 ++ Runtime/Mapper/MMC4.cs.meta | 11 + Runtime/Mapper/Mapper094.cs | 17 + Runtime/Mapper/Mapper094.cs.meta | 11 + Runtime/Mapper/Mapper155.cs | 11 + Runtime/Mapper/Mapper155.cs.meta | 11 + Runtime/Mapper/Mapper180.cs | 21 + Runtime/Mapper/Mapper180.cs.meta | 11 + Runtime/Mapper/NROM.cs | 23 + Runtime/Mapper/NROM.cs.meta | 11 + Runtime/Mapper/Nina003006.cs | 32 + Runtime/Mapper/Nina003006.cs.meta | 11 + Runtime/Mapper/UxROM.cs | 22 + Runtime/Mapper/UxROM.cs.meta | 11 + Runtime/PPU.Core.cs | 380 ++++++++ Runtime/PPU.Core.cs.meta | 11 + Runtime/PPU.Memory.cs | 130 +++ Runtime/PPU.Memory.cs.meta | 11 + Runtime/PPU.Registers.cs | 261 ++++++ Runtime/PPU.Registers.cs.meta | 11 + Runtime/PPU.cs | 10 + Runtime/PPU.cs.meta | 11 + Runtime/Renderer.meta | 8 + Runtime/Renderer/IRenderer.cs | 13 + Runtime/Renderer/IRenderer.cs.meta | 11 + Runtime/Renderer/UnityRenderer.cs | 65 ++ Runtime/Renderer/UnityRenderer.cs.meta | 11 + Runtime/UNes.cs | 251 ++++++ Runtime/UNes.cs.meta | 11 + Runtime/Utility.cs | 35 + Runtime/Utility.cs.meta | 11 + Sample.meta | 8 + Sample/Resources.meta | 8 + Sample/Resources/Mario.bytes | Bin 0 -> 40976 bytes Sample/Resources/Mario.bytes.meta | 7 + Sample/Scene.meta | 8 + Sample/Scene/UNes_Demo.unity | 825 ++++++++++++++++++ Sample/Scene/UNes_Demo.unity.meta | 7 + Sample/Texture.meta | 8 + Sample/Texture/RenderTexture.renderTexture | 37 + .../Texture/RenderTexture.renderTexture.meta | 8 + 92 files changed, 4611 insertions(+) create mode 100644 .gitignore create mode 100644 Aya.UNes.asmdef create mode 100644 Aya.UNes.asmdef.meta create mode 100644 README.md create mode 100644 Runtime.meta create mode 100644 Runtime/Addressable.cs create mode 100644 Runtime/Addressable.cs.meta create mode 100644 Runtime/CPU.Core.cs create mode 100644 Runtime/CPU.Core.cs.meta create mode 100644 Runtime/CPU.IORegisters.cs create mode 100644 Runtime/CPU.IORegisters.cs.meta create mode 100644 Runtime/CPU.Instructions.cs create mode 100644 Runtime/CPU.Instructions.cs.meta create mode 100644 Runtime/CPU.Memory.cs create mode 100644 Runtime/CPU.Memory.cs.meta create mode 100644 Runtime/CPU.Registers.cs create mode 100644 Runtime/CPU.Registers.cs.meta create mode 100644 Runtime/CPU.cs create mode 100644 Runtime/CPU.cs.meta create mode 100644 Runtime/Cartridge.cs create mode 100644 Runtime/Cartridge.cs.meta create mode 100644 Runtime/Controller.meta create mode 100644 Runtime/Controller/IController.cs create mode 100644 Runtime/Controller/IController.cs.meta create mode 100644 Runtime/Controller/NesController.cs create mode 100644 Runtime/Controller/NesController.cs.meta create mode 100644 Runtime/Emulator.cs create mode 100644 Runtime/Emulator.cs.meta create mode 100644 Runtime/Mapper.meta create mode 100644 Runtime/Mapper/AxROM.cs create mode 100644 Runtime/Mapper/AxROM.cs.meta create mode 100644 Runtime/Mapper/BaseMapper.cs create mode 100644 Runtime/Mapper/BaseMapper.cs.meta create mode 100644 Runtime/Mapper/CNROM.cs create mode 100644 Runtime/Mapper/CNROM.cs.meta create mode 100644 Runtime/Mapper/Camerica.cs create mode 100644 Runtime/Mapper/Camerica.cs.meta create mode 100644 Runtime/Mapper/ColorDreams.cs create mode 100644 Runtime/Mapper/ColorDreams.cs.meta create mode 100644 Runtime/Mapper/DxROM.cs create mode 100644 Runtime/Mapper/DxROM.cs.meta create mode 100644 Runtime/Mapper/GxROM.cs create mode 100644 Runtime/Mapper/GxROM.cs.meta create mode 100644 Runtime/Mapper/Jaleco.cs create mode 100644 Runtime/Mapper/Jaleco.cs.meta create mode 100644 Runtime/Mapper/MMC1.cs create mode 100644 Runtime/Mapper/MMC1.cs.meta create mode 100644 Runtime/Mapper/MMC2.cs create mode 100644 Runtime/Mapper/MMC2.cs.meta create mode 100644 Runtime/Mapper/MMC3.cs create mode 100644 Runtime/Mapper/MMC3.cs.meta create mode 100644 Runtime/Mapper/MMC4.cs create mode 100644 Runtime/Mapper/MMC4.cs.meta create mode 100644 Runtime/Mapper/Mapper094.cs create mode 100644 Runtime/Mapper/Mapper094.cs.meta create mode 100644 Runtime/Mapper/Mapper155.cs create mode 100644 Runtime/Mapper/Mapper155.cs.meta create mode 100644 Runtime/Mapper/Mapper180.cs create mode 100644 Runtime/Mapper/Mapper180.cs.meta create mode 100644 Runtime/Mapper/NROM.cs create mode 100644 Runtime/Mapper/NROM.cs.meta create mode 100644 Runtime/Mapper/Nina003006.cs create mode 100644 Runtime/Mapper/Nina003006.cs.meta create mode 100644 Runtime/Mapper/UxROM.cs create mode 100644 Runtime/Mapper/UxROM.cs.meta create mode 100644 Runtime/PPU.Core.cs create mode 100644 Runtime/PPU.Core.cs.meta create mode 100644 Runtime/PPU.Memory.cs create mode 100644 Runtime/PPU.Memory.cs.meta create mode 100644 Runtime/PPU.Registers.cs create mode 100644 Runtime/PPU.Registers.cs.meta create mode 100644 Runtime/PPU.cs create mode 100644 Runtime/PPU.cs.meta create mode 100644 Runtime/Renderer.meta create mode 100644 Runtime/Renderer/IRenderer.cs create mode 100644 Runtime/Renderer/IRenderer.cs.meta create mode 100644 Runtime/Renderer/UnityRenderer.cs create mode 100644 Runtime/Renderer/UnityRenderer.cs.meta create mode 100644 Runtime/UNes.cs create mode 100644 Runtime/UNes.cs.meta create mode 100644 Runtime/Utility.cs create mode 100644 Runtime/Utility.cs.meta create mode 100644 Sample.meta create mode 100644 Sample/Resources.meta create mode 100644 Sample/Resources/Mario.bytes create mode 100644 Sample/Resources/Mario.bytes.meta create mode 100644 Sample/Scene.meta create mode 100644 Sample/Scene/UNes_Demo.unity create mode 100644 Sample/Scene/UNes_Demo.unity.meta create mode 100644 Sample/Texture.meta create mode 100644 Sample/Texture/RenderTexture.renderTexture create mode 100644 Sample/Texture/RenderTexture.renderTexture.meta diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..e8306c1 --- /dev/null +++ b/.gitignore @@ -0,0 +1,35 @@ +# Addinational File +LICENSE.meta +README.md.meta +README_CN.md.meta + +/[Ll]ibrary/ +/[Tt]emp/ +/[Oo]bj/ +/[Bb]uild/ +/[Bb]uilds/ +/[Pp]rojectSettings/ProjectVersion.txt +/Assets/AssetStoreTools* + +# Autogenerated VS/MD solution and project files +ExportedObj/ +*.csproj +*.unityproj +*.sln +*.suo +*.tmp +*.user +*.userprefs +*.pidb +*.booproj +*.svd + +# Unity3D generated meta files +*.pidb.meta + +# Unity3D Generated File On Crash Reports +sysinfo.txt + +# Builds +*.apk +*.unitypackage \ No newline at end of file diff --git a/Aya.UNes.asmdef b/Aya.UNes.asmdef new file mode 100644 index 0000000..308bc69 --- /dev/null +++ b/Aya.UNes.asmdef @@ -0,0 +1,13 @@ +{ + "name": "Aya.UNes", + "references": [], + "includePlatforms": [], + "excludePlatforms": [], + "allowUnsafeCode": true, + "overrideReferences": false, + "precompiledReferences": [], + "autoReferenced": true, + "defineConstraints": [], + "versionDefines": [], + "noEngineReferences": false +} \ No newline at end of file diff --git a/Aya.UNes.asmdef.meta b/Aya.UNes.asmdef.meta new file mode 100644 index 0000000..11f4d03 --- /dev/null +++ b/Aya.UNes.asmdef.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: 3fe77f1eed9fc0847a86648f644fe815 +AssemblyDefinitionImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/README.md b/README.md new file mode 100644 index 0000000..7514485 --- /dev/null +++ b/README.md @@ -0,0 +1,3 @@ +# UNes + +本项目修改自 https://github.com/Xyene/Emulator.NES \ No newline at end of file diff --git a/Runtime.meta b/Runtime.meta new file mode 100644 index 0000000..079a930 --- /dev/null +++ b/Runtime.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: ca1eb074eb7f66946b0d974b802d7b8d +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Addressable.cs b/Runtime/Addressable.cs new file mode 100644 index 0000000..891a96d --- /dev/null +++ b/Runtime/Addressable.cs @@ -0,0 +1,62 @@ +using System.Runtime.CompilerServices; + +namespace Aya.UNes +{ + public abstract class Addressable + { + public delegate uint ReadDelegate(uint address); + + public delegate void WriteDelegate(uint address, byte val); + + protected readonly Emulator _emulator; + protected readonly ReadDelegate[] _readMap; + protected readonly WriteDelegate[] _writeMap; + protected readonly uint _addressSize; + + protected Addressable(Emulator emulator, uint addressSpace) + { + _emulator = emulator; + _addressSize = addressSpace; + _readMap = new ReadDelegate[addressSpace + 1]; + _writeMap = new WriteDelegate[addressSpace + 1]; + } + + protected virtual void InitializeMemoryMap() + { + _readMap.Fill(address => 0); + + // Some games write to addresses not mapped and expect to continue afterwards + _writeMap.Fill((address, val) => { }); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public uint ReadByte(uint address) + { + address &= _addressSize; + return _readMap[address](address); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public void WriteByte(uint address, uint val) + { + address &= _addressSize; + _writeMap[address](address, (byte)val); + } + + public void MapReadHandler(uint start, uint end, CPU.ReadDelegate func) + { + for (uint i = start; i <= end; i++) + { + _readMap[i] = func; + } + } + + public void MapWriteHandler(uint start, uint end, CPU.WriteDelegate func) + { + for (uint i = start; i <= end; i++) + { + _writeMap[i] = func; + } + } + } +} diff --git a/Runtime/Addressable.cs.meta b/Runtime/Addressable.cs.meta new file mode 100644 index 0000000..84e599c --- /dev/null +++ b/Runtime/Addressable.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8649a70e245ed1744aa620cb28504fb5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/CPU.Core.cs b/Runtime/CPU.Core.cs new file mode 100644 index 0000000..4785f0d --- /dev/null +++ b/Runtime/CPU.Core.cs @@ -0,0 +1,81 @@ +using System; + +namespace Aya.UNes +{ + sealed partial class CPU + { + public enum InterruptType + { + NMI, + IRQ, + RESET + } + + private readonly uint[] _interruptHandlerOffsets = { 0xFFFA, 0xFFFE, 0xFFFC }; + private readonly bool[] _interrupts = new bool[2]; + + public void Initialize() + { + A = 0; + X = 0; + Y = 0; + SP = 0xFD; + P = 0x24; + + PC = ReadWord(_interruptHandlerOffsets[(int) InterruptType.RESET]); + } + + public void Reset() + { + SP -= 3; + F.InterruptsDisabled = true; + } + + public void TickFromPPU() + { + if (Cycle-- > 0) return; + ExecuteSingleInstruction(); + } + + public void ExecuteSingleInstruction() + { + for (var i = 0; i < _interrupts.Length; i++) + { + if (_interrupts[i]) + { + PushWord(PC); + Push(P); + PC = ReadWord(_interruptHandlerOffsets[i]); + F.InterruptsDisabled = true; + _interrupts[i] = false; + return; + } + } + + _currentInstruction = NextByte(); + + Cycle += _opCodeDefs[_currentInstruction].Cycles; + + ResetInstructionAddressingMode(); + // if (_numExecuted > 10000 && PC - 1 == 0xFF61) + // if(_emulator.Controller.debug || 0x6E00 <= PC && PC <= 0x6EEF) + // Console.WriteLine($"{(PC - 1).ToString("X4")} {_currentInstruction.ToString("X2")} {opcodeNames[_currentInstruction]}\t\t\tA:{A.ToString("X2")} X:{X.ToString("X2")} Y:{Y.ToString("X2")} P:{P.ToString("X2")} SP:{SP.ToString("X2")}"); + + var op = _opCodes[_currentInstruction]; + if (op == null) + { + throw new ArgumentException(_currentInstruction.ToString("X2")); + } + + op(); + } + + public void TriggerInterrupt(InterruptType type) + { + if (!F.InterruptsDisabled || type == InterruptType.NMI) + { + _interrupts[(int)type] = true; + } + } + } +} diff --git a/Runtime/CPU.Core.cs.meta b/Runtime/CPU.Core.cs.meta new file mode 100644 index 0000000..5f375dc --- /dev/null +++ b/Runtime/CPU.Core.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 63ec605962bce0e4899b19d49d4ed747 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/CPU.IORegisters.cs b/Runtime/CPU.IORegisters.cs new file mode 100644 index 0000000..0fa4785 --- /dev/null +++ b/Runtime/CPU.IORegisters.cs @@ -0,0 +1,44 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Runtime.CompilerServices; +using System.Text; +using System.Threading.Tasks; + +namespace Aya.UNes +{ + sealed partial class CPU + { + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public void WriteIoRegister(uint reg, byte val) + { + switch (reg) + { + case 0x4014: // OAM DMA + _emulator.PPU.PerformDMA(val); + break; + case 0x4016: + _emulator.Controller.Strobe(val == 1); + break; + } + + if (reg <= 0x401F) + { + return; // APU write + } + + throw new NotImplementedException($"{reg:X4} = {val:X2}"); + } + + public uint ReadIORegister(uint reg) + { + switch (reg) + { + case 0x4016: + return (uint) _emulator.Controller.ReadState() & 0x1; + } + return 0x00; + //throw new NotImplementedException(); + } + } +} diff --git a/Runtime/CPU.IORegisters.cs.meta b/Runtime/CPU.IORegisters.cs.meta new file mode 100644 index 0000000..42ddf45 --- /dev/null +++ b/Runtime/CPU.IORegisters.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6ad8b3c04ef34de48b5cab3029811d9d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/CPU.Instructions.cs b/Runtime/CPU.Instructions.cs new file mode 100644 index 0000000..459d834 --- /dev/null +++ b/Runtime/CPU.Instructions.cs @@ -0,0 +1,463 @@ +using System; +using static Aya.UNes.CPU.AddressingMode; + +namespace Aya.UNes +{ + public sealed partial class CPU + { + [AttributeUsage(AttributeTargets.Method, Inherited = false, AllowMultiple = true)] + public class OpCodeDef : Attribute + { + public int OpCode; + public int Cycles = 1; + public bool PageBoundary; + public bool RMW; + public AddressingMode Mode = None; + } + + [OpCodeDef(OpCode = 0x20, Cycles = 6)] + private void JSR() + { + PushWord(PC + 1); + PC = NextWord(); + } + + [OpCodeDef(OpCode = 0x40, Cycles = 6)] + private void RTI() + { + // TODO: this dummy fetch should happen for all single-byte instructions + NextByte(); + P = Pop(); + PC = PopWord(); + } + + [OpCodeDef(OpCode = 0x60, Cycles = 6)] + private void RTS() + { + NextByte(); + PC = PopWord() + 1; + } + + [OpCodeDef(OpCode = 0xC8, Cycles = 2)] + private void INY() => Y++; + + [OpCodeDef(OpCode = 0x88, Cycles = 2)] + private void DEY() => Y--; + + [OpCodeDef(OpCode = 0xE8, Cycles = 2)] + private void INX() => X++; + + [OpCodeDef(OpCode = 0xCA, Cycles = 2, RMW = true)] + private void DEX() => X--; + + [OpCodeDef(OpCode = 0xA8, Cycles = 2)] + private void TAY() => Y = A; + + [OpCodeDef(OpCode = 0x98, Cycles = 2)] + private void TYA() => A = Y; + + [OpCodeDef(OpCode = 0xAA, Cycles = 2, RMW = true)] + private void TAX() => X = A; + + [OpCodeDef(OpCode = 0x8A, Cycles = 2, RMW = true)] + private void TXA() => A = X; + + [OpCodeDef(OpCode = 0xBA, Cycles = 2)] + private void TSX() => X = SP; + + [OpCodeDef(OpCode = 0x9A, Cycles = 2, RMW = true)] + private void TXS() => SP = X; + + [OpCodeDef(OpCode = 0x08, Cycles = 3)] + private void PHP() => Push(P | BreakSourceBit); + + [OpCodeDef(OpCode = 0x28, Cycles = 4)] + private void PLP() => P = (uint)(Pop() & ~BreakSourceBit); + + [OpCodeDef(OpCode = 0x68, Cycles = 4)] + private void PLA() => A = Pop(); + + [OpCodeDef(OpCode = 0x48, Cycles = 3)] + private void PHA() => Push(A); + + [OpCodeDef(OpCode = 0x24, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x2C, Mode = Absolute, Cycles = 4)] + private void BIT() + { + uint val = AddressRead(); + F.Overflow = (val & 0x40) > 0; + F.Zero = (val & A) == 0; + F.Negative = (val & 0x80) > 0; + } + + private void Branch(bool cond) + { + uint nPC = (uint)(PC + NextSByte() + 1); + if (cond) + { + PC = nPC; + Cycle++; + } + } + + [OpCodeDef(OpCode = 0x4C, Cycles = 3)] + [OpCodeDef(OpCode = 0x6C, Cycles = 5)] + private void JMP() + { + if (_currentInstruction == 0x4C) + { + PC = NextWord(); + } + else if (_currentInstruction == 0x6C) + { + uint off = NextWord(); + // AN INDIRECT JUMP MUST NEVER USE A VECTOR BEGINNING ON THE LAST BYTE OF A PAGE + // + // If address $3000 contains $40, $30FF contains $80, and $3100 contains $50, + // the result of JMP ($30FF) will be a transfer of control to $4080 rather than + // $5080 as you intended i.e. the 6502 took the low byte of the address from + // $30FF and the high byte from $3000. + // + // http://www.6502.org/tutorials/6502opcodes.html + uint hi = (off & 0xFF) == 0xFF ? off - 0xFF : off + 1; + uint oldPC = PC; + PC = ReadByte(off) | (ReadByte(hi) << 8); + + if ((oldPC & 0xFF00) != (PC & 0xFF00)) + { + Cycle += 2; + } + } + else + { + throw new NotImplementedException(); + } + } + + [OpCodeDef(OpCode = 0xB0, Cycles = 2)] + private void BCS() => Branch(F.Carry); + + [OpCodeDef(OpCode = 0x90, Cycles = 2)] + private void BCC() => Branch(!F.Carry); + + [OpCodeDef(OpCode = 0xF0, Cycles = 2)] + private void BEQ() => Branch(F.Zero); + + [OpCodeDef(OpCode = 0xD0, Cycles = 2)] + private void BNE() => Branch(!F.Zero); + + [OpCodeDef(OpCode = 0x70, Cycles = 2)] + private void BVS() => Branch(F.Overflow); + + [OpCodeDef(OpCode = 0x50, Cycles = 2)] + private void BVC() => Branch(!F.Overflow); + + [OpCodeDef(OpCode = 0x10, Cycles = 2)] + private void BPL() => Branch(!F.Negative); + + [OpCodeDef(OpCode = 0x30, Cycles = 2)] + private void BMI() => Branch(F.Negative); + + [OpCodeDef(OpCode = 0x81, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0x91, Mode = IndirectY, Cycles = 6)] + [OpCodeDef(OpCode = 0x95, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0x99, Mode = AbsoluteY, Cycles = 5)] + [OpCodeDef(OpCode = 0x9D, Mode = AbsoluteX, Cycles = 5)] + [OpCodeDef(OpCode = 0x85, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x8D, Mode = Absolute, Cycles = 4)] + private void STA() => AddressWrite(A); + + [OpCodeDef(OpCode = 0x96, Mode = ZeroPageY, Cycles = 4)] + [OpCodeDef(OpCode = 0x86, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x8E, Mode = Absolute, Cycles = 4)] + private void STX() => AddressWrite(X); + + [OpCodeDef(OpCode = 0x94, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0x84, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x8C, Mode = Absolute, Cycles = 4)] + private void STY() => AddressWrite(Y); + + [OpCodeDef(OpCode = 0x18, Cycles = 2)] + private void CLC() => F.Carry = false; + + [OpCodeDef(OpCode = 0x38, Cycles = 2)] + private void SEC() => F.Carry = true; + + [OpCodeDef(OpCode = 0x58, Cycles = 2)] + private void CLI() => F.InterruptsDisabled = false; + + [OpCodeDef(OpCode = 0x78, Cycles = 2)] + private void SEI() => F.InterruptsDisabled = true; + + [OpCodeDef(OpCode = 0xB8, Cycles = 2)] + private void CLV() => F.Overflow = false; + + [OpCodeDef(OpCode = 0xD8, Cycles = 2)] + private void CLD() => F.DecimalMode = false; + + [OpCodeDef(OpCode = 0xF8, Cycles = 2)] + private void SED() => F.DecimalMode = true; + + [OpCodeDef(OpCode = 0xEA, Cycles = 2)] + [OpCodeDef(OpCode = 0x1A, Cycles = 2)] // Unofficial + [OpCodeDef(OpCode = 0x3A, Cycles = 2)] // Unofficial + [OpCodeDef(OpCode = 0x5A, Cycles = 2)] // Unofficial + [OpCodeDef(OpCode = 0x7A, Cycles = 2)] // Unofficial + [OpCodeDef(OpCode = 0xDA, Cycles = 2)] // Unofficial + [OpCodeDef(OpCode = 0xFA, Cycles = 2)] // Unofficial + private void NOP() { } + + [OpCodeDef(OpCode = 0xA1, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0xA5, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0xA9, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0xAD, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0xB1, Mode = IndirectY, Cycles = 5, PageBoundary = true)] + [OpCodeDef(OpCode = 0xB5, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0xB9, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)] + [OpCodeDef(OpCode = 0xBD, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void LDA() => A = AddressRead(); + + [OpCodeDef(OpCode = 0xA0, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0xA4, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0xAC, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0xB4, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0xBC, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void LDY() => Y = AddressRead(); + + [OpCodeDef(OpCode = 0xA2, Mode = Immediate, Cycles = 2, RMW = true)] + [OpCodeDef(OpCode = 0xA6, Mode = ZeroPage, Cycles = 3, RMW = true)] + [OpCodeDef(OpCode = 0xAE, Mode = Absolute, Cycles = 4, RMW = true)] + [OpCodeDef(OpCode = 0xB6, Mode = ZeroPageY, Cycles = 4, RMW = true)] + [OpCodeDef(OpCode = 0xBE, Mode = AbsoluteY, Cycles = 4, PageBoundary = true, RMW = true)] + private void LDX() => X = AddressRead(); + + [OpCodeDef(OpCode = 0x01, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0x05, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x09, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0x0D, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0x11, Mode = IndirectY, Cycles = 5, PageBoundary = true)] + [OpCodeDef(OpCode = 0x15, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0x19, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)] + [OpCodeDef(OpCode = 0x1D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void ORA() => A |= AddressRead(); + + [OpCodeDef(OpCode = 0x21, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0x25, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x29, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0x2D, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0x31, Mode = IndirectY, Cycles = 5, PageBoundary = true)] + [OpCodeDef(OpCode = 0x35, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0x39, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)] + [OpCodeDef(OpCode = 0x3D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void AND() => A &= AddressRead(); + + [OpCodeDef(OpCode = 0x41, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0x45, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x49, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0x4D, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0x51, Mode = IndirectY, Cycles = 5, PageBoundary = true)] + [OpCodeDef(OpCode = 0x55, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0x59, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)] + [OpCodeDef(OpCode = 0x5D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void EOR() => A ^= AddressRead(); + + [OpCodeDef(OpCode = 0xE1, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0xE5, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x69, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0xE9, Mode = Immediate, Cycles = 2)] // Official duplicate of $69 + [OpCodeDef(OpCode = 0xEB, Mode = Immediate, Cycles = 2)] // Unofficial duplicate of $69 + [OpCodeDef(OpCode = 0xED, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0xF1, Mode = IndirectY, Cycles = 5, PageBoundary = true)] + [OpCodeDef(OpCode = 0xF5, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0xF9, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)] + [OpCodeDef(OpCode = 0xFD, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void SBC() => ADCImpl((byte)~AddressRead()); + + [OpCodeDef(OpCode = 0x61, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0x65, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0x69, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0x6D, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0x71, Mode = IndirectY, Cycles = 5, PageBoundary = true)] + [OpCodeDef(OpCode = 0x75, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0x79, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)] + [OpCodeDef(OpCode = 0x7D, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void ADC() => ADCImpl(AddressRead()); + + private void ADCImpl(uint val) + { + int nA = (sbyte)A + (sbyte)val + (sbyte)(F.Carry ? 1 : 0); + F.Overflow = nA < -128 || nA > 127; + F.Carry = (A + val + (F.Carry ? 1 : 0)) > 0xFF; + A = (byte)(nA & 0xFF); + } + + [OpCodeDef(OpCode = 0x00, Cycles = 7)] + private void BRK() + { + NextByte(); + Push(P | BreakSourceBit); + F.InterruptsDisabled = true; + PC = ReadByte(0xFFFE) | (ReadByte(0xFFFF) << 8); + } + + [OpCodeDef(OpCode = 0xC1, Mode = IndirectX, Cycles = 6)] + [OpCodeDef(OpCode = 0xC5, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0xC9, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0xCD, Mode = Absolute, Cycles = 4)] + [OpCodeDef(OpCode = 0xD1, Mode = IndirectY, Cycles = 5, PageBoundary = true)] + [OpCodeDef(OpCode = 0xD5, Mode = ZeroPageX, Cycles = 4)] + [OpCodeDef(OpCode = 0xD9, Mode = AbsoluteY, Cycles = 4, PageBoundary = true)] + [OpCodeDef(OpCode = 0xDD, Mode = AbsoluteX, Cycles = 4, PageBoundary = true)] + private void CMP() => CMPImpl(A); + + [OpCodeDef(OpCode = 0xE0, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0xE4, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0xEC, Mode = Absolute, Cycles = 4)] + private void CPX() => CMPImpl(X); + + [OpCodeDef(OpCode = 0xC0, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0xC4, Mode = ZeroPage, Cycles = 3)] + [OpCodeDef(OpCode = 0xCC, Mode = Absolute, Cycles = 4)] + private void CPY() => CMPImpl(Y); + + private void CMPImpl(uint reg) + { + long d = reg - (int)AddressRead(); + + F.Negative = (d & 0x80) > 0 && d != 0; + F.Carry = d >= 0; + F.Zero = d == 0; + } + + [OpCodeDef(OpCode = 0x46, Mode = ZeroPage, Cycles = 5, RMW = true)] + [OpCodeDef(OpCode = 0x4E, Mode = Absolute, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x56, Mode = ZeroPageX, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x5E, Mode = AbsoluteX, Cycles = 7, RMW = true)] + [OpCodeDef(OpCode = 0x4A, Mode = Direct, Cycles = 2, RMW = true)] + private void LSR() + { + uint D = AddressRead(); + F.Carry = (D & 0x1) > 0; + D >>= 1; + _F(D); + AddressWrite(D); + } + + [OpCodeDef(OpCode = 0x06, Mode = ZeroPage, Cycles = 5, RMW = true)] + [OpCodeDef(OpCode = 0x0E, Mode = Absolute, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x16, Mode = ZeroPageX, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x1E, Mode = AbsoluteX, Cycles = 7, RMW = true)] + [OpCodeDef(OpCode = 0x0A, Mode = Direct, Cycles = 2, RMW = true)] + private void ASL() + { + uint D = AddressRead(); + F.Carry = (D & 0x80) > 0; + D <<= 1; + _F(D); + AddressWrite(D); + } + + [OpCodeDef(OpCode = 0x66, Mode = ZeroPage, Cycles = 5, RMW = true)] + [OpCodeDef(OpCode = 0x6E, Mode = Absolute, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x76, Mode = ZeroPageX, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x7E, Mode = AbsoluteX, Cycles = 7, RMW = true)] + [OpCodeDef(OpCode = 0x6A, Mode = Direct, Cycles = 2, RMW = true)] + private void ROR() + { + uint D = AddressRead(); + bool c = F.Carry; + F.Carry = (D & 0x1) > 0; + D >>= 1; + if (c) D |= 0x80; + _F(D); + AddressWrite(D); + } + + [OpCodeDef(OpCode = 0x26, Mode = ZeroPage, Cycles = 5, RMW = true)] + [OpCodeDef(OpCode = 0x2E, Mode = Absolute, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x36, Mode = ZeroPageX, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0x3E, Mode = AbsoluteX, Cycles = 7, RMW = true)] + [OpCodeDef(OpCode = 0x2A, Mode = Direct, Cycles = 2, RMW = true)] + private void ROL() + { + uint D = AddressRead(); + bool c = F.Carry; + F.Carry = (D & 0x80) > 0; + D <<= 1; + if (c) D |= 0x1; + _F(D); + AddressWrite(D); + } + + [OpCodeDef(OpCode = 0xE6, Mode = ZeroPage, Cycles = 5, RMW = true)] + [OpCodeDef(OpCode = 0xEE, Mode = Absolute, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0xF6, Mode = ZeroPageX, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0xFE, Mode = AbsoluteX, Cycles = 7, RMW = true)] + private void INC() + { + byte D = (byte)(AddressRead() + 1); + _F(D); + AddressWrite(D); + } + + [OpCodeDef(OpCode = 0xC6, Mode = ZeroPage, Cycles = 5, RMW = true)] + [OpCodeDef(OpCode = 0xCE, Mode = Absolute, Cycles = 3, RMW = true)] + [OpCodeDef(OpCode = 0xD6, Mode = ZeroPageX, Cycles = 6, RMW = true)] + [OpCodeDef(OpCode = 0xDE, Mode = AbsoluteX, Cycles = 7, RMW = true)] + private void DEC() + { + byte D = (byte)(AddressRead() - 1); + _F(D); + AddressWrite(D); + } + + #region Unofficial Opcodes + + [OpCodeDef(OpCode = 0x80, Cycles = 2)] + [OpCodeDef(OpCode = 0x82, Cycles = 2)] + [OpCodeDef(OpCode = 0x89, Cycles = 2)] + [OpCodeDef(OpCode = 0xC2, Cycles = 2)] + [OpCodeDef(OpCode = 0xE2, Cycles = 2)] + private void SKB() => NextByte(); // Essentially a 2-byte NOP + + [OpCodeDef(OpCode = 0x0B, Mode = Immediate, Cycles = 2)] + [OpCodeDef(OpCode = 0x2B, Mode = Immediate, Cycles = 2)] + private void ANC() + { + A &= AddressRead(); + F.Carry = F.Negative; + } + + [OpCodeDef(OpCode = 0x4B, Mode = Immediate, Cycles = 2)] + private void ALR() + { + A &= AddressRead(); + F.Carry = (A & 0x1) > 0; + A >>= 1; + _F(A); + } + + [OpCodeDef(OpCode = 0x6B, Mode = Immediate, Cycles = 2)] + private void ARR() + { + A &= AddressRead(); + bool c = F.Carry; + F.Carry = (A & 0x1) > 0; + A >>= 1; + if (c) A |= 0x80; + _F(A); + } + + [OpCodeDef(OpCode = 0xAB, Mode = Immediate, Cycles = 2)] + private void ATX() + { + // This opcode ORs the A register with #$EE, ANDs the result with an immediate + // value, and then stores the result in both A and X. + A |= ReadByte(0xEE); + A &= AddressRead(); + X = A; + } + + #endregion + } +} diff --git a/Runtime/CPU.Instructions.cs.meta b/Runtime/CPU.Instructions.cs.meta new file mode 100644 index 0000000..7817694 --- /dev/null +++ b/Runtime/CPU.Instructions.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: cae8846694d75f04a8214a484032d533 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/CPU.Memory.cs b/Runtime/CPU.Memory.cs new file mode 100644 index 0000000..7c3291e --- /dev/null +++ b/Runtime/CPU.Memory.cs @@ -0,0 +1,165 @@ +using System; +using System.Linq; +using System.Runtime.CompilerServices; +using static Aya.UNes.CPU.AddressingMode; + +namespace Aya.UNes +{ + sealed partial class CPU + { + public enum AddressingMode + { + None, + Direct, + Immediate, + ZeroPage, + Absolute, + ZeroPageX, + ZeroPageY, + AbsoluteX, + AbsoluteY, + IndirectX, + IndirectY + } + + private uint? _currentMemoryAddress; + private uint _rmwValue; + + private void ResetInstructionAddressingMode() => _currentMemoryAddress = null; + + private uint _Address() + { + var def = _opCodeDefs[_currentInstruction]; + switch (def.Mode) + { + case Immediate: + return PC++; + case ZeroPage: + return NextByte(); + case Absolute: + return NextWord(); + case ZeroPageX: + return (NextByte() + X) & 0xFF; + case ZeroPageY: + return (NextByte() + Y) & 0xFF; + case AbsoluteX: + var address = NextWord(); + if (def.PageBoundary && (address & 0xFF00) != ((address + X) & 0xFF00)) + { + Cycle += 1; + } + + return address + X; + case AbsoluteY: + address = NextWord(); + if (def.PageBoundary && (address & 0xFF00) != ((address + Y) & 0xFF00)) + { + Cycle += 1; + } + + return address + Y; + case IndirectX: + var off = (NextByte() + X) & 0xFF; + return ReadByte(off) | (ReadByte((off + 1) & 0xFF) << 8); + case IndirectY: + off = NextByte() & 0xFF; + address = ReadByte(off) | (ReadByte((off + 1) & 0xFF) << 8); + if (def.PageBoundary && (address & 0xFF00) != ((address + Y) & 0xFF00)) + { + Cycle += 1; + } + + return (address + Y) & 0xFFFF; + } + throw new NotImplementedException(); + } + + public uint AddressRead() + { + if (_opCodeDefs[_currentInstruction].Mode == Direct) + { + return _rmwValue = A; + } + + if (_currentMemoryAddress == null) + { + _currentMemoryAddress = _Address(); + } + + return _rmwValue = ReadByte((uint)_currentMemoryAddress) & 0xFF; + } + + public void AddressWrite(uint val) + { + if (_opCodeDefs[_currentInstruction].Mode == Direct) + { + A = val; + } + else + { + if (_currentMemoryAddress == null) + { + _currentMemoryAddress = _Address(); + } + + if (_opCodeDefs[_currentInstruction].RMW) + { + WriteByte((uint)_currentMemoryAddress, _rmwValue); + } + + WriteByte((uint)_currentMemoryAddress, val); + } + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private uint ReadWord(uint address) => ReadByte(address) | (ReadByte(address + 1) << 8); + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private uint NextByte() => ReadByte(PC++); + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private uint NextWord() => NextByte() | (NextByte() << 8); + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private sbyte NextSByte() => (sbyte)NextByte(); + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private void Push(uint what) + { + WriteByte(0x100 + SP, what); + SP--; + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private uint Pop() + { + SP++; + return ReadByte(0x100 + SP); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private void PushWord(uint what) + { + Push(what >> 8); + Push(what & 0xFF); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private uint PopWord() => Pop() | (Pop() << 8); + + protected override void InitializeMemoryMap() + { + base.InitializeMemoryMap(); + + MapReadHandler(0x0000, 0x1FFF, address => _ram[address & 0x07FF]); + MapReadHandler(0x2000, 0x3FFF, address => _emulator.PPU.ReadRegister((address & 0x7) - 0x2000)); + MapReadHandler(0x4000, 0x4017, ReadIORegister); + + MapWriteHandler(0x0000, 0x1FFF, (address, val) => _ram[address & 0x07FF] = val); + MapWriteHandler(0x2000, 0x3FFF, (address, val) => _emulator.PPU.WriteRegister((address & 0x7) - 0x2000, val)); + MapWriteHandler(0x4000, 0x401F, WriteIoRegister); + + _emulator.Mapper.InitializeMemoryMap(this); + } + } +} diff --git a/Runtime/CPU.Memory.cs.meta b/Runtime/CPU.Memory.cs.meta new file mode 100644 index 0000000..e10a02a --- /dev/null +++ b/Runtime/CPU.Memory.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: cc18416f50415b74e96092e235a95f92 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/CPU.Registers.cs b/Runtime/CPU.Registers.cs new file mode 100644 index 0000000..dfe05fb --- /dev/null +++ b/Runtime/CPU.Registers.cs @@ -0,0 +1,89 @@ +using System.Runtime.CompilerServices; + +namespace Aya.UNes +{ + sealed partial class CPU + { + private const int CarryBit = 0x1; + private const int ZeroBit = 0x2; + private const int InterruptDisabledBit = 0x4; + private const int DecimalModeBit = 0x8; + private const int BreakSourceBit = 0x10; + private const int OverflowBit = 0x40; + private const int NegativeBit = 0x80; + + public class CPUFlags + { + public bool Negative; + public bool Overflow; + public bool BreakSource; + public bool DecimalMode; + public bool InterruptsDisabled; + public bool Zero; + public bool Carry; + } + + public readonly CPUFlags F = new CPUFlags(); + + public uint _A, _X, _Y, _SP; + public uint PC; + + public uint A + { + get => _A; + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private set => _A = _F(value & 0xFF); + } + + public uint X + { + get => _X; + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private set => _X = _F(value & 0xFF); + } + + public uint Y + { + get => _Y; + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private set => _Y = _F(value & 0xFF); + } + + public uint SP + { + get => _SP; + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private set => _SP = value & 0xFF; + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + private uint _F(uint val) + { + F.Zero = (val & 0xFF) == 0; + F.Negative = (val & 0x80) > 0; + return val; + } + + public uint P + { + get => (uint) ((F.Carry.AsByte() << 0) | + (F.Zero.AsByte() << 1) | + (F.InterruptsDisabled.AsByte() << 2) | + (F.DecimalMode.AsByte() << 3) | + (F.BreakSource.AsByte() << 4) | + (1 << 5) | + (F.Overflow.AsByte() << 6) | + (F.Negative.AsByte() << 7)); + set + { + F.Carry = (value & CarryBit) > 0; + F.Zero = (value & ZeroBit) > 0; + F.InterruptsDisabled = (value & InterruptDisabledBit) > 0; + F.DecimalMode = (value & DecimalModeBit) > 0; + F.BreakSource = (value & BreakSourceBit) > 0; + F.Overflow = (value & OverflowBit) > 0; + F.Negative = (value & NegativeBit) > 0; + } + } + } +} diff --git a/Runtime/CPU.Registers.cs.meta b/Runtime/CPU.Registers.cs.meta new file mode 100644 index 0000000..508a0d9 --- /dev/null +++ b/Runtime/CPU.Registers.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 57750595bb8e0d24babf00d51e4624bb +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/CPU.cs b/Runtime/CPU.cs new file mode 100644 index 0000000..ad35697 --- /dev/null +++ b/Runtime/CPU.cs @@ -0,0 +1,65 @@ +using System; +using System.Linq; +using System.Reflection; + +namespace Aya.UNes +{ + sealed partial class CPU : Addressable + { + private readonly byte[] _ram = new byte[0x800]; + public int Cycle; + private uint _currentInstruction; + + public delegate void OpCode(); + + private readonly OpCode[] _opCodes = new OpCode[256]; + private readonly string[] _opCodeNames = new string[256]; + private readonly OpCodeDef[] _opCodeDefs = new OpCodeDef[256]; + + public CPU(Emulator emulator) : base(emulator, 0xFFFF) + { + InitializeOpCodes(); + InitializeMemoryMap(); + Initialize(); + } + + private void InitializeOpCodes() + { + var opCodeBindings = from opCode in GetType().GetMethods(BindingFlags.NonPublic | BindingFlags.Instance) + let defs = opCode.GetCustomAttributes(typeof(OpCodeDef), false) + where defs.Length > 0 + select new + { + binding = (OpCode)Delegate.CreateDelegate(typeof(OpCode), this, opCode.Name), + name = opCode.Name, + defs = (from d in defs select (OpCodeDef)d) + }; + + foreach (var opCode in opCodeBindings) + { + foreach (var def in opCode.defs) + { + _opCodes[def.OpCode] = opCode.binding; + _opCodeNames[def.OpCode] = opCode.name; + _opCodeDefs[def.OpCode] = def; + } + } + } + + public void Execute() + { + for (var i = 0; i < 5000; i++) + { + ExecuteSingleInstruction(); + } + + uint w; + ushort x = 6000; + string z = ""; + while ((w = ReadByte(x)) != '\0') + { + z += (char)w; + } + } + } +} diff --git a/Runtime/CPU.cs.meta b/Runtime/CPU.cs.meta new file mode 100644 index 0000000..f80ca2b --- /dev/null +++ b/Runtime/CPU.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fa724afa42498dd4e80f44c3da2a5881 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Cartridge.cs b/Runtime/Cartridge.cs new file mode 100644 index 0000000..74f4e66 --- /dev/null +++ b/Runtime/Cartridge.cs @@ -0,0 +1,66 @@ +using System; + +namespace Aya.UNes +{ + public class Cartridge + { + public readonly byte[] Raw; + public readonly int PRGROMSize; + public readonly int CHRROMSize; + public readonly int PRGRAMSize; + public readonly int PRGROMOffset; + public readonly int MapperNumber; + public readonly byte[] PRGROM; + public readonly byte[] CHRROM; + public VRAMMirroringMode MirroringMode; + + public enum VRAMMirroringMode + { + Horizontal, Vertical, All, Upper, Lower + } + + public Cartridge(byte[] bytes) + { + Raw = bytes; + + var header = BitConverter.ToInt32(Raw, 0); + if (header != 0x1A53454E) // "NES" + { + throw new FormatException("unexpected header value " + header.ToString("X")); + } + + PRGROMSize = Raw[4] * 0x4000; // 16kb units + CHRROMSize = Raw[5] * 0x2000; // 8kb units + PRGRAMSize = Raw[8] * 0x2000; + + var hasTrainer = (Raw[6] & 0b100) > 0; + PRGROMOffset = 16 + (hasTrainer ? 512 : 0); + + MirroringMode = (Raw[6] & 0x1) > 0 ? VRAMMirroringMode.Vertical : VRAMMirroringMode.Horizontal; + if ((Raw[6] & 0x8) > 0) + { + MirroringMode = VRAMMirroringMode.All; + } + + MapperNumber = (Raw[6] >> 4) | (Raw[7] & 0xF0); + + PRGROM = new byte[PRGROMSize]; + Array.Copy(Raw, PRGROMOffset, PRGROM, 0, PRGROMSize); + + if (CHRROMSize == 0) + { + CHRROM = new byte[0x2000]; + } + else + { + CHRROM = new byte[CHRROMSize]; + Array.Copy(Raw, PRGROMOffset + PRGROMSize, CHRROM, 0, CHRROMSize); + } + } + + public override string ToString() + { + return $"Cartridge{{PRGROMSize={PRGROMSize}, CHRROMSize={CHRROMSize}, PRGROMOffset={PRGROMOffset}, MapperNumber={MapperNumber}}}"; + } + } +} diff --git a/Runtime/Cartridge.cs.meta b/Runtime/Cartridge.cs.meta new file mode 100644 index 0000000..4fcccd7 --- /dev/null +++ b/Runtime/Cartridge.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b45675095efbb224cad44ee16c2587ff +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Controller.meta b/Runtime/Controller.meta new file mode 100644 index 0000000..52daa48 --- /dev/null +++ b/Runtime/Controller.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 87f4900ceb2e3b24caa8eec8df6b34e7 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Controller/IController.cs b/Runtime/Controller/IController.cs new file mode 100644 index 0000000..6cff890 --- /dev/null +++ b/Runtime/Controller/IController.cs @@ -0,0 +1,15 @@ +using UnityEngine; + +namespace Aya.UNes.Controller +{ + public interface IController + { + void Strobe(bool on); + + int ReadState(); + + void PressKey(KeyCode keyCode); + + void ReleaseKey(KeyCode keyCode); + } +} diff --git a/Runtime/Controller/IController.cs.meta b/Runtime/Controller/IController.cs.meta new file mode 100644 index 0000000..097e178 --- /dev/null +++ b/Runtime/Controller/IController.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fde4f0c3566240b4fa824b147b019ff7 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Controller/NesController.cs b/Runtime/Controller/NesController.cs new file mode 100644 index 0000000..4c89a87 --- /dev/null +++ b/Runtime/Controller/NesController.cs @@ -0,0 +1,71 @@ +using System.Collections.Generic; +using UnityEngine; + +namespace Aya.UNes.Controller +{ + public class NesController : IController + { + private int _data; + private int _serialData; + private bool _strobing; + + public bool Debug; + // bit: 7 6 5 4 3 2 1 0 + // button: A B Select Start Up Down Left + + private readonly Dictionary _keyMapping = new Dictionary + { + {KeyCode.A, 7}, + {KeyCode.S, 6}, + {KeyCode.RightShift, 5}, + {KeyCode.Return, 4}, + {KeyCode.UpArrow, 3}, + {KeyCode.DownArrow, 2}, + {KeyCode.LeftArrow, 1}, + {KeyCode.RightArrow, 0}, + }; + + public void Strobe(bool on) + { + _serialData = _data; + _strobing = on; + } + + public int ReadState() + { + int ret = ((_serialData & 0x80) > 0).AsByte(); + if (!_strobing) + { + _serialData <<= 1; + _serialData &= 0xFF; + } + + return ret; + } + + public void PressKey(KeyCode keyCode) + { + if (keyCode == KeyCode.P) + { + Debug ^= true; + } + + if (!_keyMapping.ContainsKey(keyCode)) + { + return; + } + + _data |= 1 << _keyMapping[keyCode]; + } + + public void ReleaseKey(KeyCode keyCode) + { + if (!_keyMapping.ContainsKey(keyCode)) + { + return; + } + + _data &= ~(1 << _keyMapping[keyCode]); + } + } +} diff --git a/Runtime/Controller/NesController.cs.meta b/Runtime/Controller/NesController.cs.meta new file mode 100644 index 0000000..0582652 --- /dev/null +++ b/Runtime/Controller/NesController.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6a6001b2b26e33144b77fe0b02e21dc0 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Emulator.cs b/Runtime/Emulator.cs new file mode 100644 index 0000000..5bae444 --- /dev/null +++ b/Runtime/Emulator.cs @@ -0,0 +1,66 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Reflection; +using Aya.UNes.Controller; +using Aya.UNes.Mapper; + +namespace Aya.UNes +{ + public class Emulator + { + private static readonly Dictionary> Mappers = (from type in Assembly.GetExecutingAssembly().GetTypes() + let def = (MapperDef)type.GetCustomAttributes(typeof(MapperDef), true).FirstOrDefault() + where def != null + select new { def, type }).ToDictionary(a => a.def.Id, a => new KeyValuePair(a.type, a.def)); + + public IController Controller; + + public readonly CPU CPU; + + public readonly PPU PPU; + + public readonly BaseMapper Mapper; + + public readonly Cartridge Cartridge; + + public Emulator(byte[] bytes, IController controller) + { + Cartridge = new Cartridge(bytes); + if (!Mappers.ContainsKey(Cartridge.MapperNumber)) + { + throw new NotImplementedException($"unsupported mapper {Cartridge.MapperNumber}"); + } + + Mapper = (BaseMapper)Activator.CreateInstance(Mappers[Cartridge.MapperNumber].Key, this); + CPU = new CPU(this); + PPU = new PPU(this); + Controller = controller; + + // Load(); + } + + public void Save(string path) + { + using (var fs = new FileStream(path + ".sav", FileMode.Create, FileAccess.Write)) + { + Mapper.Save(fs); + } + } + + public void Load(string path) + { + var sav = path + ".sav"; + if (!File.Exists(sav)) + { + return; + } + + using (var fs = new FileStream(sav, FileMode.Open, FileAccess.Read)) + { + Mapper.Load(fs); + } + } + } +} diff --git a/Runtime/Emulator.cs.meta b/Runtime/Emulator.cs.meta new file mode 100644 index 0000000..49774f0 --- /dev/null +++ b/Runtime/Emulator.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9c56d6518f6443745a10dd7dda0ac7f9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper.meta b/Runtime/Mapper.meta new file mode 100644 index 0000000..f05ed2d --- /dev/null +++ b/Runtime/Mapper.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: d22acc98b0a7fa042bb19b9894899fb7 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/AxROM.cs b/Runtime/Mapper/AxROM.cs new file mode 100644 index 0000000..2541c27 --- /dev/null +++ b/Runtime/Mapper/AxROM.cs @@ -0,0 +1,26 @@ +using static Aya.UNes.Cartridge.VRAMMirroringMode; + +namespace Aya.UNes.Mapper +{ + [MapperDef(7)] + public class AxROM : BaseMapper + { + protected int _bankOffset; + private readonly Cartridge.VRAMMirroringMode[] _mirroringModes = { Lower, Upper }; + + public AxROM(Emulator emulator) : base(emulator) + { + _emulator.Cartridge.MirroringMode = _mirroringModes[0]; + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_bankOffset + (address - 0x8000)]); + cpu.MapWriteHandler(0x8000, 0xFFFF, (address, val) => + { + _bankOffset = (val & 0x7) * 0x8000; + _emulator.Cartridge.MirroringMode = _mirroringModes[(val >> 4) & 0x1]; + }); + } + } +} diff --git a/Runtime/Mapper/AxROM.cs.meta b/Runtime/Mapper/AxROM.cs.meta new file mode 100644 index 0000000..67ab997 --- /dev/null +++ b/Runtime/Mapper/AxROM.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 632b907c0f30abb47b218545e40ff06d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/BaseMapper.cs b/Runtime/Mapper/BaseMapper.cs new file mode 100644 index 0000000..e8d0abf --- /dev/null +++ b/Runtime/Mapper/BaseMapper.cs @@ -0,0 +1,71 @@ +using System; +using System.IO; + +namespace Aya.UNes.Mapper +{ + [AttributeUsage(AttributeTargets.Class, Inherited = false)] + public class MapperDef : Attribute + { + public int Id; + public string Name; + public string Description; + + public MapperDef() + { + + } + + public MapperDef(int id) + { + Id = id; + } + } + + public abstract class BaseMapper + { + protected readonly Emulator _emulator; + protected readonly byte[] _prgROM; + protected readonly byte[] _prgRAM = new byte[0x2000]; + protected readonly byte[] _chrROM; + protected readonly uint _lastBankOffset; + + protected BaseMapper(Emulator emulator) + { + _emulator = emulator; + var cart = emulator.Cartridge; + _prgROM = cart.PRGROM; + _chrROM = cart.CHRROM; + _lastBankOffset = (uint) _prgROM.Length - 0x4000; + } + + public virtual void InitializeMemoryMap(CPU cpu) + { + + } + + public virtual void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[address]); + ppu.MapWriteHandler(0x0000, 0x1FFF, (address, val) => _chrROM[address] = val); + } + + public virtual void ProcessCycle(int scanLine, int cycle) + { + + } + + public virtual void Save(Stream os) + { + os.Write(_prgRAM, 0, _prgRAM.Length); + } + + public virtual void Load(Stream os) + { + using (var binaryReader = new BinaryReader(os)) + { + var ram = binaryReader.ReadBytes((int)os.Length); + Array.Copy(ram, _prgRAM, ram.Length); + } + } + } +} \ No newline at end of file diff --git a/Runtime/Mapper/BaseMapper.cs.meta b/Runtime/Mapper/BaseMapper.cs.meta new file mode 100644 index 0000000..6e44d0d --- /dev/null +++ b/Runtime/Mapper/BaseMapper.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a468573a1ef266542a055638aeef33cb +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/CNROM.cs b/Runtime/Mapper/CNROM.cs new file mode 100644 index 0000000..029eb97 --- /dev/null +++ b/Runtime/Mapper/CNROM.cs @@ -0,0 +1,32 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(3)] + public class CNROM : BaseMapper + { + protected int _bankOffset; + + public CNROM(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_bankOffset + address]); + } + + public override void InitializeMemoryMap(CPU cpu) + { + if (_prgROM.Length == 0x8000) + { + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[address - 0x8000]); + } + else + { + cpu.MapReadHandler(0x8000, 0xBFFF, address => _prgROM[address - 0x8000]); + cpu.MapReadHandler(0xC000, 0xFFFF, address => _prgROM[address - 0xC000]); + } + + cpu.MapWriteHandler(0x8000, 0xFFFF, (address, val) => _bankOffset = (val & 0x3) * 0x2000); + } + } +} diff --git a/Runtime/Mapper/CNROM.cs.meta b/Runtime/Mapper/CNROM.cs.meta new file mode 100644 index 0000000..15d040c --- /dev/null +++ b/Runtime/Mapper/CNROM.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fdaa43625af49e54aacf69d2ed92544f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/Camerica.cs b/Runtime/Mapper/Camerica.cs new file mode 100644 index 0000000..6a9ad0c --- /dev/null +++ b/Runtime/Mapper/Camerica.cs @@ -0,0 +1,27 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(71)] + public class Camerica : BaseMapper + { + protected int _prgBankOffset; + + public Camerica(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x8000, 0xBFFF, address => _prgROM[_prgBankOffset + (address - 0x8000)]); + cpu.MapReadHandler(0xC000, 0xFFFF, address => _prgROM[_prgROM.Length - 0x4000 + (address - 0xC000)]); + + // Actually starts at 0x8000, but use 0x9000 for compatibility w/o submapper + cpu.MapWriteHandler(0x9000, 0x9FFF, (address, val) => + { + // TODO: Fire Hawk mirroring + }); + + // The number of bits available vary: 4 for the BF9093, 3 for the BF9097, and 2 for the BF9096. + cpu.MapWriteHandler(0xC000, 0xFFFF, (address, val) => _prgBankOffset = (val & 0xF) * 0x4000 % _prgROM.Length); + } + } +} diff --git a/Runtime/Mapper/Camerica.cs.meta b/Runtime/Mapper/Camerica.cs.meta new file mode 100644 index 0000000..2d735de --- /dev/null +++ b/Runtime/Mapper/Camerica.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a879b12017bd97949a00317043b96f54 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/ColorDreams.cs b/Runtime/Mapper/ColorDreams.cs new file mode 100644 index 0000000..57f10f8 --- /dev/null +++ b/Runtime/Mapper/ColorDreams.cs @@ -0,0 +1,29 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(11)] + public class ColorDreams : BaseMapper + { + protected int _prgBankOffset; + protected int _chrBankOffset; + + public ColorDreams(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_chrBankOffset + address]); + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_prgBankOffset + (address - 0x8000)]); + + cpu.MapWriteHandler(0x8000, 0xFFFF, (address, val) => + { + _prgBankOffset = (val & 0x3) * 0x8000; + _chrBankOffset = (val >> 4) * 0x2000; + }); + } + } +} diff --git a/Runtime/Mapper/ColorDreams.cs.meta b/Runtime/Mapper/ColorDreams.cs.meta new file mode 100644 index 0000000..172a6aa --- /dev/null +++ b/Runtime/Mapper/ColorDreams.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 50a4c8ff195db574585c43ba079d8195 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/DxROM.cs b/Runtime/Mapper/DxROM.cs new file mode 100644 index 0000000..0d6d17d --- /dev/null +++ b/Runtime/Mapper/DxROM.cs @@ -0,0 +1,39 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(206)] + public class DxROM : MMC3 + { + public DxROM(Emulator emulator) : base(emulator) + { + _prgBankingMode = PRGBankingMode.SwitchFix; + _chrBankingMode = CHRBankingMode.TwoFour; + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_prgBankOffsets[(address - 0x8000) / 0x2000] + address % 0x2000]); + + cpu.MapWriteHandler(0x8000, 0x9FFF, (address, val) => + { + if ((address & 0x1) == 0) + { + _currentBank = val & 0x7u; + } + else + { + if (_currentBank <= 1) val &= 0x1F; + else if (_currentBank <= 5) val &= 0x3F; + else val &= 0xF; + + _banks[_currentBank] = val; + UpdateOffsets(); + } + }); + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_chrBankOffsets[address / 0x400] + address % 0x400]); + } + } +} diff --git a/Runtime/Mapper/DxROM.cs.meta b/Runtime/Mapper/DxROM.cs.meta new file mode 100644 index 0000000..dd2b1c3 --- /dev/null +++ b/Runtime/Mapper/DxROM.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b8ae98e170ab70143bc7284dc01b9627 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/GxROM.cs b/Runtime/Mapper/GxROM.cs new file mode 100644 index 0000000..c435678 --- /dev/null +++ b/Runtime/Mapper/GxROM.cs @@ -0,0 +1,29 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(66)] + public class GxROM : BaseMapper + { + protected int _prgBankOffset; + protected int _chrBankOffset; + + public GxROM(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_chrBankOffset + address]); + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_prgBankOffset + (address - 0x8000)]); + + cpu.MapWriteHandler(0x8000, 0xFFFF, (address, val) => + { + _prgBankOffset = ((val >> 4) & 0x3) * 0x8000; + _chrBankOffset = (val & 0x3) * 0x2000; + }); + } + } +} diff --git a/Runtime/Mapper/GxROM.cs.meta b/Runtime/Mapper/GxROM.cs.meta new file mode 100644 index 0000000..2810704 --- /dev/null +++ b/Runtime/Mapper/GxROM.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9a06e1a753215b64886b90ecde825cec +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/Jaleco.cs b/Runtime/Mapper/Jaleco.cs new file mode 100644 index 0000000..5ccbd2d --- /dev/null +++ b/Runtime/Mapper/Jaleco.cs @@ -0,0 +1,29 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(140)] + public class Jaleco : BaseMapper + { + protected int _prgBankOffset; + protected int _chrBankOffset; + + public Jaleco(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_chrBankOffset + address]); + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_prgBankOffset + (address - 0x8000)]); + + cpu.MapWriteHandler(0x6000, 0x7FFF, (address, val) => + { + _prgBankOffset = ((val >> 4) & 0x3) * 0x8000; + _chrBankOffset = (val & 0x3) * 0x2000; + }); + } + } +} diff --git a/Runtime/Mapper/Jaleco.cs.meta b/Runtime/Mapper/Jaleco.cs.meta new file mode 100644 index 0000000..836c62f --- /dev/null +++ b/Runtime/Mapper/Jaleco.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ac2f62d216cf70643919c921900c6e70 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/MMC1.cs b/Runtime/Mapper/MMC1.cs new file mode 100644 index 0000000..7360857 --- /dev/null +++ b/Runtime/Mapper/MMC1.cs @@ -0,0 +1,170 @@ +using static Aya.UNes.Cartridge.VRAMMirroringMode; + +namespace Aya.UNes.Mapper +{ + [MapperDef(1)] + public class MMC1 : BaseMapper + { + // TODO: are MMC1 and MMC1A even different chip types? + public enum ChipType { MMC1, MMC1A, MMC1B, MMC1C } + public enum CHRBankingMode { Single, Double } + public enum PRGBankingMode { Switch32Kb, Switch16KbFixFirst, Switch16KbFixLast } + + private readonly Cartridge.VRAMMirroringMode[] _mirroringModes = { Lower, Upper, Vertical, Horizontal }; + + private readonly ChipType _type; + private CHRBankingMode _chrBankingMode; + private PRGBankingMode _prgBankingMode; + + private uint _serialData; + private int _serialPos; + + private uint _control; + + private readonly uint[] _chrBankOffsets = new uint[2]; + private readonly uint[] _chrBanks = new uint[2]; + + private readonly uint[] _prgBankOffsets = new uint[2]; + private uint _prgBank; + + private bool _prgRAMEnabled; + + private uint? _lastWritePC; + + public MMC1(Emulator emulator) : this(emulator, ChipType.MMC1B) + { + + } + + public MMC1(Emulator emulator, ChipType chipType) : base(emulator) + { + _type = chipType; + if (chipType == ChipType.MMC1B) _prgRAMEnabled = true; + UpdateControl(0x0F); + _emulator.Cartridge.MirroringMode = Horizontal; + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x6000, 0x7FFF, address => _prgRAM[address - 0x6000]); + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_prgBankOffsets[(address - 0x8000) / 0x4000] + address % 0x4000]); + + cpu.MapWriteHandler(0x6000, 0x7FFF, (address, val) => + { + // PRG RAM is always enabled on MMC1A + if (_type == ChipType.MMC1A || _prgRAMEnabled) + _prgRAM[address - 0x6000] = val; + }); + + cpu.MapWriteHandler(0x8000, 0xFFFF, (address, val) => + { + // Explicitly ignore the second write happening on consecutive cycles + // of an RMW instruction + var cycle = _emulator.CPU.PC; + if (cycle == _lastWritePC) + return; + _lastWritePC = cycle; + + if ((val & 0x80) > 0) + { + _serialData = 0; + _serialPos = 0; + UpdateControl(_control | 0x0C); + } + else + { + _serialData |= (uint)((val & 0x1) << _serialPos); + _serialPos++; + + if (_serialPos == 5) + { + // Address is incompletely decoded + address &= 0x6000; + if (address == 0x0000) + UpdateControl(_serialData); + else if (address == 0x2000) + UpdateCHRBank(0, _serialData); + else if (address == 0x4000) + UpdateCHRBank(1, _serialData); + else if (address == 0x6000) + UpdatePRGBank(_serialData); + + _serialData = 0; + _serialPos = 0; + } + } + }); + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_chrBankOffsets[address / 0x1000] + address % 0x1000]); + ppu.MapWriteHandler(0x0000, 0x1FFF, (address, val) => _chrROM[_chrBankOffsets[address / 0x1000] + address % 0x1000] = val); + } + + private void UpdateControl(uint value) + { + _control = value; + + _emulator.Cartridge.MirroringMode = _mirroringModes[value & 0x3]; + + _chrBankingMode = (CHRBankingMode)((value >> 4) & 0x1); + + var prgMode = (value >> 2) & 0x3; + // Both 0 and 1 are 32Kb switch + if (prgMode == 0) prgMode = 1; + _prgBankingMode = (PRGBankingMode)(prgMode - 1); + + UpdateCHRBank(1, _chrBanks[1]); + UpdateCHRBank(0, _chrBanks[0]); + UpdatePRGBank(_prgBank); + } + + private void UpdatePRGBank(uint value) + { + _prgBank = value; + + _prgRAMEnabled = (value & 0x10) == 0; + value &= 0xF; + + switch (_prgBankingMode) + { + case PRGBankingMode.Switch32Kb: + value >>= 1; + value *= 0x4000; + _prgBankOffsets[0] = value; + _prgBankOffsets[1] = value + 0x4000; + break; + case PRGBankingMode.Switch16KbFixFirst: + _prgBankOffsets[0] = 0; + _prgBankOffsets[1] = value * 0x4000; + break; + case PRGBankingMode.Switch16KbFixLast: + _prgBankOffsets[0] = value * 0x4000; + _prgBankOffsets[1] = _lastBankOffset; + break; + } + } + + private void UpdateCHRBank(uint bank, uint value) + { + _chrBanks[bank] = value; + + // TODO FIXME: I feel like this branch should only be taken + // when bank == 0, but this breaks Final Fantasy + // When can banking mode change without UpdateCHRBank being called? + if (_chrBankingMode == CHRBankingMode.Single) + { + value = _chrBanks[0]; + value >>= 1; + value *= 0x1000; + _chrBankOffsets[0] = value; + _chrBankOffsets[1] = value + 0x1000; + } + else + { + _chrBankOffsets[bank] = value * 0x1000; + } + } + } +} diff --git a/Runtime/Mapper/MMC1.cs.meta b/Runtime/Mapper/MMC1.cs.meta new file mode 100644 index 0000000..4c02868 --- /dev/null +++ b/Runtime/Mapper/MMC1.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d3ab06f7819df254491a7afa6a82bb43 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/MMC2.cs b/Runtime/Mapper/MMC2.cs new file mode 100644 index 0000000..db617a0 --- /dev/null +++ b/Runtime/Mapper/MMC2.cs @@ -0,0 +1,33 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(Id = 9, Description = "Mike Tyson's Punch-Out!!")] + public class MMC2 : MMC4 + { + public MMC2(Emulator emulator) : base(emulator) + { + + } + + public override void InitializeMemoryMap(CPU cpu) + { + base.InitializeMemoryMap(cpu); + + cpu.MapReadHandler(0x8000, 0xBFFF, address => _prgROM[_prgBankOffset + (address - 0x8000)]); + cpu.MapReadHandler(0xA000, 0xFFFF, address => _prgROM[_prgROM.Length - 0x4000 - 0x2000 + (address - 0xA000)]); + + cpu.MapWriteHandler(0xA000, 0xAFFF, (address, val) => _prgBankOffset = (val & 0xF) * 0x2000); + } + + protected override void GetLatch(uint address, out uint latch, out bool? on) + { + base.GetLatch(address, out latch, out on); + + // For MMC2, only 0xFD8 and 0xFE8 trigger the latch, + // not the whole range like in MMC4 + if (latch == 0 && (address & 0x3) != 0) + { + on = null; + } + } + } +} diff --git a/Runtime/Mapper/MMC2.cs.meta b/Runtime/Mapper/MMC2.cs.meta new file mode 100644 index 0000000..3270f4a --- /dev/null +++ b/Runtime/Mapper/MMC2.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 522a51dcf1b92d2498013b4d6f9accb0 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/MMC3.cs b/Runtime/Mapper/MMC3.cs new file mode 100644 index 0000000..9e61e97 --- /dev/null +++ b/Runtime/Mapper/MMC3.cs @@ -0,0 +1,182 @@ +using static Aya.UNes.Cartridge.VRAMMirroringMode; + +namespace Aya.UNes.Mapper +{ + [MapperDef(4)] + public class MMC3 : BaseMapper + { + // Different PRG RAM write/enable controls + public enum ChipType { MMC3, MMC6 } + public enum CHRBankingMode { TwoFour, FourTwo } + public enum PRGBankingMode { SwitchFix, FixSwitch } + + private readonly Cartridge.VRAMMirroringMode[] _mirroringModes = { Vertical, Horizontal }; + + private readonly ChipType _type; + protected CHRBankingMode _chrBankingMode; + protected PRGBankingMode _prgBankingMode; + + + protected readonly uint[] _chrBankOffsets = new uint[8]; + protected uint[] _prgBankOffsets; + protected readonly uint[] _banks = new uint[8]; + protected uint _currentBank; + + private uint _irqReloadValue; + private uint _irqCounter; + protected bool _irqEnabled; + + private bool _prgRAMEnabled; + + public MMC3(Emulator emulator) : this(emulator, ChipType.MMC3) + { + + } + + public MMC3(Emulator emulator, ChipType chipType) : base(emulator) + { + _type = chipType; + _prgBankOffsets = new uint[] { 0, 0x2000, _lastBankOffset, _lastBankOffset + 0x2000 }; + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x6000, 0x7FFF, address => _prgRAM[address - 0x6000]); + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_prgBankOffsets[(address - 0x8000) / 0x2000] + address % 0x2000]); + + cpu.MapWriteHandler(0x6000, 0xFFFF, WriteByte); + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_chrBankOffsets[address / 0x400] + address % 0x400]); + ppu.MapWriteHandler(0x0000, 0x1FFF, (address, val) => _chrROM[_chrBankOffsets[address / 0x400] + address % 0x400] = val); + } + + public override void ProcessCycle(int scanLine, int cycle) + { + if (_emulator.PPU.F.RenderingEnabled && cycle == 260 && (0 <= scanLine && scanLine < 240 || scanLine == -1)) + { + if (_irqCounter == 0) + { + _irqCounter = _irqReloadValue; + } + else + { + _irqCounter--; + if (_irqEnabled && _irqCounter == 0) + { + _emulator.CPU.TriggerInterrupt(CPU.InterruptType.IRQ); + } + } + } + } + + protected void WriteByte(uint addr, byte value) + { + bool even = (addr & 0x1) == 0; + + if (addr < 0x8000) + { + if (_prgRAMEnabled) + { + _prgRAM[addr - 0x6000] = value; + } + } + else if (addr < 0xA000) + { + if (even) + { + _currentBank = value & 0x7u; + _prgBankingMode = (PRGBankingMode)((value >> 6) & 0x1); + _chrBankingMode = (CHRBankingMode)((value >> 7) & 0x1); + } + else + { + _banks[_currentBank] = value; + } + + UpdateOffsets(); + } + else if (addr < 0xC000) + { + if (even) + { + _emulator.Cartridge.MirroringMode = _mirroringModes[value & 0x1]; + } + else + { + _prgRAMEnabled = (value & 0xC0) == 0x80; + } + } + else if (addr < 0xE000) + { + if (even) + { + _irqReloadValue = value; + } + else + { + _irqCounter = 0; + } + } + else + { + _irqEnabled = !even; + } + } + + protected void UpdateOffsets() + { + switch (_prgBankingMode) + { + case PRGBankingMode.SwitchFix: + _prgBankOffsets[0] = _banks[6] * 0x2000; + _prgBankOffsets[1] = _banks[7] * 0x2000; + _prgBankOffsets[2] = _lastBankOffset; + _prgBankOffsets[3] = _lastBankOffset + 0x2000; + break; + case PRGBankingMode.FixSwitch: + _prgBankOffsets[0] = _lastBankOffset; + _prgBankOffsets[1] = _banks[7] * 0x2000; + _prgBankOffsets[2] = _banks[6] * 0x2000; + _prgBankOffsets[3] = _lastBankOffset + 0x2000; + break; + } + + switch (_chrBankingMode) + { + case CHRBankingMode.TwoFour: + _chrBankOffsets[0] = _banks[0] & 0xFE; + _chrBankOffsets[1] = _banks[0] | 0x01; + _chrBankOffsets[2] = _banks[1] & 0xFE; + _chrBankOffsets[3] = _banks[1] | 0x01; + _chrBankOffsets[4] = _banks[2]; + _chrBankOffsets[5] = _banks[3]; + _chrBankOffsets[6] = _banks[4]; + _chrBankOffsets[7] = _banks[5]; + break; + case CHRBankingMode.FourTwo: + _chrBankOffsets[0] = _banks[2]; + _chrBankOffsets[1] = _banks[3]; + _chrBankOffsets[2] = _banks[4]; + _chrBankOffsets[3] = _banks[5]; + _chrBankOffsets[4] = _banks[0] & 0xFE; + _chrBankOffsets[5] = _banks[0] | 0x01; + _chrBankOffsets[6] = _banks[1] & 0xFE; + _chrBankOffsets[7] = _banks[1] | 0x01; + break; + } + + for (var i = 0; i < _prgBankOffsets.Length; i++) + { + _prgBankOffsets[i] %= (uint)_prgROM.Length; + } + + for (var i = 0; i < _chrBankOffsets.Length; i++) + { + _chrBankOffsets[i] = (uint) (_chrBankOffsets[i] * 0x400 % _chrROM.Length); + } + } + } +} diff --git a/Runtime/Mapper/MMC3.cs.meta b/Runtime/Mapper/MMC3.cs.meta new file mode 100644 index 0000000..0adfec7 --- /dev/null +++ b/Runtime/Mapper/MMC3.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 82fb09fbeba58ae458f58e52898fedb1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/MMC4.cs b/Runtime/Mapper/MMC4.cs new file mode 100644 index 0000000..8c54e4b --- /dev/null +++ b/Runtime/Mapper/MMC4.cs @@ -0,0 +1,73 @@ +using static Aya.UNes.Cartridge.VRAMMirroringMode; + +namespace Aya.UNes.Mapper +{ + [MapperDef(10)] + public class MMC4 : BaseMapper + { + protected readonly Cartridge.VRAMMirroringMode[] _mirroringModes = { Vertical, Horizontal }; + + protected int _prgBankOffset; + protected int[,] _chrBankOffsets = new int[2, 2]; + protected bool[] _latches = new bool[2]; + + public MMC4(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x6000, 0x7FFF, address => _prgRAM[address - 0x6000]); + cpu.MapReadHandler(0x8000, 0xBFFF, address => _prgROM[_prgBankOffset + (address - 0x8000)]); + cpu.MapReadHandler(0xC000, 0xFFFF, address => _prgROM[_prgROM.Length - 0x4000 + (address - 0xC000)]); + + cpu.MapWriteHandler(0x6000, 0x7FFF, (address, val) => _prgRAM[address - 0x6000] = val); + cpu.MapWriteHandler(0xA000, 0xAFFF, (address, val) => _prgBankOffset = (val & 0xF) * 0x4000); + cpu.MapWriteHandler(0xB000, 0xEFFF, (address, val) => + { + var bank = (address - 0xB000) / 0x2000; + var latch = ((address & 0x1FFF) == 0).AsByte(); + _chrBankOffsets[bank, latch] = (val & 0x1F) * 0x1000; + }); + + cpu.MapWriteHandler(0xF000, 0xFFFF, (address, val) => _emulator.Cartridge.MirroringMode = _mirroringModes[val & 0x1]); + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => + { + var bank = address / 0x1000; + var ret = _chrROM[_chrBankOffsets[bank, _latches[bank].AsByte()] + address % 0x1000]; + if ((address & 0x08) > 0) + { + GetLatch(address, out uint latch, out bool? on); + + if (on != null) + { + _latches[latch] = (bool)on; + } + } + + return ret; + }); + } + + protected virtual void GetLatch(uint address, out uint latch, out bool? on) + { + latch = (address >> 12) & 0x1; + on = null; + + address = (address >> 4) & 0xFF; + + if (address == 0xFE) + { + on = true; + } + else if (address == 0xFD) + { + on = false; + } + } + } +} diff --git a/Runtime/Mapper/MMC4.cs.meta b/Runtime/Mapper/MMC4.cs.meta new file mode 100644 index 0000000..81f2dfe --- /dev/null +++ b/Runtime/Mapper/MMC4.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 54d8f341641616d45b20c65ec4a75ff2 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/Mapper094.cs b/Runtime/Mapper/Mapper094.cs new file mode 100644 index 0000000..79c6753 --- /dev/null +++ b/Runtime/Mapper/Mapper094.cs @@ -0,0 +1,17 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(Id = 94, Description = "Senjou no Ookami")] + public class Mapper094 : UxROM + { + public Mapper094(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(CPU cpu) + { + base.InitializeMemoryMap(cpu); + + cpu.MapWriteHandler(0x8000, 0xFFFF, (address, val) => _bankOffset = (val & 0x1C) << 12); + } + } +} diff --git a/Runtime/Mapper/Mapper094.cs.meta b/Runtime/Mapper/Mapper094.cs.meta new file mode 100644 index 0000000..70a9d00 --- /dev/null +++ b/Runtime/Mapper/Mapper094.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0abeb567769f715418ae9125bd029206 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/Mapper155.cs b/Runtime/Mapper/Mapper155.cs new file mode 100644 index 0000000..899f8f2 --- /dev/null +++ b/Runtime/Mapper/Mapper155.cs @@ -0,0 +1,11 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(Id = 155, Description = "MMC1A")] + public class Mapper155 : MMC1 + { + // Mapper for games requiring MMC1A + public Mapper155(Emulator emulator) : base(emulator, ChipType.MMC1A) + { + } + } +} diff --git a/Runtime/Mapper/Mapper155.cs.meta b/Runtime/Mapper/Mapper155.cs.meta new file mode 100644 index 0000000..a775951 --- /dev/null +++ b/Runtime/Mapper/Mapper155.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: dc90c4e04b12a134c884ca3431375aac +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/Mapper180.cs b/Runtime/Mapper/Mapper180.cs new file mode 100644 index 0000000..3e10aaa --- /dev/null +++ b/Runtime/Mapper/Mapper180.cs @@ -0,0 +1,21 @@ +namespace Aya.UNes.Mapper +{ + // Mapper used strictly for Crazy Climber; logic is slighly different + [MapperDef(Id = 180, Description = "Crazy Climber")] + public class Mapper180 : UxROM + { + public Mapper180(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(CPU cpu) + { + base.InitializeMemoryMap(cpu); + + // $8000-$C000 is fixed to *first* bank + cpu.MapReadHandler(0x8000, 0xBFFF, address => _prgROM[address - 0x8000]); + // $C000-$FFFF is switchable, controlled the same as UxROM + cpu.MapReadHandler(0xC000, 0xFFFF, address => _prgROM[_bankOffset + (address - 0xC000)]); + } + } +} diff --git a/Runtime/Mapper/Mapper180.cs.meta b/Runtime/Mapper/Mapper180.cs.meta new file mode 100644 index 0000000..41282e8 --- /dev/null +++ b/Runtime/Mapper/Mapper180.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0ed79f8d409c2864bab9bc85848bc36a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/NROM.cs b/Runtime/Mapper/NROM.cs new file mode 100644 index 0000000..e254cc5 --- /dev/null +++ b/Runtime/Mapper/NROM.cs @@ -0,0 +1,23 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(0)] + public class NROM : BaseMapper + { + private readonly byte[] _addressSpace = new byte[0x2000 + 0x8000]; // Space for $2000 VRAM + $8000 PRG + + public NROM(Emulator emulator) : base(emulator) + { + for (var i = 0; i < 0x8000; i++) + { + var offset = _emulator.Cartridge.PRGROMSize == 0x4000 ? i & 0xBFFF : i; + _addressSpace[0x2000 + i] = _prgROM[offset]; + } + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x6000, 0xFFFF, address => _addressSpace[address - 0x6000]); + cpu.MapWriteHandler(0x6000, 0x7FFF, (address, val) => _addressSpace[address - 0x6000] = val); + } + } +} diff --git a/Runtime/Mapper/NROM.cs.meta b/Runtime/Mapper/NROM.cs.meta new file mode 100644 index 0000000..f09b0e2 --- /dev/null +++ b/Runtime/Mapper/NROM.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 7781217265cd36043acd57aa3a049d25 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/Nina003006.cs b/Runtime/Mapper/Nina003006.cs new file mode 100644 index 0000000..9cc9572 --- /dev/null +++ b/Runtime/Mapper/Nina003006.cs @@ -0,0 +1,32 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(79)] + public class Nina003006 : BaseMapper + { + protected int _prgBankOffset; + protected int _chrBankOffset; + + public Nina003006(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(PPU ppu) + { + ppu.MapReadHandler(0x0000, 0x1FFF, address => _chrROM[_chrBankOffset + address]); + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x8000, 0xFFFF, address => _prgROM[_prgBankOffset + (address - 0x8000)]); + + cpu.MapWriteHandler(0x4000, 0x5FFF, (address, val) => + { + if ((address & 0b1110_0001_0000_0000) == 0b0100_0001_0000_0000) + { + _prgBankOffset = ((val >> 4) & 0x3) * 0x8000; + _chrBankOffset = (val & 0x3) * 0x2000; + } + }); + } + } +} \ No newline at end of file diff --git a/Runtime/Mapper/Nina003006.cs.meta b/Runtime/Mapper/Nina003006.cs.meta new file mode 100644 index 0000000..128d6ad --- /dev/null +++ b/Runtime/Mapper/Nina003006.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4a935d078b54a6a4b9434e3478863251 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Mapper/UxROM.cs b/Runtime/Mapper/UxROM.cs new file mode 100644 index 0000000..1bb1a0d --- /dev/null +++ b/Runtime/Mapper/UxROM.cs @@ -0,0 +1,22 @@ +namespace Aya.UNes.Mapper +{ + [MapperDef(2)] + public class UxROM : BaseMapper + { + protected int _bankOffset; + + public UxROM(Emulator emulator) : base(emulator) + { + } + + public override void InitializeMemoryMap(CPU cpu) + { + cpu.MapReadHandler(0x6000, 0x7FFF, address => _prgRAM[address - 0x6000]); + cpu.MapReadHandler(0x8000, 0xBFFF, address => _prgROM[_bankOffset + (address - 0x8000)]); + cpu.MapReadHandler(0xC000, 0xFFFF, address => _prgROM[_prgROM.Length - 0x4000 + (address - 0xC000)]); + + cpu.MapWriteHandler(0x6000, 0x7FFF, (address, val) => _prgRAM[address - 0x6000] = val); + cpu.MapWriteHandler(0x8000, 0xFFFF, (address, val) => _bankOffset = (val & 0xF) * 0x4000); + } + } +} diff --git a/Runtime/Mapper/UxROM.cs.meta b/Runtime/Mapper/UxROM.cs.meta new file mode 100644 index 0000000..9b0f9dc --- /dev/null +++ b/Runtime/Mapper/UxROM.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9fa6ddb08aec2014fada61a92af067ad +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/PPU.Core.cs b/Runtime/PPU.Core.cs new file mode 100644 index 0000000..8af48f0 --- /dev/null +++ b/Runtime/PPU.Core.cs @@ -0,0 +1,380 @@ +using System; + +namespace Aya.UNes +{ + partial class PPU + { + private const int GameWidth = 256, GameHeight = 240; + + private uint _bufferPos; + public readonly uint[] RawBitmap = new uint[GameWidth * GameHeight]; + private readonly uint[] _priority = new uint[GameWidth * GameHeight]; + + // TODO: use real chroma/luma decoding + private readonly uint[] _palette = { + 0x7C7C7C, 0x0000FC, 0x0000BC, 0x4428BC, 0x940084, 0xA80020, 0xA81000, 0x881400, + 0x503000, 0x007800, 0x006800, 0x005800, 0x004058, 0x000000, 0x000000, 0x000000, + 0xBCBCBC, 0x0078F8, 0x0058F8, 0x6844FC, 0xD800CC, 0xE40058, 0xF83800, 0xE45C10, + 0xAC7C00, 0x00B800, 0x00A800, 0x00A844, 0x008888, 0x000000, 0x000000, 0x000000, + 0xF8F8F8, 0x3CBCFC, 0x6888FC, 0x9878F8, 0xF878F8, 0xF85898, 0xF87858, 0xFCA044, + 0xF8B800, 0xB8F818, 0x58D854, 0x58F898, 0x00E8D8, 0x787878, 0x000000, 0x000000, + 0xFCFCFC, 0xA4E4FC, 0xB8B8F8, 0xD8B8F8, 0xF8B8F8, 0xF8A4C0, 0xF0D0B0, 0xFCE0A8, + 0xF8D878, 0xD8F878, 0xB8F8B8, 0xB8F8D8, 0x00FCFC, 0xF8D8F8, 0x000000, 0x000000 + }; + + private int _scanLineCount = 261; + private int _cyclesPerLine = 341; + private int _cpuSyncCounter; + private readonly uint[] _scanLineOAM = new uint[8 * 4]; + private readonly bool[] _isSprite0 = new bool[8]; + private int _spriteCount; + + private long _tileShiftRegister; + private uint _currentNameTableByte; + private uint _currentHighTile, _currentLowTile; + private uint _currentColor; + + public void ProcessPixel(int x, int y) + { + ProcessBackgroundForPixel(x, y); + if (F.DrawSprites) + { + ProcessSpritesForPixel(x, y); + } + + if (y != -1) + { + _bufferPos++; + } + } + + private void CountSpritesOnLine(int scanLine) + { + _spriteCount = 0; + var height = F.TallSpritesEnabled ? 16 : 8; + + for (var idx = 0; idx < _oam.Length; idx += 4) + { + var y = _oam[idx] + 1; + if (scanLine >= y && scanLine < y + height) + { + _isSprite0[_spriteCount] = idx == 0; + _scanLineOAM[_spriteCount * 4 + 0] = _oam[idx + 0]; + _scanLineOAM[_spriteCount * 4 + 1] = _oam[idx + 1]; + _scanLineOAM[_spriteCount * 4 + 2] = _oam[idx + 2]; + _scanLineOAM[_spriteCount * 4 + 3] = _oam[idx + 3]; + _spriteCount++; + } + + if (_spriteCount == 8) + { + break; + } + } + } + + private void NextNameTableByte() + { + _currentNameTableByte = ReadByte(0x2000 | (V & 0x0FFF)); + } + + private void NextTileByte(bool hi) + { + var tileIdx = _currentNameTableByte * 16; + var address = F.PatternTableAddress + tileIdx + FineY; + + if (hi) + { + _currentHighTile = ReadByte(address + 8); + } + else + { + _currentLowTile = ReadByte(address); + } + } + + private void NextAttributeByte() + { + // Bless nesdev + var address = 0x23C0 | (V & 0x0C00) | ((V >> 4) & 0x38) | ((V >> 2) & 0x07); + _currentColor = (ReadByte(address) >> (int)((CoarseX & 2) | ((CoarseY & 2) << 1))) & 0x3; + } + + private void ShiftTileRegister() + { + for (var x = 0; x < 8; x++) + { + uint palette = ((_currentHighTile & 0x80) >> 6) | ((_currentLowTile & 0x80) >> 7); + _tileShiftRegister |= (palette + _currentColor * 4) << ((7 - x) * 4); + _currentLowTile <<= 1; + _currentHighTile <<= 1; + } + } + + private void ProcessBackgroundForPixel(int cycle, int scanLine) + { + if (cycle < 8 && !F.DrawLeftBackground || !F.DrawBackground && scanLine != -1) + { + // Maximally sketchy: if current address is in the PPU palette, then it draws that palette entry if rendering is disabled + // Otherwise, it draws $3F00 (universal bg color) + // https://www.romhacking.net/forum/index.php?topic=20554.0 + // Don't know if any game actually uses it, but a test ROM I wrote unexpectedly showed this + // corner case + RawBitmap[_bufferPos] = _palette[ReadByte(0x3F00 + ((F.BusAddress & 0x3F00) == 0x3F00 ? F.BusAddress & 0x001F : 0)) & 0x3F]; + return; + } + + var paletteEntry = (uint)(_tileShiftRegister >> 32 >> (int)((7 - X) * 4)) & 0x0F; + if (paletteEntry % 4 == 0) paletteEntry = 0; + + if (scanLine != -1) + { + _priority[_bufferPos] = paletteEntry; + RawBitmap[_bufferPos] = _palette[ReadByte(0x3F00u + paletteEntry) & 0x3F]; + } + } + + private void ProcessSpritesForPixel(int x, int scanLine) + { + for (var idx = _spriteCount * 4 - 4; idx >= 0; idx -= 4) + { + var spriteX = _scanLineOAM[idx + 3]; + var spriteY = _scanLineOAM[idx] + 1; + + // Don't draw this sprite if... + if (spriteY == 0 || // it's located at y = 0 + spriteY > 239 || // it's located past y = 239 ($EF) + x >= spriteX + 8 || // it's behind the current dot + x < spriteX || // it's ahead of the current dot + x < 8 && !F.DrawLeftSprites) // it's in the clip area, and clipping is enabled + { + continue; + } + + // amusingly enough, the PPU's palette handling is basically identical + // to that of the Gameboy / Gameboy Color, so I've sort of just copy/pasted + // handling code wholesale from my GBC emulator at + // https://github.com/Xyene/Nitrous-Emulator/blob/master/src/main/java/nitrous/lcd/LCD.java#L642 + var tileIdx = _scanLineOAM[idx + 1]; + if (F.TallSpritesEnabled) + { + tileIdx &= ~0x1u; + } + + tileIdx *= 16; + + var attribute = _scanLineOAM[idx + 2] & 0xE3; + + var palette = attribute & 0x3; + var front = (attribute & 0x20) == 0; + var flipX = (attribute & 0x40) > 0; + var flipY = (attribute & 0x80) > 0; + + var px = (int) (x - spriteX); + var line = (int) (scanLine - spriteY); + + var tableBase = F.TallSpritesEnabled ? (_scanLineOAM[idx + 1] & 1) * 0x1000 : F.SpriteTableAddress; + + if (F.TallSpritesEnabled) + { + if (line >= 8) + { + line -= 8; + if (!flipY) + { + tileIdx += 16; + } + + flipY = false; + } + + if (flipY) + { + tileIdx += 16; + } + } + + // here we handle the x and y flipping by tweaking the indices we are accessing + var logicalX = flipX ? 7 - px : px; + var logicalLine = flipY ? 7 - line : line; + + var address = (uint) (tableBase + tileIdx + logicalLine); + + // this looks bad, but it's about as readable as it's going to get + var color = (uint) ( + ( + ( + ( + // fetch upper bit from 2nd bit plane + ReadByte(address + 8) & (0x80 >> logicalX) + ) >> (7 - logicalX) + ) << 1 // this is the upper bit of the color number + ) | + ( + ( + ReadByte(address) & (0x80 >> logicalX) + ) >> (7 - logicalX) + )); // << 0, this is the lower bit of the color number + + if (color > 0) + { + var backgroundPixel = _priority[_bufferPos]; + // Sprite 0 hits... + if (!(!_isSprite0[idx / 4] || // do not occur on not-0 sprite + x < 8 && !F.DrawLeftSprites || // or if left clipping is enabled + backgroundPixel == 0 || // or if bg pixel is transparent + F.Sprite0Hit || // or if it fired this frame already + x == 255)) // or if x is 255, "for an obscure reason related to the pixel pipeline" + { + F.Sprite0Hit = true; + } + + if (F.DrawBackground && (front || backgroundPixel == 0)) + { + if (scanLine != -1) + { + RawBitmap[_bufferPos] = _palette[ReadByte(0x3F10 + palette * 4 + color) & 0x3F]; + } + } + } + } + } + + public void ProcessFrame() + { + RawBitmap.Fill(0u); + _priority.Fill(0u); + _bufferPos = 0; + + for (var i = -1; i < _scanLineCount; i++) + { + ProcessScanLine(i); + } + } + + public void ProcessScanLine(int line) + { + for (var i = 0; i < _cyclesPerLine; i++) + { + ProcessCycle(line, i); + } + } + + private int _cpuClocksSinceVBL; + private int _ppuClocksSinceVBL; + + public void ProcessCycle(int scanLine, int cycle) + { + var visibleCycle = 1 <= cycle && cycle <= 256; + var prefetchCycle = 321 <= cycle && cycle <= 336; + var fetchCycle = visibleCycle || prefetchCycle; + + if (F.VBlankStarted) + { + _ppuClocksSinceVBL++; + } + + if (0 <= scanLine && scanLine < 240 || scanLine == -1) + { + if (visibleCycle) + { + ProcessPixel(cycle - 1, scanLine); + } + + // During pixels 280 through 304 of this scanline, the vertical scroll bits are reloaded TODO: if rendering is enabled. + if (scanLine == -1 && 280 <= cycle && cycle <= 304) + { + ReloadScrollY(); + } + + if (fetchCycle) + { + _tileShiftRegister <<= 4; + + // See https://wiki.nesdev.com/w/images/d/d1/Ntsc_timing.png + // Takes 8 cycles for tile to be read, 2 per "step" + switch (cycle & 7) + { + case 1: // NT + NextNameTableByte(); + break; + case 3: // AT + NextAttributeByte(); + break; + case 5: // Tile low + NextTileByte(false); + break; + case 7: // Tile high + NextTileByte(true); + break; + case 0: // 2nd cycle of tile high fetch + if (cycle == 256) + IncrementScrollY(); + else + IncrementScrollX(); + // Begin rendering a brand new tile + ShiftTileRegister(); + break; + } + } + + if (cycle == 257) + { + ReloadScrollX(); + // 257 - 320 + // The tile data for the sprites on the next scanline are fetched here. + // TODO: stagger this over all the cycles as opposed to only on 257 + CountSpritesOnLine(scanLine + 1); + } + } + + // TODO: this is a hack; VBlank should be cleared on dot 1 of the pre-render line, + // but for some reason we're at 2272-2273 CPU clocks at that time + // (i.e., our PPU timing is off somewhere by 6-9 PPU cycles per frame) + if (F.VBlankStarted && _cpuClocksSinceVBL == 2270) + { + F.VBlankStarted = false; + _cpuClocksSinceVBL = 0; + } + + if (cycle == 1) + { + if (scanLine == 241) + { + F.VBlankStarted = true; + if (F.NMIEnabled) + { + _emulator.CPU.TriggerInterrupt(CPU.InterruptType.NMI); + } + } + + // Happens at the same time as 1st cycle of NT byte fetch + if (scanLine == -1) + { + // Console.WriteLine(_ppuClocksSinceVBL); + _ppuClocksSinceVBL = 0; + F.VBlankStarted = false; + F.Sprite0Hit = false; + F.SpriteOverflow = false; + } + } + + _emulator.Mapper.ProcessCycle(scanLine, cycle); + + if (_cpuSyncCounter + 1 == 3) + { + if (F.VBlankStarted) + { + _cpuClocksSinceVBL++; + } + + _emulator.CPU.TickFromPPU(); + _cpuSyncCounter = 0; + } + else + { + _cpuSyncCounter++; + } + } + } +} diff --git a/Runtime/PPU.Core.cs.meta b/Runtime/PPU.Core.cs.meta new file mode 100644 index 0000000..db54fa8 --- /dev/null +++ b/Runtime/PPU.Core.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c6d671ac51126cd43981be89a0309422 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/PPU.Memory.cs b/Runtime/PPU.Memory.cs new file mode 100644 index 0000000..fb14874 --- /dev/null +++ b/Runtime/PPU.Memory.cs @@ -0,0 +1,130 @@ +using System; +using System.Runtime.CompilerServices; + +namespace Aya.UNes +{ + partial class PPU + { + private readonly byte[] _oam = new byte[0x100]; + private readonly byte[] _vRam = new byte[0x2000]; + private readonly byte[] _paletteRAM = new byte[0x20]; + + private static readonly uint[][] _vRamMirrorLookup = + { + new uint[]{0, 0, 1, 1}, // H + new uint[]{0, 1, 0, 1}, // V + new uint[]{0, 1, 2, 3}, // All + new uint[]{0, 0, 0, 0}, // Upper + new uint[]{1, 1, 1, 1}, // Lower + }; + + private int _lastWrittenRegister; + + public void WriteRegister(uint reg, byte val) + { + reg &= 0xF; + _lastWrittenRegister = val & 0xFF; + switch (reg) + { + case 0x0000: + PPUCTRL = val; + return; + case 0x0001: + PPUMASK = val; + return; + case 0x0002: return; + case 0x0003: + OAMADDR = val; + return; + case 0x0004: + OAMDATA = val; + return; + case 0x005: + PPUSCROLL = val; + return; + case 0x0006: + PPUADDR = val; + return; + case 0x0007: + PPUDATA = val; + return; + } + + throw new NotImplementedException($"{reg:X4} = {val:X2}"); + } + + public byte ReadRegister(uint reg) + { + reg &= 0xF; + switch (reg) + { + case 0x0000: return (byte)_lastWrittenRegister; + case 0x0001: return (byte)_lastWrittenRegister; + case 0x0002: + return (byte)PPUSTATUS; + case 0x0003: + return (byte)OAMADDR; + case 0x0004: + return (byte)OAMDATA; + case 0x0005: return (byte)_lastWrittenRegister; + case 0x0006: return (byte)_lastWrittenRegister; + case 0x0007: + return (byte)PPUDATA; + } + throw new NotImplementedException(reg.ToString("X2")); + } + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public uint GetVRamMirror(long address) + { + long entry; + var table = Math.DivRem(address - 0x2000, 0x400, out entry); + return _vRamMirrorLookup[(int)_emulator.Cartridge.MirroringMode][table] * 0x400 + (uint)entry; + } + + protected override void InitializeMemoryMap() + { + base.InitializeMemoryMap(); + + MapReadHandler(0x2000, 0x2FFF, address => _vRam[GetVRamMirror(address)]); + MapReadHandler(0x3000, 0x3EFF, address => _vRam[GetVRamMirror(address - 0x1000)]); + MapReadHandler(0x3F00, 0x3FFF, address => + { + if (address == 0x3F10 || address == 0x3F14 || address == 0x3F18 || address == 0x3F0C) + { + address -= 0x10; + } + + return _paletteRAM[(address - 0x3F00) & 0x1F]; + }); + + MapWriteHandler(0x2000, 0x2FFF, (address, val) => _vRam[GetVRamMirror(address)] = val); + MapWriteHandler(0x3000, 0x3EFF, (address, val) => _vRam[GetVRamMirror(address - 0x1000)] = val); + MapWriteHandler(0x3F00, 0x3FFF, (address, val) => + { + if (address == 0x3F10 || address == 0x3F14 || address == 0x3F18 || address == 0x3F0C) + { + address -= 0x10; + } + + _paletteRAM[(address - 0x3F00) & 0x1F] = val; + }); + + _emulator.Mapper.InitializeMemoryMap(this); + } + + public void PerformDMA(uint from) + { + //Console.WriteLine("OAM DMA"); + from <<= 8; + for (uint i = 0; i <= 0xFF; i++) + { + _oam[F.OAMAddress] = (byte)_emulator.CPU.ReadByte(from); + from++; + F.OAMAddress++; + } + + _emulator.CPU.Cycle += 513 + _emulator.CPU.Cycle % 2; + } + } +} diff --git a/Runtime/PPU.Memory.cs.meta b/Runtime/PPU.Memory.cs.meta new file mode 100644 index 0000000..cf6d09f --- /dev/null +++ b/Runtime/PPU.Memory.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fa442177f52c5ff4a805fb4e1a30703b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/PPU.Registers.cs b/Runtime/PPU.Registers.cs new file mode 100644 index 0000000..3c37945 --- /dev/null +++ b/Runtime/PPU.Registers.cs @@ -0,0 +1,261 @@ +using System; + +namespace Aya.UNes +{ + partial class PPU + { + public class PPUFlags + { + /* PPUCTRL register */ + public bool NMIEnabled; + public bool IsMaster; + public bool TallSpritesEnabled; + public uint PatternTableAddress; + public uint SpriteTableAddress; + public uint VRAMIncrement; + + /* PPUMASK register */ + public bool GrayscaleEnabled; + public bool DrawLeftBackground; + public bool DrawLeftSprites; + public bool DrawBackground; + public bool DrawSprites; + // Flipped for PAL/Dendy + public bool EmphasizeRed; + public bool EmphasizeGreen; + public bool EmphasizeBlue; + + /* PPUSTATUS register */ + public bool VBlankStarted; + public bool Sprite0Hit; + public bool SpriteOverflow; + public bool AddressLatch; + + /* PPUADDR register */ + private uint _busAddress; + public uint BusAddress + { + get => _busAddress; + set => _busAddress = value & 0x3FFF; + } + + /* PPUDATA register */ + public uint BusData; + + /* OAMADDR register */ + private uint _oamAddress; + public uint OAMAddress + { + get => _oamAddress; + set => _oamAddress = value & 0xFF; + } + + /* PPUSCROLL registers */ + [Obsolete] + public uint ScrollX; + [Obsolete] + public uint ScrollY; + + public bool RenderingEnabled => DrawBackground || DrawSprites; + } + + public PPUFlags F = new PPUFlags(); + + private uint _v; + public uint V + { + get => _v; + set => _v = value & 0x7FFF; + } + public uint T, X; + + public uint CoarseX => V & 0x1F; + + public uint CoarseY => (V >> 5) & 0x1F; + + public uint FineY => (V >> 12) & 0x7; + + public void ReloadScrollX() => V = (V & 0xFBE0) | (T & 0x041F); + + public void ReloadScrollY() => V = (V & 0x841F) | (T & 0x7BE0); + + public void IncrementScrollX() + { + if ((V & 0x001F) == 31) // if coarse X == 31 + { + V &= ~0x001Fu; // coarse X = 0 + V ^= 0x0400; // switch horizontal nametable + } + else + { + V += 1; // increment coarse X + } + } + + public void IncrementScrollY() + { + if ((V & 0x7000) != 0x7000) // if fine Y < 7 + { + V += 0x1000; // increment fine Y + } + else + { + V &= ~0x7000u; // fine Y = 0 + + uint y = (V & 0x03E0) >> 5; // let y = coarse Y + if (y == 29) + { + y = 0; // coarse Y = 0 + V ^= 0x0800; + } + // switch vertical nametable + else if (y == 31) + { + y = 0; // coarse Y = 0, nametable not switched + } + else + { + y += 1; // increment coarse Y + } + + V = (V & ~0x03E0u) | (y << 5); // put coarse Y back into v + } + } + + public uint PPUCTRL + { + set + { + F.NMIEnabled = (value & 0x80) > 0; + F.IsMaster = (value & 0x40) > 0; + F.TallSpritesEnabled = (value & 0x20) > 0; + F.PatternTableAddress = (value & 0x10) > 0 ? 0x1000u : 0x0000; + F.SpriteTableAddress = (value & 0x08) > 0 ? 0x1000u : 0x0000; + F.VRAMIncrement = (value & 0x04) > 0 ? 32u : 1; + // yyy NN YYYYY XXXXX + // ||| || ||||| +++++--coarse X scroll + // ||| || +++++--------coarse Y scroll + // ||| ++--------------nametable select + // +++-----------------fine Y scroll + T = (T & 0xF3FF) | ((value & 0x3) << 10); // Bits 10-11 hold the base address of the nametable minus $2000 + } + } + + public uint PPUMASK + { + set + { + F.GrayscaleEnabled = (value & 0x1) > 0; + F.DrawLeftBackground = (value & 0x2) > 0; + F.DrawLeftSprites = (value & 0x4) > 0; + F.DrawBackground = (value & 0x8) > 0; + F.DrawSprites = (value & 0x10) > 0; + F.EmphasizeRed = (value & 0x20) > 0; + F.EmphasizeGreen = (value & 0x40) > 0; + F.EmphasizeBlue = (value & 0x80) > 0; + } + } + + /** $2002 **/ + public uint PPUSTATUS + { + get + { + F.AddressLatch = false; + var ret = (F.VBlankStarted.AsByte() << 7) | + (F.Sprite0Hit.AsByte() << 6) | + (F.SpriteOverflow.AsByte() << 5) | + (_lastWrittenRegister & 0x1F); + F.VBlankStarted = false; + return (uint)ret; + } + } + + /** $2006 **/ + public uint PPUADDR + { + set + { + if (F.AddressLatch) + { + T = (T & 0xFF00) | value; + F.BusAddress = T; + V = T; + } + else + { + T = (T & 0x80FF) | ((value & 0x3F) << 8); + } + + F.AddressLatch ^= true; + } + } + + /** $2005 **/ + public uint PPUSCROLL + { + set + { + if (F.AddressLatch) + { + F.ScrollY = value; + T = (T & 0x8FFF) | ((value & 0x7) << 12); + T = (T & 0xFC1F) | (value & 0xF8) << 2; + } + else + { + F.ScrollX = value; + X = value & 0x7; + T = (T & 0xFFE0) | (value >> 3); + } + + F.AddressLatch ^= true; + } + } + + private uint _readBuffer; + public uint PPUDATA + { + get + { + uint ret = ReadByte(F.BusAddress); + if (F.BusAddress < 0x3F00) + { + uint temp = _readBuffer; + _readBuffer = ret; + ret = temp; + } + else + { + // Palette read should also read VRAM into read buffer + _readBuffer = ReadByte(F.BusAddress - 0x1000); + } + + F.BusAddress += F.VRAMIncrement; + return ret; + } + set + { + F.BusData = value; + WriteByte(F.BusAddress, value); + F.BusAddress += F.VRAMIncrement; + } + } + + public uint OAMADDR + { + get => F.OAMAddress; + set => F.OAMAddress = value; + } + + public uint OAMDATA + { + get => _oam[F.OAMAddress]; + set + { + _oam[F.OAMAddress] = (byte)value; + F.OAMAddress++; + } + } + } +} diff --git a/Runtime/PPU.Registers.cs.meta b/Runtime/PPU.Registers.cs.meta new file mode 100644 index 0000000..24948ac --- /dev/null +++ b/Runtime/PPU.Registers.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0db554be3f472c940bde6ce34fc9ed7e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/PPU.cs b/Runtime/PPU.cs new file mode 100644 index 0000000..c3a08c7 --- /dev/null +++ b/Runtime/PPU.cs @@ -0,0 +1,10 @@ +namespace Aya.UNes +{ + public sealed partial class PPU : Addressable + { + public PPU(Emulator emulator) : base(emulator, 0x3FFF) + { + InitializeMemoryMap(); + } + } +} diff --git a/Runtime/PPU.cs.meta b/Runtime/PPU.cs.meta new file mode 100644 index 0000000..5006815 --- /dev/null +++ b/Runtime/PPU.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: bd14757720b242e4b92425c9d6aae16a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Renderer.meta b/Runtime/Renderer.meta new file mode 100644 index 0000000..ede6704 --- /dev/null +++ b/Runtime/Renderer.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: ff3612d2884e82145a83b869175fe872 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Renderer/IRenderer.cs b/Runtime/Renderer/IRenderer.cs new file mode 100644 index 0000000..683b552 --- /dev/null +++ b/Runtime/Renderer/IRenderer.cs @@ -0,0 +1,13 @@ +namespace Aya.UNes.Renderer +{ + public interface IRenderer + { + string RendererName { get; } + + void Draw(); + + void InitRendering(UNes nes); + + void EndRendering(); + } +} diff --git a/Runtime/Renderer/IRenderer.cs.meta b/Runtime/Renderer/IRenderer.cs.meta new file mode 100644 index 0000000..5bad6e1 --- /dev/null +++ b/Runtime/Renderer/IRenderer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6e02ede35ded0294fb7f13f60387557f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Renderer/UnityRenderer.cs b/Runtime/Renderer/UnityRenderer.cs new file mode 100644 index 0000000..3881f32 --- /dev/null +++ b/Runtime/Renderer/UnityRenderer.cs @@ -0,0 +1,65 @@ +using System.Runtime.InteropServices; +using UnityEngine; + +namespace Aya.UNes.Renderer +{ + public class UnityRenderer : IRenderer + { + public UNes UNes; + public RenderTexture RenderTexture; + + public string RendererName => "Unity"; + + private Texture2D _drawTexture; + private Color[] _pixelCache; + + public void Draw() + { + if (RenderTexture.filterMode != UNes.FilterMode) + { + RenderTexture.filterMode = UNes.FilterMode; + } + + for (var y = 0; y < UNes.GameHeight; y++) + { + for (var x = 0; x < UNes.GameWidth; x++) + { + var rawIndex = UNes.GameWidth * y + x; + var color = GetColor(UNes.RawBitmap[rawIndex]); + var texIndex = UNes.GameWidth * (UNes.GameHeight - y - 1) + x; + _pixelCache[texIndex] = color; + } + } + + _drawTexture.SetPixels(_pixelCache); + _drawTexture.Apply(); + + Graphics.Blit(_drawTexture, RenderTexture); + } + + public Color GetColor(uint value) + { + var r = 0xFF0000 & value; + r >>= 16; + var b = 0xFF & value; + var g = 0xFF00 & value; + g >>= 8; + var color = new Color(r / 255f, g / 255f, b / 255f); + return color; + } + + public void InitRendering(UNes nes) + { + UNes = nes; + RenderTexture = nes.RenderTexture; + _drawTexture = new Texture2D(UNes.GameWidth, UNes.GameHeight); + _pixelCache = new Color[UNes.GameWidth * UNes.GameHeight]; + } + + public void EndRendering() + { + + } + } +} + diff --git a/Runtime/Renderer/UnityRenderer.cs.meta b/Runtime/Renderer/UnityRenderer.cs.meta new file mode 100644 index 0000000..13e31a8 --- /dev/null +++ b/Runtime/Renderer/UnityRenderer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1f751adeb00750a43a0d2534a2301da2 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/UNes.cs b/Runtime/UNes.cs new file mode 100644 index 0000000..f7d6c47 --- /dev/null +++ b/Runtime/UNes.cs @@ -0,0 +1,251 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Linq; +using System.Threading; +using UnityEngine; +using Aya.UNes.Controller; +using Aya.UNes.Renderer; +using Debug = UnityEngine.Debug; + +namespace Aya.UNes +{ + public class UNes : MonoBehaviour + { + public string RomFile; + public RenderTexture RenderTexture; + + public const int GameWidth = 256; + public const int GameHeight = 240; + public FilterMode FilterMode = FilterMode.Point; + public bool RenderThread = true; + + + private bool _rendererRunning = true; + private readonly IController _controller = new NesController(); + + private Emulator _emu; + private bool _suspended; + + private readonly Type[] _possibleRendererList = { typeof(UnityRenderer) }; + private readonly List _availableRendererList = new List(); + + public IRenderer Renderer { get; set; } + public uint[] RawBitmap { get; set; } = new uint[GameWidth * GameHeight]; + public bool Ready { get; set; } + public bool GameStarted { get; set; } + + private Thread _renderThread; + private int _activeSpeed = 1; + + private void BootCartridge(string rom) + { + var bytes = Resources.Load(RomFile).bytes; + _emu = new Emulator(bytes, _controller); + + if (RenderThread) + { + _renderThread = new Thread(() => + { + GameStarted = true; + Console.WriteLine(_emu.Cartridge); + var s = new Stopwatch(); + var s0 = new Stopwatch(); + while (_rendererRunning) + { + if (_suspended) + { + Thread.Sleep(100); + continue; + } + + s.Restart(); + for (var i = 0; i < 60 && !_suspended; i++) + { + s0.Restart(); + lock (RawBitmap) + { + _emu.PPU.ProcessFrame(); + RawBitmap = _emu.PPU.RawBitmap; + } + + s0.Stop(); + Thread.Sleep(Math.Max((int)(980 / 60.0 - s0.ElapsedMilliseconds), 0) / _activeSpeed); + } + + s.Stop(); + Console.WriteLine($"60 frames in {s.ElapsedMilliseconds}ms"); + } + }); + + _renderThread.Start(); + } + else + { + GameStarted = true; + } + } + + #region Monobehaviour + + public void Awake() + { + FindRenderer(); + SetRenderer(_availableRendererList.Last()); + } + + public void OnEnable() + { + BootCartridge(RomFile); + } + + public void OnDisable() + { + _rendererRunning = false; + // _emu?.Save(); + } + + public void Update() + { + if (!GameStarted) return; + UpdateInput(); + UpdateRender(); + } + + #endregion + + #region Input + + public void UpdateInput() + { + HandlerKeyDown(keyCode => + { + switch (keyCode) + { + case KeyCode.F2: + _suspended = false; + break; + case KeyCode.F3: + _suspended = true; + break; + default: + _controller.PressKey(keyCode); + break; + } + }); + + HandlerKeyUp(keyCode => + { + _controller.ReleaseKey(keyCode); + }); + } + + private KeyCode[] _keyCodes + { + get + { + if (_keyCodeCache == null) + { + var array = Enum.GetValues(typeof(KeyCode)); + _keyCodeCache = new KeyCode[array.Length]; + for (var i = 0; i < array.Length; i++) + { + _keyCodeCache[i] = (KeyCode) array.GetValue(i); + } + } + + return _keyCodeCache; + } + } + private KeyCode[] _keyCodeCache; + + public void HandlerKeyDown(Action onKeyDown) + { + foreach (var keyCode in _keyCodes) + { + if (Input.GetKeyDown(keyCode)) + { + onKeyDown(keyCode); + } + } + } + + public void HandlerKeyUp(Action onKeyUp) + { + foreach (var keyCode in _keyCodes) + { + if (Input.GetKeyUp(keyCode)) + { + onKeyUp(keyCode); + } + } + } + + #endregion + + #region Render + + public void UpdateRender() + { + if (!RenderThread) + { + if (!_rendererRunning) return; + if (_suspended) return; + + _emu.PPU.ProcessFrame(); + RawBitmap = _emu.PPU.RawBitmap; + Renderer.Draw(); + } + else + { + if (!_rendererRunning) return; + if (_suspended) return; + + lock (RawBitmap) + { + Renderer.Draw(); + } + } + } + + private void SetRenderer(IRenderer render) + { + if (Renderer == render) return; + Renderer?.EndRendering(); + Renderer = render; + render.InitRendering(this); + } + + private void FindRenderer() + { + foreach (var renderType in _possibleRendererList) + { + try + { + var renderer = (IRenderer)Activator.CreateInstance(renderType); + renderer.InitRendering(this); + renderer.EndRendering(); + _availableRendererList.Add(renderer); + } + catch (Exception) + { + Console.WriteLine($"{renderType} failed to initialize"); + } + } + } + + #endregion + + public void Open(string file) + { + try + { + BootCartridge(file); + } + catch (Exception ex) + { + Debug.LogError("Error loading ROM file; either corrupt or unsupported"); + } + } + } +} diff --git a/Runtime/UNes.cs.meta b/Runtime/UNes.cs.meta new file mode 100644 index 0000000..99998ce --- /dev/null +++ b/Runtime/UNes.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5948479853ec1194497c60babb6bc213 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Runtime/Utility.cs b/Runtime/Utility.cs new file mode 100644 index 0000000..3085f25 --- /dev/null +++ b/Runtime/Utility.cs @@ -0,0 +1,35 @@ +using System; +using System.Collections.Generic; +using System.Runtime.CompilerServices; + +namespace Aya.UNes +{ + public static class Utility + { + [MethodImpl(MethodImplOptions.AggressiveInlining)] + public static byte AsByte(this bool to) + { + unsafe + { + var result = *((byte*) &to); + return result; + } + } + + public static void Fill(this T[] arr, T value) + { + for (var i = 0; i < arr.Length; i++) + { + arr[i] = value; + } + } + + public static void Map(this IEnumerable enumerator, Action go) + { + foreach (var e in enumerator) + { + go(e); + } + } + } +} diff --git a/Runtime/Utility.cs.meta b/Runtime/Utility.cs.meta new file mode 100644 index 0000000..2d1f3ff --- /dev/null +++ b/Runtime/Utility.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b326828d1b42abc419d9fe125eed8205 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Sample.meta b/Sample.meta new file mode 100644 index 0000000..46826c5 --- /dev/null +++ b/Sample.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 049e5a5b3f9646b4fab19ebeb26449a1 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Sample/Resources.meta b/Sample/Resources.meta new file mode 100644 index 0000000..dfacc8d --- /dev/null +++ b/Sample/Resources.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 481be8b9dd1218749a75bf1bd4612c04 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Sample/Resources/Mario.bytes b/Sample/Resources/Mario.bytes new file mode 100644 index 0000000000000000000000000000000000000000..878ef21b5976a87c2fc19e780381816cbc769a65 GIT binary patch literal 40976 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