From e90cdc3e992cbc5ba1d44ce445254a9cda0107e2 Mon Sep 17 00:00:00 2001 From: "ALIENJACK\\alien" Date: Wed, 3 Jul 2024 15:40:13 +0800 Subject: [PATCH] =?UTF-8?q?mynes.core=20dll=E6=94=B9=E4=BB=A3=E7=A0=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 5 + AxibugEmuOnline.Client/Assets/MyNes.Core.meta | 8 + .../Assets/MyNes.Core/Bandai.cs | 143 + .../Assets/MyNes.Core/Bandai.cs.meta | 11 + .../Assets/MyNes.Core/BankInfo.cs | 29 + .../Assets/MyNes.Core/BankInfo.cs.meta | 11 + .../Assets/MyNes.Core/BankInfoSorter.cs | 16 + .../Assets/MyNes.Core/BankInfoSorter.cs.meta | 11 + .../Assets/MyNes.Core/BlankJoypad.cs | 9 + .../Assets/MyNes.Core/BlankJoypad.cs.meta | 11 + .../MyNes.Core/BlankShortuctsHandler.cs | 10 + .../MyNes.Core/BlankShortuctsHandler.cs.meta | 11 + .../Assets/MyNes.Core/Board.cs | 1117 ++++ .../Assets/MyNes.Core/Board.cs.meta | 11 + .../Assets/MyNes.Core/BoardInfoAttribute.cs | 57 + .../MyNes.Core/BoardInfoAttribute.cs.meta | 11 + .../Assets/MyNes.Core/BoardInfoObject.cs | 15 + .../Assets/MyNes.Core/BoardInfoObject.cs.meta | 11 + .../Assets/MyNes.Core/CHRArea.cs | 14 + .../Assets/MyNes.Core/CHRArea.cs.meta | 11 + .../ComponentAce.Compression.Libs.zlib.meta | 8 + .../Adler32.cs | 72 + .../Adler32.cs.meta | 11 + .../Deflate.cs | 1379 ++++ .../Deflate.cs.meta | 11 + .../InfBlocks.cs | 658 ++ .../InfBlocks.cs.meta | 11 + .../InfCodes.cs | 666 ++ .../InfCodes.cs.meta | 11 + .../InfTree.cs | 475 ++ .../InfTree.cs.meta | 11 + .../Inflate.cs | 409 ++ .../Inflate.cs.meta | 11 + .../StaticTree.cs | 126 + .../StaticTree.cs.meta | 11 + .../SupportClass.cs | 104 + .../SupportClass.cs.meta | 11 + .../Tree.cs | 337 + .../Tree.cs.meta | 11 + .../ZInputStream.cs | 134 + .../ZInputStream.cs.meta | 11 + .../ZOutputStream.cs | 193 + .../ZOutputStream.cs.meta | 11 + .../ZStream.cs | 222 + .../ZStream.cs.meta | 11 + .../ZStreamException.cs | 16 + .../ZStreamException.cs.meta | 11 + .../zlibConst.cs | 54 + .../zlibConst.cs.meta | 11 + .../Assets/MyNes.Core/Crc32.cs | 109 + .../Assets/MyNes.Core/Crc32.cs.meta | 11 + .../Assets/MyNes.Core/EmuRegion.cs | 9 + .../Assets/MyNes.Core/EmuRegion.cs.meta | 11 + .../Assets/MyNes.Core/EmuSettings.cs | 100 + .../Assets/MyNes.Core/EmuSettings.cs.meta | 11 + .../Assets/MyNes.Core/Eprom.cs | 383 ++ .../Assets/MyNes.Core/Eprom.cs.meta | 11 + .../Assets/MyNes.Core/FFE.cs | 56 + .../Assets/MyNes.Core/FFE.cs.meta | 11 + .../Assets/MyNes.Core/GameGenie.cs | 153 + .../Assets/MyNes.Core/GameGenie.cs.meta | 11 + .../Assets/MyNes.Core/GameGenieCode.cs | 19 + .../Assets/MyNes.Core/GameGenieCode.cs.meta | 11 + .../Assets/MyNes.Core/GetIsPlaying.cs | 4 + .../Assets/MyNes.Core/GetIsPlaying.cs.meta | 11 + .../Assets/MyNes.Core/HassIssuesAttribute.cs | 8 + .../MyNes.Core/HassIssuesAttribute.cs.meta | 11 + .../Assets/MyNes.Core/HelperTools.cs | 181 + .../Assets/MyNes.Core/HelperTools.cs.meta | 11 + .../Assets/MyNes.Core/IAudioProvider.cs | 29 + .../Assets/MyNes.Core/IAudioProvider.cs.meta | 11 + .../Assets/MyNes.Core/IJoypadConnecter.cs | 18 + .../MyNes.Core/IJoypadConnecter.cs.meta | 11 + .../Assets/MyNes.Core/INes.cs | 97 + .../Assets/MyNes.Core/INes.cs.meta | 11 + .../Assets/MyNes.Core/IRom.cs | 29 + .../Assets/MyNes.Core/IRom.cs.meta | 11 + .../Assets/MyNes.Core/ISettings.cs | 149 + .../Assets/MyNes.Core/ISettings.cs.meta | 11 + .../Assets/MyNes.Core/IShortcutsHandler.cs | 7 + .../MyNes.Core/IShortcutsHandler.cs.meta | 11 + .../MyNes.Core/IVSUnisystemDIPConnecter.cs | 25 + .../IVSUnisystemDIPConnecter.cs.meta | 11 + .../Assets/MyNes.Core/IVideoProvider.cs | 41 + .../Assets/MyNes.Core/IVideoProvider.cs.meta | 11 + .../Assets/MyNes.Core/IZapperConnecter.cs | 16 + .../MyNes.Core/IZapperConnecter.cs.meta | 11 + .../Assets/MyNes.Core/MMC2.cs | 116 + .../Assets/MyNes.Core/MMC2.cs.meta | 11 + .../Assets/MyNes.Core/MMC5Pcm.cs | 86 + .../Assets/MyNes.Core/MMC5Pcm.cs.meta | 11 + .../Assets/MyNes.Core/MMC5Sqr.cs | 214 + .../Assets/MyNes.Core/MMC5Sqr.cs.meta | 11 + .../Assets/MyNes.Core/MNInterfaceLanguage.cs | 129 + .../MyNes.Core/MNInterfaceLanguage.cs.meta | 11 + .../Assets/MyNes.Core/Mapper000.cs | 7 + .../Assets/MyNes.Core/Mapper000.cs.meta | 11 + .../Assets/MyNes.Core/Mapper001.cs | 245 + .../Assets/MyNes.Core/Mapper001.cs.meta | 11 + .../Assets/MyNes.Core/Mapper002.cs | 17 + .../Assets/MyNes.Core/Mapper002.cs.meta | 11 + .../Assets/MyNes.Core/Mapper003.cs | 15 + .../Assets/MyNes.Core/Mapper003.cs.meta | 11 + .../Assets/MyNes.Core/Mapper004.cs | 224 + .../Assets/MyNes.Core/Mapper004.cs.meta | 11 + .../Assets/MyNes.Core/Mapper005.cs | 890 +++ .../Assets/MyNes.Core/Mapper005.cs.meta | 11 + .../Assets/MyNes.Core/Mapper006.cs | 21 + .../Assets/MyNes.Core/Mapper006.cs.meta | 11 + .../Assets/MyNes.Core/Mapper007.cs | 12 + .../Assets/MyNes.Core/Mapper007.cs.meta | 11 + .../Assets/MyNes.Core/Mapper008.cs | 15 + .../Assets/MyNes.Core/Mapper008.cs.meta | 11 + .../Assets/MyNes.Core/Mapper009.cs | 7 + .../Assets/MyNes.Core/Mapper009.cs.meta | 11 + .../Assets/MyNes.Core/Mapper010.cs | 25 + .../Assets/MyNes.Core/Mapper010.cs.meta | 11 + .../Assets/MyNes.Core/Mapper011.cs | 16 + .../Assets/MyNes.Core/Mapper011.cs.meta | 11 + .../Assets/MyNes.Core/Mapper013.cs | 21 + .../Assets/MyNes.Core/Mapper013.cs.meta | 11 + .../Assets/MyNes.Core/Mapper015.cs | 36 + .../Assets/MyNes.Core/Mapper015.cs.meta | 11 + .../Assets/MyNes.Core/Mapper016.cs | 7 + .../Assets/MyNes.Core/Mapper016.cs.meta | 11 + .../Assets/MyNes.Core/Mapper017.cs | 64 + .../Assets/MyNes.Core/Mapper017.cs.meta | 11 + .../Assets/MyNes.Core/Mapper018.cs | 219 + .../Assets/MyNes.Core/Mapper018.cs.meta | 11 + .../Assets/MyNes.Core/Mapper019.cs | 7 + .../Assets/MyNes.Core/Mapper019.cs.meta | 11 + .../Assets/MyNes.Core/Mapper021.cs | 254 + .../Assets/MyNes.Core/Mapper021.cs.meta | 11 + .../Assets/MyNes.Core/Mapper022.cs | 138 + .../Assets/MyNes.Core/Mapper022.cs.meta | 11 + .../Assets/MyNes.Core/Mapper023.cs | 163 + .../Assets/MyNes.Core/Mapper023.cs.meta | 11 + .../Assets/MyNes.Core/Mapper024.cs | 245 + .../Assets/MyNes.Core/Mapper024.cs.meta | 11 + .../Assets/MyNes.Core/Mapper025.cs | 254 + .../Assets/MyNes.Core/Mapper025.cs.meta | 11 + .../Assets/MyNes.Core/Mapper026.cs | 245 + .../Assets/MyNes.Core/Mapper026.cs.meta | 11 + .../Assets/MyNes.Core/Mapper032.cs | 109 + .../Assets/MyNes.Core/Mapper032.cs.meta | 11 + .../Assets/MyNes.Core/Mapper033.cs | 177 + .../Assets/MyNes.Core/Mapper033.cs.meta | 11 + .../Assets/MyNes.Core/Mapper034.cs | 50 + .../Assets/MyNes.Core/Mapper034.cs.meta | 11 + .../Assets/MyNes.Core/Mapper041.cs | 58 + .../Assets/MyNes.Core/Mapper041.cs.meta | 11 + .../Assets/MyNes.Core/Mapper042.cs | 94 + .../Assets/MyNes.Core/Mapper042.cs.meta | 11 + .../Assets/MyNes.Core/Mapper044.cs | 235 + .../Assets/MyNes.Core/Mapper044.cs.meta | 11 + .../Assets/MyNes.Core/Mapper045.cs | 294 + .../Assets/MyNes.Core/Mapper045.cs.meta | 11 + .../Assets/MyNes.Core/Mapper046.cs | 42 + .../Assets/MyNes.Core/Mapper046.cs.meta | 11 + .../Assets/MyNes.Core/Mapper047.cs | 258 + .../Assets/MyNes.Core/Mapper047.cs.meta | 11 + .../Assets/MyNes.Core/Mapper048.cs | 172 + .../Assets/MyNes.Core/Mapper048.cs.meta | 11 + .../Assets/MyNes.Core/Mapper049.cs | 272 + .../Assets/MyNes.Core/Mapper049.cs.meta | 11 + .../Assets/MyNes.Core/Mapper050.cs | 71 + .../Assets/MyNes.Core/Mapper050.cs.meta | 11 + .../Assets/MyNes.Core/Mapper051.cs | 79 + .../Assets/MyNes.Core/Mapper051.cs.meta | 11 + .../Assets/MyNes.Core/Mapper052.cs | 273 + .../Assets/MyNes.Core/Mapper052.cs.meta | 11 + .../Assets/MyNes.Core/Mapper053.cs | 58 + .../Assets/MyNes.Core/Mapper053.cs.meta | 11 + .../Assets/MyNes.Core/Mapper056.cs | 97 + .../Assets/MyNes.Core/Mapper056.cs.meta | 11 + .../Assets/MyNes.Core/Mapper057.cs | 55 + .../Assets/MyNes.Core/Mapper057.cs.meta | 11 + .../Assets/MyNes.Core/Mapper058.cs | 24 + .../Assets/MyNes.Core/Mapper058.cs.meta | 11 + .../Assets/MyNes.Core/Mapper060.cs | 67 + .../Assets/MyNes.Core/Mapper060.cs.meta | 11 + .../Assets/MyNes.Core/Mapper061.cs | 20 + .../Assets/MyNes.Core/Mapper061.cs.meta | 11 + .../Assets/MyNes.Core/Mapper062.cs | 24 + .../Assets/MyNes.Core/Mapper062.cs.meta | 11 + .../Assets/MyNes.Core/Mapper064.cs | 258 + .../Assets/MyNes.Core/Mapper064.cs.meta | 11 + .../Assets/MyNes.Core/Mapper065.cs | 94 + .../Assets/MyNes.Core/Mapper065.cs.meta | 11 + .../Assets/MyNes.Core/Mapper066.cs | 12 + .../Assets/MyNes.Core/Mapper066.cs.meta | 11 + .../Assets/MyNes.Core/Mapper067.cs | 108 + .../Assets/MyNes.Core/Mapper067.cs.meta | 11 + .../Assets/MyNes.Core/Mapper068.cs | 112 + .../Assets/MyNes.Core/Mapper068.cs.meta | 11 + .../Assets/MyNes.Core/Mapper069.cs | 259 + .../Assets/MyNes.Core/Mapper069.cs.meta | 11 + .../Assets/MyNes.Core/Mapper070.cs | 18 + .../Assets/MyNes.Core/Mapper070.cs.meta | 11 + .../Assets/MyNes.Core/Mapper071.cs | 30 + .../Assets/MyNes.Core/Mapper071.cs.meta | 11 + .../Assets/MyNes.Core/Mapper072.cs | 53 + .../Assets/MyNes.Core/Mapper072.cs.meta | 11 + .../Assets/MyNes.Core/Mapper073.cs | 111 + .../Assets/MyNes.Core/Mapper073.cs.meta | 11 + .../Assets/MyNes.Core/Mapper074.cs | 234 + .../Assets/MyNes.Core/Mapper074.cs.meta | 11 + .../Assets/MyNes.Core/Mapper075.cs | 63 + .../Assets/MyNes.Core/Mapper075.cs.meta | 11 + .../Assets/MyNes.Core/Mapper076.cs | 77 + .../Assets/MyNes.Core/Mapper076.cs.meta | 11 + .../Assets/MyNes.Core/Mapper077.cs | 23 + .../Assets/MyNes.Core/Mapper077.cs.meta | 11 + .../Assets/MyNes.Core/Mapper078.cs | 33 + .../Assets/MyNes.Core/Mapper078.cs.meta | 11 + .../Assets/MyNes.Core/Mapper079.cs | 15 + .../Assets/MyNes.Core/Mapper079.cs.meta | 11 + .../Assets/MyNes.Core/Mapper080.cs | 56 + .../Assets/MyNes.Core/Mapper080.cs.meta | 11 + .../Assets/MyNes.Core/Mapper082.cs | 70 + .../Assets/MyNes.Core/Mapper082.cs.meta | 11 + .../Assets/MyNes.Core/Mapper085.cs | 171 + .../Assets/MyNes.Core/Mapper085.cs.meta | 11 + .../Assets/MyNes.Core/Mapper086.cs | 15 + .../Assets/MyNes.Core/Mapper086.cs.meta | 11 + .../Assets/MyNes.Core/Mapper087.cs | 11 + .../Assets/MyNes.Core/Mapper087.cs.meta | 11 + .../Assets/MyNes.Core/Mapper088.cs | 67 + .../Assets/MyNes.Core/Mapper088.cs.meta | 11 + .../Assets/MyNes.Core/Mapper089.cs | 19 + .../Assets/MyNes.Core/Mapper089.cs.meta | 11 + .../Assets/MyNes.Core/Mapper090.cs | 634 ++ .../Assets/MyNes.Core/Mapper090.cs.meta | 11 + .../Assets/MyNes.Core/Mapper091.cs | 96 + .../Assets/MyNes.Core/Mapper091.cs.meta | 11 + .../Assets/MyNes.Core/Mapper092.cs | 50 + .../Assets/MyNes.Core/Mapper092.cs.meta | 11 + .../Assets/MyNes.Core/Mapper093.cs | 18 + .../Assets/MyNes.Core/Mapper093.cs.meta | 11 + .../Assets/MyNes.Core/Mapper094.cs | 17 + .../Assets/MyNes.Core/Mapper094.cs.meta | 11 + .../Assets/MyNes.Core/Mapper095.cs | 265 + .../Assets/MyNes.Core/Mapper095.cs.meta | 11 + .../Assets/MyNes.Core/Mapper096.cs | 46 + .../Assets/MyNes.Core/Mapper096.cs.meta | 11 + .../Assets/MyNes.Core/Mapper097.cs | 32 + .../Assets/MyNes.Core/Mapper097.cs.meta | 11 + .../Assets/MyNes.Core/Mapper105.cs | 200 + .../Assets/MyNes.Core/Mapper105.cs.meta | 11 + .../Assets/MyNes.Core/Mapper107.cs | 11 + .../Assets/MyNes.Core/Mapper107.cs.meta | 11 + .../Assets/MyNes.Core/Mapper112.cs | 70 + .../Assets/MyNes.Core/Mapper112.cs.meta | 11 + .../Assets/MyNes.Core/Mapper113.cs | 18 + .../Assets/MyNes.Core/Mapper113.cs.meta | 11 + .../Assets/MyNes.Core/Mapper115.cs | 259 + .../Assets/MyNes.Core/Mapper115.cs.meta | 11 + .../Assets/MyNes.Core/Mapper118.cs | 263 + .../Assets/MyNes.Core/Mapper118.cs.meta | 11 + .../Assets/MyNes.Core/Mapper119.cs | 236 + .../Assets/MyNes.Core/Mapper119.cs.meta | 11 + .../Assets/MyNes.Core/Mapper133.cs | 12 + .../Assets/MyNes.Core/Mapper133.cs.meta | 11 + .../Assets/MyNes.Core/Mapper140.cs | 12 + .../Assets/MyNes.Core/Mapper140.cs.meta | 11 + .../Assets/MyNes.Core/Mapper152.cs | 19 + .../Assets/MyNes.Core/Mapper152.cs.meta | 11 + .../Assets/MyNes.Core/Mapper154.cs | 72 + .../Assets/MyNes.Core/Mapper154.cs.meta | 11 + .../Assets/MyNes.Core/Mapper163.cs | 109 + .../Assets/MyNes.Core/Mapper163.cs.meta | 11 + .../Assets/MyNes.Core/Mapper164.cs | 35 + .../Assets/MyNes.Core/Mapper164.cs.meta | 11 + .../Assets/MyNes.Core/Mapper165.cs | 253 + .../Assets/MyNes.Core/Mapper165.cs.meta | 11 + .../Assets/MyNes.Core/Mapper180.cs | 14 + .../Assets/MyNes.Core/Mapper180.cs.meta | 11 + .../Assets/MyNes.Core/Mapper182.cs | 216 + .../Assets/MyNes.Core/Mapper182.cs.meta | 11 + .../Assets/MyNes.Core/Mapper184.cs | 12 + .../Assets/MyNes.Core/Mapper184.cs.meta | 11 + .../Assets/MyNes.Core/Mapper185.cs | 45 + .../Assets/MyNes.Core/Mapper185.cs.meta | 11 + .../Assets/MyNes.Core/Mapper189.cs | 185 + .../Assets/MyNes.Core/Mapper189.cs.meta | 11 + .../Assets/MyNes.Core/Mapper191.cs | 212 + .../Assets/MyNes.Core/Mapper191.cs.meta | 11 + .../Assets/MyNes.Core/Mapper192.cs | 225 + .../Assets/MyNes.Core/Mapper192.cs.meta | 11 + .../Assets/MyNes.Core/Mapper193.cs | 36 + .../Assets/MyNes.Core/Mapper193.cs.meta | 11 + .../Assets/MyNes.Core/Mapper194.cs | 209 + .../Assets/MyNes.Core/Mapper194.cs.meta | 11 + .../Assets/MyNes.Core/Mapper200.cs | 14 + .../Assets/MyNes.Core/Mapper200.cs.meta | 11 + .../Assets/MyNes.Core/Mapper201.cs | 12 + .../Assets/MyNes.Core/Mapper201.cs.meta | 11 + .../Assets/MyNes.Core/Mapper202.cs | 22 + .../Assets/MyNes.Core/Mapper202.cs.meta | 11 + .../Assets/MyNes.Core/Mapper203.cs | 16 + .../Assets/MyNes.Core/Mapper203.cs.meta | 11 + .../Assets/MyNes.Core/Mapper204.cs | 19 + .../Assets/MyNes.Core/Mapper204.cs.meta | 11 + .../Assets/MyNes.Core/Mapper205.cs | 275 + .../Assets/MyNes.Core/Mapper205.cs.meta | 11 + .../Assets/MyNes.Core/Mapper206.cs | 131 + .../Assets/MyNes.Core/Mapper206.cs.meta | 11 + .../Assets/MyNes.Core/Mapper207.cs | 112 + .../Assets/MyNes.Core/Mapper207.cs.meta | 11 + .../Assets/MyNes.Core/Mapper209.cs | 12 + .../Assets/MyNes.Core/Mapper209.cs.meta | 11 + .../Assets/MyNes.Core/Mapper212.cs | 19 + .../Assets/MyNes.Core/Mapper212.cs.meta | 11 + .../Assets/MyNes.Core/Mapper213.cs | 12 + .../Assets/MyNes.Core/Mapper213.cs.meta | 11 + .../Assets/MyNes.Core/Mapper214.cs | 13 + .../Assets/MyNes.Core/Mapper214.cs.meta | 11 + .../Assets/MyNes.Core/Mapper216.cs | 12 + .../Assets/MyNes.Core/Mapper216.cs.meta | 11 + .../Assets/MyNes.Core/Mapper222.cs | 48 + .../Assets/MyNes.Core/Mapper222.cs.meta | 11 + .../Assets/MyNes.Core/Mapper225.cs | 63 + .../Assets/MyNes.Core/Mapper225.cs.meta | 11 + .../Assets/MyNes.Core/Mapper226.cs | 66 + .../Assets/MyNes.Core/Mapper226.cs.meta | 11 + .../Assets/MyNes.Core/Mapper227.cs | 90 + .../Assets/MyNes.Core/Mapper227.cs.meta | 11 + .../Assets/MyNes.Core/Mapper228.cs | 61 + .../Assets/MyNes.Core/Mapper228.cs.meta | 11 + .../Assets/MyNes.Core/Mapper229.cs | 17 + .../Assets/MyNes.Core/Mapper229.cs.meta | 11 + .../Assets/MyNes.Core/Mapper230.cs | 58 + .../Assets/MyNes.Core/Mapper230.cs.meta | 11 + .../Assets/MyNes.Core/Mapper231.cs | 13 + .../Assets/MyNes.Core/Mapper231.cs.meta | 11 + .../Assets/MyNes.Core/Mapper232.cs | 42 + .../Assets/MyNes.Core/Mapper232.cs.meta | 11 + .../Assets/MyNes.Core/Mapper233.cs | 68 + .../Assets/MyNes.Core/Mapper233.cs.meta | 11 + .../Assets/MyNes.Core/Mapper240.cs | 12 + .../Assets/MyNes.Core/Mapper240.cs.meta | 11 + .../Assets/MyNes.Core/Mapper242.cs | 12 + .../Assets/MyNes.Core/Mapper242.cs.meta | 11 + .../Assets/MyNes.Core/Mapper243.cs | 88 + .../Assets/MyNes.Core/Mapper243.cs.meta | 11 + .../Assets/MyNes.Core/Mapper245.cs | 229 + .../Assets/MyNes.Core/Mapper245.cs.meta | 11 + .../Assets/MyNes.Core/Mapper246.cs | 50 + .../Assets/MyNes.Core/Mapper246.cs.meta | 11 + .../Assets/MyNes.Core/Mapper255.cs | 66 + .../Assets/MyNes.Core/Mapper255.cs.meta | 11 + .../Assets/MyNes.Core/MemReadAccess.cs | 4 + .../Assets/MyNes.Core/MemReadAccess.cs.meta | 11 + .../Assets/MyNes.Core/MemWriteAccess.cs | 4 + .../Assets/MyNes.Core/MemWriteAccess.cs.meta | 11 + .../Assets/MyNes.Core/Mirroring.cs | 11 + .../Assets/MyNes.Core/Mirroring.cs.meta | 11 + .../Assets/MyNes.Core/MyNes.Core.asmdef | 3 + .../Assets/MyNes.Core/MyNes.Core.asmdef.meta | 7 + .../Assets/MyNes.Core/MyNesMain.cs | 339 + .../Assets/MyNes.Core/MyNesMain.cs.meta | 11 + .../Assets/MyNes.Core/NTArea.cs | 10 + .../Assets/MyNes.Core/NTArea.cs.meta | 11 + .../Assets/MyNes.Core/Namcot106.cs | 553 ++ .../Assets/MyNes.Core/Namcot106.cs.meta | 11 + .../Assets/MyNes.Core/Namcot106Chnl.cs | 157 + .../Assets/MyNes.Core/Namcot106Chnl.cs.meta | 11 + .../Assets/MyNes.Core/NesCartDatabase.cs | 330 + .../Assets/MyNes.Core/NesCartDatabase.cs.meta | 11 + .../NesCartDatabaseCartridgeInfo.cs | 53 + .../NesCartDatabaseCartridgeInfo.cs.meta | 11 + .../MyNes.Core/NesCartDatabaseGameInfo.cs | 29 + .../NesCartDatabaseGameInfo.cs.meta | 11 + .../Assets/MyNes.Core/NesEmu.cs | 5939 +++++++++++++++++ .../Assets/MyNes.Core/NesEmu.cs.meta | 11 + .../Assets/MyNes.Core/PRGArea.cs | 18 + .../Assets/MyNes.Core/PRGArea.cs.meta | 11 + .../Assets/MyNes.Core/PaletteSelectSetting.cs | 10 + .../MyNes.Core/PaletteSelectSetting.cs.meta | 11 + .../Assets/MyNes.Core/RegionSetting.cs | 10 + .../Assets/MyNes.Core/RegionSetting.cs.meta | 11 + .../Assets/MyNes.Core/RenderAudioSamples.cs | 4 + .../MyNes.Core/RenderAudioSamples.cs.meta | 11 + .../Assets/MyNes.Core/RenderVideoFrame.cs | 4 + .../MyNes.Core/RenderVideoFrame.cs.meta | 11 + .../Assets/MyNes.Core/RendererSettings.cs | 106 + .../MyNes.Core/RendererSettings.cs.meta | 11 + .../Assets/MyNes.Core/SRAMBankInfo.cs | 22 + .../Assets/MyNes.Core/SRAMBankInfo.cs.meta | 11 + .../Assets/MyNes.Core/SoundDCBlockerFilter.cs | 33 + .../MyNes.Core/SoundDCBlockerFilter.cs.meta | 11 + .../Assets/MyNes.Core/SoundHighPassFilter.cs | 36 + .../MyNes.Core/SoundHighPassFilter.cs.meta | 11 + .../Assets/MyNes.Core/SoundLowPassFilter.cs | 45 + .../MyNes.Core/SoundLowPassFilter.cs.meta | 11 + .../Assets/MyNes.Core/StateHandler.cs | 197 + .../Assets/MyNes.Core/StateHandler.cs.meta | 11 + .../Assets/MyNes.Core/Sunsoft5BChnl.cs | 95 + .../Assets/MyNes.Core/Sunsoft5BChnl.cs.meta | 11 + .../Assets/MyNes.Core/TogglePause.cs | 4 + .../Assets/MyNes.Core/TogglePause.cs.meta | 11 + .../Assets/MyNes.Core/Tracer.cs | 59 + .../Assets/MyNes.Core/Tracer.cs.meta | 11 + .../Assets/MyNes.Core/TracerEventArgs.cs | 17 + .../Assets/MyNes.Core/TracerEventArgs.cs.meta | 11 + .../Assets/MyNes.Core/TracerStatus.cs | 10 + .../Assets/MyNes.Core/TracerStatus.cs.meta | 11 + .../Assets/MyNes.Core/VRC6Pulse.cs | 116 + .../Assets/MyNes.Core/VRC6Pulse.cs.meta | 11 + .../Assets/MyNes.Core/VRC6Sawtooth.cs | 114 + .../Assets/MyNes.Core/VRC6Sawtooth.cs.meta | 11 + .../Assets/MyNes.Core/WaveRecorder.cs | 302 + .../Assets/MyNes.Core/WaveRecorder.cs.meta | 11 + .../MyNes.Core/WithExternalSoundAttribute.cs | 8 + .../WithExternalSoundAttribute.cs.meta | 11 + .../Assets/MyNes.Core/ZlipWrapper.cs | 39 + .../Assets/MyNes.Core/ZlipWrapper.cs.meta | 11 + .../Assets/Plugins/Core.dll | Bin 347648 -> 0 bytes .../Assets/Plugins/Core.dll.meta | 33 - .../Assets/Scene/EmuTest.unity | 316 + .../Assets/Scene/EmuTest.unity.meta | 7 + .../Script/AxibugEmuOnline.Client.asmdef | 5 +- .../AxibugEmuOnline.Client.sln | 298 +- .../Packages/packages-lock.json | 52 +- .../ProjectSettings/AudioManager.asset | 20 + .../ProjectSettings/ClusterInputManager.asset | 6 + .../ProjectSettings/DynamicsManager.asset | 40 + .../ProjectSettings/EditorBuildSettings.asset | 8 + .../ProjectSettings/EditorSettings.asset | 47 + .../ProjectSettings/GraphicsSettings.asset | 67 + .../ProjectSettings/InputManager.asset | 296 + .../ProjectSettings/MemorySettings.asset | 35 + .../ProjectSettings/NavMeshAreas.asset | 93 + .../PackageManagerSettings.asset | 36 + .../ProjectSettings/Physics2DSettings.asset | 48 + .../ProjectSettings/PresetManager.asset | 7 + .../ProjectSettings/ProjectSettings.asset | 667 ++ .../ProjectSettings/ProjectVersion.txt | 2 + .../ProjectSettings/QualitySettings.asset | 322 + .../ProjectSettings/TagManager.asset | 43 + .../ProjectSettings/TimeManager.asset | 9 + .../UnityConnectSettings.asset | 38 + .../ProjectSettings/VFXManager.asset | 18 + .../VersionControlSettings.asset | 8 + .../UserSettings/EditorUserSettings.asset | 31 + .../UserSettings/Layouts/default-2022.dwlt | 1268 ++++ .../UserSettings/Search.index | 13 + .../UserSettings/Search.settings | 76 + 448 files changed, 36154 insertions(+), 81 deletions(-) create mode 100644 .gitignore create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs.meta create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs create mode 100644 AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs.meta delete mode 100644 AxibugEmuOnline.Client/Assets/Plugins/Core.dll delete mode 100644 AxibugEmuOnline.Client/Assets/Plugins/Core.dll.meta create mode 100644 AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity create mode 100644 AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity.meta create mode 100644 AxibugEmuOnline.Client/ProjectSettings/AudioManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/ClusterInputManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/DynamicsManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/EditorBuildSettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/EditorSettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/GraphicsSettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/InputManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/MemorySettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/NavMeshAreas.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/PackageManagerSettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/Physics2DSettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/PresetManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/ProjectSettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/ProjectVersion.txt create mode 100644 AxibugEmuOnline.Client/ProjectSettings/QualitySettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/TagManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/TimeManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/UnityConnectSettings.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/VFXManager.asset create mode 100644 AxibugEmuOnline.Client/ProjectSettings/VersionControlSettings.asset create mode 100644 AxibugEmuOnline.Client/UserSettings/EditorUserSettings.asset create mode 100644 AxibugEmuOnline.Client/UserSettings/Layouts/default-2022.dwlt create mode 100644 AxibugEmuOnline.Client/UserSettings/Search.index create mode 100644 AxibugEmuOnline.Client/UserSettings/Search.settings diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..aca88770 --- /dev/null +++ b/.gitignore @@ -0,0 +1,5 @@ +/AxibugEmuOnline.Client/*.vsconfig +/AxibugEmuOnline.Client/Library/ +/AxibugEmuOnline.Client/Temp/ +/AxibugEmuOnline.Client/.vs/ +/AxibugEmuOnline.Client/*.csproj diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core.meta new file mode 100644 index 00000000..433ca4b8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: 42793fd3c4b41384983c8c2551603557 +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs new file mode 100644 index 00000000..0eb3ef7e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs @@ -0,0 +1,143 @@ +using System.IO; + +namespace MyNes.Core +{ + + + internal abstract class Bandai : Board + { + private bool irq_enable; + + private int irq_counter; + + private Eprom eprom; + + internal override void Initialize(IRom rom) + { + base.Initialize(rom); + if (base.BoardType.ToLower().Contains("24c01")) + { + eprom = new Eprom(128); + } + else + { + eprom = new Eprom((base.MapperNumber == 16) ? 256 : 128); + } + } + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + irq_enable = false; + irq_counter = 0; + eprom.HardReset(); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + WritePRG(ref address, ref data); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF) + { + case 0: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 1: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 2: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 3: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 4: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 5: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 6: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 7: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 8: + Switch16KPRG(data, PRGArea.Area8000); + break; + case 9: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 10: + irq_enable = (data & 1) == 1; + NesEmu.IRQFlags &= -9; + break; + case 11: + irq_counter = (irq_counter & 0xFF00) | data; + break; + case 12: + irq_counter = (irq_counter & 0xFF) | (data << 8); + break; + case 13: + eprom.Write(address, data); + break; + } + } + + internal override void ReadSRM(ref ushort address, out byte value) + { + value = eprom.Read(address); + } + + internal override void OnCPUClock() + { + if (irq_enable) + { + irq_counter--; + if (irq_counter == 0) + { + NesEmu.IRQFlags |= 8; + } + if (irq_counter < 0) + { + irq_counter = 65535; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_enable); + stream.Write(irq_counter); + eprom.SaveState(stream); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_enable = stream.ReadBoolean(); + irq_counter = stream.ReadInt32(); + eprom.LoadState(stream); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs.meta new file mode 100644 index 00000000..44f5f1c5 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Bandai.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 449c08ab39c9dd440b660ac2d12fb8ff +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs new file mode 100644 index 00000000..3adbcac9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs @@ -0,0 +1,29 @@ +namespace MyNes.Core +{ + + + internal struct BankInfo + { + public bool IsRAM; + + public bool Enabled; + + public bool Writable; + + public bool IsBattery; + + public string ID; + + public byte[] DATA; + + public BankInfo(string ID, bool IsRAM, bool Writable, bool Enabled, bool IsBattery, byte[] DATA) + { + this.ID = ID; + this.IsRAM = IsRAM; + this.Writable = Writable; + this.Enabled = Enabled; + this.DATA = DATA; + this.IsBattery = IsBattery; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs.meta new file mode 100644 index 00000000..cc2618b2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfo.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e9cec9f263a14d640885246190006795 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs new file mode 100644 index 00000000..c0c8fbdf --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs @@ -0,0 +1,16 @@ +using System.Collections.Generic; + +namespace MyNes.Core +{ + internal class BankInfoSorter : IComparer + { + public int Compare(BankInfo x, BankInfo y) + { + int result = 0; + int result2 = 0; + int.TryParse(x.ID, out result); + int.TryParse(y.ID, out result2); + return result2 - result; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs.meta new file mode 100644 index 00000000..8278d2aa --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BankInfoSorter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2d41e03bf8974044fadc0b1cbcf7811e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs new file mode 100644 index 00000000..40de4151 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs @@ -0,0 +1,9 @@ +namespace MyNes.Core +{ + internal class BlankJoypad : IJoypadConnecter + { + public override void Update() + { + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs.meta new file mode 100644 index 00000000..afba8a61 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankJoypad.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d122941c868588e4e9fdb758ff7bd340 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs new file mode 100644 index 00000000..32807029 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs @@ -0,0 +1,10 @@ +namespace MyNes.Core +{ + + internal class BlankShortuctsHandler : IShortcutsHandler + { + public void Update() + { + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs.meta new file mode 100644 index 00000000..9271f363 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BlankShortuctsHandler.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d719eb3b397dd0743a86ded70de8d9dc +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs new file mode 100644 index 00000000..95c0f95d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs @@ -0,0 +1,1117 @@ +using System; +using System.Collections.Generic; +using System.IO; + +namespace MyNes.Core +{ + internal abstract class Board + { + protected byte[][] PRG_RAM; + + protected bool[] PRG_RAM_ENABLED; + + protected bool[] PRG_RAM_WRITABLE; + + protected bool[] PRG_RAM_BATTERY; + + protected byte[][] PRG_ROM; + + protected int PRG_RAM_08KB_DEFAULT_BLK_Count; + + internal int PRG_ROM_04KB_Count; + + protected int PRG_ROM_08KB_Count; + + protected int PRG_ROM_16KB_Count; + + protected int PRG_ROM_32KB_Count; + + protected int PRG_ROM_04KB_Mask; + + protected int PRG_ROM_08KB_Mask; + + protected int PRG_ROM_16KB_Mask; + + protected int PRG_ROM_32KB_Mask; + + internal int PRG_RAM_04KB_Count; + + protected int PRG_RAM_08KB_Count; + + protected int PRG_RAM_16KB_Count; + + protected int PRG_RAM_32KB_Count; + + protected int PRG_RAM_04KB_Mask; + + protected int PRG_RAM_08KB_Mask; + + protected int PRG_RAM_16KB_Mask; + + protected int PRG_RAM_32KB_Mask; + + protected bool[] PRG_AREA_BLK_RAM; + + protected int[] PRG_AREA_BLK_INDEX; + + protected int PRG_TMP_INDX; + + protected int PRG_TMP_AREA; + + protected byte[][] CHR_RAM; + + protected bool[] CHR_RAM_ENABLED; + + protected bool[] CHR_RAM_WRITABLE; + + protected bool[] CHR_RAM_BATTERY; + + protected byte[][] CHR_ROM; + + protected bool[] CHR_AREA_BLK_RAM; + + protected int[] CHR_AREA_BLK_INDEX; + + protected int CHR_TMP_INDX; + + protected int CHR_TMP_AREA; + + protected int CHR_ROM_01KB_DEFAULT_BLK_Count; + + internal int CHR_ROM_01KB_Count; + + protected int CHR_ROM_02KB_Count; + + protected int CHR_ROM_04KB_Count; + + protected int CHR_ROM_08KB_Count; + + internal int CHR_ROM_01KB_Mask; + + protected int CHR_ROM_02KB_Mask; + + protected int CHR_ROM_04KB_Mask; + + protected int CHR_ROM_08KB_Mask; + + internal int CHR_RAM_01KB_Count; + + protected int CHR_RAM_02KB_Count; + + protected int CHR_RAM_04KB_Count; + + protected int CHR_RAM_08KB_Count; + + internal int CHR_RAM_01KB_Mask; + + protected int CHR_RAM_02KB_Mask; + + protected int CHR_RAM_04KB_Mask; + + protected int CHR_RAM_08KB_Mask; + + protected byte[][] NMT_RAM; + + internal int[] NMT_AREA_BLK_INDEX; + + protected int NMT_TMP_INDX; + + protected int NMT_TMP_AREA; + + internal Mirroring NMT_DEFAULT_MIRROR; + + internal string SHA1 = ""; + + internal string CRC = ""; + + internal bool IsGameFoundOnDB; + + internal NesCartDatabaseGameInfo GameInfo; + + internal NesCartDatabaseCartridgeInfo GameCartInfo; + + internal bool SRAMSaveRequired; + + protected bool enabled_ppuA12ToggleTimer; + + protected bool ppuA12TogglesOnRaisingEdge; + + protected int old_vram_address; + + protected int new_vram_address; + + protected int ppu_cycles_timer; + + internal bool enable_external_sound; + + internal bool IsGameGenieActive; + + internal GameGenieCode[] GameGenieCodes; + + internal string BoardType { get; private set; } + + internal string BoardPCB { get; private set; } + + internal List Chips { get; private set; } + + internal string Name { get; set; } + + internal int MapperNumber { get; set; } + + internal bool HasIssues { get; set; } + + internal virtual string Issues { get; set; } + + public Board() + { + MapperNumber = -1; + PRG_RAM_08KB_DEFAULT_BLK_Count = 1; + CHR_ROM_01KB_DEFAULT_BLK_Count = 8; + LoadAttrs(); + } + + internal virtual void Initialize(IRom rom) + { + SHA1 = rom.SHA1; + SRAMSaveRequired = false; + IsGameGenieActive = false; + BoardType = "N/A"; + BoardPCB = "N/A"; + Chips = new List(); + if (NesCartDatabase.Ready) + { + Tracer.WriteLine("Looking for rom in the database .."); + GameInfo = NesCartDatabase.Find(SHA1, out IsGameFoundOnDB); + if (GameInfo.Cartridges != null) + { + foreach (NesCartDatabaseCartridgeInfo cartridge in GameInfo.Cartridges) + { + if (cartridge.SHA1.ToLower() == SHA1.ToLower()) + { + GameCartInfo = cartridge; + break; + } + } + } + if (IsGameFoundOnDB) + { + Tracer.WriteInformation("Game found in Database !!"); + Tracer.WriteLine("> Game name: " + GameInfo.Game_Name); + Tracer.WriteLine("> Game alt name: " + GameInfo.Game_AltName); + BoardType = GameCartInfo.Board_Type; + Tracer.WriteLine("> Board Type: " + BoardType); + BoardPCB = GameCartInfo.Board_Pcb; + Tracer.WriteLine("> Board Pcb: " + BoardPCB); + if (GameCartInfo.chip_type != null) + { + for (int i = 0; i < GameCartInfo.chip_type.Count; i++) + { + Console.WriteLine($"> CHIP {(i + 1).ToString()}: {GameCartInfo.chip_type[i]}"); + Chips.Add(GameCartInfo.chip_type[i]); + } + } + } + else + { + Tracer.WriteWarning("Game is not found in database ."); + } + } + Tracer.WriteLine("Initializing the board (Mapper # " + MapperNumber + ") ...."); + Tracer.WriteLine("Loading PRG ROM ..."); + PRG_AREA_BLK_RAM = new bool[16]; + PRG_AREA_BLK_INDEX = new int[16]; + PRG_ROM = new byte[0][]; + int num = 0; + for (int j = 0; j < rom.PRG.Length; j += 4096) + { + Array.Resize(ref PRG_ROM, PRG_ROM.GetLength(0) + 1); + PRG_ROM[num] = new byte[4096]; + for (int k = 0; k < 4096; k++) + { + PRG_ROM[num][k] = rom.PRG[j + k]; + } + num++; + } + PRG_ROM_04KB_Count = PRG_ROM.GetLength(0); + PRG_ROM_04KB_Mask = PRG_ROM_04KB_Count - 1; + PRG_ROM_08KB_Count = PRG_ROM_04KB_Count / 2; + PRG_ROM_08KB_Mask = PRG_ROM_08KB_Count - 1; + PRG_ROM_16KB_Count = PRG_ROM_04KB_Count / 4; + PRG_ROM_16KB_Mask = PRG_ROM_16KB_Count - 1; + PRG_ROM_32KB_Count = PRG_ROM_04KB_Count / 8; + PRG_ROM_32KB_Mask = PRG_ROM_32KB_Count - 1; + Tracer.WriteLine("PRG ROM loaded successfully."); + Tracer.WriteLine("PRG ROM Size = " + PRG_ROM_04KB_Count * 4 + "KB"); + Tracer.WriteLine("Loading PRG RAM ..."); + SRAMBankInfo[] pRGRAM8KCountFromDB = GetPRGRAM8KCountFromDB(); + PRG_RAM = new byte[0][]; + PRG_RAM_BATTERY = new bool[0]; + PRG_RAM_ENABLED = new bool[0]; + PRG_RAM_WRITABLE = new bool[0]; + SRAMBankInfo[] array = pRGRAM8KCountFromDB; + for (int l = 0; l < array.Length; l++) + { + SRAMBankInfo sRAMBankInfo = array[l]; + if (sRAMBankInfo.BATTERY) + { + SRAMSaveRequired = true; + } + int result = 0; + int.TryParse(sRAMBankInfo.SIZE.Replace("k", ""), out result); + if (result > 0) + { + int num2 = result / 2; + for (int m = 0; m < num2; m++) + { + Array.Resize(ref PRG_RAM_BATTERY, PRG_RAM_BATTERY.Length + 1); + Array.Resize(ref PRG_RAM_ENABLED, PRG_RAM_ENABLED.Length + 1); + Array.Resize(ref PRG_RAM_WRITABLE, PRG_RAM_WRITABLE.Length + 1); + Array.Resize(ref PRG_RAM, PRG_RAM.GetLength(0) + 1); + PRG_RAM[PRG_RAM.GetLength(0) - 1] = new byte[4096]; + PRG_RAM_BATTERY[PRG_RAM_BATTERY.Length - 1] = sRAMBankInfo.BATTERY; + PRG_RAM_ENABLED[PRG_RAM_ENABLED.Length - 1] = true; + PRG_RAM_WRITABLE[PRG_RAM_WRITABLE.Length - 1] = true; + } + } + } + PRG_RAM_04KB_Count = PRG_RAM.GetLength(0); + PRG_RAM_04KB_Mask = PRG_RAM_04KB_Count - 1; + PRG_RAM_08KB_Count = PRG_RAM_04KB_Count / 2; + PRG_RAM_08KB_Mask = PRG_RAM_08KB_Count - 1; + PRG_RAM_16KB_Count = PRG_RAM_04KB_Count / 4; + PRG_RAM_16KB_Mask = PRG_RAM_16KB_Count - 1; + PRG_RAM_32KB_Count = PRG_RAM_04KB_Count / 8; + PRG_RAM_32KB_Mask = PRG_RAM_32KB_Count - 1; + Tracer.WriteLine("PRG RAM loaded successfully."); + Tracer.WriteLine("PRG RAM Size = " + PRG_RAM_04KB_Count * 4 + "KB"); + if (rom.HasTrainer) + { + rom.Trainer.CopyTo(PRG_RAM[3], 0); + } + Tracer.WriteLine("Loading CHR ROM ..."); + CHR_ROM = new byte[0][]; + CHR_AREA_BLK_RAM = new bool[8]; + CHR_AREA_BLK_INDEX = new int[8]; + num = 0; + for (int n = 0; n < rom.CHR.Length; n += 1024) + { + Array.Resize(ref CHR_ROM, CHR_ROM.GetLength(0) + 1); + CHR_ROM[num] = new byte[1024]; + for (int num3 = 0; num3 < 1024; num3++) + { + CHR_ROM[num][num3] = rom.CHR[n + num3]; + } + num++; + } + CHR_ROM_01KB_Count = CHR_ROM.GetLength(0); + CHR_ROM_01KB_Mask = CHR_ROM_01KB_Count - 1; + CHR_ROM_02KB_Count = CHR_ROM_01KB_Count / 2; + CHR_ROM_02KB_Mask = CHR_ROM_02KB_Count - 1; + CHR_ROM_04KB_Count = CHR_ROM_01KB_Count / 4; + CHR_ROM_04KB_Mask = CHR_ROM_04KB_Count - 1; + CHR_ROM_08KB_Count = CHR_ROM_01KB_Count / 8; + CHR_ROM_08KB_Mask = CHR_ROM_08KB_Count - 1; + Tracer.WriteLine("CHR ROM loaded successfully."); + Tracer.WriteLine("CHR ROM Size = " + CHR_ROM_01KB_Count + "KB"); + Tracer.WriteLine("Loading CHR RAM ..."); + int cHRRAM1KCountFromDB = GetCHRRAM1KCountFromDB(); + CHR_RAM = new byte[0][]; + CHR_RAM_BATTERY = new bool[cHRRAM1KCountFromDB]; + CHR_RAM_ENABLED = new bool[cHRRAM1KCountFromDB]; + CHR_RAM_WRITABLE = new bool[cHRRAM1KCountFromDB]; + for (int num4 = 0; num4 < cHRRAM1KCountFromDB; num4++) + { + Array.Resize(ref CHR_RAM, CHR_RAM.GetLength(0) + 1); + CHR_RAM[num4] = new byte[1024]; + CHR_RAM_BATTERY[num4] = false; + CHR_RAM_ENABLED[num4] = true; + CHR_RAM_WRITABLE[num4] = true; + } + CHR_RAM_01KB_Count = CHR_RAM.GetLength(0); + CHR_RAM_01KB_Mask = CHR_RAM_01KB_Count - 1; + CHR_RAM_02KB_Count = CHR_RAM_01KB_Count / 2; + CHR_RAM_02KB_Mask = CHR_RAM_02KB_Count - 1; + CHR_RAM_04KB_Count = CHR_RAM_01KB_Count / 4; + CHR_RAM_04KB_Mask = CHR_RAM_04KB_Count - 1; + CHR_RAM_08KB_Count = CHR_RAM_01KB_Count / 8; + CHR_RAM_08KB_Mask = CHR_RAM_08KB_Count - 1; + Tracer.WriteLine("CHR RAM loaded successfully."); + Tracer.WriteLine("CHR RAM Size = " + CHR_RAM_01KB_Count + "KB"); + Tracer.WriteLine("Loading Nametables ..."); + NMT_AREA_BLK_INDEX = new int[4]; + NMT_RAM = new byte[0][]; + for (int num5 = 0; num5 < 4; num5++) + { + Array.Resize(ref NMT_RAM, NMT_RAM.GetLength(0) + 1); + NMT_RAM[num5] = new byte[1024]; + } + NMT_DEFAULT_MIRROR = rom.Mirroring; + Tracer.WriteLine("Mirroring set to " + NMT_DEFAULT_MIRROR); + Tracer.WriteLine("Board (Mapper # " + MapperNumber + ") initialized successfully."); + } + + internal virtual void HardReset() + { + Tracer.WriteLine("Hard reset board (Mapper # " + MapperNumber + ") ...."); + Tracer.WriteLine("Switching 16KB PRG RAM at 0x4000 - 0x7000"); + Toggle16KPRG_RAM(ram: true, PRGArea.Area4000); + Switch16KPRG(0, PRGArea.Area4000); + Tracer.WriteLine("Switching 32KB PRG ROM at 0x8000 - 0xF000"); + Toggle32KPRG_RAM(ram: false, PRGArea.Area8000); + Switch32KPRG(0, PRGArea.Area8000); + Tracer.WriteLine("Switching 8KB CHR " + ((CHR_ROM_01KB_Count == 0) ? "RAM" : "ROM") + " at 0x0000 - 0x1000"); + Toggle08KCHR_RAM(CHR_ROM_01KB_Count == 0); + Switch08KCHR(0); + Tracer.WriteLine("Switching to mirroring: " + NMT_DEFAULT_MIRROR); + Switch01KNMTFromMirroring(NMT_DEFAULT_MIRROR); + Tracer.WriteLine("Hard reset board (Mapper # " + MapperNumber + ") is done successfully."); + } + + internal virtual void SoftReset() + { + } + + protected virtual void LoadAttrs() + { + enable_external_sound = false; + Attribute[] customAttributes = Attribute.GetCustomAttributes(GetType()); + foreach (Attribute attribute in customAttributes) + { + if (attribute.GetType() == typeof(BoardInfoAttribute)) + { + BoardInfoAttribute boardInfoAttribute = (BoardInfoAttribute)attribute; + Name = boardInfoAttribute.Name; + MapperNumber = boardInfoAttribute.Mapper; + PRG_RAM_08KB_DEFAULT_BLK_Count = boardInfoAttribute.DefaultPRG_RAM_8KB_BanksCount; + CHR_ROM_01KB_DEFAULT_BLK_Count = boardInfoAttribute.DefaultCHR_RAM_1KB_BanksCount; + enabled_ppuA12ToggleTimer = boardInfoAttribute.Enabled_ppuA12ToggleTimer; + ppuA12TogglesOnRaisingEdge = boardInfoAttribute.PPUA12TogglesOnRaisingEdge; + } + else if (attribute.GetType() == typeof(WithExternalSoundAttribute)) + { + enable_external_sound = true; + } + else if (attribute.GetType() == typeof(HassIssuesAttribute)) + { + HasIssues = true; + } + } + } + + protected SRAMBankInfo[] GetPRGRAM8KCountFromDB() + { + Tracer.WriteLine("Retrieving PRG RAM information from database ...."); + List list = new List(); + if (IsGameFoundOnDB) + { + if (GameCartInfo.WRAMBanks.Count > 0) + { + foreach (SRAMBankInfo wRAMBank in GameCartInfo.WRAMBanks) + { + list.Add(wRAMBank); + } + } + else + { + Tracer.WriteLine("This game has no PRG RAM !"); + Tracer.WriteWarning("> Adding 8K x " + PRG_RAM_08KB_DEFAULT_BLK_Count + " PRG RAM BANKS to avoid exceptions."); + SRAMBankInfo item = new SRAMBankInfo(0, PRG_RAM_08KB_DEFAULT_BLK_Count * 8 + "k", BATTERY: true); + list.Add(item); + } + } + else + { + Tracer.WriteWarning("Could't find this game in database .... Adding 8K x " + PRG_RAM_08KB_DEFAULT_BLK_Count + " PRG RAM BANKS to avoid exceptions."); + SRAMBankInfo item2 = new SRAMBankInfo(0, PRG_RAM_08KB_DEFAULT_BLK_Count * 8 + "k", BATTERY: true); + list.Add(item2); + } + return list.ToArray(); + } + + protected int GetCHRRAM1KCountFromDB() + { + int num = 0; + Tracer.WriteLine("Retrieving CHR RAM information from database ...."); + if (IsGameFoundOnDB) + { + bool flag = false; + if (GameCartInfo.VRAM_sizes != null) + { + Tracer.WriteLine("Using database to initialize CHR RAM ....."); + foreach (string vRAM_size in GameCartInfo.VRAM_sizes) + { + int result = 0; + if (int.TryParse(vRAM_size.Replace("k", ""), out result)) + { + Tracer.WriteLine(">CHR RAM CHIP SIZE " + vRAM_size + " KB added"); + num += result; + if (num > 0) + { + flag = true; + } + } + } + } + if (!flag) + { + Tracer.WriteLine("Game not found in database to initialize CHR RAM; CHR RAM size set to " + CHR_ROM_01KB_DEFAULT_BLK_Count + " KB"); + num = CHR_ROM_01KB_DEFAULT_BLK_Count; + } + } + else + { + Tracer.WriteWarning("Game not found in database to initialize CHR RAM; CHR RAM size set to " + CHR_ROM_01KB_DEFAULT_BLK_Count + " KB"); + num = CHR_ROM_01KB_DEFAULT_BLK_Count; + } + return num; + } + + internal virtual void WriteEX(ref ushort addr, ref byte val) + { + PRG_TMP_AREA = (addr >> 12) & 0xF; + if (PRG_AREA_BLK_RAM[PRG_TMP_AREA]) + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_RAM_04KB_Mask; + if (PRG_RAM_ENABLED[PRG_TMP_INDX] && PRG_RAM_WRITABLE[PRG_TMP_INDX]) + { + PRG_RAM[PRG_TMP_INDX][addr & 0xFFF] = val; + } + } + } + + internal virtual void WriteSRM(ref ushort addr, ref byte val) + { + PRG_TMP_AREA = (addr >> 12) & 0xF; + if (PRG_AREA_BLK_RAM[PRG_TMP_AREA]) + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_RAM_04KB_Mask; + if (PRG_RAM_ENABLED[PRG_TMP_INDX] && PRG_RAM_WRITABLE[PRG_TMP_INDX]) + { + PRG_RAM[PRG_TMP_INDX][addr & 0xFFF] = val; + } + } + } + + internal virtual void WritePRG(ref ushort addr, ref byte val) + { + PRG_TMP_AREA = (addr >> 12) & 0xF; + if (PRG_AREA_BLK_RAM[PRG_TMP_AREA]) + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_RAM_04KB_Mask; + if (PRG_RAM_ENABLED[PRG_TMP_INDX] && PRG_RAM_WRITABLE[PRG_TMP_INDX]) + { + PRG_RAM[PRG_TMP_INDX][addr & 0xFFF] = val; + } + } + } + + internal virtual void ReadEX(ref ushort addr, out byte val) + { + PRG_TMP_AREA = (addr >> 12) & 0xF; + if (PRG_AREA_BLK_RAM[PRG_TMP_AREA]) + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_RAM_04KB_Mask; + if (PRG_RAM_ENABLED[PRG_TMP_INDX]) + { + val = PRG_RAM[PRG_TMP_INDX][addr & 0xFFF]; + } + else + { + val = 0; + } + } + else + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_ROM_04KB_Mask; + val = PRG_ROM[PRG_TMP_INDX][addr & 0xFFF]; + } + } + + internal virtual void ReadSRM(ref ushort addr, out byte val) + { + PRG_TMP_AREA = (addr >> 12) & 0xF; + if (PRG_AREA_BLK_RAM[PRG_TMP_AREA]) + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_RAM_04KB_Mask; + if (PRG_RAM_ENABLED[PRG_TMP_INDX]) + { + val = PRG_RAM[PRG_TMP_INDX][addr & 0xFFF]; + } + else + { + val = 0; + } + } + else + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_ROM_04KB_Mask; + val = PRG_ROM[PRG_TMP_INDX][addr & 0xFFF]; + } + } + + internal virtual void ReadPRG(ref ushort addr, out byte val) + { + PRG_TMP_AREA = (addr >> 12) & 0xF; + if (PRG_AREA_BLK_RAM[PRG_TMP_AREA]) + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_RAM_04KB_Mask; + if (PRG_RAM_ENABLED[PRG_TMP_INDX]) + { + val = PRG_RAM[PRG_TMP_INDX][addr & 0xFFF]; + } + else + { + val = 0; + } + } + else + { + PRG_TMP_INDX = PRG_AREA_BLK_INDEX[PRG_TMP_AREA] & PRG_ROM_04KB_Mask; + val = PRG_ROM[PRG_TMP_INDX][addr & 0xFFF]; + } + if (!IsGameGenieActive) + { + return; + } + GameGenieCode[] gameGenieCodes = GameGenieCodes; + for (int i = 0; i < gameGenieCodes.Length; i++) + { + GameGenieCode gameGenieCode = gameGenieCodes[i]; + if (!gameGenieCode.Enabled || gameGenieCode.Address != addr) + { + continue; + } + if (gameGenieCode.IsCompare) + { + if (gameGenieCode.Compare == val) + { + val = gameGenieCode.Value; + } + } + else + { + val = gameGenieCode.Value; + } + break; + } + } + + internal virtual void WriteCHR(ref ushort addr, ref byte val) + { + CHR_TMP_AREA = (addr >> 10) & 7; + if (CHR_AREA_BLK_RAM[CHR_TMP_AREA]) + { + CHR_TMP_INDX = CHR_AREA_BLK_INDEX[CHR_TMP_AREA] & CHR_RAM_01KB_Mask; + if (CHR_RAM_ENABLED[CHR_TMP_INDX] && CHR_RAM_WRITABLE[CHR_TMP_INDX]) + { + CHR_RAM[CHR_TMP_INDX][addr & 0x3FF] = val; + } + } + } + + internal virtual void ReadCHR(ref ushort addr, out byte val) + { + CHR_TMP_AREA = (addr >> 10) & 7; + CHR_TMP_INDX = CHR_AREA_BLK_INDEX[CHR_TMP_AREA]; + if (CHR_AREA_BLK_RAM[CHR_TMP_AREA]) + { + CHR_TMP_INDX &= CHR_RAM_01KB_Mask; + if (CHR_RAM_ENABLED[CHR_TMP_INDX]) + { + val = CHR_RAM[CHR_TMP_INDX][addr & 0x3FF]; + } + else + { + val = 0; + } + } + else + { + CHR_TMP_INDX &= CHR_ROM_01KB_Mask; + val = CHR_ROM[CHR_TMP_INDX][addr & 0x3FF]; + } + } + + internal virtual void WriteNMT(ref ushort addr, ref byte val) + { + NMT_TMP_AREA = (addr >> 10) & 3; + NMT_TMP_INDX = NMT_AREA_BLK_INDEX[NMT_TMP_AREA]; + NMT_RAM[NMT_TMP_INDX][addr & 0x3FF] = val; + } + + internal virtual void ReadNMT(ref ushort addr, out byte val) + { + NMT_TMP_AREA = (addr >> 10) & 3; + NMT_TMP_INDX = NMT_AREA_BLK_INDEX[NMT_TMP_AREA]; + val = NMT_RAM[NMT_TMP_INDX][addr & 0x3FF]; + } + + protected void Switch04KPRG(int index, PRGArea area) + { + PRG_AREA_BLK_INDEX[(uint)area] = index; + } + + protected void Switch08KPRG(int index, PRGArea area) + { + index *= 2; + PRG_AREA_BLK_INDEX[(uint)area] = index; + PRG_AREA_BLK_INDEX[(uint)(area + 1)] = index + 1; + } + + protected void Switch16KPRG(int index, PRGArea area) + { + index *= 4; + PRG_AREA_BLK_INDEX[(uint)area] = index; + PRG_AREA_BLK_INDEX[(uint)(area + 1)] = index + 1; + PRG_AREA_BLK_INDEX[(uint)(area + 2)] = index + 2; + PRG_AREA_BLK_INDEX[(uint)(area + 3)] = index + 3; + } + + protected void Switch32KPRG(int index, PRGArea area) + { + index *= 8; + PRG_AREA_BLK_INDEX[(uint)area] = index; + PRG_AREA_BLK_INDEX[(uint)(area + 1)] = index + 1; + PRG_AREA_BLK_INDEX[(uint)(area + 2)] = index + 2; + PRG_AREA_BLK_INDEX[(uint)(area + 3)] = index + 3; + PRG_AREA_BLK_INDEX[(uint)(area + 4)] = index + 4; + PRG_AREA_BLK_INDEX[(uint)(area + 5)] = index + 5; + PRG_AREA_BLK_INDEX[(uint)(area + 6)] = index + 6; + PRG_AREA_BLK_INDEX[(uint)(area + 7)] = index + 7; + } + + protected void Toggle04KPRG_RAM(bool ram, PRGArea area) + { + PRG_AREA_BLK_RAM[(uint)area] = ram; + } + + protected void Toggle08KPRG_RAM(bool ram, PRGArea area) + { + PRG_AREA_BLK_RAM[(uint)area] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 1)] = ram; + } + + protected void Toggle16KPRG_RAM(bool ram, PRGArea area) + { + PRG_AREA_BLK_RAM[(uint)area] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 1)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 2)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 3)] = ram; + } + + protected void Toggle32KPRG_RAM(bool ram, PRGArea area) + { + PRG_AREA_BLK_RAM[(uint)area] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 1)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 2)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 3)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 4)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 5)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 6)] = ram; + PRG_AREA_BLK_RAM[(uint)(area + 7)] = ram; + } + + protected void TogglePRGRAMEnable(bool enable) + { + for (int i = 0; i < PRG_RAM_ENABLED.Length; i++) + { + PRG_RAM_ENABLED[i] = enable; + } + } + + protected void TogglePRGRAMWritableEnable(bool enable) + { + for (int i = 0; i < PRG_RAM_WRITABLE.Length; i++) + { + PRG_RAM_WRITABLE[i] = enable; + } + } + + protected void Toggle04KPRG_RAM_Enabled(bool enable, int index) + { + PRG_RAM_ENABLED[index] = enable; + } + + protected void Toggle04KPRG_RAM_Writable(bool enable, int index) + { + PRG_RAM_WRITABLE[index] = enable; + } + + protected void Toggle04KPRG_RAM_Battery(bool enable, int index) + { + PRG_RAM_BATTERY[index] = enable; + } + + protected void Switch01KCHR(int index, CHRArea area) + { + CHR_AREA_BLK_INDEX[(uint)area] = index; + } + + protected void Switch02KCHR(int index, CHRArea area) + { + index *= 2; + CHR_AREA_BLK_INDEX[(uint)area] = index; + CHR_AREA_BLK_INDEX[(uint)(area + 1)] = index + 1; + } + + protected void Switch04KCHR(int index, CHRArea area) + { + index *= 4; + CHR_AREA_BLK_INDEX[(uint)area] = index; + CHR_AREA_BLK_INDEX[(uint)(area + 1)] = index + 1; + CHR_AREA_BLK_INDEX[(uint)(area + 2)] = index + 2; + CHR_AREA_BLK_INDEX[(uint)(area + 3)] = index + 3; + } + + protected void Switch08KCHR(int index) + { + index *= 8; + CHR_AREA_BLK_INDEX[0] = index; + CHR_AREA_BLK_INDEX[1] = index + 1; + CHR_AREA_BLK_INDEX[2] = index + 2; + CHR_AREA_BLK_INDEX[3] = index + 3; + CHR_AREA_BLK_INDEX[4] = index + 4; + CHR_AREA_BLK_INDEX[5] = index + 5; + CHR_AREA_BLK_INDEX[6] = index + 6; + CHR_AREA_BLK_INDEX[7] = index + 7; + } + + protected void Toggle01KCHR_RAM(bool ram, CHRArea area) + { + CHR_AREA_BLK_RAM[(uint)area] = ram; + } + + protected void Toggle02KCHR_RAM(bool ram, CHRArea area) + { + CHR_AREA_BLK_RAM[(uint)area] = ram; + CHR_AREA_BLK_RAM[(uint)(area + 1)] = ram; + } + + protected void Toggle04KCHR_RAM(bool ram, CHRArea area) + { + CHR_AREA_BLK_RAM[(uint)area] = ram; + CHR_AREA_BLK_RAM[(uint)(area + 1)] = ram; + CHR_AREA_BLK_RAM[(uint)(area + 2)] = ram; + CHR_AREA_BLK_RAM[(uint)(area + 3)] = ram; + } + + protected void Toggle08KCHR_RAM(bool ram) + { + CHR_AREA_BLK_RAM[0] = ram; + CHR_AREA_BLK_RAM[1] = ram; + CHR_AREA_BLK_RAM[2] = ram; + CHR_AREA_BLK_RAM[3] = ram; + CHR_AREA_BLK_RAM[4] = ram; + CHR_AREA_BLK_RAM[5] = ram; + CHR_AREA_BLK_RAM[6] = ram; + CHR_AREA_BLK_RAM[7] = ram; + } + + protected void Toggle01KCHR_RAM_Enabled(bool enable, int index) + { + CHR_RAM_ENABLED[index] = enable; + } + + protected void Toggle01KCHR_RAM_Writable(bool enable, int index) + { + CHR_RAM_WRITABLE[index] = enable; + } + + protected void ToggleCHRRAMWritableEnable(bool enable) + { + for (int i = 0; i < CHR_RAM_WRITABLE.Length; i++) + { + CHR_RAM_WRITABLE[i] = enable; + } + } + + protected void Toggle01KCHR_RAM_Battery(bool enable, int index) + { + CHR_RAM_BATTERY[index] = enable; + } + + protected void Switch01KNMT(int index, byte area) + { + NMT_AREA_BLK_INDEX[area] = index; + } + + protected void Switch01KNMT(byte mirroring) + { + NMT_AREA_BLK_INDEX[0] = mirroring & 3; + NMT_AREA_BLK_INDEX[1] = (mirroring >> 2) & 3; + NMT_AREA_BLK_INDEX[2] = (mirroring >> 4) & 3; + NMT_AREA_BLK_INDEX[3] = (mirroring >> 6) & 3; + } + + protected void Switch01KNMTFromMirroring(Mirroring mirroring) + { + NMT_AREA_BLK_INDEX[0] = (int)(mirroring & (Mirroring)3); + NMT_AREA_BLK_INDEX[1] = ((int)mirroring >> 2) & 3; + NMT_AREA_BLK_INDEX[2] = ((int)mirroring >> 4) & 3; + NMT_AREA_BLK_INDEX[3] = ((int)mirroring >> 6) & 3; + } + + internal virtual void OnPPUAddressUpdate(ref ushort address) + { + if (!enabled_ppuA12ToggleTimer) + { + return; + } + old_vram_address = new_vram_address; + new_vram_address = address & 0x1000; + if (ppuA12TogglesOnRaisingEdge) + { + if (old_vram_address < new_vram_address) + { + if (ppu_cycles_timer > 8) + { + OnPPUA12RaisingEdge(); + } + ppu_cycles_timer = 0; + } + } + else if (old_vram_address > new_vram_address) + { + if (ppu_cycles_timer > 8) + { + OnPPUA12RaisingEdge(); + } + ppu_cycles_timer = 0; + } + } + + internal virtual void OnCPUClock() + { + } + + internal virtual void OnPPUClock() + { + if (enabled_ppuA12ToggleTimer) + { + ppu_cycles_timer++; + } + } + + internal virtual void OnPPUA12RaisingEdge() + { + } + + internal virtual void OnPPUScanlineTick() + { + } + + internal virtual void OnAPUClockDuration() + { + } + + internal virtual void OnAPUClockEnvelope() + { + } + + internal virtual void OnAPUClockSingle() + { + } + + internal virtual void OnAPUClock() + { + } + + internal virtual double APUGetSample() + { + return 0.0; + } + + internal virtual void APUApplyChannelsSettings() + { + } + + internal void SetupGameGenie(bool IsGameGenieActive, GameGenieCode[] GameGenieCodes) + { + this.IsGameGenieActive = IsGameGenieActive; + this.GameGenieCodes = GameGenieCodes; + } + + internal virtual void WriteStateData(ref BinaryWriter bin) + { + for (int i = 0; i < PRG_RAM.Length; i++) + { + bin.Write(PRG_RAM[i]); + } + for (int j = 0; j < PRG_RAM_ENABLED.Length; j++) + { + bin.Write(PRG_RAM_ENABLED[j]); + } + for (int k = 0; k < PRG_RAM_WRITABLE.Length; k++) + { + bin.Write(PRG_RAM_WRITABLE[k]); + } + for (int l = 0; l < PRG_RAM_BATTERY.Length; l++) + { + bin.Write(PRG_RAM_BATTERY[l]); + } + for (int m = 0; m < PRG_AREA_BLK_RAM.Length; m++) + { + bin.Write(PRG_AREA_BLK_RAM[m]); + } + for (int n = 0; n < PRG_AREA_BLK_INDEX.Length; n++) + { + bin.Write(PRG_AREA_BLK_INDEX[n]); + } + bin.Write(PRG_TMP_INDX); + bin.Write(PRG_TMP_AREA); + for (int num = 0; num < CHR_RAM.Length; num++) + { + bin.Write(CHR_RAM[num]); + } + for (int num2 = 0; num2 < CHR_RAM_ENABLED.Length; num2++) + { + bin.Write(CHR_RAM_ENABLED[num2]); + } + for (int num3 = 0; num3 < CHR_RAM_WRITABLE.Length; num3++) + { + bin.Write(CHR_RAM_WRITABLE[num3]); + } + for (int num4 = 0; num4 < CHR_RAM_BATTERY.Length; num4++) + { + bin.Write(CHR_RAM_BATTERY[num4]); + } + for (int num5 = 0; num5 < CHR_AREA_BLK_RAM.Length; num5++) + { + bin.Write(CHR_AREA_BLK_RAM[num5]); + } + for (int num6 = 0; num6 < CHR_AREA_BLK_INDEX.Length; num6++) + { + bin.Write(CHR_AREA_BLK_INDEX[num6]); + } + bin.Write(CHR_TMP_INDX); + bin.Write(CHR_TMP_AREA); + for (int num7 = 0; num7 < NMT_RAM.Length; num7++) + { + bin.Write(NMT_RAM[num7]); + } + for (int num8 = 0; num8 < NMT_AREA_BLK_INDEX.Length; num8++) + { + bin.Write(NMT_AREA_BLK_INDEX[num8]); + } + bin.Write(NMT_TMP_INDX); + bin.Write(NMT_TMP_AREA); + } + + internal virtual void ReadStateData(ref BinaryReader bin) + { + for (int i = 0; i < PRG_RAM.Length; i++) + { + bin.Read(PRG_RAM[i], 0, PRG_RAM[i].Length); + } + for (int j = 0; j < PRG_RAM_ENABLED.Length; j++) + { + PRG_RAM_ENABLED[j] = bin.ReadBoolean(); + } + for (int k = 0; k < PRG_RAM_WRITABLE.Length; k++) + { + PRG_RAM_WRITABLE[k] = bin.ReadBoolean(); + } + for (int l = 0; l < PRG_RAM_BATTERY.Length; l++) + { + PRG_RAM_BATTERY[l] = bin.ReadBoolean(); + } + for (int m = 0; m < PRG_AREA_BLK_RAM.Length; m++) + { + PRG_AREA_BLK_RAM[m] = bin.ReadBoolean(); + } + for (int n = 0; n < PRG_AREA_BLK_INDEX.Length; n++) + { + PRG_AREA_BLK_INDEX[n] = bin.ReadInt32(); + } + PRG_TMP_INDX = bin.ReadInt32(); + PRG_TMP_AREA = bin.ReadInt32(); + for (int num = 0; num < CHR_RAM.Length; num++) + { + bin.Read(CHR_RAM[num], 0, CHR_RAM[num].Length); + } + for (int num2 = 0; num2 < CHR_RAM_ENABLED.Length; num2++) + { + CHR_RAM_ENABLED[num2] = bin.ReadBoolean(); + } + for (int num3 = 0; num3 < CHR_RAM_WRITABLE.Length; num3++) + { + CHR_RAM_WRITABLE[num3] = bin.ReadBoolean(); + } + for (int num4 = 0; num4 < CHR_RAM_BATTERY.Length; num4++) + { + CHR_RAM_BATTERY[num4] = bin.ReadBoolean(); + } + for (int num5 = 0; num5 < CHR_AREA_BLK_RAM.Length; num5++) + { + CHR_AREA_BLK_RAM[num5] = bin.ReadBoolean(); + } + for (int num6 = 0; num6 < CHR_AREA_BLK_INDEX.Length; num6++) + { + CHR_AREA_BLK_INDEX[num6] = bin.ReadInt32(); + } + CHR_TMP_INDX = bin.ReadInt32(); + CHR_TMP_AREA = bin.ReadInt32(); + for (int num7 = 0; num7 < NMT_RAM.Length; num7++) + { + bin.Read(NMT_RAM[num7], 0, NMT_RAM[num7].Length); + } + for (int num8 = 0; num8 < NMT_AREA_BLK_INDEX.Length; num8++) + { + NMT_AREA_BLK_INDEX[num8] = bin.ReadInt32(); + } + NMT_TMP_INDX = bin.ReadInt32(); + NMT_TMP_AREA = bin.ReadInt32(); + } + + internal void SaveSRAM(Stream stream) + { + for (int i = 0; i < PRG_RAM_04KB_Count; i++) + { + if (PRG_RAM_BATTERY[i]) + { + stream.Write(PRG_RAM[i], 0, 4096); + } + } + } + + internal byte[] GetSRAMBuffer() + { + List list = new List(); + for (int i = 0; i < PRG_RAM_04KB_Count; i++) + { + if (PRG_RAM_BATTERY[i]) + { + list.AddRange(PRG_RAM[i]); + } + } + return list.ToArray(); + } + + internal void LoadSRAM(Stream stream) + { + for (int i = 0; i < PRG_RAM_04KB_Count; i++) + { + if (PRG_RAM_BATTERY[i]) + { + stream.Read(PRG_RAM[i], 0, 4096); + } + } + } + + internal void LoadSRAM(byte[] buffer) + { + int num = 0; + for (int i = 0; i < PRG_RAM_04KB_Count; i++) + { + if (PRG_RAM_BATTERY[i]) + { + for (int j = 0; j < 4096; j++) + { + PRG_RAM[i][j] = buffer[j + num]; + } + num += 4096; + } + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs.meta new file mode 100644 index 00000000..633023e3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Board.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a649aa2f72c13234481b8a2db2b101f8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs new file mode 100644 index 00000000..f128856f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs @@ -0,0 +1,57 @@ +using System; + +namespace MyNes.Core +{ + internal class BoardInfoAttribute : Attribute + { + public string Name { get; private set; } + + public int Mapper { get; private set; } + + public int DefaultPRG_RAM_8KB_BanksCount { get; private set; } + + public int DefaultCHR_RAM_1KB_BanksCount { get; private set; } + + public bool Enabled_ppuA12ToggleTimer { get; private set; } + + public bool PPUA12TogglesOnRaisingEdge { get; private set; } + + public BoardInfoAttribute(string boardName, int inesMapperNumber) + { + Name = boardName; + Mapper = inesMapperNumber; + DefaultPRG_RAM_8KB_BanksCount = 1; + DefaultCHR_RAM_1KB_BanksCount = 8; + Enabled_ppuA12ToggleTimer = (PPUA12TogglesOnRaisingEdge = false); + } + + public BoardInfoAttribute(string boardName, int inesMapperNumber, int defaultPRG_RAM_8KB_BanksCount, int defaultCHR_RAM_1KB_BanksCount) + { + Name = boardName; + Mapper = inesMapperNumber; + DefaultPRG_RAM_8KB_BanksCount = defaultPRG_RAM_8KB_BanksCount; + DefaultCHR_RAM_1KB_BanksCount = defaultCHR_RAM_1KB_BanksCount; + Enabled_ppuA12ToggleTimer = (PPUA12TogglesOnRaisingEdge = false); + } + + public BoardInfoAttribute(string boardName, int inesMapperNumber, bool Enabled_ppuA12ToggleTimer, bool PPUA12TogglesOnRaisingEdge) + { + Name = boardName; + Mapper = inesMapperNumber; + DefaultPRG_RAM_8KB_BanksCount = 1; + DefaultCHR_RAM_1KB_BanksCount = 8; + this.Enabled_ppuA12ToggleTimer = Enabled_ppuA12ToggleTimer; + this.PPUA12TogglesOnRaisingEdge = PPUA12TogglesOnRaisingEdge; + } + + public BoardInfoAttribute(string boardName, int inesMapperNumber, int defaultPRG_RAM_8KB_BanksCount, int defaultCHR_RAM_1KB_BanksCount, bool Enabled_ppuA12ToggleTimer, bool PPUA12TogglesOnRaisingEdge) + { + Name = boardName; + Mapper = inesMapperNumber; + DefaultPRG_RAM_8KB_BanksCount = defaultPRG_RAM_8KB_BanksCount; + DefaultCHR_RAM_1KB_BanksCount = defaultCHR_RAM_1KB_BanksCount; + this.Enabled_ppuA12ToggleTimer = Enabled_ppuA12ToggleTimer; + this.PPUA12TogglesOnRaisingEdge = PPUA12TogglesOnRaisingEdge; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs.meta new file mode 100644 index 00000000..1b46dc56 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoAttribute.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 42db8567eb222464faed6061346980df +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs new file mode 100644 index 00000000..a18d060b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs @@ -0,0 +1,15 @@ +namespace MyNes.Core +{ + public class BoardInfoObject + { + public string Name { get; internal set; } + + public int MapperNumber { get; internal set; } + + public bool IsSupported { get; internal set; } + + public string Issues { get; internal set; } + + public bool HasIssues { get; internal set; } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs.meta new file mode 100644 index 00000000..35a33db8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/BoardInfoObject.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 02faf494147d32b4d9288978c3ae6145 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs new file mode 100644 index 00000000..2d66b9ae --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs @@ -0,0 +1,14 @@ +namespace MyNes.Core +{ + internal enum CHRArea : byte + { + Area0000, + Area0400, + Area0800, + Area0C00, + Area1000, + Area1400, + Area1800, + Area1C00 + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs.meta new file mode 100644 index 00000000..6fd0dbed --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/CHRArea.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 682bde94b329bbe40bca1f57566c3bdd +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib.meta new file mode 100644 index 00000000..05c18cfe --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib.meta @@ -0,0 +1,8 @@ +fileFormatVersion: 2 +guid: d8b8249d0e1fe314787bfc6d8a22318f +folderAsset: yes +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs new file mode 100644 index 00000000..82658346 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs @@ -0,0 +1,72 @@ +namespace ComponentAce.Compression.Libs.zlib +{ + internal sealed class Adler32 + { + private const int BASE = 65521; + + private const int NMAX = 5552; + + internal long adler32(long adler, byte[] buf, int index, int len) + { + if (buf == null) + { + return 1L; + } + long num = adler & 0xFFFF; + long num2 = (adler >> 16) & 0xFFFF; + while (len > 0) + { + int num3 = ((len < 5552) ? len : 5552); + len -= num3; + while (num3 >= 16) + { + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num += buf[index++] & 0xFF; + num2 += num; + num3 -= 16; + } + if (num3 != 0) + { + do + { + num += buf[index++] & 0xFF; + num2 += num; + } + while (--num3 != 0); + } + num %= 65521; + num2 %= 65521; + } + return (num2 << 16) | num; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs.meta new file mode 100644 index 00000000..17aacb18 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Adler32.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5f0afef237e01da488c3afdb23b495d4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs new file mode 100644 index 00000000..ce31c52c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs @@ -0,0 +1,1379 @@ +using System; + +namespace ComponentAce.Compression.Libs.zlib +{ + public sealed class Deflate + { + internal class Config + { + internal int good_length; + + internal int max_lazy; + + internal int nice_length; + + internal int max_chain; + + internal int func; + + internal Config(int good_length, int max_lazy, int nice_length, int max_chain, int func) + { + this.good_length = good_length; + this.max_lazy = max_lazy; + this.nice_length = nice_length; + this.max_chain = max_chain; + this.func = func; + } + } + + private const int MAX_MEM_LEVEL = 9; + + private const int Z_DEFAULT_COMPRESSION = -1; + + private const int MAX_WBITS = 15; + + private const int DEF_MEM_LEVEL = 8; + + private const int STORED = 0; + + private const int FAST = 1; + + private const int SLOW = 2; + + private static Config[] config_table; + + private static readonly string[] z_errmsg; + + private const int NeedMore = 0; + + private const int BlockDone = 1; + + private const int FinishStarted = 2; + + private const int FinishDone = 3; + + private const int PRESET_DICT = 32; + + private const int Z_FILTERED = 1; + + private const int Z_HUFFMAN_ONLY = 2; + + private const int Z_DEFAULT_STRATEGY = 0; + + private const int Z_NO_FLUSH = 0; + + private const int Z_PARTIAL_FLUSH = 1; + + private const int Z_SYNC_FLUSH = 2; + + private const int Z_FULL_FLUSH = 3; + + private const int Z_FINISH = 4; + + private const int Z_OK = 0; + + private const int Z_STREAM_END = 1; + + private const int Z_NEED_DICT = 2; + + private const int Z_ERRNO = -1; + + private const int Z_STREAM_ERROR = -2; + + private const int Z_DATA_ERROR = -3; + + private const int Z_MEM_ERROR = -4; + + private const int Z_BUF_ERROR = -5; + + private const int Z_VERSION_ERROR = -6; + + private const int INIT_STATE = 42; + + private const int BUSY_STATE = 113; + + private const int FINISH_STATE = 666; + + private const int Z_DEFLATED = 8; + + private const int STORED_BLOCK = 0; + + private const int STATIC_TREES = 1; + + private const int DYN_TREES = 2; + + private const int Z_BINARY = 0; + + private const int Z_ASCII = 1; + + private const int Z_UNKNOWN = 2; + + private const int Buf_size = 16; + + private const int REP_3_6 = 16; + + private const int REPZ_3_10 = 17; + + private const int REPZ_11_138 = 18; + + private const int MIN_MATCH = 3; + + private const int MAX_MATCH = 258; + + private static readonly int MIN_LOOKAHEAD; + + private const int MAX_BITS = 15; + + private const int D_CODES = 30; + + private const int BL_CODES = 19; + + private const int LENGTH_CODES = 29; + + private const int LITERALS = 256; + + private static readonly int L_CODES; + + private static readonly int HEAP_SIZE; + + private const int END_BLOCK = 256; + + internal ZStream strm; + + internal int status; + + internal byte[] pending_buf; + + internal int pending_buf_size; + + internal int pending_out; + + internal int pending; + + internal int noheader; + + internal byte data_type; + + internal byte method; + + internal int last_flush; + + internal int w_size; + + internal int w_bits; + + internal int w_mask; + + internal byte[] window; + + internal int window_size; + + internal short[] prev; + + internal short[] head; + + internal int ins_h; + + internal int hash_size; + + internal int hash_bits; + + internal int hash_mask; + + internal int hash_shift; + + internal int block_start; + + internal int match_length; + + internal int prev_match; + + internal int match_available; + + internal int strstart; + + internal int match_start; + + internal int lookahead; + + internal int prev_length; + + internal int max_chain_length; + + internal int max_lazy_match; + + internal int level; + + internal int strategy; + + internal int good_match; + + internal int nice_match; + + internal short[] dyn_ltree; + + internal short[] dyn_dtree; + + internal short[] bl_tree; + + internal Tree l_desc = new Tree(); + + internal Tree d_desc = new Tree(); + + internal Tree bl_desc = new Tree(); + + internal short[] bl_count = new short[16]; + + internal int[] heap = new int[2 * L_CODES + 1]; + + internal int heap_len; + + internal int heap_max; + + internal byte[] depth = new byte[2 * L_CODES + 1]; + + internal int l_buf; + + internal int lit_bufsize; + + internal int last_lit; + + internal int d_buf; + + internal int opt_len; + + internal int static_len; + + internal int matches; + + internal int last_eob_len; + + internal short bi_buf; + + internal int bi_valid; + + internal Deflate() + { + dyn_ltree = new short[HEAP_SIZE * 2]; + dyn_dtree = new short[122]; + bl_tree = new short[78]; + } + + internal void lm_init() + { + window_size = 2 * w_size; + head[hash_size - 1] = 0; + for (int i = 0; i < hash_size - 1; i++) + { + head[i] = 0; + } + max_lazy_match = config_table[level].max_lazy; + good_match = config_table[level].good_length; + nice_match = config_table[level].nice_length; + max_chain_length = config_table[level].max_chain; + strstart = 0; + block_start = 0; + lookahead = 0; + match_length = (prev_length = 2); + match_available = 0; + ins_h = 0; + } + + internal void tr_init() + { + l_desc.dyn_tree = dyn_ltree; + l_desc.stat_desc = StaticTree.static_l_desc; + d_desc.dyn_tree = dyn_dtree; + d_desc.stat_desc = StaticTree.static_d_desc; + bl_desc.dyn_tree = bl_tree; + bl_desc.stat_desc = StaticTree.static_bl_desc; + bi_buf = 0; + bi_valid = 0; + last_eob_len = 8; + init_block(); + } + + internal void init_block() + { + for (int i = 0; i < L_CODES; i++) + { + dyn_ltree[i * 2] = 0; + } + for (int j = 0; j < 30; j++) + { + dyn_dtree[j * 2] = 0; + } + for (int k = 0; k < 19; k++) + { + bl_tree[k * 2] = 0; + } + dyn_ltree[512] = 1; + opt_len = (static_len = 0); + last_lit = (matches = 0); + } + + internal void pqdownheap(short[] tree, int k) + { + int num = heap[k]; + for (int num2 = k << 1; num2 <= heap_len; num2 <<= 1) + { + if (num2 < heap_len && smaller(tree, heap[num2 + 1], heap[num2], depth)) + { + num2++; + } + if (smaller(tree, num, heap[num2], depth)) + { + break; + } + heap[k] = heap[num2]; + k = num2; + } + heap[k] = num; + } + + internal static bool smaller(short[] tree, int n, int m, byte[] depth) + { + return tree[n * 2] < tree[m * 2] || (tree[n * 2] == tree[m * 2] && depth[n] <= depth[m]); + } + + internal void scan_tree(short[] tree, int max_code) + { + int num = -1; + int num2 = tree[1]; + int num3 = 0; + int num4 = 7; + int num5 = 4; + if (num2 == 0) + { + num4 = 138; + num5 = 3; + } + tree[(max_code + 1) * 2 + 1] = (short)SupportClass.Identity(65535L); + for (int i = 0; i <= max_code; i++) + { + int num6 = num2; + num2 = tree[(i + 1) * 2 + 1]; + if (++num3 < num4 && num6 == num2) + { + continue; + } + if (num3 < num5) + { + bl_tree[num6 * 2] = (short)(bl_tree[num6 * 2] + num3); + } + else if (num6 != 0) + { + if (num6 != num) + { + bl_tree[num6 * 2]++; + } + bl_tree[32]++; + } + else if (num3 <= 10) + { + bl_tree[34]++; + } + else + { + bl_tree[36]++; + } + num3 = 0; + num = num6; + if (num2 == 0) + { + num4 = 138; + num5 = 3; + } + else if (num6 == num2) + { + num4 = 6; + num5 = 3; + } + else + { + num4 = 7; + num5 = 4; + } + } + } + + internal int build_bl_tree() + { + scan_tree(dyn_ltree, l_desc.max_code); + scan_tree(dyn_dtree, d_desc.max_code); + bl_desc.build_tree(this); + int num = 18; + while (num >= 3 && bl_tree[Tree.bl_order[num] * 2 + 1] == 0) + { + num--; + } + opt_len += 3 * (num + 1) + 5 + 5 + 4; + return num; + } + + internal void send_all_trees(int lcodes, int dcodes, int blcodes) + { + send_bits(lcodes - 257, 5); + send_bits(dcodes - 1, 5); + send_bits(blcodes - 4, 4); + for (int i = 0; i < blcodes; i++) + { + send_bits(bl_tree[Tree.bl_order[i] * 2 + 1], 3); + } + send_tree(dyn_ltree, lcodes - 1); + send_tree(dyn_dtree, dcodes - 1); + } + + internal void send_tree(short[] tree, int max_code) + { + int num = -1; + int num2 = tree[1]; + int num3 = 0; + int num4 = 7; + int num5 = 4; + if (num2 == 0) + { + num4 = 138; + num5 = 3; + } + for (int i = 0; i <= max_code; i++) + { + int num6 = num2; + num2 = tree[(i + 1) * 2 + 1]; + if (++num3 < num4 && num6 == num2) + { + continue; + } + if (num3 < num5) + { + do + { + send_code(num6, bl_tree); + } + while (--num3 != 0); + } + else if (num6 != 0) + { + if (num6 != num) + { + send_code(num6, bl_tree); + num3--; + } + send_code(16, bl_tree); + send_bits(num3 - 3, 2); + } + else if (num3 <= 10) + { + send_code(17, bl_tree); + send_bits(num3 - 3, 3); + } + else + { + send_code(18, bl_tree); + send_bits(num3 - 11, 7); + } + num3 = 0; + num = num6; + if (num2 == 0) + { + num4 = 138; + num5 = 3; + } + else if (num6 == num2) + { + num4 = 6; + num5 = 3; + } + else + { + num4 = 7; + num5 = 4; + } + } + } + + internal void put_byte(byte[] p, int start, int len) + { + Array.Copy(p, start, pending_buf, pending, len); + pending += len; + } + + internal void put_byte(byte c) + { + pending_buf[pending++] = c; + } + + internal void put_short(int w) + { + put_byte((byte)w); + put_byte((byte)SupportClass.URShift(w, 8)); + } + + internal void putShortMSB(int b) + { + put_byte((byte)(b >> 8)); + put_byte((byte)b); + } + + internal void send_code(int c, short[] tree) + { + send_bits(tree[c * 2] & 0xFFFF, tree[c * 2 + 1] & 0xFFFF); + } + + internal void send_bits(int value_Renamed, int length) + { + if (bi_valid > 16 - length) + { + bi_buf = (short)((ushort)bi_buf | (ushort)((value_Renamed << bi_valid) & 0xFFFF)); + put_short(bi_buf); + bi_buf = (short)SupportClass.URShift(value_Renamed, 16 - bi_valid); + bi_valid += length - 16; + } + else + { + bi_buf = (short)((ushort)bi_buf | (ushort)((value_Renamed << bi_valid) & 0xFFFF)); + bi_valid += length; + } + } + + internal void _tr_align() + { + send_bits(2, 3); + send_code(256, StaticTree.static_ltree); + bi_flush(); + if (1 + last_eob_len + 10 - bi_valid < 9) + { + send_bits(2, 3); + send_code(256, StaticTree.static_ltree); + bi_flush(); + } + last_eob_len = 7; + } + + internal bool _tr_tally(int dist, int lc) + { + pending_buf[d_buf + last_lit * 2] = (byte)SupportClass.URShift(dist, 8); + pending_buf[d_buf + last_lit * 2 + 1] = (byte)dist; + pending_buf[l_buf + last_lit] = (byte)lc; + last_lit++; + if (dist == 0) + { + dyn_ltree[lc * 2]++; + } + else + { + matches++; + dist--; + dyn_ltree[(Tree._length_code[lc] + 256 + 1) * 2]++; + dyn_dtree[Tree.d_code(dist) * 2]++; + } + if ((last_lit & 0x1FFF) == 0 && level > 2) + { + int num = last_lit * 8; + int num2 = strstart - block_start; + for (int i = 0; i < 30; i++) + { + num = (int)(num + dyn_dtree[i * 2] * (5L + (long)Tree.extra_dbits[i])); + } + num = SupportClass.URShift(num, 3); + if (matches < last_lit / 2 && num < num2 / 2) + { + return true; + } + } + return last_lit == lit_bufsize - 1; + } + + internal void compress_block(short[] ltree, short[] dtree) + { + int num = 0; + if (last_lit != 0) + { + do + { + int num2 = ((pending_buf[d_buf + num * 2] << 8) & 0xFF00) | (pending_buf[d_buf + num * 2 + 1] & 0xFF); + int num3 = pending_buf[l_buf + num] & 0xFF; + num++; + if (num2 == 0) + { + send_code(num3, ltree); + continue; + } + int num4 = Tree._length_code[num3]; + send_code(num4 + 256 + 1, ltree); + int num5 = Tree.extra_lbits[num4]; + if (num5 != 0) + { + num3 -= Tree.base_length[num4]; + send_bits(num3, num5); + } + num2--; + num4 = Tree.d_code(num2); + send_code(num4, dtree); + num5 = Tree.extra_dbits[num4]; + if (num5 != 0) + { + num2 -= Tree.base_dist[num4]; + send_bits(num2, num5); + } + } + while (num < last_lit); + } + send_code(256, ltree); + last_eob_len = ltree[513]; + } + + internal void set_data_type() + { + int i = 0; + int num = 0; + int num2 = 0; + for (; i < 7; i++) + { + num2 += dyn_ltree[i * 2]; + } + for (; i < 128; i++) + { + num += dyn_ltree[i * 2]; + } + for (; i < 256; i++) + { + num2 += dyn_ltree[i * 2]; + } + data_type = (byte)((num2 <= SupportClass.URShift(num, 2)) ? 1u : 0u); + } + + internal void bi_flush() + { + if (bi_valid == 16) + { + put_short(bi_buf); + bi_buf = 0; + bi_valid = 0; + } + else if (bi_valid >= 8) + { + put_byte((byte)bi_buf); + bi_buf = (short)SupportClass.URShift(bi_buf, 8); + bi_valid -= 8; + } + } + + internal void bi_windup() + { + if (bi_valid > 8) + { + put_short(bi_buf); + } + else if (bi_valid > 0) + { + put_byte((byte)bi_buf); + } + bi_buf = 0; + bi_valid = 0; + } + + internal void copy_block(int buf, int len, bool header) + { + bi_windup(); + last_eob_len = 8; + if (header) + { + put_short((short)len); + put_short((short)(~len)); + } + put_byte(window, buf, len); + } + + internal void flush_block_only(bool eof) + { + _tr_flush_block((block_start >= 0) ? block_start : (-1), strstart - block_start, eof); + block_start = strstart; + strm.flush_pending(); + } + + internal int deflate_stored(int flush) + { + int num = 65535; + if (num > pending_buf_size - 5) + { + num = pending_buf_size - 5; + } + while (true) + { + if (lookahead <= 1) + { + fill_window(); + if (lookahead == 0 && flush == 0) + { + return 0; + } + if (lookahead == 0) + { + break; + } + } + strstart += lookahead; + lookahead = 0; + int num2 = block_start + num; + if (strstart == 0 || strstart >= num2) + { + lookahead = strstart - num2; + strstart = num2; + flush_block_only(eof: false); + if (strm.avail_out == 0) + { + return 0; + } + } + if (strstart - block_start >= w_size - MIN_LOOKAHEAD) + { + flush_block_only(eof: false); + if (strm.avail_out == 0) + { + return 0; + } + } + } + flush_block_only(flush == 4); + if (strm.avail_out == 0) + { + return (flush == 4) ? 2 : 0; + } + return (flush != 4) ? 1 : 3; + } + + internal void _tr_stored_block(int buf, int stored_len, bool eof) + { + send_bits(eof ? 1 : 0, 3); + copy_block(buf, stored_len, header: true); + } + + internal void _tr_flush_block(int buf, int stored_len, bool eof) + { + int num = 0; + int num2; + int num3; + if (level > 0) + { + if (data_type == 2) + { + set_data_type(); + } + l_desc.build_tree(this); + d_desc.build_tree(this); + num = build_bl_tree(); + num2 = SupportClass.URShift(opt_len + 3 + 7, 3); + num3 = SupportClass.URShift(static_len + 3 + 7, 3); + if (num3 <= num2) + { + num2 = num3; + } + } + else + { + num2 = (num3 = stored_len + 5); + } + if (stored_len + 4 <= num2 && buf != -1) + { + _tr_stored_block(buf, stored_len, eof); + } + else if (num3 == num2) + { + send_bits(2 + (eof ? 1 : 0), 3); + compress_block(StaticTree.static_ltree, StaticTree.static_dtree); + } + else + { + send_bits(4 + (eof ? 1 : 0), 3); + send_all_trees(l_desc.max_code + 1, d_desc.max_code + 1, num + 1); + compress_block(dyn_ltree, dyn_dtree); + } + init_block(); + if (eof) + { + bi_windup(); + } + } + + internal void fill_window() + { + do + { + int num = window_size - lookahead - strstart; + int num2; + if (num == 0 && strstart == 0 && lookahead == 0) + { + num = w_size; + } + else if (num == -1) + { + num--; + } + else if (strstart >= w_size + w_size - MIN_LOOKAHEAD) + { + Array.Copy(window, w_size, window, 0, w_size); + match_start -= w_size; + strstart -= w_size; + block_start -= w_size; + num2 = hash_size; + int num3 = num2; + do + { + int num4 = head[--num3] & 0xFFFF; + head[num3] = (short)((num4 >= w_size) ? (num4 - w_size) : 0); + } + while (--num2 != 0); + num2 = w_size; + num3 = num2; + do + { + int num4 = prev[--num3] & 0xFFFF; + prev[num3] = (short)((num4 >= w_size) ? (num4 - w_size) : 0); + } + while (--num2 != 0); + num += w_size; + } + if (strm.avail_in == 0) + { + break; + } + num2 = strm.read_buf(window, strstart + lookahead, num); + lookahead += num2; + if (lookahead >= 3) + { + ins_h = window[strstart] & 0xFF; + ins_h = ((ins_h << hash_shift) ^ (window[strstart + 1] & 0xFF)) & hash_mask; + } + } + while (lookahead < MIN_LOOKAHEAD && strm.avail_in != 0); + } + + internal int deflate_fast(int flush) + { + int num = 0; + while (true) + { + if (lookahead < MIN_LOOKAHEAD) + { + fill_window(); + if (lookahead < MIN_LOOKAHEAD && flush == 0) + { + return 0; + } + if (lookahead == 0) + { + break; + } + } + if (lookahead >= 3) + { + ins_h = ((ins_h << hash_shift) ^ (window[strstart + 2] & 0xFF)) & hash_mask; + num = head[ins_h] & 0xFFFF; + prev[strstart & w_mask] = head[ins_h]; + head[ins_h] = (short)strstart; + } + if (num != 0L && ((strstart - num) & 0xFFFF) <= w_size - MIN_LOOKAHEAD && strategy != 2) + { + match_length = longest_match(num); + } + bool flag; + if (match_length >= 3) + { + flag = _tr_tally(strstart - match_start, match_length - 3); + lookahead -= match_length; + if (match_length <= max_lazy_match && lookahead >= 3) + { + match_length--; + do + { + strstart++; + ins_h = ((ins_h << hash_shift) ^ (window[strstart + 2] & 0xFF)) & hash_mask; + num = head[ins_h] & 0xFFFF; + prev[strstart & w_mask] = head[ins_h]; + head[ins_h] = (short)strstart; + } + while (--match_length != 0); + strstart++; + } + else + { + strstart += match_length; + match_length = 0; + ins_h = window[strstart] & 0xFF; + ins_h = ((ins_h << hash_shift) ^ (window[strstart + 1] & 0xFF)) & hash_mask; + } + } + else + { + flag = _tr_tally(0, window[strstart] & 0xFF); + lookahead--; + strstart++; + } + if (flag) + { + flush_block_only(eof: false); + if (strm.avail_out == 0) + { + return 0; + } + } + } + flush_block_only(flush == 4); + if (strm.avail_out == 0) + { + if (flush == 4) + { + return 2; + } + return 0; + } + return (flush != 4) ? 1 : 3; + } + + internal int deflate_slow(int flush) + { + int num = 0; + while (true) + { + if (lookahead < MIN_LOOKAHEAD) + { + fill_window(); + if (lookahead < MIN_LOOKAHEAD && flush == 0) + { + return 0; + } + if (lookahead == 0) + { + break; + } + } + if (lookahead >= 3) + { + ins_h = ((ins_h << hash_shift) ^ (window[strstart + 2] & 0xFF)) & hash_mask; + num = head[ins_h] & 0xFFFF; + prev[strstart & w_mask] = head[ins_h]; + head[ins_h] = (short)strstart; + } + prev_length = match_length; + prev_match = match_start; + match_length = 2; + if (num != 0 && prev_length < max_lazy_match && ((strstart - num) & 0xFFFF) <= w_size - MIN_LOOKAHEAD) + { + if (strategy != 2) + { + match_length = longest_match(num); + } + if (match_length <= 5 && (strategy == 1 || (match_length == 3 && strstart - match_start > 4096))) + { + match_length = 2; + } + } + if (prev_length >= 3 && match_length <= prev_length) + { + int num2 = strstart + lookahead - 3; + bool flag = _tr_tally(strstart - 1 - prev_match, prev_length - 3); + lookahead -= prev_length - 1; + prev_length -= 2; + do + { + if (++strstart <= num2) + { + ins_h = ((ins_h << hash_shift) ^ (window[strstart + 2] & 0xFF)) & hash_mask; + num = head[ins_h] & 0xFFFF; + prev[strstart & w_mask] = head[ins_h]; + head[ins_h] = (short)strstart; + } + } + while (--prev_length != 0); + match_available = 0; + match_length = 2; + strstart++; + if (flag) + { + flush_block_only(eof: false); + if (strm.avail_out == 0) + { + return 0; + } + } + } + else if (match_available != 0) + { + if (_tr_tally(0, window[strstart - 1] & 0xFF)) + { + flush_block_only(eof: false); + } + strstart++; + lookahead--; + if (strm.avail_out == 0) + { + return 0; + } + } + else + { + match_available = 1; + strstart++; + lookahead--; + } + } + if (match_available != 0) + { + bool flag = _tr_tally(0, window[strstart - 1] & 0xFF); + match_available = 0; + } + flush_block_only(flush == 4); + if (strm.avail_out == 0) + { + if (flush == 4) + { + return 2; + } + return 0; + } + return (flush != 4) ? 1 : 3; + } + + internal int longest_match(int cur_match) + { + int num = max_chain_length; + int num2 = strstart; + int num3 = prev_length; + int num4 = ((strstart > w_size - MIN_LOOKAHEAD) ? (strstart - (w_size - MIN_LOOKAHEAD)) : 0); + int num5 = nice_match; + int num6 = w_mask; + int num7 = strstart + 258; + byte b = window[num2 + num3 - 1]; + byte b2 = window[num2 + num3]; + if (prev_length >= good_match) + { + num >>= 2; + } + if (num5 > lookahead) + { + num5 = lookahead; + } + do + { + int num8 = cur_match; + if (window[num8 + num3] != b2 || window[num8 + num3 - 1] != b || window[num8] != window[num2] || window[++num8] != window[num2 + 1]) + { + continue; + } + num2 += 2; + num8++; + while (window[++num2] == window[++num8] && window[++num2] == window[++num8] && window[++num2] == window[++num8] && window[++num2] == window[++num8] && window[++num2] == window[++num8] && window[++num2] == window[++num8] && window[++num2] == window[++num8] && window[++num2] == window[++num8] && num2 < num7) + { + } + int num9 = 258 - (num7 - num2); + num2 = num7 - 258; + if (num9 > num3) + { + match_start = cur_match; + num3 = num9; + if (num9 >= num5) + { + break; + } + b = window[num2 + num3 - 1]; + b2 = window[num2 + num3]; + } + } + while ((cur_match = prev[cur_match & num6] & 0xFFFF) > num4 && --num != 0); + if (num3 <= lookahead) + { + return num3; + } + return lookahead; + } + + internal int deflateInit(ZStream strm, int level, int bits) + { + return deflateInit2(strm, level, 8, bits, 8, 0); + } + + internal int deflateInit(ZStream strm, int level) + { + return deflateInit(strm, level, 15); + } + + internal int deflateInit2(ZStream strm, int level, int method, int windowBits, int memLevel, int strategy) + { + int num = 0; + strm.msg = null; + if (level == -1) + { + level = 6; + } + if (windowBits < 0) + { + num = 1; + windowBits = -windowBits; + } + if (memLevel < 1 || memLevel > 9 || method != 8 || windowBits < 9 || windowBits > 15 || level < 0 || level > 9 || strategy < 0 || strategy > 2) + { + return -2; + } + strm.dstate = this; + noheader = num; + w_bits = windowBits; + w_size = 1 << w_bits; + w_mask = w_size - 1; + hash_bits = memLevel + 7; + hash_size = 1 << hash_bits; + hash_mask = hash_size - 1; + hash_shift = (hash_bits + 3 - 1) / 3; + window = new byte[w_size * 2]; + prev = new short[w_size]; + head = new short[hash_size]; + lit_bufsize = 1 << memLevel + 6; + pending_buf = new byte[lit_bufsize * 4]; + pending_buf_size = lit_bufsize * 4; + d_buf = lit_bufsize; + l_buf = 3 * lit_bufsize; + this.level = level; + this.strategy = strategy; + this.method = (byte)method; + return deflateReset(strm); + } + + internal int deflateReset(ZStream strm) + { + strm.total_in = (strm.total_out = 0L); + strm.msg = null; + strm.data_type = 2; + pending = 0; + pending_out = 0; + if (noheader < 0) + { + noheader = 0; + } + status = ((noheader != 0) ? 113 : 42); + strm.adler = strm._adler.adler32(0L, null, 0, 0); + last_flush = 0; + tr_init(); + lm_init(); + return 0; + } + + internal int deflateEnd() + { + if (status != 42 && status != 113 && status != 666) + { + return -2; + } + pending_buf = null; + head = null; + prev = null; + window = null; + return (status == 113) ? (-3) : 0; + } + + internal int deflateParams(ZStream strm, int _level, int _strategy) + { + int result = 0; + if (_level == -1) + { + _level = 6; + } + if (_level < 0 || _level > 9 || _strategy < 0 || _strategy > 2) + { + return -2; + } + if (config_table[level].func != config_table[_level].func && strm.total_in != 0) + { + result = strm.deflate(1); + } + if (level != _level) + { + level = _level; + max_lazy_match = config_table[level].max_lazy; + good_match = config_table[level].good_length; + nice_match = config_table[level].nice_length; + max_chain_length = config_table[level].max_chain; + } + strategy = _strategy; + return result; + } + + internal int deflateSetDictionary(ZStream strm, byte[] dictionary, int dictLength) + { + int num = dictLength; + int sourceIndex = 0; + if (dictionary == null || status != 42) + { + return -2; + } + strm.adler = strm._adler.adler32(strm.adler, dictionary, 0, dictLength); + if (num < 3) + { + return 0; + } + if (num > w_size - MIN_LOOKAHEAD) + { + num = w_size - MIN_LOOKAHEAD; + sourceIndex = dictLength - num; + } + Array.Copy(dictionary, sourceIndex, window, 0, num); + strstart = num; + block_start = num; + ins_h = window[0] & 0xFF; + ins_h = ((ins_h << hash_shift) ^ (window[1] & 0xFF)) & hash_mask; + for (int i = 0; i <= num - 3; i++) + { + ins_h = ((ins_h << hash_shift) ^ (window[i + 2] & 0xFF)) & hash_mask; + prev[i & w_mask] = head[ins_h]; + head[ins_h] = (short)i; + } + return 0; + } + + internal int deflate(ZStream strm, int flush) + { + if (flush > 4 || flush < 0) + { + return -2; + } + if (strm.next_out == null || (strm.next_in == null && strm.avail_in != 0) || (status == 666 && flush != 4)) + { + strm.msg = z_errmsg[4]; + return -2; + } + if (strm.avail_out == 0) + { + strm.msg = z_errmsg[7]; + return -5; + } + this.strm = strm; + int num = last_flush; + last_flush = flush; + if (status == 42) + { + int num2 = 8 + (w_bits - 8 << 4) << 8; + int num3 = ((level - 1) & 0xFF) >> 1; + if (num3 > 3) + { + num3 = 3; + } + num2 |= num3 << 6; + if (strstart != 0) + { + num2 |= 0x20; + } + num2 += 31 - num2 % 31; + status = 113; + putShortMSB(num2); + if (strstart != 0) + { + putShortMSB((int)SupportClass.URShift(strm.adler, 16)); + putShortMSB((int)(strm.adler & 0xFFFF)); + } + strm.adler = strm._adler.adler32(0L, null, 0, 0); + } + if (pending != 0) + { + strm.flush_pending(); + if (strm.avail_out == 0) + { + last_flush = -1; + return 0; + } + } + else if (strm.avail_in == 0 && flush <= num && flush != 4) + { + strm.msg = z_errmsg[7]; + return -5; + } + if (status == 666 && strm.avail_in != 0) + { + strm.msg = z_errmsg[7]; + return -5; + } + if (strm.avail_in != 0 || lookahead != 0 || (flush != 0 && status != 666)) + { + int num4 = -1; + switch (config_table[level].func) + { + case 0: + num4 = deflate_stored(flush); + break; + case 1: + num4 = deflate_fast(flush); + break; + case 2: + num4 = deflate_slow(flush); + break; + } + if (num4 == 2 || num4 == 3) + { + status = 666; + } + if (num4 == 0 || num4 == 2) + { + if (strm.avail_out == 0) + { + last_flush = -1; + } + return 0; + } + if (num4 == 1) + { + if (flush == 1) + { + _tr_align(); + } + else + { + _tr_stored_block(0, 0, eof: false); + if (flush == 3) + { + for (int i = 0; i < hash_size; i++) + { + head[i] = 0; + } + } + } + strm.flush_pending(); + if (strm.avail_out == 0) + { + last_flush = -1; + return 0; + } + } + } + if (flush != 4) + { + return 0; + } + if (noheader != 0) + { + return 1; + } + putShortMSB((int)SupportClass.URShift(strm.adler, 16)); + putShortMSB((int)(strm.adler & 0xFFFF)); + strm.flush_pending(); + noheader = -1; + return (pending == 0) ? 1 : 0; + } + + static Deflate() + { + z_errmsg = new string[10] { "need dictionary", "stream end", "", "file error", "stream error", "data error", "insufficient memory", "buffer error", "incompatible version", "" }; + MIN_LOOKAHEAD = 262; + L_CODES = 286; + HEAP_SIZE = 2 * L_CODES + 1; + config_table = new Config[10]; + config_table[0] = new Config(0, 0, 0, 0, 0); + config_table[1] = new Config(4, 4, 8, 4, 1); + config_table[2] = new Config(4, 5, 16, 8, 1); + config_table[3] = new Config(4, 6, 32, 32, 1); + config_table[4] = new Config(4, 4, 16, 16, 2); + config_table[5] = new Config(8, 16, 32, 32, 2); + config_table[6] = new Config(8, 16, 128, 128, 2); + config_table[7] = new Config(8, 32, 128, 256, 2); + config_table[8] = new Config(32, 128, 258, 1024, 2); + config_table[9] = new Config(32, 258, 258, 4096, 2); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs.meta new file mode 100644 index 00000000..191cf63f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Deflate.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2dfc48049b9e04b4eadd12cb50383cb8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs new file mode 100644 index 00000000..d8550daf --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs @@ -0,0 +1,658 @@ +using System; + +namespace ComponentAce.Compression.Libs.zlib +{ + internal sealed class InfBlocks + { + private const int MANY = 1440; + + private static readonly int[] inflate_mask = new int[17] + { + 0, 1, 3, 7, 15, 31, 63, 127, 255, 511, + 1023, 2047, 4095, 8191, 16383, 32767, 65535 + }; + + internal static readonly int[] border = new int[19] + { + 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, + 11, 4, 12, 3, 13, 2, 14, 1, 15 + }; + + private const int Z_OK = 0; + + private const int Z_STREAM_END = 1; + + private const int Z_NEED_DICT = 2; + + private const int Z_ERRNO = -1; + + private const int Z_STREAM_ERROR = -2; + + private const int Z_DATA_ERROR = -3; + + private const int Z_MEM_ERROR = -4; + + private const int Z_BUF_ERROR = -5; + + private const int Z_VERSION_ERROR = -6; + + private const int TYPE = 0; + + private const int LENS = 1; + + private const int STORED = 2; + + private const int TABLE = 3; + + private const int BTREE = 4; + + private const int DTREE = 5; + + private const int CODES = 6; + + private const int DRY = 7; + + private const int DONE = 8; + + private const int BAD = 9; + + internal int mode; + + internal int left; + + internal int table; + + internal int index; + + internal int[] blens; + + internal int[] bb = new int[1]; + + internal int[] tb = new int[1]; + + internal InfCodes codes; + + internal int last; + + internal int bitk; + + internal int bitb; + + internal int[] hufts; + + internal byte[] window; + + internal int end; + + internal int read; + + internal int write; + + internal object checkfn; + + internal long check; + + internal InfBlocks(ZStream z, object checkfn, int w) + { + hufts = new int[4320]; + window = new byte[w]; + end = w; + this.checkfn = checkfn; + mode = 0; + reset(z, null); + } + + internal void reset(ZStream z, long[] c) + { + if (c != null) + { + c[0] = check; + } + if (mode == 4 || mode == 5) + { + blens = null; + } + if (mode == 6) + { + codes.free(z); + } + mode = 0; + bitk = 0; + bitb = 0; + read = (write = 0); + if (checkfn != null) + { + z.adler = (check = z._adler.adler32(0L, null, 0, 0)); + } + } + + internal int proc(ZStream z, int r) + { + int num = z.next_in_index; + int num2 = z.avail_in; + int num3 = bitb; + int i = bitk; + int num4 = write; + int num5 = ((num4 < read) ? (read - num4 - 1) : (end - num4)); + while (true) + { + switch (mode) + { + case 0: + { + for (; i < 3; i += 8) + { + if (num2 != 0) + { + r = 0; + num2--; + num3 |= (z.next_in[num++] & 0xFF) << i; + continue; + } + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + int num6 = num3 & 7; + last = num6 & 1; + switch (SupportClass.URShift(num6, 1)) + { + case 0: + num3 = SupportClass.URShift(num3, 3); + i -= 3; + num6 = i & 7; + num3 = SupportClass.URShift(num3, num6); + i -= num6; + mode = 1; + break; + case 1: + { + int[] array5 = new int[1]; + int[] array6 = new int[1]; + int[][] array7 = new int[1][]; + int[][] array8 = new int[1][]; + InfTree.inflate_trees_fixed(array5, array6, array7, array8, z); + codes = new InfCodes(array5[0], array6[0], array7[0], array8[0], z); + num3 = SupportClass.URShift(num3, 3); + i -= 3; + mode = 6; + break; + } + case 2: + num3 = SupportClass.URShift(num3, 3); + i -= 3; + mode = 3; + break; + case 3: + num3 = SupportClass.URShift(num3, 3); + i -= 3; + mode = 9; + z.msg = "invalid block type"; + r = -3; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + break; + } + case 1: + for (; i < 32; i += 8) + { + if (num2 != 0) + { + r = 0; + num2--; + num3 |= (z.next_in[num++] & 0xFF) << i; + continue; + } + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + if ((SupportClass.URShift(~num3, 16) & 0xFFFF) != (num3 & 0xFFFF)) + { + mode = 9; + z.msg = "invalid stored block lengths"; + r = -3; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + left = num3 & 0xFFFF; + num3 = (i = 0); + mode = ((left != 0) ? 2 : ((last != 0) ? 7 : 0)); + break; + case 2: + { + if (num2 == 0) + { + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + if (num5 == 0) + { + if (num4 == end && read != 0) + { + num4 = 0; + num5 = ((num4 < read) ? (read - num4 - 1) : (end - num4)); + } + if (num5 == 0) + { + write = num4; + r = inflate_flush(z, r); + num4 = write; + num5 = ((num4 < read) ? (read - num4 - 1) : (end - num4)); + if (num4 == end && read != 0) + { + num4 = 0; + num5 = ((num4 < read) ? (read - num4 - 1) : (end - num4)); + } + if (num5 == 0) + { + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + } + } + r = 0; + int num6 = left; + if (num6 > num2) + { + num6 = num2; + } + if (num6 > num5) + { + num6 = num5; + } + Array.Copy(z.next_in, num, window, num4, num6); + num += num6; + num2 -= num6; + num4 += num6; + num5 -= num6; + if ((left -= num6) == 0) + { + mode = ((last != 0) ? 7 : 0); + } + break; + } + case 3: + { + for (; i < 14; i += 8) + { + if (num2 != 0) + { + r = 0; + num2--; + num3 |= (z.next_in[num++] & 0xFF) << i; + continue; + } + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + int num6 = (table = num3 & 0x3FFF); + if ((num6 & 0x1F) > 29 || ((num6 >> 5) & 0x1F) > 29) + { + mode = 9; + z.msg = "too many length or distance symbols"; + r = -3; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + num6 = 258 + (num6 & 0x1F) + ((num6 >> 5) & 0x1F); + blens = new int[num6]; + num3 = SupportClass.URShift(num3, 14); + i -= 14; + index = 0; + mode = 4; + goto case 4; + } + case 4: + { + while (index < 4 + SupportClass.URShift(table, 10)) + { + for (; i < 3; i += 8) + { + if (num2 != 0) + { + r = 0; + num2--; + num3 |= (z.next_in[num++] & 0xFF) << i; + continue; + } + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + blens[border[index++]] = num3 & 7; + num3 = SupportClass.URShift(num3, 3); + i -= 3; + } + while (index < 19) + { + blens[border[index++]] = 0; + } + bb[0] = 7; + int num6 = InfTree.inflate_trees_bits(blens, bb, tb, hufts, z); + if (num6 != 0) + { + r = num6; + if (r == -3) + { + blens = null; + mode = 9; + } + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + index = 0; + mode = 5; + goto case 5; + } + case 5: + { + int num6; + while (true) + { + num6 = table; + if (index >= 258 + (num6 & 0x1F) + ((num6 >> 5) & 0x1F)) + { + break; + } + for (num6 = bb[0]; i < num6; i += 8) + { + if (num2 != 0) + { + r = 0; + num2--; + num3 |= (z.next_in[num++] & 0xFF) << i; + continue; + } + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + if (tb[0] == -1) + { + } + num6 = hufts[(tb[0] + (num3 & inflate_mask[num6])) * 3 + 1]; + int num7 = hufts[(tb[0] + (num3 & inflate_mask[num6])) * 3 + 2]; + if (num7 < 16) + { + num3 = SupportClass.URShift(num3, num6); + i -= num6; + blens[index++] = num7; + continue; + } + int num8 = ((num7 == 18) ? 7 : (num7 - 14)); + int num9 = ((num7 == 18) ? 11 : 3); + for (; i < num6 + num8; i += 8) + { + if (num2 != 0) + { + r = 0; + num2--; + num3 |= (z.next_in[num++] & 0xFF) << i; + continue; + } + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + num3 = SupportClass.URShift(num3, num6); + i -= num6; + num9 += num3 & inflate_mask[num8]; + num3 = SupportClass.URShift(num3, num8); + i -= num8; + num8 = index; + num6 = table; + if (num8 + num9 > 258 + (num6 & 0x1F) + ((num6 >> 5) & 0x1F) || (num7 == 16 && num8 < 1)) + { + blens = null; + mode = 9; + z.msg = "invalid bit length repeat"; + r = -3; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + num7 = ((num7 == 16) ? blens[num8 - 1] : 0); + do + { + blens[num8++] = num7; + } + while (--num9 != 0); + index = num8; + } + tb[0] = -1; + int[] array = new int[1]; + int[] array2 = new int[1]; + int[] array3 = new int[1]; + int[] array4 = new int[1]; + array[0] = 9; + array2[0] = 6; + num6 = table; + num6 = InfTree.inflate_trees_dynamic(257 + (num6 & 0x1F), 1 + ((num6 >> 5) & 0x1F), blens, array, array2, array3, array4, hufts, z); + if (num6 != 0) + { + if (num6 == -3) + { + blens = null; + mode = 9; + } + r = num6; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + codes = new InfCodes(array[0], array2[0], hufts, array3[0], hufts, array4[0], z); + blens = null; + mode = 6; + goto case 6; + } + case 6: + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + if ((r = codes.proc(this, z, r)) != 1) + { + return inflate_flush(z, r); + } + r = 0; + codes.free(z); + num = z.next_in_index; + num2 = z.avail_in; + num3 = bitb; + i = bitk; + num4 = write; + num5 = ((num4 < read) ? (read - num4 - 1) : (end - num4)); + if (last == 0) + { + mode = 0; + break; + } + mode = 7; + goto case 7; + case 7: + write = num4; + r = inflate_flush(z, r); + num4 = write; + num5 = ((num4 < read) ? (read - num4 - 1) : (end - num4)); + if (read != write) + { + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + mode = 8; + goto case 8; + case 8: + r = 1; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + case 9: + r = -3; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + default: + r = -2; + bitb = num3; + bitk = i; + z.avail_in = num2; + z.total_in += num - z.next_in_index; + z.next_in_index = num; + write = num4; + return inflate_flush(z, r); + } + } + } + + internal void free(ZStream z) + { + reset(z, null); + window = null; + hufts = null; + } + + internal void set_dictionary(byte[] d, int start, int n) + { + Array.Copy(d, start, window, 0, n); + read = (write = n); + } + + internal int sync_point() + { + return (mode == 1) ? 1 : 0; + } + + internal int inflate_flush(ZStream z, int r) + { + int next_out_index = z.next_out_index; + int num = read; + int num2 = ((num <= write) ? write : end) - num; + if (num2 > z.avail_out) + { + num2 = z.avail_out; + } + if (num2 != 0 && r == -5) + { + r = 0; + } + z.avail_out -= num2; + z.total_out += num2; + if (checkfn != null) + { + z.adler = (check = z._adler.adler32(check, window, num, num2)); + } + Array.Copy(window, num, z.next_out, next_out_index, num2); + next_out_index += num2; + num += num2; + if (num == end) + { + num = 0; + if (write == end) + { + write = 0; + } + num2 = write - num; + if (num2 > z.avail_out) + { + num2 = z.avail_out; + } + if (num2 != 0 && r == -5) + { + r = 0; + } + z.avail_out -= num2; + z.total_out += num2; + if (checkfn != null) + { + z.adler = (check = z._adler.adler32(check, window, num, num2)); + } + Array.Copy(window, num, z.next_out, next_out_index, num2); + next_out_index += num2; + num += num2; + } + z.next_out_index = next_out_index; + read = num; + return r; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs.meta new file mode 100644 index 00000000..553006c1 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfBlocks.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 38ff23616a4cf9349add205940d1e8de +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs new file mode 100644 index 00000000..544a8b0a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs @@ -0,0 +1,666 @@ +using System; + +namespace ComponentAce.Compression.Libs.zlib +{ + internal sealed class InfCodes + { + private static readonly int[] inflate_mask = new int[17] + { + 0, 1, 3, 7, 15, 31, 63, 127, 255, 511, + 1023, 2047, 4095, 8191, 16383, 32767, 65535 + }; + + private const int Z_OK = 0; + + private const int Z_STREAM_END = 1; + + private const int Z_NEED_DICT = 2; + + private const int Z_ERRNO = -1; + + private const int Z_STREAM_ERROR = -2; + + private const int Z_DATA_ERROR = -3; + + private const int Z_MEM_ERROR = -4; + + private const int Z_BUF_ERROR = -5; + + private const int Z_VERSION_ERROR = -6; + + private const int START = 0; + + private const int LEN = 1; + + private const int LENEXT = 2; + + private const int DIST = 3; + + private const int DISTEXT = 4; + + private const int COPY = 5; + + private const int LIT = 6; + + private const int WASH = 7; + + private const int END = 8; + + private const int BADCODE = 9; + + internal int mode; + + internal int len; + + internal int[] tree; + + internal int tree_index = 0; + + internal int need; + + internal int lit; + + internal int get_Renamed; + + internal int dist; + + internal byte lbits; + + internal byte dbits; + + internal int[] ltree; + + internal int ltree_index; + + internal int[] dtree; + + internal int dtree_index; + + internal InfCodes(int bl, int bd, int[] tl, int tl_index, int[] td, int td_index, ZStream z) + { + mode = 0; + lbits = (byte)bl; + dbits = (byte)bd; + ltree = tl; + ltree_index = tl_index; + dtree = td; + dtree_index = td_index; + } + + internal InfCodes(int bl, int bd, int[] tl, int[] td, ZStream z) + { + mode = 0; + lbits = (byte)bl; + dbits = (byte)bd; + ltree = tl; + ltree_index = 0; + dtree = td; + dtree_index = 0; + } + + internal int proc(InfBlocks s, ZStream z, int r) + { + int num = 0; + int num2 = 0; + int num3 = 0; + num3 = z.next_in_index; + int num4 = z.avail_in; + num = s.bitb; + num2 = s.bitk; + int num5 = s.write; + int num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + while (true) + { + switch (mode) + { + case 0: + if (num6 >= 258 && num4 >= 10) + { + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + r = inflate_fast(lbits, dbits, ltree, ltree_index, dtree, dtree_index, s, z); + num3 = z.next_in_index; + num4 = z.avail_in; + num = s.bitb; + num2 = s.bitk; + num5 = s.write; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + if (r != 0) + { + mode = ((r == 1) ? 7 : 9); + break; + } + } + need = lbits; + tree = ltree; + tree_index = ltree_index; + mode = 1; + goto case 1; + case 1: + { + int num7; + for (num7 = need; num2 < num7; num2 += 8) + { + if (num4 != 0) + { + r = 0; + num4--; + num |= (z.next_in[num3++] & 0xFF) << num2; + continue; + } + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + int num8 = (tree_index + (num & inflate_mask[num7])) * 3; + num = SupportClass.URShift(num, tree[num8 + 1]); + num2 -= tree[num8 + 1]; + int num9 = tree[num8]; + if (num9 == 0) + { + lit = tree[num8 + 2]; + mode = 6; + break; + } + if (((uint)num9 & 0x10u) != 0) + { + get_Renamed = num9 & 0xF; + len = tree[num8 + 2]; + mode = 2; + break; + } + if ((num9 & 0x40) == 0) + { + need = num9; + tree_index = num8 / 3 + tree[num8 + 2]; + break; + } + if (((uint)num9 & 0x20u) != 0) + { + mode = 7; + break; + } + mode = 9; + z.msg = "invalid literal/length code"; + r = -3; + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + case 2: + { + int num7; + for (num7 = get_Renamed; num2 < num7; num2 += 8) + { + if (num4 != 0) + { + r = 0; + num4--; + num |= (z.next_in[num3++] & 0xFF) << num2; + continue; + } + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + len += num & inflate_mask[num7]; + num >>= num7; + num2 -= num7; + need = dbits; + tree = dtree; + tree_index = dtree_index; + mode = 3; + goto case 3; + } + case 3: + { + int num7; + for (num7 = need; num2 < num7; num2 += 8) + { + if (num4 != 0) + { + r = 0; + num4--; + num |= (z.next_in[num3++] & 0xFF) << num2; + continue; + } + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + int num8 = (tree_index + (num & inflate_mask[num7])) * 3; + num >>= tree[num8 + 1]; + num2 -= tree[num8 + 1]; + int num9 = tree[num8]; + if (((uint)num9 & 0x10u) != 0) + { + get_Renamed = num9 & 0xF; + dist = tree[num8 + 2]; + mode = 4; + break; + } + if ((num9 & 0x40) == 0) + { + need = num9; + tree_index = num8 / 3 + tree[num8 + 2]; + break; + } + mode = 9; + z.msg = "invalid distance code"; + r = -3; + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + case 4: + { + int num7; + for (num7 = get_Renamed; num2 < num7; num2 += 8) + { + if (num4 != 0) + { + r = 0; + num4--; + num |= (z.next_in[num3++] & 0xFF) << num2; + continue; + } + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + dist += num & inflate_mask[num7]; + num >>= num7; + num2 -= num7; + mode = 5; + goto case 5; + } + case 5: + { + int i; + for (i = num5 - dist; i < 0; i += s.end) + { + } + while (len != 0) + { + if (num6 == 0) + { + if (num5 == s.end && s.read != 0) + { + num5 = 0; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + } + if (num6 == 0) + { + s.write = num5; + r = s.inflate_flush(z, r); + num5 = s.write; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + if (num5 == s.end && s.read != 0) + { + num5 = 0; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + } + if (num6 == 0) + { + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + } + } + s.window[num5++] = s.window[i++]; + num6--; + if (i == s.end) + { + i = 0; + } + len--; + } + mode = 0; + break; + } + case 6: + if (num6 == 0) + { + if (num5 == s.end && s.read != 0) + { + num5 = 0; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + } + if (num6 == 0) + { + s.write = num5; + r = s.inflate_flush(z, r); + num5 = s.write; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + if (num5 == s.end && s.read != 0) + { + num5 = 0; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + } + if (num6 == 0) + { + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + } + } + r = 0; + s.window[num5++] = (byte)lit; + num6--; + mode = 0; + break; + case 7: + if (num2 > 7) + { + num2 -= 8; + num4++; + num3--; + } + s.write = num5; + r = s.inflate_flush(z, r); + num5 = s.write; + num6 = ((num5 < s.read) ? (s.read - num5 - 1) : (s.end - num5)); + if (s.read != s.write) + { + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + mode = 8; + goto case 8; + case 8: + r = 1; + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + case 9: + r = -3; + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + default: + r = -2; + s.bitb = num; + s.bitk = num2; + z.avail_in = num4; + z.total_in += num3 - z.next_in_index; + z.next_in_index = num3; + s.write = num5; + return s.inflate_flush(z, r); + } + } + } + + internal void free(ZStream z) + { + } + + internal int inflate_fast(int bl, int bd, int[] tl, int tl_index, int[] td, int td_index, InfBlocks s, ZStream z) + { + int next_in_index = z.next_in_index; + int num = z.avail_in; + int num2 = s.bitb; + int num3 = s.bitk; + int num4 = s.write; + int num5 = ((num4 < s.read) ? (s.read - num4 - 1) : (s.end - num4)); + int num6 = inflate_mask[bl]; + int num7 = inflate_mask[bd]; + int num11; + while (true) + { + if (num3 < 20) + { + num--; + num2 |= (z.next_in[next_in_index++] & 0xFF) << num3; + num3 += 8; + continue; + } + int num8 = num2 & num6; + int[] array = tl; + int num9 = tl_index; + int num10; + if ((num10 = array[(num9 + num8) * 3]) == 0) + { + num2 >>= array[(num9 + num8) * 3 + 1]; + num3 -= array[(num9 + num8) * 3 + 1]; + s.window[num4++] = (byte)array[(num9 + num8) * 3 + 2]; + num5--; + } + else + { + while (true) + { + num2 >>= array[(num9 + num8) * 3 + 1]; + num3 -= array[(num9 + num8) * 3 + 1]; + if (((uint)num10 & 0x10u) != 0) + { + num10 &= 0xF; + num11 = array[(num9 + num8) * 3 + 2] + (num2 & inflate_mask[num10]); + num2 >>= num10; + for (num3 -= num10; num3 < 15; num3 += 8) + { + num--; + num2 |= (z.next_in[next_in_index++] & 0xFF) << num3; + } + num8 = num2 & num7; + array = td; + num9 = td_index; + num10 = array[(num9 + num8) * 3]; + while (true) + { + num2 >>= array[(num9 + num8) * 3 + 1]; + num3 -= array[(num9 + num8) * 3 + 1]; + if (((uint)num10 & 0x10u) != 0) + { + break; + } + if ((num10 & 0x40) == 0) + { + num8 += array[(num9 + num8) * 3 + 2]; + num8 += num2 & inflate_mask[num10]; + num10 = array[(num9 + num8) * 3]; + bool flag = true; + continue; + } + z.msg = "invalid distance code"; + num11 = z.avail_in - num; + num11 = ((num3 >> 3 < num11) ? (num3 >> 3) : num11); + num += num11; + next_in_index -= num11; + num3 -= num11 << 3; + s.bitb = num2; + s.bitk = num3; + z.avail_in = num; + z.total_in += next_in_index - z.next_in_index; + z.next_in_index = next_in_index; + s.write = num4; + return -3; + } + for (num10 &= 0xF; num3 < num10; num3 += 8) + { + num--; + num2 |= (z.next_in[next_in_index++] & 0xFF) << num3; + } + int num12 = array[(num9 + num8) * 3 + 2] + (num2 & inflate_mask[num10]); + num2 >>= num10; + num3 -= num10; + num5 -= num11; + int num13; + if (num4 >= num12) + { + num13 = num4 - num12; + if (num4 - num13 > 0 && 2 > num4 - num13) + { + s.window[num4++] = s.window[num13++]; + num11--; + s.window[num4++] = s.window[num13++]; + num11--; + } + else + { + Array.Copy(s.window, num13, s.window, num4, 2); + num4 += 2; + num13 += 2; + num11 -= 2; + } + } + else + { + num13 = num4 - num12; + do + { + num13 += s.end; + } + while (num13 < 0); + num10 = s.end - num13; + if (num11 > num10) + { + num11 -= num10; + if (num4 - num13 > 0 && num10 > num4 - num13) + { + do + { + s.window[num4++] = s.window[num13++]; + } + while (--num10 != 0); + } + else + { + Array.Copy(s.window, num13, s.window, num4, num10); + num4 += num10; + num13 += num10; + num10 = 0; + } + num13 = 0; + } + } + if (num4 - num13 > 0 && num11 > num4 - num13) + { + do + { + s.window[num4++] = s.window[num13++]; + } + while (--num11 != 0); + break; + } + Array.Copy(s.window, num13, s.window, num4, num11); + num4 += num11; + num13 += num11; + num11 = 0; + break; + } + if ((num10 & 0x40) == 0) + { + num8 += array[(num9 + num8) * 3 + 2]; + num8 += num2 & inflate_mask[num10]; + if ((num10 = array[(num9 + num8) * 3]) == 0) + { + num2 >>= array[(num9 + num8) * 3 + 1]; + num3 -= array[(num9 + num8) * 3 + 1]; + s.window[num4++] = (byte)array[(num9 + num8) * 3 + 2]; + num5--; + break; + } + bool flag2 = true; + continue; + } + if (((uint)num10 & 0x20u) != 0) + { + num11 = z.avail_in - num; + num11 = ((num3 >> 3 < num11) ? (num3 >> 3) : num11); + num += num11; + next_in_index -= num11; + num3 -= num11 << 3; + s.bitb = num2; + s.bitk = num3; + z.avail_in = num; + z.total_in += next_in_index - z.next_in_index; + z.next_in_index = next_in_index; + s.write = num4; + return 1; + } + z.msg = "invalid literal/length code"; + num11 = z.avail_in - num; + num11 = ((num3 >> 3 < num11) ? (num3 >> 3) : num11); + num += num11; + next_in_index -= num11; + num3 -= num11 << 3; + s.bitb = num2; + s.bitk = num3; + z.avail_in = num; + z.total_in += next_in_index - z.next_in_index; + z.next_in_index = next_in_index; + s.write = num4; + return -3; + } + } + if (num5 < 258 || num < 10) + { + break; + } + } + num11 = z.avail_in - num; + num11 = ((num3 >> 3 < num11) ? (num3 >> 3) : num11); + num += num11; + next_in_index -= num11; + num3 -= num11 << 3; + s.bitb = num2; + s.bitk = num3; + z.avail_in = num; + z.total_in += next_in_index - z.next_in_index; + z.next_in_index = next_in_index; + s.write = num4; + return 0; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs.meta new file mode 100644 index 00000000..e6eeab34 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfCodes.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 848289b37732055419a0cc1d6f28d36a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs new file mode 100644 index 00000000..49ad715d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs @@ -0,0 +1,475 @@ +using System; + +namespace ComponentAce.Compression.Libs.zlib +{ + internal sealed class InfTree + { + private const int MANY = 1440; + + private const int Z_OK = 0; + + private const int Z_STREAM_END = 1; + + private const int Z_NEED_DICT = 2; + + private const int Z_ERRNO = -1; + + private const int Z_STREAM_ERROR = -2; + + private const int Z_DATA_ERROR = -3; + + private const int Z_MEM_ERROR = -4; + + private const int Z_BUF_ERROR = -5; + + private const int Z_VERSION_ERROR = -6; + + internal const int fixed_bl = 9; + + internal const int fixed_bd = 5; + + internal static readonly int[] fixed_tl = new int[1536] + { + 96, 7, 256, 0, 8, 80, 0, 8, 16, 84, + 8, 115, 82, 7, 31, 0, 8, 112, 0, 8, + 48, 0, 9, 192, 80, 7, 10, 0, 8, 96, + 0, 8, 32, 0, 9, 160, 0, 8, 0, 0, + 8, 128, 0, 8, 64, 0, 9, 224, 80, 7, + 6, 0, 8, 88, 0, 8, 24, 0, 9, 144, + 83, 7, 59, 0, 8, 120, 0, 8, 56, 0, + 9, 208, 81, 7, 17, 0, 8, 104, 0, 8, + 40, 0, 9, 176, 0, 8, 8, 0, 8, 136, + 0, 8, 72, 0, 9, 240, 80, 7, 4, 0, + 8, 84, 0, 8, 20, 85, 8, 227, 83, 7, + 43, 0, 8, 116, 0, 8, 52, 0, 9, 200, + 81, 7, 13, 0, 8, 100, 0, 8, 36, 0, + 9, 168, 0, 8, 4, 0, 8, 132, 0, 8, + 68, 0, 9, 232, 80, 7, 8, 0, 8, 92, + 0, 8, 28, 0, 9, 152, 84, 7, 83, 0, + 8, 124, 0, 8, 60, 0, 9, 216, 82, 7, + 23, 0, 8, 108, 0, 8, 44, 0, 9, 184, + 0, 8, 12, 0, 8, 140, 0, 8, 76, 0, + 9, 248, 80, 7, 3, 0, 8, 82, 0, 8, + 18, 85, 8, 163, 83, 7, 35, 0, 8, 114, + 0, 8, 50, 0, 9, 196, 81, 7, 11, 0, + 8, 98, 0, 8, 34, 0, 9, 164, 0, 8, + 2, 0, 8, 130, 0, 8, 66, 0, 9, 228, + 80, 7, 7, 0, 8, 90, 0, 8, 26, 0, + 9, 148, 84, 7, 67, 0, 8, 122, 0, 8, + 58, 0, 9, 212, 82, 7, 19, 0, 8, 106, + 0, 8, 42, 0, 9, 180, 0, 8, 10, 0, + 8, 138, 0, 8, 74, 0, 9, 244, 80, 7, + 5, 0, 8, 86, 0, 8, 22, 192, 8, 0, + 83, 7, 51, 0, 8, 118, 0, 8, 54, 0, + 9, 204, 81, 7, 15, 0, 8, 102, 0, 8, + 38, 0, 9, 172, 0, 8, 6, 0, 8, 134, + 0, 8, 70, 0, 9, 236, 80, 7, 9, 0, + 8, 94, 0, 8, 30, 0, 9, 156, 84, 7, + 99, 0, 8, 126, 0, 8, 62, 0, 9, 220, + 82, 7, 27, 0, 8, 110, 0, 8, 46, 0, + 9, 188, 0, 8, 14, 0, 8, 142, 0, 8, + 78, 0, 9, 252, 96, 7, 256, 0, 8, 81, + 0, 8, 17, 85, 8, 131, 82, 7, 31, 0, + 8, 113, 0, 8, 49, 0, 9, 194, 80, 7, + 10, 0, 8, 97, 0, 8, 33, 0, 9, 162, + 0, 8, 1, 0, 8, 129, 0, 8, 65, 0, + 9, 226, 80, 7, 6, 0, 8, 89, 0, 8, + 25, 0, 9, 146, 83, 7, 59, 0, 8, 121, + 0, 8, 57, 0, 9, 210, 81, 7, 17, 0, + 8, 105, 0, 8, 41, 0, 9, 178, 0, 8, + 9, 0, 8, 137, 0, 8, 73, 0, 9, 242, + 80, 7, 4, 0, 8, 85, 0, 8, 21, 80, + 8, 258, 83, 7, 43, 0, 8, 117, 0, 8, + 53, 0, 9, 202, 81, 7, 13, 0, 8, 101, + 0, 8, 37, 0, 9, 170, 0, 8, 5, 0, + 8, 133, 0, 8, 69, 0, 9, 234, 80, 7, + 8, 0, 8, 93, 0, 8, 29, 0, 9, 154, + 84, 7, 83, 0, 8, 125, 0, 8, 61, 0, + 9, 218, 82, 7, 23, 0, 8, 109, 0, 8, + 45, 0, 9, 186, 0, 8, 13, 0, 8, 141, + 0, 8, 77, 0, 9, 250, 80, 7, 3, 0, + 8, 83, 0, 8, 19, 85, 8, 195, 83, 7, + 35, 0, 8, 115, 0, 8, 51, 0, 9, 198, + 81, 7, 11, 0, 8, 99, 0, 8, 35, 0, + 9, 166, 0, 8, 3, 0, 8, 131, 0, 8, + 67, 0, 9, 230, 80, 7, 7, 0, 8, 91, + 0, 8, 27, 0, 9, 150, 84, 7, 67, 0, + 8, 123, 0, 8, 59, 0, 9, 214, 82, 7, + 19, 0, 8, 107, 0, 8, 43, 0, 9, 182, + 0, 8, 11, 0, 8, 139, 0, 8, 75, 0, + 9, 246, 80, 7, 5, 0, 8, 87, 0, 8, + 23, 192, 8, 0, 83, 7, 51, 0, 8, 119, + 0, 8, 55, 0, 9, 206, 81, 7, 15, 0, + 8, 103, 0, 8, 39, 0, 9, 174, 0, 8, + 7, 0, 8, 135, 0, 8, 71, 0, 9, 238, + 80, 7, 9, 0, 8, 95, 0, 8, 31, 0, + 9, 158, 84, 7, 99, 0, 8, 127, 0, 8, + 63, 0, 9, 222, 82, 7, 27, 0, 8, 111, + 0, 8, 47, 0, 9, 190, 0, 8, 15, 0, + 8, 143, 0, 8, 79, 0, 9, 254, 96, 7, + 256, 0, 8, 80, 0, 8, 16, 84, 8, 115, + 82, 7, 31, 0, 8, 112, 0, 8, 48, 0, + 9, 193, 80, 7, 10, 0, 8, 96, 0, 8, + 32, 0, 9, 161, 0, 8, 0, 0, 8, 128, + 0, 8, 64, 0, 9, 225, 80, 7, 6, 0, + 8, 88, 0, 8, 24, 0, 9, 145, 83, 7, + 59, 0, 8, 120, 0, 8, 56, 0, 9, 209, + 81, 7, 17, 0, 8, 104, 0, 8, 40, 0, + 9, 177, 0, 8, 8, 0, 8, 136, 0, 8, + 72, 0, 9, 241, 80, 7, 4, 0, 8, 84, + 0, 8, 20, 85, 8, 227, 83, 7, 43, 0, + 8, 116, 0, 8, 52, 0, 9, 201, 81, 7, + 13, 0, 8, 100, 0, 8, 36, 0, 9, 169, + 0, 8, 4, 0, 8, 132, 0, 8, 68, 0, + 9, 233, 80, 7, 8, 0, 8, 92, 0, 8, + 28, 0, 9, 153, 84, 7, 83, 0, 8, 124, + 0, 8, 60, 0, 9, 217, 82, 7, 23, 0, + 8, 108, 0, 8, 44, 0, 9, 185, 0, 8, + 12, 0, 8, 140, 0, 8, 76, 0, 9, 249, + 80, 7, 3, 0, 8, 82, 0, 8, 18, 85, + 8, 163, 83, 7, 35, 0, 8, 114, 0, 8, + 50, 0, 9, 197, 81, 7, 11, 0, 8, 98, + 0, 8, 34, 0, 9, 165, 0, 8, 2, 0, + 8, 130, 0, 8, 66, 0, 9, 229, 80, 7, + 7, 0, 8, 90, 0, 8, 26, 0, 9, 149, + 84, 7, 67, 0, 8, 122, 0, 8, 58, 0, + 9, 213, 82, 7, 19, 0, 8, 106, 0, 8, + 42, 0, 9, 181, 0, 8, 10, 0, 8, 138, + 0, 8, 74, 0, 9, 245, 80, 7, 5, 0, + 8, 86, 0, 8, 22, 192, 8, 0, 83, 7, + 51, 0, 8, 118, 0, 8, 54, 0, 9, 205, + 81, 7, 15, 0, 8, 102, 0, 8, 38, 0, + 9, 173, 0, 8, 6, 0, 8, 134, 0, 8, + 70, 0, 9, 237, 80, 7, 9, 0, 8, 94, + 0, 8, 30, 0, 9, 157, 84, 7, 99, 0, + 8, 126, 0, 8, 62, 0, 9, 221, 82, 7, + 27, 0, 8, 110, 0, 8, 46, 0, 9, 189, + 0, 8, 14, 0, 8, 142, 0, 8, 78, 0, + 9, 253, 96, 7, 256, 0, 8, 81, 0, 8, + 17, 85, 8, 131, 82, 7, 31, 0, 8, 113, + 0, 8, 49, 0, 9, 195, 80, 7, 10, 0, + 8, 97, 0, 8, 33, 0, 9, 163, 0, 8, + 1, 0, 8, 129, 0, 8, 65, 0, 9, 227, + 80, 7, 6, 0, 8, 89, 0, 8, 25, 0, + 9, 147, 83, 7, 59, 0, 8, 121, 0, 8, + 57, 0, 9, 211, 81, 7, 17, 0, 8, 105, + 0, 8, 41, 0, 9, 179, 0, 8, 9, 0, + 8, 137, 0, 8, 73, 0, 9, 243, 80, 7, + 4, 0, 8, 85, 0, 8, 21, 80, 8, 258, + 83, 7, 43, 0, 8, 117, 0, 8, 53, 0, + 9, 203, 81, 7, 13, 0, 8, 101, 0, 8, + 37, 0, 9, 171, 0, 8, 5, 0, 8, 133, + 0, 8, 69, 0, 9, 235, 80, 7, 8, 0, + 8, 93, 0, 8, 29, 0, 9, 155, 84, 7, + 83, 0, 8, 125, 0, 8, 61, 0, 9, 219, + 82, 7, 23, 0, 8, 109, 0, 8, 45, 0, + 9, 187, 0, 8, 13, 0, 8, 141, 0, 8, + 77, 0, 9, 251, 80, 7, 3, 0, 8, 83, + 0, 8, 19, 85, 8, 195, 83, 7, 35, 0, + 8, 115, 0, 8, 51, 0, 9, 199, 81, 7, + 11, 0, 8, 99, 0, 8, 35, 0, 9, 167, + 0, 8, 3, 0, 8, 131, 0, 8, 67, 0, + 9, 231, 80, 7, 7, 0, 8, 91, 0, 8, + 27, 0, 9, 151, 84, 7, 67, 0, 8, 123, + 0, 8, 59, 0, 9, 215, 82, 7, 19, 0, + 8, 107, 0, 8, 43, 0, 9, 183, 0, 8, + 11, 0, 8, 139, 0, 8, 75, 0, 9, 247, + 80, 7, 5, 0, 8, 87, 0, 8, 23, 192, + 8, 0, 83, 7, 51, 0, 8, 119, 0, 8, + 55, 0, 9, 207, 81, 7, 15, 0, 8, 103, + 0, 8, 39, 0, 9, 175, 0, 8, 7, 0, + 8, 135, 0, 8, 71, 0, 9, 239, 80, 7, + 9, 0, 8, 95, 0, 8, 31, 0, 9, 159, + 84, 7, 99, 0, 8, 127, 0, 8, 63, 0, + 9, 223, 82, 7, 27, 0, 8, 111, 0, 8, + 47, 0, 9, 191, 0, 8, 15, 0, 8, 143, + 0, 8, 79, 0, 9, 255 + }; + + internal static readonly int[] fixed_td = new int[96] + { + 80, 5, 1, 87, 5, 257, 83, 5, 17, 91, + 5, 4097, 81, 5, 5, 89, 5, 1025, 85, 5, + 65, 93, 5, 16385, 80, 5, 3, 88, 5, 513, + 84, 5, 33, 92, 5, 8193, 82, 5, 9, 90, + 5, 2049, 86, 5, 129, 192, 5, 24577, 80, 5, + 2, 87, 5, 385, 83, 5, 25, 91, 5, 6145, + 81, 5, 7, 89, 5, 1537, 85, 5, 97, 93, + 5, 24577, 80, 5, 4, 88, 5, 769, 84, 5, + 49, 92, 5, 12289, 82, 5, 13, 90, 5, 3073, + 86, 5, 193, 192, 5, 24577 + }; + + internal static readonly int[] cplens = new int[31] + { + 3, 4, 5, 6, 7, 8, 9, 10, 11, 13, + 15, 17, 19, 23, 27, 31, 35, 43, 51, 59, + 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, + 0 + }; + + internal static readonly int[] cplext = new int[31] + { + 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, + 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, + 4, 4, 4, 4, 5, 5, 5, 5, 0, 112, + 112 + }; + + internal static readonly int[] cpdist = new int[30] + { + 1, 2, 3, 4, 5, 7, 9, 13, 17, 25, + 33, 49, 65, 97, 129, 193, 257, 385, 513, 769, + 1025, 1537, 2049, 3073, 4097, 6145, 8193, 12289, 16385, 24577 + }; + + internal static readonly int[] cpdext = new int[30] + { + 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, + 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, + 9, 9, 10, 10, 11, 11, 12, 12, 13, 13 + }; + + internal const int BMAX = 15; + + internal static int huft_build(int[] b, int bindex, int n, int s, int[] d, int[] e, int[] t, int[] m, int[] hp, int[] hn, int[] v) + { + int[] array = new int[16]; + int[] array2 = new int[3]; + int[] array3 = new int[15]; + int[] array4 = new int[16]; + int num = 0; + int num2 = n; + do + { + array[b[bindex + num]]++; + num++; + num2--; + } + while (num2 != 0); + if (array[0] == n) + { + t[0] = -1; + m[0] = 0; + return 0; + } + int num3 = m[0]; + int i; + for (i = 1; i <= 15 && array[i] == 0; i++) + { + } + int j = i; + if (num3 < i) + { + num3 = i; + } + num2 = 15; + while (num2 != 0 && array[num2] == 0) + { + num2--; + } + int num4 = num2; + if (num3 > num2) + { + num3 = num2; + } + m[0] = num3; + int num5 = 1 << i; + while (i < num2) + { + if ((num5 -= array[i]) < 0) + { + return -3; + } + i++; + num5 <<= 1; + } + if ((num5 -= array[num2]) < 0) + { + return -3; + } + array[num2] += num5; + i = (array4[1] = 0); + num = 1; + int num6 = 2; + while (--num2 != 0) + { + i = (array4[num6] = i + array[num]); + num6++; + num++; + } + num2 = 0; + num = 0; + do + { + if ((i = b[bindex + num]) != 0) + { + v[array4[i]++] = num2; + } + num++; + } + while (++num2 < n); + n = array4[num4]; + num2 = (array4[0] = 0); + num = 0; + int num7 = -1; + int num8 = -num3; + array3[0] = 0; + int num9 = 0; + int num10 = 0; + for (; j <= num4; j++) + { + int num11 = array[j]; + while (num11-- != 0) + { + int num12; + while (j > num8 + num3) + { + num7++; + num8 += num3; + num10 = num4 - num8; + num10 = ((num10 > num3) ? num3 : num10); + if ((num12 = 1 << (i = j - num8)) > num11 + 1) + { + num12 -= num11 + 1; + num6 = j; + if (i < num10) + { + while (++i < num10 && (num12 <<= 1) > array[++num6]) + { + num12 -= array[num6]; + } + } + } + num10 = 1 << i; + if (hn[0] + num10 > 1440) + { + return -3; + } + num9 = (array3[num7] = hn[0]); + hn[0] += num10; + if (num7 != 0) + { + array4[num7] = num2; + array2[0] = (byte)i; + array2[1] = (byte)num3; + i = SupportClass.URShift(num2, num8 - num3); + array2[2] = num9 - array3[num7 - 1] - i; + Array.Copy(array2, 0, hp, (array3[num7 - 1] + i) * 3, 3); + } + else + { + t[0] = num9; + } + } + array2[1] = (byte)(j - num8); + if (num >= n) + { + array2[0] = 192; + } + else if (v[num] < s) + { + array2[0] = (byte)((v[num] >= 256) ? 96 : 0); + array2[2] = v[num++]; + } + else + { + array2[0] = (byte)(e[v[num] - s] + 16 + 64); + array2[2] = d[v[num++] - s]; + } + num12 = 1 << j - num8; + for (i = SupportClass.URShift(num2, num8); i < num10; i += num12) + { + Array.Copy(array2, 0, hp, (num9 + i) * 3, 3); + } + i = 1 << j - 1; + while ((num2 & i) != 0) + { + num2 ^= i; + i = SupportClass.URShift(i, 1); + } + num2 ^= i; + int num13 = (1 << num8) - 1; + while ((num2 & num13) != array4[num7]) + { + num7--; + num8 -= num3; + num13 = (1 << num8) - 1; + } + } + } + return (num5 != 0 && num4 != 1) ? (-5) : 0; + } + + internal static int inflate_trees_bits(int[] c, int[] bb, int[] tb, int[] hp, ZStream z) + { + int[] hn = new int[1]; + int[] v = new int[19]; + int num = huft_build(c, 0, 19, 19, null, null, tb, bb, hp, hn, v); + if (num == -3) + { + z.msg = "oversubscribed dynamic bit lengths tree"; + } + else if (num == -5 || bb[0] == 0) + { + z.msg = "incomplete dynamic bit lengths tree"; + num = -3; + } + return num; + } + + internal static int inflate_trees_dynamic(int nl, int nd, int[] c, int[] bl, int[] bd, int[] tl, int[] td, int[] hp, ZStream z) + { + int[] hn = new int[1]; + int[] v = new int[288]; + int num = huft_build(c, 0, nl, 257, cplens, cplext, tl, bl, hp, hn, v); + if (num != 0 || bl[0] == 0) + { + switch (num) + { + case -3: + z.msg = "oversubscribed literal/length tree"; + break; + default: + z.msg = "incomplete literal/length tree"; + num = -3; + break; + case -4: + break; + } + return num; + } + num = huft_build(c, nl, nd, 0, cpdist, cpdext, td, bd, hp, hn, v); + if (num != 0 || (bd[0] == 0 && nl > 257)) + { + switch (num) + { + case -3: + z.msg = "oversubscribed distance tree"; + break; + case -5: + z.msg = "incomplete distance tree"; + num = -3; + break; + default: + z.msg = "empty distance tree with lengths"; + num = -3; + break; + case -4: + break; + } + return num; + } + return 0; + } + + internal static int inflate_trees_fixed(int[] bl, int[] bd, int[][] tl, int[][] td, ZStream z) + { + bl[0] = 9; + bd[0] = 5; + tl[0] = fixed_tl; + td[0] = fixed_td; + return 0; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs.meta new file mode 100644 index 00000000..23d1b6ae --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/InfTree.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 12c1095bcb2a9e748af19115781c4483 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs new file mode 100644 index 00000000..8fac4064 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs @@ -0,0 +1,409 @@ +namespace ComponentAce.Compression.Libs.zlib +{ + internal sealed class Inflate + { + private const int MAX_WBITS = 15; + + private const int PRESET_DICT = 32; + + internal const int Z_NO_FLUSH = 0; + + internal const int Z_PARTIAL_FLUSH = 1; + + internal const int Z_SYNC_FLUSH = 2; + + internal const int Z_FULL_FLUSH = 3; + + internal const int Z_FINISH = 4; + + private const int Z_DEFLATED = 8; + + private const int Z_OK = 0; + + private const int Z_STREAM_END = 1; + + private const int Z_NEED_DICT = 2; + + private const int Z_ERRNO = -1; + + private const int Z_STREAM_ERROR = -2; + + private const int Z_DATA_ERROR = -3; + + private const int Z_MEM_ERROR = -4; + + private const int Z_BUF_ERROR = -5; + + private const int Z_VERSION_ERROR = -6; + + private const int METHOD = 0; + + private const int FLAG = 1; + + private const int DICT4 = 2; + + private const int DICT3 = 3; + + private const int DICT2 = 4; + + private const int DICT1 = 5; + + private const int DICT0 = 6; + + private const int BLOCKS = 7; + + private const int CHECK4 = 8; + + private const int CHECK3 = 9; + + private const int CHECK2 = 10; + + private const int CHECK1 = 11; + + private const int DONE = 12; + + private const int BAD = 13; + + internal int mode; + + internal int method; + + internal long[] was = new long[1]; + + internal long need; + + internal int marker; + + internal int nowrap; + + internal int wbits; + + internal InfBlocks blocks; + + private static byte[] mark = new byte[4] + { + 0, + 0, + (byte)SupportClass.Identity(255L), + (byte)SupportClass.Identity(255L) + }; + + internal int inflateReset(ZStream z) + { + if (z == null || z.istate == null) + { + return -2; + } + z.total_in = (z.total_out = 0L); + z.msg = null; + z.istate.mode = ((z.istate.nowrap != 0) ? 7 : 0); + z.istate.blocks.reset(z, null); + return 0; + } + + internal int inflateEnd(ZStream z) + { + if (blocks != null) + { + blocks.free(z); + } + blocks = null; + return 0; + } + + internal int inflateInit(ZStream z, int w) + { + z.msg = null; + blocks = null; + nowrap = 0; + if (w < 0) + { + w = -w; + nowrap = 1; + } + if (w < 8 || w > 15) + { + inflateEnd(z); + return -2; + } + wbits = w; + z.istate.blocks = new InfBlocks(z, (z.istate.nowrap != 0) ? null : this, 1 << w); + inflateReset(z); + return 0; + } + + internal int inflate(ZStream z, int f) + { + if (z == null || z.istate == null || z.next_in == null) + { + return -2; + } + f = ((f == 4) ? (-5) : 0); + int num = -5; + while (true) + { + switch (z.istate.mode) + { + case 0: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + if (((z.istate.method = z.next_in[z.next_in_index++]) & 0xF) != 8) + { + z.istate.mode = 13; + z.msg = "unknown compression method"; + z.istate.marker = 5; + break; + } + if ((z.istate.method >> 4) + 8 > z.istate.wbits) + { + z.istate.mode = 13; + z.msg = "invalid window size"; + z.istate.marker = 5; + break; + } + z.istate.mode = 1; + goto case 1; + case 1: + { + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + int num2 = z.next_in[z.next_in_index++] & 0xFF; + if (((z.istate.method << 8) + num2) % 31 != 0) + { + z.istate.mode = 13; + z.msg = "incorrect header check"; + z.istate.marker = 5; + break; + } + if ((num2 & 0x20) == 0) + { + z.istate.mode = 7; + break; + } + z.istate.mode = 2; + goto case 2; + } + case 2: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need = ((z.next_in[z.next_in_index++] & 0xFF) << 24) & -16777216; + z.istate.mode = 3; + goto case 3; + case 3: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need += (long)((ulong)((z.next_in[z.next_in_index++] & 0xFF) << 16) & 0xFF0000uL); + z.istate.mode = 4; + goto case 4; + case 4: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need += (long)((ulong)((z.next_in[z.next_in_index++] & 0xFF) << 8) & 0xFF00uL); + z.istate.mode = 5; + goto case 5; + case 5: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need += (long)((ulong)z.next_in[z.next_in_index++] & 0xFFuL); + z.adler = z.istate.need; + z.istate.mode = 6; + return 2; + case 6: + z.istate.mode = 13; + z.msg = "need dictionary"; + z.istate.marker = 0; + return -2; + case 7: + num = z.istate.blocks.proc(z, num); + switch (num) + { + case -3: + z.istate.mode = 13; + z.istate.marker = 0; + goto end_IL_004b; + case 0: + num = f; + break; + } + if (num != 1) + { + return num; + } + num = f; + z.istate.blocks.reset(z, z.istate.was); + if (z.istate.nowrap != 0) + { + z.istate.mode = 12; + break; + } + z.istate.mode = 8; + goto case 8; + case 8: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need = ((z.next_in[z.next_in_index++] & 0xFF) << 24) & -16777216; + z.istate.mode = 9; + goto case 9; + case 9: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need += (long)((ulong)((z.next_in[z.next_in_index++] & 0xFF) << 16) & 0xFF0000uL); + z.istate.mode = 10; + goto case 10; + case 10: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need += (long)((ulong)((z.next_in[z.next_in_index++] & 0xFF) << 8) & 0xFF00uL); + z.istate.mode = 11; + goto case 11; + case 11: + if (z.avail_in == 0) + { + return num; + } + num = f; + z.avail_in--; + z.total_in++; + z.istate.need += (long)((ulong)z.next_in[z.next_in_index++] & 0xFFuL); + if ((int)z.istate.was[0] != (int)z.istate.need) + { + z.istate.mode = 13; + z.msg = "incorrect data check"; + z.istate.marker = 5; + break; + } + z.istate.mode = 12; + goto case 12; + case 12: + return 1; + case 13: + return -3; + default: + { + return -2; + } + end_IL_004b: + break; + } + } + } + + internal int inflateSetDictionary(ZStream z, byte[] dictionary, int dictLength) + { + int start = 0; + int num = dictLength; + if (z == null || z.istate == null || z.istate.mode != 6) + { + return -2; + } + if (z._adler.adler32(1L, dictionary, 0, dictLength) != z.adler) + { + return -3; + } + z.adler = z._adler.adler32(0L, null, 0, 0); + if (num >= 1 << z.istate.wbits) + { + num = (1 << z.istate.wbits) - 1; + start = dictLength - num; + } + z.istate.blocks.set_dictionary(dictionary, start, num); + z.istate.mode = 7; + return 0; + } + + internal int inflateSync(ZStream z) + { + if (z == null || z.istate == null) + { + return -2; + } + if (z.istate.mode != 13) + { + z.istate.mode = 13; + z.istate.marker = 0; + } + int num; + if ((num = z.avail_in) == 0) + { + return -5; + } + int num2 = z.next_in_index; + int num3 = z.istate.marker; + while (num != 0 && num3 < 4) + { + num3 = ((z.next_in[num2] != mark[num3]) ? ((z.next_in[num2] == 0) ? (4 - num3) : 0) : (num3 + 1)); + num2++; + num--; + } + z.total_in += num2 - z.next_in_index; + z.next_in_index = num2; + z.avail_in = num; + z.istate.marker = num3; + if (num3 != 4) + { + return -3; + } + long total_in = z.total_in; + long total_out = z.total_out; + inflateReset(z); + z.total_in = total_in; + z.total_out = total_out; + z.istate.mode = 7; + return 0; + } + + internal int inflateSyncPoint(ZStream z) + { + if (z == null || z.istate == null || z.istate.blocks == null) + { + return -2; + } + return z.istate.blocks.sync_point(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs.meta new file mode 100644 index 00000000..a814ad81 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Inflate.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f0c7c284a3a696541b95af8150097d7d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs new file mode 100644 index 00000000..a5964a87 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs @@ -0,0 +1,126 @@ +namespace ComponentAce.Compression.Libs.zlib +{ + internal sealed class StaticTree + { + private const int MAX_BITS = 15; + + private const int BL_CODES = 19; + + private const int D_CODES = 30; + + private const int LITERALS = 256; + + private const int LENGTH_CODES = 29; + + private static readonly int L_CODES; + + internal const int MAX_BL_BITS = 7; + + internal static readonly short[] static_ltree; + + internal static readonly short[] static_dtree; + + internal static StaticTree static_l_desc; + + internal static StaticTree static_d_desc; + + internal static StaticTree static_bl_desc; + + internal short[] static_tree; + + internal int[] extra_bits; + + internal int extra_base; + + internal int elems; + + internal int max_length; + + internal StaticTree(short[] static_tree, int[] extra_bits, int extra_base, int elems, int max_length) + { + this.static_tree = static_tree; + this.extra_bits = extra_bits; + this.extra_base = extra_base; + this.elems = elems; + this.max_length = max_length; + } + + static StaticTree() + { + L_CODES = 286; + static_ltree = new short[576] + { + 12, 8, 140, 8, 76, 8, 204, 8, 44, 8, + 172, 8, 108, 8, 236, 8, 28, 8, 156, 8, + 92, 8, 220, 8, 60, 8, 188, 8, 124, 8, + 252, 8, 2, 8, 130, 8, 66, 8, 194, 8, + 34, 8, 162, 8, 98, 8, 226, 8, 18, 8, + 146, 8, 82, 8, 210, 8, 50, 8, 178, 8, + 114, 8, 242, 8, 10, 8, 138, 8, 74, 8, + 202, 8, 42, 8, 170, 8, 106, 8, 234, 8, + 26, 8, 154, 8, 90, 8, 218, 8, 58, 8, + 186, 8, 122, 8, 250, 8, 6, 8, 134, 8, + 70, 8, 198, 8, 38, 8, 166, 8, 102, 8, + 230, 8, 22, 8, 150, 8, 86, 8, 214, 8, + 54, 8, 182, 8, 118, 8, 246, 8, 14, 8, + 142, 8, 78, 8, 206, 8, 46, 8, 174, 8, + 110, 8, 238, 8, 30, 8, 158, 8, 94, 8, + 222, 8, 62, 8, 190, 8, 126, 8, 254, 8, + 1, 8, 129, 8, 65, 8, 193, 8, 33, 8, + 161, 8, 97, 8, 225, 8, 17, 8, 145, 8, + 81, 8, 209, 8, 49, 8, 177, 8, 113, 8, + 241, 8, 9, 8, 137, 8, 73, 8, 201, 8, + 41, 8, 169, 8, 105, 8, 233, 8, 25, 8, + 153, 8, 89, 8, 217, 8, 57, 8, 185, 8, + 121, 8, 249, 8, 5, 8, 133, 8, 69, 8, + 197, 8, 37, 8, 165, 8, 101, 8, 229, 8, + 21, 8, 149, 8, 85, 8, 213, 8, 53, 8, + 181, 8, 117, 8, 245, 8, 13, 8, 141, 8, + 77, 8, 205, 8, 45, 8, 173, 8, 109, 8, + 237, 8, 29, 8, 157, 8, 93, 8, 221, 8, + 61, 8, 189, 8, 125, 8, 253, 8, 19, 9, + 275, 9, 147, 9, 403, 9, 83, 9, 339, 9, + 211, 9, 467, 9, 51, 9, 307, 9, 179, 9, + 435, 9, 115, 9, 371, 9, 243, 9, 499, 9, + 11, 9, 267, 9, 139, 9, 395, 9, 75, 9, + 331, 9, 203, 9, 459, 9, 43, 9, 299, 9, + 171, 9, 427, 9, 107, 9, 363, 9, 235, 9, + 491, 9, 27, 9, 283, 9, 155, 9, 411, 9, + 91, 9, 347, 9, 219, 9, 475, 9, 59, 9, + 315, 9, 187, 9, 443, 9, 123, 9, 379, 9, + 251, 9, 507, 9, 7, 9, 263, 9, 135, 9, + 391, 9, 71, 9, 327, 9, 199, 9, 455, 9, + 39, 9, 295, 9, 167, 9, 423, 9, 103, 9, + 359, 9, 231, 9, 487, 9, 23, 9, 279, 9, + 151, 9, 407, 9, 87, 9, 343, 9, 215, 9, + 471, 9, 55, 9, 311, 9, 183, 9, 439, 9, + 119, 9, 375, 9, 247, 9, 503, 9, 15, 9, + 271, 9, 143, 9, 399, 9, 79, 9, 335, 9, + 207, 9, 463, 9, 47, 9, 303, 9, 175, 9, + 431, 9, 111, 9, 367, 9, 239, 9, 495, 9, + 31, 9, 287, 9, 159, 9, 415, 9, 95, 9, + 351, 9, 223, 9, 479, 9, 63, 9, 319, 9, + 191, 9, 447, 9, 127, 9, 383, 9, 255, 9, + 511, 9, 0, 7, 64, 7, 32, 7, 96, 7, + 16, 7, 80, 7, 48, 7, 112, 7, 8, 7, + 72, 7, 40, 7, 104, 7, 24, 7, 88, 7, + 56, 7, 120, 7, 4, 7, 68, 7, 36, 7, + 100, 7, 20, 7, 84, 7, 52, 7, 116, 7, + 3, 8, 131, 8, 67, 8, 195, 8, 35, 8, + 163, 8, 99, 8, 227, 8 + }; + static_dtree = new short[60] + { + 0, 5, 16, 5, 8, 5, 24, 5, 4, 5, + 20, 5, 12, 5, 28, 5, 2, 5, 18, 5, + 10, 5, 26, 5, 6, 5, 22, 5, 14, 5, + 30, 5, 1, 5, 17, 5, 9, 5, 25, 5, + 5, 5, 21, 5, 13, 5, 29, 5, 3, 5, + 19, 5, 11, 5, 27, 5, 7, 5, 23, 5 + }; + static_l_desc = new StaticTree(static_ltree, Tree.extra_lbits, 257, L_CODES, 15); + static_d_desc = new StaticTree(static_dtree, Tree.extra_dbits, 0, 30, 15); + static_bl_desc = new StaticTree(null, Tree.extra_blbits, 0, 19, 7); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs.meta new file mode 100644 index 00000000..ea36f10f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/StaticTree.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6d5351586a003ca4da795d2ce129ba8e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs new file mode 100644 index 00000000..7b9f174c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs @@ -0,0 +1,104 @@ +using System.IO; +using System.Text; + +namespace ComponentAce.Compression.Libs.zlib +{ + public class SupportClass + { + public static long Identity(long literal) + { + return literal; + } + + public static ulong Identity(ulong literal) + { + return literal; + } + + public static float Identity(float literal) + { + return literal; + } + + public static double Identity(double literal) + { + return literal; + } + + public static int URShift(int number, int bits) + { + if (number >= 0) + { + return number >> bits; + } + return (number >> bits) + (2 << ~bits); + } + + public static int URShift(int number, long bits) + { + return URShift(number, (int)bits); + } + + public static long URShift(long number, int bits) + { + if (number >= 0) + { + return number >> bits; + } + return (number >> bits) + (2L << ~bits); + } + + public static long URShift(long number, long bits) + { + return URShift(number, (int)bits); + } + + public static int ReadInput(Stream sourceStream, byte[] target, int start, int count) + { + if (target.Length == 0) + { + return 0; + } + byte[] array = new byte[target.Length]; + int num = sourceStream.Read(array, start, count); + if (num == 0) + { + return -1; + } + for (int i = start; i < start + num; i++) + { + target[i] = array[i]; + } + return num; + } + + public static int ReadInput(TextReader sourceTextReader, byte[] target, int start, int count) + { + if (target.Length == 0) + { + return 0; + } + char[] array = new char[target.Length]; + int num = sourceTextReader.Read(array, start, count); + if (num == 0) + { + return -1; + } + for (int i = start; i < start + num; i++) + { + target[i] = (byte)array[i]; + } + return num; + } + + public static byte[] ToByteArray(string sourceString) + { + return Encoding.UTF8.GetBytes(sourceString); + } + + public static char[] ToCharArray(byte[] byteArray) + { + return Encoding.UTF8.GetChars(byteArray); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs.meta new file mode 100644 index 00000000..f145dcd9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/SupportClass.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 192062774319d2845b2df445d3215268 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs new file mode 100644 index 00000000..00b12362 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs @@ -0,0 +1,337 @@ +using System; + +namespace ComponentAce.Compression.Libs.zlib +{ + internal sealed class Tree + { + private const int MAX_BITS = 15; + + private const int BL_CODES = 19; + + private const int D_CODES = 30; + + private const int LITERALS = 256; + + private const int LENGTH_CODES = 29; + + private static readonly int L_CODES = 286; + + private static readonly int HEAP_SIZE = 2 * L_CODES + 1; + + internal const int MAX_BL_BITS = 7; + + internal const int END_BLOCK = 256; + + internal const int REP_3_6 = 16; + + internal const int REPZ_3_10 = 17; + + internal const int REPZ_11_138 = 18; + + internal static readonly int[] extra_lbits = new int[29] + { + 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, + 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, + 4, 4, 4, 4, 5, 5, 5, 5, 0 + }; + + internal static readonly int[] extra_dbits = new int[30] + { + 0, 0, 0, 0, 1, 1, 2, 2, 3, 3, + 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, + 9, 9, 10, 10, 11, 11, 12, 12, 13, 13 + }; + + internal static readonly int[] extra_blbits = new int[19] + { + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 2, 3, 7 + }; + + internal static readonly byte[] bl_order = new byte[19] + { + 16, 17, 18, 0, 8, 7, 9, 6, 10, 5, + 11, 4, 12, 3, 13, 2, 14, 1, 15 + }; + + internal const int Buf_size = 16; + + internal const int DIST_CODE_LEN = 512; + + internal static readonly byte[] _dist_code = new byte[512] + { + 0, 1, 2, 3, 4, 4, 5, 5, 6, 6, + 6, 6, 7, 7, 7, 7, 8, 8, 8, 8, + 8, 8, 8, 8, 9, 9, 9, 9, 9, 9, + 9, 9, 10, 10, 10, 10, 10, 10, 10, 10, + 10, 10, 10, 10, 10, 10, 10, 10, 11, 11, + 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, + 11, 11, 11, 11, 12, 12, 12, 12, 12, 12, + 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, + 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, + 12, 12, 12, 12, 12, 12, 13, 13, 13, 13, + 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, + 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, + 13, 13, 13, 13, 13, 13, 13, 13, 14, 14, + 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, + 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, + 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, + 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, + 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, + 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, + 14, 14, 15, 15, 15, 15, 15, 15, 15, 15, + 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, + 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, + 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, + 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, + 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, + 15, 15, 15, 15, 15, 15, 0, 0, 16, 17, + 18, 18, 19, 19, 20, 20, 20, 20, 21, 21, + 21, 21, 22, 22, 22, 22, 22, 22, 22, 22, + 23, 23, 23, 23, 23, 23, 23, 23, 24, 24, + 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, + 24, 24, 24, 24, 25, 25, 25, 25, 25, 25, + 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, + 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, + 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, + 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, + 26, 26, 27, 27, 27, 27, 27, 27, 27, 27, + 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, + 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, + 27, 27, 27, 27, 28, 28, 28, 28, 28, 28, + 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, + 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, + 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, + 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, + 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, + 28, 28, 28, 28, 28, 28, 28, 28, 29, 29, + 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, + 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, + 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, + 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, + 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, + 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, + 29, 29 + }; + + internal static readonly byte[] _length_code = new byte[256] + { + 0, 1, 2, 3, 4, 5, 6, 7, 8, 8, + 9, 9, 10, 10, 11, 11, 12, 12, 12, 12, + 13, 13, 13, 13, 14, 14, 14, 14, 15, 15, + 15, 15, 16, 16, 16, 16, 16, 16, 16, 16, + 17, 17, 17, 17, 17, 17, 17, 17, 18, 18, + 18, 18, 18, 18, 18, 18, 19, 19, 19, 19, + 19, 19, 19, 19, 20, 20, 20, 20, 20, 20, + 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, + 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, + 21, 21, 21, 21, 21, 21, 22, 22, 22, 22, + 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, + 22, 22, 23, 23, 23, 23, 23, 23, 23, 23, + 23, 23, 23, 23, 23, 23, 23, 23, 24, 24, + 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, + 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, + 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, + 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, + 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, + 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, + 25, 25, 26, 26, 26, 26, 26, 26, 26, 26, + 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, + 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, + 26, 26, 26, 26, 27, 27, 27, 27, 27, 27, + 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, + 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, + 27, 27, 27, 27, 27, 28 + }; + + internal static readonly int[] base_length = new int[29] + { + 0, 1, 2, 3, 4, 5, 6, 7, 8, 10, + 12, 14, 16, 20, 24, 28, 32, 40, 48, 56, + 64, 80, 96, 112, 128, 160, 192, 224, 0 + }; + + internal static readonly int[] base_dist = new int[30] + { + 0, 1, 2, 3, 4, 6, 8, 12, 16, 24, + 32, 48, 64, 96, 128, 192, 256, 384, 512, 768, + 1024, 1536, 2048, 3072, 4096, 6144, 8192, 12288, 16384, 24576 + }; + + internal short[] dyn_tree; + + internal int max_code; + + internal StaticTree stat_desc; + + internal static int d_code(int dist) + { + return (dist < 256) ? _dist_code[dist] : _dist_code[256 + SupportClass.URShift(dist, 7)]; + } + + internal void gen_bitlen(Deflate s) + { + short[] array = dyn_tree; + short[] static_tree = stat_desc.static_tree; + int[] extra_bits = stat_desc.extra_bits; + int extra_base = stat_desc.extra_base; + int max_length = stat_desc.max_length; + int num = 0; + for (int i = 0; i <= 15; i++) + { + s.bl_count[i] = 0; + } + array[s.heap[s.heap_max] * 2 + 1] = 0; + int j; + for (j = s.heap_max + 1; j < HEAP_SIZE; j++) + { + int num2 = s.heap[j]; + int i = array[array[num2 * 2 + 1] * 2 + 1] + 1; + if (i > max_length) + { + i = max_length; + num++; + } + array[num2 * 2 + 1] = (short)i; + if (num2 <= max_code) + { + s.bl_count[i]++; + int num3 = 0; + if (num2 >= extra_base) + { + num3 = extra_bits[num2 - extra_base]; + } + short num4 = array[num2 * 2]; + s.opt_len += num4 * (i + num3); + if (static_tree != null) + { + s.static_len += num4 * (static_tree[num2 * 2 + 1] + num3); + } + } + } + if (num == 0) + { + return; + } + do + { + int i = max_length - 1; + while (s.bl_count[i] == 0) + { + i--; + } + s.bl_count[i]--; + s.bl_count[i + 1] = (short)(s.bl_count[i + 1] + 2); + s.bl_count[max_length]--; + num -= 2; + } + while (num > 0); + for (int i = max_length; i != 0; i--) + { + int num2 = s.bl_count[i]; + while (num2 != 0) + { + int num5 = s.heap[--j]; + if (num5 <= max_code) + { + if (array[num5 * 2 + 1] != i) + { + s.opt_len = (int)(s.opt_len + ((long)i - (long)array[num5 * 2 + 1]) * array[num5 * 2]); + array[num5 * 2 + 1] = (short)i; + } + num2--; + } + } + } + } + + internal void build_tree(Deflate s) + { + short[] array = dyn_tree; + short[] static_tree = stat_desc.static_tree; + int elems = stat_desc.elems; + int num = -1; + s.heap_len = 0; + s.heap_max = HEAP_SIZE; + for (int i = 0; i < elems; i++) + { + if (array[i * 2] != 0) + { + num = (s.heap[++s.heap_len] = i); + s.depth[i] = 0; + } + else + { + array[i * 2 + 1] = 0; + } + } + int num2; + while (s.heap_len < 2) + { + num2 = (s.heap[++s.heap_len] = ((num < 2) ? (++num) : 0)); + array[num2 * 2] = 1; + s.depth[num2] = 0; + s.opt_len--; + if (static_tree != null) + { + s.static_len -= static_tree[num2 * 2 + 1]; + } + } + max_code = num; + for (int i = s.heap_len / 2; i >= 1; i--) + { + s.pqdownheap(array, i); + } + num2 = elems; + do + { + int i = s.heap[1]; + s.heap[1] = s.heap[s.heap_len--]; + s.pqdownheap(array, 1); + int num3 = s.heap[1]; + s.heap[--s.heap_max] = i; + s.heap[--s.heap_max] = num3; + array[num2 * 2] = (short)(array[i * 2] + array[num3 * 2]); + s.depth[num2] = (byte)(Math.Max(s.depth[i], s.depth[num3]) + 1); + array[i * 2 + 1] = (array[num3 * 2 + 1] = (short)num2); + s.heap[1] = num2++; + s.pqdownheap(array, 1); + } + while (s.heap_len >= 2); + s.heap[--s.heap_max] = s.heap[1]; + gen_bitlen(s); + gen_codes(array, num, s.bl_count); + } + + internal static void gen_codes(short[] tree, int max_code, short[] bl_count) + { + short[] array = new short[16]; + short num = 0; + for (int i = 1; i <= 15; i++) + { + num = (array[i] = (short)(num + bl_count[i - 1] << 1)); + } + for (int j = 0; j <= max_code; j++) + { + int num2 = tree[j * 2 + 1]; + if (num2 != 0) + { + tree[j * 2] = (short)bi_reverse(array[num2]++, num2); + } + } + } + + internal static int bi_reverse(int code, int len) + { + int num = 0; + do + { + num |= code & 1; + code = SupportClass.URShift(code, 1); + num <<= 1; + } + while (--len > 0); + return SupportClass.URShift(num, 1); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs.meta new file mode 100644 index 00000000..dc2f8356 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/Tree.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 09b036b3d0426fe42bf7d0dd3bd0cca9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs new file mode 100644 index 00000000..4cc396bb --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs @@ -0,0 +1,134 @@ +using System.IO; + +namespace ComponentAce.Compression.Libs.zlib +{ + public class ZInputStream : BinaryReader + { + protected ZStream z = new ZStream(); + + protected int bufsize = 512; + + protected int flush; + + protected byte[] buf; + + protected byte[] buf1 = new byte[1]; + + protected bool compress; + + internal Stream in_Renamed = null; + + internal bool nomoreinput = false; + + public virtual int FlushMode + { + get + { + return flush; + } + set + { + flush = value; + } + } + + public virtual long TotalIn => z.total_in; + + public virtual long TotalOut => z.total_out; + + internal void InitBlock() + { + flush = 0; + buf = new byte[bufsize]; + } + + public ZInputStream(Stream in_Renamed) + : base(in_Renamed) + { + InitBlock(); + this.in_Renamed = in_Renamed; + z.inflateInit(); + compress = false; + z.next_in = buf; + z.next_in_index = 0; + z.avail_in = 0; + } + + public ZInputStream(Stream in_Renamed, int level) + : base(in_Renamed) + { + InitBlock(); + this.in_Renamed = in_Renamed; + z.deflateInit(level); + compress = true; + z.next_in = buf; + z.next_in_index = 0; + z.avail_in = 0; + } + + public override int Read() + { + if (read(buf1, 0, 1) == -1) + { + return -1; + } + return buf1[0] & 0xFF; + } + + public int read(byte[] b, int off, int len) + { + if (len == 0) + { + return 0; + } + z.next_out = b; + z.next_out_index = off; + z.avail_out = len; + int num; + do + { + if (z.avail_in == 0 && !nomoreinput) + { + z.next_in_index = 0; + z.avail_in = SupportClass.ReadInput(in_Renamed, buf, 0, bufsize); + if (z.avail_in == -1) + { + z.avail_in = 0; + nomoreinput = true; + } + } + num = ((!compress) ? z.inflate(flush) : z.deflate(flush)); + if (nomoreinput && num == -5) + { + return -1; + } + if (num != 0 && num != 1) + { + throw new ZStreamException((compress ? "de" : "in") + "flating: " + z.msg); + } + if (nomoreinput && z.avail_out == len) + { + return -1; + } + } + while (z.avail_out == len && num == 0); + return len - z.avail_out; + } + + public long skip(long n) + { + int num = 512; + if (n < num) + { + num = (int)n; + } + byte[] array = new byte[num]; + return SupportClass.ReadInput(BaseStream, array, 0, array.Length); + } + + public override void Close() + { + in_Renamed.Close(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs.meta new file mode 100644 index 00000000..e9ed2b9c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZInputStream.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 005058a4e4fb325449d934f67cfa0ce9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs new file mode 100644 index 00000000..c0c0e382 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs @@ -0,0 +1,193 @@ +using System; +using System.IO; + +namespace ComponentAce.Compression.Libs.zlib +{ + public class ZOutputStream : Stream + { + protected internal ZStream z = new ZStream(); + + protected internal int bufsize = 4096; + + protected internal int flush_Renamed_Field; + + protected internal byte[] buf; + + protected internal byte[] buf1 = new byte[1]; + + protected internal bool compress; + + private Stream out_Renamed; + + public virtual int FlushMode + { + get + { + return flush_Renamed_Field; + } + set + { + flush_Renamed_Field = value; + } + } + + public virtual long TotalIn => z.total_in; + + public virtual long TotalOut => z.total_out; + + public override bool CanRead => false; + + public override bool CanSeek => false; + + public override bool CanWrite => false; + + public override long Length => 0L; + + public override long Position + { + get + { + return 0L; + } + set + { + } + } + + private void InitBlock() + { + flush_Renamed_Field = 0; + buf = new byte[bufsize]; + } + + public ZOutputStream(Stream out_Renamed) + { + InitBlock(); + this.out_Renamed = out_Renamed; + z.inflateInit(); + compress = false; + } + + public ZOutputStream(Stream out_Renamed, int level) + { + InitBlock(); + this.out_Renamed = out_Renamed; + z.deflateInit(level); + compress = true; + } + + public void WriteByte(int b) + { + buf1[0] = (byte)b; + Write(buf1, 0, 1); + } + + public override void WriteByte(byte b) + { + WriteByte(b); + } + + public override void Write(byte[] b1, int off, int len) + { + if (len == 0) + { + return; + } + byte[] array = new byte[b1.Length]; + Array.Copy(b1, 0, array, 0, b1.Length); + z.next_in = array; + z.next_in_index = off; + z.avail_in = len; + do + { + z.next_out = buf; + z.next_out_index = 0; + z.avail_out = bufsize; + int num = ((!compress) ? z.inflate(flush_Renamed_Field) : z.deflate(flush_Renamed_Field)); + if (num != 0 && num != 1) + { + throw new ZStreamException((compress ? "de" : "in") + "flating: " + z.msg); + } + out_Renamed.Write(buf, 0, bufsize - z.avail_out); + } + while (z.avail_in > 0 || z.avail_out == 0); + } + + public virtual void finish() + { + do + { + z.next_out = buf; + z.next_out_index = 0; + z.avail_out = bufsize; + int num = ((!compress) ? z.inflate(4) : z.deflate(4)); + if (num != 1 && num != 0) + { + throw new ZStreamException((compress ? "de" : "in") + "flating: " + z.msg); + } + if (bufsize - z.avail_out > 0) + { + out_Renamed.Write(buf, 0, bufsize - z.avail_out); + } + } + while (z.avail_in > 0 || z.avail_out == 0); + try + { + Flush(); + } + catch + { + } + } + + public virtual void end() + { + if (compress) + { + z.deflateEnd(); + } + else + { + z.inflateEnd(); + } + z.free(); + z = null; + } + + public override void Close() + { + try + { + finish(); + } + catch + { + } + finally + { + end(); + out_Renamed.Close(); + out_Renamed = null; + } + } + + public override void Flush() + { + out_Renamed.Flush(); + } + + public override int Read(byte[] buffer, int offset, int count) + { + return 0; + } + + public override void SetLength(long value) + { + } + + public override long Seek(long offset, SeekOrigin origin) + { + return 0L; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs.meta new file mode 100644 index 00000000..569e09b4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZOutputStream.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 7c3c209ceb37db9499856d560dffd88e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs new file mode 100644 index 00000000..eb44bf8c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs @@ -0,0 +1,222 @@ +using System; + +namespace ComponentAce.Compression.Libs.zlib +{ + public sealed class ZStream + { + private const int MAX_WBITS = 15; + + private static readonly int DEF_WBITS = 15; + + private const int Z_NO_FLUSH = 0; + + private const int Z_PARTIAL_FLUSH = 1; + + private const int Z_SYNC_FLUSH = 2; + + private const int Z_FULL_FLUSH = 3; + + private const int Z_FINISH = 4; + + private const int MAX_MEM_LEVEL = 9; + + private const int Z_OK = 0; + + private const int Z_STREAM_END = 1; + + private const int Z_NEED_DICT = 2; + + private const int Z_ERRNO = -1; + + private const int Z_STREAM_ERROR = -2; + + private const int Z_DATA_ERROR = -3; + + private const int Z_MEM_ERROR = -4; + + private const int Z_BUF_ERROR = -5; + + private const int Z_VERSION_ERROR = -6; + + public byte[] next_in; + + public int next_in_index; + + public int avail_in; + + public long total_in; + + public byte[] next_out; + + public int next_out_index; + + public int avail_out; + + public long total_out; + + public string msg; + + internal Deflate dstate; + + internal Inflate istate; + + internal int data_type; + + public long adler; + + internal Adler32 _adler = new Adler32(); + + public int inflateInit() + { + return inflateInit(DEF_WBITS); + } + + public int inflateInit(int w) + { + istate = new Inflate(); + return istate.inflateInit(this, w); + } + + public int inflate(int f) + { + if (istate == null) + { + return -2; + } + return istate.inflate(this, f); + } + + public int inflateEnd() + { + if (istate == null) + { + return -2; + } + int result = istate.inflateEnd(this); + istate = null; + return result; + } + + public int inflateSync() + { + if (istate == null) + { + return -2; + } + return istate.inflateSync(this); + } + + public int inflateSetDictionary(byte[] dictionary, int dictLength) + { + if (istate == null) + { + return -2; + } + return istate.inflateSetDictionary(this, dictionary, dictLength); + } + + public int deflateInit(int level) + { + return deflateInit(level, 15); + } + + public int deflateInit(int level, int bits) + { + dstate = new Deflate(); + return dstate.deflateInit(this, level, bits); + } + + public int deflate(int flush) + { + if (dstate == null) + { + return -2; + } + return dstate.deflate(this, flush); + } + + public int deflateEnd() + { + if (dstate == null) + { + return -2; + } + int result = dstate.deflateEnd(); + dstate = null; + return result; + } + + public int deflateParams(int level, int strategy) + { + if (dstate == null) + { + return -2; + } + return dstate.deflateParams(this, level, strategy); + } + + public int deflateSetDictionary(byte[] dictionary, int dictLength) + { + if (dstate == null) + { + return -2; + } + return dstate.deflateSetDictionary(this, dictionary, dictLength); + } + + internal void flush_pending() + { + int pending = dstate.pending; + if (pending > avail_out) + { + pending = avail_out; + } + if (pending != 0) + { + if (dstate.pending_buf.Length <= dstate.pending_out || next_out.Length <= next_out_index || dstate.pending_buf.Length < dstate.pending_out + pending || next_out.Length < next_out_index + pending) + { + } + Array.Copy(dstate.pending_buf, dstate.pending_out, next_out, next_out_index, pending); + next_out_index += pending; + dstate.pending_out += pending; + total_out += pending; + avail_out -= pending; + dstate.pending -= pending; + if (dstate.pending == 0) + { + dstate.pending_out = 0; + } + } + } + + internal int read_buf(byte[] buf, int start, int size) + { + int num = avail_in; + if (num > size) + { + num = size; + } + if (num == 0) + { + return 0; + } + avail_in -= num; + if (dstate.noheader == 0) + { + adler = _adler.adler32(adler, next_in, next_in_index, num); + } + Array.Copy(next_in, next_in_index, buf, start, num); + next_in_index += num; + total_in += num; + return num; + } + + public void free() + { + next_in = null; + next_out = null; + msg = null; + _adler = null; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs.meta new file mode 100644 index 00000000..fd3afb84 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStream.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a7dac276d0775834da0bd77f4b3bca54 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs new file mode 100644 index 00000000..b1655f01 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs @@ -0,0 +1,16 @@ +using System.IO; + +namespace ComponentAce.Compression.Libs.zlib +{ + public class ZStreamException : IOException + { + public ZStreamException() + { + } + + public ZStreamException(string s) + : base(s) + { + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs.meta new file mode 100644 index 00000000..865aa54f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/ZStreamException.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9059b75bb33b8f546a3634f6c1f15418 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs new file mode 100644 index 00000000..b37980b0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs @@ -0,0 +1,54 @@ +namespace ComponentAce.Compression.Libs.zlib +{ + public sealed class zlibConst + { + private const string version_Renamed_Field = "1.0.2"; + + public const int Z_NO_COMPRESSION = 0; + + public const int Z_BEST_SPEED = 1; + + public const int Z_BEST_COMPRESSION = 9; + + public const int Z_DEFAULT_COMPRESSION = -1; + + public const int Z_FILTERED = 1; + + public const int Z_HUFFMAN_ONLY = 2; + + public const int Z_DEFAULT_STRATEGY = 0; + + public const int Z_NO_FLUSH = 0; + + public const int Z_PARTIAL_FLUSH = 1; + + public const int Z_SYNC_FLUSH = 2; + + public const int Z_FULL_FLUSH = 3; + + public const int Z_FINISH = 4; + + public const int Z_OK = 0; + + public const int Z_STREAM_END = 1; + + public const int Z_NEED_DICT = 2; + + public const int Z_ERRNO = -1; + + public const int Z_STREAM_ERROR = -2; + + public const int Z_DATA_ERROR = -3; + + public const int Z_MEM_ERROR = -4; + + public const int Z_BUF_ERROR = -5; + + public const int Z_VERSION_ERROR = -6; + + public static string version() + { + return "1.0.2"; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs.meta new file mode 100644 index 00000000..d5c4e969 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ComponentAce.Compression.Libs.zlib/zlibConst.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 189574d77199372408e8e294f65e987a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs new file mode 100644 index 00000000..6c99dd9c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs @@ -0,0 +1,109 @@ +using System.Security.Cryptography; + +namespace MyNes.Core +{ + public class Crc32 : HashAlgorithm + { + public const uint DefaultPolynomial = 3988292384u; + + public const uint DefaultSeed = uint.MaxValue; + + private uint hash; + + private uint seed; + + private uint[] table; + + private static uint[] defaultTable; + + public override int HashSize => 32; + + public Crc32() + { + table = InitializeTable(3988292384u); + seed = uint.MaxValue; + Initialize(); + } + + public Crc32(uint polynomial, uint seed) + { + table = InitializeTable(polynomial); + this.seed = seed; + Initialize(); + } + + public override void Initialize() + { + hash = seed; + } + + protected override void HashCore(byte[] buffer, int start, int length) + { + hash = CalculateHash(table, hash, buffer, start, length); + } + + protected override byte[] HashFinal() + { + return HashValue = UInt32ToBigEndianBytes(~hash); + } + + public static uint Compute(byte[] buffer) + { + return ~CalculateHash(InitializeTable(3988292384u), uint.MaxValue, buffer, 0, buffer.Length); + } + + public static uint Compute(uint seed, byte[] buffer) + { + return ~CalculateHash(InitializeTable(3988292384u), seed, buffer, 0, buffer.Length); + } + + public static uint Compute(uint polynomial, uint seed, byte[] buffer) + { + return ~CalculateHash(InitializeTable(polynomial), seed, buffer, 0, buffer.Length); + } + + private static uint[] InitializeTable(uint polynomial) + { + if (polynomial == 3988292384u && defaultTable != null) + { + return defaultTable; + } + uint[] array = new uint[256]; + for (int i = 0; i < 256; i++) + { + uint num = (uint)i; + for (int j = 0; j < 8; j++) + { + num = (((num & 1) != 1) ? (num >> 1) : ((num >> 1) ^ polynomial)); + } + array[i] = num; + } + if (polynomial == 3988292384u) + { + defaultTable = array; + } + return array; + } + + private static uint CalculateHash(uint[] table, uint seed, byte[] buffer, int start, int size) + { + uint num = seed; + for (int i = start; i < size; i++) + { + num = (num >> 8) ^ table[buffer[i] ^ (num & 0xFF)]; + } + return num; + } + + private byte[] UInt32ToBigEndianBytes(uint x) + { + return new byte[4] + { + (byte)((x >> 24) & 0xFFu), + (byte)((x >> 16) & 0xFFu), + (byte)((x >> 8) & 0xFFu), + (byte)(x & 0xFFu) + }; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs.meta new file mode 100644 index 00000000..1b434546 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Crc32.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6735365dc98ead64aacb1d9fa1b79cbc +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs new file mode 100644 index 00000000..2d6c875a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs @@ -0,0 +1,9 @@ +namespace MyNes.Core +{ + public enum EmuRegion + { + NTSC, + PALB, + DENDY + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs.meta new file mode 100644 index 00000000..94d37674 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuRegion.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f091eb23796c9f2469aca986fc0c75d6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs new file mode 100644 index 00000000..b55e644f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs @@ -0,0 +1,100 @@ +using System.IO; + +namespace MyNes.Core +{ + public class EmuSettings : ISettings + { + public string SnapsFolder = "Snaps"; + + public string WavesFolder = "SoundRecords"; + + public string SnapsFormat = ".png"; + + public bool SnapsReplace = false; + + public int RegionSetting = 0; + + public string StateFolder = "States"; + + public string GameGenieFolder = "GMCodes"; + + public string SRAMFolder = "Srams"; + + public bool SaveSRAMAtEmuShutdown = true; + + public EmuSettings(string path) + : base(path) + { + } + + public override void LoadSettings() + { + base.LoadSettings(); + if (MyNesMain.WorkingFolder == null) + { + MyNesMain.MakeWorkingFolder(); + } + if (SnapsFolder == "Snaps") + { + SnapsFolder = Path.Combine(MyNesMain.WorkingFolder, "Snaps"); + } + if (StateFolder == "States") + { + StateFolder = Path.Combine(MyNesMain.WorkingFolder, "States"); + } + if (GameGenieFolder == "GMCodes") + { + GameGenieFolder = Path.Combine(MyNesMain.WorkingFolder, "GMCodes"); + } + if (SRAMFolder == "Srams") + { + SRAMFolder = Path.Combine(MyNesMain.WorkingFolder, "Srams"); + } + if (WavesFolder == "SoundRecords") + { + WavesFolder = Path.Combine(MyNesMain.WorkingFolder, "SoundRecords"); + } + try + { + Directory.CreateDirectory(WavesFolder); + } + catch + { + Tracer.WriteError("Cannot create sound records folder !!"); + } + try + { + Directory.CreateDirectory(SnapsFolder); + } + catch + { + Tracer.WriteError("Cannot create snaps folder !!"); + } + try + { + Directory.CreateDirectory(StateFolder); + } + catch + { + Tracer.WriteError("Cannot create states folder !!"); + } + try + { + Directory.CreateDirectory(SRAMFolder); + } + catch + { + Tracer.WriteError("Cannot create srams folder !!"); + } + try + { + Directory.CreateDirectory(GameGenieFolder); + } + catch + { + Tracer.WriteError("Cannot create game genie codes folder !!"); + } + StateHandler.StateFolder = StateFolder; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs.meta new file mode 100644 index 00000000..273ac163 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/EmuSettings.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f3b3769eaf4c4d54f9ac64295acde90b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs new file mode 100644 index 00000000..41b481ae --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs @@ -0,0 +1,383 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + internal class Eprom + { + private enum EpromDevice + { + X24C01, + X24C02 + } + + private enum EpromMode + { + Data, + Addressing, + Idle, + Read, + Write, + Ack, + NotAck, + AckWait + } + + private byte[] data; + + private EpromMode mode = EpromMode.Data; + + private EpromMode nextmode = EpromMode.Data; + + private EpromDevice device = EpromDevice.X24C01; + + private bool psda; + + private bool pscl; + + private int output = 0; + + private int cbit = 0; + + private int caddress = 0; + + private int cdata = 0; + + private bool isRead; + + private bool cSCL; + + private bool cSDA; + + public Eprom(int memorySize) + { + Console.WriteLine("Initializing Eprom ..."); + data = new byte[memorySize]; + device = ((memorySize == 256) ? EpromDevice.X24C02 : EpromDevice.X24C01); + Console.WriteLine("Eprom memory size = " + memorySize); + Console.WriteLine("Eprom device = " + device); + } + + public void HardReset() + { + pscl = false; + psda = false; + mode = EpromMode.Idle; + nextmode = EpromMode.Idle; + cbit = 0; + caddress = 0; + cdata = 0; + isRead = false; + output = 16; + } + + public void Write(int address, byte data) + { + cSCL = (data & 0x20) == 32; + cSDA = (data & 0x40) == 64; + if (pscl && (!cSDA & psda)) + { + Start(); + } + else if (pscl && (cSDA & !psda)) + { + Stop(); + } + else if (cSCL & !pscl) + { + switch (device) + { + case EpromDevice.X24C01: + RiseX24C01((data >> 6) & 1); + break; + case EpromDevice.X24C02: + RiseX24C02((data >> 6) & 1); + break; + } + } + else if (!cSCL & pscl) + { + switch (device) + { + case EpromDevice.X24C01: + FallX24C01(); + break; + case EpromDevice.X24C02: + FallX24C02(); + break; + } + } + pscl = cSCL; + psda = cSDA; + } + + public byte Read(int address) + { + return (byte)output; + } + + private void Start() + { + switch (device) + { + case EpromDevice.X24C01: + mode = EpromMode.Addressing; + cbit = 0; + caddress = 0; + output = 16; + break; + case EpromDevice.X24C02: + mode = EpromMode.Data; + cbit = 0; + output = 16; + break; + } + } + + private void Stop() + { + mode = EpromMode.Idle; + output = 16; + } + + private void RiseX24C01(int bit) + { + switch (mode) + { + case EpromMode.Addressing: + if (cbit < 7) + { + caddress &= ~(1 << cbit); + caddress |= bit << cbit++; + } + else if (cbit < 8) + { + cbit = 8; + if (bit != 0) + { + nextmode = EpromMode.Read; + cdata = data[caddress]; + } + else + { + nextmode = EpromMode.Write; + } + } + break; + case EpromMode.Ack: + output = 0; + break; + case EpromMode.Read: + if (cbit < 8) + { + output = (((cdata & (1 << cbit++)) != 0) ? 16 : 0); + } + break; + case EpromMode.Write: + if (cbit < 8) + { + cdata &= ~(1 << cbit); + cdata |= bit << cbit++; + } + break; + case EpromMode.AckWait: + if (bit == 0) + { + nextmode = EpromMode.Idle; + } + break; + case EpromMode.Idle: + case EpromMode.NotAck: + break; + } + } + + private void RiseX24C02(int bit) + { + switch (mode) + { + case EpromMode.Data: + if (cbit < 8) + { + cdata &= ~(1 << 7 - cbit); + cdata |= bit << 7 - cbit++; + } + break; + case EpromMode.Addressing: + if (cbit < 8) + { + caddress &= ~(1 << 7 - cbit); + caddress |= bit << 7 - cbit++; + } + break; + case EpromMode.Read: + if (cbit < 8) + { + output = (((cdata & (1 << 7 - cbit++)) != 0) ? 16 : 0); + } + break; + case EpromMode.Write: + if (cbit < 8) + { + cdata &= ~(1 << 7 - cbit); + cdata |= bit << 7 - cbit++; + } + break; + case EpromMode.NotAck: + output = 16; + break; + case EpromMode.Ack: + output = 0; + break; + case EpromMode.AckWait: + if (bit == 0) + { + nextmode = EpromMode.Read; + cdata = data[caddress]; + } + break; + case EpromMode.Idle: + break; + } + } + + private void FallX24C01() + { + switch (mode) + { + case EpromMode.Addressing: + if (cbit == 8) + { + mode = EpromMode.Ack; + output = 16; + } + break; + case EpromMode.Ack: + mode = nextmode; + cbit = 0; + output = 16; + break; + case EpromMode.Read: + if (cbit == 8) + { + mode = EpromMode.AckWait; + caddress = (caddress + 1) & 0x7F; + } + break; + case EpromMode.Write: + if (cbit == 8) + { + mode = EpromMode.Ack; + nextmode = EpromMode.Idle; + data[caddress] = (byte)cdata; + caddress = (caddress + 1) & 0x7F; + } + break; + case EpromMode.Idle: + break; + } + } + + private void FallX24C02() + { + switch (mode) + { + case EpromMode.Data: + if (cbit != 8) + { + break; + } + if ((cdata & 0xA0) == 160) + { + cbit = 0; + mode = EpromMode.Ack; + isRead = (cdata & 1) == 1; + output = 16; + if (isRead) + { + nextmode = EpromMode.Read; + cdata = data[caddress]; + } + else + { + nextmode = EpromMode.Addressing; + } + } + else + { + mode = EpromMode.NotAck; + nextmode = EpromMode.Idle; + output = 16; + } + break; + case EpromMode.Addressing: + if (cbit == 8) + { + cbit = 0; + mode = EpromMode.Ack; + nextmode = (isRead ? EpromMode.Idle : EpromMode.Write); + output = 16; + } + break; + case EpromMode.Read: + if (cbit == 8) + { + mode = EpromMode.AckWait; + caddress = (caddress + 1) & 0xFF; + } + break; + case EpromMode.Write: + if (cbit == 8) + { + cbit = 0; + mode = EpromMode.Ack; + nextmode = EpromMode.Write; + data[caddress] = (byte)cdata; + caddress = (caddress + 1) & 0xFF; + } + break; + case EpromMode.NotAck: + mode = EpromMode.Idle; + cbit = 0; + output = 16; + break; + case EpromMode.Ack: + case EpromMode.AckWait: + mode = nextmode; + cbit = 0; + output = 16; + break; + case EpromMode.Idle: + break; + } + } + + public void SaveState(BinaryWriter stream) + { + stream.Write(data); + stream.Write((int)mode); + stream.Write((int)nextmode); + stream.Write(psda); + stream.Write(pscl); + stream.Write(output); + stream.Write(cbit); + stream.Write(caddress); + stream.Write(cdata); + stream.Write(isRead); + } + + public void LoadState(BinaryReader stream) + { + stream.Read(data, 0, data.Length); + mode = (EpromMode)stream.ReadInt32(); + nextmode = (EpromMode)stream.ReadInt32(); + psda = stream.ReadBoolean(); + pscl = stream.ReadBoolean(); + output = stream.ReadInt32(); + cbit = stream.ReadInt32(); + caddress = stream.ReadInt32(); + cdata = stream.ReadInt32(); + isRead = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs.meta new file mode 100644 index 00000000..bfc4ec5a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Eprom.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: bf9a039fae5474f4d9699753440312cf +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs new file mode 100644 index 00000000..9232f55f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs @@ -0,0 +1,56 @@ +using System.IO; + +namespace MyNes.Core +{ + internal abstract class FFE : Board + { + protected bool irqEnable; + + protected int irqCounter; + + internal override void WriteEX(ref ushort address, ref byte data) + { + switch (address) + { + case 17665: + irqEnable = false; + NesEmu.IRQFlags &= -9; + break; + case 17666: + irqCounter = (irqCounter & 0xFF00) | data; + break; + case 17667: + irqEnable = true; + irqCounter = (irqCounter & 0xFF) | (data << 8); + break; + } + } + + internal override void OnCPUClock() + { + if (irqEnable) + { + irqCounter++; + if (irqCounter >= 65535) + { + irqCounter = 0; + NesEmu.IRQFlags |= 8; + } + } + } + + internal override void WriteStateData(ref BinaryWriter bin) + { + base.WriteStateData(ref bin); + bin.Write(irqEnable); + bin.Write(irqCounter); + } + + internal override void ReadStateData(ref BinaryReader bin) + { + base.ReadStateData(ref bin); + irqEnable = bin.ReadBoolean(); + irqCounter = bin.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs.meta new file mode 100644 index 00000000..85c1a4e6 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/FFE.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 05457d6300519b04a966034b34ad95df +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs new file mode 100644 index 00000000..3c893a00 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs @@ -0,0 +1,153 @@ +using System.Collections.Generic; + +namespace MyNes.Core +{ + public class GameGenie + { + public string[] LettersTable = new string[16] + { + "A", "P", "Z", "L", "G", "I", "T", "Y", "E", "O", + "X", "U", "K", "S", "V", "N" + }; + + public byte[] HEXTable = new byte[16] + { + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, + 10, 11, 12, 13, 14, 15 + }; + + private List lettersTable = new List(); + + public GameGenie() + { + lettersTable = new List(LettersTable); + } + + public int GetCodeAsHEX(string code) + { + int num = 0; + int num2 = code.ToCharArray().Length - 1; + char[] array = code.ToCharArray(); + foreach (char c in array) + { + num |= HEXTable[lettersTable.IndexOf(c.ToString())] << num2 * 4; + num2--; + } + return num; + } + + public byte GetGGValue(int code, int length) + { + int num = 0; + int num2 = 0; + int num3 = 0; + int num4 = 0; + int num5 = 0; + int num6 = 0; + int num7 = 0; + int num8 = 0; + switch (length) + { + case 6: + num8 = (code & 0x800000) >> 23; + num7 = (code & 0x40000) >> 18; + num6 = (code & 0x20000) >> 17; + num5 = (code & 0x10000) >> 16; + num4 = (code & 8) >> 3; + num3 = (code & 0x400000) >> 22; + num2 = (code & 0x200000) >> 21; + num = (code & 0x100000) >> 20; + break; + case 8: + num8 = (code >> 31) & 1; + num7 = (code >> 27) & 1; + num6 = (code >> 26) & 1; + num5 = (code >> 25) & 1; + num4 = (code >> 3) & 1; + num3 = (code >> 30) & 1; + num2 = (code >> 29) & 1; + num = (code >> 28) & 1; + break; + } + return (byte)((num8 << 7) | (num7 << 6) | (num6 << 5) | (num5 << 4) | (num4 << 3) | (num3 << 2) | (num2 << 1) | num); + } + + public int GetGGAddress(int code, int length) + { + int num = 0; + int num2 = 0; + int num3 = 0; + int num4 = 0; + int num5 = 0; + int num6 = 0; + int num7 = 0; + int num8 = 0; + int num9 = 0; + int num10 = 0; + int num11 = 0; + int num12 = 0; + int num13 = 0; + int num14 = 0; + int num15 = 0; + switch (length) + { + case 6: + num15 = (code >> 10) & 1; + num14 = (code >> 9) & 1; + num13 = (code >> 8) & 1; + num12 = (code >> 7) & 1; + num11 = (code >> 2) & 1; + num10 = (code >> 1) & 1; + num9 = code & 1; + num8 = (code >> 19) & 1; + num7 = (code >> 14) & 1; + num6 = (code >> 13) & 1; + num5 = (code >> 12) & 1; + num4 = (code >> 11) & 1; + num3 = (code >> 6) & 1; + num2 = (code >> 5) & 1; + num = (code >> 4) & 1; + break; + case 8: + num15 = (code >> 18) & 1; + num14 = (code >> 17) & 1; + num13 = (code >> 16) & 1; + num12 = (code >> 15) & 1; + num11 = (code >> 10) & 1; + num10 = (code >> 9) & 1; + num9 = (code >> 8) & 1; + num8 = (code >> 25) & 1; + num7 = (code >> 22) & 1; + num6 = (code >> 21) & 1; + num5 = (code >> 20) & 1; + num4 = (code >> 19) & 1; + num3 = (code >> 14) & 1; + num2 = (code >> 13) & 1; + num = (code >> 12) & 1; + break; + } + return (num15 << 14) | (num14 << 13) | (num13 << 12) | (num12 << 11) | (num11 << 10) | (num10 << 9) | (num9 << 8) | (num8 << 7) | (num7 << 6) | (num6 << 5) | (num5 << 4) | (num4 << 3) | (num3 << 2) | (num2 << 1) | num; + } + + public byte GetGGCompareValue(int code) + { + int num = 0; + int num2 = 0; + int num3 = 0; + int num4 = 0; + int num5 = 0; + int num6 = 0; + int num7 = 0; + int num8 = 0; + num8 = (code >> 7) & 1; + num7 = (code >> 2) & 1; + num6 = (code >> 1) & 1; + num5 = code & 1; + num4 = (code >> 11) & 1; + num3 = (code >> 6) & 1; + num2 = (code >> 5) & 1; + num = (code >> 4) & 1; + return (byte)((num8 << 7) | (num7 << 6) | (num6 << 5) | (num5 << 4) | (num4 << 3) | (num3 << 2) | (num2 << 1) | num); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs.meta new file mode 100644 index 00000000..8cde2169 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenie.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c440f367166527145bec0f405ad3b587 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs new file mode 100644 index 00000000..a46c121a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs @@ -0,0 +1,19 @@ +namespace MyNes.Core +{ + public struct GameGenieCode + { + public string Name; + + public string Descreption; + + public int Address; + + public byte Compare; + + public byte Value; + + public bool IsCompare; + + public bool Enabled; + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs.meta new file mode 100644 index 00000000..2237c1de --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/GameGenieCode.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0be47c6f15a83aa48b41cf4e18b98e6d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs new file mode 100644 index 00000000..6bedd571 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs @@ -0,0 +1,4 @@ +namespace MyNes.Core +{ + internal delegate void GetIsPlaying(out bool playing); +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs.meta new file mode 100644 index 00000000..9a59786d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/GetIsPlaying.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d5e4818002a2f3140b2da0af5b1639fa +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs new file mode 100644 index 00000000..d5e29a62 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs @@ -0,0 +1,8 @@ +using System; + +namespace MyNes.Core +{ + internal class HassIssuesAttribute : Attribute + { + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs.meta new file mode 100644 index 00000000..66e381d8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/HassIssuesAttribute.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4ea93c2200789cd499484183e415c4c6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs new file mode 100644 index 00000000..78d7aaf7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs @@ -0,0 +1,181 @@ +using System.IO; +using System.Security.Cryptography; + +namespace MyNes.Core +{ + public class HelperTools + { + public static string GetFileSize(string FilePath) + { + if (File.Exists(Path.GetFullPath(FilePath))) + { + FileInfo fileInfo = new FileInfo(FilePath); + string text = " Byte"; + double num = fileInfo.Length; + if (fileInfo.Length >= 1024) + { + num = (double)fileInfo.Length / 1024.0; + text = " KB"; + } + if (num >= 1024.0) + { + num /= 1024.0; + text = " MB"; + } + if (num >= 1024.0) + { + num /= 1024.0; + text = " GB"; + } + return num.ToString("F2") + text; + } + return ""; + } + + public static string GetSize(long size) + { + string text = " Byte"; + double num = size; + if (size >= 1024) + { + num = (double)size / 1024.0; + text = " KB"; + } + if (num >= 1024.0) + { + num /= 1024.0; + text = " MB"; + } + if (num >= 1024.0) + { + num /= 1024.0; + text = " GB"; + } + if (num < 0.0) + { + return "???"; + } + return num.ToString("F2") + text; + } + + public static string GetSize(ulong size) + { + string text = " Byte"; + double num = size; + if (size >= 1024) + { + num = (double)size / 1024.0; + text = " KB"; + } + if (num >= 1024.0) + { + num /= 1024.0; + text = " MB"; + } + if (num >= 1024.0) + { + num /= 1024.0; + text = " GB"; + } + if (num < 0.0) + { + return "???"; + } + return num.ToString("F2") + text; + } + + public static long GetSizeAsBytes(string FilePath) + { + if (File.Exists(FilePath)) + { + FileInfo fileInfo = new FileInfo(FilePath); + return fileInfo.Length; + } + return 0L; + } + + public static bool IsStringContainsNumbers(string text) + { + char[] array = text.ToCharArray(); + foreach (char c in array) + { + int result = 0; + if (int.TryParse(c.ToString(), out result)) + { + return true; + } + } + return false; + } + + public static string CalculateCRC(string filePath) + { + if (File.Exists(filePath)) + { + Stream stream = new FileStream(filePath, FileMode.Open, FileAccess.Read); + byte[] buffer = new byte[stream.Length]; + stream.Read(buffer, 0, (int)stream.Length); + stream.Close(); + string text = ""; + Crc32 crc = new Crc32(); + byte[] array = crc.ComputeHash(buffer); + byte[] array2 = array; + foreach (byte b in array2) + { + text += b.ToString("x2").ToLower(); + } + return text; + } + return ""; + } + + public static string CalculateCRC(string filePath, int bytesToSkip) + { + if (File.Exists(filePath)) + { + Stream stream = new FileStream(filePath, FileMode.Open, FileAccess.Read); + stream.Read(new byte[bytesToSkip], 0, bytesToSkip); + byte[] buffer = new byte[stream.Length - bytesToSkip]; + stream.Read(buffer, 0, (int)(stream.Length - bytesToSkip)); + stream.Close(); + string text = ""; + Crc32 crc = new Crc32(); + byte[] array = crc.ComputeHash(buffer); + byte[] array2 = array; + foreach (byte b in array2) + { + text += b.ToString("x2").ToLower(); + } + return text; + } + return ""; + } + + public static string CalculateSHA1(string filePath) + { + if (File.Exists(filePath)) + { + byte[] buffer = GetBuffer(filePath); + string text = ""; + SHA1Managed sHA1Managed = new SHA1Managed(); + byte[] array = sHA1Managed.ComputeHash(buffer); + byte[] array2 = array; + foreach (byte b in array2) + { + text += b.ToString("x2").ToLower(); + } + return text; + } + return ""; + } + + public static byte[] GetBuffer(string filePath) + { + Stream stream = new FileStream(filePath, FileMode.Open, FileAccess.Read); + byte[] array = new byte[stream.Length]; + stream.Read(array, 0, (int)stream.Length); + stream.Close(); + return array; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs.meta new file mode 100644 index 00000000..1b200738 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/HelperTools.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6edadafe72c2a374fb7abbbf03b14cb6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs new file mode 100644 index 00000000..057b1a29 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs @@ -0,0 +1,29 @@ +namespace MyNes.Core +{ + public interface IAudioProvider + { + string Name { get; } + + string ID { get; } + + bool AllowBufferChange { get; } + + bool AllowFrequencyChange { get; } + + void SubmitSamples(ref short[] buffer, ref int samples_added); + + void TogglePause(bool paused); + + void GetIsPlaying(out bool playing); + + void Initialize(); + + void ShutDown(); + + void Reset(); + + void SignalToggle(bool started); + + void SetVolume(int Vol); + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs.meta new file mode 100644 index 00000000..c38a2826 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IAudioProvider.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4366c4b2360b48d4c87ff24651f8dec8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs new file mode 100644 index 00000000..e42d920f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs @@ -0,0 +1,18 @@ +namespace MyNes.Core +{ + public abstract class IJoypadConnecter + { + protected byte DATA; + + public abstract void Update(); + + public virtual void Destroy() + { + } + + public virtual byte GetData() + { + return DATA; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs.meta new file mode 100644 index 00000000..fbbe1093 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IJoypadConnecter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c613333925a94f44c85ef86668ea7636 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs new file mode 100644 index 00000000..548a5ee3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs @@ -0,0 +1,97 @@ +using System.IO; +using System.Security.Cryptography; + +namespace MyNes.Core +{ + public class INes : IRom + { + public bool HasBattery { get; private set; } + + public bool IsPlaychoice10 { get; private set; } + + public bool IsVSUnisystem { get; private set; } + + public override void Load(string fileName, bool loadDumps) + { + FileStream fileStream = new FileStream(fileName, FileMode.Open, FileAccess.Read); + if (fileStream.Length < 16) + { + fileStream.Close(); + base.IsValid = false; + return; + } + byte[] array = new byte[16]; + fileStream.Read(array, 0, 16); + byte[] buffer = new byte[fileStream.Length - 16]; + fileStream.Read(buffer, 0, (int)(fileStream.Length - 16)); + base.SHA1 = ""; + SHA1Managed sHA1Managed = new SHA1Managed(); + byte[] array2 = sHA1Managed.ComputeHash(buffer); + byte[] array3 = array2; + foreach (byte b in array3) + { + base.SHA1 += b.ToString("x2").ToLower(); + } + if (array[0] != 78 || array[1] != 69 || array[2] != 83 || array[3] != 26) + { + fileStream.Close(); + base.IsValid = false; + return; + } + base.PRGCount = array[4]; + base.CHRCount = array[5]; + switch (array[6] & 9) + { + case 0: + base.Mirroring = Mirroring.Horz; + break; + case 1: + base.Mirroring = Mirroring.Vert; + break; + case 8: + case 9: + base.Mirroring = Mirroring.Full; + break; + } + HasBattery = (array[6] & 2) != 0; + base.HasTrainer = (array[6] & 4) != 0; + if ((array[7] & 0xF) == 0) + { + base.MapperNumber = (byte)((array[7] & 0xF0) | (array[6] >> 4)); + } + else + { + base.MapperNumber = (byte)(array[6] >> 4); + } + IsVSUnisystem = (array[7] & 1) != 0; + IsPlaychoice10 = (array[7] & 2) != 0; + if (loadDumps) + { + fileStream.Seek(16L, SeekOrigin.Begin); + if (base.HasTrainer) + { + base.Trainer = new byte[512]; + fileStream.Read(base.Trainer, 0, 512); + } + else + { + base.Trainer = new byte[0]; + } + base.PRG = new byte[base.PRGCount * 16384]; + fileStream.Read(base.PRG, 0, base.PRGCount * 16384); + if (base.CHRCount > 0) + { + base.CHR = new byte[base.CHRCount * 8192]; + fileStream.Read(base.CHR, 0, base.CHRCount * 8192); + } + else + { + base.CHR = new byte[0]; + } + } + base.IsValid = true; + fileStream.Dispose(); + fileStream.Close(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs.meta new file mode 100644 index 00000000..80bb67be --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/INes.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6d7f5c4eca034d148883c87e6c6b2443 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs new file mode 100644 index 00000000..e7133614 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs @@ -0,0 +1,29 @@ +namespace MyNes.Core +{ + public abstract class IRom + { + public bool IsValid { get; set; } + + public int PRGCount { get; set; } + + public int CHRCount { get; set; } + + public int MapperNumber { get; set; } + + public Mirroring Mirroring { get; set; } + + public bool HasTrainer { get; set; } + + public byte[] PRG { get; set; } + + public byte[] CHR { get; set; } + + public byte[] Trainer { get; set; } + + public string SHA1 { get; set; } + + public virtual void Load(string fileName, bool loadDumps) + { + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs.meta new file mode 100644 index 00000000..25009e77 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IRom.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 60fcaf7174409fc448d54f85371d31fc +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs new file mode 100644 index 00000000..5375ac6e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs @@ -0,0 +1,149 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Reflection; + +namespace MyNes.Core +{ + public abstract class ISettings + { + protected string filePath; + + protected FieldInfo[] Fields; + + public ISettings(string filePath) + { + this.filePath = filePath; + } + + public virtual void LoadSettings() + { + Fields = GetType().GetFields(); + if (!File.Exists(filePath)) + { + return; + } + string[] array = File.ReadAllLines(filePath); + for (int i = 0; i < array.Length; i++) + { + string[] array2 = array[i].Split('='); + if (array2 != null && array2.Length == 2) + { + SetField(array2[0], array2[1]); + } + } + } + + public virtual void SaveSettings() + { + Fields = GetType().GetFields(); + List list = new List(); + FieldInfo[] fields = Fields; + foreach (FieldInfo fieldInfo in fields) + { + if (fieldInfo.IsPublic) + { + list.Add(fieldInfo.Name + "=" + GetFieldValue(fieldInfo)); + } + } + File.WriteAllLines(filePath, list.ToArray()); + } + + protected virtual void SetField(string fieldName, string val) + { + for (int i = 0; i < Fields.Length; i++) + { + if (!(Fields[i].Name == fieldName)) + { + continue; + } + if (Fields[i].FieldType == typeof(string)) + { + Fields[i].SetValue(this, val); + } + else if (Fields[i].FieldType == typeof(bool)) + { + Fields[i].SetValue(this, val == "1"); + } + else if (Fields[i].FieldType == typeof(int)) + { + int result = 0; + if (int.TryParse(val, out result)) + { + Fields[i].SetValue(this, result); + } + } + else if (Fields[i].FieldType == typeof(float)) + { + float result2 = 0f; + if (float.TryParse(val, out result2)) + { + Fields[i].SetValue(this, result2); + } + } + else if (Fields[i].FieldType == typeof(string[])) + { + string[] value = val.Split(new string[1] { "*" }, StringSplitOptions.RemoveEmptyEntries); + Fields[i].SetValue(this, value); + } + else + { + Tracer.WriteLine("Unknown setting type = " + Fields[i].FieldType); + } + break; + } + } + + protected virtual string GetFieldValue(string fieldName) + { + for (int i = 0; i < Fields.Length; i++) + { + if (Fields[i].Name == fieldName) + { + return GetFieldValue(Fields[i]); + } + } + return ""; + } + + protected virtual string GetFieldValue(FieldInfo field) + { + object value = field.GetValue(this); + if (field.FieldType == typeof(string)) + { + return value.ToString(); + } + if (field.FieldType == typeof(bool)) + { + return ((bool)value) ? "1" : "0"; + } + if (field.FieldType == typeof(int)) + { + return value.ToString(); + } + if (field.FieldType == typeof(float)) + { + return value.ToString(); + } + if (field.FieldType == typeof(string[])) + { + string text = ""; + string[] array = (string[])value; + if (array != null) + { + string[] array2 = array; + foreach (string text2 in array2) + { + text = text + text2 + "*"; + } + } + if (text.Length > 0) + { + return text.Substring(0, text.Length - 1); + } + return ""; + } + return ""; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs.meta new file mode 100644 index 00000000..8b0c3b2a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ISettings.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3236f262a69a32f4395a1f17537be480 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs new file mode 100644 index 00000000..bf195939 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs @@ -0,0 +1,7 @@ +namespace MyNes.Core +{ + public interface IShortcutsHandler + { + void Update(); + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs.meta new file mode 100644 index 00000000..cf4b4454 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IShortcutsHandler.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e44becc2730cc364db0b40806a1dd8cd +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs new file mode 100644 index 00000000..3a79e82b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs @@ -0,0 +1,25 @@ +namespace MyNes.Core +{ + public abstract class IVSUnisystemDIPConnecter + { + public abstract void Update(); + + public virtual void OnEmuShutdown() + { + } + + public virtual byte GetData4016() + { + return 0; + } + + public virtual byte GetData4017() + { + return 0; + } + + public virtual void Write4020(ref byte data) + { + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs.meta new file mode 100644 index 00000000..870c2f0d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVSUnisystemDIPConnecter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 593f15ab3ef85d14a8a4c462199c9775 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs new file mode 100644 index 00000000..b5100098 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs @@ -0,0 +1,41 @@ +namespace MyNes.Core +{ + public interface IVideoProvider + { + string Name { get; } + + string ID { get; } + + void WriteErrorNotification(string message, bool instant); + + void WriteInfoNotification(string message, bool instant); + + void WriteWarningNotification(string message, bool instant); + + void TakeSnapshotAs(string path, string format); + + void TakeSnapshot(); + + void Initialize(); + + void ShutDown(); + + void SignalToggle(bool started); + + void SubmitFrame(ref int[] buffer); + + void ResizeBegin(); + + void ResizeEnd(); + + void ApplyRegionChanges(); + + void Resume(); + + void ToggleAspectRatio(bool keep_aspect); + + void ToggleFPS(bool show_fps); + + void ApplyFilter(); + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs.meta new file mode 100644 index 00000000..f7a09411 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IVideoProvider.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fe8b5545022785849a0b156b01d15972 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs new file mode 100644 index 00000000..65d651d4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs @@ -0,0 +1,16 @@ +namespace MyNes.Core +{ + public abstract class IZapperConnecter + { + protected bool Trigger; + + protected bool State; + + public abstract void Update(); + + public virtual byte GetData() + { + return (byte)((Trigger ? 16u : 0u) | (State ? 8u : 0u)); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs.meta new file mode 100644 index 00000000..3a723648 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/IZapperConnecter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b7e39cb978938d8478d799f539c44c3f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs new file mode 100644 index 00000000..eeab14fd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs @@ -0,0 +1,116 @@ +using System.IO; + +namespace MyNes.Core +{ + internal abstract class MMC2 : Board + { + private byte chr_reg0A; + + private byte chr_reg0B; + + private byte chr_reg1A; + + private byte chr_reg1B; + + private byte latch_a = 254; + + private byte latch_b = 254; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask - 2, PRGArea.AreaA000); + Switch08KPRG(PRG_ROM_08KB_Mask - 1, PRGArea.AreaC000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + chr_reg0B = 4; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF000) + { + case 40960: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 45056: + chr_reg0A = data; + if (latch_a == 253) + { + Switch04KCHR(chr_reg0A, CHRArea.Area0000); + } + break; + case 49152: + chr_reg0B = data; + if (latch_a == 254) + { + Switch04KCHR(chr_reg0B, CHRArea.Area0000); + } + break; + case 53248: + chr_reg1A = data; + if (latch_b == 253) + { + Switch04KCHR(chr_reg1A, CHRArea.Area1000); + } + break; + case 57344: + chr_reg1B = data; + if (latch_b == 254) + { + Switch04KCHR(chr_reg1B, CHRArea.Area1000); + } + break; + case 61440: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + break; + } + } + + internal override void ReadCHR(ref ushort address, out byte data) + { + if ((address & 0x1FF0) == 4048 && latch_a != 253) + { + latch_a = 253; + Switch04KCHR(chr_reg0A, CHRArea.Area0000); + } + else if ((address & 0x1FF0) == 4064 && latch_a != 254) + { + latch_a = 254; + Switch04KCHR(chr_reg0B, CHRArea.Area0000); + } + else if ((address & 0x1FF0) == 8144 && latch_b != 253) + { + latch_b = 253; + Switch04KCHR(chr_reg1A, CHRArea.Area1000); + } + else if ((address & 0x1FF0) == 8160 && latch_b != 254) + { + latch_b = 254; + Switch04KCHR(chr_reg1B, CHRArea.Area1000); + } + base.ReadCHR(ref address, out data); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(chr_reg0A); + stream.Write(chr_reg0B); + stream.Write(chr_reg1A); + stream.Write(chr_reg1B); + stream.Write(latch_a); + stream.Write(latch_b); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + chr_reg0A = stream.ReadByte(); + chr_reg0B = stream.ReadByte(); + chr_reg1A = stream.ReadByte(); + chr_reg1B = stream.ReadByte(); + latch_a = stream.ReadByte(); + latch_b = stream.ReadByte(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs.meta new file mode 100644 index 00000000..f081fbd3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC2.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c2fd2154d001d8c4297526ae3b12644a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs new file mode 100644 index 00000000..39043177 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs @@ -0,0 +1,86 @@ +using System.IO; + +namespace MyNes.Core +{ + internal class MMC5Pcm + { + internal byte output; + + internal bool Outputable; + + private bool readMode; + + private bool PCMIRQenable; + + private bool irqTrip; + + internal void HardReset() + { + output = 0; + readMode = false; + PCMIRQenable = false; + irqTrip = false; + } + + internal void SoftReset() + { + HardReset(); + } + + internal void Write5010(byte data) + { + readMode = (data & 1) == 1; + PCMIRQenable = (data & 0x80) == 128; + if (PCMIRQenable && irqTrip) + { + NesEmu.IRQFlags |= 8; + } + } + + internal byte Read5010() + { + byte result = (byte)((irqTrip & PCMIRQenable) ? 128u : 0u); + irqTrip = false; + NesEmu.IRQFlags &= -9; + return result; + } + + internal void Write5011(byte data) + { + if (readMode) + { + return; + } + if (data == 0) + { + irqTrip = true; + } + else + { + irqTrip = false; + if (Outputable) + { + output = data; + } + } + if (PCMIRQenable && irqTrip) + { + NesEmu.IRQFlags |= 8; + } + } + + internal void SaveState(ref BinaryWriter stream) + { + stream.Write(readMode); + stream.Write(PCMIRQenable); + stream.Write(irqTrip); + } + + internal void LoadState(ref BinaryReader stream) + { + readMode = stream.ReadBoolean(); + PCMIRQenable = stream.ReadBoolean(); + irqTrip = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs.meta new file mode 100644 index 00000000..6cdb2d82 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Pcm.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b4c531b467bf54c4a9d0874c00316b40 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs new file mode 100644 index 00000000..d1c7bf35 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs @@ -0,0 +1,214 @@ +using System.IO; + +namespace MyNes.Core +{ + internal class MMC5Sqr + { + private byte[][] duty_cycle_sequences = new byte[4][] + { + new byte[8] { 0, 0, 0, 0, 0, 0, 0, 1 }, + new byte[8] { 0, 0, 0, 0, 0, 0, 1, 1 }, + new byte[8] { 0, 0, 0, 0, 1, 1, 1, 1 }, + new byte[8] { 1, 1, 1, 1, 1, 1, 0, 0 } + }; + + private byte[] duration_table = new byte[32] + { + 10, 254, 20, 2, 40, 4, 80, 6, 160, 8, + 60, 10, 14, 12, 26, 14, 12, 16, 24, 18, + 48, 20, 96, 22, 192, 24, 72, 26, 16, 28, + 32, 30 + }; + + private byte duty_cycle; + + private bool length_halt; + + private bool constant_volume_envelope; + + private byte volume_devider_period; + + private int timer; + + private int period_devider; + + private byte seqencer; + + private bool length_enabled; + + private int length_counter; + + private bool envelope_start_flag; + + private byte envelope_devider; + + private byte envelope_decay_level_counter; + + private byte envelope; + + internal int output; + + internal bool Outputable; + + internal void HardReset() + { + duty_cycle = 0; + length_halt = false; + constant_volume_envelope = false; + volume_devider_period = 0; + timer = 0; + period_devider = 0; + seqencer = 0; + length_enabled = false; + length_counter = 0; + envelope_start_flag = false; + envelope_devider = 0; + envelope_decay_level_counter = 0; + envelope = 0; + } + + internal void SoftReset() + { + HardReset(); + } + + internal void Clock() + { + period_devider--; + if (period_devider > 0) + { + return; + } + period_devider = timer + 1; + if (length_counter > 0) + { + if (Outputable) + { + output = duty_cycle_sequences[duty_cycle][seqencer] * envelope; + } + } + else + { + output = 0; + } + if (seqencer == 0) + { + seqencer = 7; + } + else + { + seqencer--; + } + } + + internal void ClockLength() + { + if (length_counter > 0 && !length_halt) + { + length_counter--; + } + } + + internal void ClockEnvelope() + { + if (envelope_start_flag) + { + envelope_start_flag = false; + envelope_decay_level_counter = 15; + envelope_devider = (byte)(volume_devider_period + 1); + } + else if (envelope_devider > 0) + { + envelope_devider--; + } + else + { + envelope_devider = (byte)(volume_devider_period + 1); + if (envelope_decay_level_counter > 0) + { + envelope_decay_level_counter--; + } + else if (length_halt) + { + envelope_decay_level_counter = 15; + } + } + envelope = (constant_volume_envelope ? volume_devider_period : envelope_decay_level_counter); + } + + internal void Write0(ref byte value) + { + duty_cycle = (byte)((value & 0xC0) >> 6); + volume_devider_period = (byte)(value & 0xFu); + length_halt = (value & 0x20) != 0; + constant_volume_envelope = (value & 0x10) != 0; + envelope = (constant_volume_envelope ? volume_devider_period : envelope_decay_level_counter); + } + + internal void Write2(ref byte value) + { + timer = (timer & 0xFF00) | value; + } + + internal void Write3(ref byte value) + { + timer = (timer & 0xFF) | ((value & 7) << 8); + if (length_enabled) + { + length_counter = duration_table[value >> 3]; + } + seqencer = 0; + envelope_start_flag = true; + } + + internal void WriteEnabled(bool enabled) + { + length_enabled = enabled; + if (!length_enabled) + { + length_counter = 0; + } + } + + internal bool ReadEnable() + { + return length_counter > 0; + } + + internal void WriteStateData(ref BinaryWriter bin) + { + bin.Write(duty_cycle); + bin.Write(length_halt); + bin.Write(constant_volume_envelope); + bin.Write(volume_devider_period); + bin.Write(timer); + bin.Write(period_devider); + bin.Write(seqencer); + bin.Write(length_enabled); + bin.Write(length_counter); + bin.Write(envelope_start_flag); + bin.Write(envelope_devider); + bin.Write(envelope_decay_level_counter); + bin.Write(envelope); + bin.Write(output); + } + + internal void ReadStateData(ref BinaryReader bin) + { + duty_cycle = bin.ReadByte(); + length_halt = bin.ReadBoolean(); + constant_volume_envelope = bin.ReadBoolean(); + volume_devider_period = bin.ReadByte(); + timer = bin.ReadInt32(); + period_devider = bin.ReadInt32(); + seqencer = bin.ReadByte(); + length_enabled = bin.ReadBoolean(); + length_counter = bin.ReadInt32(); + envelope_start_flag = bin.ReadBoolean(); + envelope_devider = bin.ReadByte(); + envelope_decay_level_counter = bin.ReadByte(); + envelope = bin.ReadByte(); + output = bin.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs.meta new file mode 100644 index 00000000..e80b3425 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MMC5Sqr.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 21efe36faf5e4b74592b812f08f60773 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs new file mode 100644 index 00000000..cc870c1f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs @@ -0,0 +1,129 @@ +namespace MyNes.Core +{ + public class MNInterfaceLanguage + { + public static string Message_RomInfoCanBeOnlyShown = "Rom info can be shown only when emulation is on (i.e. game is loaded)"; + + public static string Message_StateSlotSetTo = "State slot set to"; + + public static string Message_LoadStateCanBeUsedOnly = "Load state as can be used only when emulation is on (i.e. game is loaded)"; + + public static string Message_SaveStateCanBeUseOnly = "Save state as can be used only when emulation is on (i.e. game is loaded)"; + + public static string Message_HardResetCanBeUsedOnly = "Hard reset can be used only when emulation is on (i.e. game is loaded)"; + + public static string Message_SoftResetCanBeUsedOnly = "Soft reset can be used only when emulation is on (i.e. game is loaded)"; + + public static string Message_TurboCanBeToggledOnly = "Turbo can be toggled only when emulation is on (i.e. game is loaded)"; + + public static string Message_GameGenieCanBeConfiguredOnly = "Game Genie can be enabled/configured only when emulation is on (i.e. game is loaded)"; + + public static string Message_Error1 = "Can't save state, emu is off."; + + public static string Message_Error2 = "Can't save state, no rom file is loaded."; + + public static string Message_Error3 = "Can't save state while loading a state !"; + + public static string Message_Error4 = "Already saving state !!"; + + public static string Message_Error5 = "Can't load state, emu is off."; + + public static string Message_Error6 = "Can't load state, no rom file is loaded."; + + public static string Message_Error7 = "Can't load state while saving a state !"; + + public static string Message_Error8 = "Already loading state !!"; + + public static string Message_Error9 = "No state found in slot"; + + public static string Message_Error10 = "Unable load state at slot"; + + public static string Message_Error11 = "Not My Nes State File !"; + + public static string Message_Error12 = "Not compatible state file version !"; + + public static string Message_Error13 = "This state file is not for this game; not same SHA1 !"; + + public static string Message_Error14 = "IS NOT LOCATED, mapper is not supported or unable to find it."; + + public static string Message_Error15 = "will be used instead, assigned successfully."; + + public static string Message_Error16 = "Game Genie code length cannot be more than 8 letters"; + + public static string Message_Error17 = "has issues and may not function probably with this game."; + + public static string Message_Info1 = "State saved at slot"; + + public static string Message_Info2 = "State loaded from slot"; + + public static string Message_Info3 = "Snapshot saved"; + + public static string Message_Info4 = "Interface language set to"; + + public static string Message_PleaseRestartToApply = "Please restart My Nes to apply."; + + public static string Message_HardReset = "HARD RESET !"; + + public static string Message_SoftReset = "SOFT RESET !"; + + public static string Message_Paused = "PAUSED"; + + public static string Mapper = "Mapper"; + + public static string IssueMapper5 = "Split screen not implemented.\nUchuu Keibitai SDF game graphic corruption for unknown reason in the intro (not in the split screen)."; + + public static string IssueMapper6 = "Mapper 6 is not tested, issues may occur"; + + public static string IssueMapper8 = "Mapper 8 is not tested, issues may occur"; + + public static string IssueMapper33 = "Mapper 33: Akira is not working for unknown reason."; + + public static string IssueMapper44 = "In game Super Big 7 - in - 1 : Double Dragon 3 game does not work."; + + public static string IssueMapper53 = "Mapper 53 does not work with the test roms i have, maybe something wrong with the implementation or the roms themselves"; + + public static string IssueMapper56 = "Mapper 56 does not work with the test roms i have, maybe something wrong with the implementation or the roms themselves"; + + public static string IssueMapper58 = "Study and Game 32-in-1 (Ch) [!].nes needs keyboard ?"; + + public static string IssueMapper60 = "Mapper 60 does not work with the test roms i have, maybe something wrong with the implementation or the roms themselves"; + + public static string IssueMapper85 = "VRC7 sound channels are not supported"; + + public static string IssueMapper90 = "DipSwitch is not implemented, the irq modes 2-3 are not implemented yet."; + + public static string IssueMapper96 = "Mapper 96 does not function probably and needs special controller to be implemented."; + + public static string IssueMapper105 = "Game hangs on title screen !"; + + public static string IssueMapper119 = "Mapper 119 does not function probably"; + + public static string IssueMapper154 = "Game shows glitches with chr"; + + public static string IssueMapper180 = "Crazy Climber needs special controller which not implemented yet."; + + public static string IssueMapper191 = "Mapper 191 is not tested, issues may occur"; + + public static string IssueMapper193 = "Game show nothing but fighter sprite !"; + + public static string IssueMapper202 = "150 in 1: some games not work well. Is it mapper or rom dump ?"; + + public static string IssueMapper203 = "64-in-1: some games not work, maybe something wrong with the implementation or the rom itself"; + + public static string IssueMapper207 = "Fudou Myouou Den is not assigned as mapper 207 while it should be !"; + + public static string IssueMapper222 = "Mapper 222 is not tested, issues may occur"; + + public static string IssueMapper228 = "Mapper 228 does not function probably"; + + public static string IssueMapper229 = "Mapper 229 is not tested, issues may occur"; + + public static string IssueMapper230 = "Only Contra works !?"; + + public static string IssueMapper243 = "Shows glitches in some games."; + + public static string IssueMapper245 = "Graphic glitches, maybe chr switches."; + + public static string IssueMapper255 = "Mapper 255 is not tested, issues may occur"; + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs.meta new file mode 100644 index 00000000..f42a5873 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MNInterfaceLanguage.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 54f6278d37c378c4f9a6fe125f4505fd +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs new file mode 100644 index 00000000..f1d18bdd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs @@ -0,0 +1,7 @@ +namespace MyNes.Core +{ + [BoardInfo("NROM", 0)] + internal class Mapper000 : Board + { + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs.meta new file mode 100644 index 00000000..a7515e71 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper000.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 41b0e004e19b28f47812b3eab3da2622 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs new file mode 100644 index 00000000..5cd492b6 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs @@ -0,0 +1,245 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("MMC1", 1, 4, 64)] + internal class Mapper001 : Board + { + private int address_reg; + + private byte[] reg = new byte[4]; + + private byte shift = 0; + + private byte buffer = 0; + + private bool flag_p; + + private bool flag_c; + + private bool flag_s; + + private bool enable_wram_enable; + + private int prg_hijackedbit; + + private bool use_hijacked; + + private bool use_sram_switch; + + private int sram_switch_mask; + + private int cpuCycles; + + internal override void HardReset() + { + base.HardReset(); + cpuCycles = 0; + address_reg = 0; + reg = new byte[4]; + reg[0] = 12; + flag_c = false; + flag_s = (flag_p = true); + prg_hijackedbit = 0; + reg[1] = (reg[2] = (reg[3] = 0)); + buffer = 0; + shift = 0; + if (base.Chips.Contains("MMC1B") || base.Chips.Contains("MMC1B2")) + { + TogglePRGRAMEnable(enable: false); + Console.WriteLine("MMC1: SRAM Disabled."); + } + enable_wram_enable = !base.Chips.Contains("MMC1A"); + Console.WriteLine("MMC1: enable_wram_enable = " + enable_wram_enable); + use_hijacked = (PRG_ROM_16KB_Mask & 0x10) == 16; + if (use_hijacked) + { + prg_hijackedbit = 16; + } + use_sram_switch = false; + if (PRG_RAM_08KB_Count > 0) + { + use_sram_switch = true; + sram_switch_mask = (use_hijacked ? 8 : 24); + sram_switch_mask &= PRG_RAM_08KB_Mask << 3; + if (sram_switch_mask == 0) + { + use_sram_switch = false; + } + } + Switch16KPRG(0xF | prg_hijackedbit, PRGArea.AreaC000); + Console.WriteLine("MMC1: use_hijacked = " + use_hijacked); + Console.WriteLine("MMC1: use_sram_switch = " + use_sram_switch); + Console.WriteLine("MMC1: sram_switch_mask = " + sram_switch_mask.ToString("X2")); + } + + internal override void WritePRG(ref ushort address, ref byte value) + { + if (cpuCycles > 0) + { + return; + } + cpuCycles = 3; + if ((value & 0x80) == 128) + { + reg[0] |= 12; + flag_s = (flag_p = true); + shift = (buffer = 0); + return; + } + if ((value & 1) == 1) + { + buffer |= (byte)(1 << (int)shift); + } + if (++shift < 5) + { + return; + } + address_reg = (address & 0x7FFF) >> 13; + reg[address_reg] = buffer; + shift = (buffer = 0); + switch (address_reg) + { + case 0: + flag_c = (reg[0] & 0x10) != 0; + flag_p = (reg[0] & 8) != 0; + flag_s = (reg[0] & 4) != 0; + UpdatePRG(); + UpdateCHR(); + switch (reg[0] & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + } + break; + case 1: + if (!flag_c) + { + Switch08KCHR(reg[1] >> 1); + } + else + { + Switch04KCHR(reg[1], CHRArea.Area0000); + } + if (use_sram_switch) + { + Switch08KPRG((reg[1] & sram_switch_mask) >> 3, PRGArea.Area6000); + } + if (use_hijacked) + { + prg_hijackedbit = reg[1] & 0x10; + UpdatePRG(); + } + break; + case 2: + if (flag_c) + { + Switch04KCHR(reg[2], CHRArea.Area1000); + } + if (use_sram_switch) + { + Switch08KPRG((reg[2] & sram_switch_mask) >> 3, PRGArea.Area6000); + } + if (use_hijacked) + { + prg_hijackedbit = reg[2] & 0x10; + UpdatePRG(); + } + break; + case 3: + if (enable_wram_enable) + { + TogglePRGRAMEnable((reg[3] & 0x10) == 0); + } + UpdatePRG(); + break; + } + } + + private void UpdateCHR() + { + if (!flag_c) + { + Switch08KCHR(reg[1] >> 1); + } + else + { + Switch04KCHR(reg[1], CHRArea.Area0000); + Switch04KCHR(reg[2], CHRArea.Area1000); + } + if (use_sram_switch) + { + Switch08KPRG((reg[1] & sram_switch_mask) >> 3, PRGArea.Area6000); + } + } + + private void UpdatePRG() + { + if (!flag_p) + { + Switch32KPRG(((reg[3] & 0xF) | prg_hijackedbit) >> 1, PRGArea.Area8000); + } + else if (flag_s) + { + Switch16KPRG((reg[3] & 0xF) | prg_hijackedbit, PRGArea.Area8000); + Switch16KPRG(0xF | prg_hijackedbit, PRGArea.AreaC000); + } + else + { + Switch16KPRG(prg_hijackedbit, PRGArea.Area8000); + Switch16KPRG((reg[3] & 0xF) | prg_hijackedbit, PRGArea.AreaC000); + } + } + + internal override void OnCPUClock() + { + if (cpuCycles > 0) + { + cpuCycles--; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(reg); + stream.Write(shift); + stream.Write(buffer); + stream.Write(flag_p); + stream.Write(flag_c); + stream.Write(flag_s); + stream.Write(enable_wram_enable); + stream.Write(prg_hijackedbit); + stream.Write(use_hijacked); + stream.Write(use_sram_switch); + stream.Write(cpuCycles); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + stream.Read(reg, 0, reg.Length); + shift = stream.ReadByte(); + buffer = stream.ReadByte(); + flag_p = stream.ReadBoolean(); + flag_c = stream.ReadBoolean(); + flag_s = stream.ReadBoolean(); + enable_wram_enable = stream.ReadBoolean(); + prg_hijackedbit = stream.ReadInt32(); + use_hijacked = stream.ReadBoolean(); + use_sram_switch = stream.ReadBoolean(); + cpuCycles = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs.meta new file mode 100644 index 00000000..8cb1ef26 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper001.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 42f133f6a434de742b44fc92425e28b8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs new file mode 100644 index 00000000..0573b035 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs @@ -0,0 +1,17 @@ +namespace MyNes.Core +{ + [BoardInfo("UxROM", 2)] + internal class Mapper002 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort addr, ref byte val) + { + Switch16KPRG(val, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs.meta new file mode 100644 index 00000000..db979ebb --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper002.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a977999313c88f0498ab5893280de143 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs new file mode 100644 index 00000000..11684f60 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs @@ -0,0 +1,15 @@ +namespace MyNes.Core +{ + [BoardInfo("CNROM", 3)] + internal class Mapper003 : Board + { + private byte data_temp; + + internal override void WritePRG(ref ushort address, ref byte data) + { + ReadPRG(ref address, out data_temp); + data_temp &= data; + Switch08KCHR(data_temp); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs.meta new file mode 100644 index 00000000..58f60ecf --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper003.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e59d7105def01754ab2fbe6d71f36875 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs new file mode 100644 index 00000000..785b24ef --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs @@ -0,0 +1,224 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("MMC3", 4, true, true)] + internal class Mapper004 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + mmc3_alt_behavior = false; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Tracer.WriteWarning("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Tracer.WriteWarning("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Tracer.WriteWarning("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WritePRG(ref ushort addr, ref byte val) + { + switch (addr & 0xE001) + { + case 32768: + address_8001 = val & 7; + flag_c = (val & 0x80) != 0; + flag_p = (val & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = val; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = val & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((val & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((val & 0x80) != 0); + TogglePRGRAMWritableEnable((val & 0x40) == 0); + break; + case 49152: + irq_reload = val; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter bin) + { + base.WriteStateData(ref bin); + bin.Write(flag_c); + bin.Write(flag_p); + bin.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + bin.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + bin.Write(prg_reg[j]); + } + bin.Write(irq_enabled); + bin.Write(irq_counter); + bin.Write(old_irq_counter); + bin.Write(irq_reload); + bin.Write(irq_clear); + bin.Write(mmc3_alt_behavior); + } + + internal override void ReadStateData(ref BinaryReader bin) + { + base.ReadStateData(ref bin); + flag_c = bin.ReadBoolean(); + flag_p = bin.ReadBoolean(); + address_8001 = bin.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = bin.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = bin.ReadInt32(); + } + irq_enabled = bin.ReadBoolean(); + irq_counter = bin.ReadByte(); + old_irq_counter = bin.ReadInt32(); + irq_reload = bin.ReadByte(); + irq_clear = bin.ReadBoolean(); + mmc3_alt_behavior = bin.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs.meta new file mode 100644 index 00000000..656b91e6 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper004.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e2b6fcc80f0fa6941bdd8fd3fb33c886 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs new file mode 100644 index 00000000..d9f8a48d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs @@ -0,0 +1,890 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("MMC5", 5, 8, 16)] + [WithExternalSound] + [HassIssues] + internal class Mapper005 : Board + { + private int ram_protectA; + + private int ram_protectB; + + private int ExRAM_mode; + + private int[] CHROffset_spr; + + private int[] CHROffsetEX; + + private int[] CHROffsetSP; + + private int[] chrRegA; + + private int[] chrRegB; + + private int[] prgReg; + + private bool useSRAMmirroring; + + private int chr_high; + + private int chr_mode; + + private int prg_mode; + + private bool chr_setB_last; + + private byte temp_val; + + private byte temp_fill; + + private int lastAccessVRAM; + + private int paletteNo; + + private int shift; + + private int EXtilenumber; + + private byte multiplicand; + + private byte multiplier; + + private ushort product; + + private bool split_enable; + + private bool split_right; + + private int split_tile; + + private int split_yscroll; + + private bool split_doit; + + private int split_watch_tile; + + private byte irq_line; + + private byte irq_enable; + + private int irq_pending; + + private int irq_current_counter; + + private int irq_current_inframe; + + private MMC5Sqr snd_1; + + private MMC5Sqr snd_2; + + private MMC5Pcm snd_3; + + private double[][][][][] mix_table; + + internal override string Issues => MNInterfaceLanguage.IssueMapper5; + + internal override void Initialize(IRom rom) + { + base.Initialize(rom); + snd_1 = new MMC5Sqr(); + snd_2 = new MMC5Sqr(); + snd_3 = new MMC5Pcm(); + mix_table = new double[16][][][][]; + for (int i = 0; i < 16; i++) + { + mix_table[i] = new double[16][][][]; + for (int j = 0; j < 16; j++) + { + mix_table[i][j] = new double[16][][]; + for (int k = 0; k < 16; k++) + { + mix_table[i][j][k] = new double[16][]; + for (int l = 0; l < 16; l++) + { + mix_table[i][j][k][l] = new double[256]; + for (int m = 0; m < 256; m++) + { + double num = 95.88 / (8128.0 / (double)(i + j) + 100.0); + double num2 = 159.79 / (1.0 / ((double)k / 8227.0 + (double)l / 12241.0 + (double)m / 22638.0) + 100.0); + mix_table[i][j][k][l][m] = num + num2; + } + } + } + } + } + } + + internal override void HardReset() + { + base.HardReset(); + string text = SHA1.ToUpper(); + string text2 = text; + if (!(text2 == "37267833C984F176DB4B0BC9D45DABA0FFF45304")) + { + if (text2 == "800AEFE756E85A0A78CCB4DAE68EBBA5DF24BF41") + { + useSRAMmirroring = true; + } + } + else + { + useSRAMmirroring = true; + } + Console.WriteLine("MMC5: using PRG RAM mirroring = " + useSRAMmirroring); + CHROffset_spr = new int[8]; + CHROffsetEX = new int[8]; + CHROffsetSP = new int[8]; + chrRegA = new int[8]; + chrRegB = new int[4]; + prgReg = new int[4]; + prgReg[3] = PRG_ROM_08KB_Mask; + prg_mode = 3; + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.Area8000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaA000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaC000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + Switch04kCHREX(0, 0); + Switch04kCHRSP(0, 0); + Switch08kCHR_spr(0); + TogglePRGRAMWritableEnable(enable: true); + TogglePRGRAMEnable(enable: true); + APUApplyChannelsSettings(); + snd_1.HardReset(); + snd_2.HardReset(); + snd_3.HardReset(); + } + + internal override void SoftReset() + { + base.SoftReset(); + snd_1.SoftReset(); + snd_2.SoftReset(); + snd_3.SoftReset(); + } + + internal override void WriteEX(ref ushort address, ref byte value) + { + if (address >= 23552) + { + if (ExRAM_mode == 2) + { + NMT_RAM[2][address & 0x3FF] = value; + } + else if (ExRAM_mode < 2) + { + if (irq_current_inframe == 64) + { + NMT_RAM[2][address & 0x3FF] = value; + } + else + { + NMT_RAM[2][address & 0x3FF] = 0; + } + } + return; + } + switch (address) + { + case 20480: + snd_1.Write0(ref value); + break; + case 20482: + snd_1.Write2(ref value); + break; + case 20483: + snd_1.Write3(ref value); + break; + case 20484: + snd_2.Write0(ref value); + break; + case 20486: + snd_2.Write2(ref value); + break; + case 20487: + snd_2.Write3(ref value); + break; + case 20496: + snd_3.Write5010(value); + break; + case 20497: + snd_3.Write5011(value); + break; + case 20501: + snd_1.WriteEnabled((value & 1) != 0); + snd_2.WriteEnabled((value & 2) != 0); + break; + case 20736: + prg_mode = value & 3; + break; + case 20737: + chr_mode = value & 3; + break; + case 20738: + ram_protectA = value & 3; + UpdateRamProtect(); + break; + case 20739: + ram_protectB = value & 3; + UpdateRamProtect(); + break; + case 20740: + ExRAM_mode = value & 3; + break; + case 20741: + Switch01KNMT(value); + break; + case 20755: + if (!useSRAMmirroring) + { + Switch08KPRG(value & 7, PRGArea.Area6000); + } + else + { + Switch08KPRG((value >> 2) & 1, PRGArea.Area6000); + } + break; + case 20756: + if (prg_mode == 3) + { + Toggle08KPRG_RAM((value & 0x80) == 0, PRGArea.Area8000); + Switch08KPRG(value & 0x7F, PRGArea.Area8000); + } + break; + case 20757: + switch (prg_mode) + { + case 1: + Toggle16KPRG_RAM((value & 0x80) == 0, PRGArea.Area8000); + Switch16KPRG((value & 0x7F) >> 1, PRGArea.Area8000); + break; + case 2: + Toggle16KPRG_RAM((value & 0x80) == 0, PRGArea.Area8000); + Switch16KPRG((value & 0x7F) >> 1, PRGArea.Area8000); + break; + case 3: + Toggle08KPRG_RAM((value & 0x80) == 0, PRGArea.AreaA000); + Switch08KPRG(value & 0x7F, PRGArea.AreaA000); + break; + } + break; + case 20758: + { + int num = prg_mode; + int num2 = num; + if ((uint)(num2 - 2) <= 1u) + { + Toggle08KPRG_RAM((value & 0x80) == 0, PRGArea.AreaC000); + Switch08KPRG(value & 0x7F, PRGArea.AreaC000); + } + break; + } + case 20759: + switch (prg_mode) + { + case 0: + Switch32KPRG((value & 0x7C) >> 2, PRGArea.Area8000); + break; + case 1: + Switch16KPRG((value & 0x7F) >> 1, PRGArea.AreaC000); + break; + case 2: + Switch08KPRG(value & 0x7F, PRGArea.AreaE000); + break; + case 3: + Switch08KPRG(value & 0x7F, PRGArea.AreaE000); + break; + } + break; + case 20768: + chr_setB_last = false; + if (chr_mode == 3) + { + Switch01kCHR_spr(value | chr_high, 0); + } + break; + case 20769: + chr_setB_last = false; + switch (chr_mode) + { + case 2: + Switch02kCHR_spr(value | chr_high, 0); + break; + case 3: + Switch01kCHR_spr(value | chr_high, 1024); + break; + } + break; + case 20770: + chr_setB_last = false; + if (chr_mode == 3) + { + Switch01kCHR_spr(value | chr_high, 2048); + } + break; + case 20771: + chr_setB_last = false; + switch (chr_mode) + { + case 1: + Switch04kCHR_spr(value | chr_high, 0); + break; + case 2: + Switch02kCHR_spr(value | chr_high, 2048); + break; + case 3: + Switch01kCHR_spr(value | chr_high, 3072); + break; + } + break; + case 20772: + chr_setB_last = false; + if (chr_mode == 3) + { + Switch01kCHR_spr(value | chr_high, 4096); + } + break; + case 20773: + chr_setB_last = false; + switch (chr_mode) + { + case 2: + Switch02kCHR_spr(value | chr_high, 4096); + break; + case 3: + Switch01kCHR_spr(value | chr_high, 5120); + break; + } + break; + case 20774: + chr_setB_last = false; + if (chr_mode == 3) + { + Switch01kCHR_spr(value | chr_high, 6144); + } + break; + case 20775: + chr_setB_last = false; + switch (chr_mode) + { + case 0: + Switch08kCHR_spr(value | chr_high); + break; + case 1: + Switch04kCHR_spr(value | chr_high, 4096); + break; + case 2: + Switch02kCHR_spr(value | chr_high, 6144); + break; + case 3: + Switch01kCHR_spr(value | chr_high, 7168); + break; + } + break; + case 20776: + chr_setB_last = true; + if (chr_mode == 3) + { + Switch01KCHR(value | chr_high, CHRArea.Area0000); + Switch01KCHR(value | chr_high, CHRArea.Area1000); + } + break; + case 20777: + chr_setB_last = true; + switch (chr_mode) + { + case 2: + Switch02KCHR(value | chr_high, CHRArea.Area0000); + Switch02KCHR(value | chr_high, CHRArea.Area1000); + break; + case 3: + Switch01KCHR(value | chr_high, CHRArea.Area0400); + Switch01KCHR(value | chr_high, CHRArea.Area1400); + break; + } + break; + case 20778: + chr_setB_last = true; + if (chr_mode == 3) + { + Switch01KCHR(value | chr_high, CHRArea.Area0800); + Switch01KCHR(value | chr_high, CHRArea.Area1800); + } + break; + case 20779: + chr_setB_last = true; + switch (chr_mode) + { + case 0: + Switch04kCHR_bkg(value | chr_high, 0); + Switch04kCHR_bkg(value | chr_high, 4096); + break; + case 1: + Switch04KCHR(value | chr_high, CHRArea.Area0000); + Switch04KCHR(value | chr_high, CHRArea.Area1000); + break; + case 2: + Switch02KCHR(value | chr_high, CHRArea.Area0800); + Switch02KCHR(value | chr_high, CHRArea.Area1800); + break; + case 3: + Switch01KCHR(value | chr_high, CHRArea.Area0C00); + Switch01KCHR(value | chr_high, CHRArea.Area1C00); + break; + } + break; + case 20784: + chr_high = (value & 3) << 8; + break; + case 20742: + { + for (int j = 0; j < 960; j++) + { + NMT_RAM[3][j] = value; + } + break; + } + case 20743: + { + for (int i = 960; i < 1024; i++) + { + temp_fill = (byte)((uint)(2 << (value & 3)) | (value & 3u)); + temp_fill |= (byte)((temp_fill & 0xF) << 4); + NMT_RAM[3][i] = temp_fill; + } + break; + } + case 20992: + split_tile = value & 0x1F; + split_enable = (value & 0x80) == 128; + split_right = (value & 0x40) == 64; + break; + case 20993: + split_yscroll = value; + break; + case 20994: + Switch04kCHRSP(value, address & 0); + Switch04kCHRSP(value, address & 0x1000); + break; + case 20995: + irq_line = value; + break; + case 20996: + irq_enable = value; + break; + case 20997: + multiplicand = value; + product = (ushort)(multiplicand * multiplier); + break; + case 20998: + multiplier = value; + product = (ushort)(multiplicand * multiplier); + break; + } + } + + internal override void ReadEX(ref ushort address, out byte data) + { + if (address >= 23552 && ExRAM_mode >= 2) + { + data = NMT_RAM[2][address & 0x3FF]; + return; + } + switch (address) + { + case 20496: + data = snd_3.Read5010(); + break; + case 20996: + data = (byte)(irq_current_inframe | irq_pending); + irq_pending = 0; + NesEmu.IRQFlags &= -9; + break; + case 20997: + data = (byte)(product & 0xFFu); + break; + case 20998: + data = (byte)((product & 0xFF00) >> 8); + break; + case 20501: + data = (byte)((snd_1.ReadEnable() ? 1u : 0u) | (snd_2.ReadEnable() ? 2u : 0u)); + data = 0; + break; + default: + data = 0; + break; + } + } + + internal override void ReadCHR(ref ushort address, out byte data) + { + if (!NesEmu.ppu_is_sprfetch && split_enable && ExRAM_mode < 2) + { + split_watch_tile = address & 0x3F; + if (!split_right) + { + split_doit = split_watch_tile < split_tile; + } + else + { + split_doit = split_watch_tile >= split_tile; + } + if (!split_doit) + { + } + } + if (ExRAM_mode == 1) + { + if (!NesEmu.ppu_is_sprfetch) + { + EXtilenumber = NMT_RAM[2][lastAccessVRAM] & 0x3F; + Switch04kCHREX(EXtilenumber | chr_high, address & 0x1000); + data = CHR_ROM[CHROffsetEX[(address >> 10) & 7]][address & 0x3FF]; + } + else + { + data = CHR_ROM[CHROffset_spr[(address >> 10) & 7]][address & 0x3FF]; + } + } + else if (NesEmu.ppu_reg_2000_Sprite_size == 16) + { + if (!NesEmu.ppu_is_sprfetch) + { + data = CHR_ROM[CHR_AREA_BLK_INDEX[(address >> 10) & 7]][address & 0x3FF]; + } + else + { + data = CHR_ROM[CHROffset_spr[(address >> 10) & 7]][address & 0x3FF]; + } + } + else if (chr_setB_last) + { + data = CHR_ROM[CHR_AREA_BLK_INDEX[(address >> 10) & 7]][address & 0x3FF]; + } + else + { + data = CHR_ROM[CHROffset_spr[(address >> 10) & 7]][address & 0x3FF]; + } + } + + internal override void ReadNMT(ref ushort address, out byte data) + { + if (split_doit) + { + } + if (ExRAM_mode == 1) + { + if ((address & 0x3FF) <= 959) + { + lastAccessVRAM = address & 0x3FF; + } + else + { + paletteNo = NMT_RAM[2][lastAccessVRAM] & 0xC0; + shift = ((lastAccessVRAM >> 4) & 4) | (lastAccessVRAM & 2); + switch (shift) + { + case 0: + data = (byte)(paletteNo >> 6); + return; + case 2: + data = (byte)(paletteNo >> 4); + return; + case 4: + data = (byte)(paletteNo >> 2); + return; + case 6: + data = (byte)paletteNo; + return; + } + } + } + data = NMT_RAM[NMT_AREA_BLK_INDEX[(address >> 10) & 3]][address & 0x3FF]; + } + + internal override void WriteNMT(ref ushort address, ref byte value) + { + if (ExRAM_mode == 1 && (address & 0x3FF) <= 959) + { + lastAccessVRAM = address & 0x3FF; + } + NMT_RAM[NMT_AREA_BLK_INDEX[(address >> 10) & 3]][address & 0x3FF] = value; + } + + private void UpdateRamProtect() + { + TogglePRGRAMWritableEnable(ram_protectA == 2 && ram_protectB == 1); + } + + private void Switch04kCHR_bkg(int index, int where) + { + int num = (where >> 10) & 7; + index <<= 2; + CHR_AREA_BLK_INDEX[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHR_AREA_BLK_INDEX[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHR_AREA_BLK_INDEX[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHR_AREA_BLK_INDEX[num] = index & CHR_ROM_01KB_Mask; + } + + private void Switch01kCHR_spr(int index, int where) + { + CHROffset_spr[(where >> 10) & 7] = index & CHR_ROM_01KB_Mask; + } + + private void Switch02kCHR_spr(int index, int where) + { + int num = (where >> 10) & 7; + index <<= 1; + CHROffset_spr[num] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[num + 1] = index & CHR_ROM_01KB_Mask; + } + + private void Switch04kCHR_spr(int index, int where) + { + int num = (where >> 10) & 7; + index <<= 2; + CHROffset_spr[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffset_spr[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffset_spr[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffset_spr[num] = index & CHR_ROM_01KB_Mask; + } + + private void Switch08kCHR_spr(int index) + { + index <<= 3; + CHROffset_spr[0] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[1] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[2] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[3] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[4] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[5] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[6] = index & CHR_ROM_01KB_Mask; + index++; + CHROffset_spr[7] = index & CHR_ROM_01KB_Mask; + } + + private void Switch04kCHREX(int index, int where) + { + int num = (where >> 10) & 7; + index <<= 2; + CHROffsetEX[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffsetEX[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffsetEX[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffsetEX[num] = index & CHR_ROM_01KB_Mask; + } + + private void Switch04kCHRSP(int index, int where) + { + int num = (where >> 10) & 7; + index <<= 2; + CHROffsetSP[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffsetSP[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffsetSP[num] = index & CHR_ROM_01KB_Mask; + num++; + index++; + CHROffsetSP[num] = index & CHR_ROM_01KB_Mask; + } + + internal override void OnPPUScanlineTick() + { + irq_current_inframe = ((NesEmu.IsInRender() && NesEmu.IsRenderingOn()) ? 64 : 0); + if (irq_current_inframe == 0) + { + irq_current_inframe = 64; + irq_current_counter = 0; + irq_pending = 0; + NesEmu.IRQFlags &= -9; + return; + } + irq_current_counter++; + if (irq_current_counter == irq_line) + { + irq_pending = 128; + if (irq_enable == 128) + { + NesEmu.IRQFlags |= 8; + } + } + } + + internal override void OnAPUClock() + { + base.OnAPUClock(); + snd_1.Clock(); + snd_2.Clock(); + } + + internal override void OnAPUClockEnvelope() + { + base.OnAPUClockEnvelope(); + snd_1.ClockLength(); + snd_2.ClockLength(); + snd_1.ClockEnvelope(); + snd_2.ClockEnvelope(); + } + + internal override double APUGetSample() + { + return mix_table[snd_1.output][snd_2.output][0][0][snd_3.output]; + } + + internal override void APUApplyChannelsSettings() + { + base.APUApplyChannelsSettings(); + snd_1.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_MMC5_SQ1; + snd_2.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_MMC5_SQ2; + snd_3.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_MMC5_PCM; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(ram_protectA); + stream.Write(ram_protectB); + stream.Write(ExRAM_mode); + for (int i = 0; i < CHROffset_spr.Length; i++) + { + stream.Write(CHROffset_spr[i]); + } + for (int j = 0; j < CHROffsetEX.Length; j++) + { + stream.Write(CHROffsetEX[j]); + } + for (int k = 0; k < CHROffsetSP.Length; k++) + { + stream.Write(CHROffsetSP[k]); + } + for (int l = 0; l < chrRegA.Length; l++) + { + stream.Write(chrRegA[l]); + } + for (int m = 0; m < chrRegB.Length; m++) + { + stream.Write(chrRegB[m]); + } + for (int n = 0; n < prgReg.Length; n++) + { + stream.Write(prgReg[n]); + } + stream.Write(useSRAMmirroring); + stream.Write(chr_high); + stream.Write(chr_mode); + stream.Write(prg_mode); + stream.Write(chr_setB_last); + stream.Write(temp_val); + stream.Write(temp_fill); + stream.Write(lastAccessVRAM); + stream.Write(paletteNo); + stream.Write(shift); + stream.Write(EXtilenumber); + stream.Write(multiplicand); + stream.Write(multiplier); + stream.Write(product); + stream.Write(split_enable); + stream.Write(split_right); + stream.Write(split_tile); + stream.Write(split_yscroll); + stream.Write(split_doit); + stream.Write(split_watch_tile); + stream.Write(irq_line); + stream.Write(irq_enable); + stream.Write(irq_pending); + stream.Write(irq_current_counter); + stream.Write(irq_current_inframe); + snd_1.WriteStateData(ref stream); + snd_2.WriteStateData(ref stream); + snd_3.SaveState(ref stream); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + ram_protectA = stream.ReadInt32(); + ram_protectB = stream.ReadInt32(); + ExRAM_mode = stream.ReadInt32(); + for (int i = 0; i < CHROffset_spr.Length; i++) + { + CHROffset_spr[i] = stream.ReadInt32(); + } + for (int j = 0; j < CHROffsetEX.Length; j++) + { + CHROffsetEX[j] = stream.ReadInt32(); + } + for (int k = 0; k < CHROffsetSP.Length; k++) + { + CHROffsetSP[k] = stream.ReadInt32(); + } + for (int l = 0; l < chrRegA.Length; l++) + { + chrRegA[l] = stream.ReadInt32(); + } + for (int m = 0; m < chrRegB.Length; m++) + { + chrRegB[m] = stream.ReadInt32(); + } + for (int n = 0; n < prgReg.Length; n++) + { + prgReg[n] = stream.ReadInt32(); + } + useSRAMmirroring = stream.ReadBoolean(); + chr_high = stream.ReadInt32(); + chr_mode = stream.ReadInt32(); + prg_mode = stream.ReadInt32(); + chr_setB_last = stream.ReadBoolean(); + temp_val = stream.ReadByte(); + temp_fill = stream.ReadByte(); + lastAccessVRAM = stream.ReadInt32(); + paletteNo = stream.ReadInt32(); + shift = stream.ReadInt32(); + EXtilenumber = stream.ReadInt32(); + multiplicand = stream.ReadByte(); + multiplier = stream.ReadByte(); + product = stream.ReadUInt16(); + split_enable = stream.ReadBoolean(); + split_right = stream.ReadBoolean(); + split_tile = stream.ReadInt32(); + split_yscroll = stream.ReadInt32(); + split_doit = stream.ReadBoolean(); + split_watch_tile = stream.ReadInt32(); + irq_line = stream.ReadByte(); + irq_enable = stream.ReadByte(); + irq_pending = stream.ReadInt32(); + irq_current_counter = stream.ReadInt32(); + irq_current_inframe = stream.ReadInt32(); + snd_1.ReadStateData(ref stream); + snd_2.ReadStateData(ref stream); + snd_3.LoadState(ref stream); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs.meta new file mode 100644 index 00000000..bdd56936 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper005.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: bf04ed9f8d2fd1e4bb54763b3bc7bcfa +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs new file mode 100644 index 00000000..c16526cf --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs @@ -0,0 +1,21 @@ +namespace MyNes.Core +{ + [BoardInfo("FFE F4xxx", 6)] + [HassIssues] + internal class Mapper006 : FFE + { + internal override string Issues => MNInterfaceLanguage.IssueMapper6; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(7, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(data & 3); + Switch16KPRG((data >> 2) & 0xF, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs.meta new file mode 100644 index 00000000..d05dd0f5 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper006.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a4104e887a80ef44f865e981d84b4d01 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs new file mode 100644 index 00000000..5e2a319c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("AxROM", 7)] + internal class Mapper007 : Board + { + internal override void WritePRG(ref ushort addr, ref byte val) + { + Switch01KNMTFromMirroring(((val & 0x10) == 16) ? Mirroring.OneScB : Mirroring.OneScA); + Switch32KPRG(val & 7, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs.meta new file mode 100644 index 00000000..f142f132 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper007.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9829484ad2c88fd4ca0539447d126725 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs new file mode 100644 index 00000000..3a44584b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs @@ -0,0 +1,15 @@ +namespace MyNes.Core +{ + [BoardInfo("FFE F3xxx", 8)] + [HassIssues] + internal class Mapper008 : FFE + { + internal override string Issues => MNInterfaceLanguage.IssueMapper8; + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch32KPRG((data >> 4) & 3, PRGArea.Area8000); + Switch08KCHR(data & 3); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs.meta new file mode 100644 index 00000000..b7e31ce1 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper008.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e8b28ac4454361544919b7f879ab32d4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs new file mode 100644 index 00000000..fe7a816d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs @@ -0,0 +1,7 @@ +namespace MyNes.Core +{ + [BoardInfo("MMC2", 9)] + internal class Mapper009 : MMC2 + { + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs.meta new file mode 100644 index 00000000..c6bd0b75 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper009.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b96edc1b48f9f6f44bcfa04f1d3bda1d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs new file mode 100644 index 00000000..22399d0d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs @@ -0,0 +1,25 @@ +namespace MyNes.Core +{ + [BoardInfo("MMC4", 10)] + internal class Mapper010 : MMC2 + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(0, PRGArea.Area8000); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if ((address & 0xF000) == 40960) + { + Switch16KPRG(data, PRGArea.Area8000); + } + else + { + base.WritePRG(ref address, ref data); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs.meta new file mode 100644 index 00000000..f544b11c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper010.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4b1f6a01ae2d9b840b1bf3e87ba875f3 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs new file mode 100644 index 00000000..f0144a0c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs @@ -0,0 +1,16 @@ +namespace MyNes.Core +{ + [BoardInfo("Color Dreams", 11)] + internal class Mapper011 : Board + { + private byte writeData; + + internal override void WritePRG(ref ushort address, ref byte data) + { + ReadPRG(ref address, out writeData); + writeData &= data; + Switch32KPRG(writeData & 3, PRGArea.Area8000); + Switch08KCHR((writeData >> 4) & 0xF); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs.meta new file mode 100644 index 00000000..36e2e45d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper011.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5c0a3239622d8e841b4cc5d0e35f15ae +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs new file mode 100644 index 00000000..206cfbd1 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs @@ -0,0 +1,21 @@ +namespace MyNes.Core +{ + [BoardInfo("CPROM", 13, 1, 16)] + internal class Mapper013 : Board + { + private byte writeData; + + internal override void HardReset() + { + base.HardReset(); + Toggle08KCHR_RAM(ram: true); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + ReadPRG(ref address, out writeData); + writeData &= data; + Switch04KCHR(writeData & 3, CHRArea.Area1000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs.meta new file mode 100644 index 00000000..ad302e81 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper013.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6244402647756ed4c899d2d9e2b48754 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs new file mode 100644 index 00000000..b3b12469 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs @@ -0,0 +1,36 @@ +namespace MyNes.Core +{ + [BoardInfo("100-in-1 Contra Function 16", 15)] + internal class Mapper015 : Board + { + private int temp; + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 3) + { + case 0: + Switch16KPRG(data & 0x3F, PRGArea.Area8000); + Switch16KPRG((data & 0x3F) | 1, PRGArea.AreaC000); + break; + case 1: + Switch16KPRG(data & 0x3F, PRGArea.Area8000); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + break; + case 2: + temp = data << 1; + temp = ((data & 0x3F) << 1) | ((data >> 7) & 1); + Switch08KPRG(temp, PRGArea.Area8000); + Switch08KPRG(temp, PRGArea.AreaA000); + Switch08KPRG(temp, PRGArea.AreaC000); + Switch08KPRG(temp, PRGArea.AreaE000); + break; + case 3: + Switch16KPRG(data & 0x3F, PRGArea.Area8000); + Switch16KPRG(data & 0x3F, PRGArea.AreaC000); + break; + } + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs.meta new file mode 100644 index 00000000..94eb1ed3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper015.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 91c36800c0238a84cbe2f1313237d9df +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs new file mode 100644 index 00000000..f72f00cb --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs @@ -0,0 +1,7 @@ +namespace MyNes.Core +{ + [BoardInfo("Bandai", 16)] + internal class Mapper016 : Bandai + { + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs.meta new file mode 100644 index 00000000..52881ef7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper016.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0a10f96e86bb11143ba3f4bc30388390 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs new file mode 100644 index 00000000..e6734a37 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs @@ -0,0 +1,64 @@ +namespace MyNes.Core +{ + [BoardInfo("FFE F8xxx", 17)] + internal class Mapper017 : FFE + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + switch (address) + { + case 17668: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 17669: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 17670: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 17671: + Switch08KPRG(data, PRGArea.AreaE000); + break; + case 17680: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 17681: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 17682: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 17683: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 17684: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 17685: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 17686: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 17687: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 17672: + case 17673: + case 17674: + case 17675: + case 17676: + case 17677: + case 17678: + case 17679: + break; + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs.meta new file mode 100644 index 00000000..601a986e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper017.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a53431c37c141204e9148d581878b56e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs new file mode 100644 index 00000000..c65aeb49 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs @@ -0,0 +1,219 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Jaleco SS8806", 18)] + internal class Mapper018 : Board + { + private int[] prg_reg; + + private int[] chr_reg; + + private int irqRelaod; + + private int irqCounter; + + private bool irqEnable; + + private int irqMask; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + prg_reg = new int[3]; + chr_reg = new int[8]; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF003) + { + case 32768: + prg_reg[0] = (prg_reg[0] & 0xF0) | (data & 0xF); + Switch08KPRG(prg_reg[0], PRGArea.Area8000); + break; + case 32769: + prg_reg[0] = (prg_reg[0] & 0xF) | ((data & 0xF) << 4); + Switch08KPRG(prg_reg[0], PRGArea.Area8000); + break; + case 32770: + prg_reg[1] = (prg_reg[1] & 0xF0) | (data & 0xF); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + break; + case 32771: + prg_reg[1] = (prg_reg[1] & 0xF) | ((data & 0xF) << 4); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + break; + case 36864: + prg_reg[2] = (prg_reg[2] & 0xF0) | (data & 0xF); + Switch08KPRG(prg_reg[2], PRGArea.AreaC000); + break; + case 36865: + prg_reg[2] = (prg_reg[2] & 0xF) | ((data & 0xF) << 4); + Switch08KPRG(prg_reg[2], PRGArea.AreaC000); + break; + case 40960: + chr_reg[0] = (chr_reg[0] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[0], CHRArea.Area0000); + break; + case 40961: + chr_reg[0] = (chr_reg[0] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[0], CHRArea.Area0000); + break; + case 40962: + chr_reg[1] = (chr_reg[1] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[1], CHRArea.Area0400); + break; + case 40963: + chr_reg[1] = (chr_reg[1] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[1], CHRArea.Area0400); + break; + case 45056: + chr_reg[2] = (chr_reg[2] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[2], CHRArea.Area0800); + break; + case 45057: + chr_reg[2] = (chr_reg[2] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[2], CHRArea.Area0800); + break; + case 45058: + chr_reg[3] = (chr_reg[3] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[3], CHRArea.Area0C00); + break; + case 45059: + chr_reg[3] = (chr_reg[3] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[3], CHRArea.Area0C00); + break; + case 49152: + chr_reg[4] = (chr_reg[4] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[4], CHRArea.Area1000); + break; + case 49153: + chr_reg[4] = (chr_reg[4] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[4], CHRArea.Area1000); + break; + case 49154: + chr_reg[5] = (chr_reg[5] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[5], CHRArea.Area1400); + break; + case 49155: + chr_reg[5] = (chr_reg[5] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[5], CHRArea.Area1400); + break; + case 53248: + chr_reg[6] = (chr_reg[6] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[6], CHRArea.Area1800); + break; + case 53249: + chr_reg[6] = (chr_reg[6] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[6], CHRArea.Area1800); + break; + case 53250: + chr_reg[7] = (chr_reg[7] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_reg[7], CHRArea.Area1C00); + break; + case 53251: + chr_reg[7] = (chr_reg[7] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_reg[7], CHRArea.Area1C00); + break; + case 57344: + irqRelaod = (irqRelaod & 0xFFF0) | (data & 0xF); + break; + case 57345: + irqRelaod = (irqRelaod & 0xFF0F) | ((data & 0xF) << 4); + break; + case 57346: + irqRelaod = (irqRelaod & 0xF0FF) | ((data & 0xF) << 8); + break; + case 57347: + irqRelaod = (irqRelaod & 0xFFF) | ((data & 0xF) << 12); + break; + case 61440: + irqCounter = irqRelaod; + NesEmu.IRQFlags &= -9; + break; + case 61441: + irqEnable = (data & 1) == 1; + if ((data & 8) == 8) + { + irqMask = 15; + } + else if ((data & 4) == 4) + { + irqMask = 255; + } + else if ((data & 2) == 2) + { + irqMask = 4095; + } + else + { + irqMask = 65535; + } + NesEmu.IRQFlags &= -9; + break; + case 61442: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + } + } + + internal override void OnCPUClock() + { + if (irqEnable && (irqCounter & irqMask) > 0 && (--irqCounter & irqMask) == 0) + { + irqEnable = false; + NesEmu.IRQFlags |= 8; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + for (int i = 0; i < prg_reg.Length; i++) + { + stream.Write(prg_reg[i]); + } + for (int j = 0; j < chr_reg.Length; j++) + { + stream.Write(chr_reg[j]); + } + stream.Write(irqRelaod); + stream.Write(irqCounter); + stream.Write(irqEnable); + stream.Write(irqMask); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + for (int i = 0; i < prg_reg.Length; i++) + { + prg_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < chr_reg.Length; j++) + { + chr_reg[j] = stream.ReadInt32(); + } + irqRelaod = stream.ReadInt32(); + irqCounter = stream.ReadInt32(); + irqEnable = stream.ReadBoolean(); + irqMask = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs.meta new file mode 100644 index 00000000..3fc21f01 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper018.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b9409e3543eded64ba7d391f604d4157 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs new file mode 100644 index 00000000..25d41635 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs @@ -0,0 +1,7 @@ +namespace MyNes.Core +{ + [BoardInfo("Namcot 106", 19, 1, 256)] + internal class Mapper019 : Namcot106 + { + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs.meta new file mode 100644 index 00000000..86d96db8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper019.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e12370c4d1f15a246ad180d702f92a1a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs new file mode 100644 index 00000000..5bd6e944 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs @@ -0,0 +1,254 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC4", 21)] + internal class Mapper021 : Board + { + private bool prg_mode; + + private byte prg_reg0; + + private int[] chr_Reg; + + private int irq_reload; + + private int irq_counter; + + private int prescaler; + + private bool irq_mode_cycle; + + private bool irq_enable; + + private bool irq_enable_on_ak; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + prescaler = 341; + chr_Reg = new int[8]; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + case 32770: + case 32772: + case 32774: + case 32832: + case 32896: + case 32960: + prg_reg0 = data; + Switch08KPRG(prg_mode ? (PRG_ROM_08KB_Mask - 1) : (prg_reg0 & 0x1F), PRGArea.Area8000); + Switch08KPRG(prg_mode ? (prg_reg0 & 0x1F) : (PRG_ROM_08KB_Mask - 1), PRGArea.AreaC000); + break; + case 36864: + case 36866: + case 36928: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 36868: + case 36870: + case 36992: + case 37056: + prg_mode = (data & 2) == 2; + Switch08KPRG(prg_mode ? (PRG_ROM_08KB_Mask - 1) : (prg_reg0 & 0x1F), PRGArea.Area8000); + Switch08KPRG(prg_mode ? (prg_reg0 & 0x1F) : (PRG_ROM_08KB_Mask - 1), PRGArea.AreaC000); + break; + case 40960: + case 40962: + case 40964: + case 40966: + case 41024: + case 41088: + case 41152: + Switch08KPRG(data & 0x1F, PRGArea.AreaA000); + break; + case 45056: + chr_Reg[0] = (chr_Reg[0] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[0], CHRArea.Area0000); + break; + case 45058: + case 45120: + chr_Reg[0] = (chr_Reg[0] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[0], CHRArea.Area0000); + break; + case 45060: + case 45184: + chr_Reg[1] = (chr_Reg[1] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[1], CHRArea.Area0400); + break; + case 45062: + case 45248: + chr_Reg[1] = (chr_Reg[1] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[1], CHRArea.Area0400); + break; + case 49152: + chr_Reg[2] = (chr_Reg[2] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[2], CHRArea.Area0800); + break; + case 49154: + case 49216: + chr_Reg[2] = (chr_Reg[2] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[2], CHRArea.Area0800); + break; + case 49156: + case 49280: + chr_Reg[3] = (chr_Reg[3] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[3], CHRArea.Area0C00); + break; + case 49158: + case 49344: + chr_Reg[3] = (chr_Reg[3] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[3], CHRArea.Area0C00); + break; + case 53248: + chr_Reg[4] = (chr_Reg[4] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[4], CHRArea.Area1000); + break; + case 53250: + case 53312: + chr_Reg[4] = (chr_Reg[4] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[4], CHRArea.Area1000); + break; + case 53252: + case 53376: + chr_Reg[5] = (chr_Reg[5] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[5], CHRArea.Area1400); + break; + case 53254: + case 53440: + chr_Reg[5] = (chr_Reg[5] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[5], CHRArea.Area1400); + break; + case 57344: + chr_Reg[6] = (chr_Reg[6] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[6], CHRArea.Area1800); + break; + case 57346: + case 57408: + chr_Reg[6] = (chr_Reg[6] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[6], CHRArea.Area1800); + break; + case 57348: + case 57472: + chr_Reg[7] = (chr_Reg[7] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[7], CHRArea.Area1C00); + break; + case 57350: + case 57536: + chr_Reg[7] = (chr_Reg[7] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[7], CHRArea.Area1C00); + break; + case 61440: + irq_reload = (irq_reload & 0xF0) | (data & 0xF); + break; + case 61442: + case 61504: + irq_reload = (irq_reload & 0xF) | ((data & 0xF) << 4); + break; + case 61444: + case 61568: + irq_mode_cycle = (data & 4) == 4; + irq_enable = (data & 2) == 2; + irq_enable_on_ak = (data & 1) == 1; + if (irq_enable) + { + irq_counter = irq_reload; + prescaler = 341; + } + NesEmu.IRQFlags &= -9; + break; + case 61446: + case 61632: + NesEmu.IRQFlags &= -9; + irq_enable = irq_enable_on_ak; + break; + } + } + + internal override void OnCPUClock() + { + if (!irq_enable) + { + return; + } + if (!irq_mode_cycle) + { + if (prescaler > 0) + { + prescaler -= 3; + return; + } + prescaler = 341; + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + else + { + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prg_mode); + stream.Write(prg_reg0); + for (int i = 0; i < chr_Reg.Length; i++) + { + stream.Write(chr_Reg[i]); + } + stream.Write(irq_reload); + stream.Write(irq_counter); + stream.Write(prescaler); + stream.Write(irq_mode_cycle); + stream.Write(irq_enable); + stream.Write(irq_enable_on_ak); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prg_mode = stream.ReadBoolean(); + prg_reg0 = stream.ReadByte(); + for (int i = 0; i < chr_Reg.Length; i++) + { + chr_Reg[i] = stream.ReadInt32(); + } + irq_reload = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + prescaler = stream.ReadInt32(); + irq_mode_cycle = stream.ReadBoolean(); + irq_enable = stream.ReadBoolean(); + irq_enable_on_ak = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs.meta new file mode 100644 index 00000000..59f7adaa --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper021.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0579b5a05fd8d6f4e8351ee140dda132 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs new file mode 100644 index 00000000..29aa8b43 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs @@ -0,0 +1,138 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC2", 22)] + internal class Mapper022 : Board + { + private int[] chr_Reg; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + chr_Reg = new int[8]; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + case 32769: + case 32770: + case 32771: + Switch08KPRG(data & 0xF, PRGArea.Area8000); + break; + case 36864: + case 36865: + case 36866: + case 36867: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 40960: + case 40961: + case 40962: + case 40963: + Switch08KPRG(data & 0xF, PRGArea.AreaA000); + break; + case 45056: + chr_Reg[0] = (chr_Reg[0] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[0] >> 1, CHRArea.Area0000); + break; + case 45058: + chr_Reg[0] = (chr_Reg[0] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[0] >> 1, CHRArea.Area0000); + break; + case 45057: + chr_Reg[1] = (chr_Reg[1] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[1] >> 1, CHRArea.Area0400); + break; + case 45059: + chr_Reg[1] = (chr_Reg[1] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[1] >> 1, CHRArea.Area0400); + break; + case 49152: + chr_Reg[2] = (chr_Reg[2] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[2] >> 1, CHRArea.Area0800); + break; + case 49154: + chr_Reg[2] = (chr_Reg[2] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[2] >> 1, CHRArea.Area0800); + break; + case 49153: + chr_Reg[3] = (chr_Reg[3] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[3] >> 1, CHRArea.Area0C00); + break; + case 49155: + chr_Reg[3] = (chr_Reg[3] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[3] >> 1, CHRArea.Area0C00); + break; + case 53248: + chr_Reg[4] = (chr_Reg[4] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[4] >> 1, CHRArea.Area1000); + break; + case 53250: + chr_Reg[4] = (chr_Reg[4] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[4] >> 1, CHRArea.Area1000); + break; + case 53249: + chr_Reg[5] = (chr_Reg[5] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[5] >> 1, CHRArea.Area1400); + break; + case 53251: + chr_Reg[5] = (chr_Reg[5] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[5] >> 1, CHRArea.Area1400); + break; + case 57344: + chr_Reg[6] = (chr_Reg[6] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[6] >> 1, CHRArea.Area1800); + break; + case 57346: + chr_Reg[6] = (chr_Reg[6] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[6] >> 1, CHRArea.Area1800); + break; + case 57345: + chr_Reg[7] = (chr_Reg[7] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[7] >> 1, CHRArea.Area1C00); + break; + case 57347: + chr_Reg[7] = (chr_Reg[7] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[7] >> 1, CHRArea.Area1C00); + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + for (int i = 0; i < chr_Reg.Length; i++) + { + stream.Write(chr_Reg[i]); + } + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + for (int i = 0; i < chr_Reg.Length; i++) + { + chr_Reg[i] = stream.ReadInt32(); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs.meta new file mode 100644 index 00000000..ddd31b16 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper022.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 37f7c4432f871c64287860405074e0cb +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs new file mode 100644 index 00000000..b540714d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs @@ -0,0 +1,163 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC2", 23)] + internal class Mapper023 : Board + { + private int[] chr_Reg; + + private byte security; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + chr_Reg = new int[8]; + security = 0; + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (address == 24576) + { + security = (byte)(data & 1u); + } + } + + internal override void ReadSRM(ref ushort address, out byte data) + { + if (address == 24576) + { + data = security; + } + else + { + data = 0; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + case 32769: + case 32770: + case 32771: + Switch08KPRG(data & 0xF, PRGArea.Area8000); + break; + case 36864: + case 36865: + case 36866: + case 36867: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 40960: + case 40961: + case 40962: + case 40963: + Switch08KPRG(data & 0xF, PRGArea.AreaA000); + break; + case 45056: + chr_Reg[0] = (chr_Reg[0] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[0], CHRArea.Area0000); + break; + case 45057: + chr_Reg[0] = (chr_Reg[0] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[0], CHRArea.Area0000); + break; + case 45058: + chr_Reg[1] = (chr_Reg[1] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[1], CHRArea.Area0400); + break; + case 45059: + chr_Reg[1] = (chr_Reg[1] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[1], CHRArea.Area0400); + break; + case 49152: + chr_Reg[2] = (chr_Reg[2] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[2], CHRArea.Area0800); + break; + case 49153: + chr_Reg[2] = (chr_Reg[2] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[2], CHRArea.Area0800); + break; + case 49154: + chr_Reg[3] = (chr_Reg[3] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[3], CHRArea.Area0C00); + break; + case 49155: + chr_Reg[3] = (chr_Reg[3] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[3], CHRArea.Area0C00); + break; + case 53248: + chr_Reg[4] = (chr_Reg[4] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[4], CHRArea.Area1000); + break; + case 53249: + chr_Reg[4] = (chr_Reg[4] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[4], CHRArea.Area1000); + break; + case 53250: + chr_Reg[5] = (chr_Reg[5] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[5], CHRArea.Area1400); + break; + case 53251: + chr_Reg[5] = (chr_Reg[5] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[5], CHRArea.Area1400); + break; + case 57344: + chr_Reg[6] = (chr_Reg[6] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[6], CHRArea.Area1800); + break; + case 57345: + chr_Reg[6] = (chr_Reg[6] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[6], CHRArea.Area1800); + break; + case 57346: + chr_Reg[7] = (chr_Reg[7] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[7], CHRArea.Area1C00); + break; + case 57347: + chr_Reg[7] = (chr_Reg[7] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[7], CHRArea.Area1C00); + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + for (int i = 0; i < chr_Reg.Length; i++) + { + stream.Write(chr_Reg[i]); + } + stream.Write(security); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + for (int i = 0; i < chr_Reg.Length; i++) + { + chr_Reg[i] = stream.ReadInt32(); + } + security = stream.ReadByte(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs.meta new file mode 100644 index 00000000..11e063f9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper023.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ad20b02baf8d45749a5f2ffffcba13f5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs new file mode 100644 index 00000000..29315b7c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs @@ -0,0 +1,245 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC6", 24)] + [WithExternalSound] + internal class Mapper024 : Board + { + private int irq_reload; + + private int irq_counter; + + private int prescaler; + + private bool irq_mode_cycle; + + private bool irq_enable; + + private bool irq_enable_on_ak; + + private VRC6Pulse snd_1; + + private VRC6Pulse snd_2; + + private VRC6Sawtooth snd_3; + + private double[] audio_pulse_table; + + private double[] audio_tnd_table; + + internal override void Initialize(IRom rom) + { + base.Initialize(rom); + snd_1 = new VRC6Pulse(); + snd_2 = new VRC6Pulse(); + snd_3 = new VRC6Sawtooth(); + audio_pulse_table = new double[32]; + for (int i = 0; i < 32; i++) + { + audio_pulse_table[i] = 95.52 / (8128.0 / (double)i + 100.0); + } + audio_tnd_table = new double[204]; + for (int j = 0; j < 204; j++) + { + audio_tnd_table[j] = 163.67 / (24329.0 / (double)j + 100.0); + } + } + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + APUApplyChannelsSettings(); + snd_1.HardReset(); + snd_2.HardReset(); + snd_3.HardReset(); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + case 32769: + case 32770: + case 32771: + Switch16KPRG(data, PRGArea.Area8000); + break; + case 36864: + snd_1.Write0(ref data); + break; + case 36865: + snd_1.Write1(ref data); + break; + case 36866: + snd_1.Write2(ref data); + break; + case 40960: + snd_2.Write0(ref data); + break; + case 40961: + snd_2.Write1(ref data); + break; + case 40962: + snd_2.Write2(ref data); + break; + case 45056: + snd_3.Write0(ref data); + break; + case 45057: + snd_3.Write1(ref data); + break; + case 45058: + snd_3.Write2(ref data); + break; + case 45059: + switch ((data & 0xC) >> 2) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 49152: + case 49153: + case 49154: + case 49155: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 53248: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 53249: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 53250: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 53251: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 57344: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 57345: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 57346: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 57347: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 61440: + irq_reload = data; + break; + case 61441: + irq_mode_cycle = (data & 4) == 4; + irq_enable = (data & 2) == 2; + irq_enable_on_ak = (data & 1) == 1; + if (irq_enable) + { + irq_counter = irq_reload; + prescaler = 341; + } + NesEmu.IRQFlags &= -9; + break; + case 61442: + NesEmu.IRQFlags &= -9; + irq_enable = irq_enable_on_ak; + break; + } + } + + internal override void OnCPUClock() + { + if (!irq_enable) + { + return; + } + if (!irq_mode_cycle) + { + if (prescaler > 0) + { + prescaler -= 3; + return; + } + prescaler = 341; + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + else + { + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + } + + internal override void OnAPUClockSingle() + { + base.OnAPUClockSingle(); + snd_1.ClockSingle(); + snd_2.ClockSingle(); + snd_3.ClockSingle(); + } + + internal override void APUApplyChannelsSettings() + { + base.APUApplyChannelsSettings(); + snd_1.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_VRC6_SQ1; + snd_2.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_VRC6_SQ2; + snd_3.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_VRC6_SAW; + } + + internal override double APUGetSample() + { + return audio_pulse_table[snd_1.output + snd_2.output] + audio_tnd_table[snd_3.output]; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_reload); + stream.Write(irq_counter); + stream.Write(prescaler); + stream.Write(irq_mode_cycle); + stream.Write(irq_enable); + stream.Write(irq_enable_on_ak); + snd_1.SaveState(ref stream); + snd_2.SaveState(ref stream); + snd_3.SaveState(ref stream); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_reload = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + prescaler = stream.ReadInt32(); + irq_mode_cycle = stream.ReadBoolean(); + irq_enable = stream.ReadBoolean(); + irq_enable_on_ak = stream.ReadBoolean(); + snd_1.LoadState(ref stream); + snd_2.LoadState(ref stream); + snd_3.LoadState(ref stream); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs.meta new file mode 100644 index 00000000..5dbecfc6 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper024.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 555beee22dceafd4598f922cfd1aca63 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs new file mode 100644 index 00000000..9583013a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs @@ -0,0 +1,254 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC4", 25)] + internal class Mapper025 : Board + { + private bool prg_mode; + + private byte prg_reg0; + + private int[] chr_Reg; + + private int irq_reload; + + private int irq_counter; + + private int prescaler; + + private bool irq_mode_cycle; + + private bool irq_enable; + + private bool irq_enable_on_ak; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + prescaler = 341; + chr_Reg = new int[8]; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + case 32769: + case 32770: + case 32771: + case 32772: + case 32776: + case 32780: + prg_reg0 = data; + Switch08KPRG(prg_mode ? (PRG_ROM_08KB_Mask - 1) : (prg_reg0 & 0x1F), PRGArea.Area8000); + Switch08KPRG(prg_mode ? (prg_reg0 & 0x1F) : (PRG_ROM_08KB_Mask - 1), PRGArea.AreaC000); + break; + case 36864: + case 36866: + case 36872: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 36865: + case 36867: + case 36868: + case 36876: + prg_mode = (data & 2) == 2; + Switch08KPRG(prg_mode ? (PRG_ROM_08KB_Mask - 1) : (prg_reg0 & 0x1F), PRGArea.Area8000); + Switch08KPRG(prg_mode ? (prg_reg0 & 0x1F) : (PRG_ROM_08KB_Mask - 1), PRGArea.AreaC000); + break; + case 40960: + case 40961: + case 40962: + case 40963: + case 40964: + case 40968: + case 40972: + Switch08KPRG(data & 0x1F, PRGArea.AreaA000); + break; + case 45056: + chr_Reg[0] = (chr_Reg[0] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[0], CHRArea.Area0000); + break; + case 45058: + case 45064: + chr_Reg[0] = (chr_Reg[0] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[0], CHRArea.Area0000); + break; + case 45057: + case 45060: + chr_Reg[1] = (chr_Reg[1] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[1], CHRArea.Area0400); + break; + case 45059: + case 45068: + chr_Reg[1] = (chr_Reg[1] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[1], CHRArea.Area0400); + break; + case 49152: + chr_Reg[2] = (chr_Reg[2] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[2], CHRArea.Area0800); + break; + case 49154: + case 49160: + chr_Reg[2] = (chr_Reg[2] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[2], CHRArea.Area0800); + break; + case 49153: + case 49156: + chr_Reg[3] = (chr_Reg[3] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[3], CHRArea.Area0C00); + break; + case 49155: + case 49164: + chr_Reg[3] = (chr_Reg[3] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[3], CHRArea.Area0C00); + break; + case 53248: + chr_Reg[4] = (chr_Reg[4] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[4], CHRArea.Area1000); + break; + case 53250: + case 53256: + chr_Reg[4] = (chr_Reg[4] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[4], CHRArea.Area1000); + break; + case 53249: + case 53252: + chr_Reg[5] = (chr_Reg[5] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[5], CHRArea.Area1400); + break; + case 53251: + case 53260: + chr_Reg[5] = (chr_Reg[5] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[5], CHRArea.Area1400); + break; + case 57344: + chr_Reg[6] = (chr_Reg[6] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[6], CHRArea.Area1800); + break; + case 57346: + case 57352: + chr_Reg[6] = (chr_Reg[6] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[6], CHRArea.Area1800); + break; + case 57345: + case 57348: + chr_Reg[7] = (chr_Reg[7] & 0xF0) | (data & 0xF); + Switch01KCHR(chr_Reg[7], CHRArea.Area1C00); + break; + case 57347: + case 57356: + chr_Reg[7] = (chr_Reg[7] & 0xF) | ((data & 0xF) << 4); + Switch01KCHR(chr_Reg[7], CHRArea.Area1C00); + break; + case 61440: + irq_reload = (irq_reload & 0xF0) | (data & 0xF); + break; + case 61442: + case 61448: + irq_reload = (irq_reload & 0xF) | ((data & 0xF) << 4); + break; + case 61441: + case 61444: + irq_mode_cycle = (data & 4) == 4; + irq_enable = (data & 2) == 2; + irq_enable_on_ak = (data & 1) == 1; + if (irq_enable) + { + irq_counter = irq_reload; + prescaler = 341; + } + NesEmu.IRQFlags &= -9; + break; + case 61443: + case 61452: + NesEmu.IRQFlags &= -9; + irq_enable = irq_enable_on_ak; + break; + } + } + + internal override void OnCPUClock() + { + if (!irq_enable) + { + return; + } + if (!irq_mode_cycle) + { + if (prescaler > 0) + { + prescaler -= 3; + return; + } + prescaler = 341; + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + else + { + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prg_mode); + stream.Write(prg_reg0); + for (int i = 0; i < chr_Reg.Length; i++) + { + stream.Write(chr_Reg[i]); + } + stream.Write(irq_reload); + stream.Write(irq_counter); + stream.Write(prescaler); + stream.Write(irq_mode_cycle); + stream.Write(irq_enable); + stream.Write(irq_enable_on_ak); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prg_mode = stream.ReadBoolean(); + prg_reg0 = stream.ReadByte(); + for (int i = 0; i < chr_Reg.Length; i++) + { + chr_Reg[i] = stream.ReadInt32(); + } + irq_reload = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + prescaler = stream.ReadInt32(); + irq_mode_cycle = stream.ReadBoolean(); + irq_enable = stream.ReadBoolean(); + irq_enable_on_ak = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs.meta new file mode 100644 index 00000000..35a992b9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper025.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c486d1b95f6d6fb4a922255ec293358e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs new file mode 100644 index 00000000..72c4bbcd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs @@ -0,0 +1,245 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC6", 26)] + [WithExternalSound] + internal class Mapper026 : Board + { + private int irq_reload; + + private int irq_counter; + + private int prescaler; + + private bool irq_mode_cycle; + + private bool irq_enable; + + private bool irq_enable_on_ak; + + private VRC6Pulse snd_1; + + private VRC6Pulse snd_2; + + private VRC6Sawtooth snd_3; + + private double[] audio_pulse_table; + + private double[] audio_tnd_table; + + internal override void Initialize(IRom rom) + { + base.Initialize(rom); + snd_1 = new VRC6Pulse(); + snd_2 = new VRC6Pulse(); + snd_3 = new VRC6Sawtooth(); + audio_pulse_table = new double[32]; + for (int i = 0; i < 32; i++) + { + audio_pulse_table[i] = 95.52 / (8128.0 / (double)i + 100.0); + } + audio_tnd_table = new double[204]; + for (int j = 0; j < 204; j++) + { + audio_tnd_table[j] = 163.67 / (24329.0 / (double)j + 100.0); + } + } + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + APUApplyChannelsSettings(); + snd_1.HardReset(); + snd_2.HardReset(); + snd_3.HardReset(); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + case 32769: + case 32770: + case 32771: + Switch16KPRG(data, PRGArea.Area8000); + break; + case 36864: + snd_1.Write0(ref data); + break; + case 36866: + snd_1.Write1(ref data); + break; + case 36865: + snd_1.Write2(ref data); + break; + case 40960: + snd_2.Write0(ref data); + break; + case 40962: + snd_2.Write1(ref data); + break; + case 40961: + snd_2.Write2(ref data); + break; + case 45056: + snd_3.Write0(ref data); + break; + case 45058: + snd_3.Write1(ref data); + break; + case 45057: + snd_3.Write2(ref data); + break; + case 45059: + switch ((data & 0xC) >> 2) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 49152: + case 49153: + case 49154: + case 49155: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 53248: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 53250: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 53249: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 53251: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 57344: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 57346: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 57345: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 57347: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 61440: + irq_reload = data; + break; + case 61442: + irq_mode_cycle = (data & 4) == 4; + irq_enable = (data & 2) == 2; + irq_enable_on_ak = (data & 1) == 1; + if (irq_enable) + { + irq_counter = irq_reload; + prescaler = 341; + } + NesEmu.IRQFlags &= -9; + break; + case 61441: + NesEmu.IRQFlags &= -9; + irq_enable = irq_enable_on_ak; + break; + } + } + + internal override void OnCPUClock() + { + if (!irq_enable) + { + return; + } + if (!irq_mode_cycle) + { + if (prescaler > 0) + { + prescaler -= 3; + return; + } + prescaler = 341; + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + else + { + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + } + + internal override void OnAPUClockSingle() + { + base.OnAPUClockSingle(); + snd_1.ClockSingle(); + snd_2.ClockSingle(); + snd_3.ClockSingle(); + } + + internal override void APUApplyChannelsSettings() + { + base.APUApplyChannelsSettings(); + snd_1.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_VRC6_SQ1; + snd_2.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_VRC6_SQ2; + snd_3.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_VRC6_SAW; + } + + internal override double APUGetSample() + { + return audio_pulse_table[snd_1.output + snd_2.output] + audio_tnd_table[snd_3.output]; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_reload); + stream.Write(irq_counter); + stream.Write(prescaler); + stream.Write(irq_mode_cycle); + stream.Write(irq_enable); + stream.Write(irq_enable_on_ak); + snd_1.SaveState(ref stream); + snd_2.SaveState(ref stream); + snd_3.SaveState(ref stream); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_reload = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + prescaler = stream.ReadInt32(); + irq_mode_cycle = stream.ReadBoolean(); + irq_enable = stream.ReadBoolean(); + irq_enable_on_ak = stream.ReadBoolean(); + snd_1.LoadState(ref stream); + snd_2.LoadState(ref stream); + snd_3.LoadState(ref stream); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs.meta new file mode 100644 index 00000000..dcf86326 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper026.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: db4baec7ce5fb7143a58e2698f6a5e31 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs new file mode 100644 index 00000000..eb5d7e73 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs @@ -0,0 +1,109 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Irem G-101", 32)] + internal class Mapper032 : Board + { + private bool prg_mode; + + private byte prg_reg0; + + private bool enable_mirroring_switch; + + internal override void HardReset() + { + base.HardReset(); + enable_mirroring_switch = true; + if (SHA1 == "7E4180432726A433C46BA2206D9E13B32761C11E") + { + enable_mirroring_switch = false; + Switch01KNMTFromMirroring(Mirroring.OneScA); + } + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF007) + { + case 32768: + case 32769: + case 32770: + case 32771: + case 32772: + case 32773: + case 32774: + case 32775: + prg_reg0 = data; + Switch08KPRG((!prg_mode) ? prg_reg0 : 0, PRGArea.Area8000); + Switch08KPRG(prg_mode ? prg_reg0 : (PRG_ROM_08KB_Mask - 1), PRGArea.AreaC000); + break; + case 36864: + case 36865: + case 36866: + case 36867: + case 36868: + case 36869: + case 36870: + case 36871: + prg_mode = (data & 2) == 2; + Switch08KPRG((!prg_mode) ? prg_reg0 : 0, PRGArea.Area8000); + Switch08KPRG(prg_mode ? prg_reg0 : (PRG_ROM_08KB_Mask - 1), PRGArea.AreaC000); + if (enable_mirroring_switch) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40960: + case 40961: + case 40962: + case 40963: + case 40964: + case 40965: + case 40966: + case 40967: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 45056: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 45057: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 45058: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 45059: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 45060: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 45061: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 45062: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 45063: + Switch01KCHR(data, CHRArea.Area1C00); + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prg_mode); + stream.Write(prg_reg0); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prg_mode = stream.ReadBoolean(); + prg_reg0 = stream.ReadByte(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs.meta new file mode 100644 index 00000000..9991fdc9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper032.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e87da889f2fa51f49a00fb850557d445 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs new file mode 100644 index 00000000..b1c91fd4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs @@ -0,0 +1,177 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Taito TC0190/TC0350", 33)] + [HassIssues] + internal class Mapper033 : Board + { + private bool MODE; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override string Issues => MNInterfaceLanguage.IssueMapper33; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + MODE = true; + if (IsGameFoundOnDB) + { + foreach (string chip in base.Chips) + { + if (chip.Contains("TC0190")) + { + MODE = false; + ppuA12TogglesOnRaisingEdge = true; + enabled_ppuA12ToggleTimer = true; + break; + } + } + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + mmc3_alt_behavior = false; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (!MODE) + { + switch (address & 0xE003) + { + case 32768: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 32769: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 32770: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 32771: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 40960: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 40961: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 40962: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 40963: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 49152: + irq_reload = (byte)(data ^ 0xFFu); + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 49154: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 49155: + irq_enabled = true; + break; + case 57344: + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.Horz : Mirroring.Vert); + break; + } + } + else + { + switch (address & 0xA003) + { + case 32768: + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.Horz : Mirroring.Vert); + Switch08KPRG(data & 0x3F, PRGArea.Area8000); + break; + case 32769: + Switch08KPRG(data & 0x3F, PRGArea.AreaA000); + break; + case 32770: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 32771: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 40960: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 40961: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 40962: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 40963: + Switch01KCHR(data, CHRArea.Area1C00); + break; + } + } + } + + internal override void OnPPUA12RaisingEdge() + { + if (!MODE) + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs.meta new file mode 100644 index 00000000..2076edf2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper033.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0d361d388950f6c4780d369471cd60d6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs new file mode 100644 index 00000000..646157ed --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs @@ -0,0 +1,50 @@ +namespace MyNes.Core +{ + [BoardInfo("BxROM/NINA-001", 34)] + internal class Mapper034 : Board + { + private bool BxROM; + + private byte writeData; + + internal override void HardReset() + { + base.HardReset(); + BxROM = true; + if (base.BoardType.Contains("NINA")) + { + BxROM = false; + } + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + base.WriteSRM(ref address, ref data); + if (!BxROM) + { + switch (address) + { + case 32765: + Switch32KPRG(data, PRGArea.Area8000); + break; + case 32766: + Switch04KCHR(data, CHRArea.Area0000); + break; + case 32767: + Switch04KCHR(data, CHRArea.Area1000); + break; + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (BxROM) + { + ReadPRG(ref address, out writeData); + writeData &= data; + Switch32KPRG(writeData, PRGArea.Area8000); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs.meta new file mode 100644 index 00000000..f25520fa --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper034.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d2c0d2bbce836014c834aa850db5c74e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs new file mode 100644 index 00000000..31d1ad4e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs @@ -0,0 +1,58 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Caltron 6-in-1", 41)] + internal class Mapper041 : Board + { + private bool enableReg = false; + + private int vromReg = 0; + + internal override void HardReset() + { + base.HardReset(); + vromReg = 0; + enableReg = true; + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (address <= 26623) + { + Switch32KPRG(address & 7, PRGArea.Area8000); + enableReg = (address & 4) == 4; + vromReg = (vromReg & 3) | ((address >> 1) & 0xC); + Switch08KCHR(vromReg); + Switch01KNMTFromMirroring(((address & 0x20) == 32) ? Mirroring.Horz : Mirroring.Vert); + } + else + { + base.WriteSRM(ref address, ref data); + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (enableReg) + { + vromReg = (vromReg & 0xC) | (data & 3); + Switch08KCHR(vromReg); + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(enableReg); + stream.Write(vromReg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + enableReg = stream.ReadBoolean(); + vromReg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs.meta new file mode 100644 index 00000000..a70d39f9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper041.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d7289b55e4c007a4c926f7b572f2469c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs new file mode 100644 index 00000000..bf1f406c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs @@ -0,0 +1,94 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Mario Baby", 42)] + internal class Mapper042 : Board + { + private int SRAM_PRG_Page = 0; + + private bool irqEnable = false; + + private int irqCounter = 0; + + internal override void HardReset() + { + base.HardReset(); + Switch32KPRG(PRG_ROM_32KB_Mask, PRGArea.Area8000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (address == 32768) + { + Switch08KCHR(data); + return; + } + if (address == 61440) + { + SRAM_PRG_Page = data << 13; + return; + } + switch (address & 0xE003) + { + case 57344: + Switch08KPRG(data, PRGArea.Area6000); + break; + case 57345: + if ((data & 8) == 8) + { + Switch01KNMTFromMirroring(Mirroring.Horz); + } + else + { + Switch01KNMTFromMirroring(Mirroring.Vert); + } + break; + case 57346: + irqEnable = (data & 2) == 2; + if (!irqEnable) + { + irqCounter = 0; + } + NesEmu.IRQFlags &= -9; + break; + } + } + + internal override void OnCPUClock() + { + if (!irqEnable) + { + return; + } + int num = irqCounter++; + if ((irqCounter & 0x6000) != (num & 0x6000)) + { + if ((irqCounter & 0x6000) == 24576) + { + NesEmu.IRQFlags |= 8; + } + else + { + NesEmu.IRQFlags &= -9; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(SRAM_PRG_Page); + stream.Write(irqEnable); + stream.Write(irqCounter); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + SRAM_PRG_Page = stream.ReadInt32(); + irqEnable = stream.ReadBoolean(); + irqCounter = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs.meta new file mode 100644 index 00000000..d26c363a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper042.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8d14e7ff8c821fd40a2583903904425c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs new file mode 100644 index 00000000..01a6f5ee --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs @@ -0,0 +1,235 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("7-in-1 MMC3 Port A001h", 44, true, true)] + [HassIssues] + internal class Mapper044 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int block; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + private int prg_and; + + private int prg_or; + + private int chr_and; + + private int chr_or; + + internal override string Issues => MNInterfaceLanguage.IssueMapper44; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_and = 15; + prg_or = 0; + chr_and = 127; + chr_or = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + block = data & 7; + prg_and = ((block > 5) ? 31 : 15); + prg_or = ((block < 5) ? (block << 4) : 96); + chr_and = ((block > 5) ? 255 : 127); + chr_or = ((block < 5) ? (block << 7) : 768); + SetupPRG(); + SetupCHR(); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area0000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area0800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area1000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area1400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area1800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area1C00); + } + else + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area1000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area1800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area0000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area0400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area0800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG((prg_reg[flag_p ? 2 : 0] & prg_and) | prg_or, PRGArea.Area8000); + Switch08KPRG((prg_reg[1] & prg_and) | prg_or, PRGArea.AreaA000); + Switch08KPRG((prg_reg[(!flag_p) ? 2 : 0] & prg_and) | prg_or, PRGArea.AreaC000); + Switch08KPRG((prg_reg[3] & prg_and) | prg_or, PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + stream.Write(block); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(prg_and); + stream.Write(prg_or); + stream.Write(chr_and); + stream.Write(chr_or); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + block = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + prg_and = stream.ReadInt32(); + prg_or = stream.ReadInt32(); + chr_and = stream.ReadInt32(); + chr_or = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs.meta new file mode 100644 index 00000000..a8a9ae77 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper044.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5454ce1b5791e5c4bbf39820d9fd4bac +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs new file mode 100644 index 00000000..dd83b697 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs @@ -0,0 +1,294 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("X-in-1 MMC3 Port 6000hx4", 45, true, true)] + internal class Mapper045 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + private bool locked; + + private int regCounter; + + private int prg_and; + + private int prg_or; + + private int chr_and; + + private int chr_or; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_and = 63; + prg_or = 0; + chr_and = 4095; + chr_or = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + locked = false; + regCounter = 0; + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (locked) + { + base.WriteSRM(ref address, ref data); + return; + } + switch (regCounter) + { + case 0: + chr_or = (chr_or & 0xFF00) | data; + SetupCHR(); + break; + case 1: + prg_or = data; + SetupPRG(); + break; + case 2: + if ((data & 8) == 8) + { + chr_and = (1 << (data & 7) + 1) - 1; + } + else + { + chr_and = ((data <= 0) ? (-1) : 0); + } + chr_or = (chr_or & 0xFF) | ((data & 0xF0) << 4); + SetupCHR(); + break; + case 3: + locked = (data & 0x40) == 64; + prg_and = (data & 0x3F) ^ 0x3F; + SetupPRG(); + break; + } + regCounter++; + if (regCounter > 3) + { + regCounter = 0; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area0000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area0800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area1000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area1400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area1800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area1C00); + } + else + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area1000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area1800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area0000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area0400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area0800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG((prg_reg[flag_p ? 2 : 0] & prg_and) | prg_or, PRGArea.Area8000); + Switch08KPRG((prg_reg[1] & prg_and) | prg_or, PRGArea.AreaA000); + Switch08KPRG((prg_reg[(!flag_p) ? 2 : 0] & prg_and) | prg_or, PRGArea.AreaC000); + Switch08KPRG((prg_reg[3] & prg_and) | prg_or, PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(prg_and); + stream.Write(prg_or); + stream.Write(chr_and); + stream.Write(chr_or); + stream.Write(locked); + stream.Write(regCounter); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + prg_and = stream.ReadInt32(); + prg_or = stream.ReadInt32(); + chr_and = stream.ReadInt32(); + chr_or = stream.ReadInt32(); + locked = stream.ReadBoolean(); + regCounter = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs.meta new file mode 100644 index 00000000..cf9dbfc9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper045.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 06873d727dc7ca6468474a46df099612 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs new file mode 100644 index 00000000..3afdda64 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs @@ -0,0 +1,42 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("15-in-1 Color Dreams", 46)] + internal class Mapper046 : Board + { + private int prg_reg; + + private int chr_reg; + + internal override void WriteSRM(ref ushort address, ref byte data) + { + prg_reg = (prg_reg & 1) | ((data << 1) & 0x1E); + chr_reg = (chr_reg & 7) | ((data >> 1) & 0x78); + Switch08KCHR(chr_reg); + Switch32KPRG(prg_reg, PRGArea.Area8000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + prg_reg = (data & 1) | (prg_reg & 0x1E); + chr_reg = ((data >> 4) & 7) | (chr_reg & 0x78); + Switch08KCHR(chr_reg); + Switch32KPRG(prg_reg, PRGArea.Area8000); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prg_reg); + stream.Write(chr_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prg_reg = stream.ReadInt32(); + chr_reg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs.meta new file mode 100644 index 00000000..47801902 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper046.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ae6badd4ec9ec79468957a52726ef1ca +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs new file mode 100644 index 00000000..e81fbfb7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs @@ -0,0 +1,258 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("2-in-1 MMC3 Port 6000h", 47, true, true)] + internal class Mapper047 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + private int block; + + private int prg_and; + + private int prg_or; + + private int chr_and; + + private int chr_or; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_and = 15; + prg_or = 0; + chr_and = 127; + chr_or = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (PRG_RAM_ENABLED[PRG_AREA_BLK_INDEX[0]] && PRG_RAM_WRITABLE[PRG_AREA_BLK_INDEX[0]]) + { + block = data & 1; + prg_or = block << 4; + chr_or = block << 7; + SetupCHR(); + SetupPRG(); + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area0000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area0800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area1000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area1400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area1800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area1C00); + } + else + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area1000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area1800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area0000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area0400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area0800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG((prg_reg[flag_p ? 2 : 0] & prg_and) | prg_or, PRGArea.Area8000); + Switch08KPRG((prg_reg[1] & prg_and) | prg_or, PRGArea.AreaA000); + Switch08KPRG((prg_reg[(!flag_p) ? 2 : 0] & prg_and) | prg_or, PRGArea.AreaC000); + Switch08KPRG((prg_reg[3] & prg_and) | prg_or, PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + stream.Write(block); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(prg_and); + stream.Write(prg_or); + stream.Write(chr_and); + stream.Write(chr_or); + stream.Write(irq_enabled); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + block = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + prg_and = stream.ReadInt32(); + prg_or = stream.ReadInt32(); + chr_and = stream.ReadInt32(); + chr_or = stream.ReadInt32(); + irq_enabled = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs.meta new file mode 100644 index 00000000..218f3ce6 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper047.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2279e07deeabd9645a738f56d0b1a9d0 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs new file mode 100644 index 00000000..943b319e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs @@ -0,0 +1,172 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Taito TC0190/TC0350", 48, true, true)] + internal class Mapper048 : Board + { + private bool MODE; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + MODE = false; + if (IsGameFoundOnDB) + { + foreach (string chip in base.Chips) + { + if (chip.Contains("TC0350")) + { + MODE = true; + break; + } + } + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + mmc3_alt_behavior = false; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (!MODE) + { + switch (address & 0xE003) + { + case 32768: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 32769: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 32770: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 32771: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 40960: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 40961: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 40962: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 40963: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 49152: + irq_reload = (byte)(data ^ 0xFFu); + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 49154: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 49155: + irq_enabled = true; + break; + case 57344: + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.Horz : Mirroring.Vert); + break; + } + } + else + { + switch (address & 0xA003) + { + case 32768: + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.Horz : Mirroring.Vert); + Switch08KPRG(data & 0x3F, PRGArea.Area8000); + break; + case 32769: + Switch08KPRG(data & 0x3F, PRGArea.AreaA000); + break; + case 32770: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 32771: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 40960: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 40961: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 40962: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 40963: + Switch01KCHR(data, CHRArea.Area1C00); + break; + } + } + } + + internal override void OnPPUA12RaisingEdge() + { + if (!MODE) + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs.meta new file mode 100644 index 00000000..caa1e267 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper048.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 208670702cbdc4541894f8509f031ef3 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs new file mode 100644 index 00000000..caf34f59 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs @@ -0,0 +1,272 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("4-in-1 MMC3 Port 6xxxh", 49, true, true)] + internal class Mapper049 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + private bool prg_32Mode; + + private int prg_32Page; + + private int prg_and; + + private int prg_or; + + private int chr_and; + + private int chr_or; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + prg_32Mode = false; + prg_32Page = 0; + address_8001 = 0; + prg_and = 15; + prg_or = 0; + chr_and = 127; + chr_or = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (PRG_RAM_ENABLED[PRG_AREA_BLK_INDEX[0]] && PRG_RAM_WRITABLE[PRG_AREA_BLK_INDEX[0]]) + { + prg_32Mode = (data & 1) == 1; + prg_or = (data >> 2) & 0x30; + chr_or = (data & 0xC0) << 1; + prg_32Page = (data >> 4) & 3; + SetupCHR(); + SetupPRG(); + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area0000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area0800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area1000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area1400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area1800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area1C00); + } + else + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area1000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area1800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area0000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area0400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area0800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + if (prg_32Mode) + { + Switch08KPRG((prg_reg[flag_p ? 2 : 0] & prg_and) | prg_or, PRGArea.Area8000); + Switch08KPRG((prg_reg[1] & prg_and) | prg_or, PRGArea.AreaA000); + Switch08KPRG((prg_reg[(!flag_p) ? 2 : 0] & prg_and) | prg_or, PRGArea.AreaC000); + Switch08KPRG((prg_reg[3] & prg_and) | prg_or, PRGArea.AreaE000); + } + else + { + Switch32KPRG(prg_32Page, PRGArea.Area8000); + } + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(prg_and); + stream.Write(prg_or); + stream.Write(chr_and); + stream.Write(chr_or); + stream.Write(prg_32Mode); + stream.Write(prg_32Page); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + prg_and = stream.ReadInt32(); + prg_or = stream.ReadInt32(); + chr_and = stream.ReadInt32(); + chr_or = stream.ReadInt32(); + prg_32Mode = stream.ReadBoolean(); + prg_32Page = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs.meta new file mode 100644 index 00000000..54db8d23 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper049.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c1e562cabf4561140a8f52fe37210dd3 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs new file mode 100644 index 00000000..637f7722 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs @@ -0,0 +1,71 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("FDS-Port - Alt. Levels", 50)] + internal class Mapper050 : Board + { + private int prg_page; + + private int irq_counter; + + private bool irq_enabled; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(15, PRGArea.Area6000); + Switch08KPRG(8, PRGArea.Area8000); + Switch08KPRG(9, PRGArea.AreaA000); + Switch08KPRG(11, PRGArea.AreaE000); + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + switch (address & 0x4120) + { + case 16416: + prg_page = (data & 8) | ((data & 1) << 2) | ((data >> 1) & 3); + Switch08KPRG(prg_page, PRGArea.AreaC000); + break; + case 16672: + irq_enabled = (data & 1) == 1; + if (!irq_enabled) + { + irq_counter = 0; + NesEmu.IRQFlags &= -9; + } + break; + } + } + + internal override void OnCPUClock() + { + if (irq_enabled) + { + irq_counter++; + if (irq_counter == 4096) + { + NesEmu.IRQFlags |= 8; + irq_counter = 0; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prg_page); + stream.Write(irq_counter); + stream.Write(irq_enabled); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prg_page = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + irq_enabled = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs.meta new file mode 100644 index 00000000..9b6523c7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper050.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: bb91df250655ebb4f873fc1084576301 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs new file mode 100644 index 00000000..e769f4dc --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs @@ -0,0 +1,79 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("11-in-1", 51)] + internal class Mapper051 : Board + { + private int bank = 0; + + private int mode = 1; + + private int offset; + + internal override void HardReset() + { + base.HardReset(); + bank = 0; + mode = 1; + offset = 0; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE000) + { + case 32768: + case 57344: + bank = data & 0xF; + UpdateBanks(); + break; + case 49152: + bank = data & 0xF; + mode = ((data >> 3) & 2) | (mode & 1); + UpdateBanks(); + break; + } + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + mode = ((data >> 3) & 2) | ((data >> 1) & 1); + UpdateBanks(); + } + + private void UpdateBanks() + { + offset = 0; + if ((mode & 1) == 1) + { + Switch32KPRG(bank, PRGArea.Area8000); + offset = 35; + } + else + { + Switch08KPRG((bank << 1) | (mode >> 1), PRGArea.Area8000); + Switch08KPRG((bank << 1) | 7, PRGArea.Area8000); + offset = 47; + } + Switch08KPRG(offset | (bank << 2), PRGArea.Area6000); + Switch01KNMTFromMirroring((mode == 3) ? Mirroring.Horz : Mirroring.Vert); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(bank); + stream.Write(mode); + stream.Write(offset); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + bank = stream.ReadInt32(); + mode = stream.ReadInt32(); + offset = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs.meta new file mode 100644 index 00000000..2e474c04 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper051.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e399d70fe48a5db4ba1c813a6e7b46fa +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs new file mode 100644 index 00000000..de148672 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs @@ -0,0 +1,273 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("7-in-1 MMC3 Port 6800h with SRAM", 52, true, true)] + internal class Mapper052 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + private int prg_and; + + private int prg_or; + + private int chr_and; + + private int chr_or; + + private bool locked; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + locked = false; + address_8001 = 0; + prg_and = 31; + prg_or = 0; + chr_and = 255; + chr_or = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void SoftReset() + { + HardReset(); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (!locked) + { + if (PRG_RAM_ENABLED[PRG_AREA_BLK_INDEX[0]] && PRG_RAM_WRITABLE[PRG_AREA_BLK_INDEX[0]]) + { + locked = true; + prg_and = ((data << 1) & 0x10) ^ 0x1F; + prg_or = ((data & 6) | ((data >> 3) & data & 1)) << 4; + chr_and = ((data & 0x40) << 1) ^ 0xFF; + chr_or = (((data >> 3) & 4) | ((data >> 1) & 2) | ((data >> 6) & (data >> 4) & 1)) << 7; + SetupCHR(); + SetupPRG(); + } + } + else + { + base.WriteSRM(ref address, ref data); + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area0000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area0800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area1000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area1400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area1800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area1C00); + } + else + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area1000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area1800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area0000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area0400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area0800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG((prg_reg[flag_p ? 2 : 0] & prg_and) | prg_or, PRGArea.Area8000); + Switch08KPRG((prg_reg[1] & prg_and) | prg_or, PRGArea.AreaA000); + Switch08KPRG((prg_reg[(!flag_p) ? 2 : 0] & prg_and) | prg_or, PRGArea.AreaC000); + Switch08KPRG((prg_reg[3] & prg_and) | prg_or, PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(prg_and); + stream.Write(prg_or); + stream.Write(chr_and); + stream.Write(chr_or); + stream.Write(locked); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + prg_and = stream.ReadInt32(); + prg_or = stream.ReadInt32(); + chr_and = stream.ReadInt32(); + chr_or = stream.ReadInt32(); + locked = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs.meta new file mode 100644 index 00000000..7dca79d0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper052.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fe5026b8a1bdd5942876adb7451904ff +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs new file mode 100644 index 00000000..191b40dc --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs @@ -0,0 +1,58 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 53)] + [HassIssues] + internal class Mapper053 : Board + { + private byte[] regs = new byte[2]; + + private bool epromFirst; + + internal override string Issues => MNInterfaceLanguage.IssueMapper53; + + internal override void HardReset() + { + base.HardReset(); + regs = new byte[2]; + epromFirst = true; + Switch08KPRG(0, PRGArea.Area6000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + regs[1] = data; + UpdatePrg(); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + regs[0] = data; + UpdatePrg(); + Switch01KNMTFromMirroring(((data & 0x20) == 32) ? Mirroring.Horz : Mirroring.Vert); + } + + private void UpdatePrg() + { + int num = (regs[0] << 3) & 0x78; + Switch08KPRG(((num << 1) | 0xF) + (epromFirst ? 4 : 0), PRGArea.Area6000); + Switch16KPRG(((regs[0] & 0x10) == 16) ? ((num | (regs[1] & 7)) + (epromFirst ? 2 : 0)) : ((!epromFirst) ? 128 : 0), PRGArea.Area8000); + Switch16KPRG(((regs[0] & 0x10) == 16) ? ((num | 7) + (epromFirst ? 2 : 0)) : (epromFirst ? 1 : 129), PRGArea.AreaC000); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(regs); + stream.Write(epromFirst); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + stream.Read(regs, 0, 2); + epromFirst = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs.meta new file mode 100644 index 00000000..9a603b98 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper053.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fa0f9f5e47b4fd64997db593a5a91357 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs new file mode 100644 index 00000000..559e256b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs @@ -0,0 +1,97 @@ +namespace MyNes.Core +{ + [BoardInfo("Pirate SMB3", 56)] + internal class Mapper056 : Board + { + private int irqCounter = 0; + + private int irqLatch = 0; + + private bool irqEnabled = false; + + private int irqControl = 0; + + private int switchControl = 0; + + internal override string Issues => MNInterfaceLanguage.IssueMapper56; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + irqLatch = 0; + irqCounter = 0; + irqControl = 0; + irqEnabled = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (address < 61440) + { + switch (address & 0xE000) + { + case 32768: + irqLatch = (irqLatch & 0xFFF0) | (data & 0xF); + break; + case 36864: + irqLatch = (irqLatch & 0xFF0F) | ((data & 0xF) << 4); + break; + case 40960: + irqLatch = (irqLatch & 0xF0FF) | ((data & 0xF) << 8); + break; + case 45056: + irqLatch = (irqLatch & 0xFFF) | ((data & 0xF) << 12); + break; + case 49152: + irqControl = data & 5; + irqEnabled = (data & 2) == 2; + if (irqEnabled) + { + irqCounter = irqLatch; + } + NesEmu.IRQFlags &= -9; + break; + case 53248: + irqEnabled = (irqControl & 1) == 1; + NesEmu.IRQFlags &= -9; + break; + case 57344: + switchControl = data; + break; + } + return; + } + int num = (switchControl & 0xF) - 1; + if (num < 3) + { + Switch08KPRG((data & 0xF) | (PRG_AREA_BLK_INDEX[(num >> 13) + 1] & 0x10), (PRGArea)num); + } + switch (address & 0xC00) + { + case 0: + address &= 3; + if (address < 3) + { + Switch08KPRG((data & 0xF) | (PRG_AREA_BLK_INDEX[(num >> 13) + 1] & 0x10), (PRGArea)address); + } + break; + case 2048: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Vert : Mirroring.Horz); + break; + case 3072: + Switch01KCHR(data, (CHRArea)(address & 7u)); + break; + } + } + + internal override void OnCPUClock() + { + if (irqEnabled && irqCounter++ == 65535) + { + irqCounter = irqLatch; + NesEmu.IRQFlags |= 8; + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs.meta new file mode 100644 index 00000000..717ff544 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper056.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 83d2f8d6bd1261249ae318f6c67df658 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs new file mode 100644 index 00000000..26fe2e09 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs @@ -0,0 +1,55 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("6-in-1 (SuperGK)", 57)] + internal class Mapper057 : Board + { + private int chr_aaa; + + private int chr_bbb; + + private int chr_hhh; + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0x8800) + { + case 32768: + chr_aaa = data & 7; + chr_hhh = (data & 0x40) >> 3; + break; + case 34816: + chr_bbb = data & 7; + if ((data & 0x10) == 16) + { + Switch32KPRG((data & 0xE0) >> 6, PRGArea.Area8000); + } + else + { + Switch16KPRG((data & 0xE0) >> 5, PRGArea.Area8000); + Switch16KPRG((data & 0xE0) >> 5, PRGArea.AreaC000); + } + Switch01KNMTFromMirroring(((data & 8) == 8) ? Mirroring.Horz : Mirroring.Vert); + break; + } + Switch08KCHR(chr_hhh | (chr_aaa | chr_bbb)); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(chr_aaa); + stream.Write(chr_bbb); + stream.Write(chr_hhh); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + chr_aaa = stream.ReadInt32(); + chr_bbb = stream.ReadInt32(); + chr_hhh = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs.meta new file mode 100644 index 00000000..6d6fe5e0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper057.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: bf918deb9af239143bd0e1fbcad64911 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs new file mode 100644 index 00000000..1cc600ec --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs @@ -0,0 +1,24 @@ +namespace MyNes.Core +{ + [BoardInfo("68-in-1 (Game Star)", 58)] + [HassIssues] + internal class Mapper058 : Board + { + internal override string Issues => MNInterfaceLanguage.IssueMapper58; + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR((address >> 3) & 7); + if ((address & 0x40) == 0) + { + Switch32KPRG((address & 7) >> 1, PRGArea.Area8000); + } + else + { + Switch16KPRG(address & 7, PRGArea.Area8000); + Switch16KPRG(address & 7, PRGArea.AreaC000); + } + Switch01KNMTFromMirroring(((address & 0x80) == 128) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs.meta new file mode 100644 index 00000000..3fff5118 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper058.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d5f3c51f1ab08764aa3af28f3048bfa5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs new file mode 100644 index 00000000..2ab0e2c2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs @@ -0,0 +1,67 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 60)] + [HassIssues] + internal class Mapper060 : Board + { + private int latch = 0; + + private byte menu = 0; + + internal override string Issues => MNInterfaceLanguage.IssueMapper60; + + internal override void HardReset() + { + base.HardReset(); + latch = 0; + menu = 0; + } + + internal override void SoftReset() + { + base.SoftReset(); + latch = 0; + menu = (byte)((uint)(menu + 1) & 3u); + Switch08KCHR(menu); + Switch16KPRG(menu, PRGArea.Area8000); + Switch16KPRG(menu, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + latch = address & 0x100; + Switch01KNMTFromMirroring(((address & 8) == 8) ? Mirroring.Horz : Mirroring.Vert); + Switch16KPRG((address >> 4) & ~((~address >> 7) & 1), PRGArea.Area8000); + Switch16KPRG((address >> 4) | ((~address >> 7) & 1), PRGArea.AreaC000); + Switch08KCHR(address); + } + + internal override void ReadPRG(ref ushort address, out byte data) + { + if (latch == 0) + { + base.ReadPRG(ref address, out data); + } + else + { + data = menu; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(latch); + stream.Write(menu); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + latch = stream.ReadInt32(); + menu = stream.ReadByte(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs.meta new file mode 100644 index 00000000..14648f02 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper060.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6f5e014a4df93af4db13624605f4e7f5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs new file mode 100644 index 00000000..8d65c61f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs @@ -0,0 +1,20 @@ +namespace MyNes.Core +{ + [BoardInfo("20-in-1", 61)] + internal class Mapper061 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + if ((address & 0x10) == 0) + { + Switch32KPRG(address & 0xF, PRGArea.Area8000); + } + else + { + Switch16KPRG(((address & 0xF) << 1) | ((address & 0x20) >> 5), PRGArea.Area8000); + Switch16KPRG(((address & 0xF) << 1) | ((address & 0x20) >> 5), PRGArea.AreaC000); + } + Switch01KNMTFromMirroring(((address & 0x80) == 128) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs.meta new file mode 100644 index 00000000..990dcd55 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper061.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d371dd307b7ac2044893482d5eb2c28e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs new file mode 100644 index 00000000..caf9ad90 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs @@ -0,0 +1,24 @@ +namespace MyNes.Core +{ + [BoardInfo("Super 700-in-1", 62)] + internal class Mapper062 : Board + { + private int prg_page; + + internal override void WritePRG(ref ushort address, ref byte data) + { + prg_page = ((address & 0x3F00) >> 8) | (address & 0x40); + Switch08KCHR(((address & 0x1F) << 2) | (data & 3)); + if ((address & 0x20) == 32) + { + Switch16KPRG(prg_page, PRGArea.Area8000); + Switch16KPRG(prg_page, PRGArea.AreaC000); + } + else + { + Switch32KPRG(prg_page >> 1, PRGArea.Area8000); + } + Switch01KNMTFromMirroring(((address & 0x80) == 128) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs.meta new file mode 100644 index 00000000..b32919e4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper062.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ea3055fddea86eb47b1bbbc8c0c9c873 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs new file mode 100644 index 00000000..eec8d92e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs @@ -0,0 +1,258 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Tengen RAMBO-1", 64, true, true)] + internal class Mapper064 : Board + { + private bool flag_c; + + private bool flag_p; + + private bool flag_k; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private byte irq_reload; + + private bool irq_mode; + + private bool irq_clear; + + private int irq_prescaler; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = (flag_k = false)); + address_8001 = 0; + prg_reg = new int[3]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = 2; + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + SetupPRG(); + chr_reg = new int[8]; + for (int i = 0; i < 8; i++) + { + chr_reg[i] = i; + } + SetupCHR(); + irq_enabled = false; + irq_counter = 0; + irq_prescaler = 0; + irq_mode = false; + irq_reload = byte.MaxValue; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 0xF; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + flag_k = (data & 0x20) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data; + SetupPRG(); + break; + case 8: + chr_reg[6] = data; + SetupCHR(); + break; + case 9: + chr_reg[7] = data; + SetupCHR(); + break; + case 15: + prg_reg[2] = data; + SetupPRG(); + break; + case 10: + case 11: + case 12: + case 13: + case 14: + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 49152: + irq_reload = data; + break; + case 49153: + irq_mode = (data & 1) == 1; + irq_clear = true; + irq_prescaler = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + if (!flag_k) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + } + else + { + Switch01KCHR(chr_reg[0], CHRArea.Area0000); + Switch01KCHR(chr_reg[6], CHRArea.Area0400); + Switch01KCHR(chr_reg[1], CHRArea.Area0800); + Switch01KCHR(chr_reg[7], CHRArea.Area0C00); + } + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + if (!flag_k) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + } + else + { + Switch01KCHR(chr_reg[0], CHRArea.Area1000); + Switch01KCHR(chr_reg[6], CHRArea.Area1400); + Switch01KCHR(chr_reg[1], CHRArea.Area1800); + Switch01KCHR(chr_reg[7], CHRArea.Area1C00); + } + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[(!flag_p) ? 1u : 0u], PRGArea.AreaA000); + Switch08KPRG(prg_reg[flag_p ? 1 : 2], PRGArea.AreaC000); + } + + internal override void OnPPUA12RaisingEdge() + { + ClockIRQ(); + } + + internal override void OnCPUClock() + { + if (irq_mode) + { + irq_prescaler++; + if (irq_prescaler == 4) + { + irq_prescaler = 0; + ClockIRQ(); + } + } + } + + private void ClockIRQ() + { + if (irq_clear) + { + irq_counter = (byte)(irq_reload + 1); + irq_clear = false; + } + else if (irq_counter == 0) + { + irq_counter = irq_reload; + } + else if (--irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(irq_prescaler); + stream.Write(irq_mode); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + irq_prescaler = stream.ReadInt32(); + irq_mode = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs.meta new file mode 100644 index 00000000..a5b7a769 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper064.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: a696f836221fe3646b3200f3e6bcb87c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs new file mode 100644 index 00000000..7b4f0065 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs @@ -0,0 +1,94 @@ +namespace MyNes.Core +{ + [BoardInfo("Irem H-3001", 65)] + internal class Mapper065 : Board + { + private bool irq_enable; + + private int irq_reload; + + private int irq_counter; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(0, PRGArea.Area8000); + Switch08KPRG(1, PRGArea.AreaA000); + Switch08KPRG(254, PRGArea.AreaC000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 36865: + Switch01KNMTFromMirroring(((data & 0x80) == 128) ? Mirroring.Horz : Mirroring.Vert); + break; + case 36867: + irq_enable = (data & 0x80) == 128; + NesEmu.IRQFlags &= -9; + break; + case 36868: + irq_counter = irq_reload; + NesEmu.IRQFlags &= -9; + break; + case 36869: + irq_reload = (irq_reload & 0xFF) | (data << 8); + break; + case 36870: + irq_reload = (irq_reload & 0xFF00) | data; + break; + case 40960: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 49152: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 45056: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 45057: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 45058: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 45059: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 45060: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 45061: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 45062: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 45063: + Switch01KCHR(data, CHRArea.Area1C00); + break; + } + } + + internal override void OnCPUClock() + { + if (irq_enable) + { + if (irq_counter > 0) + { + irq_counter--; + } + else if (irq_counter == 0) + { + irq_counter = -1; + NesEmu.IRQFlags |= 8; + } + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs.meta new file mode 100644 index 00000000..767b2804 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper065.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 314d0c6b3f522d84fb090126194b88f9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs new file mode 100644 index 00000000..40784a70 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("GxROM", 66)] + internal class Mapper066 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch32KPRG((data >> 4) & 3, PRGArea.Area8000); + Switch08KCHR(data & 3); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs.meta new file mode 100644 index 00000000..8698e8de --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper066.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1fc42abf998812f40b3ab620cb1cae65 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs new file mode 100644 index 00000000..bebc0005 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs @@ -0,0 +1,108 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Sunsoft 3", 67)] + internal class Mapper067 : Board + { + private bool irq_enabled; + + private int irq_counter; + + private bool odd; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + irq_enabled = false; + irq_counter = 65535; + odd = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF800) + { + case 34816: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 38912: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 43008: + Switch02KCHR(data, CHRArea.Area1000); + break; + case 47104: + Switch02KCHR(data, CHRArea.Area1800); + break; + case 51200: + if (!odd) + { + irq_counter = (irq_counter & 0xFF) | (data << 8); + } + else + { + irq_counter = (irq_counter & 0xFF00) | data; + } + odd = !odd; + break; + case 55296: + irq_enabled = (data & 0x10) == 16; + odd = false; + NesEmu.IRQFlags &= -9; + break; + case 59392: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 63488: + Switch16KPRG(data, PRGArea.Area8000); + break; + } + } + + internal override void OnCPUClock() + { + if (irq_enabled) + { + irq_counter--; + if (irq_counter == 0) + { + irq_counter = 65535; + NesEmu.IRQFlags |= 8; + irq_enabled = false; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(odd); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadInt32(); + odd = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs.meta new file mode 100644 index 00000000..68a00d4b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper067.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6ce6078bf7001f448b966f6e9d3a9ff8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs new file mode 100644 index 00000000..7aed440b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs @@ -0,0 +1,112 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Sunsoft 4", 68)] + internal class Mapper068 : Board + { + private bool flag_r; + + private bool flag_m; + + private int nt_reg0; + + private int nt_reg1; + + private int temp; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF000) + { + case 32768: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 36864: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 40960: + Switch02KCHR(data, CHRArea.Area1000); + break; + case 45056: + Switch02KCHR(data, CHRArea.Area1800); + break; + case 49152: + nt_reg0 = (data & 0x7F) | 0x80; + break; + case 53248: + nt_reg1 = (data & 0x7F) | 0x80; + break; + case 57344: + flag_r = (data & 0x10) == 16; + flag_m = (data & 1) == 1; + Switch01KNMTFromMirroring(flag_m ? Mirroring.Horz : Mirroring.Vert); + break; + case 61440: + Switch16KPRG(data, PRGArea.Area8000); + break; + } + } + + internal override void ReadNMT(ref ushort address, out byte data) + { + if (!flag_r) + { + data = NMT_RAM[NMT_AREA_BLK_INDEX[(address >> 10) & 3]][address & 0x3FF]; + return; + } + switch ((address >> 10) & 3) + { + case 0: + data = CHR_ROM[nt_reg0][address & 0x3FF]; + break; + case 1: + data = CHR_ROM[flag_m ? nt_reg0 : nt_reg1][address & 0x3FF]; + break; + case 2: + data = CHR_ROM[flag_m ? nt_reg1 : nt_reg0][address & 0x3FF]; + break; + case 3: + data = CHR_ROM[nt_reg1][address & 0x3FF]; + break; + default: + data = 0; + break; + } + } + + internal override void WriteNMT(ref ushort address, ref byte data) + { + if (!flag_r) + { + base.WriteNMT(ref address, ref data); + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_r); + stream.Write(flag_m); + stream.Write(nt_reg0); + stream.Write(nt_reg1); + stream.Write(temp); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_r = stream.ReadBoolean(); + flag_m = stream.ReadBoolean(); + nt_reg0 = stream.ReadInt32(); + nt_reg1 = stream.ReadInt32(); + temp = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs.meta new file mode 100644 index 00000000..d0ac7f7a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper068.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 22f5878561b23c24badfb5a921bd895e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs new file mode 100644 index 00000000..521b451d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs @@ -0,0 +1,259 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("FME-7/Sunsoft 5B", 69)] + [WithExternalSound] + internal class Mapper069 : Board + { + private int address_A000; + + private int address_E000; + + private int irq_counter; + + private bool irq_count_enabled; + + private bool irq_trigger_enabled; + + private Sunsoft5BChnl snd_1; + + private Sunsoft5BChnl snd_2; + + private Sunsoft5BChnl snd_3; + + private double[] audio_pulse_table; + + private double[] audio_tnd_table; + + internal override void Initialize(IRom rom) + { + base.Initialize(rom); + snd_1 = new Sunsoft5BChnl(); + snd_2 = new Sunsoft5BChnl(); + snd_3 = new Sunsoft5BChnl(); + audio_pulse_table = new double[32]; + for (int i = 0; i < 32; i++) + { + audio_pulse_table[i] = 95.52 / (8128.0 / (double)i + 100.0); + } + audio_tnd_table = new double[204]; + for (int j = 0; j < 204; j++) + { + audio_tnd_table[j] = 163.67 / (24329.0 / (double)j + 100.0); + } + } + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + address_A000 = 0; + irq_counter = 65535; + irq_count_enabled = false; + irq_trigger_enabled = false; + APUApplyChannelsSettings(); + snd_1.HardReset(); + snd_2.HardReset(); + snd_3.HardReset(); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE000) + { + case 32768: + address_A000 = data & 0xF; + break; + case 40960: + switch (address_A000) + { + case 0: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 1: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 2: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 3: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 4: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 5: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 6: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 7: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 8: + TogglePRGRAMEnable((data & 0x80) == 128); + if ((data & 0x40u) != 0) + { + Toggle08KPRG_RAM(ram: true, PRGArea.Area6000); + Switch08KPRG(data & 0x3F & PRG_RAM_08KB_Mask, PRGArea.Area6000); + } + else + { + Toggle08KPRG_RAM(ram: false, PRGArea.Area6000); + Switch08KPRG(data & 0x3F, PRGArea.Area6000); + } + break; + case 9: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 10: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 11: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 12: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 13: + irq_count_enabled = (data & 0x80) == 128; + irq_trigger_enabled = (data & 1) == 1; + if (!irq_trigger_enabled) + { + NesEmu.IRQFlags &= -9; + } + break; + case 14: + irq_counter = (irq_counter & 0xFF00) | data; + break; + case 15: + irq_counter = (irq_counter & 0xFF) | (data << 8); + break; + } + break; + case 49152: + address_E000 = data & 0xF; + break; + case 57344: + switch (address_E000) + { + case 0: + snd_1.Write0(ref data); + break; + case 1: + snd_1.Write1(ref data); + break; + case 2: + snd_2.Write0(ref data); + break; + case 3: + snd_2.Write1(ref data); + break; + case 4: + snd_3.Write0(ref data); + break; + case 5: + snd_3.Write1(ref data); + break; + case 7: + snd_1.Enabled = (data & 1) == 0; + snd_2.Enabled = (data & 2) == 0; + snd_3.Enabled = (data & 4) == 0; + break; + case 8: + snd_1.Volume = (byte)(data & 0xFu); + break; + case 9: + snd_2.Volume = (byte)(data & 0xFu); + break; + case 10: + snd_3.Volume = (byte)(data & 0xFu); + break; + case 6: + break; + } + break; + } + } + + internal override void OnCPUClock() + { + if (!irq_count_enabled) + { + return; + } + irq_counter--; + if (irq_counter <= 0) + { + irq_counter = 65535; + if (irq_trigger_enabled) + { + NesEmu.IRQFlags |= 8; + } + } + } + + internal override double APUGetSample() + { + return audio_pulse_table[snd_1.output + snd_2.output] + audio_tnd_table[snd_3.output]; + } + + internal override void OnAPUClockSingle() + { + base.OnAPUClockSingle(); + snd_1.ClockSingle(); + snd_2.ClockSingle(); + snd_3.ClockSingle(); + } + + internal override void APUApplyChannelsSettings() + { + base.APUApplyChannelsSettings(); + snd_1.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_SUN1; + snd_2.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_SUN2; + snd_3.Outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_SUN3; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(address_A000); + stream.Write(address_E000); + stream.Write(irq_counter); + stream.Write(irq_count_enabled); + stream.Write(irq_trigger_enabled); + snd_1.SaveState(ref stream); + snd_2.SaveState(ref stream); + snd_3.SaveState(ref stream); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + address_A000 = stream.ReadInt32(); + address_E000 = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + irq_count_enabled = stream.ReadBoolean(); + irq_trigger_enabled = stream.ReadBoolean(); + snd_1.LoadState(ref stream); + snd_2.LoadState(ref stream); + snd_3.LoadState(ref stream); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs.meta new file mode 100644 index 00000000..5ea67234 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper069.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3aa95ff6ff61dbe4da50f18baa928fe4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs new file mode 100644 index 00000000..41cb3df3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs @@ -0,0 +1,18 @@ +namespace MyNes.Core +{ + [BoardInfo("Bandai", 70)] + internal class Mapper070 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch16KPRG((data >> 4) & 0xF, PRGArea.Area8000); + Switch08KCHR(data & 0xF); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs.meta new file mode 100644 index 00000000..a734c670 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper070.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0e3aaa8620d44434b941fe8fc0580ca1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs new file mode 100644 index 00000000..2427cc19 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs @@ -0,0 +1,30 @@ +namespace MyNes.Core +{ + [BoardInfo("Camerica", 71)] + internal class Mapper071 : Board + { + private bool fireHawk; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + fireHawk = SHA1.ToUpper() == "334781C830F135CF30A33E392D8AAA4AFDC223F9"; + } + + internal override void WritePRG(ref ushort addr, ref byte val) + { + if (addr < 40960) + { + if (fireHawk) + { + Switch01KNMTFromMirroring(((val & 0x10) == 16) ? Mirroring.OneScB : Mirroring.OneScA); + } + } + else if (addr >= 49152) + { + Switch16KPRG(val, PRGArea.Area8000); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs.meta new file mode 100644 index 00000000..09beb83d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper071.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5c123475e1ad66a4f839b1768a5a0d07 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs new file mode 100644 index 00000000..4870d6cd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs @@ -0,0 +1,53 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Jaleco Early Mapper 0", 72)] + internal class Mapper072 : Board + { + private byte writeData; + + private int chr_reg; + + private int prg_reg; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + writeData = 0; + chr_reg = (prg_reg = 0); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch ((data >> 6) & 3) + { + case 0: + Switch08KCHR(chr_reg); + Switch16KPRG(prg_reg, PRGArea.Area8000); + break; + case 1: + chr_reg = data & 0xF; + break; + case 2: + prg_reg = data & 0xF; + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(chr_reg); + stream.Write(prg_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + chr_reg = stream.ReadInt32(); + prg_reg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs.meta new file mode 100644 index 00000000..ab97a4e9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper072.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2523f31241f5556429834dbf0281aa69 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs new file mode 100644 index 00000000..3d004c6e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs @@ -0,0 +1,111 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC3", 73)] + internal class Mapper073 : Board + { + private bool irq_mode_8; + + private bool irq_enable; + + private bool irq_enable_on_ak; + + private int irq_reload; + + private int irq_counter; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + irq_mode_8 = false; + irq_enable = false; + irq_enable_on_ak = false; + irq_reload = 0; + irq_counter = 0; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF000) + { + case 32768: + irq_reload = (irq_reload & 0xFFF0) | (data & 0xF); + break; + case 36864: + irq_reload = (irq_reload & 0xFF0F) | ((data & 0xF) << 4); + break; + case 40960: + irq_reload = (irq_reload & 0xF0FF) | ((data & 0xF) << 8); + break; + case 45056: + irq_reload = (irq_reload & 0xFFF) | ((data & 0xF) << 12); + break; + case 49152: + irq_mode_8 = (data & 4) == 4; + irq_enable = (data & 2) == 2; + irq_enable_on_ak = (data & 1) == 1; + if (irq_enable) + { + irq_counter = irq_reload; + } + NesEmu.IRQFlags &= -9; + break; + case 53248: + irq_enable = irq_enable_on_ak; + NesEmu.IRQFlags &= -9; + break; + case 61440: + Switch16KPRG(data & 0xF, PRGArea.Area8000); + break; + } + } + + internal override void OnCPUClock() + { + if (!irq_enable) + { + return; + } + if (irq_mode_8) + { + irq_counter = (irq_counter & 0xFF00) | (byte)((irq_counter & 0xFF) + 1); + if ((byte)(irq_counter & 0xFF) == byte.MaxValue) + { + NesEmu.IRQFlags |= 8; + irq_counter = (irq_counter & 0xFF00) | (irq_reload & 0xFF); + } + } + else + { + irq_counter++; + if (irq_counter == 65535) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_mode_8); + stream.Write(irq_enable); + stream.Write(irq_enable_on_ak); + stream.Write(irq_reload); + stream.Write(irq_counter); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_mode_8 = stream.ReadBoolean(); + irq_enable = stream.ReadBoolean(); + irq_enable_on_ak = stream.ReadBoolean(); + irq_reload = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs.meta new file mode 100644 index 00000000..6a0a43c4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper073.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 07ac7400d8c8fc74991b21d6733ba3f4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs new file mode 100644 index 00000000..8795f90b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs @@ -0,0 +1,234 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Pirate MMC3 variant", 74, true, true)] + internal class Mapper074 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Toggle02KCHR_RAM(chr_reg[0] != 8 && chr_reg[0] != 9, CHRArea.Area0000); + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Toggle02KCHR_RAM(chr_reg[1] != 8 && chr_reg[1] != 9, CHRArea.Area0800); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Toggle02KCHR_RAM(chr_reg[2] != 8 && chr_reg[2] != 9, CHRArea.Area1000); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Toggle02KCHR_RAM(chr_reg[3] != 8 && chr_reg[3] != 9, CHRArea.Area1400); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Toggle02KCHR_RAM(chr_reg[4] != 8 && chr_reg[4] != 9, CHRArea.Area1800); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Toggle02KCHR_RAM(chr_reg[5] != 8 && chr_reg[5] != 9, CHRArea.Area1C00); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Toggle02KCHR_RAM(chr_reg[0] != 8 && chr_reg[0] != 9, CHRArea.Area1000); + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Toggle02KCHR_RAM(chr_reg[1] != 8 && chr_reg[1] != 9, CHRArea.Area1800); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Toggle02KCHR_RAM(chr_reg[2] != 8 && chr_reg[2] != 9, CHRArea.Area0000); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Toggle02KCHR_RAM(chr_reg[3] != 8 && chr_reg[3] != 9, CHRArea.Area0400); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Toggle02KCHR_RAM(chr_reg[4] != 8 && chr_reg[4] != 9, CHRArea.Area0800); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Toggle02KCHR_RAM(chr_reg[5] != 8 && chr_reg[5] != 9, CHRArea.Area0C00); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs.meta new file mode 100644 index 00000000..1ee12b94 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper074.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: de1bb7ed83cdb22499c854992fae8e27 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs new file mode 100644 index 00000000..6604ceb1 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs @@ -0,0 +1,63 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC1", 75)] + internal class Mapper075 : Board + { + private int chr0_reg; + + private int chr1_reg; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF000) + { + case 32768: + Switch08KPRG(data & 0xF, PRGArea.Area8000); + break; + case 40960: + Switch08KPRG(data & 0xF, PRGArea.AreaA000); + break; + case 49152: + Switch08KPRG(data & 0xF, PRGArea.AreaC000); + break; + case 36864: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + chr0_reg = (chr0_reg & 0xF) | ((data & 2) << 3); + Switch04KCHR(chr0_reg, CHRArea.Area0000); + chr1_reg = (chr1_reg & 0xF) | ((data & 4) << 2); + Switch04KCHR(chr1_reg, CHRArea.Area1000); + break; + case 57344: + chr0_reg = (chr0_reg & 0x10) | (data & 0xF); + Switch04KCHR(chr0_reg, CHRArea.Area0000); + break; + case 61440: + chr1_reg = (chr1_reg & 0x10) | (data & 0xF); + Switch04KCHR(chr1_reg, CHRArea.Area1000); + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(chr0_reg); + stream.Write(chr1_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + chr0_reg = stream.ReadInt32(); + chr1_reg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs.meta new file mode 100644 index 00000000..128672ea --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper075.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 96976ae7823c77e4d99355570ab4ff15 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs new file mode 100644 index 00000000..690dfffa --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs @@ -0,0 +1,77 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Namco 109", 76)] + internal class Mapper076 : Board + { + private int address_8001; + + private bool prg_a; + + private byte prg_reg; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask - 1, PRGArea.AreaC000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + address_8001 = 0; + prg_a = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + prg_a = (data & 0x40) == 64; + Switch08KPRG(prg_reg, prg_a ? PRGArea.AreaC000 : PRGArea.Area8000); + break; + case 32769: + switch (address_8001) + { + case 2: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 3: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 4: + Switch02KCHR(data, CHRArea.Area1000); + break; + case 5: + Switch02KCHR(data, CHRArea.Area1800); + break; + case 6: + Switch08KPRG(prg_reg = data, prg_a ? PRGArea.AreaC000 : PRGArea.Area8000); + break; + case 7: + Switch08KPRG(data, PRGArea.AreaA000); + break; + } + break; + case 40960: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(address_8001); + stream.Write(prg_a); + stream.Write(prg_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + address_8001 = stream.ReadInt32(); + prg_a = stream.ReadBoolean(); + prg_reg = stream.ReadByte(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs.meta new file mode 100644 index 00000000..74841651 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper076.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: eaed30d1f6adc7f4dbb3092f65b7b614 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs new file mode 100644 index 00000000..7922a49b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs @@ -0,0 +1,23 @@ +namespace MyNes.Core +{ + [BoardInfo("Irem", 77)] + internal class Mapper077 : Board + { + internal override void HardReset() + { + base.HardReset(); + Toggle02KCHR_RAM(ram: true, CHRArea.Area0800); + Switch02KCHR(0, CHRArea.Area0800); + Toggle02KCHR_RAM(ram: true, CHRArea.Area1000); + Switch02KCHR(1, CHRArea.Area1000); + Toggle02KCHR_RAM(ram: true, CHRArea.Area1800); + Switch02KCHR(2, CHRArea.Area1800); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch02KCHR((data >> 4) & 0xF, CHRArea.Area0000); + Switch32KPRG(data & 0xF, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs.meta new file mode 100644 index 00000000..7cecfd27 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper077.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e75ccafc4c3062a48bf1344deff1c242 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs new file mode 100644 index 00000000..4d21f8a9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs @@ -0,0 +1,33 @@ +namespace MyNes.Core +{ + [BoardInfo("Irem 74HC161/32", 78)] + internal class Mapper078 : Board + { + private bool mirroring_mode_single; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + mirroring_mode_single = false; + if (base.BoardType == "JALECO-JF-16") + { + mirroring_mode_single = true; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR((data >> 4) & 0xF); + Switch16KPRG(data & 7, PRGArea.Area8000); + if (mirroring_mode_single) + { + Switch01KNMTFromMirroring(((data & 8) == 8) ? Mirroring.OneScB : Mirroring.OneScA); + } + else + { + Switch01KNMTFromMirroring(((data & 8) == 8) ? Mirroring.Vert : Mirroring.Horz); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs.meta new file mode 100644 index 00000000..f49e71bb --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper078.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 194bdddd89a72a641840f7ef60b36f75 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs new file mode 100644 index 00000000..3fcf8177 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs @@ -0,0 +1,15 @@ +namespace MyNes.Core +{ + [BoardInfo("AVE Nina-3", 79)] + internal class Mapper079 : Board + { + internal override void WriteEX(ref ushort address, ref byte data) + { + if ((address ^ 0x4100) == 0) + { + Switch32KPRG((data >> 3) & 7, PRGArea.Area8000); + Switch08KCHR((data & 7) | ((data >> 3) & 8)); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs.meta new file mode 100644 index 00000000..16e89a3b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper079.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 239bf400a91a2ad42b07c18327d7ab73 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs new file mode 100644 index 00000000..d161be24 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs @@ -0,0 +1,56 @@ +namespace MyNes.Core +{ + [BoardInfo("Taito X-005", 80)] + internal class Mapper080 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + switch (address) + { + case 32496: + Switch02KCHR(data >> 1, CHRArea.Area0000); + break; + case 32497: + Switch02KCHR(data >> 1, CHRArea.Area0800); + break; + case 32498: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 32499: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 32500: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 32501: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 32502: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Vert : Mirroring.Horz); + break; + case 32506: + case 32507: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 32508: + case 32509: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 32510: + case 32511: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 32503: + case 32504: + case 32505: + break; + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs.meta new file mode 100644 index 00000000..e9ca94b8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper080.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 80c33ae23427747408bafc4c5a86721a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs new file mode 100644 index 00000000..22500ff2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs @@ -0,0 +1,70 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Taito X1-17 ", 82)] + internal class Mapper082 : Board + { + private bool chr_mode; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + switch (address) + { + case 32496: + Switch02KCHR(data >> 1, chr_mode ? CHRArea.Area1000 : CHRArea.Area0000); + break; + case 32497: + Switch02KCHR(data >> 1, chr_mode ? CHRArea.Area1800 : CHRArea.Area0800); + break; + case 32498: + Switch01KCHR(data, (!chr_mode) ? CHRArea.Area1000 : CHRArea.Area0000); + break; + case 32499: + Switch01KCHR(data, chr_mode ? CHRArea.Area0400 : CHRArea.Area1400); + break; + case 32500: + Switch01KCHR(data, chr_mode ? CHRArea.Area0800 : CHRArea.Area1800); + break; + case 32501: + Switch01KCHR(data, chr_mode ? CHRArea.Area0C00 : CHRArea.Area1C00); + break; + case 32502: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Vert : Mirroring.Horz); + chr_mode = (data & 2) == 2; + break; + case 32506: + Switch08KPRG(data >> 2, PRGArea.Area8000); + break; + case 32507: + Switch08KPRG(data >> 2, PRGArea.AreaA000); + break; + case 32508: + Switch08KPRG(data >> 2, PRGArea.AreaC000); + break; + case 32503: + case 32504: + case 32505: + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(chr_mode); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + chr_mode = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs.meta new file mode 100644 index 00000000..f5c8b1a0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper082.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 62b681ec1c812a64e8ebb96cc8c47d01 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs new file mode 100644 index 00000000..e6e682d4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs @@ -0,0 +1,171 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("VRC7", 85)] + [HassIssues] + internal class Mapper085 : Board + { + private int irq_reload; + + private int irq_counter; + + private int prescaler; + + private bool irq_mode_cycle; + + private bool irq_enable; + + private bool irq_enable_on_ak; + + internal override string Issues => MNInterfaceLanguage.IssueMapper85; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + irq_reload = 0; + prescaler = 341; + irq_counter = 0; + irq_mode_cycle = false; + irq_enable = false; + irq_enable_on_ak = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address) + { + case 32768: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 32776: + case 32784: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 36864: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 40960: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 40968: + case 40976: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 45056: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 45064: + case 45072: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 49152: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 49160: + case 49168: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 53248: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 53256: + case 53264: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 57344: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 57352: + case 57360: + irq_reload = data; + break; + case 61440: + irq_mode_cycle = (data & 4) == 4; + irq_enable = (data & 2) == 2; + irq_enable_on_ak = (data & 1) == 1; + if (irq_enable) + { + irq_counter = irq_reload; + prescaler = 341; + } + NesEmu.IRQFlags &= -9; + break; + case 61448: + case 61456: + NesEmu.IRQFlags &= -9; + irq_enable = irq_enable_on_ak; + break; + } + } + + internal override void OnCPUClock() + { + if (!irq_enable) + { + return; + } + if (!irq_mode_cycle) + { + if (prescaler > 0) + { + prescaler -= 3; + return; + } + prescaler = 341; + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + else + { + irq_counter++; + if (irq_counter == 255) + { + NesEmu.IRQFlags |= 8; + irq_counter = irq_reload; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prescaler); + stream.Write(irq_counter); + stream.Write(irq_mode_cycle); + stream.Write(irq_reload); + stream.Write(irq_enable); + stream.Write(irq_enable_on_ak); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prescaler = stream.ReadInt32(); + irq_counter = stream.ReadInt32(); + irq_mode_cycle = stream.ReadBoolean(); + irq_reload = stream.ReadInt32(); + irq_enable = stream.ReadBoolean(); + irq_enable_on_ak = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs.meta new file mode 100644 index 00000000..032fd665 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper085.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6afe7d8088d58b1418070fa2e6f42cff +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs new file mode 100644 index 00000000..410ea5fc --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs @@ -0,0 +1,15 @@ +namespace MyNes.Core +{ + [BoardInfo("Jaleco Early Mapper 2", 86)] + internal class Mapper086 : Board + { + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (address < 28672) + { + Switch32KPRG((data >> 4) & 3, PRGArea.Area8000); + Switch08KCHR((data & 7) | ((data >> 4) & 4)); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs.meta new file mode 100644 index 00000000..34a3cb19 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper086.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 57129c0244d9b7246a087748b41fdc5b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs new file mode 100644 index 00000000..7764a7db --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs @@ -0,0 +1,11 @@ +namespace MyNes.Core +{ + [BoardInfo("Jaleco/Konami", 87)] + internal class Mapper087 : Board + { + internal override void WriteSRM(ref ushort address, ref byte data) + { + Switch08KCHR(((data & 2) >> 1) | ((data & 1) << 1)); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs.meta new file mode 100644 index 00000000..7b9cd585 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper087.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6340ee74cbb8f2d46a4bc6954246758a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs new file mode 100644 index 00000000..31cdc9b0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs @@ -0,0 +1,67 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Namco 118", 88)] + internal class Mapper088 : Board + { + private int address_8001; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0x8001) + { + case 32768: + address_8001 = data & 7; + break; + case 32769: + switch (address_8001) + { + case 0: + Switch02KCHR((data & 0x3F) >> 1, CHRArea.Area0000); + break; + case 1: + Switch02KCHR((data & 0x3F) >> 1, CHRArea.Area0800); + break; + case 2: + Switch01KCHR(data | 0x40, CHRArea.Area1000); + break; + case 3: + Switch01KCHR(data | 0x40, CHRArea.Area1400); + break; + case 4: + Switch01KCHR(data | 0x40, CHRArea.Area1800); + break; + case 5: + Switch01KCHR(data | 0x40, CHRArea.Area1C00); + break; + case 6: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 7: + Switch08KPRG(data, PRGArea.AreaA000); + break; + } + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(address_8001); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + address_8001 = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs.meta new file mode 100644 index 00000000..d9e387c4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper088.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 12caa8670cf3a6d468dd168c39fef9b6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs new file mode 100644 index 00000000..17eae889 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs @@ -0,0 +1,19 @@ +namespace MyNes.Core +{ + [BoardInfo("Sunsoft Early", 89)] + internal class Mapper089 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR((data & 7) | ((data >> 4) & 8)); + Switch16KPRG((data >> 4) & 7, PRGArea.Area8000); + Switch01KNMTFromMirroring(((data & 8) == 8) ? Mirroring.OneScB : Mirroring.OneScA); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs.meta new file mode 100644 index 00000000..daf4860f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper089.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 84e15b9d69ed162448c7748a15d700ba +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs new file mode 100644 index 00000000..dbc7371d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs @@ -0,0 +1,634 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Pirate MMC5-style", 90)] + [HassIssues] + internal class Mapper090 : Board + { + protected bool MAPPER90MODE; + + private int[] prg_reg; + + private int[] chr_reg; + + private int[] nt_reg; + + private int prg_mode; + + private int chr_mode; + + private bool chr_block_mode; + + private int chr_block; + + private bool chr_m; + + private bool flag_s; + + private int irqCounter = 0; + + private bool IrqEnable = false; + + private bool irqCountDownMode = false; + + private bool irqCountUpMode = false; + + private bool irqFunkyMode = false; + + private bool irqPrescalerSize = false; + + private int irqSource = 0; + + private int irqPrescaler = 0; + + private int irqPrescalerXOR = 0; + + private byte irqFunkyModeReg = 0; + + private byte Dipswitch; + + private byte multiplication_a; + + private byte multiplication_b; + + private ushort multiplication; + + private byte RAM5803; + + private bool nt_advanced_enable; + + private bool nt_rom_only; + + private int nt_ram_select; + + internal override string Issues => MNInterfaceLanguage.IssueMapper90; + + internal override void HardReset() + { + base.HardReset(); + MAPPER90MODE = true; + prg_reg = new int[4]; + chr_reg = new int[8]; + nt_reg = new int[4]; + prg_mode = (chr_mode = 0); + for (int i = 0; i < 4; i++) + { + prg_reg[i] = i; + nt_reg[i] = i; + } + for (int j = 0; j < 8; j++) + { + chr_reg[j] = j; + } + SetupPRG(); + SetupCHR(); + Dipswitch = 0; + irqCounter = 0; + IrqEnable = false; + irqCountDownMode = false; + irqCountUpMode = false; + irqFunkyMode = false; + irqPrescalerSize = false; + irqSource = 0; + irqPrescaler = 0; + irqPrescalerXOR = 0; + irqFunkyModeReg = 0; + RAM5803 = 0; + flag_s = false; + multiplication_a = 0; + multiplication_b = 0; + multiplication = 0; + } + + internal override void SoftReset() + { + base.SoftReset(); + if (Dipswitch == 0) + { + Dipswitch = byte.MaxValue; + } + else + { + Dipswitch = 0; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF007) + { + case 32768: + case 32769: + case 32770: + case 32771: + case 32772: + case 32773: + case 32774: + case 32775: + prg_reg[address & 3] = data & 0x7F; + SetupPRG(); + break; + case 36864: + case 36865: + case 36866: + case 36867: + case 36868: + case 36869: + case 36870: + case 36871: + chr_reg[address & 7] = (chr_reg[address & 7] & 0xFF00) | data; + SetupCHR(); + break; + case 40960: + case 40961: + case 40962: + case 40963: + case 40964: + case 40965: + case 40966: + case 40967: + chr_reg[address & 7] = (chr_reg[address & 7] & 0xFF) | (data << 8); + SetupCHR(); + break; + case 45056: + case 45057: + case 45058: + case 45059: + nt_reg[address & 3] = (nt_reg[address & 3] & 0xFF00) | data; + break; + case 45060: + case 45061: + case 45062: + case 45063: + nt_reg[address & 3] = (nt_reg[address & 3] & 0xFF) | (data << 8); + break; + case 49152: + IrqEnable = (data & 1) == 1; + if (!IrqEnable) + { + NesEmu.IRQFlags &= -9; + } + break; + case 49153: + irqCountDownMode = (data & 0x80) == 128; + irqCountUpMode = (data & 0x40) == 64; + irqFunkyMode = (data & 8) == 8; + irqPrescalerSize = (data & 4) == 4; + irqSource = data & 3; + break; + case 49154: + IrqEnable = false; + NesEmu.IRQFlags &= -9; + break; + case 49155: + IrqEnable = true; + break; + case 49156: + irqPrescaler = data ^ irqPrescalerXOR; + break; + case 49157: + irqCounter = data ^ irqPrescalerXOR; + break; + case 49158: + irqPrescalerXOR = data; + break; + case 49159: + irqFunkyModeReg = data; + break; + case 53248: + flag_s = (data & 0x80) == 128; + prg_mode = data & 7; + chr_mode = (data >> 3) & 3; + nt_advanced_enable = (data & 0x20) == 32; + nt_rom_only = (data & 0x40) == 64; + SetupPRG(); + SetupCHR(); + break; + case 53249: + switch (data & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 53250: + nt_ram_select = data & 0x80; + break; + case 53251: + chr_m = (data & 0x80) == 128; + chr_block_mode = (data & 0x20) == 32; + chr_block = (data & 0x1F) << 8; + SetupCHR(); + break; + } + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + } + + internal override void ReadSRM(ref ushort address, out byte data) + { + if (flag_s) + { + base.ReadSRM(ref address, out data); + } + else + { + data = 0; + } + } + + internal override void ReadEX(ref ushort address, out byte data) + { + switch (address) + { + case 20480: + data = Dipswitch; + break; + case 22528: + data = (byte)(multiplication & 0xFFu); + break; + case 22529: + data = (byte)((multiplication & 0xFF00) >> 8); + break; + case 22531: + data = RAM5803; + break; + default: + data = 0; + break; + } + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + switch (address) + { + case 22528: + multiplication_a = data; + multiplication = (ushort)(multiplication_a * multiplication_b); + break; + case 22529: + multiplication_b = data; + multiplication = (ushort)(multiplication_a * multiplication_b); + break; + case 22531: + RAM5803 = data; + break; + case 22530: + break; + } + } + + internal override void ReadNMT(ref ushort address, out byte data) + { + if (MAPPER90MODE) + { + data = NMT_RAM[NMT_AREA_BLK_INDEX[(address >> 10) & 3]][address & 0x3FF]; + } + if (!nt_advanced_enable) + { + data = NMT_RAM[NMT_AREA_BLK_INDEX[(address >> 10) & 3]][address & 0x3FF]; + } + else if (nt_rom_only) + { + data = CHR_ROM[nt_reg[(address >> 10) & 3]][address & 0x3FF]; + } + else if ((nt_reg[(address >> 10) & 3] & 0x80) != nt_ram_select) + { + data = CHR_ROM[nt_reg[(address >> 10) & 3]][address & 0x3FF]; + } + else + { + data = NMT_RAM[nt_reg[(address >> 10) & 3] & 1][address & 0x3FF]; + } + } + + internal override void WriteNMT(ref ushort address, ref byte data) + { + if (MAPPER90MODE) + { + NMT_RAM[NMT_AREA_BLK_INDEX[(address >> 10) & 3]][address & 0x3FF] = data; + } + else if (!nt_advanced_enable) + { + NMT_RAM[NMT_AREA_BLK_INDEX[(address >> 10) & 3]][address & 0x3FF] = data; + } + else if (!nt_rom_only && (nt_reg[(address >> 10) & 3] & 0x80) == nt_ram_select) + { + NMT_RAM[nt_reg[(address >> 10) & 3] & 1][address & 0x3FF] = data; + } + } + + private void SetupPRG() + { + switch (prg_mode) + { + case 0: + Switch08KPRG(prg_reg[3] * 4 + 3, PRGArea.Area6000); + Switch32KPRG(PRG_ROM_32KB_Mask, PRGArea.Area8000); + break; + case 1: + Switch08KPRG(prg_reg[3] * 2 + 1, PRGArea.Area6000); + Switch16KPRG(prg_reg[1], PRGArea.Area8000); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + break; + case 2: + Switch08KPRG(prg_reg[3], PRGArea.Area6000); + Switch08KPRG(prg_reg[0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[2], PRGArea.AreaC000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + break; + case 3: + Switch08KPRG(ReverseByte(prg_reg[3]), PRGArea.Area6000); + Switch08KPRG(ReverseByte(prg_reg[0]), PRGArea.Area8000); + Switch08KPRG(ReverseByte(prg_reg[1]), PRGArea.AreaA000); + Switch08KPRG(ReverseByte(prg_reg[2]), PRGArea.AreaC000); + Switch08KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaE000); + break; + case 4: + Switch08KPRG(prg_reg[3] * 4 + 3, PRGArea.Area6000); + Switch32KPRG(prg_reg[3], PRGArea.Area8000); + break; + case 5: + Switch08KPRG(prg_reg[3] * 2 + 1, PRGArea.Area6000); + Switch16KPRG(prg_reg[1], PRGArea.Area8000); + Switch16KPRG(prg_reg[3], PRGArea.AreaC000); + break; + case 6: + Switch08KPRG(prg_reg[3], PRGArea.Area6000); + Switch08KPRG(prg_reg[0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[2], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + break; + case 7: + Switch08KPRG(ReverseByte(prg_reg[3]), PRGArea.Area6000); + Switch08KPRG(ReverseByte(prg_reg[0]), PRGArea.Area8000); + Switch08KPRG(ReverseByte(prg_reg[1]), PRGArea.AreaA000); + Switch08KPRG(ReverseByte(prg_reg[2]), PRGArea.AreaC000); + Switch08KPRG(ReverseByte(prg_reg[3]), PRGArea.AreaE000); + break; + } + } + + private void SetupCHR() + { + switch (chr_mode) + { + case 0: + if (chr_block_mode) + { + Switch08KCHR(chr_reg[0]); + } + else + { + Switch08KCHR((chr_reg[0] & 0xFF) | chr_block); + } + break; + case 1: + if (chr_block_mode) + { + Switch04KCHR(chr_reg[0], CHRArea.Area0000); + Switch04KCHR(chr_reg[4], CHRArea.Area1000); + } + else + { + Switch04KCHR((chr_reg[0] & 0xFF) | chr_block, CHRArea.Area0000); + Switch04KCHR((chr_reg[4] & 0xFF) | chr_block, CHRArea.Area1000); + } + break; + case 2: + if (chr_block_mode) + { + Switch02KCHR(chr_reg[0], CHRArea.Area0000); + Switch02KCHR(chr_m ? chr_reg[0] : chr_reg[2], CHRArea.Area0800); + Switch02KCHR(chr_reg[4], CHRArea.Area1000); + Switch02KCHR(chr_reg[6], CHRArea.Area1800); + } + else + { + Switch02KCHR((chr_reg[0] & 0xFF) | chr_block, CHRArea.Area0000); + Switch02KCHR(((chr_m ? chr_reg[0] : chr_reg[2]) & 0xFF) | chr_block, CHRArea.Area0800); + Switch02KCHR((chr_reg[4] & 0xFF) | chr_block, CHRArea.Area1000); + Switch02KCHR((chr_reg[6] & 0xFF) | chr_block, CHRArea.Area1800); + } + break; + case 3: + if (chr_block_mode) + { + Switch01KCHR(chr_reg[0], CHRArea.Area0000); + Switch01KCHR(chr_reg[1], CHRArea.Area0400); + Switch01KCHR(chr_m ? chr_reg[0] : chr_reg[2], CHRArea.Area0800); + Switch01KCHR(chr_m ? chr_reg[1] : chr_reg[3], CHRArea.Area0C00); + Switch01KCHR(chr_reg[4], CHRArea.Area1000); + Switch01KCHR(chr_reg[5], CHRArea.Area1400); + Switch01KCHR(chr_reg[6], CHRArea.Area1800); + Switch01KCHR(chr_reg[7], CHRArea.Area1C00); + } + else + { + Switch01KCHR((chr_reg[0] & 0xFF) | chr_block, CHRArea.Area0000); + Switch01KCHR((chr_reg[1] & 0xFF) | chr_block, CHRArea.Area0400); + Switch01KCHR(((chr_m ? chr_reg[0] : chr_reg[2]) & 0xFF) | chr_block, CHRArea.Area0800); + Switch01KCHR(((chr_m ? chr_reg[1] : chr_reg[3]) & 0xFF) | chr_block, CHRArea.Area0C00); + Switch01KCHR((chr_reg[4] & 0xFF) | chr_block, CHRArea.Area1000); + Switch01KCHR((chr_reg[5] & 0xFF) | chr_block, CHRArea.Area1400); + Switch01KCHR((chr_reg[6] & 0xFF) | chr_block, CHRArea.Area1800); + Switch01KCHR((chr_reg[7] & 0xFF) | chr_block, CHRArea.Area1C00); + } + break; + } + } + + private byte ReverseByte(int value) + { + return (byte)((uint)(((value & 0x40) >> 6) | ((value & 0x20) >> 4) | ((value & 0x10) >> 2)) | ((uint)value & 8u) | (uint)((value & 4) << 2) | (uint)((value & 2) << 4) | (uint)((value & 1) << 6)); + } + + internal override void OnCPUClock() + { + if (irqSource != 0) + { + return; + } + if (irqPrescalerSize) + { + irqPrescaler = (irqPrescaler & 0xF8) | (((irqPrescaler & 7) + 1) & 7); + if ((irqPrescaler & 7) == 7) + { + ClockIRQCounter(); + } + } + else + { + irqPrescaler++; + if (irqPrescaler == 255) + { + ClockIRQCounter(); + } + } + } + + internal override void OnPPUAddressUpdate(ref ushort address) + { + if (irqSource != 1) + { + return; + } + old_vram_address = new_vram_address; + new_vram_address = address & 0x1000; + if (old_vram_address >= new_vram_address) + { + return; + } + if (irqPrescalerSize) + { + irqPrescaler = (irqPrescaler & 0xF8) | (((irqPrescaler & 7) + 1) & 7); + if ((irqPrescaler & 7) == 7) + { + ClockIRQCounter(); + } + } + else + { + irqPrescaler++; + if (irqPrescaler == 255) + { + ClockIRQCounter(); + } + } + } + + private void ClockIRQCounter() + { + if (irqCountDownMode && irqCountUpMode) + { + return; + } + if (irqCountDownMode) + { + irqCounter--; + if (irqCounter == 0) + { + irqCounter = 255; + if (IrqEnable) + { + NesEmu.IRQFlags |= 8; + } + } + } + else + { + if (!irqCountUpMode) + { + return; + } + irqCounter++; + if (irqCounter == 255) + { + irqCounter = 0; + if (IrqEnable) + { + NesEmu.IRQFlags |= 8; + } + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + for (int i = 0; i < prg_reg.Length; i++) + { + stream.Write(prg_reg[i]); + } + for (int j = 0; j < chr_reg.Length; j++) + { + stream.Write(chr_reg[j]); + } + for (int k = 0; k < nt_reg.Length; k++) + { + stream.Write(nt_reg[k]); + } + stream.Write(prg_mode); + stream.Write(chr_mode); + stream.Write(chr_block_mode); + stream.Write(chr_block); + stream.Write(chr_m); + stream.Write(flag_s); + stream.Write(irqCounter); + stream.Write(IrqEnable); + stream.Write(irqCountDownMode); + stream.Write(irqCountUpMode); + stream.Write(irqFunkyMode); + stream.Write(irqPrescalerSize); + stream.Write(irqSource); + stream.Write(irqPrescaler); + stream.Write(irqPrescalerXOR); + stream.Write(irqFunkyModeReg); + stream.Write(Dipswitch); + stream.Write(multiplication_a); + stream.Write(multiplication_b); + stream.Write(multiplication); + stream.Write(RAM5803); + stream.Write(nt_advanced_enable); + stream.Write(nt_rom_only); + stream.Write(nt_ram_select); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + for (int i = 0; i < prg_reg.Length; i++) + { + prg_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < chr_reg.Length; j++) + { + chr_reg[j] = stream.ReadInt32(); + } + for (int k = 0; k < nt_reg.Length; k++) + { + nt_reg[k] = stream.ReadInt32(); + } + prg_mode = stream.ReadInt32(); + chr_mode = stream.ReadInt32(); + chr_block_mode = stream.ReadBoolean(); + chr_block = stream.ReadInt32(); + chr_m = stream.ReadBoolean(); + flag_s = stream.ReadBoolean(); + irqCounter = stream.ReadInt32(); + IrqEnable = stream.ReadBoolean(); + irqCountDownMode = stream.ReadBoolean(); + irqCountUpMode = stream.ReadBoolean(); + irqFunkyMode = stream.ReadBoolean(); + irqPrescalerSize = stream.ReadBoolean(); + irqSource = stream.ReadInt32(); + irqPrescaler = stream.ReadInt32(); + irqPrescalerXOR = stream.ReadInt32(); + irqFunkyModeReg = stream.ReadByte(); + Dipswitch = stream.ReadByte(); + multiplication_a = stream.ReadByte(); + multiplication_b = stream.ReadByte(); + multiplication = stream.ReadUInt16(); + RAM5803 = stream.ReadByte(); + nt_advanced_enable = stream.ReadBoolean(); + nt_rom_only = stream.ReadBoolean(); + nt_ram_select = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs.meta new file mode 100644 index 00000000..a1534dd9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper090.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4562c28692c896049be0384ad0e20aba +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs new file mode 100644 index 00000000..c3f994aa --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs @@ -0,0 +1,96 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("HK-SF3", 91, true, true)] + internal class Mapper091 : Board + { + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + switch (address & 0x7003) + { + case 24576: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 24577: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 24578: + Switch02KCHR(data, CHRArea.Area1000); + break; + case 24579: + Switch02KCHR(data, CHRArea.Area1800); + break; + case 28672: + Switch08KPRG(data & 0xF, PRGArea.Area8000); + break; + case 28673: + Switch08KPRG(data & 0xF, PRGArea.AreaA000); + break; + case 28674: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 28675: + irq_enabled = true; + irq_reload = 7; + irq_counter = 0; + break; + } + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs.meta new file mode 100644 index 00000000..d9ca0cec --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper091.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 81a22386348f4944eb3d2efeb823456c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs new file mode 100644 index 00000000..fd8f9a74 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs @@ -0,0 +1,50 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Jaleco Early Mapper 1", 92)] + internal class Mapper092 : Board + { + private int chr_reg; + + private int prg_reg; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(0, PRGArea.Area8000); + chr_reg = (prg_reg = 0); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch ((data >> 6) & 3) + { + case 0: + Switch08KCHR(chr_reg); + Switch16KPRG(prg_reg, PRGArea.AreaC000); + break; + case 1: + chr_reg = data & 0xF; + break; + case 2: + prg_reg = data & 0xF; + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(chr_reg); + stream.Write(prg_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + chr_reg = stream.ReadInt32(); + prg_reg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs.meta new file mode 100644 index 00000000..765b4a8f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper092.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 571f1d5188986574cb8a8b7e10b1268a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs new file mode 100644 index 00000000..bc67e0bd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs @@ -0,0 +1,18 @@ +namespace MyNes.Core +{ + [BoardInfo("74161/32", 93)] + internal class Mapper093 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch16KPRG((data >> 4) & 0xF, PRGArea.Area8000); + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs.meta new file mode 100644 index 00000000..98d1fa36 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper093.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e51bf4286fd1f7d4c85e8595173e5d94 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs new file mode 100644 index 00000000..40a5810e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs @@ -0,0 +1,17 @@ +namespace MyNes.Core +{ + [BoardInfo("74161/32", 94)] + internal class Mapper094 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch16KPRG((data >> 2) & 7, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs.meta new file mode 100644 index 00000000..dc7e6e45 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper094.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8957c987f3b199c4ba6259065bd5601e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs new file mode 100644 index 00000000..2b704fdd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs @@ -0,0 +1,265 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Namcot MMC3-Style", 95, true, true)] + internal class Mapper095 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void ReadNMT(ref ushort address, out byte value) + { + if (flag_c) + { + value = NMT_RAM[(chr_reg[((address >> 10) & 3) + 2] & 0x20) >> 5][address & 0x3FF]; + return; + } + switch ((address >> 10) & 3) + { + case 0: + case 1: + value = NMT_RAM[(chr_reg[0] & 0x20) >> 5][address & 0x3FF]; + break; + case 2: + case 3: + value = NMT_RAM[(chr_reg[1] & 0x20) >> 5][address & 0x3FF]; + break; + default: + value = 0; + break; + } + } + + internal override void WriteNMT(ref ushort address, ref byte data) + { + if (flag_c) + { + NMT_RAM[(chr_reg[((address >> 10) & 3) + 2] & 0x20) >> 5][address & 0x3FF] = data; + return; + } + switch ((address >> 10) & 3) + { + case 0: + case 1: + NMT_RAM[(chr_reg[0] & 0x20) >> 5][address & 0x3FF] = data; + break; + case 2: + case 3: + NMT_RAM[(chr_reg[1] & 0x20) >> 5][address & 0x3FF] = data; + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs.meta new file mode 100644 index 00000000..c4f3a1be --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper095.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b71e88068e12bd34fb195cee2cb0dc65 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs new file mode 100644 index 00000000..0fd1f2fe --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs @@ -0,0 +1,46 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("74161/32", 96, 1, 32)] + internal class Mapper096 : Board + { + private int flag_c; + + internal override string Issues => MNInterfaceLanguage.IssueMapper96; + + internal override void HardReset() + { + base.HardReset(); + flag_c = 0; + Switch04KCHR(3, CHRArea.Area1000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch32KPRG(data & 3, PRGArea.Area8000); + flag_c = (((data & 4) == 4) ? 1 : 0); + Switch04KCHR(3, CHRArea.Area1000); + } + + internal override void OnPPUAddressUpdate(ref ushort address) + { + if ((address & 0x3FF) < 960 && (address & 0x1000) == 0) + { + Switch04KCHR(((address & 0x300) >> 8) | flag_c, CHRArea.Area0000); + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs.meta new file mode 100644 index 00000000..43592ef8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper096.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f860cb5fe74fd8242bc105c669f9d9db +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs new file mode 100644 index 00000000..e83e7314 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs @@ -0,0 +1,32 @@ +namespace MyNes.Core +{ + [BoardInfo("Irem - PRG HI", 97)] + internal class Mapper097 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.Area8000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch16KPRG(data & 0xF, PRGArea.AreaC000); + switch ((address >> 6) & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs.meta new file mode 100644 index 00000000..664e0ca7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper097.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b05cc673e685f8444a65d76550f91aa0 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs new file mode 100644 index 00000000..ea346312 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs @@ -0,0 +1,200 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("NES-EVENT", 105)] + [HassIssues] + internal class Mapper105 : Board + { + private int DipSwitchNumber; + + private byte[] reg = new byte[4]; + + private byte shift = 0; + + private byte buffer = 0; + + private bool flag_p; + + private bool flag_s; + + private bool flag_o; + + private int reg_a; + + private int reg_b; + + private bool irq_control; + + private bool initialized; + + private int irq_counter; + + private int dipswitches; + + internal override string Issues => MNInterfaceLanguage.IssueMapper105; + + internal override void HardReset() + { + base.HardReset(); + TogglePRGRAMEnable(enable: true); + reg = new byte[4]; + reg[0] = 12; + flag_s = (flag_p = true); + reg[1] = (reg[2] = (reg[3] = 0)); + buffer = 0; + shift = 0; + initialized = false; + DipSwitchNumber = 0; + dipswitches = 0x20000000 | (DipSwitchNumber << 22); + } + + internal override void SoftReset() + { + DipSwitchNumber = (DipSwitchNumber + 1) & 0xF; + dipswitches = 0x20000000 | (DipSwitchNumber << 22); + } + + internal override void WritePRG(ref ushort address, ref byte value) + { + if ((value & 0x80) == 128) + { + reg[0] |= 12; + flag_s = (flag_p = true); + shift = (buffer = 0); + return; + } + if ((value & 1) == 1) + { + buffer |= (byte)(1 << (int)shift); + } + if (++shift < 5) + { + return; + } + address = (ushort)((address & 0x7FFF) >> 13); + reg[address] = buffer; + shift = (buffer = 0); + switch (address) + { + case 0: + flag_p = (reg[0] & 8) != 0; + flag_s = (reg[0] & 4) != 0; + UpdatePRG(); + switch (reg[0] & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.OneScA); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + } + break; + case 1: + irq_control = (reg[1] & 0x10) == 16; + if (irq_control) + { + initialized = true; + irq_counter = 0; + NesEmu.IRQFlags &= -9; + } + else + { + Switch32KPRG(0, PRGArea.Area8000); + } + flag_o = (reg[1] & 8) == 8; + reg_a = (reg[1] >> 1) & 3; + UpdatePRG(); + break; + case 3: + TogglePRGRAMEnable((reg[3] & 0x10) == 0); + reg_b = reg[3] & 0xF; + UpdatePRG(); + break; + case 2: + break; + } + } + + private void UpdatePRG() + { + if (initialized) + { + if (!flag_o) + { + Switch32KPRG(reg_a, PRGArea.Area8000); + } + else if (!flag_p) + { + Switch32KPRG((reg_b >> 1) + 4, PRGArea.Area8000); + } + else if (!flag_s) + { + Switch16KPRG(8, PRGArea.Area8000); + Switch16KPRG(reg_b + 8, PRGArea.AreaC000); + } + else + { + Switch16KPRG(reg_b + 8, PRGArea.Area8000); + Switch16KPRG(15, PRGArea.AreaC000); + } + } + } + + internal override void OnCPUClock() + { + if (!irq_control) + { + irq_counter++; + if (irq_counter == dipswitches) + { + irq_counter = 0; + NesEmu.IRQFlags |= 8; + } + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(DipSwitchNumber); + stream.Write(reg); + stream.Write(shift); + stream.Write(buffer); + stream.Write(flag_p); + stream.Write(flag_s); + stream.Write(flag_o); + stream.Write(reg_a); + stream.Write(reg_b); + stream.Write(irq_control); + stream.Write(initialized); + stream.Write(irq_counter); + stream.Write(dipswitches); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + DipSwitchNumber = stream.ReadInt32(); + stream.Read(reg, 0, reg.Length); + shift = stream.ReadByte(); + buffer = stream.ReadByte(); + flag_p = stream.ReadBoolean(); + flag_s = stream.ReadBoolean(); + flag_o = stream.ReadBoolean(); + reg_a = stream.ReadInt32(); + reg_b = stream.ReadInt32(); + irq_control = stream.ReadBoolean(); + initialized = stream.ReadBoolean(); + irq_counter = stream.ReadInt32(); + dipswitches = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs.meta new file mode 100644 index 00000000..9c366cb4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper105.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: cfb3de8d2c5d1464c87dc027df07e70c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs new file mode 100644 index 00000000..f5e332de --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs @@ -0,0 +1,11 @@ +namespace MyNes.Core +{ + internal class Mapper107 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch32KPRG(data >> 1, PRGArea.Area8000); + Switch08KCHR(data); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs.meta new file mode 100644 index 00000000..7a3c0395 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper107.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: fb362e4e113e6464bb7ea3f78943eeb1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs new file mode 100644 index 00000000..1e04e7cf --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs @@ -0,0 +1,70 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Asder", 112)] + internal class Mapper112 : Board + { + private int address_A000; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_A000 = data & 7; + break; + case 40960: + switch (address_A000) + { + case 0: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 1: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 2: + Switch02KCHR(data >> 1, CHRArea.Area0000); + break; + case 3: + Switch02KCHR(data >> 1, CHRArea.Area0800); + break; + case 4: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 5: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 6: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 7: + Switch01KCHR(data, CHRArea.Area1C00); + break; + } + break; + case 57344: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(address_A000); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + address_A000 = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs.meta new file mode 100644 index 00000000..682ebb99 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper112.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 95343c95a55d7de4785d4fa775ffde3b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs new file mode 100644 index 00000000..d8517c23 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs @@ -0,0 +1,18 @@ +namespace MyNes.Core +{ + [BoardInfo("Sachen/Hacker/Nina", 113)] + internal class Mapper113 : Board + { + internal override void WriteEX(ref ushort address, ref byte data) + { + int num = address & 0x4100; + int num2 = num; + if (num2 == 16640) + { + Switch08KCHR((data & 7) | ((data & 0x40) >> 3)); + Switch32KPRG((data >> 3) & 7, PRGArea.Area8000); + Switch01KNMTFromMirroring(((data & 0x80) == 128) ? Mirroring.Vert : Mirroring.Horz); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs.meta new file mode 100644 index 00000000..417f79d8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper113.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4bb984682499b7b4883f1ca9a4f9bd71 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs new file mode 100644 index 00000000..080e95f8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs @@ -0,0 +1,259 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("MMC3 Cart Saint", 115, true, true)] + internal class Mapper115 : Board + { + private bool flag_c; + + private bool flag_p; + + private bool flag_o; + + private int chr_block; + + private int prg_secreg; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_o = (flag_c = (flag_p = false)); + address_8001 = 0; + chr_block = 0; + prg_secreg = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + switch (address & 0x6001) + { + case 24576: + flag_o = (data & 0x80) == 128; + prg_secreg = data & 0xF; + SetupPRG(); + break; + case 24577: + chr_block = (data & 1) << 8; + SetupCHR(); + break; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR((chr_reg[0] >> 1) | chr_block, CHRArea.Area0000); + Switch02KCHR((chr_reg[1] >> 1) | chr_block, CHRArea.Area0800); + Switch01KCHR(chr_reg[2] | chr_block, CHRArea.Area1000); + Switch01KCHR(chr_reg[3] | chr_block, CHRArea.Area1400); + Switch01KCHR(chr_reg[4] | chr_block, CHRArea.Area1800); + Switch01KCHR(chr_reg[5] | chr_block, CHRArea.Area1C00); + } + else + { + Switch02KCHR((chr_reg[0] >> 1) | chr_block, CHRArea.Area1000); + Switch02KCHR((chr_reg[1] >> 1) | chr_block, CHRArea.Area1800); + Switch01KCHR(chr_reg[2] | chr_block, CHRArea.Area0000); + Switch01KCHR(chr_reg[3] | chr_block, CHRArea.Area0400); + Switch01KCHR(chr_reg[4] | chr_block, CHRArea.Area0800); + Switch01KCHR(chr_reg[5] | chr_block, CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + if (!flag_o) + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + } + else + { + Switch16KPRG(prg_secreg, PRGArea.Area8000); + } + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(flag_o); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(chr_block); + stream.Write(prg_secreg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + flag_o = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + chr_block = stream.ReadInt32(); + prg_secreg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs.meta new file mode 100644 index 00000000..2e6e78fa --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper115.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: cd032d0a8b03c6345b983161007e83c5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs new file mode 100644 index 00000000..236730f4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs @@ -0,0 +1,263 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("MMC3 TLSROM", 118, true, true)] + internal class Mapper118 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + internal override void ReadNMT(ref ushort address, out byte value) + { + switch ((address >> 10) & 3) + { + case 0: + value = NMT_RAM[(chr_reg[flag_c ? 2 : 0] >> 7) & 1][address & 0x3FF]; + break; + case 1: + value = NMT_RAM[(chr_reg[flag_c ? 3 : 0] >> 7) & 1][address & 0x3FF]; + break; + case 2: + value = NMT_RAM[(chr_reg[(!flag_c) ? 1 : 4] >> 7) & 1][address & 0x3FF]; + break; + case 3: + value = NMT_RAM[(chr_reg[(!flag_c) ? 1 : 5] >> 7) & 1][address & 0x3FF]; + break; + default: + value = 0; + break; + } + } + + internal override void WriteNMT(ref ushort address, ref byte data) + { + switch ((address >> 10) & 3) + { + case 0: + NMT_RAM[(chr_reg[flag_c ? 2 : 0] >> 7) & 1][address & 0x3FF] = data; + break; + case 1: + NMT_RAM[(chr_reg[flag_c ? 3 : 0] >> 7) & 1][address & 0x3FF] = data; + break; + case 2: + NMT_RAM[(chr_reg[(!flag_c) ? 1 : 4] >> 7) & 1][address & 0x3FF] = data; + break; + case 3: + NMT_RAM[(chr_reg[(!flag_c) ? 1 : 5] >> 7) & 1][address & 0x3FF] = data; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs.meta new file mode 100644 index 00000000..aca92d88 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper118.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6aa2fe8842a72ee4782e1df1077cfb77 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs new file mode 100644 index 00000000..828cd56e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs @@ -0,0 +1,236 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("TQROM", 119, 1, 8, true, true)] + internal class Mapper119 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override string Issues => MNInterfaceLanguage.IssueMapper119; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Toggle02KCHR_RAM((chr_reg[0] & 0x40) != 0, CHRArea.Area0000); + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Toggle02KCHR_RAM((chr_reg[1] & 0x40) != 0, CHRArea.Area0800); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Toggle02KCHR_RAM((chr_reg[2] & 0x40) != 0, CHRArea.Area1000); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Toggle02KCHR_RAM((chr_reg[3] & 0x40) != 0, CHRArea.Area1400); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Toggle02KCHR_RAM((chr_reg[4] & 0x40) != 0, CHRArea.Area1800); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Toggle02KCHR_RAM((chr_reg[5] & 0x40) != 0, CHRArea.Area1C00); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Toggle02KCHR_RAM((chr_reg[0] & 0x40) != 0, CHRArea.Area1000); + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Toggle02KCHR_RAM((chr_reg[1] & 0x40) != 0, CHRArea.Area1800); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Toggle02KCHR_RAM((chr_reg[2] & 0x40) != 0, CHRArea.Area0000); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Toggle02KCHR_RAM((chr_reg[3] & 0x40) != 0, CHRArea.Area0400); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Toggle02KCHR_RAM((chr_reg[4] & 0x40) != 0, CHRArea.Area0800); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Toggle02KCHR_RAM((chr_reg[5] & 0x40) != 0, CHRArea.Area0C00); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs.meta new file mode 100644 index 00000000..1c2f177d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper119.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 7286dc403cf3750449c7c3ec5b6e8067 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs new file mode 100644 index 00000000..8df37d77 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Sachen", 133)] + internal class Mapper133 : Board + { + internal override void WriteEX(ref ushort address, ref byte data) + { + Switch08KCHR(data & 3); + Switch32KPRG(data >> 2, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs.meta new file mode 100644 index 00000000..d70a1c49 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper133.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9b03fd27304f16b4dafa17d7d1750ee1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs new file mode 100644 index 00000000..49031e76 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 140)] + internal class Mapper140 : Board + { + internal override void WriteSRM(ref ushort address, ref byte data) + { + Switch08KCHR(data & 0xF); + Switch32KPRG((data >> 4) & 3, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs.meta new file mode 100644 index 00000000..d562dd2a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper140.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3cba32549a9133c49bbc34b948b557d9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs new file mode 100644 index 00000000..2fd20b0a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs @@ -0,0 +1,19 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 152)] + internal class Mapper152 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(data & 0xF); + Switch16KPRG((data >> 4) & 7, PRGArea.Area8000); + Switch01KNMTFromMirroring(((data & 0x80) == 128) ? Mirroring.OneScB : Mirroring.OneScA); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs.meta new file mode 100644 index 00000000..75e1a5a0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper152.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 67722d43f058e6f4a9dac4938c2f1173 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs new file mode 100644 index 00000000..34f00201 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs @@ -0,0 +1,72 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("NAMCOT-3453", 154)] + [HassIssues] + internal class Mapper154 : Board + { + private int address_8001; + + internal override string Issues => MNInterfaceLanguage.IssueMapper154; + + internal override void HardReset() + { + base.HardReset(); + address_8001 = 0; + Switch16KPRG(PRG_ROM_16KB_Mask, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0x8001) + { + case 32768: + address_8001 = data & 7; + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.OneScB : Mirroring.OneScA); + break; + case 32769: + switch (address_8001) + { + case 0: + Switch02KCHR((data & 0x3F) >> 1, CHRArea.Area0000); + break; + case 1: + Switch02KCHR((data & 0x3F) >> 1, CHRArea.Area0800); + break; + case 2: + Switch01KCHR(data | 0x40, CHRArea.Area1000); + break; + case 3: + Switch01KCHR(data | 0x40, CHRArea.Area1400); + break; + case 4: + Switch01KCHR(data | 0x40, CHRArea.Area1800); + break; + case 5: + Switch01KCHR(data | 0x40, CHRArea.Area1C00); + break; + case 6: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 7: + Switch08KPRG(data, PRGArea.AreaA000); + break; + } + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(address_8001); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + address_8001 = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs.meta new file mode 100644 index 00000000..82403ba2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper154.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 42da22d35d44cf4418466ac94e6877e1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs new file mode 100644 index 00000000..859cbad8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs @@ -0,0 +1,109 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 163)] + internal class Mapper163 : Board + { + internal byte prg_reg; + + internal byte security_reg; + + internal bool security_trigger; + + internal byte security_reg101; + + internal bool do_chr_ram; + + internal override void HardReset() + { + base.HardReset(); + security_reg101 = 1; + security_trigger = false; + do_chr_ram = true; + Switch32KPRG(15, PRGArea.Area8000); + Toggle08KCHR_RAM(ram: true); + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + if (address == 20737) + { + if (security_reg101 != 0 && data == 0) + { + security_trigger = true; + } + security_reg101 = data; + } + else + { + if (address < 20480) + { + return; + } + switch (address & 0x7300) + { + case 20480: + prg_reg = (byte)((prg_reg & 0xF0u) | (data & 0xFu)); + Switch32KPRG(prg_reg, PRGArea.Area8000); + do_chr_ram = (data & 0x80) != 0; + if (!do_chr_ram && NesEmu.ppu_clock_v < 128) + { + Switch08KCHR(0); + } + break; + case 20736: + if (data == 6) + { + Switch32KPRG(3, PRGArea.Area8000); + } + break; + case 20992: + prg_reg = (byte)((prg_reg & 0xFu) | (uint)((data & 0xF) << 4)); + Switch32KPRG(prg_reg, PRGArea.Area8000); + break; + case 21248: + security_reg = data; + break; + } + } + } + + internal override void ReadEX(ref ushort addr, out byte val) + { + val = 0; + if (addr < 20480) + { + return; + } + switch (addr & 0x1E14) + { + case 20736: + val = security_reg; + break; + case 21760: + if (security_trigger) + { + val = security_reg; + } + break; + } + } + + internal override void OnPPUScanlineTick() + { + base.OnPPUScanlineTick(); + if (do_chr_ram && NesEmu.IsRenderingOn()) + { + if (NesEmu.ppu_clock_v == 127) + { + Switch04KCHR(1, CHRArea.Area0000); + Switch04KCHR(1, CHRArea.Area1000); + } + if (NesEmu.ppu_clock_v == 237) + { + Switch04KCHR(0, CHRArea.Area0000); + Switch04KCHR(0, CHRArea.Area1000); + } + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs.meta new file mode 100644 index 00000000..c16f6b7d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper163.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 12f496f2eca79314b9d5bf0da4d6c9dc +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs new file mode 100644 index 00000000..9e86ed9a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs @@ -0,0 +1,35 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 164)] + internal class Mapper164 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch32KPRG(255, PRGArea.Area8000); + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + if (address >= 20480) + { + int num = address & 0xF000; + int num2 = num; + if (num2 == 20480) + { + Switch32KPRG(data, PRGArea.Area8000); + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + int num = address & 0xF000; + int num2 = num; + if (num2 == 53248) + { + Switch32KPRG(data, PRGArea.Area8000); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs.meta new file mode 100644 index 00000000..ccbd0598 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper164.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: da6f9ced886f85647927cc4e20e053ce +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs new file mode 100644 index 00000000..7c928751 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs @@ -0,0 +1,253 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 165, true, true)] + internal class Mapper165 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private byte latch_a = 254; + + private byte latch_b = 254; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + internal override void ReadCHR(ref ushort address, out byte data) + { + if ((address & 0x1FF0) == 4048 && latch_a != 253) + { + latch_a = 253; + Switch04KCHR(chr_reg[0] >> 2, CHRArea.Area0000); + } + else if ((address & 0x1FF0) == 4064 && latch_a != 254) + { + latch_a = 254; + Switch04KCHR(chr_reg[1] >> 2, CHRArea.Area0000); + } + else if ((address & 0x1FF0) == 8144 && latch_b != 253) + { + latch_b = 253; + Switch04KCHR(chr_reg[2] >> 2, CHRArea.Area1000); + } + else if ((address & 0x1FF0) == 8160 && latch_b != 254) + { + latch_b = 254; + Switch04KCHR(chr_reg[4] >> 2, CHRArea.Area1000); + } + base.ReadCHR(ref address, out data); + } + + private void SetupCHR() + { + if (latch_a == 253) + { + Switch04KCHR(chr_reg[0] >> 2, CHRArea.Area0000); + } + if (latch_a == 254) + { + Switch04KCHR(chr_reg[1] >> 2, CHRArea.Area0000); + } + if (latch_b == 253) + { + Switch04KCHR(chr_reg[2] >> 2, CHRArea.Area1000); + } + if (latch_b == 254) + { + Switch04KCHR(chr_reg[4] >> 2, CHRArea.Area1000); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(latch_a); + stream.Write(latch_b); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + latch_a = stream.ReadByte(); + latch_b = stream.ReadByte(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs.meta new file mode 100644 index 00000000..ea827a51 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper165.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 01df600d7fe9ad7419fea7c8fd3e3eb1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs new file mode 100644 index 00000000..6ec511ae --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs @@ -0,0 +1,14 @@ +namespace MyNes.Core +{ + [BoardInfo("Nihon Bussan", 180)] + [HassIssues] + internal class Mapper180 : Board + { + internal override string Issues => MNInterfaceLanguage.IssueMapper180; + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch16KPRG(data & 7, PRGArea.AreaC000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs.meta new file mode 100644 index 00000000..9f1fc09f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper180.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1ffae0c30d3a5dd48af658c47bc53a38 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs new file mode 100644 index 00000000..49c7fda3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs @@ -0,0 +1,216 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 182, true, true)] + internal class Mapper182 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + mmc3_alt_behavior = false; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32769: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40960: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 49152: + switch (address_8001) + { + case 0: + chr_reg[0] = data; + SetupCHR(); + break; + case 1: + chr_reg[3] = data; + SetupCHR(); + break; + case 2: + chr_reg[1] = data; + SetupCHR(); + break; + case 3: + chr_reg[5] = data; + SetupCHR(); + break; + case 4: + prg_reg[0] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + case 5: + prg_reg[1] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + case 6: + chr_reg[2] = data; + SetupCHR(); + break; + case 7: + chr_reg[4] = data; + SetupCHR(); + break; + } + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + irq_reload = data; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs.meta new file mode 100644 index 00000000..55001c4d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper182.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: db03437852d4eb942a0ea6a97e2169bf +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs new file mode 100644 index 00000000..943a192e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Sunsoft", 184)] + internal class Mapper184 : Board + { + internal override void WriteSRM(ref ushort address, ref byte data) + { + Switch04KCHR(data & 7, CHRArea.Area0000); + Switch04KCHR((data >> 4) & 7, CHRArea.Area1000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs.meta new file mode 100644 index 00000000..f10c0476 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper184.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3ea85ab17c144364994c80fc92ddafe5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs new file mode 100644 index 00000000..8a3f1275 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs @@ -0,0 +1,45 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 185)] + internal class Mapper185 : Board + { + private bool lockchr; + + internal override void HardReset() + { + base.HardReset(); + lockchr = false; + } + + internal override void ReadCHR(ref ushort address, out byte data) + { + if (!lockchr) + { + base.ReadCHR(ref address, out data); + } + else + { + data = byte.MaxValue; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + lockchr = (data & 0xF) == 0 || data == 19; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(lockchr); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + lockchr = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs.meta new file mode 100644 index 00000000..cd6b4488 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper185.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8c3aabf4c02e11a4988508a35f600696 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs new file mode 100644 index 00000000..4bd5d571 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs @@ -0,0 +1,185 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("MMC3 Variant", 189, true, true)] + internal class Mapper189 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = 0; + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + Switch32KPRG(((data & 0xF0) >> 4) | (data & 0xF), PRGArea.Area8000); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + Switch32KPRG(((data & 0xF0) >> 4) | (data & 0xF), PRGArea.Area8000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + break; + case 32769: + { + int num = address_8001; + int num2 = num; + if ((uint)num2 <= 5u) + { + chr_reg[address_8001] = data; + SetupCHR(); + } + break; + } + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + stream.Write(prg_reg); + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + prg_reg = stream.ReadInt32(); + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs.meta new file mode 100644 index 00000000..9b37abb3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper189.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 16049f0fed2d5614188f842c620127a9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs new file mode 100644 index 00000000..b7a77764 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs @@ -0,0 +1,212 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Pirate MMC3 variant", 191, true, true)] + [HassIssues] + internal class Mapper191 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + internal override string Issues => MNInterfaceLanguage.IssueMapper191; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Toggle02KCHR_RAM((chr_reg[0] & 0x80) == 0, CHRArea.Area0000); + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Toggle02KCHR_RAM((chr_reg[1] & 0x80) == 0, CHRArea.Area0800); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Toggle02KCHR_RAM((chr_reg[2] & 0x80) == 0, CHRArea.Area1000); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Toggle02KCHR_RAM((chr_reg[3] & 0x80) == 0, CHRArea.Area1400); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Toggle02KCHR_RAM((chr_reg[4] & 0x80) == 0, CHRArea.Area1800); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Toggle02KCHR_RAM((chr_reg[5] & 0x80) == 0, CHRArea.Area1C00); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Toggle02KCHR_RAM((chr_reg[0] & 0x80) == 0, CHRArea.Area1000); + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Toggle02KCHR_RAM((chr_reg[1] & 0x80) == 0, CHRArea.Area1800); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Toggle02KCHR_RAM((chr_reg[2] & 0x80) == 0, CHRArea.Area0000); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Toggle02KCHR_RAM((chr_reg[3] & 0x80) == 0, CHRArea.Area0400); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Toggle02KCHR_RAM((chr_reg[4] & 0x80) == 0, CHRArea.Area0800); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Toggle02KCHR_RAM((chr_reg[5] & 0x80) == 0, CHRArea.Area0C00); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs.meta new file mode 100644 index 00000000..55a5c037 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper191.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0938567bea9d60e46b861a10275b4484 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs new file mode 100644 index 00000000..f62dc4fc --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs @@ -0,0 +1,225 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Pirate MMC3 variant", 192, true, true)] + internal class Mapper192 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool isRam; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + isRam = chr_reg[0] >= 8 && chr_reg[0] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area0000); + Switch02KCHR(isRam ? (chr_reg[0] - 8 >> 1) : (chr_reg[0] >> 1), CHRArea.Area0000); + isRam = chr_reg[1] >= 8 && chr_reg[1] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area0000); + Switch02KCHR(isRam ? (chr_reg[1] - 8 >> 1) : (chr_reg[1] >> 1), CHRArea.Area0800); + isRam = chr_reg[2] >= 8 && chr_reg[2] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area1000); + Switch01KCHR(isRam ? (chr_reg[2] - 8) : chr_reg[2], CHRArea.Area1000); + isRam = chr_reg[3] >= 8 && chr_reg[3] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area1400); + Switch01KCHR(isRam ? (chr_reg[3] - 8) : chr_reg[3], CHRArea.Area1400); + isRam = chr_reg[4] >= 8 && chr_reg[4] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area1800); + Switch01KCHR(isRam ? (chr_reg[4] - 8) : chr_reg[4], CHRArea.Area1800); + isRam = chr_reg[5] >= 8 && chr_reg[5] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area1C00); + Switch01KCHR(isRam ? (chr_reg[5] - 8) : chr_reg[5], CHRArea.Area1C00); + } + else + { + isRam = chr_reg[0] >= 8 && chr_reg[0] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area1000); + Switch02KCHR(isRam ? (chr_reg[0] - 8 >> 1) : (chr_reg[0] >> 1), CHRArea.Area1000); + isRam = chr_reg[1] >= 8 && chr_reg[1] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area1800); + Switch02KCHR(isRam ? (chr_reg[1] - 8 >> 1) : (chr_reg[1] >> 1), CHRArea.Area1800); + isRam = chr_reg[2] >= 8 && chr_reg[2] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area0000); + Switch01KCHR(isRam ? (chr_reg[2] - 8) : chr_reg[2], CHRArea.Area0000); + isRam = chr_reg[3] >= 8 && chr_reg[3] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area0400); + Switch01KCHR(isRam ? (chr_reg[3] - 8) : chr_reg[3], CHRArea.Area0400); + isRam = chr_reg[4] >= 8 && chr_reg[4] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area0800); + Switch01KCHR(isRam ? (chr_reg[4] - 8) : chr_reg[4], CHRArea.Area0800); + isRam = chr_reg[5] >= 8 && chr_reg[5] <= 11; + Toggle02KCHR_RAM(isRam, CHRArea.Area0C00); + Switch01KCHR(isRam ? (chr_reg[5] - 8) : chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(isRam); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + isRam = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs.meta new file mode 100644 index 00000000..354492cb --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper192.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4fa6279cecd943f42baf6d4e8a966cb4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs new file mode 100644 index 00000000..6e6f39b8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs @@ -0,0 +1,36 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 193, 1, 32)] + [HassIssues] + internal class Mapper193 : Board + { + internal override string Issues => MNInterfaceLanguage.IssueMapper193; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask - 2, PRGArea.AreaA000); + Switch08KPRG(PRG_ROM_08KB_Mask - 1, PRGArea.AreaC000); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + switch (address & 0x6003) + { + case 24576: + Switch04KCHR(data >> 2, CHRArea.Area0000); + break; + case 24577: + Switch02KCHR(data >> 1, CHRArea.Area1000); + break; + case 24578: + Switch02KCHR(data >> 1, CHRArea.Area1800); + break; + case 24579: + Switch08KPRG(data, PRGArea.Area8000); + break; + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs.meta new file mode 100644 index 00000000..512e4aa7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper193.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f76460c502274ab4a8cac566ce0f0126 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs new file mode 100644 index 00000000..1f2a785a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs @@ -0,0 +1,209 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Pirate MMC3 variant", 194, true, true)] + internal class Mapper194 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Toggle02KCHR_RAM(chr_reg[0] > 1, CHRArea.Area0000); + Switch02KCHR((chr_reg[0] > 1) ? (chr_reg[0] - 1 >> 1) : (chr_reg[0] >> 1), CHRArea.Area0000); + Toggle02KCHR_RAM(chr_reg[1] > 1, CHRArea.Area0800); + Switch02KCHR((chr_reg[1] > 1) ? (chr_reg[1] - 1 >> 1) : (chr_reg[1] >> 1), CHRArea.Area0800); + Toggle02KCHR_RAM(chr_reg[2] > 1, CHRArea.Area1000); + Switch01KCHR((chr_reg[2] > 1) ? (chr_reg[2] - 1) : chr_reg[2], CHRArea.Area1000); + Toggle02KCHR_RAM(chr_reg[3] > 1, CHRArea.Area1400); + Switch01KCHR((chr_reg[3] > 1) ? (chr_reg[3] - 1) : chr_reg[3], CHRArea.Area1400); + Toggle02KCHR_RAM(chr_reg[4] > 1, CHRArea.Area1800); + Switch01KCHR((chr_reg[4] > 1) ? (chr_reg[4] - 1) : chr_reg[4], CHRArea.Area1800); + Toggle02KCHR_RAM(chr_reg[5] > 1, CHRArea.Area1C00); + Switch01KCHR((chr_reg[5] > 1) ? (chr_reg[5] - 1) : chr_reg[5], CHRArea.Area1C00); + } + else + { + Toggle02KCHR_RAM(chr_reg[0] > 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Toggle02KCHR_RAM(chr_reg[1] > 1, CHRArea.Area1800); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Toggle02KCHR_RAM(chr_reg[2] > 1, CHRArea.Area0000); + Switch01KCHR((chr_reg[2] > 1) ? (chr_reg[2] - 1) : chr_reg[2], CHRArea.Area0000); + Toggle02KCHR_RAM(chr_reg[3] > 1, CHRArea.Area0400); + Switch01KCHR((chr_reg[3] > 1) ? (chr_reg[3] - 1) : chr_reg[3], CHRArea.Area0400); + Toggle02KCHR_RAM(chr_reg[4] > 1, CHRArea.Area0800); + Switch01KCHR((chr_reg[4] > 1) ? (chr_reg[4] - 1) : chr_reg[4], CHRArea.Area0800); + Toggle02KCHR_RAM(chr_reg[5] > 1, CHRArea.Area0C00); + Switch01KCHR((chr_reg[5] > 1) ? (chr_reg[5] - 1) : chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs.meta new file mode 100644 index 00000000..3dc67526 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper194.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 81ac03f4648c10c47b5903542095eef4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs new file mode 100644 index 00000000..4967f1bd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs @@ -0,0 +1,14 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 200)] + internal class Mapper200 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(address & 7); + Switch16KPRG(address & 7, PRGArea.Area8000); + Switch16KPRG(address & 7, PRGArea.AreaC000); + Switch01KNMTFromMirroring(((address & 8) == 8) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs.meta new file mode 100644 index 00000000..82706b1d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper200.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: cf60a6b944a2cd646b955ef1890527fe +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs new file mode 100644 index 00000000..a1aa22f2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 201)] + internal class Mapper201 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(address & 0xFF); + Switch32KPRG(address & 0xFF, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs.meta new file mode 100644 index 00000000..8978d1c2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper201.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0fa34e45527799b4090d44ffa7f88235 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs new file mode 100644 index 00000000..dc485dfd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs @@ -0,0 +1,22 @@ +namespace MyNes.Core +{ + [BoardInfo("150-in-1", 202)] + [HassIssues] + internal class Mapper202 : Board + { + internal override string Issues => MNInterfaceLanguage.IssueMapper202; + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch01KNMTFromMirroring(((address & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + Switch08KCHR((address >> 1) & 7); + if ((address & 0xC) == 12) + { + Switch32KPRG(3, PRGArea.Area8000); + return; + } + Switch16KPRG((address >> 1) & 7, PRGArea.Area8000); + Switch16KPRG((address >> 1) & 7, PRGArea.AreaC000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs.meta new file mode 100644 index 00000000..054b5a01 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper202.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 70c735d1069ff314dbf88ff45dbdb8b2 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs new file mode 100644 index 00000000..a1a20199 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs @@ -0,0 +1,16 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 203)] + [HassIssues] + internal class Mapper203 : Board + { + internal override string Issues => MNInterfaceLanguage.IssueMapper203; + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(data & 3); + Switch16KPRG(data & 0x3F, PRGArea.Area8000); + Switch16KPRG(data & 0x3F, PRGArea.AreaC000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs.meta new file mode 100644 index 00000000..79c73322 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper203.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 16cfb93389e5f6a4084c106625e4879f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs new file mode 100644 index 00000000..bbff677b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs @@ -0,0 +1,19 @@ +namespace MyNes.Core +{ + [BoardInfo("64-in-1", 204)] + internal class Mapper204 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch01KNMTFromMirroring(((address & 0x10) == 16) ? Mirroring.Horz : Mirroring.Vert); + Switch08KCHR(address & 7); + if ((address & 6) == 6) + { + Switch32KPRG(3, PRGArea.Area8000); + return; + } + Switch16KPRG(address & 7, PRGArea.Area8000); + Switch16KPRG(address & 7, PRGArea.AreaC000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs.meta new file mode 100644 index 00000000..6c034820 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper204.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5d5dcc6e4a9329745868b6b2ef452dc5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs new file mode 100644 index 00000000..efb548a2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs @@ -0,0 +1,275 @@ +using System; +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 205, true, true)] + internal class Mapper205 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private int chr_or; + + private int chr_and; + + private int prg_or; + + private int prg_and; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_and = 31; + prg_or = 0; + chr_and = 255; + chr_or = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + if (IsGameFoundOnDB) + { + switch (GameCartInfo.chip_type[0].ToLower()) + { + case "mmc3a": + mmc3_alt_behavior = true; + Console.WriteLine("Chip= MMC3 A, MMC3 IQR mode switched to RevA"); + break; + case "mmc3b": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 B, MMC3 IQR mode switched to RevB"); + break; + case "mmc3c": + mmc3_alt_behavior = false; + Console.WriteLine("Chip= MMC3 C, MMC3 IQR mode switched to RevB"); + break; + } + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + switch (data & 3) + { + case 0: + prg_and = 31; + prg_or = 0; + chr_and = 255; + chr_or = 0; + break; + case 1: + prg_and = 31; + prg_or = 16; + chr_and = 255; + chr_or = 128; + break; + case 2: + prg_and = 15; + prg_or = 32; + chr_and = 127; + chr_or = 256; + break; + case 3: + prg_and = 15; + prg_or = 48; + chr_and = 127; + chr_or = 384; + break; + } + SetupCHR(); + SetupPRG(); + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area0000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area0800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area1000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area1400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area1800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area1C00); + } + else + { + Switch02KCHR(((chr_reg[0] & chr_and) | chr_or) >> 1, CHRArea.Area1000); + Switch02KCHR(((chr_reg[1] & chr_and) | chr_or) >> 1, CHRArea.Area1800); + Switch01KCHR((chr_reg[2] & chr_and) | chr_or, CHRArea.Area0000); + Switch01KCHR((chr_reg[3] & chr_and) | chr_or, CHRArea.Area0400); + Switch01KCHR((chr_reg[4] & chr_and) | chr_or, CHRArea.Area0800); + Switch01KCHR((chr_reg[5] & chr_and) | chr_or, CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG((prg_reg[flag_p ? 2 : 0] & prg_and) | prg_or, PRGArea.Area8000); + Switch08KPRG((prg_reg[1] & prg_and) | prg_or, PRGArea.AreaA000); + Switch08KPRG((prg_reg[(!flag_p) ? 2 : 0] & prg_and) | prg_or, PRGArea.AreaC000); + Switch08KPRG((prg_reg[3] & prg_and) | prg_or, PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(chr_or); + stream.Write(chr_and); + stream.Write(prg_or); + stream.Write(prg_and); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + chr_or = stream.ReadInt32(); + chr_and = stream.ReadInt32(); + prg_or = stream.ReadInt32(); + prg_and = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs.meta new file mode 100644 index 00000000..2108df88 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper205.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 10f689fe32f13114fbc616cb3a04dd0f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs new file mode 100644 index 00000000..d50894c7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs @@ -0,0 +1,131 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 206)] + internal class Mapper206 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + SetupPRG(); + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + case 7: + prg_reg[address_8001 - 6] = data & PRG_ROM_08KB_Mask; + SetupPRG(); + break; + } + break; + } + } + + private void SetupCHR() + { + if (!flag_c) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + + private void SetupPRG() + { + Switch08KPRG(prg_reg[flag_p ? 2 : 0], PRGArea.Area8000); + Switch08KPRG(prg_reg[1], PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0], PRGArea.AreaC000); + Switch08KPRG(prg_reg[3], PRGArea.AreaE000); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs.meta new file mode 100644 index 00000000..ae267178 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper206.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: dd0ebc01d4ccb5144be2ae2380e63a4b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs new file mode 100644 index 00000000..4f7778ab --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs @@ -0,0 +1,112 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 207)] + [HassIssues] + internal class Mapper207 : Board + { + private int mirroring0; + + private int mirroring1; + + internal override string Issues => MNInterfaceLanguage.IssueMapper207; + + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + switch (address) + { + case 32496: + Switch02KCHR(data & 0x3F, CHRArea.Area0000); + mirroring0 = (data >> 7) & 1; + break; + case 32497: + Switch02KCHR(data & 0x3F, CHRArea.Area0800); + mirroring1 = (data >> 7) & 1; + break; + case 32498: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 32499: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 32500: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 32501: + Switch01KCHR(data, CHRArea.Area1C00); + break; + case 32506: + case 32507: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 32508: + case 32509: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 32510: + case 32511: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 32502: + case 32503: + case 32504: + case 32505: + break; + } + } + + internal override void ReadNMT(ref ushort address, out byte data) + { + switch ((address >> 10) & 3) + { + case 0: + case 1: + data = NMT_RAM[mirroring0][address & 0x3FF]; + break; + case 2: + case 3: + data = NMT_RAM[mirroring1][address & 0x3FF]; + break; + default: + data = 0; + break; + } + } + + internal override void WriteNMT(ref ushort address, ref byte data) + { + switch ((address >> 10) & 3) + { + case 0: + case 1: + NMT_RAM[mirroring0][address & 0x3FF] = data; + break; + case 2: + case 3: + NMT_RAM[mirroring1][address & 0x3FF] = data; + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(mirroring0); + stream.Write(mirroring1); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + mirroring0 = stream.ReadInt32(); + mirroring1 = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs.meta new file mode 100644 index 00000000..8905a373 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper207.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8120e93e983c6a240aa549981107c6f9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs new file mode 100644 index 00000000..5b4d75ff --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Pirate MMC5-style", 209)] + internal class Mapper209 : Mapper090 + { + internal override void HardReset() + { + MAPPER90MODE = false; + base.HardReset(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs.meta new file mode 100644 index 00000000..a769ed78 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper209.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6e1cfeabf90d9d241a85d23d367fd146 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs new file mode 100644 index 00000000..9b4f7fb7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs @@ -0,0 +1,19 @@ +namespace MyNes.Core +{ + [BoardInfo("Super HIK 300-in-1 (1994)", 212)] + internal class Mapper212 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(address & 7); + Switch01KNMTFromMirroring(((address & 8) == 8) ? Mirroring.Horz : Mirroring.Vert); + if ((address & 0x4000) == 16384) + { + Switch32KPRG((address >> 1) & 3, PRGArea.Area8000); + return; + } + Switch16KPRG(address & 7, PRGArea.Area8000); + Switch16KPRG(address & 7, PRGArea.AreaC000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs.meta new file mode 100644 index 00000000..f4712eaa --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper212.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9b0a6452db095d64d9b655ea1a70de61 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs new file mode 100644 index 00000000..781dc65f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("9999999-in-1", 213)] + internal class Mapper213 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR((address >> 3) & 7); + Switch32KPRG((address >> 1) & 3, PRGArea.Area8000); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs.meta new file mode 100644 index 00000000..0b17a906 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper213.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 5f55d2933322aa24986e3c886539b729 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs new file mode 100644 index 00000000..e45d15ae --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs @@ -0,0 +1,13 @@ +namespace MyNes.Core +{ + [BoardInfo("X-in-1", 214)] + internal class Mapper214 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch16KPRG(address >> 2, PRGArea.Area8000); + Switch16KPRG(address >> 2, PRGArea.AreaC000); + Switch08KCHR(address); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs.meta new file mode 100644 index 00000000..ab69bd18 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper214.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c7b5c32585db1e640b6b297c40f0df87 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs new file mode 100644 index 00000000..edd16c22 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 216)] + internal class Mapper216 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch32KPRG(address, PRGArea.Area8000); + Switch08KCHR(address >> 1); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs.meta new file mode 100644 index 00000000..bccb44d4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper216.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e384cf0eb5fa0b04eb8453de825d16f1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs new file mode 100644 index 00000000..c21c8938 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs @@ -0,0 +1,48 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 222)] + internal class Mapper222 : Board + { + internal override string Issues => MNInterfaceLanguage.IssueMapper222; + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF003) + { + case 32768: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 36864: + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + break; + case 40960: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 45056: + Switch01KCHR(data, CHRArea.Area0000); + break; + case 45058: + Switch01KCHR(data, CHRArea.Area0400); + break; + case 49152: + Switch01KCHR(data, CHRArea.Area0800); + break; + case 49154: + Switch01KCHR(data, CHRArea.Area0C00); + break; + case 53248: + Switch01KCHR(data, CHRArea.Area1000); + break; + case 53250: + Switch01KCHR(data, CHRArea.Area1400); + break; + case 57344: + Switch01KCHR(data, CHRArea.Area1800); + break; + case 57346: + Switch01KCHR(data, CHRArea.Area1C00); + break; + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs.meta new file mode 100644 index 00000000..fe1dc074 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper222.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 84f2bbbe7be8a35479c83cded3cdd0f4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs new file mode 100644 index 00000000..29ab413a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs @@ -0,0 +1,63 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 225)] + internal class Mapper225 : Board + { + private byte[] RAM; + + internal override void HardReset() + { + base.HardReset(); + RAM = new byte[4]; + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + if (address >= 22528) + { + RAM[address & 3] = (byte)(data & 0xFu); + } + } + + internal override void ReadEX(ref ushort address, out byte value) + { + if (address >= 22528) + { + value = RAM[address & 3]; + } + else + { + value = 0; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(address & 0x3F); + if ((address & 0x1000) == 4096) + { + Switch16KPRG((address >> 6) & 0x3F, PRGArea.Area8000); + Switch16KPRG((address >> 6) & 0x3F, PRGArea.AreaC000); + } + else + { + Switch32KPRG(((address >> 6) & 0x3F) >> 1, PRGArea.Area8000); + } + Switch01KNMTFromMirroring(((address & 0x2000) == 8192) ? Mirroring.Horz : Mirroring.Vert); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(RAM); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + stream.Read(RAM, 0, RAM.Length); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs.meta new file mode 100644 index 00000000..c9bc3f1f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper225.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 805833a504a7b7e44b9bc02878580e21 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs new file mode 100644 index 00000000..3fe1ec42 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs @@ -0,0 +1,66 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 226)] + internal class Mapper226 : Board + { + private int prg_reg; + + private bool prg_mode; + + internal override void HardReset() + { + base.HardReset(); + prg_reg = 0; + prg_mode = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0x8001) + { + case 32768: + prg_reg = (data & 0x1F) | ((data & 0x80) >> 2) | (prg_reg & 0xC0); + prg_mode = (data & 0x20) == 32; + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.Vert : Mirroring.Horz); + if (prg_mode) + { + Switch16KPRG(prg_reg, PRGArea.Area8000); + Switch16KPRG(prg_reg, PRGArea.AreaC000); + } + else + { + Switch32KPRG(prg_reg >> 1, PRGArea.Area8000); + } + break; + case 32769: + prg_reg = ((data & 1) << 6) | (prg_reg & 0x3F); + if (prg_mode) + { + Switch16KPRG(prg_reg, PRGArea.Area8000); + Switch16KPRG(prg_reg, PRGArea.AreaC000); + } + else + { + Switch32KPRG(prg_reg >> 1, PRGArea.Area8000); + } + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prg_reg); + stream.Write(prg_mode); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prg_reg = stream.ReadInt32(); + prg_mode = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs.meta new file mode 100644 index 00000000..194189b7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper226.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d2d0d6a9786e1eb488a45d8b784279b9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs new file mode 100644 index 00000000..b838088b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs @@ -0,0 +1,90 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 227)] + internal class Mapper227 : Board + { + private bool flag_o; + + private bool flag_s; + + private bool flag_l; + + private int prg_reg; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(0, PRGArea.AreaC000); + flag_o = false; + flag_s = false; + flag_l = false; + prg_reg = 0; + ToggleCHRRAMWritableEnable(enable: true); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + flag_s = (address & 1) == 1; + flag_o = (address & 0x80) == 128; + flag_l = (address & 0x200) == 512; + prg_reg = ((address >> 2) & 0x1F) | ((address >> 3) & 0x20); + Switch01KNMTFromMirroring(((address & 2) == 2) ? Mirroring.Horz : Mirroring.Vert); + ToggleCHRRAMWritableEnable(!flag_o); + if (flag_o) + { + if (!flag_s) + { + Switch16KPRG(prg_reg, PRGArea.Area8000); + Switch16KPRG(prg_reg, PRGArea.AreaC000); + } + else + { + Switch32KPRG(prg_reg >> 1, PRGArea.Area8000); + } + } + else if (!flag_l) + { + if (!flag_s) + { + Switch16KPRG(prg_reg, PRGArea.Area8000); + Switch16KPRG(prg_reg & 0x38, PRGArea.AreaC000); + } + else + { + Switch16KPRG(prg_reg & 0x3E, PRGArea.Area8000); + Switch16KPRG(prg_reg & 0x38, PRGArea.AreaC000); + } + } + else if (!flag_s) + { + Switch16KPRG(prg_reg, PRGArea.Area8000); + Switch16KPRG(prg_reg | 7, PRGArea.AreaC000); + } + else + { + Switch16KPRG(prg_reg & 0x3E, PRGArea.Area8000); + Switch16KPRG(prg_reg | 7, PRGArea.AreaC000); + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_o); + stream.Write(flag_s); + stream.Write(flag_l); + stream.Write(prg_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_o = stream.ReadBoolean(); + flag_s = stream.ReadBoolean(); + flag_l = stream.ReadBoolean(); + prg_reg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs.meta new file mode 100644 index 00000000..4ac38b8b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper227.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 675c53fc4556f454094683ba8b663cca +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs new file mode 100644 index 00000000..f5596d01 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs @@ -0,0 +1,61 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 228)] + [HassIssues] + internal class Mapper228 : Board + { + private byte[] RAM; + + private int bank; + + internal override string Issues => MNInterfaceLanguage.IssueMapper228; + + internal override void HardReset() + { + base.HardReset(); + RAM = new byte[4]; + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + RAM[address & 3] = (byte)(data & 0xFu); + } + + internal override void ReadEX(ref ushort address, out byte data) + { + data = RAM[address & 3]; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(((address & 0xF) << 2) | (data & 3)); + Switch01KNMTFromMirroring(((address & 0x2000) == 8192) ? Mirroring.Horz : Mirroring.Vert); + bank = ((address >> 7) & 0x1F) + ((address >> 7) & (address >> 8) & 0x10); + if ((address & 0x20) == 32) + { + Switch16KPRG((bank << 2) | ((address >> 5) & 2), PRGArea.Area8000); + Switch16KPRG((bank << 2) | ((address >> 5) & 2), PRGArea.AreaC000); + } + else + { + Switch32KPRG(bank, PRGArea.Area8000); + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(RAM); + stream.Write(bank); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + stream.Read(RAM, 0, RAM.Length); + bank = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs.meta new file mode 100644 index 00000000..2578d37a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper228.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d66a7ccd1090c894ba6919d9fa505c79 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs new file mode 100644 index 00000000..84ffd917 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs @@ -0,0 +1,17 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 229)] + [HassIssues] + internal class Mapper229 : Board + { + internal override string Issues => MNInterfaceLanguage.IssueMapper229; + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch01KNMTFromMirroring(((address & 0x20) == 32) ? Mirroring.Horz : Mirroring.Vert); + Switch16KPRG(((address & 0x1E) == 30) ? (address & 0x1F) : 0, PRGArea.Area8000); + Switch16KPRG(((address & 0x1E) != 30) ? 1 : (address & 0x1F), PRGArea.AreaC000); + Switch08KCHR(address); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs.meta new file mode 100644 index 00000000..abbb34fd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper229.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8123f098f1b3113469e3407bbd436d4f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs new file mode 100644 index 00000000..879ff319 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs @@ -0,0 +1,58 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 230)] + [HassIssues] + internal class Mapper230 : Board + { + private bool contraMode = false; + + internal override string Issues => MNInterfaceLanguage.IssueMapper230; + + internal override void HardReset() + { + base.HardReset(); + contraMode = true; + Switch16KPRG(0, PRGArea.Area8000); + Switch16KPRG(7, PRGArea.AreaC000); + } + + internal override void SoftReset() + { + base.SoftReset(); + contraMode = !contraMode; + if (contraMode) + { + Switch16KPRG(0, PRGArea.Area8000); + Switch16KPRG(7, PRGArea.AreaC000); + Switch01KNMTFromMirroring(Mirroring.Vert); + } + else + { + Switch08KCHR(0); + Switch16KPRG(8, PRGArea.Area8000); + Switch16KPRG(39, PRGArea.AreaC000); + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (contraMode) + { + Switch16KPRG(data & 7, PRGArea.Area8000); + Switch16KPRG(7, PRGArea.AreaC000); + Switch01KNMTFromMirroring(Mirroring.Vert); + return; + } + Switch01KNMTFromMirroring(((data & 0x40) == 64) ? Mirroring.Vert : Mirroring.Horz); + if ((data & 0x20) == 32) + { + Switch16KPRG((data & 0x1F) + 8, PRGArea.Area8000); + Switch16KPRG((data & 0x1F) + 8, PRGArea.AreaC000); + } + else + { + Switch32KPRG(((data & 0x1F) >> 1) + 4, PRGArea.Area8000); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs.meta new file mode 100644 index 00000000..cc4df397 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper230.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f6425637e54797c45abebe7b1c8e30dc +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs new file mode 100644 index 00000000..98743452 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs @@ -0,0 +1,13 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 231)] + internal class Mapper231 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch16KPRG(address & 0x1E, PRGArea.Area8000); + Switch16KPRG((address & 0x1E) | ((address >> 5) & 1), PRGArea.AreaC000); + Switch01KNMTFromMirroring(((address & 0x80) == 128) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs.meta new file mode 100644 index 00000000..e4f30842 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper231.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: be52d26a4ead22042b0265a6db3b2a6a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs new file mode 100644 index 00000000..4d48bf45 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs @@ -0,0 +1,42 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 232)] + internal class Mapper232 : Board + { + private int prg_reg; + + internal override void HardReset() + { + base.HardReset(); + Switch16KPRG(3, PRGArea.AreaC000); + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + if (address < 49152) + { + prg_reg = ((data & 0x18) >> 1) | (prg_reg & 3); + } + else + { + prg_reg = (prg_reg & 0xC) | (data & 3); + } + Switch16KPRG(prg_reg, PRGArea.Area8000); + Switch16KPRG(3 | (prg_reg & 0xC), PRGArea.AreaC000); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(prg_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + prg_reg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs.meta new file mode 100644 index 00000000..a6090df2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper232.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: dafbc365bc8eb164a9fcac860585e366 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs new file mode 100644 index 00000000..7e55904b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs @@ -0,0 +1,68 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 233)] + internal class Mapper233 : Board + { + private int title = 0; + + private int bank; + + internal override void HardReset() + { + base.HardReset(); + bank = (title = 0); + } + + internal override void SoftReset() + { + base.SoftReset(); + bank = 0; + title ^= 32; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + bank = data & 0x1F; + if ((data & 0x20) == 32) + { + Switch16KPRG(title | bank, PRGArea.Area8000); + Switch16KPRG(title | bank, PRGArea.AreaC000); + } + else + { + Switch32KPRG((title >> 1) | (bank >> 1), PRGArea.Area8000); + } + switch ((data >> 6) & 3) + { + case 0: + Switch01KNMT(128); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 2: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(title); + stream.Write(bank); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + title = stream.ReadInt32(); + bank = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs.meta new file mode 100644 index 00000000..400a17e3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper233.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e907b2b34ef04f145be24c3f2eada20f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs new file mode 100644 index 00000000..18912221 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 240)] + internal class Mapper240 : Board + { + internal override void WriteEX(ref ushort address, ref byte data) + { + Switch32KPRG((data >> 4) & 0xF, PRGArea.Area8000); + Switch08KCHR(data & 0xF); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs.meta new file mode 100644 index 00000000..52cd3f72 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper240.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: eb9d4ffbec8733541a56bc8469e2b86f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs new file mode 100644 index 00000000..f305de53 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs @@ -0,0 +1,12 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 242)] + internal class Mapper242 : Board + { + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch32KPRG((address >> 3) & 0xF, PRGArea.Area8000); + Switch01KNMTFromMirroring(((address & 2) == 2) ? Mirroring.Horz : Mirroring.Vert); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs.meta new file mode 100644 index 00000000..1334878e --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper242.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d62ffe85351435340a8e3c42f84dee3b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs new file mode 100644 index 00000000..39433279 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs @@ -0,0 +1,88 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Sachen Poker", 243)] + [HassIssues] + internal class Mapper243 : Board + { + private int addr; + + private int chr_reg; + + internal override string Issues => MNInterfaceLanguage.IssueMapper243; + + internal override void HardReset() + { + base.HardReset(); + addr = (chr_reg = 0); + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + if (address >= 20480 || address < 16416) + { + return; + } + switch (address & 0x4101) + { + case 16640: + addr = data & 7; + break; + case 16641: + switch (addr) + { + case 2: + chr_reg = ((data << 3) & 8) | (chr_reg & 7); + Switch08KCHR(chr_reg); + break; + case 4: + chr_reg = (data & 1) | (chr_reg & 0xE); + Switch08KCHR(chr_reg); + break; + case 5: + Switch32KPRG(data & 7, PRGArea.Area8000); + break; + case 6: + chr_reg = ((data & 3) << 1) | (chr_reg & 9); + Switch08KCHR(chr_reg); + break; + case 7: + switch ((data >> 1) & 3) + { + case 0: + Switch01KNMTFromMirroring(Mirroring.Horz); + break; + case 1: + Switch01KNMTFromMirroring(Mirroring.Vert); + break; + case 2: + Switch01KNMT(14); + break; + case 3: + Switch01KNMTFromMirroring(Mirroring.OneScB); + break; + } + break; + case 3: + break; + } + break; + } + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(addr); + stream.Write(chr_reg); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + addr = stream.ReadInt32(); + chr_reg = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs.meta new file mode 100644 index 00000000..0e971e17 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper243.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d1322eaba98d208409847802768cbbc6 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs new file mode 100644 index 00000000..4d48c1ea --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs @@ -0,0 +1,229 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("C&E", 245, true, true)] + [HassIssues] + internal class Mapper245 : Board + { + private bool flag_c; + + private bool flag_p; + + private int address_8001; + + private int[] chr_reg; + + private int[] prg_reg; + + private int prg_or; + + private bool irq_enabled; + + private byte irq_counter; + + private int old_irq_counter; + + private byte irq_reload; + + private bool irq_clear; + + private bool mmc3_alt_behavior; + + internal override string Issues => MNInterfaceLanguage.IssueMapper245; + + internal override void HardReset() + { + base.HardReset(); + flag_c = (flag_p = false); + address_8001 = 0; + prg_reg = new int[4]; + prg_or = 0; + prg_reg[0] = 0; + prg_reg[1] = 1; + prg_reg[2] = PRG_ROM_08KB_Mask - 1; + prg_reg[3] = PRG_ROM_08KB_Mask; + chr_reg = new int[6]; + for (int i = 0; i < 6; i++) + { + chr_reg[i] = 0; + } + SetupPRG(); + SetupCHR(); + irq_enabled = false; + irq_counter = 0; + irq_reload = byte.MaxValue; + old_irq_counter = 0; + irq_clear = false; + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xE001) + { + case 32768: + address_8001 = data & 7; + flag_c = (data & 0x80) != 0; + flag_p = (data & 0x40) != 0; + SetupCHR(); + SetupPRG(); + break; + case 32769: + switch (address_8001) + { + case 0: + case 1: + case 2: + case 3: + case 4: + case 5: + chr_reg[address_8001] = data; + SetupCHR(); + break; + case 6: + prg_reg[0] = data & 0x3F; + SetupPRG(); + break; + case 7: + prg_reg[1] = data & 0x3F; + SetupPRG(); + break; + } + break; + case 40960: + if (NMT_DEFAULT_MIRROR != Mirroring.Full) + { + Switch01KNMTFromMirroring(((data & 1) == 1) ? Mirroring.Horz : Mirroring.Vert); + } + break; + case 40961: + TogglePRGRAMEnable((data & 0x80) != 0); + TogglePRGRAMWritableEnable((data & 0x40) == 0); + break; + case 49152: + irq_reload = data; + break; + case 49153: + if (mmc3_alt_behavior) + { + irq_clear = true; + } + irq_counter = 0; + break; + case 57344: + irq_enabled = false; + NesEmu.IRQFlags &= -9; + break; + case 57345: + irq_enabled = true; + break; + } + } + + private void SetupCHR() + { + if (CHR_ROM_01KB_Count > 0) + { + if (!flag_c) + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area0000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area0800); + Switch01KCHR(chr_reg[2], CHRArea.Area1000); + Switch01KCHR(chr_reg[3], CHRArea.Area1400); + Switch01KCHR(chr_reg[4], CHRArea.Area1800); + Switch01KCHR(chr_reg[5], CHRArea.Area1C00); + } + else + { + Switch02KCHR(chr_reg[0] >> 1, CHRArea.Area1000); + Switch02KCHR(chr_reg[1] >> 1, CHRArea.Area1800); + Switch01KCHR(chr_reg[2], CHRArea.Area0000); + Switch01KCHR(chr_reg[3], CHRArea.Area0400); + Switch01KCHR(chr_reg[4], CHRArea.Area0800); + Switch01KCHR(chr_reg[5], CHRArea.Area0C00); + } + } + else if (!flag_c) + { + Switch04KCHR(0, CHRArea.Area0000); + Switch04KCHR(1, CHRArea.Area1000); + } + else + { + Switch04KCHR(1, CHRArea.Area0000); + Switch04KCHR(0, CHRArea.Area1000); + } + } + + private void SetupPRG() + { + prg_or = (chr_reg[0] & 2) << 5; + Switch08KPRG(prg_reg[flag_p ? 2 : 0] | prg_or, PRGArea.Area8000); + Switch08KPRG(prg_reg[1] | prg_or, PRGArea.AreaA000); + Switch08KPRG(prg_reg[(!flag_p) ? 2 : 0] | prg_or, PRGArea.AreaC000); + Switch08KPRG(prg_reg[3] | prg_or, PRGArea.AreaE000); + } + + internal override void OnPPUA12RaisingEdge() + { + old_irq_counter = irq_counter; + if (irq_counter == 0 || irq_clear) + { + irq_counter = irq_reload; + } + else + { + irq_counter--; + } + if ((!mmc3_alt_behavior || old_irq_counter != 0 || irq_clear) && irq_counter == 0 && irq_enabled) + { + NesEmu.IRQFlags |= 8; + } + irq_clear = false; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(flag_c); + stream.Write(flag_p); + stream.Write(address_8001); + for (int i = 0; i < chr_reg.Length; i++) + { + stream.Write(chr_reg[i]); + } + for (int j = 0; j < prg_reg.Length; j++) + { + stream.Write(prg_reg[j]); + } + stream.Write(irq_enabled); + stream.Write(irq_counter); + stream.Write(old_irq_counter); + stream.Write(irq_reload); + stream.Write(irq_clear); + stream.Write(prg_or); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + flag_c = stream.ReadBoolean(); + flag_p = stream.ReadBoolean(); + address_8001 = stream.ReadInt32(); + for (int i = 0; i < chr_reg.Length; i++) + { + chr_reg[i] = stream.ReadInt32(); + } + for (int j = 0; j < prg_reg.Length; j++) + { + prg_reg[j] = stream.ReadInt32(); + } + irq_enabled = stream.ReadBoolean(); + irq_counter = stream.ReadByte(); + old_irq_counter = stream.ReadInt32(); + irq_reload = stream.ReadByte(); + irq_clear = stream.ReadBoolean(); + prg_or = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs.meta new file mode 100644 index 00000000..040ac428 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper245.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6ab59a7c8ec0352498d385808f3136a5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs new file mode 100644 index 00000000..71991abf --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs @@ -0,0 +1,50 @@ +namespace MyNes.Core +{ + [BoardInfo("Unknown", 246)] + internal class Mapper246 : Board + { + internal override void HardReset() + { + base.HardReset(); + Switch08KPRG(255, PRGArea.AreaE000); + } + + internal override void WriteSRM(ref ushort address, ref byte data) + { + if (address < 26624) + { + switch (address) + { + case 24576: + Switch08KPRG(data, PRGArea.Area8000); + break; + case 24577: + Switch08KPRG(data, PRGArea.AreaA000); + break; + case 24578: + Switch08KPRG(data, PRGArea.AreaC000); + break; + case 24579: + Switch08KPRG(data, PRGArea.AreaE000); + break; + case 24580: + Switch02KCHR(data, CHRArea.Area0000); + break; + case 24581: + Switch02KCHR(data, CHRArea.Area0800); + break; + case 24582: + Switch02KCHR(data, CHRArea.Area1000); + break; + case 24583: + Switch02KCHR(data, CHRArea.Area1800); + break; + } + } + else + { + base.WriteSRM(ref address, ref data); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs.meta new file mode 100644 index 00000000..46221cae --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper246.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b64643bcf29c76246b9f648dea863909 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs new file mode 100644 index 00000000..b54016c3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs @@ -0,0 +1,66 @@ +using System.IO; + +namespace MyNes.Core +{ + [BoardInfo("Unknown", 255)] + [HassIssues] + internal class Mapper255 : Board + { + private byte[] RAM; + + internal override string Issues => MNInterfaceLanguage.IssueMapper255; + + internal override void HardReset() + { + base.HardReset(); + RAM = new byte[4]; + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + if (address >= 22528) + { + RAM[address & 3] = (byte)(data & 0xFu); + } + } + + internal override void ReadEX(ref ushort address, out byte data) + { + if (address >= 22528) + { + data = RAM[address & 3]; + } + else + { + data = 0; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + Switch08KCHR(address & 0x3F); + if ((address & 0x1000) == 4096) + { + Switch16KPRG((address >> 6) & 0x3F, PRGArea.Area8000); + Switch16KPRG((address >> 6) & 0x3F, PRGArea.AreaC000); + } + else + { + Switch32KPRG(((address >> 6) & 0x3F) >> 1, PRGArea.Area8000); + } + Switch01KNMTFromMirroring(((address & 0x2000) == 8192) ? Mirroring.Horz : Mirroring.Vert); + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(RAM); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + stream.Read(RAM, 0, RAM.Length); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs.meta new file mode 100644 index 00000000..2014a577 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mapper255.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 076b4701a252e1a4aab4efd4c5aede0b +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs new file mode 100644 index 00000000..6f199bca --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs @@ -0,0 +1,4 @@ +namespace MyNes.Core +{ + internal delegate void MemReadAccess(ref ushort addr, out byte value); +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs.meta new file mode 100644 index 00000000..93f9e4f4 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemReadAccess.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 71b28062ea32d1f41bc7a9b990155078 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs new file mode 100644 index 00000000..e63e2dc5 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs @@ -0,0 +1,4 @@ +namespace MyNes.Core +{ + internal delegate void MemWriteAccess(ref ushort addr, ref byte value); +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs.meta new file mode 100644 index 00000000..1b3d5aad --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MemWriteAccess.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: d731f056b42a0974bbee54ffc9333053 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs new file mode 100644 index 00000000..d637bea5 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs @@ -0,0 +1,11 @@ +namespace MyNes.Core +{ + public enum Mirroring : byte + { + Horz = 80, + Vert = 68, + OneScA = 0, + OneScB = 85, + Full = 228 + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs.meta new file mode 100644 index 00000000..c033bb36 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Mirroring.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f22679fb7ecffc440a9aaea88226619d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef new file mode 100644 index 00000000..7690f954 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef @@ -0,0 +1,3 @@ +{ + "name": "MyNes.Core" +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef.meta new file mode 100644 index 00000000..0ab097f9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNes.Core.asmdef.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: 0c194730510bd1b4fad0398ccfe4235b +AssemblyDefinitionImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs new file mode 100644 index 00000000..4a2c4bf1 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs @@ -0,0 +1,339 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Reflection; + +namespace MyNes.Core +{ + public class MyNesMain + { + public static EmuSettings EmuSettings { get; private set; } + + public static RendererSettings RendererSettings { get; private set; } + + public static string AppPath { get; private set; } + + public static string WorkingFolder { get; private set; } + + internal static List Boards { get; private set; } + + public static List VideoProviders { get; private set; } + + public static List AudioProviders { get; private set; } + + public static IVideoProvider VideoProvider { get; private set; } + + public static IAudioProvider AudioProvider { get; private set; } + + public static WaveRecorder WaveRecorder { get; private set; } + + public static void Initialize(bool setupRenderers) + { + Tracer.WriteLine("Initializing My Nes Core ...."); + AppPath = Path.GetDirectoryName(Environment.GetCommandLineArgs()[0]); + if (AppPath == "") + { + AppPath = Path.GetFullPath(".\\"); + } + WorkingFolder = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.Personal), "MyNes"); + Directory.CreateDirectory(WorkingFolder); + Tracer.WriteLine("Loading emu settings ..."); + EmuSettings = new EmuSettings(Path.Combine(WorkingFolder, "emusettings.ini")); + EmuSettings.LoadSettings(); + Tracer.WriteLine("Emu settings loaded successfully."); + Tracer.WriteLine("Loading renderer settings ..."); + RendererSettings = new RendererSettings(Path.Combine(WorkingFolder, "renderersettings.ini")); + RendererSettings.LoadSettings(); + Tracer.WriteLine("Renderer settings loaded successfully."); + Tracer.WriteLine("Locating boards and providers ..."); + WaveRecorder = new WaveRecorder(); + Boards = new List(); + VideoProviders = new List(); + AudioProviders = new List(); + string[] files = Directory.GetFiles(Path.GetDirectoryName(Environment.GetCommandLineArgs()[0]), "*", SearchOption.AllDirectories); + string[] array = files; + foreach (string text in array) + { + try + { + if (!(Path.GetExtension(text).ToLower() == ".exe") && !(Path.GetExtension(text).ToLower() == ".dll")) + { + continue; + } + Tracer.WriteLine("Reading assembly: " + text); + Assembly assembly = Assembly.LoadFile(text); + if (!(assembly != null)) + { + continue; + } + Type[] types = assembly.GetTypes(); + Type[] array2 = types; + foreach (Type type in array2) + { + if (type.IsSubclassOf(typeof(Board)) && !type.IsAbstract) + { + Board board = Activator.CreateInstance(type) as Board; + Boards.Add(board); + Tracer.WriteLine("Board added: " + board.Name + " [ Mapper " + board.MapperNumber + "]"); + } + else if (type.GetInterface("MyNes.Core.IVideoProvider") != null) + { + IVideoProvider videoProvider = Activator.CreateInstance(type) as IVideoProvider; + VideoProviders.Add(videoProvider); + Tracer.WriteLine("Video provider added: " + videoProvider.Name + " [" + videoProvider.ID + "]"); + } + else if (type.GetInterface("MyNes.Core.IAudioProvider") != null) + { + IAudioProvider audioProvider = Activator.CreateInstance(type) as IAudioProvider; + AudioProviders.Add(audioProvider); + Tracer.WriteLine("Audio provider added: " + audioProvider.Name + " [" + audioProvider.ID + "]"); + } + } + } + catch (Exception ex) + { + Tracer.WriteLine("ERROR: " + ex.ToString()); + } + } + Tracer.WriteInformation("Done."); + Tracer.WriteInformation("Total of " + Boards.Count + " board found."); + Tracer.WriteInformation("Total of " + VideoProviders.Count + " video provider found."); + Tracer.WriteInformation("Total of " + AudioProviders.Count + " audio provider found."); + if (setupRenderers) + { + SetVideoProvider(); + SetAudioProvider(); + SetRenderingMethods(); + } + NesEmu.Initialize(); + } + + public static void Shutdown() + { + if (NesEmu.ON) + { + NesEmu.ShutDown(); + } + if (VideoProvider != null) + { + VideoProvider.ShutDown(); + } + if (AudioProvider != null) + { + AudioProvider.ShutDown(); + } + Tracer.WriteLine("Saving settings ..."); + EmuSettings.SaveSettings(); + RendererSettings.SaveSettings(); + Tracer.WriteLine("Settings saved successfully."); + Tracer.WriteLine("Exiting My Nes."); + } + + internal static bool IsBoardExist(int mapper) + { + foreach (Board board in Boards) + { + if (board.MapperNumber == mapper) + { + return true; + } + } + return false; + } + + internal static Board GetBoard(int mapper) + { + foreach (Board board in Boards) + { + if (board.MapperNumber == mapper) + { + return board; + } + } + return null; + } + + public static void MakeWorkingFolder() + { + WorkingFolder = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.Personal), "MyNes"); + Directory.CreateDirectory(WorkingFolder); + } + + public static BoardInfoObject[] GetBoardsList(bool includeUnsupportedMappers) + { + List list = new List(); + if (includeUnsupportedMappers) + { + for (int i = 0; i < 256; i++) + { + bool flag = false; + foreach (Board board in Boards) + { + if (board.MapperNumber == i) + { + BoardInfoObject boardInfoObject = new BoardInfoObject(); + boardInfoObject.Name = board.Name; + boardInfoObject.MapperNumber = board.MapperNumber; + boardInfoObject.IsSupported = true; + boardInfoObject.HasIssues = board.HasIssues; + boardInfoObject.Issues = board.Issues; + list.Add(boardInfoObject); + flag = true; + break; + } + } + if (!flag) + { + BoardInfoObject boardInfoObject2 = new BoardInfoObject(); + boardInfoObject2.Name = "N/A"; + boardInfoObject2.MapperNumber = i; + boardInfoObject2.IsSupported = false; + boardInfoObject2.HasIssues = false; + boardInfoObject2.Issues = ""; + list.Add(boardInfoObject2); + } + } + } + else + { + foreach (Board board2 in Boards) + { + if (board2.MapperNumber >= 0) + { + BoardInfoObject boardInfoObject3 = new BoardInfoObject(); + boardInfoObject3.Name = board2.Name; + boardInfoObject3.MapperNumber = board2.MapperNumber; + boardInfoObject3.IsSupported = true; + boardInfoObject3.HasIssues = board2.HasIssues; + boardInfoObject3.Issues = board2.Issues; + list.Add(boardInfoObject3); + } + } + } + return list.ToArray(); + } + + public static IVideoProvider GetVideoProvider(string id) + { + foreach (IVideoProvider videoProvider in VideoProviders) + { + if (videoProvider.ID == id) + { + return videoProvider; + } + } + return null; + } + + public static IAudioProvider GetAudioProvider(string id) + { + foreach (IAudioProvider audioProvider in AudioProviders) + { + if (audioProvider.ID == id) + { + return audioProvider; + } + } + return null; + } + + public static void SetVideoProvider() + { + Tracer.WriteLine("Looking for the video provider that set in the settings..."); + VideoProvider = GetVideoProvider(RendererSettings.Video_ProviderID); + if (VideoProvider == null) + { + Tracer.WriteError("ERROR: cannot find the video provider that set in the settings"); + Tracer.WriteWarning("Deciding video provider"); + if (VideoProviders.Count > 0) + { + RendererSettings.Video_ProviderID = VideoProviders[0].ID; + VideoProvider = VideoProviders[0]; + if (VideoProvider != null) + { + Tracer.WriteInformation("Video provider set to " + VideoProvider.Name + " [" + VideoProvider.ID + "]"); + VideoProvider.Initialize(); + } + else + { + Tracer.WriteError("ERROR: cannot set video provider."); + } + } + else + { + Tracer.WriteError("ERROR: cannot set video provider, no video providers located."); + } + } + else + { + Tracer.WriteInformation("Video provider set to " + VideoProvider.Name + " [" + VideoProvider.ID + "]"); + VideoProvider.Initialize(); + } + } + + public static void SetAudioProvider() + { + Tracer.WriteLine("Looking for the audio provider that set in the settings..."); + AudioProvider = GetAudioProvider(RendererSettings.Audio_ProviderID); + if (AudioProvider == null) + { + Tracer.WriteError("ERROR: cannot find the audio provider that set in the settings"); + Tracer.WriteWarning("Deciding audio provider"); + if (AudioProviders.Count > 0) + { + RendererSettings.Audio_ProviderID = AudioProviders[0].ID; + AudioProvider = AudioProviders[0]; + if (AudioProvider != null) + { + Tracer.WriteInformation("Audio provider set to " + AudioProvider.Name + " [" + AudioProvider.ID + "]"); + AudioProvider.Initialize(); + } + else + { + Tracer.WriteError("ERROR: cannot set audio provider."); + } + } + else + { + Tracer.WriteError("ERROR: cannot set audio provider, no audio providers located."); + } + } + else + { + Tracer.WriteInformation("Audio provider set to " + AudioProvider.Name + " [" + AudioProvider.ID + "]"); + AudioProvider.Initialize(); + } + } + + public static void SetRenderingMethods() + { + if (VideoProvider != null && AudioProvider != null) + { + NesEmu.SetupRenderingMethods(VideoProvider.SubmitFrame, AudioProvider.SubmitSamples, AudioProvider.TogglePause, AudioProvider.GetIsPlaying); + } + else + { + Tracer.WriteError("ERROR: unable to setup rendering methods, one (or both) of the providers is not set (video and/or audio provider)"); + } + } + + public static void RecordWave() + { + if (NesEmu.ON && NesEmu.SoundEnabled) + { + string text = Path.Combine(EmuSettings.WavesFolder, Path.GetFileNameWithoutExtension(NesEmu.CurrentFilePath) + ".wav"); + int num = 0; + while (File.Exists(text)) + { + text = Path.Combine(EmuSettings.WavesFolder, Path.GetFileNameWithoutExtension(NesEmu.CurrentFilePath) + "_" + num + ".wav"); + num++; + } + WaveRecorder.Record(text, 1, 16, RendererSettings.Audio_Frequency); + } + else if (VideoProvider != null) + { + VideoProvider.WriteErrorNotification("Cannot record sound when the emu is off/sound is not enabled.", instant: false); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs.meta new file mode 100644 index 00000000..7be0a547 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/MyNesMain.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2ef58a78e1bd9734bb820da2c5e1e471 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs new file mode 100644 index 00000000..7635f4cc --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs @@ -0,0 +1,10 @@ +namespace MyNes.Core +{ + internal enum NTArea : byte + { + Area2000NT0, + Area2400NT1, + Area2800NT2, + Area2C00NT3 + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs.meta new file mode 100644 index 00000000..13480cbd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NTArea.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c3f55745b33b3de4992cb06a69a1bbce +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs new file mode 100644 index 00000000..b5f0558d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs @@ -0,0 +1,553 @@ +using System.IO; + +namespace MyNes.Core +{ + [WithExternalSound] + internal abstract class Namcot106 : Board + { + private int irq_counter; + + private bool irq_enable; + + private bool disables_chr_ram_A; + + private bool disables_chr_ram_B; + + private bool enable_mirroring_switch; + + private bool enable_N106_sound; + + private int temp_nmt; + + private Namcot106Chnl[] sound_channels; + + private byte soundReg; + + public int enabledChannels; + + private int enabledChannels1; + + private int channelIndex; + + private byte temp_val; + + private byte temp_i; + + public byte[] EXRAM; + + private bool[] sound_channels_enable; + + private double soundOut; + + private int sound_out_div; + + internal override void HardReset() + { + base.HardReset(); + EXRAM = new byte[128]; + Switch08KPRG(PRG_ROM_08KB_Mask, PRGArea.AreaE000); + enable_mirroring_switch = (enable_N106_sound = base.MapperNumber == 19); + switch (SHA1.ToUpper()) + { + case "97E7E61EECB73CB1EA0C15AE51E65EA56301A685": + case "3D554F55411AB2DDD1A87E7583E643970DB784F3": + case "7FA51058307DB50825C2D3A3A98C0DA554BC3C92": + case "1C476C795CFC17E987C22FFD6F09BAF1396ED2C9": + enable_mirroring_switch = false; + enable_N106_sound = false; + break; + } + if (enable_N106_sound) + { + sound_channels = new Namcot106Chnl[8]; + sound_channels_enable = new bool[8]; + for (int i = 0; i < 8; i++) + { + sound_channels[i] = new Namcot106Chnl(this); + sound_channels[i].HardReset(); + } + soundReg = 0; + enabledChannels = 0; + channelIndex = 0; + APUApplyChannelsSettings(); + } + } + + internal override void WriteEX(ref ushort address, ref byte data) + { + switch (address & 0xF800) + { + case 18432: + if (soundReg >= 64) + { + switch (soundReg & 0x7F) + { + case 64: + sound_channels[0].WriteA(ref data); + break; + case 66: + sound_channels[0].WriteB(ref data); + break; + case 68: + sound_channels[0].WriteC(ref data); + break; + case 70: + sound_channels[0].WriteD(ref data); + break; + case 71: + sound_channels[0].WriteE(ref data); + break; + case 72: + sound_channels[1].WriteA(ref data); + break; + case 74: + sound_channels[1].WriteB(ref data); + break; + case 76: + sound_channels[1].WriteC(ref data); + break; + case 78: + sound_channels[1].WriteD(ref data); + break; + case 79: + sound_channels[1].WriteE(ref data); + break; + case 80: + sound_channels[2].WriteA(ref data); + break; + case 82: + sound_channels[2].WriteB(ref data); + break; + case 84: + sound_channels[2].WriteC(ref data); + break; + case 86: + sound_channels[2].WriteD(ref data); + break; + case 87: + sound_channels[2].WriteE(ref data); + break; + case 88: + sound_channels[3].WriteA(ref data); + break; + case 90: + sound_channels[3].WriteB(ref data); + break; + case 92: + sound_channels[3].WriteC(ref data); + break; + case 94: + sound_channels[3].WriteD(ref data); + break; + case 95: + sound_channels[3].WriteE(ref data); + break; + case 96: + sound_channels[4].WriteA(ref data); + break; + case 98: + sound_channels[4].WriteB(ref data); + break; + case 100: + sound_channels[4].WriteC(ref data); + break; + case 102: + sound_channels[4].WriteD(ref data); + break; + case 103: + sound_channels[4].WriteE(ref data); + break; + case 104: + sound_channels[5].WriteA(ref data); + break; + case 106: + sound_channels[5].WriteB(ref data); + break; + case 108: + sound_channels[5].WriteC(ref data); + break; + case 110: + sound_channels[5].WriteD(ref data); + break; + case 111: + sound_channels[5].WriteE(ref data); + break; + case 112: + sound_channels[6].WriteA(ref data); + break; + case 114: + sound_channels[6].WriteB(ref data); + break; + case 116: + sound_channels[6].WriteC(ref data); + break; + case 118: + sound_channels[6].WriteD(ref data); + break; + case 119: + sound_channels[6].WriteE(ref data); + break; + case 120: + sound_channels[7].WriteA(ref data); + break; + case 122: + sound_channels[7].WriteB(ref data); + break; + case 124: + sound_channels[7].WriteC(ref data); + break; + case 126: + sound_channels[7].WriteD(ref data); + break; + case 127: + sound_channels[7].WriteE(ref data); + enabledChannels = (data & 0x70) >> 4; + channelIndex = 0; + enabledChannels1 = enabledChannels + 1; + temp_i = 7; + while (temp_i >= 0 && enabledChannels1 > 0) + { + sound_channels[temp_i].Enabled = true; + enabledChannels1--; + temp_i--; + } + break; + } + } + EXRAM[soundReg & 0x7F] = data; + if ((soundReg & 0x80) == 128) + { + soundReg = (byte)(((uint)(soundReg + 1) & 0x7Fu) | 0x80u); + } + break; + case 20480: + NesEmu.IRQFlags &= -9; + irq_counter = (irq_counter & 0x7F00) | data; + break; + case 22528: + NesEmu.IRQFlags &= -9; + irq_counter = (irq_counter & 0xFF) | ((data & 0x7F) << 8); + irq_enable = (data & 0x80) == 128; + break; + } + } + + internal override void ReadEX(ref ushort address, out byte value) + { + switch (address & 0xF800) + { + case 18432: + value = EXRAM[soundReg & 0x7F]; + if ((soundReg & 0x80) == 128) + { + soundReg = (byte)(((uint)(soundReg + 1) & 0x7Fu) | 0x80u); + } + break; + case 20480: + NesEmu.IRQFlags &= -9; + value = (byte)((uint)irq_counter & 0xFFu); + break; + case 22528: + NesEmu.IRQFlags &= -9; + value = (byte)((irq_enable ? 128u : 0u) | (uint)((irq_counter & 0x7F00) >> 8)); + break; + default: + value = 0; + break; + } + } + + internal override void WritePRG(ref ushort address, ref byte data) + { + switch (address & 0xF800) + { + case 32768: + if (!disables_chr_ram_A) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area0000); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area0000); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area0000); + Switch01KCHR(data, CHRArea.Area0000); + } + break; + case 34816: + if (!disables_chr_ram_A) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area0400); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area0400); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area0400); + Switch01KCHR(data, CHRArea.Area0400); + } + break; + case 36864: + if (!disables_chr_ram_A) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area0800); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area0800); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area0800); + Switch01KCHR(data, CHRArea.Area0800); + } + break; + case 38912: + if (!disables_chr_ram_A) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area0C00); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area0C00); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area0C00); + Switch01KCHR(data, CHRArea.Area0C00); + } + break; + case 40960: + if (!disables_chr_ram_B) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area1000); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area1000); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area1000); + Switch01KCHR(data, CHRArea.Area1000); + } + break; + case 43008: + if (!disables_chr_ram_B) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area1400); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area1400); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area1400); + Switch01KCHR(data, CHRArea.Area1400); + } + break; + case 45056: + if (!disables_chr_ram_B) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area1800); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area1800); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area1800); + Switch01KCHR(data, CHRArea.Area1800); + } + break; + case 47104: + if (!disables_chr_ram_B) + { + Toggle01KCHR_RAM(data >= 224, CHRArea.Area1C00); + Switch01KCHR((data >= 224) ? (data - 224) : data, CHRArea.Area1C00); + } + else + { + Toggle01KCHR_RAM(ram: false, CHRArea.Area1C00); + Switch01KCHR(data, CHRArea.Area1C00); + } + break; + case 49152: + if (enable_mirroring_switch) + { + NMT_AREA_BLK_INDEX[0] = data; + } + break; + case 51200: + if (enable_mirroring_switch) + { + NMT_AREA_BLK_INDEX[1] = data; + } + break; + case 53248: + if (enable_mirroring_switch) + { + NMT_AREA_BLK_INDEX[2] = data; + } + break; + case 55296: + if (enable_mirroring_switch) + { + NMT_AREA_BLK_INDEX[3] = data; + } + break; + case 57344: + Switch08KPRG(data & 0x3F, PRGArea.Area8000); + break; + case 59392: + Switch08KPRG(data & 0x3F, PRGArea.AreaA000); + disables_chr_ram_A = (data & 0x40) == 64; + disables_chr_ram_B = (data & 0x80) == 128; + break; + case 61440: + Switch08KPRG(data & 0x3F, PRGArea.AreaC000); + break; + case 63488: + soundReg = data; + break; + } + } + + internal override void ReadNMT(ref ushort address, out byte data) + { + if (enable_mirroring_switch) + { + temp_nmt = NMT_AREA_BLK_INDEX[(address >> 10) & 3]; + if (temp_nmt >= 224) + { + data = NMT_RAM[(temp_nmt - 224) & 1][address & 0x3FF]; + } + else + { + data = CHR_ROM[temp_nmt][address & 0x3FF]; + } + } + else + { + base.ReadNMT(ref address, out data); + } + } + + internal override void WriteNMT(ref ushort address, ref byte data) + { + if (enable_mirroring_switch) + { + temp_nmt = NMT_AREA_BLK_INDEX[(address >> 10) & 3]; + if (temp_nmt >= 224) + { + NMT_RAM[(temp_nmt - 224) & 1][address & 0x3FF] = data; + } + } + else + { + base.WriteNMT(ref address, ref data); + } + } + + internal override void OnCPUClock() + { + if (irq_enable) + { + if (irq_counter == 32767) + { + NesEmu.IRQFlags |= 8; + irq_counter = 0; + } + else + { + irq_counter++; + } + } + } + + internal override void OnAPUClockSingle() + { + if (sound_channels != null) + { + for (int i = 0; i < sound_channels.Length; i++) + { + sound_channels[i].ClockSingle(); + } + } + } + + internal override double APUGetSample() + { + soundOut = 0.0; + sound_out_div = 0; + if (enabledChannels > 0) + { + for (int i = 0; i < sound_channels.Length; i++) + { + if (sound_channels[i].Enabled && sound_channels_enable[i]) + { + if (sound_channels[i].clocks > 0) + { + sound_channels[i].output = sound_channels[i].output_av / sound_channels[i].clocks; + } + sound_channels[i].clocks = (sound_channels[i].output_av = 0); + soundOut += sound_channels[i].output; + sound_out_div++; + } + } + soundOut = soundOut / 8.0 / 225.0; + } + return soundOut; + } + + internal override void APUApplyChannelsSettings() + { + base.APUApplyChannelsSettings(); + sound_channels_enable[0] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT1; + sound_channels_enable[1] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT2; + sound_channels_enable[2] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT3; + sound_channels_enable[3] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT4; + sound_channels_enable[4] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT5; + sound_channels_enable[5] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT6; + sound_channels_enable[6] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT7; + sound_channels_enable[7] = MyNesMain.RendererSettings.Audio_ChannelEnabled_NMT8; + } + + internal override void WriteStateData(ref BinaryWriter stream) + { + base.WriteStateData(ref stream); + stream.Write(irq_counter); + stream.Write(irq_enable); + stream.Write(disables_chr_ram_A); + stream.Write(disables_chr_ram_B); + stream.Write(enable_mirroring_switch); + stream.Write(enable_N106_sound); + stream.Write(temp_nmt); + if (enable_N106_sound) + { + for (int i = 0; i < sound_channels.Length; i++) + { + sound_channels[i].SaveState(stream); + } + } + stream.Write(soundReg); + stream.Write(enabledChannels); + stream.Write(enabledChannels1); + stream.Write(channelIndex); + stream.Write(temp_val); + stream.Write(temp_i); + stream.Write(EXRAM); + } + + internal override void ReadStateData(ref BinaryReader stream) + { + base.ReadStateData(ref stream); + irq_counter = stream.ReadInt32(); + irq_enable = stream.ReadBoolean(); + disables_chr_ram_A = stream.ReadBoolean(); + disables_chr_ram_B = stream.ReadBoolean(); + enable_mirroring_switch = stream.ReadBoolean(); + enable_N106_sound = stream.ReadBoolean(); + temp_nmt = stream.ReadInt32(); + if (enable_N106_sound) + { + for (int i = 0; i < sound_channels.Length; i++) + { + sound_channels[i].LoadState(stream); + } + } + soundReg = stream.ReadByte(); + enabledChannels = stream.ReadInt32(); + enabledChannels1 = stream.ReadInt32(); + channelIndex = stream.ReadInt32(); + temp_val = stream.ReadByte(); + temp_i = stream.ReadByte(); + stream.Read(EXRAM, 0, EXRAM.Length); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs.meta new file mode 100644 index 00000000..c0890af0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: dd8e287326f773a47afba9c9ca5dac87 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs new file mode 100644 index 00000000..ca27e691 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs @@ -0,0 +1,157 @@ +using System.IO; + +namespace MyNes.Core +{ + internal class Namcot106Chnl + { + private Namcot106 namcot; + + private int freqTimer; + + private int frequency; + + public int output; + + public int output_av; + + public int clocks; + + private int cycles; + + private int InstrumentLength; + + private byte InstrumentAddress; + + private int startPoint; + + private int endPoint; + + private int readPoint; + + public int volume; + + private bool freez; + + public bool Enabled { get; set; } + + public Namcot106Chnl(Namcot106 namcot) + { + this.namcot = namcot; + } + + public void HardReset() + { + } + + private void UpdateFrequency() + { + if (frequency > 0) + { + freqTimer = 983040 * (namcot.enabledChannels + 1) / frequency; + freez = false; + } + else + { + freez = true; + output_av = 0; + } + } + + private void UpdatePlaybackParameters() + { + startPoint = InstrumentAddress; + endPoint = InstrumentAddress + 4 * (8 - InstrumentLength); + readPoint = InstrumentAddress; + } + + public void WriteA(ref byte data) + { + frequency = (frequency & 0xFFFF00) | data; + UpdateFrequency(); + } + + public void WriteB(ref byte data) + { + frequency = (frequency & 0xFF00FF) | (data << 8); + UpdateFrequency(); + } + + public void WriteC(ref byte data) + { + frequency = (frequency & 0xFFFF) | ((data & 3) << 12); + InstrumentLength = (data >> 2) & 7; + UpdateFrequency(); + UpdatePlaybackParameters(); + } + + public void WriteD(ref byte data) + { + InstrumentAddress = data; + UpdatePlaybackParameters(); + } + + public void WriteE(ref byte data) + { + volume = data & 0xF; + } + + public void ClockSingle() + { + if (freez || --cycles > 0) + { + return; + } + cycles = freqTimer; + if (readPoint >= startPoint && readPoint <= endPoint) + { + if (Enabled && !freez) + { + if ((readPoint & 1) == 0) + { + output_av += (namcot.EXRAM[readPoint] & 0xF) * volume; + } + else + { + output_av += ((namcot.EXRAM[readPoint] >> 4) & 0xF) * volume; + } + } + readPoint++; + } + else + { + readPoint = startPoint; + } + clocks++; + } + + public void SaveState(BinaryWriter stream) + { + stream.Write(freqTimer); + stream.Write(frequency); + stream.Write(output); + stream.Write(cycles); + stream.Write(InstrumentLength); + stream.Write(InstrumentAddress); + stream.Write(startPoint); + stream.Write(endPoint); + stream.Write(readPoint); + stream.Write(volume); + stream.Write(freez); + } + + public void LoadState(BinaryReader stream) + { + freqTimer = stream.ReadInt32(); + frequency = stream.ReadInt32(); + output = stream.ReadByte(); + cycles = stream.ReadInt32(); + InstrumentLength = stream.ReadInt32(); + InstrumentAddress = stream.ReadByte(); + startPoint = stream.ReadInt32(); + endPoint = stream.ReadInt32(); + readPoint = stream.ReadInt32(); + volume = stream.ReadInt32(); + freez = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs.meta new file mode 100644 index 00000000..86980e70 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Namcot106Chnl.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 609941ad8b89ba84c81aab729de7eded +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs new file mode 100644 index 00000000..b9f109db --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs @@ -0,0 +1,330 @@ +using System.Collections.Generic; +using System.IO; +using System.Xml; + +namespace MyNes.Core +{ + public class NesCartDatabase + { + private static List _databaseRoms = new List(); + + public static string DBVersion = ""; + + public static string DBConformance = ""; + + public static string DBAgent = ""; + + public static string DBAuthor = ""; + + public static string DBTimeStamp = ""; + + public static bool Ready = false; + + public static List DatabaseRoms => _databaseRoms; + + public static void LoadDatabase(string fileName, out bool success) + { + success = false; + Ready = false; + if (!File.Exists(fileName)) + { + return; + } + _databaseRoms.Clear(); + Stream stream = new FileStream(fileName, FileMode.Open, FileAccess.Read); + XmlReaderSettings xmlReaderSettings = new XmlReaderSettings(); + xmlReaderSettings.DtdProcessing = DtdProcessing.Ignore; + xmlReaderSettings.IgnoreWhitespace = true; + XmlReader xmlReader = XmlReader.Create(stream, xmlReaderSettings); + NesCartDatabaseGameInfo nesCartDatabaseGameInfo = default(NesCartDatabaseGameInfo); + nesCartDatabaseGameInfo.Cartridges = new List(); + nesCartDatabaseGameInfo.Game_AltName = ""; + nesCartDatabaseGameInfo.Game_Catalog = ""; + nesCartDatabaseGameInfo.Game_Class = ""; + nesCartDatabaseGameInfo.Game_Developer = ""; + nesCartDatabaseGameInfo.Game_Name = ""; + nesCartDatabaseGameInfo.Game_Players = ""; + nesCartDatabaseGameInfo.Game_Publisher = ""; + nesCartDatabaseGameInfo.Game_Region = ""; + nesCartDatabaseGameInfo.Game_ReleaseDate = ""; + while (xmlReader.Read()) + { + if ((xmlReader.Name == "xml") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("version")) + { + DBVersion = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("conformance")) + { + DBConformance = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("agent")) + { + DBAgent = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("author")) + { + DBAuthor = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("timestamp")) + { + DBTimeStamp = xmlReader.Value; + } + } + else + { + if (!((xmlReader.Name == "game") & xmlReader.IsStartElement())) + { + continue; + } + nesCartDatabaseGameInfo = default(NesCartDatabaseGameInfo); + if (xmlReader.MoveToAttribute("name")) + { + nesCartDatabaseGameInfo.Game_Name = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("altname")) + { + nesCartDatabaseGameInfo.Game_AltName = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("class")) + { + nesCartDatabaseGameInfo.Game_Class = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("catalog")) + { + nesCartDatabaseGameInfo.Game_Catalog = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("publisher")) + { + nesCartDatabaseGameInfo.Game_Publisher = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("developer")) + { + nesCartDatabaseGameInfo.Game_Developer = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("region")) + { + nesCartDatabaseGameInfo.Game_Region = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("players")) + { + nesCartDatabaseGameInfo.Game_Players = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("date")) + { + nesCartDatabaseGameInfo.Game_ReleaseDate = xmlReader.Value; + } + NesCartDatabaseCartridgeInfo nesCartDatabaseCartridgeInfo = new NesCartDatabaseCartridgeInfo(); + nesCartDatabaseCartridgeInfo.PAD_h = ""; + nesCartDatabaseCartridgeInfo.PAD_v = ""; + nesCartDatabaseCartridgeInfo.PRG_crc = ""; + nesCartDatabaseCartridgeInfo.PRG_name = ""; + nesCartDatabaseCartridgeInfo.PRG_sha1 = ""; + nesCartDatabaseCartridgeInfo.PRG_size = ""; + nesCartDatabaseCartridgeInfo.chip_type = new List(); + nesCartDatabaseCartridgeInfo.CHR_crc = ""; + nesCartDatabaseCartridgeInfo.CHR_name = ""; + nesCartDatabaseCartridgeInfo.CHR_sha1 = ""; + nesCartDatabaseCartridgeInfo.CHR_size = ""; + nesCartDatabaseCartridgeInfo.CIC_type = ""; + nesCartDatabaseCartridgeInfo.Board_Mapper = ""; + nesCartDatabaseCartridgeInfo.Board_Pcb = ""; + nesCartDatabaseCartridgeInfo.Board_Type = ""; + nesCartDatabaseCartridgeInfo.VRAM_sizes = new List(); + nesCartDatabaseCartridgeInfo.WRAMBanks = new List(); + while (xmlReader.Read()) + { + if ((xmlReader.Name == "game") & !xmlReader.IsStartElement()) + { + _databaseRoms.Add(nesCartDatabaseGameInfo); + break; + } + if ((xmlReader.Name == "cartridge") & xmlReader.IsStartElement()) + { + if (nesCartDatabaseGameInfo.Cartridges == null) + { + nesCartDatabaseGameInfo.Cartridges = new List(); + } + nesCartDatabaseCartridgeInfo = new NesCartDatabaseCartridgeInfo(); + nesCartDatabaseCartridgeInfo.PAD_h = ""; + nesCartDatabaseCartridgeInfo.PAD_v = ""; + nesCartDatabaseCartridgeInfo.PRG_crc = ""; + nesCartDatabaseCartridgeInfo.PRG_name = ""; + nesCartDatabaseCartridgeInfo.PRG_sha1 = ""; + nesCartDatabaseCartridgeInfo.PRG_size = ""; + nesCartDatabaseCartridgeInfo.chip_type = new List(); + nesCartDatabaseCartridgeInfo.CHR_crc = ""; + nesCartDatabaseCartridgeInfo.CHR_name = ""; + nesCartDatabaseCartridgeInfo.CHR_sha1 = ""; + nesCartDatabaseCartridgeInfo.CHR_size = ""; + nesCartDatabaseCartridgeInfo.CIC_type = ""; + nesCartDatabaseCartridgeInfo.Board_Mapper = ""; + nesCartDatabaseCartridgeInfo.Board_Pcb = ""; + nesCartDatabaseCartridgeInfo.Board_Type = ""; + nesCartDatabaseCartridgeInfo.VRAM_sizes = new List(); + nesCartDatabaseCartridgeInfo.WRAMBanks = new List(); + if (xmlReader.MoveToAttribute("system")) + { + nesCartDatabaseCartridgeInfo.System = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("crc")) + { + nesCartDatabaseCartridgeInfo.CRC = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("sha1")) + { + nesCartDatabaseCartridgeInfo.SHA1 = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("dump")) + { + nesCartDatabaseCartridgeInfo.Dump = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("dumper")) + { + nesCartDatabaseCartridgeInfo.Dumper = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("datedumped")) + { + nesCartDatabaseCartridgeInfo.DateDumped = xmlReader.Value; + } + } + else if ((xmlReader.Name == "cartridge") & !xmlReader.IsStartElement()) + { + nesCartDatabaseGameInfo.Cartridges.Add(nesCartDatabaseCartridgeInfo); + } + else if ((xmlReader.Name == "board") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("type")) + { + nesCartDatabaseCartridgeInfo.Board_Type = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("pcb")) + { + nesCartDatabaseCartridgeInfo.Board_Pcb = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("mapper")) + { + nesCartDatabaseCartridgeInfo.Board_Mapper = xmlReader.Value; + } + } + else if ((xmlReader.Name == "prg") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("name")) + { + nesCartDatabaseCartridgeInfo.PRG_name = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("size")) + { + nesCartDatabaseCartridgeInfo.PRG_size = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("crc")) + { + nesCartDatabaseCartridgeInfo.PRG_crc = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("sha1")) + { + nesCartDatabaseCartridgeInfo.PRG_sha1 = xmlReader.Value; + } + } + else if ((xmlReader.Name == "chr") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("name")) + { + nesCartDatabaseCartridgeInfo.CHR_name = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("size")) + { + nesCartDatabaseCartridgeInfo.CHR_size = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("crc")) + { + nesCartDatabaseCartridgeInfo.CHR_crc = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("sha1")) + { + nesCartDatabaseCartridgeInfo.CHR_sha1 = xmlReader.Value; + } + } + else if ((xmlReader.Name == "vram") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("size")) + { + nesCartDatabaseCartridgeInfo.VRAM_sizes.Add(xmlReader.Value); + } + } + else if ((xmlReader.Name == "wram") & xmlReader.IsStartElement()) + { + string sIZE = ""; + bool bATTERY = false; + int result = 0; + if (xmlReader.MoveToAttribute("size")) + { + sIZE = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("battery")) + { + bATTERY = xmlReader.Value == "1"; + } + if (xmlReader.MoveToAttribute("id")) + { + int.TryParse(xmlReader.Value, out result); + } + nesCartDatabaseCartridgeInfo.WRAMBanks.Add(new SRAMBankInfo(result, sIZE, bATTERY)); + } + else if ((xmlReader.Name == "chip") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("type")) + { + if (nesCartDatabaseCartridgeInfo.chip_type == null) + { + nesCartDatabaseCartridgeInfo.chip_type = new List(); + } + nesCartDatabaseCartridgeInfo.chip_type.Add(xmlReader.Value); + } + } + else if ((xmlReader.Name == "cic") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("type")) + { + nesCartDatabaseCartridgeInfo.CIC_type = xmlReader.Value; + } + } + else if ((xmlReader.Name == "pad") & xmlReader.IsStartElement()) + { + if (xmlReader.MoveToAttribute("h")) + { + nesCartDatabaseCartridgeInfo.PAD_h = xmlReader.Value; + } + if (xmlReader.MoveToAttribute("v")) + { + nesCartDatabaseCartridgeInfo.PAD_v = xmlReader.Value; + } + } + } + } + } + Ready = true; + success = true; + xmlReader.Close(); + stream.Close(); + } + + public static NesCartDatabaseGameInfo Find(string Cart_sha1, out bool found) + { + found = false; + foreach (NesCartDatabaseGameInfo databaseRom in _databaseRoms) + { + foreach (NesCartDatabaseCartridgeInfo cartridge in databaseRom.Cartridges) + { + if (cartridge.SHA1.ToLower() == Cart_sha1.ToLower()) + { + found = true; + return databaseRom; + } + } + } + return default(NesCartDatabaseGameInfo); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs.meta new file mode 100644 index 00000000..513535bb --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabase.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 3e7570468c0a69b41b986d9624f1e8ac +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs new file mode 100644 index 00000000..49dd6bdb --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs @@ -0,0 +1,53 @@ +using System.Collections.Generic; + +namespace MyNes.Core +{ + public class NesCartDatabaseCartridgeInfo + { + public string System; + + public string CRC; + + public string SHA1; + + public string Dump; + + public string Dumper; + + public string DateDumped; + + public string Board_Type; + + public string Board_Pcb; + + public string Board_Mapper; + + public List VRAM_sizes; + + public List WRAMBanks; + + public string PRG_name; + + public string PRG_size; + + public string PRG_crc; + + public string PRG_sha1; + + public string CHR_name; + + public string CHR_size; + + public string CHR_crc; + + public string CHR_sha1; + + public List chip_type; + + public string CIC_type; + + public string PAD_h; + + public string PAD_v; + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs.meta new file mode 100644 index 00000000..115630f3 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseCartridgeInfo.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 6bd7965d8d1e7144995a612d1a42d260 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs new file mode 100644 index 00000000..00f4376c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs @@ -0,0 +1,29 @@ +using System.Collections.Generic; + +namespace MyNes.Core +{ + public struct NesCartDatabaseGameInfo + { + public string Game_Name; + + public string Game_AltName; + + public string Game_Class; + + public string Game_Catalog; + + public string Game_Publisher; + + public string Game_Developer; + + public string Game_Region; + + public string Game_Players; + + public string Game_ReleaseDate; + + public List Cartridges; + + public static NesCartDatabaseGameInfo Empty => default(NesCartDatabaseGameInfo); + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs.meta new file mode 100644 index 00000000..ea2beb7d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesCartDatabaseGameInfo.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 304f3d587a65dad48a326a1431d975e1 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs new file mode 100644 index 00000000..d6bfc264 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs @@ -0,0 +1,5939 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.IO; +using System.Runtime.InteropServices; +using System.Threading; +using System.Xml; + +namespace MyNes.Core +{ + public class NesEmu + { + [StructLayout(LayoutKind.Explicit)] + private struct CPURegister + { + [FieldOffset(0)] + internal byte l; + + [FieldOffset(1)] + internal byte h; + + [FieldOffset(0)] + internal ushort v; + } + + private enum RequestMode + { + None, + HardReset, + SoftReset, + LoadState, + SaveState, + TakeSnapshot + } + + private static int[][] dmc_freq_table = new int[3][] + { + new int[16] + { + 428, 380, 340, 320, 286, 254, 226, 214, 190, 160, + 142, 128, 106, 84, 72, 54 + }, + new int[16] + { + 398, 354, 316, 298, 276, 236, 210, 198, 176, 148, + 132, 118, 98, 78, 66, 50 + }, + new int[16] + { + 428, 380, 340, 320, 286, 254, 226, 214, 190, 160, + 142, 128, 106, 84, 72, 54 + } + }; + + private static int dmc_output_a; + + private static int dmc_output; + + private static int dmc_period_devider; + + private static bool dmc_irq_enabled; + + private static bool dmc_loop_flag; + + private static byte dmc_rate_index; + + private static ushort dmc_addr_refresh; + + private static int dmc_size_refresh; + + private static bool dmc_dmaEnabled; + + private static byte dmc_dmaByte; + + private static int dmc_dmaBits; + + private static bool dmc_bufferFull; + + private static byte dmc_dmaBuffer; + + private static int dmc_dmaSize; + + private static ushort dmc_dmaAddr; + + private static ushort[][] nos_freq_table = new ushort[3][] + { + new ushort[16] + { + 4, 8, 16, 32, 64, 96, 128, 160, 202, 254, + 380, 508, 762, 1016, 2034, 4068 + }, + new ushort[16] + { + 4, 7, 14, 30, 60, 88, 118, 148, 188, 236, + 354, 472, 708, 944, 1890, 3778 + }, + new ushort[16] + { + 4, 8, 16, 32, 64, 96, 128, 160, 202, 254, + 380, 508, 762, 1016, 2034, 4068 + } + }; + + private static bool nos_length_halt; + + private static bool nos_constant_volume_envelope; + + private static byte nos_volume_devider_period; + + private static ushort nos_timer; + + private static bool nos_mode; + + private static int nos_period_devider; + + private static bool nos_length_enabled; + + private static int nos_length_counter; + + private static bool nos_envelope_start_flag; + + private static byte nos_envelope_devider; + + private static byte nos_envelope_decay_level_counter; + + private static byte nos_envelope; + + private static int nos_output; + + private static int nos_shift_reg; + + private static int nos_feedback; + + private static bool nos_ignore_reload; + + private static byte[][] sq_duty_cycle_sequences = new byte[4][] + { + new byte[8] { 0, 0, 0, 0, 0, 0, 0, 1 }, + new byte[8] { 0, 0, 0, 0, 0, 0, 1, 1 }, + new byte[8] { 0, 0, 0, 0, 1, 1, 1, 1 }, + new byte[8] { 1, 1, 1, 1, 1, 1, 0, 0 } + }; + + private static byte[] sq_duration_table = new byte[32] + { + 10, 254, 20, 2, 40, 4, 80, 6, 160, 8, + 60, 10, 14, 12, 26, 14, 12, 16, 24, 18, + 48, 20, 96, 22, 192, 24, 72, 26, 16, 28, + 32, 30 + }; + + private static byte sq1_duty_cycle; + + private static bool sq1_length_halt; + + private static bool sq1_constant_volume_envelope; + + private static byte sq1_volume_devider_period; + + private static bool sq1_sweep_enable; + + private static byte sq1_sweep_devider_period; + + private static bool sq1_sweep_negate; + + private static byte sq1_sweep_shift_count; + + private static int sq1_timer; + + private static int sq1_period_devider; + + private static byte sq1_seqencer; + + private static bool sq1_length_enabled; + + private static int sq1_length_counter; + + private static bool sq1_envelope_start_flag; + + private static byte sq1_envelope_devider; + + private static byte sq1_envelope_decay_level_counter; + + private static byte sq1_envelope; + + private static int sq1_sweep_counter; + + private static bool sq1_sweep_reload; + + private static int sq1_sweep_change; + + private static bool sq1_valid_freq; + + private static int sq1_output; + + private static bool sq1_ignore_reload; + + private static byte[] trl_step_seq = new byte[32] + { + 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, + 5, 4, 3, 2, 1, 0, 0, 1, 2, 3, + 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, + 14, 15 + }; + + private static bool trl_liner_control_flag; + + private static byte trl_liner_control_reload; + + private static ushort trl_timer; + + private static bool trl_length_enabled; + + private static byte trl_length_counter; + + private static bool trl_liner_control_reload_flag; + + private static byte trl_liner_counter; + + private static int trl_output; + + private static int trl_period_devider; + + private static int trl_step; + + private static bool trl_ignore_reload; + + private static byte apu_reg_io_db; + + private static byte apu_reg_io_addr; + + private static bool apu_reg_access_happened; + + private static bool apu_reg_access_w; + + private static Action[] apu_reg_update_func; + + private static Action[] apu_reg_read_func; + + private static Action[] apu_reg_write_func; + + private static Action apu_update_playback_func; + + private static bool apu_odd_cycle; + + private static bool apu_irq_enabled; + + private static bool apu_irq_flag; + + private static bool apu_irq_do_it; + + internal static bool apu_irq_delta_occur; + + private static bool apu_seq_mode; + + private static int apu_ferq_f; + + private static int apu_ferq_l; + + private static int apu_ferq_e; + + private static int apu_cycle_f; + + private static int apu_cycle_f_t; + + private static int apu_cycle_e; + + private static int apu_cycle_l; + + private static bool apu_odd_l; + + private static bool apu_check_irq; + + private static bool apu_do_env; + + private static bool apu_do_length; + + public static bool SoundEnabled; + + public static double audio_playback_amplitude = 1.5; + + public static int audio_playback_peek_limit = 124; + + private static bool audio_playback_dac_initialized; + + public static double cpu_speed; + + public static double cpu_clock_per_frame; + + internal static double apu_target_samples_count_per_frame; + + private static short[] audio_samples; + + private static int audio_w_pos; + + private static int audio_samples_added; + + internal static int audio_samples_count; + + private static int[][][][][] mix_table; + + private static double audio_x; + + private static double audio_y; + + private static double audio_y_av; + + private static double audio_y_timer; + + public static double audio_timer_ratio = 40.0; + + private static double audio_timer; + + private static SoundLowPassFilter audio_low_pass_filter_14K; + + private static SoundHighPassFilter audio_high_pass_filter_90; + + private static SoundHighPassFilter audio_high_pass_filter_440; + + private static SoundDCBlockerFilter audio_dc_blocker_filter; + + private static bool audio_sq1_outputable; + + private static bool audio_sq2_outputable; + + private static bool audio_nos_outputable; + + private static bool audio_trl_outputable; + + private static bool audio_dmc_outputable; + + private static bool audio_signal_outputed; + + private static bool apu_use_external_sound; + + private static CPURegister cpu_reg_pc; + + private static CPURegister cpu_reg_sp; + + private static CPURegister cpu_reg_ea; + + private static byte cpu_reg_a; + + private static byte cpu_reg_x; + + private static byte cpu_reg_y; + + private static bool cpu_flag_n; + + private static bool cpu_flag_v; + + private static bool cpu_flag_d; + + private static bool cpu_flag_i; + + private static bool cpu_flag_z; + + private static bool cpu_flag_c; + + private static byte cpu_m; + + private static byte cpu_opcode; + + private static byte cpu_byte_temp; + + private static int cpu_int_temp; + + private static int cpu_int_temp1; + + private static byte cpu_dummy; + + private static bool cpu_bool_tmp; + + private static CPURegister temp_add; + + private static bool CPU_IRQ_PIN; + + private static bool CPU_NMI_PIN; + + private static bool cpu_suspend_nmi; + + private static bool cpu_suspend_irq; + + private static Action[] cpu_addressings; + + private static Action[] cpu_instructions; + + private static int dma_DMCDMAWaitCycles; + + private static int dma_OAMDMAWaitCycles; + + private static bool dma_isOamDma; + + private static int dma_oamdma_i; + + private static bool dma_DMCOn; + + private static bool dma_OAMOn; + + private static bool dma_DMC_occurring; + + private static bool dma_OAM_occurring; + + private static int dma_OAMFinishCounter; + + private static ushort dma_Oamaddress; + + private static int dma_OAMCYCLE; + + private static byte dma_latch; + + private static byte dma_dummy; + + private static ushort reg_2004; + + internal static int IRQFlags = 0; + + private static bool PPU_NMI_Current; + + private static bool PPU_NMI_Old; + + private const int IRQ_APU = 1; + + internal const int IRQ_DMC = 2; + + internal const int IRQ_BOARD = 8; + + private static ushort InterruptVector; + + private static byte[] mem_wram; + + internal static Board mem_board; + + private static MemReadAccess[] mem_read_accesses; + + private static MemWriteAccess[] mem_write_accesses; + + private static bool BUS_RW; + + private static ushort BUS_ADDRESS; + + private static string SRAMFileName; + + public static string GMFileName; + + private static int PORT0; + + private static int PORT1; + + private static int inputStrobe; + + private static IJoypadConnecter joypad1; + + private static IJoypadConnecter joypad2; + + private static IJoypadConnecter joypad3; + + private static IJoypadConnecter joypad4; + + private static IShortcutsHandler shortucts; + + public static bool IsFourPlayers; + + private static byte[] reverseLookup = new byte[256] + { + 0, 128, 64, 192, 32, 160, 96, 224, 16, 144, + 80, 208, 48, 176, 112, 240, 8, 136, 72, 200, + 40, 168, 104, 232, 24, 152, 88, 216, 56, 184, + 120, 248, 4, 132, 68, 196, 36, 164, 100, 228, + 20, 148, 84, 212, 52, 180, 116, 244, 12, 140, + 76, 204, 44, 172, 108, 236, 28, 156, 92, 220, + 60, 188, 124, 252, 2, 130, 66, 194, 34, 162, + 98, 226, 18, 146, 82, 210, 50, 178, 114, 242, + 10, 138, 74, 202, 42, 170, 106, 234, 26, 154, + 90, 218, 58, 186, 122, 250, 6, 134, 70, 198, + 38, 166, 102, 230, 22, 150, 86, 214, 54, 182, + 118, 246, 14, 142, 78, 206, 46, 174, 110, 238, + 30, 158, 94, 222, 62, 190, 126, 254, 1, 129, + 65, 193, 33, 161, 97, 225, 17, 145, 81, 209, + 49, 177, 113, 241, 9, 137, 73, 201, 41, 169, + 105, 233, 25, 153, 89, 217, 57, 185, 121, 249, + 5, 133, 69, 197, 37, 165, 101, 229, 21, 149, + 85, 213, 53, 181, 117, 245, 13, 141, 77, 205, + 45, 173, 109, 237, 29, 157, 93, 221, 61, 189, + 125, 253, 3, 131, 67, 195, 35, 163, 99, 227, + 19, 147, 83, 211, 51, 179, 115, 243, 11, 139, + 75, 203, 43, 171, 107, 235, 27, 155, 91, 219, + 59, 187, 123, 251, 7, 135, 71, 199, 39, 167, + 103, 231, 23, 151, 87, 215, 55, 183, 119, 247, + 15, 143, 79, 207, 47, 175, 111, 239, 31, 159, + 95, 223, 63, 191, 127, 255 + }; + + private static Action[] ppu_v_clocks; + + private static Action[] ppu_h_clocks; + + private static Action[] ppu_bkg_fetches; + + private static Action[] ppu_spr_fetches; + + private static Action[] ppu_oam_phases; + + private static int[] ppu_bkg_pixels; + + private static int[] ppu_spr_pixels; + + private static int[] ppu_screen_pixels; + + private static int ppu_clock_h; + + internal static ushort ppu_clock_v; + + private static ushort ppu_clock_vblank_start; + + private static ushort ppu_clock_vblank_end; + + private static bool ppu_use_odd_cycle; + + private static bool ppu_use_odd_swap; + + private static bool ppu_odd_swap_done; + + private static bool ppu_is_nmi_time; + + private static bool ppu_frame_finished; + + private static byte[] ppu_oam_bank; + + private static byte[] ppu_oam_bank_secondary; + + private static byte[] ppu_palette_bank; + + private static byte ppu_reg_io_db; + + private static byte ppu_reg_io_addr; + + private static bool ppu_reg_access_happened; + + private static bool ppu_reg_access_w; + + private static Action[] ppu_reg_update_func; + + private static Action[] ppu_reg_read_func; + + private static byte ppu_reg_2000_vram_address_increament; + + private static ushort ppu_reg_2000_sprite_pattern_table_address_for_8x8_sprites; + + private static ushort ppu_reg_2000_background_pattern_table_address; + + internal static byte ppu_reg_2000_Sprite_size; + + private static bool ppu_reg_2000_VBI; + + private static bool ppu_reg_2001_show_background_in_leftmost_8_pixels_of_screen; + + private static bool ppu_reg_2001_show_sprites_in_leftmost_8_pixels_of_screen; + + private static bool ppu_reg_2001_show_background; + + private static bool ppu_reg_2001_show_sprites; + + private static bool ppu_reg_2001_grayscale; + + private static int ppu_reg_2001_emphasis; + + private static bool ppu_reg_2002_SpriteOverflow; + + private static bool ppu_reg_2002_Sprite0Hit; + + private static bool ppu_reg_2002_VblankStartedFlag; + + private static byte ppu_reg_2003_oam_addr; + + private static ushort ppu_vram_addr; + + private static byte ppu_vram_data; + + private static ushort ppu_vram_addr_temp; + + private static ushort ppu_vram_addr_access_temp; + + private static bool ppu_vram_flip_flop; + + private static byte ppu_vram_finex; + + private static ushort ppu_bkgfetch_nt_addr; + + private static byte ppu_bkgfetch_nt_data; + + private static ushort ppu_bkgfetch_at_addr; + + private static byte ppu_bkgfetch_at_data; + + private static ushort ppu_bkgfetch_lb_addr; + + private static byte ppu_bkgfetch_lb_data; + + private static ushort ppu_bkgfetch_hb_addr; + + private static byte ppu_bkgfetch_hb_data; + + private static int ppu_sprfetch_slot; + + private static byte ppu_sprfetch_y_data; + + private static byte ppu_sprfetch_t_data; + + private static byte ppu_sprfetch_at_data; + + private static byte ppu_sprfetch_x_data; + + private static ushort ppu_sprfetch_lb_addr; + + private static byte ppu_sprfetch_lb_data; + + private static ushort ppu_sprfetch_hb_addr; + + private static byte ppu_sprfetch_hb_data; + + internal static bool ppu_is_sprfetch; + + private static int ppu_bkg_render_i; + + private static int ppu_bkg_render_pos; + + private static int ppu_bkg_render_tmp_val; + + private static int ppu_bkg_current_pixel; + + private static int ppu_spr_current_pixel; + + private static int ppu_current_pixel; + + private static int ppu_render_x; + + private static int ppu_render_y; + + private static byte ppu_oamev_n; + + private static byte ppu_oamev_m; + + private static bool ppu_oamev_compare; + + private static byte ppu_oamev_slot; + + private static byte ppu_fetch_data; + + private static byte ppu_phase_index; + + private static bool ppu_sprite0_should_hit; + + private static int ppu_palette_temp__val; + + private static int ppu_color_temp__level; + + private static int ppu_color_temp__red; + + private static int ppu_color_temp__blue; + + private static int ppu_color_temp__green; + + private static bool ppu_color_temp__e_red; + + private static bool ppu_color_temp__e_blue; + + private static bool ppu_color_temp__e_green; + + private static byte[] ppu_color_temp_blue__sequence = new byte[16] + { + 7, 7, 6, 5, 4, 3, 2, 1, 2, 3, + 4, 5, 6, 0, 0, 0 + }; + + private static byte[] ppu_color_temp_red___sequence = new byte[16] + { + 7, 1, 2, 3, 4, 5, 6, 7, 6, 5, + 4, 3, 2, 0, 0, 0 + }; + + private static byte[] ppu_color_temp_green_sequence = new byte[16] + { + 7, 0, 0, 0, 1, 2, 3, 4, 5, 6, + 7, 6, 5, 0, 0, 0 + }; + + private static int ppu_temp_comparator; + + public static bool ON; + + public static bool PAUSED; + + public static bool isPaused; + + public static string CurrentFilePath; + + public static bool FrameLimiterEnabled; + + private static Thread mainThread; + + private static double fps_time_last; + + private static double fps_time_start; + + private static double fps_time_token; + + private static double fps_time_dead; + + private static double fps_time_period; + + private static double fps_time_frame_time; + + private static double emu_time_target_fps = 60.0; + + private static bool emu_frame_clocking_mode; + + private static bool emu_frame_done; + + private static bool render_initialized; + + private static RenderVideoFrame render_video; + + private static RenderAudioSamples render_audio; + + private static TogglePause render_audio_toggle_pause; + + private static GetIsPlaying render_audio_get_is_playing; + + private static bool render_audio_is_playing; + + public static EmuRegion Region; + + internal static Action EmuClockComponents; + + internal static int pal_add_cycle = 0; + + private static int SystemIndex; + + private static RequestMode emu_request_mode = RequestMode.None; + + public static bool FrameSkipEnabled; + + public static int FrameSkipInterval; + + private static int FrameSkipCounter; + + private static byte sq2_duty_cycle; + + private static bool sq2_length_halt; + + private static bool sq2_constant_volume_envelope; + + private static byte sq2_volume_devider_period; + + private static bool sq2_sweep_enable; + + private static byte sq2_sweep_devider_period; + + private static bool sq2_sweep_negate; + + private static byte sq2_sweep_shift_count; + + private static int sq2_timer; + + private static int sq2_period_devider; + + private static byte sq2_seqencer; + + private static bool sq2_length_enabled; + + private static int sq2_length_counter; + + private static bool sq2_envelope_start_flag; + + private static byte sq2_envelope_devider; + + private static byte sq2_envelope_decay_level_counter; + + private static byte sq2_envelope; + + private static int sq2_sweep_counter; + + private static bool sq2_sweep_reload; + + private static int sq2_sweep_change; + + private static bool sq2_valid_freq; + + private static int sq2_output; + + private static bool sq2_ignore_reload; + + private static byte register_p + { + get + { + return (byte)((cpu_flag_n ? 128u : 0u) | (cpu_flag_v ? 64u : 0u) | (cpu_flag_d ? 8u : 0u) | (cpu_flag_i ? 4u : 0u) | (cpu_flag_z ? 2u : 0u) | (cpu_flag_c ? 1u : 0u) | 0x20u); + } + set + { + cpu_flag_n = (value & 0x80) != 0; + cpu_flag_v = (value & 0x40) != 0; + cpu_flag_d = (value & 8) != 0; + cpu_flag_i = (value & 4) != 0; + cpu_flag_z = (value & 2) != 0; + cpu_flag_c = (value & 1) != 0; + } + } + + public static GameGenieCode[] GameGenieCodes + { + get + { + if (mem_board != null) + { + return mem_board.GameGenieCodes; + } + return null; + } + } + + public static bool IsGameGenieActive + { + get + { + if (mem_board != null) + { + return mem_board.IsGameGenieActive; + } + return false; + } + set + { + if (mem_board != null) + { + mem_board.IsGameGenieActive = value; + } + } + } + + public static bool IsGameFoundOnDB + { + get + { + if (mem_board != null) + { + return mem_board.IsGameFoundOnDB; + } + return false; + } + } + + public static NesCartDatabaseGameInfo GameInfo + { + get + { + if (mem_board != null) + { + return mem_board.GameInfo; + } + return NesCartDatabaseGameInfo.Empty; + } + } + + public static NesCartDatabaseCartridgeInfo GameCartInfo + { + get + { + if (mem_board != null) + { + return mem_board.GameCartInfo; + } + return new NesCartDatabaseCartridgeInfo(); + } + } + + public static string SHA1 => mem_board.SHA1; + + public static event EventHandler EmuShutdown; + + private static void DMCHardReset() + { + dmc_output_a = 0; + dmc_output = 0; + dmc_period_devider = 0; + dmc_loop_flag = false; + dmc_rate_index = 0; + dmc_irq_enabled = false; + dmc_dmaAddr = 49152; + dmc_addr_refresh = 49152; + dmc_size_refresh = 0; + dmc_dmaBits = 1; + dmc_dmaByte = 1; + dmc_period_devider = 0; + dmc_dmaEnabled = false; + dmc_bufferFull = false; + dmc_dmaSize = 0; + } + + private static void DMCSoftReset() + { + DMCHardReset(); + } + + private static void DMCClock() + { + dmc_period_devider--; + if (dmc_period_devider > 0) + { + return; + } + dmc_period_devider = dmc_freq_table[SystemIndex][dmc_rate_index]; + if (dmc_dmaEnabled) + { + if (((uint)dmc_dmaByte & (true ? 1u : 0u)) != 0) + { + if (dmc_output_a <= 125) + { + dmc_output_a += 2; + } + } + else if (dmc_output_a >= 2) + { + dmc_output_a -= 2; + } + dmc_dmaByte >>= 1; + } + dmc_dmaBits--; + if (dmc_dmaBits == 0) + { + dmc_dmaBits = 8; + if (dmc_bufferFull) + { + dmc_bufferFull = false; + dmc_dmaEnabled = true; + dmc_dmaByte = dmc_dmaBuffer; + if (dmc_dmaSize > 0) + { + AssertDMCDMA(); + } + } + else + { + dmc_dmaEnabled = false; + } + } + if (audio_dmc_outputable) + { + dmc_output = dmc_output_a; + } + audio_signal_outputed = true; + } + + private static void DMCDoDMA() + { + dmc_bufferFull = true; + Read(ref dmc_dmaAddr, out dmc_dmaBuffer); + if (dmc_dmaAddr == ushort.MaxValue) + { + dmc_dmaAddr = 32768; + } + else + { + dmc_dmaAddr++; + } + if (dmc_dmaSize > 0) + { + dmc_dmaSize--; + } + if (dmc_dmaSize == 0) + { + if (dmc_loop_flag) + { + dmc_dmaSize = dmc_size_refresh; + dmc_dmaAddr = dmc_addr_refresh; + } + else if (dmc_irq_enabled) + { + IRQFlags |= 2; + apu_irq_delta_occur = true; + } + } + } + + private static void APUOnRegister4010() + { + if (apu_reg_access_w) + { + dmc_irq_enabled = (apu_reg_io_db & 0x80) != 0; + dmc_loop_flag = (apu_reg_io_db & 0x40) != 0; + if (!dmc_irq_enabled) + { + apu_irq_delta_occur = false; + IRQFlags &= -3; + } + dmc_rate_index = (byte)(apu_reg_io_db & 0xFu); + } + } + + private static void APUOnRegister4011() + { + if (apu_reg_access_w) + { + dmc_output_a = (byte)(apu_reg_io_db & 0x7F); + } + } + + private static void APUOnRegister4012() + { + if (apu_reg_access_w) + { + dmc_addr_refresh = (ushort)((uint)(apu_reg_io_db << 6) | 0xC000u); + } + } + + private static void APUOnRegister4013() + { + if (apu_reg_access_w) + { + dmc_size_refresh = (apu_reg_io_db << 4) | 1; + } + } + + private static void DMCOn4015() + { + apu_irq_delta_occur = false; + IRQFlags &= -3; + } + + private static void DMCRead4015() + { + if (dmc_dmaSize > 0) + { + apu_reg_io_db = (byte)((apu_reg_io_db & 0xEFu) | 0x10u); + } + } + + private static void DMCWriteState(ref BinaryWriter bin) + { + bin.Write(dmc_output_a); + bin.Write(dmc_output); + bin.Write(dmc_period_devider); + bin.Write(dmc_irq_enabled); + bin.Write(dmc_loop_flag); + bin.Write(dmc_rate_index); + bin.Write(dmc_addr_refresh); + bin.Write(dmc_size_refresh); + bin.Write(dmc_dmaEnabled); + bin.Write(dmc_dmaByte); + bin.Write(dmc_dmaBits); + bin.Write(dmc_bufferFull); + bin.Write(dmc_dmaBuffer); + bin.Write(dmc_dmaSize); + bin.Write(dmc_dmaAddr); + } + + private static void DMCReadState(ref BinaryReader bin) + { + dmc_output_a = bin.ReadInt32(); + dmc_output = bin.ReadInt32(); + dmc_period_devider = bin.ReadInt32(); + dmc_irq_enabled = bin.ReadBoolean(); + dmc_loop_flag = bin.ReadBoolean(); + dmc_rate_index = bin.ReadByte(); + dmc_addr_refresh = bin.ReadUInt16(); + dmc_size_refresh = bin.ReadInt32(); + dmc_dmaEnabled = bin.ReadBoolean(); + dmc_dmaByte = bin.ReadByte(); + dmc_dmaBits = bin.ReadInt32(); + dmc_bufferFull = bin.ReadBoolean(); + dmc_dmaBuffer = bin.ReadByte(); + dmc_dmaSize = bin.ReadInt32(); + dmc_dmaAddr = bin.ReadUInt16(); + } + + private static void NOSHardReset() + { + nos_length_halt = false; + nos_constant_volume_envelope = false; + nos_volume_devider_period = 0; + nos_shift_reg = 1; + nos_timer = 0; + nos_mode = false; + nos_period_devider = 0; + nos_length_enabled = false; + nos_length_counter = 0; + nos_envelope_start_flag = false; + nos_envelope_devider = 0; + nos_envelope_decay_level_counter = 0; + nos_envelope = 0; + nos_output = 0; + nos_feedback = 0; + nos_ignore_reload = false; + } + + private static void NOSSoftReset() + { + NOSHardReset(); + } + + private static void NOSClock() + { + nos_period_devider--; + if (nos_period_devider > 0) + { + return; + } + nos_period_devider = nos_timer; + if (nos_mode) + { + nos_feedback = ((nos_shift_reg >> 6) & 1) ^ (nos_shift_reg & 1); + } + else + { + nos_feedback = ((nos_shift_reg >> 1) & 1) ^ (nos_shift_reg & 1); + } + nos_shift_reg >>= 1; + nos_shift_reg = (nos_shift_reg & 0x3FFF) | ((nos_feedback & 1) << 14); + if (nos_length_counter > 0 && (nos_shift_reg & 1) == 0) + { + if (audio_nos_outputable) + { + nos_output = nos_envelope; + } + } + else + { + nos_output = 0; + } + audio_signal_outputed = true; + } + + private static void NOSClockLength() + { + if (nos_length_counter > 0 && !nos_length_halt) + { + nos_length_counter--; + if (apu_reg_access_happened && apu_reg_io_addr == 15 && apu_reg_access_w) + { + nos_ignore_reload = true; + } + } + } + + private static void NOSClockEnvelope() + { + if (nos_envelope_start_flag) + { + nos_envelope_start_flag = false; + nos_envelope_decay_level_counter = 15; + nos_envelope_devider = (byte)(nos_volume_devider_period + 1); + } + else if (nos_envelope_devider > 0) + { + nos_envelope_devider--; + } + else + { + nos_envelope_devider = (byte)(nos_volume_devider_period + 1); + if (nos_envelope_decay_level_counter > 0) + { + nos_envelope_decay_level_counter--; + } + else if (nos_length_halt) + { + nos_envelope_decay_level_counter = 15; + } + } + nos_envelope = (nos_constant_volume_envelope ? nos_volume_devider_period : nos_envelope_decay_level_counter); + } + + private static void APUOnRegister400C() + { + if (apu_reg_access_w) + { + nos_volume_devider_period = (byte)(apu_reg_io_db & 0xFu); + nos_length_halt = (apu_reg_io_db & 0x20) != 0; + nos_constant_volume_envelope = (apu_reg_io_db & 0x10) != 0; + nos_envelope = (nos_constant_volume_envelope ? nos_volume_devider_period : nos_envelope_decay_level_counter); + } + } + + private static void APUOnRegister400D() + { + } + + private static void APUOnRegister400E() + { + if (apu_reg_access_w) + { + nos_timer = (ushort)(nos_freq_table[SystemIndex][apu_reg_io_db & 0xF] / 2); + nos_mode = (apu_reg_io_db & 0x80) == 128; + } + } + + private static void APUOnRegister400F() + { + if (apu_reg_access_w) + { + if (nos_length_enabled && !nos_ignore_reload) + { + nos_length_counter = sq_duration_table[apu_reg_io_db >> 3]; + } + if (nos_ignore_reload) + { + nos_ignore_reload = false; + } + nos_envelope_start_flag = true; + } + } + + private static void NOSOn4015() + { + nos_length_enabled = (apu_reg_io_db & 8) != 0; + if (!nos_length_enabled) + { + nos_length_counter = 0; + } + } + + private static void NOSRead4015() + { + if (nos_length_counter > 0) + { + apu_reg_io_db = (byte)((apu_reg_io_db & 0xF7u) | 8u); + } + } + + private static void NOSWriteState(ref BinaryWriter bin) + { + bin.Write(nos_length_halt); + bin.Write(nos_constant_volume_envelope); + bin.Write(nos_volume_devider_period); + bin.Write(nos_timer); + bin.Write(nos_mode); + bin.Write(nos_period_devider); + bin.Write(nos_length_enabled); + bin.Write(nos_length_counter); + bin.Write(nos_envelope_start_flag); + bin.Write(nos_envelope_devider); + bin.Write(nos_envelope_decay_level_counter); + bin.Write(nos_envelope); + bin.Write(nos_output); + bin.Write(nos_shift_reg); + bin.Write(nos_feedback); + bin.Write(nos_ignore_reload); + } + + private static void NOSReadState(ref BinaryReader bin) + { + nos_length_halt = bin.ReadBoolean(); + nos_constant_volume_envelope = bin.ReadBoolean(); + nos_volume_devider_period = bin.ReadByte(); + nos_timer = bin.ReadUInt16(); + nos_mode = bin.ReadBoolean(); + nos_period_devider = bin.ReadInt32(); + nos_length_enabled = bin.ReadBoolean(); + nos_length_counter = bin.ReadInt32(); + nos_envelope_start_flag = bin.ReadBoolean(); + nos_envelope_devider = bin.ReadByte(); + nos_envelope_decay_level_counter = bin.ReadByte(); + nos_envelope = bin.ReadByte(); + nos_output = bin.ReadInt32(); + nos_shift_reg = bin.ReadInt32(); + nos_feedback = bin.ReadInt32(); + nos_ignore_reload = bin.ReadBoolean(); + } + + private static void SQ1HardReset() + { + sq1_duty_cycle = 0; + sq1_length_halt = false; + sq1_constant_volume_envelope = false; + sq1_volume_devider_period = 0; + sq1_sweep_enable = false; + sq1_sweep_devider_period = 0; + sq1_sweep_negate = false; + sq1_sweep_shift_count = 0; + sq1_timer = 0; + sq1_period_devider = 0; + sq1_seqencer = 0; + sq1_length_enabled = false; + sq1_length_counter = 0; + sq1_envelope_start_flag = false; + sq1_envelope_devider = 0; + sq1_envelope_decay_level_counter = 0; + sq1_envelope = 0; + sq1_sweep_counter = 0; + sq1_sweep_reload = false; + sq1_sweep_change = 0; + sq1_valid_freq = false; + sq1_output = 0; + sq1_ignore_reload = false; + } + + private static void SQ1SoftReset() + { + SQ1HardReset(); + } + + private static void SQ1Clock() + { + sq1_period_devider--; + if (sq1_period_devider > 0) + { + return; + } + sq1_period_devider = sq1_timer + 1; + sq1_seqencer = (byte)((uint)(sq1_seqencer + 1) & 7u); + if (sq1_length_counter > 0 && sq1_valid_freq) + { + if (audio_sq1_outputable) + { + sq1_output = sq_duty_cycle_sequences[sq1_duty_cycle][sq1_seqencer] * sq1_envelope; + } + } + else + { + sq1_output = 0; + } + audio_signal_outputed = true; + } + + private static void SQ1ClockLength() + { + if (sq1_length_counter > 0 && !sq1_length_halt) + { + sq1_length_counter--; + if (apu_reg_access_happened && apu_reg_io_addr == 3 && apu_reg_access_w) + { + sq1_ignore_reload = true; + } + } + sq1_sweep_counter--; + if (sq1_sweep_counter == 0) + { + sq1_sweep_counter = sq1_sweep_devider_period + 1; + if (sq1_sweep_enable && sq1_sweep_shift_count > 0 && sq1_valid_freq) + { + sq1_sweep_change = sq1_timer >> (int)sq1_sweep_shift_count; + sq1_timer += (sq1_sweep_negate ? (~sq1_sweep_change) : sq1_sweep_change); + SQ1CalculateValidFreq(); + } + } + if (sq1_sweep_reload) + { + sq1_sweep_counter = sq1_sweep_devider_period + 1; + sq1_sweep_reload = false; + } + } + + private static void SQ1ClockEnvelope() + { + if (sq1_envelope_start_flag) + { + sq1_envelope_start_flag = false; + sq1_envelope_decay_level_counter = 15; + sq1_envelope_devider = (byte)(sq1_volume_devider_period + 1); + } + else if (sq1_envelope_devider > 0) + { + sq1_envelope_devider--; + } + else + { + sq1_envelope_devider = (byte)(sq1_volume_devider_period + 1); + if (sq1_envelope_decay_level_counter > 0) + { + sq1_envelope_decay_level_counter--; + } + else if (sq1_length_halt) + { + sq1_envelope_decay_level_counter = 15; + } + } + sq1_envelope = (sq1_constant_volume_envelope ? sq1_volume_devider_period : sq1_envelope_decay_level_counter); + } + + private static void APUOnRegister4000() + { + if (apu_reg_access_w) + { + sq1_duty_cycle = (byte)((apu_reg_io_db & 0xC0) >> 6); + sq1_volume_devider_period = (byte)(apu_reg_io_db & 0xFu); + sq1_length_halt = (apu_reg_io_db & 0x20) != 0; + sq1_constant_volume_envelope = (apu_reg_io_db & 0x10) != 0; + sq1_envelope = (sq1_constant_volume_envelope ? sq1_volume_devider_period : sq1_envelope_decay_level_counter); + } + } + + private static void APUOnRegister4001() + { + if (apu_reg_access_w) + { + sq1_sweep_enable = (apu_reg_io_db & 0x80) == 128; + sq1_sweep_devider_period = (byte)((uint)(apu_reg_io_db >> 4) & 7u); + sq1_sweep_negate = (apu_reg_io_db & 8) == 8; + sq1_sweep_shift_count = (byte)(apu_reg_io_db & 7u); + sq1_sweep_reload = true; + SQ1CalculateValidFreq(); + } + } + + private static void APUOnRegister4002() + { + if (apu_reg_access_w) + { + sq1_timer = (sq1_timer & 0xFF00) | apu_reg_io_db; + SQ1CalculateValidFreq(); + } + } + + private static void APUOnRegister4003() + { + if (apu_reg_access_w) + { + sq1_timer = (sq1_timer & 0xFF) | ((apu_reg_io_db & 7) << 8); + if (sq1_length_enabled && !sq1_ignore_reload) + { + sq1_length_counter = sq_duration_table[apu_reg_io_db >> 3]; + } + if (sq1_ignore_reload) + { + sq1_ignore_reload = false; + } + sq1_seqencer = 0; + sq1_envelope_start_flag = true; + SQ1CalculateValidFreq(); + } + } + + private static void SQ1On4015() + { + sq1_length_enabled = (apu_reg_io_db & 1) != 0; + if (!sq1_length_enabled) + { + sq1_length_counter = 0; + } + } + + private static void SQ1Read4015() + { + if (sq1_length_counter > 0) + { + apu_reg_io_db = (byte)((apu_reg_io_db & 0xFEu) | 1u); + } + } + + private static void SQ1CalculateValidFreq() + { + sq1_valid_freq = sq1_timer >= 8 && (sq1_sweep_negate || ((sq1_timer + (sq1_timer >> (int)sq1_sweep_shift_count)) & 0x800) == 0); + } + + private static void SQ1WriteState(ref BinaryWriter bin) + { + bin.Write(sq1_duty_cycle); + bin.Write(sq1_length_halt); + bin.Write(sq1_constant_volume_envelope); + bin.Write(sq1_volume_devider_period); + bin.Write(sq1_sweep_enable); + bin.Write(sq1_sweep_devider_period); + bin.Write(sq1_sweep_negate); + bin.Write(sq1_sweep_shift_count); + bin.Write(sq1_timer); + bin.Write(sq1_period_devider); + bin.Write(sq1_seqencer); + bin.Write(sq1_length_enabled); + bin.Write(sq1_length_counter); + bin.Write(sq1_envelope_start_flag); + bin.Write(sq1_envelope_devider); + bin.Write(sq1_envelope_decay_level_counter); + bin.Write(sq1_envelope); + bin.Write(sq1_sweep_counter); + bin.Write(sq1_sweep_reload); + bin.Write(sq1_sweep_change); + bin.Write(sq1_valid_freq); + bin.Write(sq1_output); + bin.Write(sq1_ignore_reload); + } + + private static void SQ1ReadState(ref BinaryReader bin) + { + sq1_duty_cycle = bin.ReadByte(); + sq1_length_halt = bin.ReadBoolean(); + sq1_constant_volume_envelope = bin.ReadBoolean(); + sq1_volume_devider_period = bin.ReadByte(); + sq1_sweep_enable = bin.ReadBoolean(); + sq1_sweep_devider_period = bin.ReadByte(); + sq1_sweep_negate = bin.ReadBoolean(); + sq1_sweep_shift_count = bin.ReadByte(); + sq1_timer = bin.ReadInt32(); + sq1_period_devider = bin.ReadInt32(); + sq1_seqencer = bin.ReadByte(); + sq1_length_enabled = bin.ReadBoolean(); + sq1_length_counter = bin.ReadInt32(); + sq1_envelope_start_flag = bin.ReadBoolean(); + sq1_envelope_devider = bin.ReadByte(); + sq1_envelope_decay_level_counter = bin.ReadByte(); + sq1_envelope = bin.ReadByte(); + sq1_sweep_counter = bin.ReadInt32(); + sq1_sweep_reload = bin.ReadBoolean(); + sq1_sweep_change = bin.ReadInt32(); + sq1_valid_freq = bin.ReadBoolean(); + sq1_output = bin.ReadInt32(); + sq1_ignore_reload = bin.ReadBoolean(); + } + + private static void TRLHardReset() + { + trl_liner_control_flag = false; + trl_liner_control_reload = 0; + trl_timer = 0; + trl_length_enabled = false; + trl_length_counter = 0; + trl_liner_control_reload_flag = false; + trl_liner_counter = 0; + trl_output = 0; + trl_period_devider = 0; + trl_step = 0; + trl_ignore_reload = false; + } + + private static void TRLSoftReset() + { + TRLHardReset(); + } + + private static void TRLClock() + { + trl_period_devider--; + if (trl_period_devider > 0) + { + return; + } + trl_period_devider = trl_timer + 1; + if (trl_length_counter > 0 && trl_liner_counter > 0 && trl_timer >= 4) + { + trl_step++; + trl_step &= 31; + if (audio_trl_outputable) + { + trl_output = trl_step_seq[trl_step]; + } + } + audio_signal_outputed = true; + } + + private static void TRLClockLength() + { + if (trl_length_counter > 0 && !trl_liner_control_flag) + { + trl_length_counter--; + if (apu_reg_access_happened && apu_reg_io_addr == 11 && apu_reg_access_w) + { + trl_ignore_reload = true; + } + } + } + + private static void TRLClockEnvelope() + { + if (trl_liner_control_reload_flag) + { + trl_liner_counter = trl_liner_control_reload; + } + else if (trl_liner_counter > 0) + { + trl_liner_counter--; + } + if (!trl_liner_control_flag) + { + trl_liner_control_reload_flag = false; + } + } + + private static void APUOnRegister4008() + { + if (apu_reg_access_w) + { + trl_liner_control_flag = (apu_reg_io_db & 0x80) == 128; + trl_liner_control_reload = (byte)(apu_reg_io_db & 0x7Fu); + } + } + + private static void APUOnRegister4009() + { + } + + private static void APUOnRegister400A() + { + if (apu_reg_access_w) + { + trl_timer = (ushort)((trl_timer & 0x7F00u) | apu_reg_io_db); + } + } + + private static void APUOnRegister400B() + { + if (apu_reg_access_w) + { + trl_timer = (ushort)((trl_timer & 0xFFu) | (uint)((apu_reg_io_db & 7) << 8)); + if (trl_length_enabled && !trl_ignore_reload) + { + trl_length_counter = sq_duration_table[apu_reg_io_db >> 3]; + } + if (trl_ignore_reload) + { + trl_ignore_reload = false; + } + trl_liner_control_reload_flag = true; + } + } + + private static void TRLOn4015() + { + trl_length_enabled = (apu_reg_io_db & 4) != 0; + if (!trl_length_enabled) + { + trl_length_counter = 0; + } + } + + private static void TRLRead4015() + { + if (trl_length_counter > 0) + { + apu_reg_io_db = (byte)((apu_reg_io_db & 0xFBu) | 4u); + } + } + + private static void TRLWriteState(ref BinaryWriter bin) + { + bin.Write(trl_liner_control_flag); + bin.Write(trl_liner_control_reload); + bin.Write(trl_timer); + bin.Write(trl_length_enabled); + bin.Write(trl_length_counter); + bin.Write(trl_liner_control_reload_flag); + bin.Write(trl_liner_counter); + bin.Write(trl_output); + bin.Write(trl_period_devider); + bin.Write(trl_step); + bin.Write(trl_ignore_reload); + } + + private static void TRLReadState(ref BinaryReader bin) + { + trl_liner_control_flag = bin.ReadBoolean(); + trl_liner_control_reload = bin.ReadByte(); + trl_timer = bin.ReadUInt16(); + trl_length_enabled = bin.ReadBoolean(); + trl_length_counter = bin.ReadByte(); + trl_liner_control_reload_flag = bin.ReadBoolean(); + trl_liner_counter = bin.ReadByte(); + trl_output = bin.ReadInt32(); + trl_period_devider = bin.ReadInt32(); + trl_step = bin.ReadInt32(); + trl_ignore_reload = bin.ReadBoolean(); + } + + private static void APUInitialize() + { + apu_reg_update_func = new Action[32]; + apu_reg_read_func = new Action[32]; + apu_reg_write_func = new Action[32]; + for (int i = 0; i < 32; i++) + { + apu_reg_update_func[i] = APUBlankAccess; + apu_reg_read_func[i] = APUBlankAccess; + apu_reg_write_func[i] = APUBlankAccess; + } + apu_reg_update_func[0] = APUOnRegister4000; + apu_reg_update_func[1] = APUOnRegister4001; + apu_reg_update_func[2] = APUOnRegister4002; + apu_reg_update_func[3] = APUOnRegister4003; + apu_reg_update_func[4] = APUOnRegister4004; + apu_reg_update_func[5] = APUOnRegister4005; + apu_reg_update_func[6] = APUOnRegister4006; + apu_reg_update_func[7] = APUOnRegister4007; + apu_reg_update_func[8] = APUOnRegister4008; + apu_reg_update_func[9] = APUOnRegister4009; + apu_reg_update_func[10] = APUOnRegister400A; + apu_reg_update_func[11] = APUOnRegister400B; + apu_reg_update_func[12] = APUOnRegister400C; + apu_reg_update_func[13] = APUOnRegister400D; + apu_reg_update_func[14] = APUOnRegister400E; + apu_reg_update_func[15] = APUOnRegister400F; + apu_reg_update_func[16] = APUOnRegister4010; + apu_reg_update_func[17] = APUOnRegister4011; + apu_reg_update_func[18] = APUOnRegister4012; + apu_reg_update_func[19] = APUOnRegister4013; + apu_reg_update_func[21] = APUOnRegister4015; + apu_reg_update_func[22] = APUOnRegister4016; + apu_reg_update_func[23] = APUOnRegister4017; + apu_reg_read_func[21] = APURead4015; + apu_reg_read_func[22] = APURead4016; + apu_reg_read_func[23] = APURead4017; + apu_reg_write_func[20] = APUWrite4014; + apu_reg_write_func[21] = APUWrite4015; + audio_low_pass_filter_14K = new SoundLowPassFilter(0.00815686); + audio_high_pass_filter_90 = new SoundHighPassFilter(0.999835); + audio_high_pass_filter_440 = new SoundHighPassFilter(0.996039); + audio_dc_blocker_filter = new SoundDCBlockerFilter(0.995); + apu_update_playback_func = APUUpdatePlaybackWithFilters; + } + + public static void ApplyAudioSettings(bool all = true) + { + SoundEnabled = MyNesMain.RendererSettings.Audio_SoundEnabled; + audio_sq1_outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_SQ1; + audio_sq2_outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_SQ2; + audio_nos_outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_NOZ; + audio_trl_outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_TRL; + audio_dmc_outputable = MyNesMain.RendererSettings.Audio_ChannelEnabled_DMC; + if (apu_use_external_sound) + { + mem_board.APUApplyChannelsSettings(); + } + if (all) + { + CalculateAudioPlaybackValues(); + } + } + + private static void APUHardReset() + { + apu_reg_io_db = 0; + apu_reg_io_addr = 0; + apu_reg_access_happened = false; + apu_reg_access_w = false; + apu_seq_mode = false; + apu_odd_cycle = true; + apu_cycle_f_t = 0; + apu_cycle_e = 4; + apu_cycle_f = 4; + apu_cycle_l = 4; + apu_odd_l = false; + apu_check_irq = false; + apu_do_env = false; + apu_do_length = false; + switch (Region) + { + case EmuRegion.NTSC: + cpu_speed = 1789773.0; + cpu_clock_per_frame = 29780.5; + apu_ferq_f = 14914; + apu_ferq_e = 3728; + apu_ferq_l = 7456; + break; + case EmuRegion.PALB: + cpu_speed = 1662607.0; + cpu_clock_per_frame = 33247.5; + apu_ferq_f = 14914; + apu_ferq_e = 3728; + apu_ferq_l = 7456; + break; + case EmuRegion.DENDY: + cpu_speed = 1773448.0; + cpu_clock_per_frame = 35464.0; + apu_ferq_f = 14914; + apu_ferq_e = 3728; + apu_ferq_l = 7456; + break; + } + Tracer.WriteLine("NES: cpu speed = " + cpu_speed); + SQ1HardReset(); + SQ2HardReset(); + NOSHardReset(); + DMCHardReset(); + TRLHardReset(); + apu_irq_enabled = true; + apu_irq_flag = false; + reg_2004 = 8196; + CalculateAudioPlaybackValues(); + apu_use_external_sound = mem_board.enable_external_sound; + if (apu_use_external_sound) + { + Tracer.WriteInformation("External sound channels has been enabled on apu."); + } + } + + private static void APUSoftReset() + { + apu_reg_io_db = 0; + apu_reg_io_addr = 0; + apu_reg_access_happened = false; + apu_reg_access_w = false; + apu_seq_mode = false; + apu_odd_cycle = false; + apu_cycle_f_t = 0; + apu_cycle_e = 4; + apu_cycle_f = 4; + apu_cycle_l = 4; + apu_odd_l = false; + apu_check_irq = false; + apu_do_env = false; + apu_do_length = false; + apu_irq_enabled = true; + apu_irq_flag = false; + SQ1SoftReset(); + SQ2SoftReset(); + TRLSoftReset(); + NOSSoftReset(); + DMCSoftReset(); + } + + private static void APUIORead(ref ushort addr, out byte value) + { + if (addr >= 16416) + { + mem_board.ReadEX(ref addr, out value); + return; + } + apu_reg_io_addr = (byte)(addr & 0x1Fu); + apu_reg_access_happened = true; + apu_reg_access_w = false; + apu_reg_read_func[apu_reg_io_addr](); + value = apu_reg_io_db; + } + + private static void APUIOWrite(ref ushort addr, ref byte value) + { + if (addr >= 16416) + { + mem_board.WriteEX(ref addr, ref value); + return; + } + apu_reg_io_addr = (byte)(addr & 0x1Fu); + apu_reg_io_db = value; + apu_reg_access_w = true; + apu_reg_access_happened = true; + apu_reg_write_func[apu_reg_io_addr](); + } + + private static void APUBlankAccess() + { + } + + private static void APUWrite4014() + { + dma_Oamaddress = (ushort)(apu_reg_io_db << 8); + AssertOAMDMA(); + } + + private static void APUWrite4015() + { + if ((apu_reg_io_db & 0x10u) != 0) + { + if (dmc_dmaSize == 0) + { + dmc_dmaSize = dmc_size_refresh; + dmc_dmaAddr = dmc_addr_refresh; + } + } + else + { + dmc_dmaSize = 0; + } + if (!dmc_bufferFull && dmc_dmaSize > 0) + { + AssertDMCDMA(); + } + } + + private static void APUOnRegister4015() + { + if (apu_reg_access_w) + { + SQ1On4015(); + SQ2On4015(); + NOSOn4015(); + TRLOn4015(); + DMCOn4015(); + } + else + { + apu_irq_flag = false; + IRQFlags &= -2; + } + } + + private static void APUOnRegister4016() + { + if (!apu_reg_access_w) + { + return; + } + if (inputStrobe > (apu_reg_io_db & 1)) + { + if (IsFourPlayers) + { + PORT0 = (joypad3.GetData() << 8) | joypad1.GetData() | 0x1010000; + PORT1 = (joypad4.GetData() << 8) | joypad2.GetData() | 0x2020000; + } + else + { + PORT0 = joypad1.GetData() | 0x1010100; + PORT1 = joypad2.GetData() | 0x2020200; + } + } + inputStrobe = apu_reg_io_db & 1; + } + + private static void APUOnRegister4017() + { + if (apu_reg_access_w) + { + apu_seq_mode = (apu_reg_io_db & 0x80) != 0; + apu_irq_enabled = (apu_reg_io_db & 0x40) == 0; + apu_cycle_e = -1; + apu_cycle_l = -1; + apu_cycle_f = -1; + apu_odd_l = false; + apu_do_length = apu_seq_mode; + apu_do_env = apu_seq_mode; + apu_check_irq = false; + if (!apu_irq_enabled) + { + apu_irq_flag = false; + IRQFlags &= -2; + } + } + } + + private static void APURead4015() + { + apu_reg_io_db &= 32; + SQ1Read4015(); + SQ2Read4015(); + NOSRead4015(); + TRLRead4015(); + DMCRead4015(); + if (apu_irq_flag) + { + apu_reg_io_db = (byte)((apu_reg_io_db & 0xBFu) | 0x40u); + } + if (apu_irq_delta_occur) + { + apu_reg_io_db = (byte)((apu_reg_io_db & 0x7Fu) | 0x80u); + } + } + + private static void APURead4016() + { + apu_reg_io_db = (byte)((uint)PORT0 & 1u); + PORT0 >>= 1; + } + + private static void APURead4017() + { + apu_reg_io_db = (byte)((uint)PORT1 & 1u); + PORT1 >>= 1; + } + + private static void APUClock() + { + apu_odd_cycle = !apu_odd_cycle; + if (apu_do_env) + { + APUClockEnvelope(); + } + if (apu_do_length) + { + APUClockDuration(); + } + if (apu_odd_cycle) + { + apu_cycle_f++; + if (apu_cycle_f >= apu_ferq_f) + { + apu_cycle_f = -1; + apu_check_irq = true; + apu_cycle_f_t = 3; + } + apu_cycle_e++; + if (apu_cycle_e >= apu_ferq_e) + { + apu_cycle_e = -1; + if (apu_check_irq) + { + if (!apu_seq_mode) + { + apu_do_env = true; + } + else + { + apu_cycle_e = 4; + } + } + else + { + apu_do_env = true; + } + } + apu_cycle_l++; + if (apu_cycle_l >= apu_ferq_l) + { + apu_odd_l = !apu_odd_l; + apu_cycle_l = (apu_odd_l ? (-2) : (-1)); + if (apu_check_irq && apu_seq_mode) + { + apu_cycle_l = 3730; + apu_odd_l = true; + } + else + { + apu_do_length = true; + } + } + SQ1Clock(); + SQ2Clock(); + NOSClock(); + if (apu_use_external_sound) + { + mem_board.OnAPUClock(); + } + if (apu_reg_access_happened) + { + apu_reg_access_happened = false; + apu_reg_update_func[apu_reg_io_addr](); + } + } + TRLClock(); + DMCClock(); + if (apu_check_irq) + { + if (!apu_seq_mode) + { + APUCheckIRQ(); + } + apu_cycle_f_t--; + if (apu_cycle_f_t == 0) + { + apu_check_irq = false; + } + } + if (apu_use_external_sound) + { + mem_board.OnAPUClockSingle(); + } + apu_update_playback_func(); + } + + private static void APUClockDuration() + { + SQ1ClockLength(); + SQ2ClockLength(); + NOSClockLength(); + TRLClockLength(); + if (apu_use_external_sound) + { + mem_board.OnAPUClockDuration(); + } + apu_do_length = false; + } + + private static void APUClockEnvelope() + { + SQ1ClockEnvelope(); + SQ2ClockEnvelope(); + NOSClockEnvelope(); + TRLClockEnvelope(); + if (apu_use_external_sound) + { + mem_board.OnAPUClockEnvelope(); + } + apu_do_env = false; + } + + private static void APUCheckIRQ() + { + if (apu_irq_enabled) + { + apu_irq_flag = true; + } + if (apu_irq_flag) + { + IRQFlags |= 1; + } + } + + private static void CalculateAudioPlaybackValues() + { + apu_target_samples_count_per_frame = (double)MyNesMain.RendererSettings.Audio_Frequency / emu_time_target_fps; + audio_timer_ratio = cpu_clock_per_frame / apu_target_samples_count_per_frame; + cpu_speed = audio_timer_ratio * (double)MyNesMain.RendererSettings.Audio_Frequency; + audio_playback_peek_limit = MyNesMain.RendererSettings.Audio_InternalPeekLimit; + audio_samples_count = MyNesMain.RendererSettings.Audio_InternalSamplesCount; + audio_playback_amplitude = MyNesMain.RendererSettings.Audio_PlaybackAmplitude; + audio_samples = new short[audio_samples_count]; + audio_w_pos = 0; + audio_samples_added = 0; + audio_timer = 0.0; + audio_x = (audio_y = 0.0); + Tracer.WriteLine("AUDIO: frequency = " + MyNesMain.RendererSettings.Audio_Frequency); + Tracer.WriteLine("AUDIO: timer ratio = " + audio_timer_ratio); + Tracer.WriteLine("AUDIO: internal samples count = " + audio_samples_count); + Tracer.WriteLine("AUDIO: amplitude = " + audio_playback_amplitude); + double num = 0.0; + if (MyNesMain.RendererSettings.Audio_EnableFilters) + { + apu_update_playback_func = APUUpdatePlaybackWithFilters; + num = cpu_speed / 14000.0; + audio_low_pass_filter_14K = new SoundLowPassFilter(SoundLowPassFilter.GetK(num, 14000.0)); + num = cpu_speed / 90.0; + audio_high_pass_filter_90 = new SoundHighPassFilter(SoundHighPassFilter.GetK(num, 90.0)); + num = cpu_speed / 440.0; + audio_high_pass_filter_440 = new SoundHighPassFilter(SoundHighPassFilter.GetK(num, 440.0)); + } + else + { + apu_update_playback_func = APUUpdatePlaybackWithoutFilters; + } + InitializeDACTables(force_intitialize: false); + } + + public static void InitializeDACTables(bool force_intitialize) + { + if (audio_playback_dac_initialized && !force_intitialize) + { + return; + } + int[] array = new int[5]; + mix_table = new int[16][][][][]; + for (int i = 0; i < 16; i++) + { + mix_table[i] = new int[16][][][]; + for (int j = 0; j < 16; j++) + { + mix_table[i][j] = new int[16][][]; + for (int k = 0; k < 16; k++) + { + mix_table[i][j][k] = new int[16][]; + for (int l = 0; l < 16; l++) + { + mix_table[i][j][k][l] = new int[128]; + for (int m = 0; m < 128; m++) + { + if (MyNesMain.RendererSettings.Audio_UseDefaultMixer) + { + double num = 95.88 / (8128.0 / (double)(i + j) + 100.0); + double num2 = 159.79 / (1.0 / ((double)k / 8227.0 + (double)l / 12241.0 + (double)m / 22638.0) + 100.0); + mix_table[i][j][k][l][m] = (int)Math.Ceiling((num + num2) * audio_playback_amplitude); + continue; + } + GetPrec(i, 255, 2048, out array[0]); + GetPrec(j, 255, 2048, out array[1]); + GetPrec(l, 255, 2048, out array[2]); + GetPrec(k, 255, 2048, out array[3]); + GetPrec(m, 255, 2048, out array[4]); + array[4] /= 2; + int num3 = array[0] + array[1] + array[2] + array[3] + array[4]; + num3 /= 5; + mix_table[i][j][k][l][m] = num3; + } + } + } + } + } + audio_playback_dac_initialized = true; + } + + private static void APUUpdatePlaybackWithFilters() + { + if (!SoundEnabled) + { + return; + } + audio_x = mix_table[sq1_output][sq2_output][trl_output][nos_output][dmc_output]; + if (apu_use_external_sound) + { + audio_x = (audio_x + mem_board.APUGetSample() * audio_playback_amplitude) / 2.0; + } + audio_high_pass_filter_90.DoFiltering(audio_x, out audio_y); + audio_high_pass_filter_440.DoFiltering(audio_y, out audio_y); + audio_low_pass_filter_14K.DoFiltering(audio_y, out audio_y); + audio_y_av += audio_y; + audio_y_timer += 1.0; + audio_timer += 1.0; + if (!(audio_timer >= audio_timer_ratio)) + { + return; + } + if (audio_y_timer > 0.0) + { + audio_y = audio_y_av / audio_y_timer; + } + else + { + audio_y = 0.0; + } + audio_y_av = 0.0; + audio_y_timer = 0.0; + audio_timer -= audio_timer_ratio; + if (audio_w_pos < audio_samples_count) + { + if (audio_y > (double)audio_playback_peek_limit) + { + audio_y = audio_playback_peek_limit; + } + if (audio_y < (double)(-audio_playback_peek_limit)) + { + audio_y = -audio_playback_peek_limit; + } + audio_samples[audio_w_pos] = (short)audio_y; + if (MyNesMain.WaveRecorder.IsRecording) + { + MyNesMain.WaveRecorder.AddSample((short)audio_y); + } + audio_w_pos++; + audio_samples_added++; + } + audio_y = 0.0; + } + + private static void APUUpdatePlaybackWithoutFilters() + { + if (!SoundEnabled) + { + return; + } + audio_y = mix_table[sq1_output][sq2_output][trl_output][nos_output][dmc_output] / 2; + if (apu_use_external_sound) + { + audio_y = (audio_y + mem_board.APUGetSample() * audio_playback_amplitude) / 2.0; + } + audio_y_av += audio_y; + audio_y_timer += 1.0; + audio_timer += 1.0; + if (!(audio_timer >= audio_timer_ratio)) + { + return; + } + if (audio_y_timer > 0.0) + { + audio_y = audio_y_av / audio_y_timer; + } + else + { + audio_y = 0.0; + } + audio_y_av = 0.0; + audio_y_timer = 0.0; + audio_timer -= audio_timer_ratio; + if (audio_w_pos < audio_samples_count) + { + if (audio_y > (double)audio_playback_peek_limit) + { + audio_y = audio_playback_peek_limit; + } + if (audio_y < (double)(-audio_playback_peek_limit)) + { + audio_y = -audio_playback_peek_limit; + } + audio_samples[audio_w_pos] = (short)audio_y; + if (MyNesMain.WaveRecorder.IsRecording) + { + MyNesMain.WaveRecorder.AddSample((short)audio_y); + } + audio_w_pos++; + audio_samples_added++; + } + audio_y = 0.0; + } + + private static void GetPrec(int inVal, int inMax, int outMax, out int val) + { + val = outMax * inVal / inMax; + } + + private static void APUWriteState(ref BinaryWriter bin) + { + bin.Write(apu_reg_io_db); + bin.Write(apu_reg_io_addr); + bin.Write(apu_reg_access_happened); + bin.Write(apu_reg_access_w); + bin.Write(apu_odd_cycle); + bin.Write(apu_irq_enabled); + bin.Write(apu_irq_flag); + bin.Write(apu_irq_delta_occur); + bin.Write(apu_seq_mode); + bin.Write(apu_ferq_f); + bin.Write(apu_ferq_l); + bin.Write(apu_ferq_e); + bin.Write(apu_cycle_f); + bin.Write(apu_cycle_e); + bin.Write(apu_cycle_l); + bin.Write(apu_odd_l); + bin.Write(apu_cycle_f_t); + bin.Write(apu_check_irq); + bin.Write(apu_do_env); + bin.Write(apu_do_length); + SQ1WriteState(ref bin); + SQ2WriteState(ref bin); + NOSWriteState(ref bin); + TRLWriteState(ref bin); + DMCWriteState(ref bin); + } + + private static void APUReadState(ref BinaryReader bin) + { + apu_reg_io_db = bin.ReadByte(); + apu_reg_io_addr = bin.ReadByte(); + apu_reg_access_happened = bin.ReadBoolean(); + apu_reg_access_w = bin.ReadBoolean(); + apu_odd_cycle = bin.ReadBoolean(); + apu_irq_enabled = bin.ReadBoolean(); + apu_irq_flag = bin.ReadBoolean(); + apu_irq_delta_occur = bin.ReadBoolean(); + apu_seq_mode = bin.ReadBoolean(); + apu_ferq_f = bin.ReadInt32(); + apu_ferq_l = bin.ReadInt32(); + apu_ferq_e = bin.ReadInt32(); + apu_cycle_f = bin.ReadInt32(); + apu_cycle_e = bin.ReadInt32(); + apu_cycle_l = bin.ReadInt32(); + apu_odd_l = bin.ReadBoolean(); + apu_cycle_f_t = bin.ReadInt32(); + apu_check_irq = bin.ReadBoolean(); + apu_do_env = bin.ReadBoolean(); + apu_do_length = bin.ReadBoolean(); + SQ1ReadState(ref bin); + SQ2ReadState(ref bin); + NOSReadState(ref bin); + TRLReadState(ref bin); + DMCReadState(ref bin); + } + + private static byte register_pb() + { + return (byte)((cpu_flag_n ? 128u : 0u) | (cpu_flag_v ? 64u : 0u) | (cpu_flag_d ? 8u : 0u) | (cpu_flag_i ? 4u : 0u) | (cpu_flag_z ? 2u : 0u) | (cpu_flag_c ? 1u : 0u) | 0x30u); + } + + private static void CPUInitialize() + { + cpu_addressings = new Action[256] + { + Imp____, IndX_R_, ImA____, IndX_W_, Zpg_R__, Zpg_R__, Zpg_RW_, Zpg_W__, ImA____, Imm____, + ImA____, Imm____, Abs_R__, Abs_R__, Abs_RW_, Abs_W__, Imp____, IndY_R_, Imp____, IndY_W_, + ZpgX_R_, ZpgX_R_, ZpgX_RW, ZpgX_W_, ImA____, AbsY_R_, ImA____, AbsY_W_, AbsX_R_, AbsX_R_, + AbsX_RW, AbsX_W_, Imp____, IndX_R_, ImA____, IndX_W_, Zpg_R__, Zpg_R__, Zpg_RW_, Zpg_W__, + ImA____, Imm____, ImA____, Imm____, Abs_R__, Abs_R__, Abs_RW_, Abs_W__, Imp____, IndY_R_, + Imp____, IndY_W_, ZpgX_R_, ZpgX_R_, ZpgX_RW, ZpgX_W_, ImA____, AbsY_R_, ImA____, AbsY_W_, + AbsX_R_, AbsX_R_, AbsX_RW, AbsX_W_, ImA____, IndX_R_, ImA____, IndX_W_, Zpg_R__, Zpg_R__, + Zpg_RW_, Zpg_W__, ImA____, Imm____, ImA____, Imm____, Abs_W__, Abs_R__, Abs_RW_, Abs_W__, + Imp____, IndY_R_, Imp____, IndY_W_, ZpgX_R_, ZpgX_R_, ZpgX_RW, ZpgX_W_, ImA____, AbsY_R_, + ImA____, AbsY_W_, AbsX_R_, AbsX_R_, AbsX_RW, AbsX_W_, ImA____, IndX_R_, ImA____, IndX_W_, + Zpg_R__, Zpg_R__, Zpg_RW_, Zpg_W__, ImA____, Imm____, ImA____, Imm____, Imp____, Abs_R__, + Abs_RW_, Abs_W__, Imp____, IndY_R_, Imp____, IndY_W_, ZpgX_R_, ZpgX_R_, ZpgX_RW, ZpgX_W_, + ImA____, AbsY_R_, ImA____, AbsY_W_, AbsX_R_, AbsX_R_, AbsX_RW, AbsX_W_, Imm____, IndX_W_, + Imm____, IndX_W_, Zpg_W__, Zpg_W__, Zpg_W__, Zpg_W__, ImA____, Imm____, ImA____, Imm____, + Abs_W__, Abs_W__, Abs_W__, Abs_W__, Imp____, IndY_W_, Imp____, IndY_W_, ZpgX_W_, ZpgX_W_, + ZpgY_W_, ZpgY_W_, ImA____, AbsY_W_, ImA____, AbsY_W_, Abs_W__, AbsX_W_, Abs_W__, AbsY_W_, + Imm____, IndX_R_, Imm____, IndX_R_, Zpg_R__, Zpg_R__, Zpg_R__, Zpg_R__, ImA____, Imm____, + ImA____, Imm____, Abs_R__, Abs_R__, Abs_R__, Abs_R__, Imp____, IndY_R_, Imp____, IndY_R_, + ZpgX_R_, ZpgX_R_, ZpgY_R_, ZpgY_R_, ImA____, AbsY_R_, ImA____, AbsY_R_, AbsX_R_, AbsX_R_, + AbsY_R_, AbsY_R_, Imm____, IndX_R_, Imm____, IndX_R_, Zpg_R__, Zpg_R__, Zpg_RW_, Zpg_R__, + ImA____, Imm____, ImA____, Imm____, Abs_R__, Abs_R__, Abs_RW_, Abs_R__, Imp____, IndY_R_, + Imp____, IndY_RW, ZpgX_R_, ZpgX_R_, ZpgX_RW, ZpgX_RW, ImA____, AbsY_R_, ImA____, AbsY_RW, + AbsX_R_, AbsX_R_, AbsX_RW, AbsX_RW, Imm____, IndX_R_, Imm____, IndX_W_, Zpg_R__, Zpg_R__, + Zpg_RW_, Zpg_W__, ImA____, Imm____, ImA____, Imm____, Abs_R__, Abs_R__, Abs_RW_, Abs_W__, + Imp____, IndY_R_, Imp____, IndY_W_, ZpgX_R_, ZpgX_R_, ZpgX_RW, ZpgX_W_, ImA____, AbsY_R_, + ImA____, AbsY_W_, AbsX_R_, AbsX_R_, AbsX_RW, AbsX_W_ + }; + cpu_instructions = new Action[256] + { + BRK__, ORA__, NOP__, SLO__, NOP__, ORA__, ASL_M, SLO__, PHP__, ORA__, + ASL_A, ANC__, NOP__, ORA__, ASL_M, SLO__, BPL__, ORA__, NOP__, SLO__, + NOP__, ORA__, ASL_M, SLO__, CLC__, ORA__, NOP__, SLO__, NOP__, ORA__, + ASL_M, SLO__, JSR__, AND__, NOP__, RLA__, BIT__, AND__, ROL_M, RLA__, + PLP__, AND__, ROL_A, ANC__, BIT__, AND__, ROL_M, RLA__, BMI__, AND__, + NOP__, RLA__, NOP__, AND__, ROL_M, RLA__, SEC__, AND__, NOP__, RLA__, + NOP__, AND__, ROL_M, RLA__, RTI__, EOR__, NOP__, SRE__, NOP__, EOR__, + LSR_M, SRE__, PHA__, EOR__, LSR_A, ALR__, JMP__, EOR__, LSR_M, SRE__, + BVC__, EOR__, NOP__, SRE__, NOP__, EOR__, LSR_M, SRE__, CLI__, EOR__, + NOP__, SRE__, NOP__, EOR__, LSR_M, SRE__, RTS__, ADC__, NOP__, RRA__, + NOP__, ADC__, ROR_M, RRA__, PLA__, ADC__, ROR_A, ARR__, JMP_I, ADC__, + ROR_M, RRA__, BVS__, ADC__, NOP__, RRA__, NOP__, ADC__, ROR_M, RRA__, + SEI__, ADC__, NOP__, RRA__, NOP__, ADC__, ROR_M, RRA__, NOP__, STA__, + NOP__, SAX__, STY__, STA__, STX__, SAX__, DEY__, NOP__, TXA__, XAA__, + STY__, STA__, STX__, SAX__, BCC__, STA__, NOP__, AHX__, STY__, STA__, + STX__, SAX__, TYA__, STA__, TXS__, XAS__, SHY__, STA__, SHX__, AHX__, + LDY__, LDA__, LDX__, LAX__, LDY__, LDA__, LDX__, LAX__, TAY__, LDA__, + TAX__, LAX__, LDY__, LDA__, LDX__, LAX__, BCS__, LDA__, NOP__, LAX__, + LDY__, LDA__, LDX__, LAX__, CLV__, LDA__, TSX__, LAR__, LDY__, LDA__, + LDX__, LAX__, CPY__, CMP__, NOP__, DCP__, CPY__, CMP__, DEC__, DCP__, + INY__, CMP__, DEX__, AXS__, CPY__, CMP__, DEC__, DCP__, BNE__, CMP__, + NOP__, DCP__, NOP__, CMP__, DEC__, DCP__, CLD__, CMP__, NOP__, DCP__, + NOP__, CMP__, DEC__, DCP__, CPX__, SBC__, NOP__, ISC__, CPX__, SBC__, + INC__, ISC__, INX__, SBC__, NOP__, SBC__, CPX__, SBC__, INC__, ISC__, + BEQ__, SBC__, NOP__, ISC__, NOP__, SBC__, INC__, ISC__, SED__, SBC__, + NOP__, ISC__, NOP__, SBC__, INC__, ISC__ + }; + } + + private static void CPUClock() + { + Read(ref cpu_reg_pc.v, out cpu_opcode); + cpu_reg_pc.v++; + cpu_addressings[cpu_opcode](); + cpu_instructions[cpu_opcode](); + if (CPU_IRQ_PIN || CPU_NMI_PIN) + { + Read(ref cpu_reg_pc.v, out cpu_dummy); + Read(ref cpu_reg_pc.v, out cpu_dummy); + Interrupt(); + } + } + + private static void CPUHardReset() + { + cpu_reg_a = 0; + cpu_reg_x = 0; + cpu_reg_y = 0; + cpu_reg_sp.l = 253; + cpu_reg_sp.h = 1; + ushort addr = 65532; + mem_board.ReadPRG(ref addr, out cpu_reg_pc.l); + addr++; + mem_board.ReadPRG(ref addr, out cpu_reg_pc.h); + register_p = 0; + cpu_flag_i = true; + cpu_reg_ea.v = 0; + cpu_opcode = 0; + CPU_IRQ_PIN = false; + CPU_NMI_PIN = false; + cpu_suspend_nmi = false; + cpu_suspend_irq = false; + IRQFlags = 0; + } + + private static void CPUSoftReset() + { + cpu_flag_i = true; + cpu_reg_sp.v -= 3; + ushort addr = 65532; + Read(ref addr, out cpu_reg_pc.l); + addr++; + Read(ref addr, out cpu_reg_pc.h); + } + + private static void Imp____() + { + } + + private static void IndX_R_() + { + temp_add.h = 0; + Read(ref cpu_reg_pc.v, out temp_add.l); + cpu_reg_pc.v++; + Read(ref temp_add.v, out cpu_dummy); + temp_add.l += cpu_reg_x; + Read(ref temp_add.v, out cpu_reg_ea.l); + temp_add.l++; + Read(ref temp_add.v, out cpu_reg_ea.h); + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void IndX_W_() + { + temp_add.h = 0; + Read(ref cpu_reg_pc.v, out temp_add.l); + cpu_reg_pc.v++; + Read(ref temp_add.v, out cpu_dummy); + temp_add.l += cpu_reg_x; + Read(ref temp_add.v, out cpu_reg_ea.l); + temp_add.l++; + Read(ref temp_add.v, out cpu_reg_ea.h); + } + + private static void IndX_RW() + { + temp_add.h = 0; + Read(ref cpu_reg_pc.v, out temp_add.l); + cpu_reg_pc.v++; + Read(ref temp_add.v, out cpu_dummy); + temp_add.l += cpu_reg_x; + Read(ref temp_add.v, out cpu_reg_ea.l); + temp_add.l++; + Read(ref temp_add.v, out cpu_reg_ea.h); + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void IndY_R_() + { + temp_add.h = 0; + Read(ref cpu_reg_pc.v, out temp_add.l); + cpu_reg_pc.v++; + Read(ref temp_add.v, out cpu_reg_ea.l); + temp_add.l++; + Read(ref temp_add.v, out cpu_reg_ea.h); + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_m); + if (cpu_reg_ea.l < cpu_reg_y) + { + cpu_reg_ea.h++; + Read(ref cpu_reg_ea.v, out cpu_m); + } + } + + private static void IndY_W_() + { + temp_add.h = 0; + Read(ref cpu_reg_pc.v, out temp_add.l); + cpu_reg_pc.v++; + Read(ref temp_add.v, out cpu_reg_ea.l); + temp_add.l++; + Read(ref temp_add.v, out cpu_reg_ea.h); + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_m); + if (cpu_reg_ea.l < cpu_reg_y) + { + cpu_reg_ea.h++; + } + } + + private static void IndY_RW() + { + temp_add.h = 0; + Read(ref cpu_reg_pc.v, out temp_add.l); + cpu_reg_pc.v++; + Read(ref temp_add.v, out cpu_reg_ea.l); + temp_add.l++; + Read(ref temp_add.v, out cpu_reg_ea.h); + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_dummy); + if (cpu_reg_ea.l < cpu_reg_y) + { + cpu_reg_ea.h++; + } + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void Zpg_R__() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void Zpg_W__() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + } + + private static void Zpg_RW_() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void ZpgX_R_() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_x; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void ZpgX_W_() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_x; + } + + private static void ZpgX_RW() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_x; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void ZpgY_R_() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void ZpgY_W_() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_y; + } + + private static void ZpgY_RW() + { + cpu_reg_ea.h = 0; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void Imm____() + { + Read(ref cpu_reg_pc.v, out cpu_m); + cpu_reg_pc.v++; + } + + private static void ImA____() + { + Read(ref cpu_reg_pc.v, out cpu_dummy); + } + + private static void Abs_R__() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void Abs_W__() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + } + + private static void Abs_RW_() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void AbsX_R_() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + cpu_reg_ea.l += cpu_reg_x; + Read(ref cpu_reg_ea.v, out cpu_m); + if (cpu_reg_ea.l < cpu_reg_x) + { + cpu_reg_ea.h++; + Read(ref cpu_reg_ea.v, out cpu_m); + } + } + + private static void AbsX_W_() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + cpu_reg_ea.l += cpu_reg_x; + Read(ref cpu_reg_ea.v, out cpu_m); + if (cpu_reg_ea.l < cpu_reg_x) + { + cpu_reg_ea.h++; + } + } + + private static void AbsX_RW() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + cpu_reg_ea.l += cpu_reg_x; + Read(ref cpu_reg_ea.v, out cpu_dummy); + if (cpu_reg_ea.l < cpu_reg_x) + { + cpu_reg_ea.h++; + } + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void AbsY_R_() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_m); + if (cpu_reg_ea.l < cpu_reg_y) + { + cpu_reg_ea.h++; + Read(ref cpu_reg_ea.v, out cpu_m); + } + } + + private static void AbsY_W_() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_m); + if (cpu_reg_ea.l < cpu_reg_y) + { + cpu_reg_ea.h++; + } + } + + private static void AbsY_RW() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v++; + cpu_reg_ea.l += cpu_reg_y; + Read(ref cpu_reg_ea.v, out cpu_m); + if (cpu_reg_ea.l < cpu_reg_y) + { + cpu_reg_ea.h++; + } + Read(ref cpu_reg_ea.v, out cpu_m); + } + + private static void Interrupt() + { + Push(ref cpu_reg_pc.h); + Push(ref cpu_reg_pc.l); + cpu_dummy = ((cpu_opcode == 0) ? register_pb() : register_p); + Push(ref cpu_dummy); + temp_add.v = InterruptVector; + cpu_suspend_nmi = true; + cpu_flag_i = true; + CPU_NMI_PIN = false; + Read(ref temp_add.v, out cpu_reg_pc.l); + temp_add.v++; + Read(ref temp_add.v, out cpu_reg_pc.h); + cpu_suspend_nmi = false; + } + + private static void Branch(ref bool condition) + { + Read(ref cpu_reg_pc.v, out cpu_byte_temp); + cpu_reg_pc.v++; + if (!condition) + { + return; + } + cpu_suspend_irq = true; + Read(ref cpu_reg_pc.v, out cpu_dummy); + cpu_reg_pc.l += cpu_byte_temp; + cpu_suspend_irq = false; + if (cpu_byte_temp >= 128) + { + if (cpu_reg_pc.l >= cpu_byte_temp) + { + Read(ref cpu_reg_pc.v, out cpu_dummy); + cpu_reg_pc.h--; + } + } + else if (cpu_reg_pc.l < cpu_byte_temp) + { + Read(ref cpu_reg_pc.v, out cpu_dummy); + cpu_reg_pc.h++; + } + } + + private static void Push(ref byte val) + { + Write(ref cpu_reg_sp.v, ref val); + cpu_reg_sp.l--; + } + + private static void Pull(out byte val) + { + cpu_reg_sp.l++; + Read(ref cpu_reg_sp.v, out val); + } + + private static void ADC__() + { + cpu_int_temp = cpu_reg_a + cpu_m + (cpu_flag_c ? 1 : 0); + cpu_flag_v = ((cpu_int_temp ^ cpu_reg_a) & (cpu_int_temp ^ cpu_m) & 0x80) != 0; + cpu_flag_n = (cpu_int_temp & 0x80) != 0; + cpu_flag_z = (cpu_int_temp & 0xFF) == 0; + cpu_flag_c = cpu_int_temp >> 8 != 0; + cpu_reg_a = (byte)((uint)cpu_int_temp & 0xFFu); + } + + private static void AHX__() + { + cpu_byte_temp = (byte)((uint)(cpu_reg_a & cpu_reg_x) & 7u); + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + } + + private static void ALR__() + { + cpu_reg_a &= cpu_m; + cpu_flag_c = (cpu_reg_a & 1) != 0; + cpu_reg_a >>= 1; + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void ANC__() + { + cpu_reg_a &= cpu_m; + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + cpu_flag_z = cpu_reg_a == 0; + cpu_flag_c = (cpu_reg_a & 0x80) != 0; + } + + private static void AND__() + { + cpu_reg_a &= cpu_m; + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void ARR__() + { + cpu_reg_a = (byte)((uint)((cpu_m & cpu_reg_a) >> 1) | (cpu_flag_c ? 128u : 0u)); + cpu_flag_z = (cpu_reg_a & 0xFF) == 0; + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + cpu_flag_c = (cpu_reg_a & 0x40) != 0; + cpu_flag_v = (((cpu_reg_a << 1) ^ cpu_reg_a) & 0x40) != 0; + } + + private static void AXS__() + { + cpu_int_temp = (cpu_reg_a & cpu_reg_x) - cpu_m; + cpu_flag_n = (cpu_int_temp & 0x80) != 0; + cpu_flag_z = (cpu_int_temp & 0xFF) == 0; + cpu_flag_c = ~cpu_int_temp >> 8 != 0; + cpu_reg_x = (byte)((uint)cpu_int_temp & 0xFFu); + } + + private static void ASL_M() + { + cpu_flag_c = (cpu_m & 0x80) == 128; + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_m = (byte)((uint)(cpu_m << 1) & 0xFEu); + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_flag_n = (cpu_m & 0x80) == 128; + cpu_flag_z = cpu_m == 0; + } + + private static void ASL_A() + { + cpu_flag_c = (cpu_reg_a & 0x80) == 128; + cpu_reg_a = (byte)((uint)(cpu_reg_a << 1) & 0xFEu); + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void BCC__() + { + cpu_bool_tmp = !cpu_flag_c; + Branch(ref cpu_bool_tmp); + } + + private static void BCS__() + { + Branch(ref cpu_flag_c); + } + + private static void BEQ__() + { + Branch(ref cpu_flag_z); + } + + private static void BIT__() + { + cpu_flag_n = (cpu_m & 0x80) != 0; + cpu_flag_v = (cpu_m & 0x40) != 0; + cpu_flag_z = (cpu_m & cpu_reg_a) == 0; + } + + private static void BRK__() + { + Read(ref cpu_reg_pc.v, out cpu_dummy); + cpu_reg_pc.v++; + Interrupt(); + } + + private static void BPL__() + { + cpu_bool_tmp = !cpu_flag_n; + Branch(ref cpu_bool_tmp); + } + + private static void BNE__() + { + cpu_bool_tmp = !cpu_flag_z; + Branch(ref cpu_bool_tmp); + } + + private static void BMI__() + { + Branch(ref cpu_flag_n); + } + + private static void BVC__() + { + cpu_bool_tmp = !cpu_flag_v; + Branch(ref cpu_bool_tmp); + } + + private static void BVS__() + { + Branch(ref cpu_flag_v); + } + + private static void SED__() + { + cpu_flag_d = true; + } + + private static void CLC__() + { + cpu_flag_c = false; + } + + private static void CLD__() + { + cpu_flag_d = false; + } + + private static void CLV__() + { + cpu_flag_v = false; + } + + private static void CMP__() + { + cpu_int_temp = cpu_reg_a - cpu_m; + cpu_flag_n = (cpu_int_temp & 0x80) == 128; + cpu_flag_c = cpu_reg_a >= cpu_m; + cpu_flag_z = cpu_int_temp == 0; + } + + private static void CPX__() + { + cpu_int_temp = cpu_reg_x - cpu_m; + cpu_flag_n = (cpu_int_temp & 0x80) == 128; + cpu_flag_c = cpu_reg_x >= cpu_m; + cpu_flag_z = cpu_int_temp == 0; + } + + private static void CPY__() + { + cpu_int_temp = cpu_reg_y - cpu_m; + cpu_flag_n = (cpu_int_temp & 0x80) == 128; + cpu_flag_c = cpu_reg_y >= cpu_m; + cpu_flag_z = cpu_int_temp == 0; + } + + private static void CLI__() + { + cpu_flag_i = false; + } + + private static void DCP__() + { + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_m--; + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_int_temp = cpu_reg_a - cpu_m; + cpu_flag_n = (cpu_int_temp & 0x80) != 0; + cpu_flag_z = cpu_int_temp == 0; + cpu_flag_c = ~cpu_int_temp >> 8 != 0; + } + + private static void DEC__() + { + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_m--; + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_flag_n = (cpu_m & 0x80) == 128; + cpu_flag_z = cpu_m == 0; + } + + private static void DEY__() + { + cpu_reg_y--; + cpu_flag_z = cpu_reg_y == 0; + cpu_flag_n = (cpu_reg_y & 0x80) == 128; + } + + private static void DEX__() + { + cpu_reg_x--; + cpu_flag_z = cpu_reg_x == 0; + cpu_flag_n = (cpu_reg_x & 0x80) == 128; + } + + private static void EOR__() + { + cpu_reg_a ^= cpu_m; + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void INC__() + { + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_m++; + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_flag_n = (cpu_m & 0x80) == 128; + cpu_flag_z = cpu_m == 0; + } + + private static void INX__() + { + cpu_reg_x++; + cpu_flag_z = cpu_reg_x == 0; + cpu_flag_n = (cpu_reg_x & 0x80) == 128; + } + + private static void INY__() + { + cpu_reg_y++; + cpu_flag_n = (cpu_reg_y & 0x80) == 128; + cpu_flag_z = cpu_reg_y == 0; + } + + private static void ISC__() + { + Read(ref cpu_reg_ea.v, out cpu_byte_temp); + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_byte_temp++; + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_int_temp = cpu_byte_temp ^ 0xFF; + cpu_int_temp1 = cpu_reg_a + cpu_int_temp + (cpu_flag_c ? 1 : 0); + cpu_flag_n = (cpu_int_temp1 & 0x80) != 0; + cpu_flag_v = ((cpu_int_temp1 ^ cpu_reg_a) & (cpu_int_temp1 ^ cpu_int_temp) & 0x80) != 0; + cpu_flag_z = (cpu_int_temp1 & 0xFF) == 0; + cpu_flag_c = cpu_int_temp1 >> 8 != 0; + cpu_reg_a = (byte)((uint)cpu_int_temp1 & 0xFFu); + } + + private static void JMP__() + { + cpu_reg_pc.v = cpu_reg_ea.v; + } + + private static void JMP_I() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + Read(ref cpu_reg_ea.v, out cpu_reg_pc.l); + cpu_reg_ea.l++; + Read(ref cpu_reg_ea.v, out cpu_reg_pc.h); + } + + private static void JSR__() + { + Read(ref cpu_reg_pc.v, out cpu_reg_ea.l); + cpu_reg_pc.v++; + Write(ref cpu_reg_sp.v, ref cpu_reg_ea.l); + Push(ref cpu_reg_pc.h); + Push(ref cpu_reg_pc.l); + Read(ref cpu_reg_pc.v, out cpu_reg_ea.h); + cpu_reg_pc.v = cpu_reg_ea.v; + } + + private static void LAR__() + { + cpu_reg_sp.l &= cpu_m; + cpu_reg_a = cpu_reg_sp.l; + cpu_reg_x = cpu_reg_sp.l; + cpu_flag_n = (cpu_reg_sp.l & 0x80) != 0; + cpu_flag_z = (cpu_reg_sp.l & 0xFF) == 0; + } + + private static void LAX__() + { + cpu_reg_x = (cpu_reg_a = cpu_m); + cpu_flag_n = (cpu_reg_x & 0x80) != 0; + cpu_flag_z = (cpu_reg_x & 0xFF) == 0; + } + + private static void LDA__() + { + cpu_reg_a = cpu_m; + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void LDX__() + { + cpu_reg_x = cpu_m; + cpu_flag_n = (cpu_reg_x & 0x80) == 128; + cpu_flag_z = cpu_reg_x == 0; + } + + private static void LDY__() + { + cpu_reg_y = cpu_m; + cpu_flag_n = (cpu_reg_y & 0x80) == 128; + cpu_flag_z = cpu_reg_y == 0; + } + + private static void LSR_A() + { + cpu_flag_c = (cpu_reg_a & 1) == 1; + cpu_reg_a >>= 1; + cpu_flag_z = cpu_reg_a == 0; + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + } + + private static void LSR_M() + { + cpu_flag_c = (cpu_m & 1) == 1; + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_m >>= 1; + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_flag_z = cpu_m == 0; + cpu_flag_n = (cpu_m & 0x80) != 0; + } + + private static void NOP__() + { + } + + private static void ORA__() + { + cpu_reg_a |= cpu_m; + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void PHA__() + { + Push(ref cpu_reg_a); + } + + private static void PHP__() + { + cpu_dummy = register_pb(); + Push(ref cpu_dummy); + } + + private static void PLA__() + { + Read(ref cpu_reg_sp.v, out cpu_dummy); + Pull(out cpu_reg_a); + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void PLP__() + { + Read(ref cpu_reg_sp.v, out cpu_dummy); + Pull(out cpu_dummy); + register_p = cpu_dummy; + } + + private static void RLA__() + { + Read(ref cpu_reg_ea.v, out cpu_byte_temp); + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_dummy = (byte)((uint)(cpu_byte_temp << 1) | (cpu_flag_c ? 1u : 0u)); + Write(ref cpu_reg_ea.v, ref cpu_dummy); + cpu_flag_n = (cpu_dummy & 0x80) != 0; + cpu_flag_z = (cpu_dummy & 0xFF) == 0; + cpu_flag_c = (cpu_byte_temp & 0x80) != 0; + cpu_reg_a &= cpu_dummy; + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + cpu_flag_z = (cpu_reg_a & 0xFF) == 0; + } + + private static void ROL_A() + { + cpu_byte_temp = (byte)((uint)(cpu_reg_a << 1) | (cpu_flag_c ? 1u : 0u)); + cpu_flag_n = (cpu_byte_temp & 0x80) != 0; + cpu_flag_z = (cpu_byte_temp & 0xFF) == 0; + cpu_flag_c = (cpu_reg_a & 0x80) != 0; + cpu_reg_a = cpu_byte_temp; + } + + private static void ROL_M() + { + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_byte_temp = (byte)((uint)(cpu_m << 1) | (cpu_flag_c ? 1u : 0u)); + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_flag_n = (cpu_byte_temp & 0x80) != 0; + cpu_flag_z = (cpu_byte_temp & 0xFF) == 0; + cpu_flag_c = (cpu_m & 0x80) != 0; + } + + private static void ROR_A() + { + cpu_byte_temp = (byte)((uint)(cpu_reg_a >> 1) | (cpu_flag_c ? 128u : 0u)); + cpu_flag_n = (cpu_byte_temp & 0x80) != 0; + cpu_flag_z = (cpu_byte_temp & 0xFF) == 0; + cpu_flag_c = (cpu_reg_a & 1) != 0; + cpu_reg_a = cpu_byte_temp; + } + + private static void ROR_M() + { + Write(ref cpu_reg_ea.v, ref cpu_m); + cpu_byte_temp = (byte)((uint)(cpu_m >> 1) | (cpu_flag_c ? 128u : 0u)); + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_flag_n = (cpu_byte_temp & 0x80) != 0; + cpu_flag_z = (cpu_byte_temp & 0xFF) == 0; + cpu_flag_c = (cpu_m & 1) != 0; + } + + private static void RRA__() + { + Read(ref cpu_reg_ea.v, out cpu_byte_temp); + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_dummy = (byte)((uint)(cpu_byte_temp >> 1) | (cpu_flag_c ? 128u : 0u)); + Write(ref cpu_reg_ea.v, ref cpu_dummy); + cpu_flag_n = (cpu_dummy & 0x80) != 0; + cpu_flag_z = (cpu_dummy & 0xFF) == 0; + cpu_flag_c = (cpu_byte_temp & 1) != 0; + cpu_byte_temp = cpu_dummy; + cpu_int_temp = cpu_reg_a + cpu_byte_temp + (cpu_flag_c ? 1 : 0); + cpu_flag_n = (cpu_int_temp & 0x80) != 0; + cpu_flag_v = ((cpu_int_temp ^ cpu_reg_a) & (cpu_int_temp ^ cpu_byte_temp) & 0x80) != 0; + cpu_flag_z = (cpu_int_temp & 0xFF) == 0; + cpu_flag_c = cpu_int_temp >> 8 != 0; + cpu_reg_a = (byte)cpu_int_temp; + } + + private static void RTI__() + { + Read(ref cpu_reg_sp.v, out cpu_dummy); + Pull(out cpu_dummy); + register_p = cpu_dummy; + Pull(out cpu_reg_pc.l); + Pull(out cpu_reg_pc.h); + } + + private static void RTS__() + { + Read(ref cpu_reg_sp.v, out cpu_dummy); + Pull(out cpu_reg_pc.l); + Pull(out cpu_reg_pc.h); + cpu_reg_pc.v++; + Read(ref cpu_reg_pc.v, out cpu_dummy); + } + + private static void SAX__() + { + cpu_dummy = (byte)(cpu_reg_x & cpu_reg_a); + Write(ref cpu_reg_ea.v, ref cpu_dummy); + } + + private static void SBC__() + { + cpu_m ^= byte.MaxValue; + cpu_int_temp = cpu_reg_a + cpu_m + (cpu_flag_c ? 1 : 0); + cpu_flag_n = (cpu_int_temp & 0x80) != 0; + cpu_flag_v = ((cpu_int_temp ^ cpu_reg_a) & (cpu_int_temp ^ cpu_m) & 0x80) != 0; + cpu_flag_z = (cpu_int_temp & 0xFF) == 0; + cpu_flag_c = cpu_int_temp >> 8 != 0; + cpu_reg_a = (byte)cpu_int_temp; + } + + private static void SEC__() + { + cpu_flag_c = true; + } + + private static void SEI__() + { + cpu_flag_i = true; + } + + private static void SHX__() + { + cpu_byte_temp = (byte)(cpu_reg_x & (cpu_reg_ea.h + 1)); + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_y; + if (cpu_reg_ea.l < cpu_reg_y) + { + cpu_reg_ea.h = cpu_byte_temp; + } + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + } + + private static void SHY__() + { + cpu_byte_temp = (byte)(cpu_reg_y & (cpu_reg_ea.h + 1)); + Read(ref cpu_reg_ea.v, out cpu_dummy); + cpu_reg_ea.l += cpu_reg_x; + if (cpu_reg_ea.l < cpu_reg_x) + { + cpu_reg_ea.h = cpu_byte_temp; + } + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + } + + private static void SLO__() + { + Read(ref cpu_reg_ea.v, out cpu_byte_temp); + cpu_flag_c = (cpu_byte_temp & 0x80) != 0; + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_byte_temp <<= 1; + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_flag_n = (cpu_byte_temp & 0x80) != 0; + cpu_flag_z = (cpu_byte_temp & 0xFF) == 0; + cpu_reg_a |= cpu_byte_temp; + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + cpu_flag_z = (cpu_reg_a & 0xFF) == 0; + } + + private static void SRE__() + { + Read(ref cpu_reg_ea.v, out cpu_byte_temp); + cpu_flag_c = (cpu_byte_temp & 1) != 0; + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_byte_temp >>= 1; + Write(ref cpu_reg_ea.v, ref cpu_byte_temp); + cpu_flag_n = (cpu_byte_temp & 0x80) != 0; + cpu_flag_z = (cpu_byte_temp & 0xFF) == 0; + cpu_reg_a ^= cpu_byte_temp; + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + cpu_flag_z = (cpu_reg_a & 0xFF) == 0; + } + + private static void STA__() + { + Write(ref cpu_reg_ea.v, ref cpu_reg_a); + } + + private static void STX__() + { + Write(ref cpu_reg_ea.v, ref cpu_reg_x); + } + + private static void STY__() + { + Write(ref cpu_reg_ea.v, ref cpu_reg_y); + } + + private static void TAX__() + { + cpu_reg_x = cpu_reg_a; + cpu_flag_n = (cpu_reg_x & 0x80) == 128; + cpu_flag_z = cpu_reg_x == 0; + } + + private static void TAY__() + { + cpu_reg_y = cpu_reg_a; + cpu_flag_n = (cpu_reg_y & 0x80) == 128; + cpu_flag_z = cpu_reg_y == 0; + } + + private static void TSX__() + { + cpu_reg_x = cpu_reg_sp.l; + cpu_flag_n = (cpu_reg_x & 0x80) != 0; + cpu_flag_z = cpu_reg_x == 0; + } + + private static void TXA__() + { + cpu_reg_a = cpu_reg_x; + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void TXS__() + { + cpu_reg_sp.l = cpu_reg_x; + } + + private static void TYA__() + { + cpu_reg_a = cpu_reg_y; + cpu_flag_n = (cpu_reg_a & 0x80) == 128; + cpu_flag_z = cpu_reg_a == 0; + } + + private static void XAA__() + { + cpu_reg_a = (byte)(cpu_reg_x & cpu_m); + cpu_flag_n = (cpu_reg_a & 0x80) != 0; + cpu_flag_z = (cpu_reg_a & 0xFF) == 0; + } + + private static void XAS__() + { + cpu_reg_sp.l = (byte)(cpu_reg_a & cpu_reg_x); + Write(ref cpu_reg_ea.v, ref cpu_reg_sp.l); + } + + private static void CPUWriteState(ref BinaryWriter bin) + { + bin.Write(cpu_reg_pc.v); + bin.Write(cpu_reg_sp.v); + bin.Write(cpu_reg_ea.v); + bin.Write(cpu_reg_a); + bin.Write(cpu_reg_x); + bin.Write(cpu_reg_y); + bin.Write(cpu_flag_n); + bin.Write(cpu_flag_v); + bin.Write(cpu_flag_d); + bin.Write(cpu_flag_i); + bin.Write(cpu_flag_z); + bin.Write(cpu_flag_c); + bin.Write(cpu_m); + bin.Write(cpu_opcode); + bin.Write(cpu_byte_temp); + bin.Write(cpu_int_temp); + bin.Write(cpu_int_temp1); + bin.Write(cpu_dummy); + bin.Write(cpu_bool_tmp); + bin.Write(temp_add.v); + bin.Write(CPU_IRQ_PIN); + bin.Write(CPU_NMI_PIN); + bin.Write(cpu_suspend_nmi); + bin.Write(cpu_suspend_irq); + } + + private static void CPUReadState(ref BinaryReader bin) + { + cpu_reg_pc.v = bin.ReadUInt16(); + cpu_reg_sp.v = bin.ReadUInt16(); + cpu_reg_ea.v = bin.ReadUInt16(); + cpu_reg_a = bin.ReadByte(); + cpu_reg_x = bin.ReadByte(); + cpu_reg_y = bin.ReadByte(); + cpu_flag_n = bin.ReadBoolean(); + cpu_flag_v = bin.ReadBoolean(); + cpu_flag_d = bin.ReadBoolean(); + cpu_flag_i = bin.ReadBoolean(); + cpu_flag_z = bin.ReadBoolean(); + cpu_flag_c = bin.ReadBoolean(); + cpu_m = bin.ReadByte(); + cpu_opcode = bin.ReadByte(); + cpu_byte_temp = bin.ReadByte(); + cpu_int_temp = bin.ReadInt32(); + cpu_int_temp1 = bin.ReadInt32(); + cpu_dummy = bin.ReadByte(); + cpu_bool_tmp = bin.ReadBoolean(); + temp_add.v = bin.ReadUInt16(); + CPU_IRQ_PIN = bin.ReadBoolean(); + CPU_NMI_PIN = bin.ReadBoolean(); + cpu_suspend_nmi = bin.ReadBoolean(); + cpu_suspend_irq = bin.ReadBoolean(); + } + + private static void DMAHardReset() + { + dma_DMCDMAWaitCycles = 0; + dma_OAMDMAWaitCycles = 0; + dma_isOamDma = false; + dma_oamdma_i = 0; + dma_DMCOn = false; + dma_OAMOn = false; + dma_DMC_occurring = false; + dma_OAM_occurring = false; + dma_OAMFinishCounter = 0; + dma_Oamaddress = 0; + dma_OAMCYCLE = 0; + dma_latch = 0; + reg_2004 = 8196; + } + + private static void DMASoftReset() + { + dma_DMCDMAWaitCycles = 0; + dma_OAMDMAWaitCycles = 0; + dma_isOamDma = false; + dma_oamdma_i = 0; + dma_DMCOn = false; + dma_OAMOn = false; + dma_DMC_occurring = false; + dma_OAM_occurring = false; + dma_OAMFinishCounter = 0; + dma_Oamaddress = 0; + dma_OAMCYCLE = 0; + dma_latch = 0; + } + + internal static void AssertDMCDMA() + { + if (dma_OAM_occurring) + { + if (dma_OAMCYCLE < 508) + { + dma_DMCDMAWaitCycles = (BUS_RW ? 1 : 0); + } + else + { + dma_DMCDMAWaitCycles = 4 - (512 - dma_OAMCYCLE); + } + } + else + { + if (dma_DMC_occurring) + { + return; + } + dma_DMCDMAWaitCycles = (BUS_RW ? 3 : 2); + if (dma_OAMFinishCounter == 3) + { + dma_DMCDMAWaitCycles++; + } + } + dma_isOamDma = false; + dma_DMCOn = true; + } + + private static void AssertOAMDMA() + { + if (!dma_OAM_occurring) + { + dma_OAMDMAWaitCycles = (apu_odd_cycle ? 1 : 2); + dma_isOamDma = true; + dma_OAMOn = true; + } + } + + private static void DMAClock() + { + if (dma_OAMFinishCounter > 0) + { + dma_OAMFinishCounter--; + } + if (!BUS_RW) + { + if (dma_DMCDMAWaitCycles > 0) + { + dma_DMCDMAWaitCycles--; + } + if (dma_OAMDMAWaitCycles > 0) + { + dma_OAMDMAWaitCycles--; + } + return; + } + if (dma_DMCOn) + { + dma_DMC_occurring = true; + dma_DMCOn = false; + if (dma_DMCDMAWaitCycles > 0) + { + if (BUS_ADDRESS == 16406 || BUS_ADDRESS == 16407) + { + Read(ref BUS_ADDRESS, out dma_dummy); + dma_DMCDMAWaitCycles--; + while (dma_DMCDMAWaitCycles > 0) + { + EmuClockComponents(); + dma_DMCDMAWaitCycles--; + } + } + else + { + if (dma_DMCDMAWaitCycles > 0) + { + EmuClockComponents(); + dma_DMCDMAWaitCycles--; + } + while (dma_DMCDMAWaitCycles > 0) + { + Read(ref BUS_ADDRESS, out dma_dummy); + dma_DMCDMAWaitCycles--; + } + } + } + DMCDoDMA(); + dma_DMC_occurring = false; + } + if (!dma_OAMOn) + { + return; + } + dma_OAM_occurring = true; + dma_OAMOn = false; + if (dma_OAMDMAWaitCycles > 0) + { + if (BUS_ADDRESS == 16406 || BUS_ADDRESS == 16407) + { + Read(ref BUS_ADDRESS, out dma_dummy); + dma_OAMDMAWaitCycles--; + while (dma_OAMDMAWaitCycles > 0) + { + EmuClockComponents(); + dma_OAMDMAWaitCycles--; + } + } + else + { + if (dma_OAMDMAWaitCycles > 0) + { + EmuClockComponents(); + dma_OAMDMAWaitCycles--; + } + while (dma_OAMDMAWaitCycles > 0) + { + Read(ref BUS_ADDRESS, out dma_dummy); + dma_OAMDMAWaitCycles--; + } + } + } + dma_OAMCYCLE = 0; + for (dma_oamdma_i = 0; dma_oamdma_i < 256; dma_oamdma_i++) + { + Read(ref dma_Oamaddress, out dma_latch); + dma_OAMCYCLE++; + Write(ref reg_2004, ref dma_latch); + dma_OAMCYCLE++; + dma_Oamaddress = (ushort)(++dma_Oamaddress & 0xFFFFu); + } + dma_OAMCYCLE = 0; + dma_OAMFinishCounter = 5; + dma_OAM_occurring = false; + } + + private static void DMAWriteState(ref BinaryWriter bin) + { + bin.Write(dma_DMCDMAWaitCycles); + bin.Write(dma_OAMDMAWaitCycles); + bin.Write(dma_isOamDma); + bin.Write(dma_oamdma_i); + bin.Write(dma_DMCOn); + bin.Write(dma_OAMOn); + bin.Write(dma_DMC_occurring); + bin.Write(dma_OAM_occurring); + bin.Write(dma_OAMFinishCounter); + bin.Write(dma_Oamaddress); + bin.Write(dma_OAMCYCLE); + bin.Write(dma_latch); + bin.Write(dma_dummy); + } + + private static void DMAReadState(ref BinaryReader bin) + { + dma_DMCDMAWaitCycles = bin.ReadInt32(); + dma_OAMDMAWaitCycles = bin.ReadInt32(); + dma_isOamDma = bin.ReadBoolean(); + dma_oamdma_i = bin.ReadInt32(); + dma_DMCOn = bin.ReadBoolean(); + dma_OAMOn = bin.ReadBoolean(); + dma_DMC_occurring = bin.ReadBoolean(); + dma_OAM_occurring = bin.ReadBoolean(); + dma_OAMFinishCounter = bin.ReadInt32(); + dma_Oamaddress = bin.ReadUInt16(); + dma_OAMCYCLE = bin.ReadInt32(); + dma_latch = bin.ReadByte(); + dma_dummy = bin.ReadByte(); + } + + private static void PollInterruptStatus() + { + if (!cpu_suspend_nmi) + { + if (PPU_NMI_Current & !PPU_NMI_Old) + { + CPU_NMI_PIN = true; + } + PPU_NMI_Old = (PPU_NMI_Current = false); + } + if (!cpu_suspend_irq) + { + CPU_IRQ_PIN = !cpu_flag_i && IRQFlags != 0; + } + if (CPU_NMI_PIN) + { + InterruptVector = 65530; + } + else + { + InterruptVector = 65534; + } + } + + private static void InterruptsWriteState(ref BinaryWriter bin) + { + bin.Write(IRQFlags); + bin.Write(PPU_NMI_Current); + bin.Write(PPU_NMI_Old); + bin.Write(InterruptVector); + } + + private static void InterruptsReadState(ref BinaryReader bin) + { + IRQFlags = bin.ReadInt32(); + PPU_NMI_Current = bin.ReadBoolean(); + PPU_NMI_Old = bin.ReadBoolean(); + InterruptVector = bin.ReadUInt16(); + } + + public static void SetupGameGenie(bool IsGameGenieActive, GameGenieCode[] GameGenieCodes) + { + if (mem_board != null) + { + mem_board.SetupGameGenie(IsGameGenieActive, GameGenieCodes); + } + } + + private static void MEMInitialize(IRom rom) + { + Tracer.WriteLine("Looking for mapper # " + rom.MapperNumber + "...."); + if (MyNesMain.IsBoardExist(rom.MapperNumber)) + { + Tracer.WriteLine("Mapper # " + rom.MapperNumber + " located, assigning..."); + mem_board = MyNesMain.GetBoard(rom.MapperNumber); + Tracer.WriteInformation("Mapper # " + rom.MapperNumber + " assigned successfully."); + if (mem_board.HasIssues) + { + Tracer.WriteWarning(MNInterfaceLanguage.Mapper + " # " + mem_board.MapperNumber + " [" + mem_board.Name + "] " + MNInterfaceLanguage.Message_Error17); + MyNesMain.VideoProvider.WriteWarningNotification(MNInterfaceLanguage.Mapper + " # " + mem_board.MapperNumber + " [" + mem_board.Name + "] " + MNInterfaceLanguage.Message_Error17, instant: false); + } + } + else + { + Tracer.WriteError("Mapper # " + rom.MapperNumber + " IS NOT LOCATED, mapper is not supported or unable to find it."); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Mapper + " # " + rom.MapperNumber + " " + MNInterfaceLanguage.Message_Error14, instant: false); + mem_board = MyNesMain.GetBoard(0); + Tracer.WriteWarning("Mapper # 0 [NROM] will be used instead, assigned successfully."); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Mapper + " # 0 [NROM] " + MNInterfaceLanguage.Message_Error15, instant: false); + } + mem_read_accesses = new MemReadAccess[65536]; + mem_write_accesses = new MemWriteAccess[65536]; + MEMMap(MEMReadWRAM, new ushort[2] { 0, 4096 }); + MEMMap(MEMWriteWRAM, new ushort[2] { 0, 4096 }); + MEMMap(PPUIORead, new ushort[2] { 8192, 12288 }); + MEMMap(PPUIOWrite, new ushort[2] { 8192, 12288 }); + MEMMap(APUIORead, new ushort[1] { 16384 }); + MEMMap(APUIOWrite, new ushort[1] { 16384 }); + MEMMap(mem_board.ReadEX, new ushort[1] { 20480 }); + MEMMap(mem_board.WriteEX, new ushort[1] { 20480 }); + MEMMap(mem_board.ReadSRM, new ushort[2] { 24576, 28672 }); + MEMMap(mem_board.WriteSRM, new ushort[2] { 24576, 28672 }); + MEMMap(mem_board.ReadPRG, new ushort[8] { 32768, 36864, 40960, 45056, 49152, 53248, 57344, 61440 }); + MEMMap(mem_board.WritePRG, new ushort[8] { 32768, 36864, 40960, 45056, 49152, 53248, 57344, 61440 }); + mem_board.Initialize(rom); + mem_wram = new byte[2048]; + } + + private static void MEMHardReset() + { + mem_wram = new byte[2048]; + mem_wram[8] = 247; + mem_wram[9] = 239; + mem_wram[10] = 223; + mem_wram[15] = 191; + Tracer.WriteLine("Reading SRAM ..."); + SRAMFileName = Path.Combine(MyNesMain.EmuSettings.SRAMFolder, Path.GetFileNameWithoutExtension(CurrentFilePath) + ".srm"); + if (File.Exists(SRAMFileName)) + { + Stream stream = new FileStream(SRAMFileName, FileMode.Open, FileAccess.Read); + byte[] array = new byte[stream.Length]; + stream.Read(array, 0, array.Length); + stream.Flush(); + stream.Close(); + byte[] outData = new byte[0]; + ZlipWrapper.DecompressData(array, out outData); + mem_board.LoadSRAM(outData); + Tracer.WriteLine("SRAM read successfully."); + } + else + { + Tracer.WriteLine("SRAM file not found; rom has no SRAM or file not exist."); + } + ReloadGameGenieCodes(); + mem_board.HardReset(); + } + + public static void ReloadGameGenieCodes() + { + Tracer.WriteLine("Reading game genie codes (if available)...."); + GMFileName = Path.Combine(MyNesMain.EmuSettings.GameGenieFolder, Path.GetFileNameWithoutExtension(CurrentFilePath) + ".txt"); + mem_board.GameGenieCodes = new GameGenieCode[0]; + if (File.Exists(GMFileName)) + { + XmlReaderSettings xmlReaderSettings = new XmlReaderSettings(); + xmlReaderSettings.DtdProcessing = DtdProcessing.Ignore; + xmlReaderSettings.IgnoreWhitespace = true; + XmlReader xmlReader = XmlReader.Create(GMFileName, xmlReaderSettings); + xmlReader.Read(); + xmlReader.Read(); + if (xmlReader.Name != "MyNesGameGenieCodesList") + { + xmlReader.Close(); + return; + } + GameGenie gameGenie = new GameGenie(); + List list = new List(); + while (xmlReader.Read()) + { + if (xmlReader.Name == "Code") + { + GameGenieCode item = default(GameGenieCode); + item.Enabled = true; + xmlReader.MoveToAttribute("code"); + item.Name = xmlReader.Value.ToString(); + if (item.Name.Length == 6) + { + item.Address = gameGenie.GetGGAddress(gameGenie.GetCodeAsHEX(item.Name), 6) | 0x8000; + item.Value = gameGenie.GetGGValue(gameGenie.GetCodeAsHEX(item.Name), 6); + item.IsCompare = false; + } + else + { + item.Address = gameGenie.GetGGAddress(gameGenie.GetCodeAsHEX(item.Name), 8) | 0x8000; + item.Value = gameGenie.GetGGValue(gameGenie.GetCodeAsHEX(item.Name), 8); + item.Compare = gameGenie.GetGGCompareValue(gameGenie.GetCodeAsHEX(item.Name)); + item.IsCompare = true; + } + list.Add(item); + } + } + xmlReader.Close(); + if (list.Count > 0) + { + mem_board.GameGenieCodes = list.ToArray(); + Tracer.WriteInformation("Game Genie codes loaded successfully, total of " + list.Count); + } + else + { + Tracer.WriteError("There is no Game Genie code in the file to load."); + } + } + else + { + Tracer.WriteWarning("No Game Genie file found for this game."); + } + } + + private static void MEMMap(MemReadAccess readAccess, ushort[] addresses) + { + for (int i = 0; i < addresses.Length; i++) + { + mem_read_accesses[(addresses[i] & 0xF000) >> 12] = readAccess; + } + } + + private static void MEMMap(MemWriteAccess writeAccess, ushort[] addresses) + { + for (int i = 0; i < addresses.Length; i++) + { + mem_write_accesses[(addresses[i] & 0xF000) >> 12] = writeAccess; + } + } + + private static void MEMReadWRAM(ref ushort addr, out byte value) + { + value = mem_wram[addr & 0x7FF]; + } + + private static void MEMWriteWRAM(ref ushort addr, ref byte value) + { + mem_wram[addr & 0x7FF] = value; + } + + internal static void Read(ref ushort addr, out byte value) + { + BUS_RW = true; + BUS_ADDRESS = addr; + EmuClockComponents(); + mem_read_accesses[(addr & 0xF000) >> 12](ref addr, out value); + } + + private static void Write(ref ushort addr, ref byte value) + { + BUS_RW = false; + BUS_ADDRESS = addr; + EmuClockComponents(); + mem_write_accesses[(addr & 0xF000) >> 12](ref addr, ref value); + } + + internal static void SaveSRAM() + { + if (mem_board != null && MyNesMain.EmuSettings.SaveSRAMAtEmuShutdown && mem_board.SRAMSaveRequired) + { + Tracer.WriteLine("Saving SRAM ..."); + byte[] outData = new byte[0]; + ZlipWrapper.CompressData(mem_board.GetSRAMBuffer(), out outData); + Stream stream = new FileStream(SRAMFileName, FileMode.Create, FileAccess.Write); + stream.Write(outData, 0, outData.Length); + stream.Flush(); + stream.Close(); + Tracer.WriteLine("SRAM saved successfully."); + } + } + + private static void MEMWriteState(ref BinaryWriter bin) + { + mem_board.WriteStateData(ref bin); + bin.Write(mem_wram); + bin.Write(BUS_RW); + bin.Write(BUS_ADDRESS); + } + + private static void MEMReadState(ref BinaryReader bin) + { + mem_board.ReadStateData(ref bin); + bin.Read(mem_wram, 0, mem_wram.Length); + BUS_RW = bin.ReadBoolean(); + BUS_ADDRESS = bin.ReadUInt16(); + } + + private static void PORTSInitialize() + { + if (joypad1 == null) + { + joypad1 = new BlankJoypad(); + } + if (joypad2 == null) + { + joypad2 = new BlankJoypad(); + } + if (joypad3 == null) + { + joypad3 = new BlankJoypad(); + } + if (joypad4 == null) + { + joypad4 = new BlankJoypad(); + } + if (shortucts == null) + { + shortucts = new BlankShortuctsHandler(); + } + } + + public static void SetupShortcutsHandler(IShortcutsHandler hh) + { + shortucts = hh; + } + + public static void SetupControllers(IJoypadConnecter joy1, IJoypadConnecter joy2, IJoypadConnecter joy3, IJoypadConnecter joy4) + { + joypad1 = joy1; + joypad2 = joy2; + joypad3 = joy3; + joypad4 = joy4; + } + + public static void SetupVSUnisystemDIP(IVSUnisystemDIPConnecter uni) + { + } + + public static void SetupControllersP1(IJoypadConnecter joy) + { + joypad1 = joy; + } + + public static void SetupControllersP2(IJoypadConnecter joy) + { + joypad2 = joy; + } + + public static void SetupControllersP3(IJoypadConnecter joy) + { + joypad3 = joy; + } + + public static void SetupControllersP4(IJoypadConnecter joy) + { + joypad4 = joy; + } + + public static void DestroyJoypads() + { + if (joypad1 == null) + { + joypad1 = new BlankJoypad(); + } + else + { + joypad1.Destroy(); + } + if (joypad2 == null) + { + joypad2 = new BlankJoypad(); + } + else + { + joypad1.Destroy(); + } + if (joypad3 == null) + { + joypad3 = new BlankJoypad(); + } + else + { + joypad1.Destroy(); + } + if (joypad4 == null) + { + joypad4 = new BlankJoypad(); + } + else + { + joypad1.Destroy(); + } + } + + private static void PORTWriteState(ref BinaryWriter bin) + { + bin.Write(PORT0); + bin.Write(PORT1); + bin.Write(inputStrobe); + } + + private static void PORTReadState(ref BinaryReader bin) + { + PORT0 = bin.ReadInt32(); + PORT1 = bin.ReadInt32(); + inputStrobe = bin.ReadInt32(); + } + + private static void PPUInitialize() + { + ppu_reg_update_func = new Action[8] { PPUOnRegister2000, PPUOnRegister2001, PPUOnRegister2002, PPUOnRegister2003, PPUOnRegister2004, PPUOnRegister2005, PPUOnRegister2006, PPUOnRegister2007 }; + ppu_reg_read_func = new Action[8] { PPURead2000, PPURead2001, PPURead2002, PPURead2003, PPURead2004, PPURead2005, PPURead2006, PPURead2007 }; + ppu_bkg_fetches = new Action[8] { PPUBKFetch0, PPUBKFetch1, PPUBKFetch2, PPUBKFetch3, PPUBKFetch4, PPUBKFetch5, PPUBKFetch6, PPUBKFetch7 }; + ppu_spr_fetches = new Action[8] { PPUBKFetch0, PPUBKFetch1, PPUBKFetch2, PPUBKFetch3, PPUSPRFetch0, PPUSPRFetch1, PPUSPRFetch2, PPUSPRFetch3 }; + ppu_oam_phases = new Action[9] { PPUOamPhase0, PPUOamPhase1, PPUOamPhase2, PPUOamPhase3, PPUOamPhase4, PPUOamPhase5, PPUOamPhase6, PPUOamPhase7, PPUOamPhase8 }; + ppu_h_clocks = new Action[341]; + ppu_h_clocks[0] = PPUHClock_000_Idle; + for (int i = 1; i < 257; i++) + { + ppu_h_clocks[i] = PPUHClock_1_256_BKGClocks; + } + for (int j = 257; j < 321; j++) + { + ppu_h_clocks[j] = PPUHClock_257_320_SPRClocks; + } + for (int k = 321; k < 337; k++) + { + ppu_h_clocks[k] = PPUHClock_321_336_DUMClocks; + } + for (int l = 337; l < 341; l++) + { + ppu_h_clocks[l] = PPUHClock_337_340_DUMClocks; + } + ppu_v_clocks = new Action[320]; + for (int m = 0; m < 240; m++) + { + ppu_v_clocks[m] = PPUScanlineRender; + } + ppu_v_clocks[240] = PPUScanlineVBLANK; + ppu_oam_bank = new byte[256]; + ppu_oam_bank_secondary = new byte[32]; + ppu_palette_bank = new byte[32]; + ppu_bkg_pixels = new int[512]; + ppu_spr_pixels = new int[512]; + ppu_screen_pixels = new int[61440]; + } + + private static void PPUHardReset() + { + ppu_reg_2001_grayscale = false; + switch (Region) + { + case EmuRegion.NTSC: + ppu_clock_vblank_start = 241; + ppu_clock_vblank_end = 261; + ppu_use_odd_cycle = true; + break; + case EmuRegion.PALB: + ppu_clock_vblank_start = 241; + ppu_clock_vblank_end = 311; + ppu_use_odd_cycle = false; + break; + case EmuRegion.DENDY: + { + ppu_clock_vblank_start = 291; + ppu_clock_vblank_end = 311; + for (int i = 241; i <= 290; i++) + { + ppu_v_clocks[i] = PPUScanlineVBLANK; + } + ppu_use_odd_cycle = false; + break; + } + } + ppu_v_clocks[ppu_clock_vblank_start] = PPUScanlineVBLANKStart; + for (int j = ppu_clock_vblank_start + 1; j <= ppu_clock_vblank_end - 1; j++) + { + ppu_v_clocks[j] = PPUScanlineVBLANK; + } + ppu_v_clocks[ppu_clock_vblank_end] = PPUScanlineVBLANKEnd; + ppu_oam_bank = new byte[256]; + ppu_oam_bank_secondary = new byte[32]; + PPUOamReset(); + ppu_palette_bank = new byte[32] + { + 9, 1, 0, 1, 0, 2, 2, 13, 8, 16, + 8, 36, 0, 0, 4, 44, 9, 1, 52, 3, + 0, 4, 0, 20, 8, 58, 0, 2, 0, 32, + 44, 8 + }; + ppu_reg_io_db = 0; + ppu_reg_io_addr = 0; + ppu_reg_access_happened = false; + ppu_reg_access_w = false; + ppu_reg_2000_vram_address_increament = 1; + ppu_reg_2000_sprite_pattern_table_address_for_8x8_sprites = 0; + ppu_reg_2000_background_pattern_table_address = 0; + ppu_reg_2000_Sprite_size = 0; + ppu_reg_2000_VBI = false; + ppu_reg_2001_show_background_in_leftmost_8_pixels_of_screen = false; + ppu_reg_2001_show_sprites_in_leftmost_8_pixels_of_screen = false; + ppu_reg_2001_show_background = false; + ppu_reg_2001_show_sprites = false; + ppu_reg_2001_grayscale = false; + ppu_reg_2001_emphasis = 0; + ppu_reg_2002_SpriteOverflow = false; + ppu_reg_2002_Sprite0Hit = false; + ppu_reg_2002_VblankStartedFlag = false; + ppu_reg_2003_oam_addr = 0; + ppu_is_sprfetch = false; + ppu_use_odd_swap = false; + ppu_clock_h = 0; + ppu_clock_v = 0; + } + + private static void PPUClock() + { + mem_board.OnPPUClock(); + ppu_v_clocks[ppu_clock_v](); + ppu_clock_h++; + if (ppu_clock_h >= 341) + { + mem_board.OnPPUScanlineTick(); + if (ppu_clock_v == ppu_clock_vblank_end) + { + ppu_clock_v = 0; + } + else + { + ppu_clock_v++; + } + ppu_clock_h = 0; + } + if (ppu_reg_access_happened) + { + ppu_reg_access_happened = false; + ppu_reg_update_func[ppu_reg_io_addr](); + } + } + + public static int GetPixel(int x, int y) + { + return ppu_screen_pixels[y * 256 + x]; + } + + private static void PPUScanlineRender() + { + ppu_h_clocks[ppu_clock_h](); + } + + private static void PPUScanlineVBLANKStart() + { + ppu_is_nmi_time = (ppu_clock_h >= 1) & (ppu_clock_h <= 3); + if (ppu_is_nmi_time) + { + if (ppu_clock_h == 1) + { + ppu_reg_2002_VblankStartedFlag = true; + ppu_frame_finished = true; + } + PPU_NMI_Current = ppu_reg_2002_VblankStartedFlag & ppu_reg_2000_VBI; + } + } + + private static void PPUScanlineVBLANKEnd() + { + ppu_is_nmi_time = (ppu_clock_h >= 1) & (ppu_clock_h <= 3); + if (ppu_clock_h == 1) + { + ppu_reg_2002_Sprite0Hit = false; + ppu_reg_2002_VblankStartedFlag = false; + ppu_reg_2002_SpriteOverflow = false; + } + PPUScanlineRender(); + if (ppu_use_odd_cycle && ppu_clock_h == 339) + { + ppu_use_odd_swap = !ppu_use_odd_swap; + if (!ppu_use_odd_swap & (ppu_reg_2001_show_background || ppu_reg_2001_show_sprites)) + { + ppu_odd_swap_done = true; + ppu_clock_h++; + } + } + } + + private static void PPUScanlineVBLANK() + { + } + + private static void PPUHClock_000_Idle() + { + if (ppu_odd_swap_done) + { + ppu_bkg_fetches[1](); + ppu_odd_swap_done = false; + } + } + + private static void PPUHClock_1_256_BKGClocks() + { + if (ppu_reg_2001_show_background || ppu_reg_2001_show_sprites) + { + if (ppu_clock_v != ppu_clock_vblank_end) + { + if (ppu_clock_h > 0 && ppu_clock_h < 65) + { + ppu_oam_bank_secondary[(ppu_clock_h - 1) & 0x1F] = byte.MaxValue; + } + else + { + if (ppu_clock_h == 65) + { + PPUOamReset(); + } + if (((ppu_clock_h - 1) & 1) == 0) + { + PPUOamEvFetch(); + } + else + { + ppu_oam_phases[ppu_phase_index](); + } + if (ppu_clock_h == 256) + { + PPUOamClear(); + } + } + } + ppu_bkg_fetches[(ppu_clock_h - 1) & 7](); + if (ppu_clock_v < 240) + { + RenderPixel(); + } + } + else if (ppu_clock_v < 240) + { + ppu_screen_pixels[ppu_clock_h - 1 + ppu_clock_v * 256] = MakeRGBColor(ppu_vram_addr); + } + } + + private static void PPUHClock_257_320_SPRClocks() + { + if (ppu_reg_2001_show_background || ppu_reg_2001_show_sprites) + { + ppu_spr_fetches[(ppu_clock_h - 1) & 7](); + if (ppu_clock_h == 257) + { + ppu_vram_addr = (ushort)((ppu_vram_addr & 0x7BE0u) | (ppu_vram_addr_temp & 0x41Fu)); + } + if (ppu_clock_v == ppu_clock_vblank_end && ppu_clock_h >= 280 && ppu_clock_h <= 304) + { + ppu_vram_addr = (ushort)((ppu_vram_addr & 0x41Fu) | (ppu_vram_addr_temp & 0x7BE0u)); + } + } + } + + private static void PPUHClock_321_336_DUMClocks() + { + if (ppu_reg_2001_show_background || ppu_reg_2001_show_sprites) + { + ppu_bkg_fetches[(ppu_clock_h - 1) & 7](); + } + } + + private static void PPUHClock_337_340_DUMClocks() + { + if (ppu_reg_2001_show_background || ppu_reg_2001_show_sprites) + { + ppu_bkg_fetches[(ppu_clock_h - 1) & 1](); + } + } + + private static void PPUBKFetch0() + { + ppu_bkgfetch_nt_addr = (ushort)(0x2000u | (ppu_vram_addr & 0xFFFu)); + mem_board.OnPPUAddressUpdate(ref ppu_bkgfetch_nt_addr); + } + + private static void PPUBKFetch1() + { + mem_board.ReadNMT(ref ppu_bkgfetch_nt_addr, out ppu_bkgfetch_nt_data); + } + + private static void PPUBKFetch2() + { + ppu_bkgfetch_at_addr = (ushort)(0x23C0u | (ppu_vram_addr & 0xC00u) | ((uint)(ppu_vram_addr >> 4) & 0x38u) | ((uint)(ppu_vram_addr >> 2) & 7u)); + mem_board.OnPPUAddressUpdate(ref ppu_bkgfetch_at_addr); + } + + private static void PPUBKFetch3() + { + mem_board.ReadNMT(ref ppu_bkgfetch_at_addr, out ppu_bkgfetch_at_data); + ppu_bkgfetch_at_data = (byte)(ppu_bkgfetch_at_data >> (((ppu_vram_addr >> 4) & 4) | (ppu_vram_addr & 2))); + } + + private static void PPUBKFetch4() + { + ppu_bkgfetch_lb_addr = (ushort)((uint)(ppu_reg_2000_background_pattern_table_address | (ppu_bkgfetch_nt_data << 4)) | ((uint)(ppu_vram_addr >> 12) & 7u)); + mem_board.OnPPUAddressUpdate(ref ppu_bkgfetch_lb_addr); + } + + private static void PPUBKFetch5() + { + mem_board.ReadCHR(ref ppu_bkgfetch_lb_addr, out ppu_bkgfetch_lb_data); + } + + private static void PPUBKFetch6() + { + ppu_bkgfetch_hb_addr = (ushort)((uint)(ppu_reg_2000_background_pattern_table_address | (ppu_bkgfetch_nt_data << 4)) | 8u | ((uint)(ppu_vram_addr >> 12) & 7u)); + mem_board.OnPPUAddressUpdate(ref ppu_bkgfetch_hb_addr); + } + + private static void PPUBKFetch7() + { + mem_board.ReadCHR(ref ppu_bkgfetch_hb_addr, out ppu_bkgfetch_hb_data); + ppu_bkg_render_pos = ppu_clock_h + 8; + ppu_bkg_render_pos %= 336; + if (ppu_clock_h == 256) + { + if ((ppu_vram_addr & 0x7000) != 28672) + { + ppu_vram_addr += 4096; + } + else + { + ppu_vram_addr ^= 28672; + switch (ppu_vram_addr & 0x3E0) + { + case 928: + ppu_vram_addr ^= 2976; + break; + case 992: + ppu_vram_addr ^= 992; + break; + default: + ppu_vram_addr += 32; + break; + } + } + } + else if ((ppu_vram_addr & 0x1F) == 31) + { + ppu_vram_addr ^= 1055; + } + else + { + ppu_vram_addr++; + } + for (ppu_bkg_render_i = 0; ppu_bkg_render_i < 8; ppu_bkg_render_i++) + { + ppu_bkg_render_tmp_val = ((ppu_bkgfetch_at_data << 2) & 0xC) | ((ppu_bkgfetch_lb_data >> 7) & 1) | ((ppu_bkgfetch_hb_data >> 6) & 2); + ppu_bkg_pixels[ppu_bkg_render_i + ppu_bkg_render_pos] = ppu_bkg_render_tmp_val; + ppu_bkgfetch_lb_data <<= 1; + ppu_bkgfetch_hb_data <<= 1; + } + } + + private static void PPUSPRFetch0() + { + ppu_sprfetch_slot = (ppu_clock_h - 1 >> 3) & 7; + ppu_sprfetch_slot = 7 - ppu_sprfetch_slot; + ppu_sprfetch_y_data = ppu_oam_bank_secondary[ppu_sprfetch_slot * 4]; + ppu_sprfetch_t_data = ppu_oam_bank_secondary[ppu_sprfetch_slot * 4 + 1]; + ppu_sprfetch_at_data = ppu_oam_bank_secondary[ppu_sprfetch_slot * 4 + 2]; + ppu_sprfetch_x_data = ppu_oam_bank_secondary[ppu_sprfetch_slot * 4 + 3]; + ppu_temp_comparator = (ppu_clock_v - ppu_sprfetch_y_data) ^ (((ppu_sprfetch_at_data & 0x80u) != 0) ? 15 : 0); + if (ppu_reg_2000_Sprite_size == 16) + { + ppu_sprfetch_lb_addr = (ushort)(((uint)(ppu_sprfetch_t_data << 12) & 0x1000u) | ((uint)(ppu_sprfetch_t_data << 4) & 0xFE0u) | ((uint)(ppu_temp_comparator << 1) & 0x10u) | ((uint)ppu_temp_comparator & 7u)); + } + else + { + ppu_sprfetch_lb_addr = (ushort)((uint)(ppu_reg_2000_sprite_pattern_table_address_for_8x8_sprites | (ppu_sprfetch_t_data << 4)) | ((uint)ppu_temp_comparator & 7u)); + } + mem_board.OnPPUAddressUpdate(ref ppu_sprfetch_lb_addr); + } + + private static void PPUSPRFetch1() + { + ppu_is_sprfetch = true; + mem_board.ReadCHR(ref ppu_sprfetch_lb_addr, out ppu_sprfetch_lb_data); + ppu_is_sprfetch = false; + if ((ppu_sprfetch_at_data & 0x40u) != 0) + { + ppu_sprfetch_lb_data = reverseLookup[ppu_sprfetch_lb_data]; + } + } + + private static void PPUSPRFetch2() + { + ppu_sprfetch_hb_addr = (ushort)(ppu_sprfetch_lb_addr | 8u); + mem_board.OnPPUAddressUpdate(ref ppu_sprfetch_hb_addr); + } + + private static void PPUSPRFetch3() + { + ppu_is_sprfetch = true; + mem_board.ReadCHR(ref ppu_sprfetch_hb_addr, out ppu_sprfetch_hb_data); + ppu_is_sprfetch = false; + if ((ppu_sprfetch_at_data & 0x40u) != 0) + { + ppu_sprfetch_hb_data = reverseLookup[ppu_sprfetch_hb_data]; + } + if (ppu_sprfetch_x_data == byte.MaxValue) + { + return; + } + for (ppu_bkg_render_i = 0; ppu_bkg_render_i < 8; ppu_bkg_render_i++) + { + if (ppu_sprfetch_x_data < byte.MaxValue) + { + ppu_bkg_render_tmp_val = ((ppu_sprfetch_at_data << 2) & 0xC) | ((ppu_sprfetch_lb_data >> 7) & 1) | ((ppu_sprfetch_hb_data >> 6) & 2); + if (((uint)ppu_bkg_render_tmp_val & 3u) != 0) + { + ppu_spr_pixels[ppu_sprfetch_x_data] = ppu_bkg_render_tmp_val; + if (ppu_sprfetch_slot == 0 && ppu_sprite0_should_hit) + { + ppu_spr_pixels[ppu_sprfetch_x_data] |= 16384; + } + if ((ppu_sprfetch_at_data & 0x20) == 0) + { + ppu_spr_pixels[ppu_sprfetch_x_data] |= 32768; + } + } + ppu_sprfetch_lb_data <<= 1; + ppu_sprfetch_hb_data <<= 1; + ppu_sprfetch_x_data++; + } + } + } + + private static void PPUOamReset() + { + ppu_oamev_n = 0; + ppu_oamev_m = 0; + ppu_oamev_slot = 0; + ppu_phase_index = 0; + ppu_sprite0_should_hit = false; + } + + private static void PPUOamClear() + { + for (int i = 0; i < ppu_spr_pixels.Length; i++) + { + ppu_spr_pixels[i] = 0; + } + } + + private static void PPUOamEvFetch() + { + ppu_fetch_data = ppu_oam_bank[ppu_oamev_n * 4 + ppu_oamev_m]; + } + + private static void PPUOamPhase0() + { + ppu_oamev_compare = ppu_clock_v >= ppu_fetch_data && ppu_clock_v < ppu_fetch_data + ppu_reg_2000_Sprite_size; + if (ppu_oamev_compare) + { + ppu_oam_bank_secondary[ppu_oamev_slot * 4] = ppu_fetch_data; + ppu_oamev_m = 1; + ppu_phase_index++; + if (ppu_oamev_n == 0) + { + ppu_sprite0_should_hit = true; + } + } + else + { + ppu_oamev_m = 0; + ppu_oamev_n++; + if (ppu_oamev_n == 64) + { + ppu_oamev_n = 0; + ppu_phase_index = 8; + } + } + } + + private static void PPUOamPhase1() + { + ppu_oam_bank_secondary[ppu_oamev_slot * 4 + ppu_oamev_m] = ppu_fetch_data; + ppu_oamev_m = 2; + ppu_phase_index++; + } + + private static void PPUOamPhase2() + { + ppu_oam_bank_secondary[ppu_oamev_slot * 4 + ppu_oamev_m] = ppu_fetch_data; + ppu_oamev_m = 3; + ppu_phase_index++; + } + + private static void PPUOamPhase3() + { + ppu_oam_bank_secondary[ppu_oamev_slot * 4 + ppu_oamev_m] = ppu_fetch_data; + ppu_oamev_m = 0; + ppu_oamev_n++; + ppu_oamev_slot++; + if (ppu_oamev_n == 64) + { + ppu_oamev_n = 0; + ppu_phase_index = 8; + } + else if (ppu_oamev_slot < 8) + { + ppu_phase_index = 0; + } + else if (ppu_oamev_slot == 8) + { + ppu_phase_index = 4; + } + } + + private static void PPUOamPhase4() + { + ppu_oamev_compare = ppu_clock_v >= ppu_fetch_data && ppu_clock_v < ppu_fetch_data + ppu_reg_2000_Sprite_size; + if (ppu_oamev_compare) + { + ppu_oamev_m = 1; + ppu_phase_index++; + ppu_reg_2002_SpriteOverflow = true; + return; + } + ppu_oamev_m++; + if (ppu_oamev_m == 4) + { + ppu_oamev_m = 0; + } + ppu_oamev_n++; + if (ppu_oamev_n == 64) + { + ppu_oamev_n = 0; + ppu_phase_index = 8; + } + else + { + ppu_phase_index = 4; + } + } + + private static void PPUOamPhase5() + { + ppu_oamev_m = 2; + ppu_phase_index++; + } + + private static void PPUOamPhase6() + { + ppu_oamev_m = 3; + ppu_phase_index++; + } + + private static void PPUOamPhase7() + { + ppu_oamev_m = 0; + ppu_oamev_n++; + if (ppu_oamev_n == 64) + { + ppu_oamev_n = 0; + } + ppu_phase_index = 8; + } + + private static void PPUOamPhase8() + { + ppu_oamev_n++; + if (ppu_oamev_n >= 64) + { + ppu_oamev_n = 0; + } + } + + private static void RenderPixel() + { + if (ppu_clock_v == ppu_clock_vblank_end) + { + return; + } + ppu_render_x = ppu_clock_h - 1; + ppu_render_y = ppu_clock_v * 256; + if (ppu_render_x < 8) + { + if (ppu_reg_2001_show_background_in_leftmost_8_pixels_of_screen) + { + ppu_bkg_current_pixel = 0x3F00 | ppu_bkg_pixels[ppu_render_x + ppu_vram_finex]; + } + else + { + ppu_bkg_current_pixel = 16128; + } + if (ppu_reg_2001_show_sprites_in_leftmost_8_pixels_of_screen) + { + ppu_spr_current_pixel = 0x3F10 | ppu_spr_pixels[ppu_render_x]; + } + else + { + ppu_spr_current_pixel = 16144; + } + } + else + { + if (!ppu_reg_2001_show_background) + { + ppu_bkg_current_pixel = 16128; + } + else + { + ppu_bkg_current_pixel = 0x3F00 | ppu_bkg_pixels[ppu_render_x + ppu_vram_finex]; + } + if (!ppu_reg_2001_show_sprites || ppu_clock_v == 0) + { + ppu_spr_current_pixel = 16144; + } + else + { + ppu_spr_current_pixel = 0x3F10 | ppu_spr_pixels[ppu_render_x]; + } + } + ppu_current_pixel = 0; + if (((uint)ppu_spr_current_pixel & 0x8000u) != 0) + { + ppu_current_pixel = ppu_spr_current_pixel; + } + else + { + ppu_current_pixel = ppu_bkg_current_pixel; + } + if ((ppu_bkg_current_pixel & 3) == 0) + { + ppu_current_pixel = ppu_spr_current_pixel; + } + else if ((ppu_spr_current_pixel & 3) == 0) + { + ppu_current_pixel = ppu_bkg_current_pixel; + } + else if (((uint)ppu_spr_pixels[ppu_render_x] & 0x4000u) != 0) + { + ppu_reg_2002_Sprite0Hit = true; + } + ppu_screen_pixels[ppu_render_x + ppu_render_y] = MakeRGBColor(ppu_current_pixel); + } + + private static int MakeRGBColor(int pixel) + { + pixel = ((pixel < 16128) ? 15 : ((((uint)ppu_current_pixel & 3u) != 0) ? ppu_palette_bank[pixel & 0x1F] : ppu_palette_bank[pixel & 0xC])); + ppu_palette_temp__val = pixel & 0xF; + ppu_color_temp__level = (pixel & 0x30) << 2; + if (ppu_reg_2001_grayscale) + { + ppu_palette_temp__val = 0; + } + ppu_color_temp__blue = (ppu_color_temp_blue__sequence[ppu_palette_temp__val] << 3) | ppu_color_temp__level; + ppu_color_temp__red = (ppu_color_temp_red___sequence[ppu_palette_temp__val] << 3) | ppu_color_temp__level; + ppu_color_temp__green = (ppu_color_temp_green_sequence[ppu_palette_temp__val] << 3) | ppu_color_temp__level; + if (SystemIndex == 0) + { + ppu_color_temp__e_red = (ppu_reg_2001_emphasis & 1) != 0; + } + else + { + ppu_color_temp__e_green = (ppu_reg_2001_emphasis & 1) != 0; + } + if (SystemIndex == 0) + { + ppu_color_temp__e_green = (ppu_reg_2001_emphasis & 2) != 0; + } + else + { + ppu_color_temp__e_red = (ppu_reg_2001_emphasis & 2) != 0; + } + ppu_color_temp__e_blue = (ppu_reg_2001_emphasis & 4) != 0; + if (ppu_color_temp__e_blue) + { + ppu_color_temp__blue |= 7; + } + if (ppu_color_temp__e_red) + { + ppu_color_temp__red |= 7; + } + if (ppu_color_temp__e_green) + { + ppu_color_temp__green |= 7; + } + return (ppu_color_temp__red << 16) | (ppu_color_temp__green << 8) | ppu_color_temp__blue; + } + + private static void PPUIORead(ref ushort addr, out byte value) + { + ppu_reg_io_addr = (byte)(addr & 7u); + ppu_reg_access_happened = true; + ppu_reg_access_w = false; + ppu_reg_read_func[ppu_reg_io_addr](); + value = ppu_reg_io_db; + } + + private static void PPUIOWrite(ref ushort addr, ref byte value) + { + ppu_reg_io_addr = (byte)(addr & 7u); + ppu_reg_io_db = value; + ppu_reg_access_w = true; + ppu_reg_access_happened = true; + } + + private static void PPUOnRegister2000() + { + if (ppu_reg_access_w) + { + ppu_vram_addr_temp = (ushort)((ppu_vram_addr_temp & 0x73FFu) | (uint)((ppu_reg_io_db & 3) << 10)); + if ((ppu_reg_io_db & 4u) != 0) + { + ppu_reg_2000_vram_address_increament = 32; + } + else + { + ppu_reg_2000_vram_address_increament = 1; + } + if ((ppu_reg_io_db & 8u) != 0) + { + ppu_reg_2000_sprite_pattern_table_address_for_8x8_sprites = 4096; + } + else + { + ppu_reg_2000_sprite_pattern_table_address_for_8x8_sprites = 0; + } + if ((ppu_reg_io_db & 0x10u) != 0) + { + ppu_reg_2000_background_pattern_table_address = 4096; + } + else + { + ppu_reg_2000_background_pattern_table_address = 0; + } + if ((ppu_reg_io_db & 0x20u) != 0) + { + ppu_reg_2000_Sprite_size = 16; + } + else + { + ppu_reg_2000_Sprite_size = 8; + } + if (!ppu_reg_2000_VBI && (ppu_reg_io_db & 0x80u) != 0 && ppu_reg_2002_VblankStartedFlag) + { + PPU_NMI_Current = true; + } + ppu_reg_2000_VBI = (ppu_reg_io_db & 0x80) != 0; + if (!ppu_reg_2000_VBI && ppu_is_nmi_time) + { + PPU_NMI_Current = false; + } + } + } + + private static void PPUOnRegister2001() + { + if (ppu_reg_access_w) + { + ppu_reg_2001_show_background_in_leftmost_8_pixels_of_screen = (ppu_reg_io_db & 2) != 0; + ppu_reg_2001_show_sprites_in_leftmost_8_pixels_of_screen = (ppu_reg_io_db & 4) != 0; + ppu_reg_2001_show_background = (ppu_reg_io_db & 8) != 0; + ppu_reg_2001_show_sprites = (ppu_reg_io_db & 0x10) != 0; + ppu_reg_2001_grayscale = (ppu_reg_io_db & 1) != 0; + ppu_reg_2001_emphasis = (ppu_reg_io_db & 0xE0) >> 5; + } + } + + private static void PPUOnRegister2002() + { + if (!ppu_reg_access_w) + { + ppu_vram_flip_flop = false; + ppu_reg_2002_VblankStartedFlag = false; + if (ppu_clock_v == ppu_clock_vblank_start) + { + PPU_NMI_Current = ppu_reg_2002_VblankStartedFlag & ppu_reg_2000_VBI; + } + } + } + + private static void PPUOnRegister2003() + { + if (ppu_reg_access_w) + { + ppu_reg_2003_oam_addr = ppu_reg_io_db; + } + } + + private static void PPUOnRegister2004() + { + if (ppu_reg_access_w) + { + if (ppu_clock_v < 240 && IsRenderingOn()) + { + ppu_reg_io_db = byte.MaxValue; + } + if ((ppu_reg_2003_oam_addr & 3) == 2) + { + ppu_reg_io_db &= 227; + } + ppu_oam_bank[ppu_reg_2003_oam_addr] = ppu_reg_io_db; + ppu_reg_2003_oam_addr = (byte)((uint)(ppu_reg_2003_oam_addr + 1) & 0xFFu); + } + } + + private static void PPUOnRegister2005() + { + if (ppu_reg_access_w) + { + if (!ppu_vram_flip_flop) + { + ppu_vram_addr_temp = (ushort)((ppu_vram_addr_temp & 0x7FE0u) | (uint)((ppu_reg_io_db & 0xF8) >> 3)); + ppu_vram_finex = (byte)(ppu_reg_io_db & 7u); + } + else + { + ppu_vram_addr_temp = (ushort)((ppu_vram_addr_temp & 0xC1Fu) | (uint)((ppu_reg_io_db & 7) << 12) | (uint)((ppu_reg_io_db & 0xF8) << 2)); + } + ppu_vram_flip_flop = !ppu_vram_flip_flop; + } + } + + private static void PPUOnRegister2006() + { + if (ppu_reg_access_w) + { + if (!ppu_vram_flip_flop) + { + ppu_vram_addr_temp = (ushort)((ppu_vram_addr_temp & 0xFFu) | (uint)((ppu_reg_io_db & 0x3F) << 8)); + } + else + { + ppu_vram_addr_temp = (ushort)((ppu_vram_addr_temp & 0x7F00u) | ppu_reg_io_db); + ppu_vram_addr = ppu_vram_addr_temp; + mem_board.OnPPUAddressUpdate(ref ppu_vram_addr); + } + ppu_vram_flip_flop = !ppu_vram_flip_flop; + } + } + + private static void PPUOnRegister2007() + { + if (ppu_reg_access_w) + { + ppu_vram_addr_access_temp = (ushort)(ppu_vram_addr & 0x3FFFu); + if (ppu_vram_addr_access_temp < 8192) + { + mem_board.WriteCHR(ref ppu_vram_addr_access_temp, ref ppu_reg_io_db); + } + else if (ppu_vram_addr_access_temp < 16128) + { + mem_board.WriteNMT(ref ppu_vram_addr_access_temp, ref ppu_reg_io_db); + } + else if ((ppu_vram_addr_access_temp & 3u) != 0) + { + ppu_palette_bank[ppu_vram_addr_access_temp & 0x1F] = ppu_reg_io_db; + } + else + { + ppu_palette_bank[ppu_vram_addr_access_temp & 0xC] = ppu_reg_io_db; + } + } + else + { + if ((ppu_vram_addr & 0x3F00) == 16128) + { + ppu_vram_addr_access_temp = (ushort)(ppu_vram_addr & 0x2FFFu); + } + else + { + ppu_vram_addr_access_temp = (ushort)(ppu_vram_addr & 0x3FFFu); + } + if (ppu_vram_addr_access_temp < 8192) + { + mem_board.ReadCHR(ref ppu_vram_addr_access_temp, out ppu_vram_data); + } + else if (ppu_vram_addr_access_temp < 16128) + { + mem_board.ReadNMT(ref ppu_vram_addr_access_temp, out ppu_vram_data); + } + } + ppu_vram_addr = (ushort)((uint)(ppu_vram_addr + ppu_reg_2000_vram_address_increament) & 0x7FFFu); + mem_board.OnPPUAddressUpdate(ref ppu_vram_addr); + } + + private static void PPURead2000() + { + } + + private static void PPURead2001() + { + } + + private static void PPURead2002() + { + ppu_reg_io_db = (byte)((ppu_reg_io_db & 0xDFu) | (ppu_reg_2002_SpriteOverflow ? 32u : 0u)); + ppu_reg_io_db = (byte)((ppu_reg_io_db & 0xBFu) | (ppu_reg_2002_Sprite0Hit ? 64u : 0u)); + ppu_reg_io_db = (byte)((ppu_reg_io_db & 0x7Fu) | (ppu_reg_2002_VblankStartedFlag ? 128u : 0u)); + } + + private static void PPURead2003() + { + } + + private static void PPURead2004() + { + ppu_reg_io_db = ppu_oam_bank[ppu_reg_2003_oam_addr]; + if (ppu_clock_v < 240 && IsRenderingOn()) + { + if (ppu_clock_h < 64) + { + ppu_reg_io_db = byte.MaxValue; + } + else if (ppu_clock_h < 192) + { + ppu_reg_io_db = ppu_oam_bank[(ppu_clock_h - 64 << 1) & 0xFC]; + } + else if (ppu_clock_h < 256) + { + ppu_reg_io_db = (((ppu_clock_h & 1) == 1) ? ppu_oam_bank[252] : ppu_oam_bank[(ppu_clock_h - 192 << 1) & 0xFC]); + } + else if (ppu_clock_h < 320) + { + ppu_reg_io_db = byte.MaxValue; + } + else + { + ppu_reg_io_db = ppu_oam_bank[0]; + } + } + } + + private static void PPURead2005() + { + } + + private static void PPURead2006() + { + } + + private static void PPURead2007() + { + ppu_vram_addr_access_temp = (ushort)(ppu_vram_addr & 0x3FFFu); + if (ppu_vram_addr_access_temp < 16128) + { + ppu_reg_io_db = ppu_vram_data; + } + else if ((ppu_vram_addr_access_temp & 3u) != 0) + { + ppu_reg_io_db = ppu_palette_bank[ppu_vram_addr_access_temp & 0x1F]; + } + else + { + ppu_reg_io_db = ppu_palette_bank[ppu_vram_addr_access_temp & 0xC]; + } + } + + internal static bool IsRenderingOn() + { + return ppu_reg_2001_show_background || ppu_reg_2001_show_sprites; + } + + internal static bool IsInRender() + { + return ppu_clock_v < 240 || ppu_clock_v == ppu_clock_vblank_end; + } + + private static void PPUWriteState(ref BinaryWriter bin) + { + bin.Write(ppu_clock_h); + bin.Write(ppu_clock_v); + bin.Write(ppu_clock_vblank_start); + bin.Write(ppu_clock_vblank_end); + bin.Write(ppu_use_odd_cycle); + bin.Write(ppu_use_odd_swap); + bin.Write(ppu_is_nmi_time); + bin.Write(ppu_frame_finished); + bin.Write(ppu_oam_bank); + bin.Write(ppu_oam_bank_secondary); + bin.Write(ppu_palette_bank); + bin.Write(ppu_reg_io_db); + bin.Write(ppu_reg_io_addr); + bin.Write(ppu_reg_access_happened); + bin.Write(ppu_reg_access_w); + bin.Write(ppu_reg_2000_vram_address_increament); + bin.Write(ppu_reg_2000_sprite_pattern_table_address_for_8x8_sprites); + bin.Write(ppu_reg_2000_background_pattern_table_address); + bin.Write(ppu_reg_2000_Sprite_size); + bin.Write(ppu_reg_2000_VBI); + bin.Write(ppu_reg_2001_show_background_in_leftmost_8_pixels_of_screen); + bin.Write(ppu_reg_2001_show_sprites_in_leftmost_8_pixels_of_screen); + bin.Write(ppu_reg_2001_show_background); + bin.Write(ppu_reg_2001_show_sprites); + bin.Write(ppu_reg_2001_grayscale ? 1 : 0); + bin.Write(ppu_reg_2001_emphasis); + bin.Write(ppu_reg_2002_SpriteOverflow); + bin.Write(ppu_reg_2002_Sprite0Hit); + bin.Write(ppu_reg_2002_VblankStartedFlag); + bin.Write(ppu_reg_2003_oam_addr); + bin.Write(ppu_vram_addr); + bin.Write(ppu_vram_data); + bin.Write(ppu_vram_addr_temp); + bin.Write(ppu_vram_addr_access_temp); + bin.Write(ppu_vram_flip_flop); + bin.Write(ppu_vram_finex); + bin.Write(ppu_bkgfetch_nt_addr); + bin.Write(ppu_bkgfetch_nt_data); + bin.Write(ppu_bkgfetch_at_addr); + bin.Write(ppu_bkgfetch_at_data); + bin.Write(ppu_bkgfetch_lb_addr); + bin.Write(ppu_bkgfetch_lb_data); + bin.Write(ppu_bkgfetch_hb_addr); + bin.Write(ppu_bkgfetch_hb_data); + bin.Write(ppu_sprfetch_slot); + bin.Write(ppu_sprfetch_y_data); + bin.Write(ppu_sprfetch_t_data); + bin.Write(ppu_sprfetch_at_data); + bin.Write(ppu_sprfetch_x_data); + bin.Write(ppu_sprfetch_lb_addr); + bin.Write(ppu_sprfetch_lb_data); + bin.Write(ppu_sprfetch_hb_addr); + bin.Write(ppu_sprfetch_hb_data); + bin.Write(ppu_bkg_render_i); + bin.Write(ppu_bkg_render_pos); + bin.Write(ppu_bkg_render_tmp_val); + bin.Write(ppu_bkg_current_pixel); + bin.Write(ppu_spr_current_pixel); + bin.Write(ppu_current_pixel); + bin.Write(ppu_render_x); + bin.Write(0); + bin.Write(ppu_oamev_n); + bin.Write(ppu_oamev_m); + bin.Write(ppu_oamev_compare); + bin.Write(ppu_oamev_slot); + bin.Write(ppu_fetch_data); + bin.Write(ppu_phase_index); + bin.Write(ppu_sprite0_should_hit); + } + + private static void PPUReadState(ref BinaryReader bin) + { + ppu_clock_h = bin.ReadInt32(); + ppu_clock_v = bin.ReadUInt16(); + ppu_clock_vblank_start = bin.ReadUInt16(); + ppu_clock_vblank_end = bin.ReadUInt16(); + ppu_use_odd_cycle = bin.ReadBoolean(); + ppu_use_odd_swap = bin.ReadBoolean(); + ppu_is_nmi_time = bin.ReadBoolean(); + ppu_frame_finished = bin.ReadBoolean(); + bin.Read(ppu_oam_bank, 0, ppu_oam_bank.Length); + bin.Read(ppu_oam_bank_secondary, 0, ppu_oam_bank_secondary.Length); + bin.Read(ppu_palette_bank, 0, ppu_palette_bank.Length); + ppu_reg_io_db = bin.ReadByte(); + ppu_reg_io_addr = bin.ReadByte(); + ppu_reg_access_happened = bin.ReadBoolean(); + ppu_reg_access_w = bin.ReadBoolean(); + ppu_reg_2000_vram_address_increament = bin.ReadByte(); + ppu_reg_2000_sprite_pattern_table_address_for_8x8_sprites = bin.ReadUInt16(); + ppu_reg_2000_background_pattern_table_address = bin.ReadUInt16(); + ppu_reg_2000_Sprite_size = bin.ReadByte(); + ppu_reg_2000_VBI = bin.ReadBoolean(); + ppu_reg_2001_show_background_in_leftmost_8_pixels_of_screen = bin.ReadBoolean(); + ppu_reg_2001_show_sprites_in_leftmost_8_pixels_of_screen = bin.ReadBoolean(); + ppu_reg_2001_show_background = bin.ReadBoolean(); + ppu_reg_2001_show_sprites = bin.ReadBoolean(); + ppu_reg_2001_grayscale = bin.ReadInt32() == 1; + ppu_reg_2001_emphasis = bin.ReadInt32(); + ppu_reg_2002_SpriteOverflow = bin.ReadBoolean(); + ppu_reg_2002_Sprite0Hit = bin.ReadBoolean(); + ppu_reg_2002_VblankStartedFlag = bin.ReadBoolean(); + ppu_reg_2003_oam_addr = bin.ReadByte(); + ppu_vram_addr = bin.ReadUInt16(); + ppu_vram_data = bin.ReadByte(); + ppu_vram_addr_temp = bin.ReadUInt16(); + ppu_vram_addr_access_temp = bin.ReadUInt16(); + ppu_vram_flip_flop = bin.ReadBoolean(); + ppu_vram_finex = bin.ReadByte(); + ppu_bkgfetch_nt_addr = bin.ReadUInt16(); + ppu_bkgfetch_nt_data = bin.ReadByte(); + ppu_bkgfetch_at_addr = bin.ReadUInt16(); + ppu_bkgfetch_at_data = bin.ReadByte(); + ppu_bkgfetch_lb_addr = bin.ReadUInt16(); + ppu_bkgfetch_lb_data = bin.ReadByte(); + ppu_bkgfetch_hb_addr = bin.ReadUInt16(); + ppu_bkgfetch_hb_data = bin.ReadByte(); + ppu_sprfetch_slot = bin.ReadInt32(); + ppu_sprfetch_y_data = bin.ReadByte(); + ppu_sprfetch_t_data = bin.ReadByte(); + ppu_sprfetch_at_data = bin.ReadByte(); + ppu_sprfetch_x_data = bin.ReadByte(); + ppu_sprfetch_lb_addr = bin.ReadUInt16(); + ppu_sprfetch_lb_data = bin.ReadByte(); + ppu_sprfetch_hb_addr = bin.ReadUInt16(); + ppu_sprfetch_hb_data = bin.ReadByte(); + ppu_bkg_render_i = bin.ReadInt32(); + ppu_bkg_render_pos = bin.ReadInt32(); + ppu_bkg_render_tmp_val = bin.ReadInt32(); + ppu_bkg_current_pixel = bin.ReadInt32(); + ppu_spr_current_pixel = bin.ReadInt32(); + ppu_current_pixel = bin.ReadInt32(); + ppu_render_x = bin.ReadInt32(); + bin.ReadInt32(); + ppu_oamev_n = bin.ReadByte(); + ppu_oamev_m = bin.ReadByte(); + ppu_oamev_compare = bin.ReadBoolean(); + ppu_oamev_slot = bin.ReadByte(); + ppu_fetch_data = bin.ReadByte(); + ppu_phase_index = bin.ReadByte(); + ppu_sprite0_should_hit = bin.ReadBoolean(); + } + + internal static void CheckGame(string fileName, out bool valid) + { + string text = Path.GetExtension(fileName).ToLower(); + string text2 = text; + if (text2 == ".nes") + { + Tracer.WriteLine("Checking INES header ..."); + INes nes = new INes(); + nes.Load(fileName, loadDumps: false); + valid = nes.IsValid; + Tracer.WriteLine("INES header is valid."); + } + else + { + Tracer.WriteWarning("File format is not supported. Format: " + Path.GetExtension(fileName)); + valid = false; + } + } + + internal static void Initialize() + { + Tracer.WriteLine("Loading database file ..."); + string text = Path.Combine(MyNesMain.AppPath, "database.xml"); + if (File.Exists(text)) + { + bool success = false; + NesCartDatabase.LoadDatabase(text, out success); + if (success) + { + Tracer.WriteInformation("Nes Cart database file loaded successfully."); + } + else + { + Tracer.WriteError("Error loading Nes Cart database file."); + } + } + else + { + Tracer.WriteWarning("Nes Cart database file cannot be located at " + text); + } + EmuClockComponents = EmuClockComponentsNTSC; + FrameLimiterEnabled = true; + CPUInitialize(); + PPUInitialize(); + APUInitialize(); + PORTSInitialize(); + } + + internal static void SetupRenderingMethods(RenderVideoFrame renderVideo, RenderAudioSamples renderAudio, TogglePause renderTogglePause, GetIsPlaying renderGetIsPlaying) + { + render_initialized = false; + render_video = renderVideo; + render_audio = renderAudio; + render_audio_toggle_pause = renderTogglePause; + render_audio_get_is_playing = renderGetIsPlaying; + render_initialized = render_video != null && render_audio != null && render_audio_toggle_pause != null && render_audio_get_is_playing != null; + if (render_initialized) + { + Tracer.WriteInformation("Renderer methods initialized successfully."); + return; + } + Tracer.WriteError("ERROR RENDERER INITIALIZING !!"); + Tracer.WriteError("Faild to initialize the renderers methods. Please use the method 'SetupRenderingMethods' to initialize the renderers methods before you can run the emulation."); + } + + public static void LoadGame(string fileName, out bool success) + { + if (!render_initialized) + { + Tracer.WriteError("NO RENDERER INITIALIZED !! EMU CANNOT BE INTIALIZED WITHOUT A RENDERER !!"); + Tracer.WriteError("Please use the method 'SetupRenderingMethods' to initialize the renderers methods before you can run the emulation."); + success = false; + return; + } + string text = Path.GetExtension(fileName).ToLower(); + string text2 = text; + if (text2 == ".nes") + { + Tracer.WriteLine("Checking INES header ..."); + INes nes = new INes(); + nes.Load(fileName, loadDumps: true); + if (nes.IsValid) + { + emu_request_mode = RequestMode.None; + CurrentFilePath = fileName; + if (ON) + { + ShutDown(); + } + Tracer.WriteLine("INES header is valid, loading game ..."); + MEMInitialize(nes); + ApplyRegionSetting(); + ApplyAudioSettings(); + ApplyFrameSkipSettings(); + PORTSInitialize(); + hardReset(); + Tracer.WriteLine("EMU is ready."); + success = true; + emu_frame_clocking_mode = !MyNesMain.RendererSettings.UseEmuThread; + ON = true; + PAUSED = false; + isPaused = false; + FrameLimiterEnabled = true; + if (MyNesMain.RendererSettings.UseEmuThread) + { + Tracer.WriteLine("Running in a thread ... using custom frame limiter."); + mainThread = new Thread(EmuClock); + mainThread.Start(); + } + MyNesMain.VideoProvider.SignalToggle(started: true); + MyNesMain.AudioProvider.SignalToggle(started: true); + } + else + { + success = false; + } + } + else + { + success = false; + } + } + + public static void HardReset() + { + PAUSED = true; + emu_request_mode = RequestMode.HardReset; + } + + private static void hardReset() + { + if (MyNesMain.WaveRecorder.IsRecording) + { + MyNesMain.WaveRecorder.Stop(); + } + render_audio_toggle_pause(paused: true); + switch (Region) + { + case EmuRegion.NTSC: + emu_time_target_fps = 60.0988; + break; + case EmuRegion.PALB: + case EmuRegion.DENDY: + emu_time_target_fps = 50.0; + break; + } + fps_time_period = 1.0 / emu_time_target_fps; + MEMHardReset(); + CPUHardReset(); + PPUHardReset(); + APUHardReset(); + DMAHardReset(); + render_audio_toggle_pause(paused: false); + MyNesMain.VideoProvider.WriteWarningNotification(MNInterfaceLanguage.Message_HardReset, instant: false); + } + + public static void SoftReset() + { + PAUSED = true; + emu_request_mode = RequestMode.SoftReset; + } + + private static void softReset() + { + CPUSoftReset(); + APUSoftReset(); + MyNesMain.VideoProvider.WriteWarningNotification(MNInterfaceLanguage.Message_SoftReset, instant: false); + } + + public static void SaveState() + { + PAUSED = true; + emu_request_mode = RequestMode.SaveState; + } + + public static void LoadState() + { + PAUSED = true; + emu_request_mode = RequestMode.LoadState; + } + + internal static void TakeSnapshot() + { + PAUSED = true; + emu_request_mode = RequestMode.TakeSnapshot; + } + + public static void ShutDown() + { + MyNesMain.VideoProvider.SignalToggle(started: false); + MyNesMain.AudioProvider.SignalToggle(started: false); + if (MyNesMain.WaveRecorder.IsRecording) + { + MyNesMain.WaveRecorder.Stop(); + } + render_audio_get_is_playing(out render_audio_is_playing); + if (render_audio_is_playing) + { + render_audio_toggle_pause(paused: true); + } + Tracer.WriteLine("Shutting down the emulation core..."); + ON = false; + if (mainThread != null) + { + Tracer.WriteLine("Aborting thread .."); + while (mainThread.IsAlive) + { + } + mainThread.Abort(); + mainThread = null; + } + SaveSRAM(); + Tracer.WriteInformation("Emulation core shutdown successfully."); + NesEmu.EmuShutdown?.Invoke(null, new EventArgs()); + } + + public static void EMUClockFrame() + { + if (PAUSED) + { + render_audio_get_is_playing(out render_audio_is_playing); + if (render_audio_is_playing) + { + render_audio_toggle_pause(paused: true); + } + shortucts.Update(); + switch (emu_request_mode) + { + case RequestMode.HardReset: + hardReset(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.SoftReset: + softReset(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.SaveState: + StateHandler.SaveState(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.LoadState: + StateHandler.LoadState(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.TakeSnapshot: + MyNesMain.VideoProvider.TakeSnapshot(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + } + isPaused = true; + return; + } + emu_frame_done = false; + while (!ppu_frame_finished && ON) + { + CPUClock(); + } + if (!FrameSkipEnabled) + { + render_video(ref ppu_screen_pixels); + } + else + { + FrameSkipCounter++; + if (FrameSkipCounter >= FrameSkipInterval) + { + render_video(ref ppu_screen_pixels); + FrameSkipCounter = 0; + } + } + isPaused = false; + ppu_frame_finished = false; + emu_frame_done = true; + joypad1.Update(); + joypad2.Update(); + if (IsFourPlayers) + { + joypad3.Update(); + joypad4.Update(); + } + shortucts.Update(); + if (SoundEnabled) + { + render_audio_get_is_playing(out render_audio_is_playing); + if (!render_audio_is_playing) + { + render_audio_toggle_pause(paused: false); + } + render_audio(ref audio_samples, ref audio_samples_added); + audio_w_pos = 0; + audio_samples_added = 0; + audio_timer = 0.0; + } + fps_time_token = GetTime() - fps_time_start; + fps_time_last = GetTime(); + fps_time_frame_time = fps_time_last - fps_time_start; + fps_time_start = GetTime(); + } + + public static void EMUClockSamples(int samples_required, out int audio_added_samples) + { + int num = audio_samples_added; + while (samples_required > 0 && audio_w_pos < audio_samples_count) + { + CPUClock(); + if (audio_samples_added - num >= 1) + { + samples_required--; + } + num = audio_samples_added; + } + audio_added_samples = audio_samples_added; + } + + private static void EmuClock() + { + while (ON) + { + if (!PAUSED) + { + CPUClock(); + if (ppu_frame_finished) + { + FrameFinished(); + } + continue; + } + render_audio_get_is_playing(out render_audio_is_playing); + if (render_audio_is_playing) + { + render_audio_toggle_pause(paused: true); + } + Thread.Sleep(100); + shortucts.Update(); + switch (emu_request_mode) + { + case RequestMode.HardReset: + hardReset(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.SoftReset: + softReset(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.SaveState: + StateHandler.SaveState(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.LoadState: + StateHandler.LoadState(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + case RequestMode.TakeSnapshot: + MyNesMain.VideoProvider.TakeSnapshot(); + PAUSED = false; + emu_request_mode = RequestMode.None; + break; + } + isPaused = true; + } + } + + internal static void EmuClockComponentsNTSC() + { + PPUClock(); + PollInterruptStatus(); + PPUClock(); + PPUClock(); + APUClock(); + DMAClock(); + mem_board.OnCPUClock(); + } + + internal static void EmuClockComponentsPALB() + { + PPUClock(); + PollInterruptStatus(); + PPUClock(); + PPUClock(); + pal_add_cycle++; + if (pal_add_cycle == 5) + { + pal_add_cycle = 0; + PPUClock(); + } + APUClock(); + DMAClock(); + mem_board.OnCPUClock(); + } + + internal static void ApplyFrameSkipSettings() + { + FrameSkipEnabled = MyNesMain.RendererSettings.FrameSkipEnabled; + FrameSkipInterval = MyNesMain.RendererSettings.FrameSkipInterval; + } + + private static void FrameFinished() + { + if (!FrameSkipEnabled) + { + render_video(ref ppu_screen_pixels); + } + else + { + FrameSkipCounter++; + if (FrameSkipCounter >= FrameSkipInterval) + { + render_video(ref ppu_screen_pixels); + FrameSkipCounter = 0; + } + } + isPaused = false; + ppu_frame_finished = false; + emu_frame_done = true; + joypad1.Update(); + joypad2.Update(); + if (IsFourPlayers) + { + joypad3.Update(); + joypad4.Update(); + } + shortucts.Update(); + if (SoundEnabled) + { + render_audio_get_is_playing(out render_audio_is_playing); + if (!render_audio_is_playing) + { + render_audio_toggle_pause(paused: false); + } + render_audio(ref audio_samples, ref audio_samples_added); + audio_w_pos = 0; + audio_samples_added = 0; + audio_timer = 0.0; + } + fps_time_token = GetTime() - fps_time_start; + if (FrameLimiterEnabled) + { + if (fps_time_token > 0.0) + { + fps_time_dead = fps_time_period - fps_time_token; + if (fps_time_dead > 0.0) + { + Thread.Sleep((int)Math.Floor(fps_time_dead * 1000.0)); + fps_time_dead = GetTime() - fps_time_start; + while (fps_time_period - fps_time_dead > 0.0) + { + fps_time_dead = GetTime() - fps_time_start; + } + } + } + fps_time_last = GetTime(); + fps_time_frame_time = fps_time_last - fps_time_start; + } + fps_time_start = GetTime(); + } + + private static double GetTime() + { + return (double)Stopwatch.GetTimestamp() / (double)Stopwatch.Frequency; + } + + public static void GetSpeedValues(out double frame_time, out double immediate_frame_time) + { + frame_time = fps_time_token; + immediate_frame_time = fps_time_frame_time; + } + + public static void SetFramePeriod(ref double period) + { + fps_time_period = period; + } + + public static void GetTargetFPS(out double fps) + { + fps = emu_time_target_fps; + } + + public static void ApplyRegionSetting() + { + switch ((RegionSetting)MyNesMain.EmuSettings.RegionSetting) + { + case RegionSetting.AUTO: + { + Tracer.WriteLine("REGION = AUTO"); + Region = EmuRegion.NTSC; + EmuClockComponents = EmuClockComponentsNTSC; + bool flag = false; + if (mem_board != null && mem_board.IsGameFoundOnDB) + { + Tracer.WriteLine("REGION SELECTION IS FROM DATABASE !!"); + if (mem_board.GameCartInfo.System.ToUpper().Contains("PAL")) + { + Region = EmuRegion.PALB; + EmuClockComponents = EmuClockComponentsPALB; + flag = true; + } + else if (mem_board.GameCartInfo.System.ToUpper().Contains("DENDY")) + { + Region = EmuRegion.DENDY; + EmuClockComponents = EmuClockComponentsNTSC; + flag = true; + } + else + { + Region = EmuRegion.NTSC; + EmuClockComponents = EmuClockComponentsNTSC; + flag = true; + } + } + if (!flag && CurrentFilePath.Contains("(E)")) + { + Region = EmuRegion.PALB; + EmuClockComponents = EmuClockComponentsPALB; + } + Tracer.WriteLine("REGION SELECTED: " + Region); + break; + } + case RegionSetting.ForceNTSC: + Tracer.WriteLine("REGION: FORCE NTSC"); + Region = EmuRegion.NTSC; + EmuClockComponents = EmuClockComponentsNTSC; + break; + case RegionSetting.ForcePALB: + Tracer.WriteLine("REGION: FORCE PALB"); + Region = EmuRegion.PALB; + EmuClockComponents = EmuClockComponentsPALB; + break; + case RegionSetting.ForceDENDY: + Tracer.WriteLine("REGION: FORCE DENDY"); + Region = EmuRegion.DENDY; + EmuClockComponents = EmuClockComponentsNTSC; + break; + } + SystemIndex = (int)Region; + } + + internal static void WriteStateData(ref BinaryWriter bin) + { + APUWriteState(ref bin); + CPUWriteState(ref bin); + DMAWriteState(ref bin); + InterruptsWriteState(ref bin); + MEMWriteState(ref bin); + PORTWriteState(ref bin); + PPUWriteState(ref bin); + } + + internal static void ReadStateData(ref BinaryReader bin) + { + APUReadState(ref bin); + CPUReadState(ref bin); + DMAReadState(ref bin); + InterruptsReadState(ref bin); + MEMReadState(ref bin); + PORTReadState(ref bin); + PPUReadState(ref bin); + } + + private static void SQ2HardReset() + { + sq2_duty_cycle = 0; + sq2_length_halt = false; + sq2_constant_volume_envelope = false; + sq2_volume_devider_period = 0; + sq2_sweep_enable = false; + sq2_sweep_devider_period = 0; + sq2_sweep_negate = false; + sq2_sweep_shift_count = 0; + sq2_timer = 0; + sq2_period_devider = 0; + sq2_seqencer = 0; + sq2_length_enabled = false; + sq2_length_counter = 0; + sq2_envelope_start_flag = false; + sq2_envelope_devider = 0; + sq2_envelope_decay_level_counter = 0; + sq2_envelope = 0; + sq2_sweep_counter = 0; + sq2_sweep_reload = false; + sq2_sweep_change = 0; + sq2_valid_freq = false; + sq2_output = 0; + sq2_ignore_reload = false; + } + + private static void SQ2SoftReset() + { + SQ2HardReset(); + } + + private static void SQ2Clock() + { + sq2_period_devider--; + if (sq2_period_devider > 0) + { + return; + } + sq2_period_devider = sq2_timer + 1; + sq2_seqencer = (byte)((uint)(sq2_seqencer + 1) & 7u); + if (sq2_length_counter > 0 && sq2_valid_freq) + { + if (audio_sq2_outputable) + { + sq2_output = sq_duty_cycle_sequences[sq2_duty_cycle][sq2_seqencer] * sq2_envelope; + } + } + else + { + sq2_output = 0; + } + audio_signal_outputed = true; + } + + private static void SQ2ClockLength() + { + if (sq2_length_counter > 0 && !sq2_length_halt) + { + sq2_length_counter--; + if (apu_reg_access_happened && apu_reg_io_addr == 7 && apu_reg_access_w) + { + sq2_ignore_reload = true; + } + } + sq2_sweep_counter--; + if (sq2_sweep_counter == 0) + { + sq2_sweep_counter = sq2_sweep_devider_period + 1; + if (sq2_sweep_enable && sq2_sweep_shift_count > 0 && sq2_valid_freq) + { + sq2_sweep_change = sq2_timer >> (int)sq2_sweep_shift_count; + sq2_timer += (sq2_sweep_negate ? (-sq2_sweep_change) : sq2_sweep_change); + SQ2CalculateValidFreq(); + } + } + else if (sq2_sweep_reload) + { + sq2_sweep_counter = sq2_sweep_devider_period + 1; + sq2_sweep_reload = false; + } + } + + private static void SQ2ClockEnvelope() + { + if (sq2_envelope_start_flag) + { + sq2_envelope_start_flag = false; + sq2_envelope_decay_level_counter = 15; + sq2_envelope_devider = (byte)(sq2_volume_devider_period + 1); + } + else if (sq2_envelope_devider > 0) + { + sq2_envelope_devider--; + } + else + { + sq2_envelope_devider = (byte)(sq2_volume_devider_period + 1); + if (sq2_envelope_decay_level_counter > 0) + { + sq2_envelope_decay_level_counter--; + } + else if (sq2_length_halt) + { + sq2_envelope_decay_level_counter = 15; + } + } + sq2_envelope = (sq2_constant_volume_envelope ? sq2_volume_devider_period : sq2_envelope_decay_level_counter); + } + + private static void APUOnRegister4004() + { + if (apu_reg_access_w) + { + sq2_duty_cycle = (byte)((apu_reg_io_db & 0xC0) >> 6); + sq2_volume_devider_period = (byte)(apu_reg_io_db & 0xFu); + sq2_length_halt = (apu_reg_io_db & 0x20) != 0; + sq2_constant_volume_envelope = (apu_reg_io_db & 0x10) != 0; + sq2_envelope = (sq2_constant_volume_envelope ? sq2_volume_devider_period : sq2_envelope_decay_level_counter); + } + } + + private static void APUOnRegister4005() + { + if (apu_reg_access_w) + { + sq2_sweep_enable = (apu_reg_io_db & 0x80) == 128; + sq2_sweep_devider_period = (byte)((uint)(apu_reg_io_db >> 4) & 7u); + sq2_sweep_negate = (apu_reg_io_db & 8) == 8; + sq2_sweep_shift_count = (byte)(apu_reg_io_db & 7u); + sq2_sweep_reload = true; + SQ2CalculateValidFreq(); + } + } + + private static void APUOnRegister4006() + { + if (apu_reg_access_w) + { + sq2_timer = (sq2_timer & 0xFF00) | apu_reg_io_db; + SQ2CalculateValidFreq(); + } + } + + private static void APUOnRegister4007() + { + if (apu_reg_access_w) + { + sq2_timer = (sq2_timer & 0xFF) | ((apu_reg_io_db & 7) << 8); + if (sq2_length_enabled && !sq2_ignore_reload) + { + sq2_length_counter = sq_duration_table[apu_reg_io_db >> 3]; + } + if (sq2_ignore_reload) + { + sq2_ignore_reload = false; + } + sq2_seqencer = 0; + sq2_envelope_start_flag = true; + SQ2CalculateValidFreq(); + } + } + + private static void SQ2On4015() + { + sq2_length_enabled = (apu_reg_io_db & 2) != 0; + if (!sq2_length_enabled) + { + sq2_length_counter = 0; + } + } + + private static void SQ2Read4015() + { + if (sq2_length_counter > 0) + { + apu_reg_io_db = (byte)((apu_reg_io_db & 0xFDu) | 2u); + } + } + + private static void SQ2CalculateValidFreq() + { + sq2_valid_freq = sq2_timer >= 8 && (sq2_sweep_negate || ((sq2_timer + (sq2_timer >> (int)sq2_sweep_shift_count)) & 0x800) == 0); + } + + private static void SQ2WriteState(ref BinaryWriter bin) + { + bin.Write(sq2_duty_cycle); + bin.Write(sq2_length_halt); + bin.Write(sq2_constant_volume_envelope); + bin.Write(sq2_volume_devider_period); + bin.Write(sq2_sweep_enable); + bin.Write(sq2_sweep_devider_period); + bin.Write(sq2_sweep_negate); + bin.Write(sq2_sweep_shift_count); + bin.Write(sq2_timer); + bin.Write(sq2_period_devider); + bin.Write(sq2_seqencer); + bin.Write(sq2_length_enabled); + bin.Write(sq2_length_counter); + bin.Write(sq2_envelope_start_flag); + bin.Write(sq2_envelope_devider); + bin.Write(sq2_envelope_decay_level_counter); + bin.Write(sq2_envelope); + bin.Write(sq2_sweep_counter); + bin.Write(sq2_sweep_reload); + bin.Write(sq2_sweep_change); + bin.Write(sq2_valid_freq); + bin.Write(sq2_output); + bin.Write(sq2_ignore_reload); + } + + private static void SQ2ReadState(ref BinaryReader bin) + { + sq2_duty_cycle = bin.ReadByte(); + sq2_length_halt = bin.ReadBoolean(); + sq2_constant_volume_envelope = bin.ReadBoolean(); + sq2_volume_devider_period = bin.ReadByte(); + sq2_sweep_enable = bin.ReadBoolean(); + sq2_sweep_devider_period = bin.ReadByte(); + sq2_sweep_negate = bin.ReadBoolean(); + sq2_sweep_shift_count = bin.ReadByte(); + sq2_timer = bin.ReadInt32(); + sq2_period_devider = bin.ReadInt32(); + sq2_seqencer = bin.ReadByte(); + sq2_length_enabled = bin.ReadBoolean(); + sq2_length_counter = bin.ReadInt32(); + sq2_envelope_start_flag = bin.ReadBoolean(); + sq2_envelope_devider = bin.ReadByte(); + sq2_envelope_decay_level_counter = bin.ReadByte(); + sq2_envelope = bin.ReadByte(); + sq2_sweep_counter = bin.ReadInt32(); + sq2_sweep_reload = bin.ReadBoolean(); + sq2_sweep_change = bin.ReadInt32(); + sq2_valid_freq = bin.ReadBoolean(); + sq2_output = bin.ReadInt32(); + sq2_ignore_reload = bin.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs.meta new file mode 100644 index 00000000..7482bdc2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/NesEmu.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0ec9e5a66367f0e47b07118c94860367 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs new file mode 100644 index 00000000..a5782753 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs @@ -0,0 +1,18 @@ +namespace MyNes.Core +{ + internal enum PRGArea : byte + { + Area4000 = 4, + Area5000, + Area6000, + Area7000, + Area8000, + Area9000, + AreaA000, + AreaB000, + AreaC000, + AreaD000, + AreaE000, + AreaF000 + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs.meta new file mode 100644 index 00000000..32839e39 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/PRGArea.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: dd5f1d06c4c4b974992327c971b3e73c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs new file mode 100644 index 00000000..28958478 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs @@ -0,0 +1,10 @@ +namespace MyNes.Core +{ + public enum PaletteSelectSetting + { + AUTO, + ForceNTSC, + ForcePALB, + File + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs.meta new file mode 100644 index 00000000..13d79b9b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/PaletteSelectSetting.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1288fd99cf9cd5b49a99b7de9ac5b778 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs new file mode 100644 index 00000000..3c07fdcf --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs @@ -0,0 +1,10 @@ +namespace MyNes.Core +{ + public enum RegionSetting + { + AUTO, + ForceNTSC, + ForcePALB, + ForceDENDY + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs.meta new file mode 100644 index 00000000..10a2dee2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RegionSetting.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4e8c869e43765a14cb6c803a65bdf3fe +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs new file mode 100644 index 00000000..6992b3b2 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs @@ -0,0 +1,4 @@ +namespace MyNes.Core +{ + internal delegate void RenderAudioSamples(ref short[] buffer, ref int samples_added); +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs.meta new file mode 100644 index 00000000..6d259fd1 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderAudioSamples.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 2c87ada4318a5b54eb32c459316a236c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs new file mode 100644 index 00000000..82dbc3dd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs @@ -0,0 +1,4 @@ +namespace MyNes.Core +{ + internal delegate void RenderVideoFrame(ref int[] buffer); +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs.meta new file mode 100644 index 00000000..a77008ab --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RenderVideoFrame.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 368bc7a67bdd3134781f923b3685629c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs new file mode 100644 index 00000000..3e6e2131 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs @@ -0,0 +1,106 @@ +namespace MyNes.Core +{ + public class RendererSettings : ISettings + { + public string Video_ProviderID = ""; + + public bool Vid_AutoStretch = true; + + public bool Vid_Res_Upscale = true; + + public int Vid_Res_W = 640; + + public int Vid_Res_H = 480; + + public int Vid_StretchMultiply = 2; + + public bool Vid_KeepAspectRatio = true; + + public bool Vid_ShowFPS = false; + + public bool Vid_Fullscreen = false; + + public bool Vid_HardwareVertexProcessing = false; + + public bool Vid_VSync = false; + + public bool Vid_ShowNotifications = true; + + public int Vid_Filter = 1; + + public bool FrameSkipEnabled = false; + + public int FrameSkipInterval = 2; + + public bool UseEmuThread = true; + + public string Audio_ProviderID = ""; + + public bool Audio_EnableFilters = true; + + public int Audio_Volume = 100; + + public bool Audio_SoundEnabled = true; + + public int Audio_Frequency = 44100; + + public int Audio_InternalSamplesCount = 4096; + + public int Audio_InternalPeekLimit = 124; + + public int Audio_PlaybackAmplitude = 200; + + public int Audio_PlaybackBufferSizeInKB = 8; + + public bool Audio_UseDefaultMixer = true; + + public bool Audio_ChannelEnabled_SQ1 = true; + + public bool Audio_ChannelEnabled_SQ2 = true; + + public bool Audio_ChannelEnabled_NOZ = true; + + public bool Audio_ChannelEnabled_TRL = true; + + public bool Audio_ChannelEnabled_DMC = true; + + public bool Audio_ChannelEnabled_MMC5_SQ1 = true; + + public bool Audio_ChannelEnabled_MMC5_SQ2 = true; + + public bool Audio_ChannelEnabled_MMC5_PCM = true; + + public bool Audio_ChannelEnabled_VRC6_SQ1 = true; + + public bool Audio_ChannelEnabled_VRC6_SQ2 = true; + + public bool Audio_ChannelEnabled_VRC6_SAW = true; + + public bool Audio_ChannelEnabled_SUN1 = true; + + public bool Audio_ChannelEnabled_SUN2 = true; + + public bool Audio_ChannelEnabled_SUN3 = true; + + public bool Audio_ChannelEnabled_NMT1 = true; + + public bool Audio_ChannelEnabled_NMT2 = true; + + public bool Audio_ChannelEnabled_NMT3 = true; + + public bool Audio_ChannelEnabled_NMT4 = true; + + public bool Audio_ChannelEnabled_NMT5 = true; + + public bool Audio_ChannelEnabled_NMT6 = true; + + public bool Audio_ChannelEnabled_NMT7 = true; + + public bool Audio_ChannelEnabled_NMT8 = true; + + public RendererSettings(string path) + : base(path) + { + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs.meta new file mode 100644 index 00000000..f8ecf0fd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/RendererSettings.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 65409a509e1cd8343b1a67971176f0df +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs new file mode 100644 index 00000000..80c644ef --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs @@ -0,0 +1,22 @@ +namespace MyNes.Core +{ + public struct SRAMBankInfo + { + public int id; + + public string SIZE; + + public bool BATTERY; + + public SRAMBankInfo(int id, string SIZE, bool BATTERY) + { + this.SIZE = SIZE; + if (SIZE == "0kb") + { + SIZE = "8kb"; + } + this.BATTERY = BATTERY; + this.id = id; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs.meta new file mode 100644 index 00000000..e8368229 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SRAMBankInfo.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 712dc9b2310b1de45b4136197e974a90 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs new file mode 100644 index 00000000..9e2fd28b --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs @@ -0,0 +1,33 @@ +namespace MyNes.Core +{ + internal class SoundDCBlockerFilter + { + private double R; + + private double y; + + private double y_1; + + private double x; + + private double x_1; + + public SoundDCBlockerFilter(double R) + { + this.R = R; + } + + public void Reset() + { + y = (y_1 = (x = (x_1 = 0.0))); + } + + public void DoFiltering(double sample, out double filtered) + { + x = sample; + filtered = (y = x - x_1 + R * y_1); + x_1 = x; + y_1 = y; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs.meta new file mode 100644 index 00000000..fe30c288 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundDCBlockerFilter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 45a36044c5210bb498dc54c616867e2a +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs new file mode 100644 index 00000000..8102bb66 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs @@ -0,0 +1,36 @@ +using System; + +namespace MyNes.Core +{ + internal class SoundHighPassFilter + { + private double K; + + private double y_1; + + private double x_1; + + public SoundHighPassFilter(double k) + { + K = k; + } + + public void Reset() + { + y_1 = (x_1 = 0.0); + } + + public void DoFiltering(double sample, out double filtered) + { + filtered = K * y_1 + K * (sample - x_1); + x_1 = sample; + y_1 = filtered; + } + + public static double GetK(double dt, double fc) + { + double num = Math.PI * 2.0 * dt * fc; + return num / (num + 1.0); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs.meta new file mode 100644 index 00000000..3edbcd67 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundHighPassFilter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 57d536cd18a24114691257432ce2b8b9 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs new file mode 100644 index 00000000..5a39d40f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs @@ -0,0 +1,45 @@ +using System; + +namespace MyNes.Core +{ + internal class SoundLowPassFilter + { + private double K; + + private double K_1; + + private double y; + + private double y_1; + + private double x; + + private double x_1; + + public SoundLowPassFilter(double k) + { + K = k; + K_1 = 1.0 - k; + } + + public void Reset(double k) + { + y = (y_1 = (x = (x_1 = 0.0))); + K = k; + K_1 = 1.0 - k; + } + + public void DoFiltering(double sample, out double filtered) + { + filtered = K * sample + K_1 * y_1; + x_1 = sample; + y_1 = filtered; + } + + public static double GetK(double dt, double fc) + { + double num = Math.PI * 2.0 * dt * fc; + return num / (num + 1.0); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs.meta new file mode 100644 index 00000000..aeb419f7 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/SoundLowPassFilter.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 0556466d0c6a1ef44a16c28d90524615 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs new file mode 100644 index 00000000..656a200d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs @@ -0,0 +1,197 @@ +using System; +using System.IO; +using System.Text; + +namespace MyNes.Core +{ + public class StateHandler + { + public static int Slot = 0; + + internal static string StateFolder = "States"; + + private const byte state_version = 7; + + private static bool IsSavingState = false; + + private static bool IsLoadingState = false; + + public static void SaveState(string fileName, bool saveImage) + { + if (!NesEmu.ON) + { + Tracer.WriteError("Can't save state, emu is off."); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error1, instant: false); + return; + } + if (!File.Exists(NesEmu.CurrentFilePath)) + { + Tracer.WriteError("Can't save state, no rom file is loaded."); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error2, instant: false); + return; + } + if (IsLoadingState) + { + Tracer.WriteError("Can't save state while loading a state !"); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error3, instant: false); + return; + } + if (IsSavingState) + { + Tracer.WriteError("Already saving state !!"); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error4, instant: false); + return; + } + IsSavingState = true; + Stream output = new MemoryStream(); + BinaryWriter bin = new BinaryWriter(output); + bin.Write(Encoding.ASCII.GetBytes("MNS")); + bin.Write((byte)7); + for (int i = 0; i < NesEmu.SHA1.Length; i += 2) + { + string value = NesEmu.SHA1.Substring(i, 2).ToUpper(); + bin.Write(Convert.ToByte(value, 16)); + } + NesEmu.WriteStateData(ref bin); + byte[] outData = new byte[0]; + ZlipWrapper.CompressData(((MemoryStream)bin.BaseStream).GetBuffer(), out outData); + Stream stream = new FileStream(fileName, FileMode.Create, FileAccess.Write); + stream.Write(outData, 0, outData.Length); + MyNesMain.VideoProvider.TakeSnapshotAs(fileName.Replace(".mns", ".jpg"), ".jpg"); + bin.Flush(); + bin.Close(); + stream.Flush(); + stream.Close(); + IsSavingState = false; + Tracer.WriteInformation("State saved at slot " + Slot); + MyNesMain.VideoProvider.WriteInfoNotification(MNInterfaceLanguage.Message_Info1 + " " + Slot, instant: false); + } + + public static void SaveState(int Slot) + { + if (StateFolder == "States") + { + StateFolder = Path.Combine(MyNesMain.WorkingFolder, "States"); + } + Directory.CreateDirectory(StateFolder); + string fileName = Path.Combine(StateFolder, Path.GetFileNameWithoutExtension(NesEmu.CurrentFilePath)) + "_" + Slot + ".mns"; + SaveState(fileName, saveImage: false); + } + + public static void SaveState() + { + SaveState(Slot); + } + + public static void LoadState() + { + LoadState(Slot); + } + + public static void LoadState(string fileName) + { + if (!NesEmu.ON) + { + Tracer.WriteError("Can't load state, emu is off."); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error5, instant: false); + return; + } + if (!File.Exists(NesEmu.CurrentFilePath)) + { + Tracer.WriteError("Can't load state, no rom file is loaded."); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error6, instant: false); + return; + } + if (IsSavingState) + { + Tracer.WriteError("Can't load state while saving a state !"); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error7, instant: false); + return; + } + if (IsLoadingState) + { + Tracer.WriteError("Already loading state !!"); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error8, instant: false); + return; + } + if (!File.Exists(fileName)) + { + Tracer.WriteError("No state found in slot " + Slot); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error9 + " " + Slot, instant: false); + return; + } + IsLoadingState = true; + Stream stream = new FileStream(fileName, FileMode.Open, FileAccess.Read); + byte[] array = new byte[stream.Length]; + byte[] outData = new byte[0]; + stream.Read(array, 0, array.Length); + stream.Close(); + ZlipWrapper.DecompressData(array, out outData); + BinaryReader bin = new BinaryReader(new MemoryStream(outData)); + byte[] array2 = new byte[3]; + bin.Read(array2, 0, array2.Length); + if (Encoding.ASCII.GetString(array2) != "MNS") + { + Tracer.WriteError("Unable load state at slot " + Slot + "; Not My Nes State File !"); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error10 + " " + Slot + "; " + MNInterfaceLanguage.Message_Error11, instant: false); + IsLoadingState = false; + return; + } + if (bin.ReadByte() != 7) + { + Tracer.WriteError("Unable load state at slot " + Slot + "; Not compatible state file version !"); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error10 + " " + Slot + "; " + MNInterfaceLanguage.Message_Error12, instant: false); + IsLoadingState = false; + return; + } + string text = ""; + for (int i = 0; i < NesEmu.SHA1.Length; i += 2) + { + text += bin.ReadByte().ToString("X2"); + } + if (text.ToLower() != NesEmu.SHA1.ToLower()) + { + Tracer.WriteError("Unable load state at slot " + Slot + "; This state file is not for this game; not same SHA1 !"); + MyNesMain.VideoProvider.WriteErrorNotification(MNInterfaceLanguage.Message_Error10 + " " + Slot + "; " + MNInterfaceLanguage.Message_Error13, instant: false); + IsLoadingState = false; + } + else + { + NesEmu.ReadStateData(ref bin); + bin.Close(); + IsLoadingState = false; + Tracer.WriteInformation("State loaded from slot " + Slot); + MyNesMain.VideoProvider.WriteInfoNotification(MNInterfaceLanguage.Message_Info2 + " " + Slot, instant: false); + } + } + + public static void LoadState(int Slot) + { + if (StateFolder == "States") + { + StateFolder = Path.Combine(MyNesMain.WorkingFolder, "States"); + } + Directory.CreateDirectory(StateFolder); + string fileName = Path.Combine(StateFolder, Path.GetFileNameWithoutExtension(NesEmu.CurrentFilePath)) + "_" + Slot + ".mns"; + LoadState(fileName); + } + + public static string GetStateFile(int slot) + { + if (File.Exists(NesEmu.CurrentFilePath)) + { + return Path.Combine(StateFolder, Path.GetFileNameWithoutExtension(NesEmu.CurrentFilePath)) + "_" + slot + ".mns"; + } + return ""; + } + + public static string GetStateImageFile(int slot) + { + if (File.Exists(NesEmu.CurrentFilePath)) + { + return Path.Combine(StateFolder, Path.GetFileNameWithoutExtension(NesEmu.CurrentFilePath)) + "_" + slot + ".jpg"; + } + return ""; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs.meta new file mode 100644 index 00000000..005d25da --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/StateHandler.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: bfc53d94a1834ed4d9206ff9f45d7d74 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs new file mode 100644 index 00000000..1f269dbd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs @@ -0,0 +1,95 @@ +using System.IO; + +namespace MyNes.Core +{ + internal class Sunsoft5BChnl + { + internal bool Enabled; + + internal byte Volume; + + private int dutyStep = 0; + + private int freqTimer; + + private int frequency; + + private int cycles; + + internal int output; + + internal bool Outputable; + + internal void HardReset() + { + } + + internal void SoftReset() + { + } + + internal void Write0(ref byte data) + { + frequency = (frequency & 0xF00) | data; + freqTimer = (frequency + 1) * 2; + } + + internal void Write1(ref byte data) + { + frequency = (frequency & 0xFF) | ((data & 0xF) << 8); + freqTimer = (frequency + 1) * 2; + } + + internal void ClockSingle() + { + if (--cycles > 0) + { + return; + } + cycles = freqTimer; + dutyStep = (dutyStep + 1) & 0x1F; + if (dutyStep <= 15) + { + if (Enabled) + { + if (Outputable) + { + output = Volume; + } + else + { + output = 0; + } + } + else + { + output = 0; + } + } + else + { + output = 0; + } + } + + internal void SaveState(ref BinaryWriter stream) + { + stream.Write(Enabled); + stream.Write(Volume); + stream.Write(dutyStep); + stream.Write(freqTimer); + stream.Write(frequency); + stream.Write(cycles); + } + + internal void LoadState(ref BinaryReader stream) + { + Enabled = stream.ReadBoolean(); + Volume = stream.ReadByte(); + dutyStep = stream.ReadInt32(); + freqTimer = stream.ReadInt32(); + frequency = stream.ReadInt32(); + cycles = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs.meta new file mode 100644 index 00000000..885619b6 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Sunsoft5BChnl.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 539fe437ee1b9e14eadfb0477280528d +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs new file mode 100644 index 00000000..b10ad383 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs @@ -0,0 +1,4 @@ +namespace MyNes.Core +{ + internal delegate void TogglePause(bool paused); +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs.meta new file mode 100644 index 00000000..f8e8b657 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/TogglePause.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c1371973a9935de4192a2d7b9556184c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs new file mode 100644 index 00000000..1d130390 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs @@ -0,0 +1,59 @@ +using System; + +namespace MyNes.Core +{ + public sealed class Tracer + { + public static event EventHandler EventRaised; + + public static void WriteLine(string message) + { + Tracer.EventRaised?.Invoke(null, new TracerEventArgs(message, TracerStatus.Normal)); + } + + public static void WriteLine(string message, string category) + { + Tracer.EventRaised?.Invoke(null, new TracerEventArgs($"{category}: {message}", TracerStatus.Normal)); + } + + public static void WriteLine(string message, TracerStatus status) + { + Tracer.EventRaised?.Invoke(null, new TracerEventArgs(message, status)); + } + + public static void WriteLine(string message, string category, TracerStatus status) + { + Tracer.EventRaised?.Invoke(null, new TracerEventArgs($"{category}: {message}", status)); + } + + public static void WriteError(string message) + { + WriteLine(message, TracerStatus.Error); + } + + public static void WriteError(string message, string category) + { + WriteLine(message, category, TracerStatus.Error); + } + + public static void WriteWarning(string message) + { + WriteLine(message, TracerStatus.Warning); + } + + public static void WriteWarning(string message, string category) + { + WriteLine(message, category, TracerStatus.Warning); + } + + public static void WriteInformation(string message) + { + WriteLine(message, TracerStatus.Infromation); + } + + public static void WriteInformation(string message, string category) + { + WriteLine(message, category, TracerStatus.Infromation); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs.meta new file mode 100644 index 00000000..92463c33 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/Tracer.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4a65f5b3f03ae904e99821cebd149964 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs new file mode 100644 index 00000000..e7f3ff60 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs @@ -0,0 +1,17 @@ +using System; + +namespace MyNes.Core +{ + public class TracerEventArgs : EventArgs + { + public string Message { get; private set; } + + public TracerStatus Status { get; private set; } + + public TracerEventArgs(string message, TracerStatus status) + { + Message = message; + Status = status; + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs.meta new file mode 100644 index 00000000..22ecc606 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerEventArgs.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 1aabf28e631b0dc42bf07ea3700eaa1c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs new file mode 100644 index 00000000..d0992942 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs @@ -0,0 +1,10 @@ +namespace MyNes.Core +{ + public enum TracerStatus + { + Normal, + Error, + Warning, + Infromation + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs.meta new file mode 100644 index 00000000..e846c1e9 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/TracerStatus.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 52a5cc3ce27c2a04cb04aa200cc3c798 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs new file mode 100644 index 00000000..789ea01a --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs @@ -0,0 +1,116 @@ +using System.IO; + +namespace MyNes.Core +{ + internal class VRC6Pulse + { + private int dutyForm; + + private int dutyStep; + + private bool enabled = true; + + internal bool Outputable; + + private bool mode = false; + + private byte volume; + + private int freqTimer; + + private int frequency; + + private int cycles; + + internal int output; + + internal void HardReset() + { + dutyForm = 0; + dutyStep = 15; + enabled = true; + mode = false; + output = 0; + } + + internal void Write0(ref byte data) + { + mode = (data & 0x80) == 128; + dutyForm = (data & 0x70) >> 4; + volume = (byte)(data & 0xFu); + } + + internal void Write1(ref byte data) + { + frequency = (frequency & 0xF00) | data; + } + + internal void Write2(ref byte data) + { + frequency = (frequency & 0xFF) | ((data & 0xF) << 8); + enabled = (data & 0x80) == 128; + } + + internal void ClockSingle() + { + if (--cycles > 0) + { + return; + } + cycles = (frequency << 1) + 2; + if (!enabled) + { + return; + } + if (mode) + { + output = volume; + return; + } + dutyStep--; + if (dutyStep < 0) + { + dutyStep = 15; + } + if (dutyStep <= dutyForm) + { + if (Outputable) + { + output = volume; + } + else + { + output = 0; + } + } + else + { + output = 0; + } + } + + internal void SaveState(ref BinaryWriter stream) + { + stream.Write(dutyForm); + stream.Write(dutyStep); + stream.Write(enabled); + stream.Write(mode); + stream.Write(volume); + stream.Write(freqTimer); + stream.Write(frequency); + stream.Write(cycles); + } + + internal void LoadState(ref BinaryReader stream) + { + dutyForm = stream.ReadInt32(); + dutyStep = stream.ReadInt32(); + enabled = stream.ReadBoolean(); + mode = stream.ReadBoolean(); + volume = stream.ReadByte(); + freqTimer = stream.ReadInt32(); + frequency = stream.ReadInt32(); + cycles = stream.ReadInt32(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs.meta new file mode 100644 index 00000000..82f6312d --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Pulse.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 74455848e758856438177d90dada5bd7 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs new file mode 100644 index 00000000..23b2e565 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs @@ -0,0 +1,114 @@ +using System.IO; + +namespace MyNes.Core +{ + internal class VRC6Sawtooth + { + private byte AccumRate; + + private int accumClock; + + private byte accumulationRegister; + + private int frequency; + + private int freqTimer; + + private int cycles; + + private bool enabled; + + internal int output; + + internal bool Outputable; + + internal void HardReset() + { + } + + private void UpdateFrequency() + { + freqTimer = (frequency + 1) * 2; + } + + internal void Write0(ref byte data) + { + AccumRate = (byte)(data & 0x3Fu); + } + + internal void Write1(ref byte data) + { + frequency = (frequency & 0xF00) | data; + UpdateFrequency(); + } + + internal void Write2(ref byte data) + { + frequency = (frequency & 0xFF) | ((data & 0xF) << 8); + enabled = (data & 0x80) == 128; + UpdateFrequency(); + } + + internal void ClockSingle() + { + if (--cycles > 0) + { + return; + } + cycles = freqTimer; + if (enabled) + { + accumClock++; + switch (++accumClock) + { + case 2: + case 4: + case 6: + case 8: + case 10: + case 12: + accumulationRegister += AccumRate; + break; + case 14: + accumulationRegister = 0; + accumClock = 0; + break; + } + if (Outputable) + { + output = (accumulationRegister >> 3) & 0x1F; + } + else + { + output = 0; + } + } + else + { + output = 0; + } + } + + internal void SaveState(ref BinaryWriter stream) + { + stream.Write(AccumRate); + stream.Write(accumClock); + stream.Write(accumulationRegister); + stream.Write(frequency); + stream.Write(freqTimer); + stream.Write(cycles); + stream.Write(enabled); + } + + internal void LoadState(ref BinaryReader stream) + { + AccumRate = stream.ReadByte(); + accumClock = stream.ReadInt32(); + accumulationRegister = stream.ReadByte(); + frequency = stream.ReadInt32(); + freqTimer = stream.ReadInt32(); + cycles = stream.ReadInt32(); + enabled = stream.ReadBoolean(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs.meta new file mode 100644 index 00000000..1578d023 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/VRC6Sawtooth.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 382f7496025cd8846bf6459a0ad7f45f +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs new file mode 100644 index 00000000..e90b79fe --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs @@ -0,0 +1,302 @@ +using System; +using System.IO; +using System.Text; + +namespace MyNes.Core +{ + public class WaveRecorder + { + private string _fileName; + + private Stream STR; + + private bool _IsRecording = false; + + private int SIZE = 0; + + private int NoOfSamples = 0; + + private int _Time = 0; + + private int TimeSamples = 0; + + private short channels; + + private short bitsPerSample; + + private int Frequency; + + public int Time => _Time; + + public bool IsRecording => _IsRecording; + + public void Record(string FilePath, short channels, short bitsPerSample, int Frequency) + { + _fileName = FilePath; + this.channels = channels; + this.bitsPerSample = bitsPerSample; + this.Frequency = Frequency; + _Time = 0; + STR = new FileStream(FilePath, FileMode.Create); + ASCIIEncoding aSCIIEncoding = new ASCIIEncoding(); + STR.Write(aSCIIEncoding.GetBytes("RIFF"), 0, 4); + STR.WriteByte(0); + STR.WriteByte(0); + STR.WriteByte(0); + STR.WriteByte(0); + STR.Write(aSCIIEncoding.GetBytes("WAVE"), 0, 4); + STR.Write(aSCIIEncoding.GetBytes("fmt "), 0, 4); + STR.WriteByte(16); + STR.WriteByte(0); + STR.WriteByte(0); + STR.WriteByte(0); + STR.WriteByte(1); + STR.WriteByte(0); + STR.WriteByte((byte)((uint)channels & 0xFFu)); + STR.WriteByte((byte)((channels & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)Frequency & 0xFFu)); + STR.WriteByte((byte)((Frequency & 0xFF00) >> 8)); + STR.WriteByte((byte)((Frequency & 0xFF0000) >> 16)); + STR.WriteByte((byte)((Frequency & 0xFF000000u) >> 24)); + int num = Frequency * channels * (bitsPerSample / 8); + STR.WriteByte((byte)((uint)num & 0xFFu)); + STR.WriteByte((byte)((num & 0xFF00) >> 8)); + STR.WriteByte((byte)((num & 0xFF0000) >> 16)); + STR.WriteByte((byte)((num & 0xFF000000u) >> 24)); + short num2 = (short)(channels * (bitsPerSample / 8)); + STR.WriteByte((byte)((uint)num2 & 0xFFu)); + STR.WriteByte((byte)((num2 & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)bitsPerSample & 0xFFu)); + STR.WriteByte((byte)((bitsPerSample & 0xFF00) >> 8)); + STR.Write(aSCIIEncoding.GetBytes("data"), 0, 4); + STR.WriteByte(0); + STR.WriteByte(0); + STR.WriteByte(0); + STR.WriteByte(0); + _IsRecording = true; + } + + public void AddBuffer(ref byte[] buffer) + { + for (int i = 0; i < buffer.Length; i++) + { + switch (channels) + { + case 1: + switch (bitsPerSample) + { + case 8: + STR.WriteByte(buffer[i]); + NoOfSamples++; + TimeSamples++; + if (TimeSamples >= Frequency) + { + _Time++; + TimeSamples = 0; + } + break; + case 16: + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + NoOfSamples++; + TimeSamples++; + if (TimeSamples >= Frequency) + { + _Time++; + TimeSamples = 0; + } + break; + case 32: + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + NoOfSamples++; + TimeSamples++; + if (TimeSamples >= Frequency) + { + _Time++; + TimeSamples = 0; + } + break; + } + break; + case 2: + switch (bitsPerSample) + { + case 8: + STR.WriteByte(buffer[i]); + STR.WriteByte(buffer[i]); + NoOfSamples++; + TimeSamples++; + if (TimeSamples >= Frequency) + { + _Time++; + TimeSamples = 0; + } + break; + case 16: + STR.WriteByte(buffer[i]); + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + STR.WriteByte(buffer[i]); + NoOfSamples++; + TimeSamples++; + if (TimeSamples >= Frequency) + { + _Time++; + TimeSamples = 0; + } + break; + case 32: + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + STR.WriteByte(buffer[i]); + i++; + STR.WriteByte(buffer[i]); + NoOfSamples++; + TimeSamples++; + if (TimeSamples >= Frequency) + { + _Time++; + TimeSamples = 0; + } + break; + } + break; + } + } + } + + public void AddSample(int Sample) + { + if (!_IsRecording) + { + return; + } + switch (channels) + { + case 1: + switch (bitsPerSample) + { + case 8: + AddSample_mono_08(Sample); + break; + case 16: + AddSample_mono_16(Sample); + break; + case 32: + AddSample_mono_32(Sample); + break; + } + break; + case 2: + switch (bitsPerSample) + { + case 8: + AddSample_stereo_08(Sample); + break; + case 16: + AddSample_stereo_16(Sample); + break; + case 32: + AddSample_stereo_32(Sample); + break; + } + break; + } + NoOfSamples++; + TimeSamples++; + if (TimeSamples >= Frequency) + { + _Time++; + TimeSamples = 0; + } + } + + private void AddSample_mono_08(int Sample) + { + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + } + + private void AddSample_mono_16(int Sample) + { + STR.WriteByte((byte)((Sample & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + } + + private void AddSample_mono_32(int Sample) + { + STR.WriteByte((byte)((Sample & 0xFF000000u) >> 24)); + STR.WriteByte((byte)((Sample & 0xFF0000) >> 16)); + STR.WriteByte((byte)((Sample & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + } + + private void AddSample_stereo_08(int Sample) + { + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + } + + private void AddSample_stereo_16(int Sample) + { + STR.WriteByte((byte)((Sample & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + STR.WriteByte((byte)((Sample & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + } + + private void AddSample_stereo_32(int Sample) + { + STR.WriteByte((byte)((Sample & 0xFF000000u) >> 24)); + STR.WriteByte((byte)((Sample & 0xFF0000) >> 16)); + STR.WriteByte((byte)((Sample & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + STR.WriteByte((byte)((Sample & 0xFF000000u) >> 24)); + STR.WriteByte((byte)((Sample & 0xFF0000) >> 16)); + STR.WriteByte((byte)((Sample & 0xFF00) >> 8)); + STR.WriteByte((byte)((uint)Sample & 0xFFu)); + } + + public void Stop() + { + if (_IsRecording & (STR != null)) + { + NoOfSamples *= channels * (bitsPerSample / 8); + SIZE = NoOfSamples + 36; + byte[] array = new byte[4]; + byte[] array2 = new byte[4]; + array = BitConverter.GetBytes(SIZE); + if (!BitConverter.IsLittleEndian) + { + Array.Reverse(array); + } + array2 = BitConverter.GetBytes(NoOfSamples); + if (!BitConverter.IsLittleEndian) + { + Array.Reverse(array2); + } + _IsRecording = false; + STR.Position = 4L; + STR.Write(array, 0, 4); + STR.Position = 40L; + STR.Write(array2, 0, 4); + STR.Close(); + MyNesMain.VideoProvider.WriteInfoNotification("Sound file saved at " + Path.GetFileName(_fileName), instant: false); + } + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs.meta new file mode 100644 index 00000000..6ea774f0 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/WaveRecorder.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 889c64e645ae5744e9af609edb7b2db8 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs new file mode 100644 index 00000000..3d5b9823 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs @@ -0,0 +1,8 @@ +using System; + +namespace MyNes.Core +{ + internal class WithExternalSoundAttribute : Attribute + { + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs.meta new file mode 100644 index 00000000..b6aed36c --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/WithExternalSoundAttribute.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 8939062f5478c624088de22b10b3dda5 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs b/AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs new file mode 100644 index 00000000..26e02fe1 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs @@ -0,0 +1,39 @@ +using System.IO; +using ComponentAce.Compression.Libs.zlib; + +namespace MyNes.Core +{ + internal class ZlipWrapper + { + internal static void CompressData(byte[] inData, out byte[] outData) + { + using MemoryStream memoryStream = new MemoryStream(); + using ZOutputStream zOutputStream = new ZOutputStream(memoryStream, -1); + using Stream input = new MemoryStream(inData); + CopyStream(input, zOutputStream); + zOutputStream.finish(); + outData = memoryStream.ToArray(); + } + + internal static void DecompressData(byte[] inData, out byte[] outData) + { + using MemoryStream memoryStream = new MemoryStream(); + using ZOutputStream zOutputStream = new ZOutputStream(memoryStream); + using Stream input = new MemoryStream(inData); + CopyStream(input, zOutputStream); + zOutputStream.finish(); + outData = memoryStream.ToArray(); + } + + internal static void CopyStream(Stream input, Stream output) + { + byte[] buffer = new byte[2000]; + int count; + while ((count = input.Read(buffer, 0, 2000)) > 0) + { + output.Write(buffer, 0, count); + } + output.Flush(); + } + } +} diff --git a/AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs.meta b/AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs.meta new file mode 100644 index 00000000..107f3ee8 --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/MyNes.Core/ZlipWrapper.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: e1fde431f4bc3d548b77ff7467e14bc0 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/Plugins/Core.dll b/AxibugEmuOnline.Client/Assets/Plugins/Core.dll deleted file mode 100644 index 0f42ed5b2a6a17cf713d8879df93d44910568dda..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 347648 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໻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%0Af(!#y2deG> zYt$Iv`1VzybeMXpWNo3WH>dJP=x4IH+8yl>?2fhwoPBm2*CAX$k=Njw4wW;=4h*F$ z&H1`qRk5a83rVC+_S9RbV!6^3=(Op|j1_3!k}oYZv~N&HbKi(Lh|QsNBWJxUI9EAp zMRaYd?Y`>XRXZ_b;&+sMwMt-5XlG;xG1N_LyDeRu+zEFjXN_INbR7;WOkXVC-2BO6 zCNfBO0sImch03oTvbCQT$!eAs@V=-{&3WMpVwP@~GqXHidd&~v6)!b`Jv7W;v3m6D zem04F1&e}(lObNNu_!s&AL^ck&fuXO)pHK28<{$<^oiEZbE^U>{Xtxk6tbf_)#|E) zB7vqz%n_L-E%IxEpjdvJb&#Mr^eb(212$o`SRqt|xr^(@+G49#rDc^)1)${gP%wf# zcT1w-1mW5!h!}u^iMb&d_wcNt1h%$6F zgR>YxG9$VV#SVi8pph>|xhCWmrr?3mI9E|^I>HLm&Ly#Vu*Ij&w_OWGOvn}uHH(F^ zg4c<$F>hg8K23X1TF(6s2SuKa=_m2x@KcpJ_*5zwF`bbn(%h*u^R>B(4(la#w^)g^ zCoN2yfM`z>)}!{j4`W+JZ|@YJtP~x4G=1SPOXG5tN>P>s8)dL<9+Vhl!JGiiBZc|A zQ1DWOh+n357(T{%ER8!uodjGT=p^7(hE76Th@D(GnaXySoJy`h5cZ51el6R`bw!3& zj^vzQF(o6NY{cNrbKc1y$qaQu5-A}{Y&!(M!a`Z-98ab-aZKDILPmP;b*PA4pvH!< zt&+u3cPxbBFlwkNM`p3nR3nR{1c2MXbwu)+5y^c=!k~1U(wM4+~TAo-44hLJ33&!v7-alEfyVsoVL^O)$Mi&x~?6- z#f^3VdcGYH8?Ui2-x4pr&Q)&GPzA%UJ+(m3YN}B`BI~kdz>MNrShhnUR|w7M2$`-B z8sZ9}C9V*f;tH`Xt{@uY3ZXTw5Srr(wcFz&%8ejt&`^_&!-}J;4nJJsuoEgZg$Mf* zwjE^-jY%f@;6tO-pg8%^fg|h$Ihlf5Z^#J|QkIMs9MV$@h}yPLafS9&TC*8z2QJR` zHsdZRl162^LZg72w|0yk0@9b#T*K0i!ff3Q?_UQyZw${(RTn3@)U{b1a-&cx&lMVn z+ELa$5Uv47;7@Bu*lD<;?bhzrc4W@_90eBKxk{6+*mj(q5KA+<0F_7vHn zvA1m#1uhC-V<3v=cS76XP7~#}XBrNpu(Yo`i5s8WTY zrctdpI5n*J9t<3qg31L&-==m`C*+*M?g!qaX2U2=f^&AU&Jw`d@7#N`IpqmA4-6$i zp+nmHEctjv&K2eqc6MRBz2*u*ZMZ`4+7!$DY=pgMlVj8->qkGMAnX+${lMiG$mixc z_jLrer4|RwGH4Sx1ueoX+M=V5z;y>Fon)k4fQmaGa0-bB0#ij?SZt0>W6da)kK);O zT4`LIR@16JJ*~?1X{>fQentigCsTH@TpvlQ>WFrYL;2m}M}*$v!A=&Z;%cQ+Q#mtV zCKfB%j0_!4cjq+|t4JG7qTCh{4ysU)O>Y}VZo1o`antF~nJCXFaZ(aN364LBVZEg! z25$x*hq3qI6Db26BO*aT=Z4AD3GG&d47_c@jmu53H__xSMS;q}daa1yFmZ3E8wC)~Q3T2C>Wf>#N zVNan^V<26}tbs#;IJ0t&X=LWLL(b(#Tsq?xtsMiqh?`@B%=-4%?_98C#dG zT@;K7W(r)UEO)YQA(+D!1~G2xNGgDIs|4-J8@Obr2W30FvquUIE*`D&QCB4PY#bQ! z;aI7@;qo2l1J zd2!^Z8XUtGicYn#Y8?(NdL4xuSBVZ=C_SOo!mW!|FxRTj%TAXT2yQB64$JqWq%LTH zeR)SZ)>vq9GR3!1 ztY3mtr->Gz;=+`HSH&W5ZaD;fok5MmudKwMY#)L^Cy)k3r4ioJw$KqACMr|!LAfUI zaFK`P8o@wk7pheuatN){N0E0MCy$)`3D-Q+lXY0G8PNEaZJiU;`#>vb?*gr$y$7^{ zrAo17p(w<{Ac+=6KYs&z*qTic8K~-VMF=dM+aJpjj=M1!m39DL@Yw?HCRm)*t);ph z5-G7_L`-zn+&JY5M_Tndnp!H)6XIf9y)ef|rF{jeqHnQOM0lXMFi`pgUg7YpRM4Ew z1zcKz&kfslwNR@-m*Mi%AyHyl=jL&|UO1#i>$q5dNKt`KW2?ubC%l)2n@MAjF1Hd;EbW?4Hgg)q z&5eaQjpD>pBk-wPHz@Rhfg(Y~&Bu3mn&<$ky6mT_mSjy5GpY)81UuNixiwy|F4pRExSgWr z+vq6I@~Yz2fd2^?Og4N24Lq2d_E^j1Fy2tn6O0I>T~n-u+hZh)dydzrWRpsYb;}GX zn#C%QFNfjV(COW`g{nfGkO@1Ei)mFF%^IcqBbPF#}F$6E4(kl$i zCw}IRs>FrT*(GG%5iM(+=42THL^I~3H!+X6Nb<5_VNUJmLWiegj4;3j!=iIqrl4o7 z;Qk@&n3rt(we#MJ;ugI^;)*_xfhn+)|ieU7ZkJWW<~#^Xjm8rA~f7xyIB$vSM4p_(Xm*k@xJ z!^?2MXhBks!^lC19VR-v=IXVY-fGt2880xwaHR!5Kn8mT1Su&nZ@H=Ix<`Qh5LTaQ zoQYzEPUJ<1a3+Dh4J_jFa%mxRm?5~GdHRGj1Sw0deB9zL=i5>~J(f z6?hCsgh#`CinDVBi!|15Yq^Fc!ycwJB*%q{tHBuj1|Vag`LMi?G-&x6tu}2rhv^Nf z(NSXWg6aLIn+OL#?YMNx+5|}Z%6YmAn8v*yz={m_k-i_y7C-jtH zU7m(0im#-d57eMUw%y=q%Y}p1LZ#5zA@`(h#V`RksJ=PPy9Xugs*yFP<#@fR1Kb`s zfn2%JD9(_EdSSR$b(KP>^=LwvX18&4;}-wj|X$HzbpFDAM^w zJY@dWt(}M*(b&6^!@SrAJ%zi9%wH{Wg3}@1sNm9`L{gmz!d{~QEtkZhb0fz~P{IUF zhKqF_iJ7+$oO31#X&X?^X6#DJ*2y~iG++%Dai0`JA$yD+xJd1$S&nvIkb{mSmo$D} zMjBehx{WW2&=rK!1{dMf{4E?TPEHt-P_3Xq8F7zdpWmol6(x#V%(Z~fVS%><=hf`I zqEoDi4JejV)g-{mG6~lqZsG8x3S%=j&soI)d=Utg9yFz-Bfy7U_U^O@=P1Y61Zm74%*$sTQE|xZPufUv{6K* zN5738uZFd~G_gQ5#MsA0DPZO-faS8}_FP)$yc~jeS=eDhw0EL>!NnmuoOo@;Av&CH zB6|i4P?+16HmfrxQfKnul@KSRrD3NuyvgD)IYf+@5V7OU$&OcFJ{3WA-$vEWlL9{r zZgJr7vDv|q*~_zLtj(BXIG@5@u!cStD&NAZTnyk$$PU0lqNA5&@tjOzA+ZQ4n3N6p zIV84L=i3riRwy0!>g>yOlHn2(M`a0#qrQa1{n-*yhmX#>M<)f?CyP$09o{*cb4U~# z+&0u@4e7p43e|ci1#_^I0BzYx7!n24k!dG|nRZf;X(s`hE+OzZUR(j~YjIXs^tOdl zyLMa|776I2j1&%)J1IHnMeG%tjA7Bw%XxIHgZI%Jee76vX0(>6OqR9=oietMzCmRA$RDv{Tq@Fq&b zxqeC%n^ioZARGt0s0H{Smn-b^lu`Xyc^A{I`P?X;)BQ9cgGo5tmBko&vN>eHS&6 z^{SlKLN1jh`CHUE(OGIgmYqFVN||cNwe!Gh^VoLPu$nVMzvJX0?3Wa6f>uppUp8dr zhc&{KIbPyQo8q^@cF;xK4(H)GjxXG_;rbC5$)H!)iq<__Drz?hub}9x#7l!~XmZmO zd4upLAx22Xwz;67GYgdbk(+#0(B;*Ep~gP@>rnw+M8}6+V5=oZCa{=SIc?6%r3Bv z#2nx6!4l8A0NPGd;xOa)?_qd5#K}5F5sac-^Sf|+LSI%KfuEs)*ZhiyutAn{l6pgS z?Rel*`wQ5@K#$B7!}&rLj^<1B#T%wnR(`@j}q~R)9r` zXy`NR)AJ0;;iAyD@V1oNQj~{t>c1Nj4zgb313|t(Pe+JN2rVVZEB0=Vgu{WP`Zarr z=B__?OG8W%c5(By>*GyaCTYUqgw1_@ zQBT39-uGcc0A8v^XEQ${H}8eoMWZT_Gw`Np01F+$cET>uK0K0ssn6%i4PH?1iso4{ zY#F>OV#POkjncj!=-e=Giq7G3AxF8qpU}jPm+xyXA!1XZWA&{CI3-4enHz2#dm&j# zKc=esDL0{^>*ca@6Q>T=7O3ey1cDvP^AhbY$CJ7kmK|KraGzUY46+k0*~c1)YoGC^ z4Rey$C{PD@#^FcdO_c)X244%%MEy}U7ZC=&6|Hr@y-jbbN$xS^B4HEW@Udwdx9{oM zVLf;P}A%XEkvVQ9i}C!XNwcUo|cns^dwKVadrX)fm`V8GclgQo5wkMK5;a`DM3e% z426}_av#|-sG7New-x7{I}%NW6LR~%#xeNI{$s_%V^D!_6${ z$#AE3hqv+TB6D~zr9>$*8fc5&xro&z;SNV2_P`LN3a8JsEcQ)q5v|+7!8|^Hn@{c7 zCabkOCLE6E@%7Vude=yITqu|ihhxjeg`!9o!ljDw2`inP!D#Wq^37#N2XOgN2l(M8 zu4ND^dBa}gmd$7PkG?_RTsn6YhxTMjP`e;4-=?NOlUNMf%T*Xsli1-yKj@2nxQ$qt zYbu;W1G3xG)|Ey$jhafXQ1XONLYO!;ijzFB1mg>p2wSdm+i$jI+_^2)6jO`dCbxr5 z3nWtFd{KXk0A@G+y*b{5);=@cIn;>_*f3mX_U(Bq$2>b(A4GHEzuifoxEd1runSYV(vrO`z4MV(ZNcD$K#NwGZdA|oIB=T~#4!~6>>RsL@ zhx!4(Mv;5jLct?8k`)v!ZsDl7?7FOAb3rG@bl7&2YsWM05r_^H1{p^Ak zObgX&(hA^|5a%!uBVL%bd36#V){BjhpU$^!zUM*>(U2?(G2NtsTJLp&j7z;1p^*$lH`zC(hENx^~u( z>VUrEVpltu!!e>A2(<%fgF9fh1874#fNHk`sCGMmYPSP+P1}^FQ*1*z5V!5vk30uc zyC!nHE~Z2#<>r`mR0@yEY|Cv7g{%vLf$f!6d(TkMBZP|+v@lf>@+cc8`&%Cph8SrJ zD!{jT7clqm%Beo&g0@$p^DP}Fl8963ahx~F2RQ7Pq%5Te*I)S%-k!jVA@=G;YuLVr zBY1TV-fYlt4MLv!WtvhQB-j3V7JE4N=2^n7ZL)-i3|;`ih~d^eG?_ePGgq5qnN7R~ z+v3&`dx4Iw3UnwFR0D^L(3W80f`bh~Mt|lR=b?uyje1R@AS?1I2vc7c(o2|yTSCQ; z^4E7Ki4tuT3l^F)!ZHaOKrWvJ)=lvM+bIvPVL>%i$t{YblA@ z^QOj<65AxF#5T$4iDgK<17m;bke|zRm2MZ+9WCP^!=u-6dk$E;DV+zIEH{NcI-MU0 zkg+{I2h^iK#VU@5#EZoCIX=kD1b3>Ze_7&;tt2Pzq z4)U!dHx+M>;YU0=d=xlV1BF%qRE_?QSktEHixv`;zU4|s+uwF%>kqcdyj1$mtHzyE zd6`+(%{GmTEwAtQN}*!G+n+ViELaUSf`e{ZJ?X8&_w8UdDBge<1Hx8f`Dq;a+geNq zcu502p2FoxL^*Yl&^GJq0J!&nIJthuSI=RW%VVf1{u40*6h9+|wHvW=AtPHswe%(e zw?;%g$XyHu>`T|M1g-l5#Po#$Vsc}E2<?yZ;Qn**X<5sSu&Vzp2jMixrn zo`q7=S}0$I7=^iG?;?1d#Fpqsl@DC2ZSiOsmJa*?wFSAqdRTU~ayon1JqSwYn=c%; z7LP@uylZIfoe-ZdRS*kl!EG4$IGxkDg8l^35og_V2-CPkB!`^!b|icY_M$k3(Z$i| z5obUlZbzhobR`_MYI4-dS8gdD#dwbtjH0 za972`>8+a|?hyM#N8;)Sd6<`&$hh>1AdiLFl0~~u z-7Q&!SD)?;`Mq5^#nu>?WBqX?9mWeU{6$`HbN8ijt@C&;%&$f)TE<{I%!3UbL!nz# zm~?E?%%UwW&Zg6BEY9NvIh+*CF3K+o@ckgU1cRF|z-+r{a-lJ$FGA?|?C@25S8Ut+ z$Ey;UIPz*wxs!>Tpd8PMX?_Ku(uV?tqO8kzEy3nH*QtxvYNtcp{k??@rPT-3aVVy6)y3phFVnW z{3a+3aNwJt;3*@fr1+EZ)8MNjOd3vxkfx#3hNaw2%=!4}3vw!S^1r--{F-V(P2(R+ zuSf|kaLoamlUz}Y4O12m%jzicO6E`mA1)?J2^EEpT}r(UOU>Ypb{Xaw!*E#J1@^a;IS_xJ8BdS+qfIBVS!=UI2X&n44oza?!oaJjqi; z4ueRGVM49z;72~;r8SsBec2GNU=cWb+m;^8Ym|mdp@}@RD9yzv+^=1FO$b2%V zuh;2&!l})xmaokjrcR!|4&A-#lp9>W+|#5mI4^MdN+fJ$zCm?FMqqY{g zE&wSw0yyZ50%Z#B7NnyF&~uQ|O=(dwSKaf6`BNJ(7fNwO-HP<;8oNcD`}o!oYC4EK zD5;b=&flVhIrEe=f8E?|UE$*FVZN$I zbn>$OdY-r2z&SWCbz-eK!`LQ{g5w=7s#dp;(Qr0m>x{n8;p{WNc6>K4U zlcCM{Q){-Mz5J)n9qEp}hPsQImTh5AU5zyCOUex1^s`E^a=5lLPWLQz4nC`k)J!#v z5N`jY2T|fGs}4Pb>L$p?YgwPy52Dv|z(&#clXwo}iC@J5p23K){LWTT>kXnNb?GOT z#ul=@y2iHlW>4B9PF+TQ&y`Ed4Eqo@^05Zp{SW}}O{mcX>N28OPtG8=gW8h1ZvnN? z?PZSC!+O>%?XMT?lMT90x=U1>CagT|QO%(~2I?~?T~EOGLN{!!Pq<0!eDznn)a zjzqo9-?k3yKBe^Nxph5qp}jzR!k$y)8pMd%b7J!+nL!;na|R(* zoMW^m^bDZZ;0V70v}rsGVhOPK?a{o_YQOG*mgoj`?FeLSRMKv<^4{j;Y@*IO-(jmx zYK=`(YdG&jSLN-LgQ}gjvsRoPT05SuhbaY=INA=B8C>rLRE0^=rW~O`sfU(hmS*?H za^;ejI*j_ADcgOmrP{6&`UAfp!qHhizjnIz2)A_wbJufg#;OhSyDSI%u#`W8eA}aL zX@Fpxf@3qWI!@Owdc>%wv9<+GX%E*HTf&Dhf?N%ju0mF6SFIGU>)l@QP|ipexl-HP zEu5ZRs2ptF{}K@@#>m&Y?~UKiZ;Y z+(c^XE!t8nixS?QpFO@Ws7rQakejyewH|lZ_{Efo{@eFr_&G=P3aquV){(4tdnMsa zW=nAvfYwtBb5+tS6E%Tty>RZcm#LvENJG1h*2)XW)vGelNYvL8NIwB78pY~C8vnS) zjN)$+fB0!XK>P*=DBBRT3IEg`ugan*3Ji>SuBU?9Tq@J;oM=Yry0mL>8pCQM70xyY=@`UUo9EmHOd^cSQXO$Q zgog2FikNGDZ3<^G|7my*Vye&(HWYr(654^&LPUENeyrSN+!IFp}JkVCmnG6D>piNlixs z4DZV@1|}_DYQl)C{v1Z1T1Je8v+%opsFQ8RDdeW*2WfV28Rhm*FFtg3P-dBuqh}kl z)Mf6nv|&JvuNNWd+3QcAm)(M8)4uBTY1^$?HuVdXY`1pVv^!5%a(jlaWNK6UFYSjR z^k?T1Y|kJX+LRL8gyGn+r}dn}?Csg2ZiZ~O4RRG+YCh??%hBPAXU}r3q1Md3Omi%a z-y-^;qLiuU>~cBS8tw&6hU=HsitTW(>C5GA)$husyermpWa12RJ>xp#Hy&-H${r7D zIGT~1KQyw~fBhQ+&L!CPbe93liBW){tZ1lX<+iJ`aI9lwn z;f8JLXmxWJoL0Ob_scssZA&H7nPbTu>fSl~&TtcV869$3JLuKftrO6?RikiZ8%ixu zT9OFg#LmBLBo6UQ=@-RBL{a5p#9jVHM4XOmS&2Ays#6@7^V{ks zb7QQHID3({7b^Egx+y+mt(fUWq3-folI?M?R24^eH}PR&r=*D9vnzYZGulht0E>U=MZgkZ4Zs3mdI7*PWBLqWEwB`zU8L6mD}rFbGU6O zq3S)%Qxm)Ybd|B!=Goy2*^Abfu)GilSJ0ncX+m8=DO?!@Cs(B!S+c|PHkv;83Ivjl zK7w-N%W9vZ5#;ctk*UZO6f`EaV=VN0v`R^q&$Y}|O!%}<<(7n#(O7paeRvIegeB_3 zYeV*tmUh^Z(OYIIx_%@%hMhr3b27-2#XkFIWU(P@WEr6im8IR`vkT--8@2SuEpR1mOCL(X3%o17}zJP}dvQo*t@AZ%Pmt8{b1Ns+LcBjnNo3 za?!{-pjpuhIQn>Qb)=E-4rEpB)8e*foH~7JrQfCAYfxKA+P>bxcDwrK;md);`*7E* zJ$zoZ|2(?H;lpU!)e(1k>IrFj?OMv(wegWsTCI%Mifx~0?Wk|XRD!|LCz&48e=4bM z588+S6y~6m$(T5)_v)?ed?2XYTK~=Gys93y5x5Co(ha8D%pR6@9oy9*7M9bLEzV%^7;SUm(apV%}Qw`SQc zxH!%p4u3mbq5P$p4K#c^Y1}Z3p2%{XAI_d4*hM22>l6 zu4-bE*x5O;Ig?bQb8aB4rujO6Z&GRCy}qQ}0W}!HnWPTkOk~gOH8NgcLfAAq2;XBC z=&JWig4L!=*K(4okz}aKXme)EQ^xBO!arQh^PXVLl=8#K-2231W{GMJolh56Ek4<0 zA-mk91)wgow4^1YHbBce^Xc4XbG5twZA^mnCz=+&I$2P%9hDZy76Nm4qU9kih-vw( z$E4ItT=U>oJX-LY%XJ=qTB4dyI4uXxGx4${(PK+xwizO;p1HPBYp_RlM@hY4*=zwZ zwv&!zRCjB0-DB>Q>rSLl`da8}Gx_c&jFz|NCHz_xR7=xBU{F{CuI0*Zu0c-6xK^WBhBZLvB@l?A93-C84cx&LCpE6?x4*l70tpCr`HqYrqsRH*?|?P2kMwT^Dq z!J2c-$VMw@u2lbB;@q%?!4_}qM5x+&Ba6=Wcc|*t!vCu^b4PVxGW?9SUr>IRyoG%g zw~nsgJ}4vqPeg5fJS%TO$Kzjj3(VTy<5+Q9#^!b*kkEfg4A|bhA+h-zx1MhRyw>G zUq|ND-`E{s8Li;t;m3-tomD4kPx236#kK=iyKj{I3hGd;0iVWj!=7B$vCcK%Q_Qi* zl?*d6ORTGdg9Jw(6l!pfpoOZ?%t|s<{Qz@g{C+=0B4Eqx|6jHPR&ka z{V?D+!}F`T%OeG`KeVeF2_^1ytR+PaqpeZ0OG;|SDoI;**1)NDsg_$KW9RmoT}yan z+RiPV8L~C6wsqKzB%57(fJCv0w3D; zi4H|xt#qR+uUg*0+FDg~o8IOLcBsp%MdOs>WYQ4DntZ*JM1v#EG|ok8Sy=Uhsv(HLEa7Oa)QQwsUglhE!>kvh7Ax z5$yS+Sj$C~59&Z#)m+;3SlgRDa^nAvK2S>H9b6{U=g^$5TWII&K`k-ry#jV-ZD$NKEqHB_~Bit5>u`)C7NZyeox z*t$Pjt;ZBDT}0jWCjhnPr?x6u6Qs80FY24!xDk5LsmUj-hn5kiJaVV8XspHaAa^=L z?R8pK&YeRqrq=0B81|dfV%2W8x+-B$SGgK$!URF^i|^*wzx3Me3$JskY78=Asq(X|AWIzCJQ1g0BXnbw zH~c1|Y*cTlM3qW4MFqnmHaR4;Pr-{(qk&1TE`Gyj&|3uJ@4 zcywo&31FGHH5$>aDJ4oMNJ+6Rl|RjlT89#{vba!5;2LjQbQBQ}b2S`E_&HJgFhp|1y%;f0May@@e zF)Dk{jftA{*gPdlTzx7Nq%xUIW0Wh`TVWy5*h;Havn`YOf^-oV(NfBpRK6|_N~;l) z^s}@xRAnlm+L>atepc5EHZUkzHnk>uE6;&2!#my9*&1a!tm5?dpvUR$og#=-tU>dRwimVwR!T=1# z<&cQtDYsNUT94mSxn5qWauY%lb0!Fjr;Nb) za@0tuYfF_IN|k*Wt(&o4A60x@QdtTxY}qq~76C&MKJrHfW#$Yc+Da8wFk^yD9F)fd zS!t@ADXkW%X#w@uTtVOyssJ{l}*mIQqZ)^_J zlY{2Tf&S|9VQDams12pPF{M=`Yy%OSdvN8Ww5OG_B4;YBNM%>5+|jHwTlOyF-zJBU zAPEJ9Ve^9^pQ_K7qI^r`efHn7lF^8~s%Q)K0r7VikBi2Yi>Y!kTSQ?mO_fWPTLswY zTOr(^<#GVnU#h%1RmZ8ABH<{E9;;As7|p`vko~zJT~?Lavp{K7URtCu6gmI(s9dp& zerwB~CFGJPx57u(Zsm?^!hS1gQ&cyurSdwc_w$eDb=ilekl{~m?Aa%q`*znjC-vT95RY{to!mEb`25z5#vow!H)sXD4iw8|p&BMR!& z#*69>aVnBxNj)8fw2J@4%SH*(6y;j>uHiqi62&O%-#0{=Rc%~4Smx9VgUvZf+K&ZY{>qMp)f+o))lj*qtK&`0u>r0-l?yFPCk zp^4a4j+^R@qKDe!E%j=i%^2v=JRtSMqsXWnl>bEgH%S&tv04Xr>c2^vG|VQq*;3S? zH20`cOD?K5h7S$ML>IHv7z{rI%nR);H6d!T7N>M3y+D$YEC!ZREz(%HicxkbM94k; zl(kHCmL4T-VoSrB9TODj--cX9Ly-~A4qXg6(-lq4VhnFVd4491jZtZtXZE>n4WMb$ zgG`!)So9dpVJd7cPY$w}lX51V59#sDP#{~N$14qFO9QFWR%IoiInnH(mG%rA5#tn zA%oGw0A&FgJ(-8Q=n60(Sc7u0oNh@174-zObQc;(sjFyaSg1s~G!maF@}M zXtq!=e5P|OsBNMQUa zfn;tBP9XUWgiQ*`N=q;eV3YV1NJ9vvs|%rx941N3gc;e}0NHFZ7&pvz(61+!MUh*H z)F{+na+TCXYI{Z{rD&Cuu}vV{i!26{|15pKdiiuc5QVv7lrQH}MNzT(&~DNkxfmEnoE_hD}<4duvi{xinwEjvMpz$5|u_L6=5d3c3VDny^kFPHYtr?wZ<^`*{Ky()e^=rnv)9?-F=`bpg7L7RaXSe>dDeP6b{D#~`pC*pjn0o7jf114Gz;0#**_B5dyjMR7NwE>bjW zbXjGz&_*UxNhrGQq3}0F$=G1%*)59VHeFt;%NupMQV`x@^?N+*S;F*A`=$Q&_8Z#!vKYxn6TItZHmbxnX7EXgKJ@ z)7es|;gp|`>&-1;HPC=!o|6V`H`MI0)JY#k9Hq{39v1vU%<_un=lqvC8{&+)E4#qW zMA8(?5+^JW#s`|78wR8nlm^WRra5I;uYQn*N3)~R ziEPCx7+DL=1tU|TE)K(j@@pte-U`_2(s~owdZAkkx&^~Zg=QX8xxayg99|_K@W=OqMckTVH2#^U&1giPG4p)NHzWQtSr#a`+)!c^ z3D;sXvC-m+M4w`W`V#(WN}(A5YUu|_n9`Ck5oTEvNNGgG3{x6p@R;b2frKi=EfR@X zZj(>+w@DT{UrT*$LVc~x=A9C&Im(K~jE_N@%k|A_zp7~5n(-~elDiB^^AV&`Xck;_ zlgm-0IgCb-%Hc&pIwv1uIhV@^nH-*JCNaZ|Tp_M2Rqm0-N|k#jX~D$e_cX=~IPM`% z(b0pVhfuWq5=jpl6u{oxLtq#fU)?*&6h`G9o+=JQQkI0>BgLs@7uB-!b`>d`4~)_@ zb~5Uv8ImoEU5%C%fy&*c#91J-Ii)FLBQwp)pBl3kVW=#3+Jq}vW46?yr?I*3^f?$0ZoZb_!;9FIE$!bTuK;NR%S0(jLtk@1^;+gTn{%wO>cN1hA zpO+cf8BZgEnIiH*zGGs(3&S^_ckGPyAwlAa2e-$)Q`r;Am^9nM;JDr$7wk5lG>h-M zBktMS-ybj8w6p)b!EN#4{w;%BV!q|l9k=4m;1v(Z**?&^rMt6j)7*iM+2?e?*%Ahgb7pkUm@}ulZN}`jx%?n13?|N4v0#n)OzNdO zcAh`o4W=$(G=+|eV}~BiALQd?{kvE2ZLnsAiHDxVooLLuLKxJoIBg}rR+AgY9UpFP z*Ha8$a0-B#4};9w%V1*gnB4X%48|MBU{PT@#2*qciDNc4&6J*ou-MKp$SqvBAYRyc z*=3h$jB2CpQM4MZdK?{a*kYG(L_JI6YmUF!yG~3Hj$|6mSFz*y`uy204RnKGJfill zB2JBVl?$8tkzsIHds|!U;P%#big`Pu%y{9&+s(&A;`T0$+D%64=kdkY{=piN=OU2C z;3oRQsr_5|T1nj3H+ODZmqz75a99}RQ8S;7ird?|G&DErH5W;}?zG-c<_WmuZr1o= zcZc|QT@(Df&O`mXu8HPu4CyV|Ij}8W)Y{(G9vj-Eke8A1nYs{$F_B|`qP5%4-@fD0?Xykbu^6OPgZidW z+_!vQ2Q3(PXc#nGZO7C2xaq*oMN5z2mnS$CV%JO1;dJ60N+v}N1a8ML9< zs|-x9+-V_tpX47lA|D^^2Fb_ATt-+Y3fR4<}4JZ|}4LvU57ED$X83 zd8(mMPg@d$r!r{mQ+gYCq!OSe3`MX}G$00SK8>Z}MV{Bn! zgxkBjJCFIYVNnTnTW21A#ZXmeY^V2CbMNKufm*_MY6;(~C49e@@Pk^y4{HfOswMom zmhh8W!cS`nKdUADwwCaFLy!t=1EKq;8UDdgX{_nqCoT9BH#Oz(=h|pkGELmq2cEZ4 zXbQlb1lM7Lefz)1brvqz%QveJ|@DNW_OI6!H1l$HZ8>j$J1J(fQw!45vEbC3cGGIS29e4|{75F7E z33vst9{3h;0&q9*bl^cC1zZix1MUNk0A2%Z0=^F*b?`jkG~kQCEZ`l$PT==I9(WdT zD)1@bSl~^-(}14?6M>fjPXWFG91nZ|xE%N!P!Bv4SO|O^coOh>;2hw`zZ?@O5mx$cYu?C4*|~r9s)G|-2?Oh_X1OZ zR|EaP_kb4Qxxi}R3qU(?7jPl)J0J(#04xDM2^<650Sp2^0}cgV3Y-ai9q0z$59|j1 z2gm@|0too&pbXp! zoDF;%I1#u9xDxmWz%nRPkzDY6{O9BM;rHQx3I9v@9rzvi@5Fy6{x1Ao_T_hkqRYoAKX_e?I>C_`ksa1^!|9hvB~*|K<2+;h%;7P5f`- zKNk_s#{V$> zRrpuoAH+Y1&oAPGQvgm%1sj28U=`2?TmWQ&#lSS+JYWKF1~3=645$O<14jZ|fU&?D zpcA+VNE5?epcnWEFcr8R*bMvth@$#TU7FRwbe$HQ>7KyqG8ye5YX6rtFXLGAGcI4_ z8fskQqc~QKAJLifDo^0j^aLG)v6fP4%$Tvbak%k`J0x)vYTThUZsMc3!yd&=dV*0*v12hXVSJ0Q(*UcLFZ}t^qCqG;Yvra2{Y*2P&uRi*MU_p!t?n zpKrVKg5PGZUwr?xH=g&?2`@S0fw}Ly?5}m#%>U?-uif&)u`gKj<<56q^vA}VmhGSZ zmaV^>^osT0I^pi8KbX3D-hD^BX4Cge&pYjlv)-}u_xWd?`l(~z^t7K(eA!dJar_4^ z|6Bbt7k>OnuRrI<<6pGyt6lH8i+Mw zJa_dM+V8sXcexvueDat(27h+wOV9jz_xpGMPv+VMA3N%G13wz~!nI$S{qBqZRJeKh zXIkI7?N{Yn&;Is__gwjp^xoc&Ouc>c55_#d?@Jx;-1Ucs8<&3ixHq5wi^E=i);FL0 z!7KhAUAO2HN5A3RpB(by)4w+7y_f!_={YMuH}mZ~eslP%HhlNw4_|e#jy6}M*E)e# zdI_4k8|`fh9|%91x--2gy03_u=(FsYE{$IYsNHFP)d(~L6Mz_K15N=p0u^8%pum-X zCISV;t-{v`Nl|DDZE$Y-61)XpvtUC%>0{3@3r`N@pNY?7@V7?Pg?a`J02bx3{+(#d}>^nHNI*G7;>ab(4m zjq{aHAEZ|wB&QEb#|LidM=S_eSn-PoHBeh^-IM})rq!*gzb$)0zSuai z*w{Rw)HZhG_ij2O~wyRb6 z8v*I30^A1d0}cQ~t`gkBgFAWfc^>?BBM&6I5e84~R`JLQZw6FXgi9Z)AL5tpCmT#!oHq_W_Da{K|`*cs2^b36F)~#DAL*oaoL8uOv6ZPf2j`9{^nZDtkHcsGiFS zr^y&6{#yW*t>jTz%ZcYQgIQdrtM15&|2%_Pd|HBw|4xrCJ~{c<8_eR1 z6I}ei1y%uyU*#_+e(76IxXN5@Te#|SqZNAKMFYg%MQtjf0My1enf(c|Fs@nc0o@3R~yXY^AcSAKLnio(y5&I?JBUv zCmO2N{|msK{(BR;_%HYH;}cx*e;JT26!#cFPW+nl%LzZ$U^f1DCAjE*dw-TkKh@*^ zBe;`aHd9Vv)z`=g9}nzHaJ(V?cjyKuo=Xg7@va1y{QCi?f4jP2dCu|ppXBks#iRT7 zJ-J-Qi~aWDR7d-N3vF8&9Bjew~;>WpyWm(Jvbw*apG?*(`A`1b!o=yKw3H<-=u zec(<%etqaqc*H-&TQkp6x9zm0o2g>5yMjsF7)F1nxJn?3q+ zkN6$>Z`R`UQZT z@HT^4{of0&xD>x{|2J|kC;nInPV&FnV3wYshNt4uPf2j`e-Ci+*C%wvd4`8iNO18# z7w7{dkE`e6zYLHQKG$GYo-csA`qPro#dDsAPfKvcf0suWpPc;b4QBDh2`>KM0ZxA3 z{!8C-3Og2X_U8xSPJY+!#B&iKC%n^OR{l?dJNbQkxFz8c|B)X59Uk46=NgYb-{b!o zxRYOT$;n@DFpE!0aOL-_^bq1{2v3H{(XD6$>Tr5 zptQ+oSvK{aGIURFD5p;7)$u{;%`+$Aj+!l&+l9e}}y_^h-RvE5Rkt zX8@;v-yWXh@juDqf2&9L?a4De`a+NYSKv;5r6(sjpK36RPfl<>zZG!$-|X=`!^2NX zaPfZ|aQgS{|8|dmhR1)8M=yH#vpsx;$NvxTjew~;p?x6vr87Cr9XTP>=ul;I97o z_J6m>-|g{#30%66exwIE$vkK z2Z%@b;~oEhnlMg&-~JCkms6Za8O)Z~n-g4gzdc^-(HD69zW{ggD~z1{vkhkPa)K+r zF9%%xS2@Uu|4M^d{KNzo|2F|w-+cSO&Es$N_&?~;{rukS(U*Jte+PH+r#&8DwlUCM z{a1a~Gu4NwfSlytWiX3(B)H`N1mN`V+yBiTzl%rwZ}8}858vzIy$LSDK+*SFI>{zJg`0ZLcS>EG#D^h?bX%QGjzCC}#or+?r6pX>1-?eV|eqx<^2&Z95# z_2p1%rk`rqvFT;<^>C%E{(3po9|a#Z{~5+30*6I}cs_UKI> z{u~cqnc(6-nBY&u(>Ns!IDN@-Gy^(GG6ARq#sa*dh_gTZb_FN_{LqH6NhYWV#sfuQ zGQc@aY;A`5K!&xOjQJbKl^@G-b|gbu8O9_}GIr{}`XG=auuk4L0PyLss9lue=d*8 zr|h&q^;MvLN1*%*6p=vs6{z16s67ZYE)~c=85@;s8u0@?-VW8!pyLKQP@p3N+WK!! zTub6gi5rtR?d$tad;Gq0`~A*$`~J?a+5R7KTXP&>#OPa4(>}%RN9yBq0KXL=vLt5XFZmH>5y9 zCedLGv;n688!3~a%}!3+1pf_=A2W~*vRWCQ8yOC>t}%ixzQw>96>OQ5pSX#2onY(e zt2S`eH?Q?ArInm3*kMsK#XFMZY9b=?cl#XI~x z@Y0Q5(i?ELwBUtbXL6ypVCs#Ha&wTR8<@kNK6x{P)q8z!T5cA#A~YpybFZ&tZuKN| zsmwsrdK0PK<(y;)IeDvZEQ?z_7mzRM;apa&E`nzyE#`b^O!-N7{a{NdY1JORQ&lCi z7+ha!+XAjPbm}|R_Fdw-o0ZJ?jwNM3ZCeBNkte<5b0=d))e`HDH+fr`$fa_Tw5|^7 zPPN3!Doy%%oNLNt6=C%q1bzMPB2Oz1Ulu2svKSt&wnQGOJQmWUglI|cF;$-2mku5| lj@ig=-)gG2WB;G~57R)K#-*GCaEo65VdDLtIR9D={4b2bp5Oog diff --git a/AxibugEmuOnline.Client/Assets/Plugins/Core.dll.meta b/AxibugEmuOnline.Client/Assets/Plugins/Core.dll.meta deleted file mode 100644 index 698e108d..00000000 --- a/AxibugEmuOnline.Client/Assets/Plugins/Core.dll.meta +++ /dev/null @@ -1,33 +0,0 @@ -fileFormatVersion: 2 -guid: 986b9dc19d701a94abe3e4d4b80a70ae -PluginImporter: - externalObjects: {} - serializedVersion: 2 - iconMap: {} - executionOrder: {} - defineConstraints: [] - isPreloaded: 0 - isOverridable: 0 - isExplicitlyReferenced: 0 - validateReferences: 1 - platformData: - - first: - Any: - second: - enabled: 1 - settings: {} - - first: - Editor: Editor - second: - enabled: 0 - settings: - DefaultValueInitialized: true - - first: - Windows Store Apps: WindowsStoreApps - second: - enabled: 0 - settings: - CPU: AnyCPU - userData: - assetBundleName: - assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity b/AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity new file mode 100644 index 00000000..66bdcabd --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity @@ -0,0 +1,316 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!29 &1 +OcclusionCullingSettings: + m_ObjectHideFlags: 0 + serializedVersion: 2 + m_OcclusionBakeSettings: + smallestOccluder: 5 + smallestHole: 0.25 + backfaceThreshold: 100 + m_SceneGUID: 00000000000000000000000000000000 + m_OcclusionCullingData: {fileID: 0} +--- !u!104 &2 +RenderSettings: + m_ObjectHideFlags: 0 + serializedVersion: 9 + m_Fog: 0 + m_FogColor: {r: 0.5, g: 0.5, b: 0.5, a: 1} + m_FogMode: 3 + m_FogDensity: 0.01 + m_LinearFogStart: 0 + m_LinearFogEnd: 300 + m_AmbientSkyColor: {r: 0.212, g: 0.227, b: 0.259, a: 1} + m_AmbientEquatorColor: {r: 0.114, g: 0.125, b: 0.133, a: 1} + m_AmbientGroundColor: {r: 0.047, g: 0.043, b: 0.035, a: 1} + m_AmbientIntensity: 1 + m_AmbientMode: 0 + m_SubtractiveShadowColor: {r: 0.42, g: 0.478, b: 0.627, a: 1} + m_SkyboxMaterial: {fileID: 10304, guid: 0000000000000000f000000000000000, type: 0} + m_HaloStrength: 0.5 + m_FlareStrength: 1 + m_FlareFadeSpeed: 3 + m_HaloTexture: {fileID: 0} + m_SpotCookie: {fileID: 10001, guid: 0000000000000000e000000000000000, type: 0} + m_DefaultReflectionMode: 0 + m_DefaultReflectionResolution: 128 + m_ReflectionBounces: 1 + m_ReflectionIntensity: 1 + m_CustomReflection: {fileID: 0} + m_Sun: {fileID: 0} + m_UseRadianceAmbientProbe: 0 +--- !u!157 &3 +LightmapSettings: + m_ObjectHideFlags: 0 + serializedVersion: 12 + m_GIWorkflowMode: 1 + m_GISettings: + serializedVersion: 2 + m_BounceScale: 1 + m_IndirectOutputScale: 1 + m_AlbedoBoost: 1 + m_EnvironmentLightingMode: 0 + m_EnableBakedLightmaps: 1 + m_EnableRealtimeLightmaps: 0 + m_LightmapEditorSettings: + serializedVersion: 12 + m_Resolution: 2 + m_BakeResolution: 40 + m_AtlasSize: 1024 + m_AO: 0 + m_AOMaxDistance: 1 + m_CompAOExponent: 1 + m_CompAOExponentDirect: 0 + m_ExtractAmbientOcclusion: 0 + m_Padding: 2 + m_LightmapParameters: {fileID: 0} + m_LightmapsBakeMode: 1 + m_TextureCompression: 1 + m_FinalGather: 0 + m_FinalGatherFiltering: 1 + m_FinalGatherRayCount: 256 + m_ReflectionCompression: 2 + m_MixedBakeMode: 2 + m_BakeBackend: 1 + m_PVRSampling: 1 + m_PVRDirectSampleCount: 32 + m_PVRSampleCount: 512 + m_PVRBounces: 2 + m_PVREnvironmentSampleCount: 256 + m_PVREnvironmentReferencePointCount: 2048 + m_PVRFilteringMode: 1 + m_PVRDenoiserTypeDirect: 1 + m_PVRDenoiserTypeIndirect: 1 + m_PVRDenoiserTypeAO: 1 + m_PVRFilterTypeDirect: 0 + m_PVRFilterTypeIndirect: 0 + m_PVRFilterTypeAO: 0 + m_PVREnvironmentMIS: 1 + m_PVRCulling: 1 + m_PVRFilteringGaussRadiusDirect: 1 + m_PVRFilteringGaussRadiusIndirect: 5 + m_PVRFilteringGaussRadiusAO: 2 + m_PVRFilteringAtrousPositionSigmaDirect: 0.5 + m_PVRFilteringAtrousPositionSigmaIndirect: 2 + m_PVRFilteringAtrousPositionSigmaAO: 1 + m_ExportTrainingData: 0 + m_TrainingDataDestination: TrainingData + m_LightProbeSampleCountMultiplier: 4 + m_LightingDataAsset: {fileID: 0} + m_LightingSettings: {fileID: 0} +--- !u!196 &4 +NavMeshSettings: + serializedVersion: 2 + m_ObjectHideFlags: 0 + m_BuildSettings: + serializedVersion: 3 + agentTypeID: 0 + agentRadius: 0.5 + agentHeight: 2 + agentSlope: 45 + agentClimb: 0.4 + ledgeDropHeight: 0 + maxJumpAcrossDistance: 0 + minRegionArea: 2 + manualCellSize: 0 + cellSize: 0.16666667 + manualTileSize: 0 + tileSize: 256 + buildHeightMesh: 0 + maxJobWorkers: 0 + preserveTilesOutsideBounds: 0 + debug: + m_Flags: 0 + m_NavMeshData: {fileID: 0} +--- !u!1 &708549044 +GameObject: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + serializedVersion: 6 + m_Component: + - component: {fileID: 708549046} + - component: {fileID: 708549045} + m_Layer: 0 + m_Name: Directional Light + m_TagString: Untagged + m_Icon: {fileID: 0} + m_NavMeshLayer: 0 + m_StaticEditorFlags: 0 + m_IsActive: 1 +--- !u!108 &708549045 +Light: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 708549044} + m_Enabled: 1 + serializedVersion: 10 + m_Type: 1 + m_Shape: 0 + m_Color: {r: 1, g: 0.95686275, b: 0.8392157, a: 1} + m_Intensity: 1 + m_Range: 10 + m_SpotAngle: 30 + m_InnerSpotAngle: 21.80208 + m_CookieSize: 10 + m_Shadows: + m_Type: 2 + m_Resolution: -1 + m_CustomResolution: -1 + m_Strength: 1 + m_Bias: 0.05 + m_NormalBias: 0.4 + m_NearPlane: 0.2 + m_CullingMatrixOverride: + e00: 1 + e01: 0 + e02: 0 + e03: 0 + e10: 0 + e11: 1 + e12: 0 + e13: 0 + e20: 0 + e21: 0 + e22: 1 + e23: 0 + e30: 0 + e31: 0 + e32: 0 + e33: 1 + m_UseCullingMatrixOverride: 0 + m_Cookie: {fileID: 0} + m_DrawHalo: 0 + m_Flare: {fileID: 0} + m_RenderMode: 0 + m_CullingMask: + serializedVersion: 2 + m_Bits: 4294967295 + m_RenderingLayerMask: 1 + m_Lightmapping: 4 + m_LightShadowCasterMode: 0 + m_AreaSize: {x: 1, y: 1} + m_BounceIntensity: 1 + m_ColorTemperature: 6570 + m_UseColorTemperature: 0 + m_BoundingSphereOverride: {x: 0, y: 0, z: 0, w: 0} + m_UseBoundingSphereOverride: 0 + m_UseViewFrustumForShadowCasterCull: 1 + m_ShadowRadius: 0 + m_ShadowAngle: 0 +--- !u!4 &708549046 +Transform: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 708549044} + serializedVersion: 2 + m_LocalRotation: {x: 0.40821788, y: -0.23456968, z: 0.10938163, w: 0.8754261} + m_LocalPosition: {x: 0, y: 3, z: 0} + m_LocalScale: {x: 1, y: 1, z: 1} + m_ConstrainProportionsScale: 0 + m_Children: [] + m_Father: {fileID: 0} + m_LocalEulerAnglesHint: {x: 50, y: -30, z: 0} +--- !u!1 &1232273651 +GameObject: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + serializedVersion: 6 + m_Component: + - component: {fileID: 1232273654} + - component: {fileID: 1232273653} + - component: {fileID: 1232273652} + m_Layer: 0 + m_Name: Main Camera + m_TagString: MainCamera + m_Icon: {fileID: 0} + m_NavMeshLayer: 0 + m_StaticEditorFlags: 0 + m_IsActive: 1 +--- !u!81 &1232273652 +AudioListener: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 1232273651} + m_Enabled: 1 +--- !u!20 &1232273653 +Camera: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 1232273651} + m_Enabled: 1 + serializedVersion: 2 + m_ClearFlags: 1 + m_BackGroundColor: {r: 0.19215687, g: 0.3019608, b: 0.4745098, a: 0} + m_projectionMatrixMode: 1 + m_GateFitMode: 2 + m_FOVAxisMode: 0 + m_Iso: 200 + m_ShutterSpeed: 0.005 + m_Aperture: 16 + m_FocusDistance: 10 + m_FocalLength: 50 + m_BladeCount: 5 + m_Curvature: {x: 2, y: 11} + m_BarrelClipping: 0.25 + m_Anamorphism: 0 + m_SensorSize: {x: 36, y: 24} + m_LensShift: {x: 0, y: 0} + m_NormalizedViewPortRect: + serializedVersion: 2 + x: 0 + y: 0 + width: 1 + height: 1 + near clip plane: 0.3 + far clip plane: 1000 + field of view: 60 + orthographic: 0 + orthographic size: 5 + m_Depth: -1 + m_CullingMask: + serializedVersion: 2 + m_Bits: 4294967295 + m_RenderingPath: -1 + m_TargetTexture: {fileID: 0} + m_TargetDisplay: 0 + m_TargetEye: 3 + m_HDR: 1 + m_AllowMSAA: 1 + m_AllowDynamicResolution: 0 + m_ForceIntoRT: 0 + m_OcclusionCulling: 1 + m_StereoConvergence: 10 + m_StereoSeparation: 0.022 +--- !u!4 &1232273654 +Transform: + m_ObjectHideFlags: 0 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 1232273651} + serializedVersion: 2 + m_LocalRotation: {x: 0, y: 0, z: 0, w: 1} + m_LocalPosition: {x: 0, y: 1, z: -10} + m_LocalScale: {x: 1, y: 1, z: 1} + m_ConstrainProportionsScale: 0 + m_Children: [] + m_Father: {fileID: 0} + m_LocalEulerAnglesHint: {x: 0, y: 0, z: 0} +--- !u!1660057539 &9223372036854775807 +SceneRoots: + m_ObjectHideFlags: 0 + m_Roots: + - {fileID: 1232273654} + - {fileID: 708549046} diff --git a/AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity.meta b/AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity.meta new file mode 100644 index 00000000..17bcd88f --- /dev/null +++ b/AxibugEmuOnline.Client/Assets/Scene/EmuTest.unity.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: 3dba71db67877594a9b5b77059205ec8 +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/AxibugEmuOnline.Client/Assets/Script/AxibugEmuOnline.Client.asmdef b/AxibugEmuOnline.Client/Assets/Script/AxibugEmuOnline.Client.asmdef index 7bdf3cc8..72c8ac05 100644 --- a/AxibugEmuOnline.Client/Assets/Script/AxibugEmuOnline.Client.asmdef +++ b/AxibugEmuOnline.Client/Assets/Script/AxibugEmuOnline.Client.asmdef @@ -1,6 +1,9 @@ { "name": "AxibugEmuOnline.Client", - "references": [], + "rootNamespace": "", + "references": [ + "GUID:0c194730510bd1b4fad0398ccfe4235b" + ], "includePlatforms": [], "excludePlatforms": [], "allowUnsafeCode": true, diff --git a/AxibugEmuOnline.Client/AxibugEmuOnline.Client.sln b/AxibugEmuOnline.Client/AxibugEmuOnline.Client.sln index 8141836e..f2b666da 100644 --- a/AxibugEmuOnline.Client/AxibugEmuOnline.Client.sln +++ b/AxibugEmuOnline.Client/AxibugEmuOnline.Client.sln @@ -1,20 +1,278 @@ - -Microsoft Visual Studio Solution File, Format Version 12.00 -# Visual Studio 15 -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AxibugEmuOnline.Client", "AxibugEmuOnline.Client.csproj", "{59C211A4-B928-7505-AF2E-6C604DAB3622}" -EndProject -Global - GlobalSection(SolutionConfigurationPlatforms) = preSolution - Debug|Any CPU = Debug|Any CPU - Release|Any CPU = Release|Any CPU - EndGlobalSection - GlobalSection(ProjectConfigurationPlatforms) = postSolution - {59C211A4-B928-7505-AF2E-6C604DAB3622}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {59C211A4-B928-7505-AF2E-6C604DAB3622}.Debug|Any CPU.Build.0 = Debug|Any CPU - {59C211A4-B928-7505-AF2E-6C604DAB3622}.Release|Any CPU.ActiveCfg = Release|Any CPU - {59C211A4-B928-7505-AF2E-6C604DAB3622}.Release|Any CPU.Build.0 = Release|Any CPU - EndGlobalSection - GlobalSection(SolutionProperties) = preSolution - HideSolutionNode = FALSE - EndGlobalSection -EndGlobal + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio 15 +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Timeline", "Unity.Timeline.csproj", "{B0DF349E-3E77-3469-A306-9F1B5D72D605}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Timeline.Editor", "Unity.Timeline.Editor.csproj", "{9D62BFE8-89CA-74D4-0926-0EF38577DFB2}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.TextMeshPro.Editor", "Unity.TextMeshPro.Editor.csproj", "{CA8BC370-E9DD-5EB9-20CD-BBC42D4DAFE9}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Internal", "Unity.Services.Core.Internal.csproj", "{1DE06E8A-0589-760E-E9EA-87943C9B27A2}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.Flow", "Unity.VisualScripting.Flow.csproj", "{50133F68-097B-E953-ED70-07F18E27CE56}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UnityEditor.UI", "UnityEditor.UI.csproj", "{EBE93344-EA1D-224C-E130-32FEBD7EAB84}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "MyNes.Core", "MyNes.Core.csproj", "{D958CC78-341F-B644-4447-8EE8CF037D34}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.PlasticSCM.Editor", "Unity.PlasticSCM.Editor.csproj", "{01C7700A-E278-B0E2-E414-4B4EA427F2CE}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.CollabProxy.Editor", "Unity.CollabProxy.Editor.csproj", "{5CB06AAE-940C-0F2C-D02D-5AB39C9E0AA8}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.Core", "Unity.VisualScripting.Core.csproj", "{32FF793E-2C1B-8992-FD16-735748086B4C}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.Core.Editor", "Unity.VisualScripting.Core.Editor.csproj", "{60BCA52D-8C02-4F9B-1F2B-5B166D854B69}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.Flow.Editor", "Unity.VisualScripting.Flow.Editor.csproj", "{8EDC679A-229C-DDC1-7C08-C3D3AEA73F90}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UnityEditor.TestRunner", "UnityEditor.TestRunner.csproj", "{F32BAC49-F2F1-CE2D-0167-1F424878F925}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.SettingsProvider.Editor", "Unity.VisualScripting.SettingsProvider.Editor.csproj", "{DFF0A2FD-8EAD-ED52-E0DC-D486AF73C847}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UnityEngine.TestRunner", "UnityEngine.TestRunner.csproj", "{0555AF0D-ECDE-BFBD-BBB5-8DC1A5197960}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Environments.Editor", "Unity.Services.Core.Environments.Editor.csproj", "{06C2DDE3-F045-CEE5-3F03-831228DDA3A9}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Editor", "Unity.Services.Core.Editor.csproj", "{7A646232-1EC4-2599-A1CD-7CF8DD15966E}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core", "Unity.Services.Core.csproj", "{49DBA999-DF45-268E-63B2-FA94C0D1B87C}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UnityEngine.UI", "UnityEngine.UI.csproj", "{2053DD51-3AE8-A5EB-65D6-04B16AA3CE66}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualStudio.Editor", "Unity.VisualStudio.Editor.csproj", "{368FC675-4200-A025-912A-59AFDAA29018}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.TestTools.CodeCoverage.Editor.OpenCover.Model", "Unity.TestTools.CodeCoverage.Editor.OpenCover.Model.csproj", "{DA00146E-B874-B45F-06C9-500B0401220C}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Rider.Editor", "Unity.Rider.Editor.csproj", "{6E165AB7-E216-7010-DFED-1C1BC39B970D}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Environments.Internal", "Unity.Services.Core.Environments.Internal.csproj", "{D172DD08-1354-1BA0-9525-0E869B5020BD}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Configuration", "Unity.Services.Core.Configuration.csproj", "{8F9BD7C0-7ACF-A18C-5A88-229D8B094F3F}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.State.Editor", "Unity.VisualScripting.State.Editor.csproj", "{F23A84DD-FF43-F524-83B9-D17532BC1DE8}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.TestTools.CodeCoverage.Editor", "Unity.TestTools.CodeCoverage.Editor.csproj", "{649C131C-6FE6-E748-C932-651E89DFADFE}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Configuration.Editor", "Unity.Services.Core.Configuration.Editor.csproj", "{CC93A1BB-FB4C-9BE7-0832-CF4AF1B69A92}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.State", "Unity.VisualScripting.State.csproj", "{E912C16D-738B-1DB1-3456-24751CAB3849}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.TextMeshPro", "Unity.TextMeshPro.csproj", "{FFF816B4-207D-AC0E-3B30-C9E0CA06D257}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Networking", "Unity.Services.Core.Networking.csproj", "{BD55EBED-7E44-C621-7FF4-FD7709A438F6}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Performance.Profile-Analyzer.Editor", "Unity.Performance.Profile-Analyzer.Editor.csproj", "{F57AB79F-532A-EBC2-871B-64C9FD27FEA9}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Scheduler", "Unity.Services.Core.Scheduler.csproj", "{2598AAB1-DA3E-A2AE-B060-D1BAF7BBE4BC}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VSCode.Editor", "Unity.VSCode.Editor.csproj", "{5A8F066D-8FFA-AFC1-1E8E-0DE9AEAE676F}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.TestTools.CodeCoverage.Editor.OpenCover.Mono.Reflection", "Unity.TestTools.CodeCoverage.Editor.OpenCover.Mono.Reflection.csproj", "{469C328E-E07E-04A6-A566-E2ABA6732CBE}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Settings.Editor", "Unity.Settings.Editor.csproj", "{541572F4-E051-F4F8-9B39-87BFA550E13A}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AxibugEmuOnline.Client", "AxibugEmuOnline.Client.csproj", "{59C211A4-B928-7505-AF2E-6C604DAB3622}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Registration", "Unity.Services.Core.Registration.csproj", "{A5C76622-FE9B-BD41-A430-5208CCE95FDA}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Telemetry", "Unity.Services.Core.Telemetry.csproj", "{7F93B805-8A3C-D81B-FEE7-A4CE542AB2D2}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Analytics", "Unity.Services.Core.Analytics.csproj", "{83045A6C-05F6-CC92-820D-281C04862F7C}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Threading", "Unity.Services.Core.Threading.csproj", "{458B6228-8612-4186-4E92-36797B16DFA3}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.EditorCoroutines.Editor", "Unity.EditorCoroutines.Editor.csproj", "{7F94681B-B8E0-DF52-D282-9411B6632E77}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Device", "Unity.Services.Core.Device.csproj", "{98604E75-FAB4-80FC-A539-DE0A0CFD0687}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.Services.Core.Environments", "Unity.Services.Core.Environments.csproj", "{8576736C-D7B9-E3D7-203C-46D059601F81}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Unity.VisualScripting.Shared.Editor", "Unity.VisualScripting.Shared.Editor.csproj", "{26BB76BB-57D1-3F9B-44BC-3ECB5B9EEF49}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Release|Any CPU = Release|Any CPU + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {B0DF349E-3E77-3469-A306-9F1B5D72D605}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {B0DF349E-3E77-3469-A306-9F1B5D72D605}.Debug|Any CPU.Build.0 = Debug|Any CPU + {B0DF349E-3E77-3469-A306-9F1B5D72D605}.Release|Any CPU.ActiveCfg = Release|Any CPU + {B0DF349E-3E77-3469-A306-9F1B5D72D605}.Release|Any CPU.Build.0 = Release|Any CPU + {9D62BFE8-89CA-74D4-0926-0EF38577DFB2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {9D62BFE8-89CA-74D4-0926-0EF38577DFB2}.Debug|Any CPU.Build.0 = Debug|Any CPU + {9D62BFE8-89CA-74D4-0926-0EF38577DFB2}.Release|Any CPU.ActiveCfg = Release|Any CPU + {9D62BFE8-89CA-74D4-0926-0EF38577DFB2}.Release|Any CPU.Build.0 = Release|Any CPU + {CA8BC370-E9DD-5EB9-20CD-BBC42D4DAFE9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {CA8BC370-E9DD-5EB9-20CD-BBC42D4DAFE9}.Debug|Any CPU.Build.0 = Debug|Any CPU + {CA8BC370-E9DD-5EB9-20CD-BBC42D4DAFE9}.Release|Any CPU.ActiveCfg = Release|Any CPU + {CA8BC370-E9DD-5EB9-20CD-BBC42D4DAFE9}.Release|Any CPU.Build.0 = Release|Any CPU + {1DE06E8A-0589-760E-E9EA-87943C9B27A2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {1DE06E8A-0589-760E-E9EA-87943C9B27A2}.Debug|Any CPU.Build.0 = Debug|Any CPU + {1DE06E8A-0589-760E-E9EA-87943C9B27A2}.Release|Any CPU.ActiveCfg = Release|Any CPU + {1DE06E8A-0589-760E-E9EA-87943C9B27A2}.Release|Any CPU.Build.0 = Release|Any CPU + {50133F68-097B-E953-ED70-07F18E27CE56}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {50133F68-097B-E953-ED70-07F18E27CE56}.Debug|Any CPU.Build.0 = Debug|Any CPU + {50133F68-097B-E953-ED70-07F18E27CE56}.Release|Any CPU.ActiveCfg = Release|Any CPU + {50133F68-097B-E953-ED70-07F18E27CE56}.Release|Any CPU.Build.0 = Release|Any CPU + {EBE93344-EA1D-224C-E130-32FEBD7EAB84}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {EBE93344-EA1D-224C-E130-32FEBD7EAB84}.Debug|Any CPU.Build.0 = Debug|Any CPU + {EBE93344-EA1D-224C-E130-32FEBD7EAB84}.Release|Any CPU.ActiveCfg = Release|Any CPU + {EBE93344-EA1D-224C-E130-32FEBD7EAB84}.Release|Any CPU.Build.0 = Release|Any CPU + {D958CC78-341F-B644-4447-8EE8CF037D34}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {D958CC78-341F-B644-4447-8EE8CF037D34}.Debug|Any CPU.Build.0 = Debug|Any CPU + {D958CC78-341F-B644-4447-8EE8CF037D34}.Release|Any CPU.ActiveCfg = Release|Any CPU + {D958CC78-341F-B644-4447-8EE8CF037D34}.Release|Any CPU.Build.0 = Release|Any CPU + {01C7700A-E278-B0E2-E414-4B4EA427F2CE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {01C7700A-E278-B0E2-E414-4B4EA427F2CE}.Debug|Any CPU.Build.0 = Debug|Any CPU + {01C7700A-E278-B0E2-E414-4B4EA427F2CE}.Release|Any CPU.ActiveCfg = Release|Any CPU + {01C7700A-E278-B0E2-E414-4B4EA427F2CE}.Release|Any CPU.Build.0 = Release|Any CPU + {5CB06AAE-940C-0F2C-D02D-5AB39C9E0AA8}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {5CB06AAE-940C-0F2C-D02D-5AB39C9E0AA8}.Debug|Any CPU.Build.0 = Debug|Any CPU + {5CB06AAE-940C-0F2C-D02D-5AB39C9E0AA8}.Release|Any CPU.ActiveCfg = Release|Any CPU + {5CB06AAE-940C-0F2C-D02D-5AB39C9E0AA8}.Release|Any CPU.Build.0 = Release|Any CPU + {32FF793E-2C1B-8992-FD16-735748086B4C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {32FF793E-2C1B-8992-FD16-735748086B4C}.Debug|Any CPU.Build.0 = Debug|Any CPU + {32FF793E-2C1B-8992-FD16-735748086B4C}.Release|Any CPU.ActiveCfg = Release|Any CPU + {32FF793E-2C1B-8992-FD16-735748086B4C}.Release|Any CPU.Build.0 = Release|Any CPU + {60BCA52D-8C02-4F9B-1F2B-5B166D854B69}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {60BCA52D-8C02-4F9B-1F2B-5B166D854B69}.Debug|Any CPU.Build.0 = Debug|Any CPU + {60BCA52D-8C02-4F9B-1F2B-5B166D854B69}.Release|Any CPU.ActiveCfg = Release|Any CPU + {60BCA52D-8C02-4F9B-1F2B-5B166D854B69}.Release|Any CPU.Build.0 = Release|Any CPU + {8EDC679A-229C-DDC1-7C08-C3D3AEA73F90}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {8EDC679A-229C-DDC1-7C08-C3D3AEA73F90}.Debug|Any CPU.Build.0 = Debug|Any CPU + {8EDC679A-229C-DDC1-7C08-C3D3AEA73F90}.Release|Any CPU.ActiveCfg = Release|Any CPU + {8EDC679A-229C-DDC1-7C08-C3D3AEA73F90}.Release|Any CPU.Build.0 = Release|Any CPU + {F32BAC49-F2F1-CE2D-0167-1F424878F925}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {F32BAC49-F2F1-CE2D-0167-1F424878F925}.Debug|Any CPU.Build.0 = Debug|Any CPU + {F32BAC49-F2F1-CE2D-0167-1F424878F925}.Release|Any CPU.ActiveCfg = Release|Any CPU + {F32BAC49-F2F1-CE2D-0167-1F424878F925}.Release|Any CPU.Build.0 = Release|Any CPU + {DFF0A2FD-8EAD-ED52-E0DC-D486AF73C847}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {DFF0A2FD-8EAD-ED52-E0DC-D486AF73C847}.Debug|Any CPU.Build.0 = Debug|Any CPU + {DFF0A2FD-8EAD-ED52-E0DC-D486AF73C847}.Release|Any CPU.ActiveCfg = Release|Any CPU + {DFF0A2FD-8EAD-ED52-E0DC-D486AF73C847}.Release|Any CPU.Build.0 = Release|Any CPU + {0555AF0D-ECDE-BFBD-BBB5-8DC1A5197960}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {0555AF0D-ECDE-BFBD-BBB5-8DC1A5197960}.Debug|Any CPU.Build.0 = Debug|Any CPU + {0555AF0D-ECDE-BFBD-BBB5-8DC1A5197960}.Release|Any CPU.ActiveCfg = Release|Any CPU + {0555AF0D-ECDE-BFBD-BBB5-8DC1A5197960}.Release|Any CPU.Build.0 = Release|Any CPU + {06C2DDE3-F045-CEE5-3F03-831228DDA3A9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {06C2DDE3-F045-CEE5-3F03-831228DDA3A9}.Debug|Any CPU.Build.0 = Debug|Any CPU + {06C2DDE3-F045-CEE5-3F03-831228DDA3A9}.Release|Any CPU.ActiveCfg = Release|Any CPU + {06C2DDE3-F045-CEE5-3F03-831228DDA3A9}.Release|Any CPU.Build.0 = Release|Any CPU + {7A646232-1EC4-2599-A1CD-7CF8DD15966E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {7A646232-1EC4-2599-A1CD-7CF8DD15966E}.Debug|Any CPU.Build.0 = Debug|Any CPU + {7A646232-1EC4-2599-A1CD-7CF8DD15966E}.Release|Any CPU.ActiveCfg = Release|Any CPU + {7A646232-1EC4-2599-A1CD-7CF8DD15966E}.Release|Any CPU.Build.0 = Release|Any CPU + {49DBA999-DF45-268E-63B2-FA94C0D1B87C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {49DBA999-DF45-268E-63B2-FA94C0D1B87C}.Debug|Any CPU.Build.0 = Debug|Any CPU + {49DBA999-DF45-268E-63B2-FA94C0D1B87C}.Release|Any CPU.ActiveCfg = Release|Any CPU + {49DBA999-DF45-268E-63B2-FA94C0D1B87C}.Release|Any CPU.Build.0 = Release|Any CPU + {2053DD51-3AE8-A5EB-65D6-04B16AA3CE66}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {2053DD51-3AE8-A5EB-65D6-04B16AA3CE66}.Debug|Any CPU.Build.0 = Debug|Any CPU + {2053DD51-3AE8-A5EB-65D6-04B16AA3CE66}.Release|Any CPU.ActiveCfg = Release|Any CPU + {2053DD51-3AE8-A5EB-65D6-04B16AA3CE66}.Release|Any CPU.Build.0 = Release|Any CPU + {368FC675-4200-A025-912A-59AFDAA29018}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {368FC675-4200-A025-912A-59AFDAA29018}.Debug|Any CPU.Build.0 = Debug|Any CPU + {368FC675-4200-A025-912A-59AFDAA29018}.Release|Any CPU.ActiveCfg = Release|Any CPU + {368FC675-4200-A025-912A-59AFDAA29018}.Release|Any CPU.Build.0 = Release|Any CPU + {DA00146E-B874-B45F-06C9-500B0401220C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {DA00146E-B874-B45F-06C9-500B0401220C}.Debug|Any CPU.Build.0 = Debug|Any CPU + {DA00146E-B874-B45F-06C9-500B0401220C}.Release|Any CPU.ActiveCfg = Release|Any CPU + {DA00146E-B874-B45F-06C9-500B0401220C}.Release|Any CPU.Build.0 = Release|Any CPU + {6E165AB7-E216-7010-DFED-1C1BC39B970D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {6E165AB7-E216-7010-DFED-1C1BC39B970D}.Debug|Any CPU.Build.0 = Debug|Any CPU + {6E165AB7-E216-7010-DFED-1C1BC39B970D}.Release|Any CPU.ActiveCfg = Release|Any CPU + {6E165AB7-E216-7010-DFED-1C1BC39B970D}.Release|Any CPU.Build.0 = Release|Any CPU + {D172DD08-1354-1BA0-9525-0E869B5020BD}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {D172DD08-1354-1BA0-9525-0E869B5020BD}.Debug|Any CPU.Build.0 = Debug|Any CPU + {D172DD08-1354-1BA0-9525-0E869B5020BD}.Release|Any CPU.ActiveCfg = Release|Any CPU + {D172DD08-1354-1BA0-9525-0E869B5020BD}.Release|Any CPU.Build.0 = Release|Any CPU + {8F9BD7C0-7ACF-A18C-5A88-229D8B094F3F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {8F9BD7C0-7ACF-A18C-5A88-229D8B094F3F}.Debug|Any CPU.Build.0 = Debug|Any CPU + {8F9BD7C0-7ACF-A18C-5A88-229D8B094F3F}.Release|Any CPU.ActiveCfg = Release|Any CPU + {8F9BD7C0-7ACF-A18C-5A88-229D8B094F3F}.Release|Any CPU.Build.0 = Release|Any CPU + {F23A84DD-FF43-F524-83B9-D17532BC1DE8}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {F23A84DD-FF43-F524-83B9-D17532BC1DE8}.Debug|Any CPU.Build.0 = Debug|Any CPU + {F23A84DD-FF43-F524-83B9-D17532BC1DE8}.Release|Any CPU.ActiveCfg = Release|Any CPU + {F23A84DD-FF43-F524-83B9-D17532BC1DE8}.Release|Any CPU.Build.0 = Release|Any CPU + {649C131C-6FE6-E748-C932-651E89DFADFE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {649C131C-6FE6-E748-C932-651E89DFADFE}.Debug|Any CPU.Build.0 = Debug|Any CPU + {649C131C-6FE6-E748-C932-651E89DFADFE}.Release|Any CPU.ActiveCfg = Release|Any CPU + {649C131C-6FE6-E748-C932-651E89DFADFE}.Release|Any CPU.Build.0 = Release|Any CPU + {CC93A1BB-FB4C-9BE7-0832-CF4AF1B69A92}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {CC93A1BB-FB4C-9BE7-0832-CF4AF1B69A92}.Debug|Any CPU.Build.0 = Debug|Any CPU + {CC93A1BB-FB4C-9BE7-0832-CF4AF1B69A92}.Release|Any CPU.ActiveCfg = Release|Any CPU + {CC93A1BB-FB4C-9BE7-0832-CF4AF1B69A92}.Release|Any CPU.Build.0 = Release|Any CPU + {E912C16D-738B-1DB1-3456-24751CAB3849}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {E912C16D-738B-1DB1-3456-24751CAB3849}.Debug|Any CPU.Build.0 = Debug|Any CPU + {E912C16D-738B-1DB1-3456-24751CAB3849}.Release|Any CPU.ActiveCfg = Release|Any CPU + {E912C16D-738B-1DB1-3456-24751CAB3849}.Release|Any CPU.Build.0 = Release|Any CPU + {FFF816B4-207D-AC0E-3B30-C9E0CA06D257}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {FFF816B4-207D-AC0E-3B30-C9E0CA06D257}.Debug|Any CPU.Build.0 = Debug|Any CPU + {FFF816B4-207D-AC0E-3B30-C9E0CA06D257}.Release|Any CPU.ActiveCfg = Release|Any CPU + {FFF816B4-207D-AC0E-3B30-C9E0CA06D257}.Release|Any CPU.Build.0 = Release|Any CPU + {BD55EBED-7E44-C621-7FF4-FD7709A438F6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {BD55EBED-7E44-C621-7FF4-FD7709A438F6}.Debug|Any CPU.Build.0 = Debug|Any CPU + {BD55EBED-7E44-C621-7FF4-FD7709A438F6}.Release|Any CPU.ActiveCfg = Release|Any CPU + {BD55EBED-7E44-C621-7FF4-FD7709A438F6}.Release|Any CPU.Build.0 = Release|Any CPU + {F57AB79F-532A-EBC2-871B-64C9FD27FEA9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {F57AB79F-532A-EBC2-871B-64C9FD27FEA9}.Debug|Any CPU.Build.0 = Debug|Any CPU + {F57AB79F-532A-EBC2-871B-64C9FD27FEA9}.Release|Any CPU.ActiveCfg = Release|Any CPU + {F57AB79F-532A-EBC2-871B-64C9FD27FEA9}.Release|Any CPU.Build.0 = Release|Any CPU + {2598AAB1-DA3E-A2AE-B060-D1BAF7BBE4BC}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {2598AAB1-DA3E-A2AE-B060-D1BAF7BBE4BC}.Debug|Any CPU.Build.0 = Debug|Any CPU + {2598AAB1-DA3E-A2AE-B060-D1BAF7BBE4BC}.Release|Any CPU.ActiveCfg = Release|Any CPU + {2598AAB1-DA3E-A2AE-B060-D1BAF7BBE4BC}.Release|Any CPU.Build.0 = Release|Any CPU + {5A8F066D-8FFA-AFC1-1E8E-0DE9AEAE676F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {5A8F066D-8FFA-AFC1-1E8E-0DE9AEAE676F}.Debug|Any CPU.Build.0 = Debug|Any CPU + {5A8F066D-8FFA-AFC1-1E8E-0DE9AEAE676F}.Release|Any CPU.ActiveCfg = Release|Any CPU + {5A8F066D-8FFA-AFC1-1E8E-0DE9AEAE676F}.Release|Any CPU.Build.0 = Release|Any CPU + {469C328E-E07E-04A6-A566-E2ABA6732CBE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {469C328E-E07E-04A6-A566-E2ABA6732CBE}.Debug|Any CPU.Build.0 = Debug|Any CPU + {469C328E-E07E-04A6-A566-E2ABA6732CBE}.Release|Any CPU.ActiveCfg = Release|Any CPU + {469C328E-E07E-04A6-A566-E2ABA6732CBE}.Release|Any CPU.Build.0 = Release|Any CPU + {541572F4-E051-F4F8-9B39-87BFA550E13A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {541572F4-E051-F4F8-9B39-87BFA550E13A}.Debug|Any CPU.Build.0 = Debug|Any CPU + {541572F4-E051-F4F8-9B39-87BFA550E13A}.Release|Any CPU.ActiveCfg = Release|Any CPU + {541572F4-E051-F4F8-9B39-87BFA550E13A}.Release|Any CPU.Build.0 = Release|Any CPU + {59C211A4-B928-7505-AF2E-6C604DAB3622}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {59C211A4-B928-7505-AF2E-6C604DAB3622}.Debug|Any CPU.Build.0 = Debug|Any CPU + {59C211A4-B928-7505-AF2E-6C604DAB3622}.Release|Any CPU.ActiveCfg = Release|Any CPU + {59C211A4-B928-7505-AF2E-6C604DAB3622}.Release|Any CPU.Build.0 = Release|Any CPU + {A5C76622-FE9B-BD41-A430-5208CCE95FDA}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {A5C76622-FE9B-BD41-A430-5208CCE95FDA}.Debug|Any CPU.Build.0 = Debug|Any CPU + {A5C76622-FE9B-BD41-A430-5208CCE95FDA}.Release|Any CPU.ActiveCfg = Release|Any CPU + {A5C76622-FE9B-BD41-A430-5208CCE95FDA}.Release|Any CPU.Build.0 = Release|Any CPU + {7F93B805-8A3C-D81B-FEE7-A4CE542AB2D2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {7F93B805-8A3C-D81B-FEE7-A4CE542AB2D2}.Debug|Any CPU.Build.0 = Debug|Any CPU + {7F93B805-8A3C-D81B-FEE7-A4CE542AB2D2}.Release|Any CPU.ActiveCfg = Release|Any CPU + {7F93B805-8A3C-D81B-FEE7-A4CE542AB2D2}.Release|Any CPU.Build.0 = Release|Any CPU + {83045A6C-05F6-CC92-820D-281C04862F7C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {83045A6C-05F6-CC92-820D-281C04862F7C}.Debug|Any CPU.Build.0 = Debug|Any CPU + {83045A6C-05F6-CC92-820D-281C04862F7C}.Release|Any CPU.ActiveCfg = Release|Any CPU + {83045A6C-05F6-CC92-820D-281C04862F7C}.Release|Any CPU.Build.0 = Release|Any CPU + {458B6228-8612-4186-4E92-36797B16DFA3}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {458B6228-8612-4186-4E92-36797B16DFA3}.Debug|Any CPU.Build.0 = Debug|Any CPU + {458B6228-8612-4186-4E92-36797B16DFA3}.Release|Any CPU.ActiveCfg = Release|Any CPU + {458B6228-8612-4186-4E92-36797B16DFA3}.Release|Any CPU.Build.0 = Release|Any CPU + {7F94681B-B8E0-DF52-D282-9411B6632E77}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {7F94681B-B8E0-DF52-D282-9411B6632E77}.Debug|Any CPU.Build.0 = Debug|Any CPU + {7F94681B-B8E0-DF52-D282-9411B6632E77}.Release|Any CPU.ActiveCfg = Release|Any CPU + {7F94681B-B8E0-DF52-D282-9411B6632E77}.Release|Any CPU.Build.0 = Release|Any CPU + {98604E75-FAB4-80FC-A539-DE0A0CFD0687}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {98604E75-FAB4-80FC-A539-DE0A0CFD0687}.Debug|Any CPU.Build.0 = Debug|Any CPU + {98604E75-FAB4-80FC-A539-DE0A0CFD0687}.Release|Any CPU.ActiveCfg = Release|Any CPU + {98604E75-FAB4-80FC-A539-DE0A0CFD0687}.Release|Any CPU.Build.0 = Release|Any CPU + {8576736C-D7B9-E3D7-203C-46D059601F81}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {8576736C-D7B9-E3D7-203C-46D059601F81}.Debug|Any CPU.Build.0 = Debug|Any CPU + {8576736C-D7B9-E3D7-203C-46D059601F81}.Release|Any CPU.ActiveCfg = Release|Any CPU + {8576736C-D7B9-E3D7-203C-46D059601F81}.Release|Any CPU.Build.0 = Release|Any CPU + {26BB76BB-57D1-3F9B-44BC-3ECB5B9EEF49}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {26BB76BB-57D1-3F9B-44BC-3ECB5B9EEF49}.Debug|Any CPU.Build.0 = Debug|Any CPU + {26BB76BB-57D1-3F9B-44BC-3ECB5B9EEF49}.Release|Any CPU.ActiveCfg = Release|Any CPU + {26BB76BB-57D1-3F9B-44BC-3ECB5B9EEF49}.Release|Any CPU.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection +EndGlobal diff --git a/AxibugEmuOnline.Client/Packages/packages-lock.json b/AxibugEmuOnline.Client/Packages/packages-lock.json index bab524c7..6ae76026 100644 --- a/AxibugEmuOnline.Client/Packages/packages-lock.json +++ b/AxibugEmuOnline.Client/Packages/packages-lock.json @@ -28,18 +28,18 @@ "depth": 0, "source": "builtin", "dependencies": { - "com.unity.ide.visualstudio": "2.0.14", - "com.unity.ide.rider": "3.0.13", + "com.unity.ide.visualstudio": "2.0.22", + "com.unity.ide.rider": "3.0.28", "com.unity.ide.vscode": "1.2.5", "com.unity.editorcoroutines": "1.0.0", - "com.unity.performance.profile-analyzer": "1.1.1", - "com.unity.test-framework": "1.1.31", - "com.unity.testtools.codecoverage": "1.0.1" + "com.unity.performance.profile-analyzer": "1.2.2", + "com.unity.test-framework": "1.1.33", + "com.unity.testtools.codecoverage": "1.2.5" } }, "com.unity.ide.rider": { - "version": "3.0.13", - "depth": 0, + "version": "3.0.28", + "depth": 1, "source": "registry", "dependencies": { "com.unity.ext.nunit": "1.0.6" @@ -47,8 +47,8 @@ "url": "https://packages.unity.cn" }, "com.unity.ide.visualstudio": { - "version": "2.0.14", - "depth": 0, + "version": "2.0.22", + "depth": 1, "source": "registry", "dependencies": { "com.unity.test-framework": "1.1.9" @@ -62,32 +62,41 @@ "dependencies": {}, "url": "https://packages.unity.cn" }, + "com.unity.nuget.newtonsoft-json": { + "version": "3.2.1", + "depth": 2, + "source": "registry", + "dependencies": {}, + "url": "https://packages.unity.cn" + }, "com.unity.performance.profile-analyzer": { - "version": "1.1.1", + "version": "1.2.2", "depth": 1, "source": "registry", "dependencies": {}, "url": "https://packages.unity.cn" }, "com.unity.services.core": { - "version": "1.0.1", + "version": "1.12.5", "depth": 1, "source": "registry", "dependencies": { - "com.unity.modules.unitywebrequest": "1.0.0" + "com.unity.modules.unitywebrequest": "1.0.0", + "com.unity.nuget.newtonsoft-json": "3.2.1", + "com.unity.modules.androidjni": "1.0.0" }, "url": "https://packages.unity.cn" }, "com.unity.settings-manager": { - "version": "1.0.3", + "version": "2.0.1", "depth": 2, "source": "registry", "dependencies": {}, "url": "https://packages.unity.cn" }, "com.unity.test-framework": { - "version": "1.1.31", - "depth": 0, + "version": "1.1.33", + "depth": 1, "source": "registry", "dependencies": { "com.unity.ext.nunit": "1.0.6", @@ -97,7 +106,7 @@ "url": "https://packages.unity.cn" }, "com.unity.testtools.codecoverage": { - "version": "1.0.1", + "version": "1.2.5", "depth": 1, "source": "registry", "dependencies": { @@ -278,17 +287,6 @@ "version": "1.0.0", "depth": 0, "source": "builtin", - "dependencies": { - "com.unity.modules.ui": "1.0.0", - "com.unity.modules.imgui": "1.0.0", - "com.unity.modules.jsonserialize": "1.0.0", - "com.unity.modules.uielementsnative": "1.0.0" - } - }, - "com.unity.modules.uielementsnative": { - "version": "1.0.0", - "depth": 1, - "source": "builtin", "dependencies": { "com.unity.modules.ui": "1.0.0", "com.unity.modules.imgui": "1.0.0", diff --git a/AxibugEmuOnline.Client/ProjectSettings/AudioManager.asset b/AxibugEmuOnline.Client/ProjectSettings/AudioManager.asset new file mode 100644 index 00000000..df1e8090 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/AudioManager.asset @@ -0,0 +1,20 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!11 &1 +AudioManager: + m_ObjectHideFlags: 0 + serializedVersion: 2 + m_Volume: 1 + Rolloff Scale: 1 + Doppler Factor: 1 + Default Speaker Mode: 2 + m_SampleRate: 0 + m_DSPBufferSize: 1024 + m_VirtualVoiceCount: 512 + m_RealVoiceCount: 32 + m_EnableOutputSuspension: 1 + m_SpatializerPlugin: + m_AmbisonicDecoderPlugin: + m_DisableAudio: 0 + m_VirtualizeEffects: 1 + m_RequestedDSPBufferSize: 0 diff --git a/AxibugEmuOnline.Client/ProjectSettings/ClusterInputManager.asset b/AxibugEmuOnline.Client/ProjectSettings/ClusterInputManager.asset new file mode 100644 index 00000000..e7886b26 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/ClusterInputManager.asset @@ -0,0 +1,6 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!236 &1 +ClusterInputManager: + m_ObjectHideFlags: 0 + m_Inputs: [] diff --git a/AxibugEmuOnline.Client/ProjectSettings/DynamicsManager.asset b/AxibugEmuOnline.Client/ProjectSettings/DynamicsManager.asset new file mode 100644 index 00000000..3e5f55bc --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/DynamicsManager.asset @@ -0,0 +1,40 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!55 &1 +PhysicsManager: + m_ObjectHideFlags: 0 + serializedVersion: 14 + m_Gravity: {x: 0, y: -9.81, z: 0} + m_DefaultMaterial: {fileID: 0} + m_BounceThreshold: 2 + m_DefaultMaxDepenetrationVelocity: 10 + m_SleepThreshold: 0.005 + m_DefaultContactOffset: 0.01 + m_DefaultSolverIterations: 6 + m_DefaultSolverVelocityIterations: 1 + m_QueriesHitBackfaces: 0 + m_QueriesHitTriggers: 1 + m_EnableAdaptiveForce: 0 + m_ClothInterCollisionDistance: 0.1 + m_ClothInterCollisionStiffness: 0.2 + m_ContactsGeneration: 1 + m_LayerCollisionMatrix: ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff + m_SimulationMode: 0 + m_AutoSyncTransforms: 0 + m_ReuseCollisionCallbacks: 0 + m_InvokeCollisionCallbacks: 1 + m_ClothInterCollisionSettingsToggle: 0 + m_ClothGravity: {x: 0, y: -9.81, z: 0} + m_ContactPairsMode: 0 + m_BroadphaseType: 0 + m_WorldBounds: + m_Center: {x: 0, y: 0, z: 0} + m_Extent: {x: 250, y: 250, z: 250} + m_WorldSubdivisions: 8 + m_FrictionType: 0 + m_EnableEnhancedDeterminism: 0 + m_EnableUnifiedHeightmaps: 1 + m_ImprovedPatchFriction: 0 + m_SolverType: 0 + m_DefaultMaxAngularSpeed: 50 + m_FastMotionThreshold: 3.4028235e+38 diff --git a/AxibugEmuOnline.Client/ProjectSettings/EditorBuildSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/EditorBuildSettings.asset new file mode 100644 index 00000000..0147887e --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/EditorBuildSettings.asset @@ -0,0 +1,8 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!1045 &1 +EditorBuildSettings: + m_ObjectHideFlags: 0 + serializedVersion: 2 + m_Scenes: [] + m_configObjects: {} diff --git a/AxibugEmuOnline.Client/ProjectSettings/EditorSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/EditorSettings.asset new file mode 100644 index 00000000..67bdff5a --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/EditorSettings.asset @@ -0,0 +1,47 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!159 &1 +EditorSettings: + m_ObjectHideFlags: 0 + serializedVersion: 12 + m_SerializationMode: 2 + m_LineEndingsForNewScripts: 2 + m_DefaultBehaviorMode: 0 + m_PrefabRegularEnvironment: {fileID: 0} + m_PrefabUIEnvironment: {fileID: 0} + m_SpritePackerMode: 0 + m_SpritePackerCacheSize: 10 + m_SpritePackerPaddingPower: 1 + m_Bc7TextureCompressor: 0 + m_EtcTextureCompressorBehavior: 1 + m_EtcTextureFastCompressor: 1 + m_EtcTextureNormalCompressor: 2 + m_EtcTextureBestCompressor: 4 + m_ProjectGenerationIncludedExtensions: txt;xml;fnt;cd;asmdef;asmref;rsp;java;cpp;c;mm;m;h + m_ProjectGenerationRootNamespace: + m_EnableTextureStreamingInEditMode: 1 + m_EnableTextureStreamingInPlayMode: 1 + m_EnableEditorAsyncCPUTextureLoading: 0 + m_AsyncShaderCompilation: 1 + m_PrefabModeAllowAutoSave: 1 + m_EnterPlayModeOptionsEnabled: 0 + m_EnterPlayModeOptions: 3 + m_GameObjectNamingDigits: 1 + m_GameObjectNamingScheme: 0 + m_AssetNamingUsesSpace: 1 + m_InspectorUseIMGUIDefaultInspector: 0 + m_UseLegacyProbeSampleCount: 0 + m_SerializeInlineMappingsOnOneLine: 1 + m_DisableCookiesInLightmapper: 0 + m_AssetPipelineMode: 1 + m_RefreshImportMode: 0 + m_CacheServerMode: 0 + m_CacheServerEndpoint: + m_CacheServerNamespacePrefix: default + m_CacheServerEnableDownload: 1 + m_CacheServerEnableUpload: 1 + m_CacheServerEnableAuth: 0 + m_CacheServerEnableTls: 0 + m_CacheServerValidationMode: 2 + m_CacheServerDownloadBatchSize: 128 + m_EnableEnlightenBakedGI: 0 diff --git a/AxibugEmuOnline.Client/ProjectSettings/GraphicsSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/GraphicsSettings.asset new file mode 100644 index 00000000..2d513492 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/GraphicsSettings.asset @@ -0,0 +1,67 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!30 &1 +GraphicsSettings: + m_ObjectHideFlags: 0 + serializedVersion: 15 + m_Deferred: + m_Mode: 1 + m_Shader: {fileID: 69, guid: 0000000000000000f000000000000000, type: 0} + m_DeferredReflections: + m_Mode: 1 + m_Shader: {fileID: 74, guid: 0000000000000000f000000000000000, type: 0} + m_ScreenSpaceShadows: + m_Mode: 1 + m_Shader: {fileID: 64, guid: 0000000000000000f000000000000000, type: 0} + m_DepthNormals: + m_Mode: 1 + m_Shader: {fileID: 62, guid: 0000000000000000f000000000000000, type: 0} + m_MotionVectors: + m_Mode: 1 + m_Shader: {fileID: 75, guid: 0000000000000000f000000000000000, type: 0} + m_LightHalo: + m_Mode: 1 + m_Shader: {fileID: 105, guid: 0000000000000000f000000000000000, type: 0} + m_LensFlare: + m_Mode: 1 + m_Shader: {fileID: 102, guid: 0000000000000000f000000000000000, type: 0} + m_VideoShadersIncludeMode: 2 + m_AlwaysIncludedShaders: + - {fileID: 7, guid: 0000000000000000f000000000000000, type: 0} + - {fileID: 15104, guid: 0000000000000000f000000000000000, type: 0} + - {fileID: 15105, guid: 0000000000000000f000000000000000, type: 0} + - {fileID: 15106, guid: 0000000000000000f000000000000000, type: 0} + - {fileID: 10753, guid: 0000000000000000f000000000000000, type: 0} + - {fileID: 10770, guid: 0000000000000000f000000000000000, type: 0} + - {fileID: 10783, guid: 0000000000000000f000000000000000, type: 0} + m_PreloadedShaders: [] + m_PreloadShadersBatchTimeLimit: -1 + m_SpritesDefaultMaterial: {fileID: 10754, guid: 0000000000000000f000000000000000, type: 0} + m_CustomRenderPipeline: {fileID: 0} + m_TransparencySortMode: 0 + m_TransparencySortAxis: {x: 0, y: 0, z: 1} + m_DefaultRenderingPath: 1 + m_DefaultMobileRenderingPath: 1 + m_TierSettings: [] + m_LightmapStripping: 0 + m_FogStripping: 0 + m_InstancingStripping: 0 + m_BrgStripping: 0 + m_LightmapKeepPlain: 1 + m_LightmapKeepDirCombined: 1 + m_LightmapKeepDynamicPlain: 1 + m_LightmapKeepDynamicDirCombined: 1 + m_LightmapKeepShadowMask: 1 + m_LightmapKeepSubtractive: 1 + m_FogKeepLinear: 1 + m_FogKeepExp: 1 + m_FogKeepExp2: 1 + m_AlbedoSwatchInfos: [] + m_LightsUseLinearIntensity: 0 + m_LightsUseColorTemperature: 0 + m_DefaultRenderingLayerMask: 1 + m_LogWhenShaderIsCompiled: 0 + m_SRPDefaultSettings: {} + m_LightProbeOutsideHullStrategy: 1 + m_CameraRelativeLightCulling: 0 + m_CameraRelativeShadowCulling: 0 diff --git a/AxibugEmuOnline.Client/ProjectSettings/InputManager.asset b/AxibugEmuOnline.Client/ProjectSettings/InputManager.asset new file mode 100644 index 00000000..8068b205 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/InputManager.asset @@ -0,0 +1,296 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!13 &1 +InputManager: + m_ObjectHideFlags: 0 + serializedVersion: 2 + m_Axes: + - serializedVersion: 3 + m_Name: Horizontal + descriptiveName: + descriptiveNegativeName: + negativeButton: left + positiveButton: right + altNegativeButton: a + altPositiveButton: d + gravity: 3 + dead: 0.001 + sensitivity: 3 + snap: 1 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Vertical + descriptiveName: + descriptiveNegativeName: + negativeButton: down + positiveButton: up + altNegativeButton: s + altPositiveButton: w + gravity: 3 + dead: 0.001 + sensitivity: 3 + snap: 1 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Fire1 + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: left ctrl + altNegativeButton: + altPositiveButton: mouse 0 + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Fire2 + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: left alt + altNegativeButton: + altPositiveButton: mouse 1 + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Fire3 + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: left shift + altNegativeButton: + altPositiveButton: mouse 2 + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Jump + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: space + altNegativeButton: + altPositiveButton: + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Mouse X + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: + altNegativeButton: + altPositiveButton: + gravity: 0 + dead: 0 + sensitivity: 0.1 + snap: 0 + invert: 0 + type: 1 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Mouse Y + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: + altNegativeButton: + altPositiveButton: + gravity: 0 + dead: 0 + sensitivity: 0.1 + snap: 0 + invert: 0 + type: 1 + axis: 1 + joyNum: 0 + - serializedVersion: 3 + m_Name: Mouse ScrollWheel + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: + altNegativeButton: + altPositiveButton: + gravity: 0 + dead: 0 + sensitivity: 0.1 + snap: 0 + invert: 0 + type: 1 + axis: 2 + joyNum: 0 + - serializedVersion: 3 + m_Name: Horizontal + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: + altNegativeButton: + altPositiveButton: + gravity: 0 + dead: 0.19 + sensitivity: 1 + snap: 0 + invert: 0 + type: 2 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Vertical + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: + altNegativeButton: + altPositiveButton: + gravity: 0 + dead: 0.19 + sensitivity: 1 + snap: 0 + invert: 1 + type: 2 + axis: 1 + joyNum: 0 + - serializedVersion: 3 + m_Name: Fire1 + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: joystick button 0 + altNegativeButton: + altPositiveButton: + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Fire2 + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: joystick button 1 + altNegativeButton: + altPositiveButton: + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Fire3 + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: joystick button 2 + altNegativeButton: + altPositiveButton: + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Jump + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: joystick button 3 + altNegativeButton: + altPositiveButton: + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Submit + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: return + altNegativeButton: + altPositiveButton: joystick button 0 + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Submit + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: enter + altNegativeButton: + altPositiveButton: space + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + - serializedVersion: 3 + m_Name: Cancel + descriptiveName: + descriptiveNegativeName: + negativeButton: + positiveButton: escape + altNegativeButton: + altPositiveButton: joystick button 1 + gravity: 1000 + dead: 0.001 + sensitivity: 1000 + snap: 0 + invert: 0 + type: 0 + axis: 0 + joyNum: 0 + m_UsePhysicalKeys: 1 diff --git a/AxibugEmuOnline.Client/ProjectSettings/MemorySettings.asset b/AxibugEmuOnline.Client/ProjectSettings/MemorySettings.asset new file mode 100644 index 00000000..5b5facec --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/MemorySettings.asset @@ -0,0 +1,35 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!387306366 &1 +MemorySettings: + m_ObjectHideFlags: 0 + m_EditorMemorySettings: + m_MainAllocatorBlockSize: -1 + m_ThreadAllocatorBlockSize: -1 + m_MainGfxBlockSize: -1 + m_ThreadGfxBlockSize: -1 + m_CacheBlockSize: -1 + m_TypetreeBlockSize: -1 + m_ProfilerBlockSize: -1 + m_ProfilerEditorBlockSize: -1 + m_BucketAllocatorGranularity: -1 + m_BucketAllocatorBucketsCount: -1 + m_BucketAllocatorBlockSize: -1 + m_BucketAllocatorBlockCount: -1 + m_ProfilerBucketAllocatorGranularity: -1 + m_ProfilerBucketAllocatorBucketsCount: -1 + m_ProfilerBucketAllocatorBlockSize: -1 + m_ProfilerBucketAllocatorBlockCount: -1 + m_TempAllocatorSizeMain: -1 + m_JobTempAllocatorBlockSize: -1 + m_BackgroundJobTempAllocatorBlockSize: -1 + m_JobTempAllocatorReducedBlockSize: -1 + m_TempAllocatorSizeGIBakingWorker: -1 + m_TempAllocatorSizeNavMeshWorker: -1 + m_TempAllocatorSizeAudioWorker: -1 + m_TempAllocatorSizeCloudWorker: -1 + m_TempAllocatorSizeGfx: -1 + m_TempAllocatorSizeJobWorker: -1 + m_TempAllocatorSizeBackgroundWorker: -1 + m_TempAllocatorSizePreloadManager: -1 + m_PlatformMemorySettings: {} diff --git a/AxibugEmuOnline.Client/ProjectSettings/NavMeshAreas.asset b/AxibugEmuOnline.Client/ProjectSettings/NavMeshAreas.asset new file mode 100644 index 00000000..2e2e3696 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/NavMeshAreas.asset @@ -0,0 +1,93 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!126 &1 +NavMeshProjectSettings: + m_ObjectHideFlags: 0 + serializedVersion: 2 + areas: + - name: Walkable + cost: 1 + - name: Not Walkable + cost: 1 + - name: Jump + cost: 2 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + - name: + cost: 1 + m_LastAgentTypeID: -887442657 + m_Settings: + - serializedVersion: 3 + agentTypeID: 0 + agentRadius: 0.5 + agentHeight: 2 + agentSlope: 45 + agentClimb: 0.75 + ledgeDropHeight: 0 + maxJumpAcrossDistance: 0 + minRegionArea: 2 + manualCellSize: 0 + cellSize: 0.16666667 + manualTileSize: 0 + tileSize: 256 + buildHeightMesh: 0 + maxJobWorkers: 0 + preserveTilesOutsideBounds: 0 + debug: + m_Flags: 0 + m_SettingNames: + - Humanoid diff --git a/AxibugEmuOnline.Client/ProjectSettings/PackageManagerSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/PackageManagerSettings.asset new file mode 100644 index 00000000..95c895df --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/PackageManagerSettings.asset @@ -0,0 +1,36 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!114 &1 +MonoBehaviour: + m_ObjectHideFlags: 53 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 13964, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_EnablePreReleasePackages: 0 + m_AdvancedSettingsExpanded: 1 + m_ScopedRegistriesSettingsExpanded: 1 + m_SeeAllPackageVersions: 0 + m_DismissPreviewPackagesInUse: 0 + oneTimeWarningShown: 0 + m_Registries: + - m_Id: main + m_Name: + m_Url: https://packages.unity.cn + m_Scopes: [] + m_IsDefault: 1 + m_Capabilities: 7 + m_ConfigSource: 0 + m_UserSelectedRegistryName: + m_UserAddingNewScopedRegistry: 0 + m_RegistryInfoDraft: + m_Modified: 0 + m_ErrorMessage: + m_UserModificationsInstanceId: -892 + m_OriginalInstanceId: -894 + m_LoadAssets: 0 diff --git a/AxibugEmuOnline.Client/ProjectSettings/Physics2DSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/Physics2DSettings.asset new file mode 100644 index 00000000..b71b65c4 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/Physics2DSettings.asset @@ -0,0 +1,48 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!19 &1 +Physics2DSettings: + m_ObjectHideFlags: 0 + serializedVersion: 6 + m_Gravity: {x: 0, y: -9.81} + m_DefaultMaterial: {fileID: 0} + m_VelocityIterations: 8 + m_PositionIterations: 3 + m_VelocityThreshold: 1 + m_MaxLinearCorrection: 0.2 + m_MaxAngularCorrection: 8 + m_MaxTranslationSpeed: 100 + m_MaxRotationSpeed: 360 + m_BaumgarteScale: 0.2 + m_BaumgarteTimeOfImpactScale: 0.75 + m_TimeToSleep: 0.5 + m_LinearSleepTolerance: 0.01 + m_AngularSleepTolerance: 2 + m_DefaultContactOffset: 0.01 + m_JobOptions: + serializedVersion: 2 + useMultithreading: 0 + useConsistencySorting: 0 + m_InterpolationPosesPerJob: 100 + m_NewContactsPerJob: 30 + m_CollideContactsPerJob: 100 + m_ClearFlagsPerJob: 200 + m_ClearBodyForcesPerJob: 200 + m_SyncDiscreteFixturesPerJob: 50 + m_SyncContinuousFixturesPerJob: 50 + m_FindNearestContactsPerJob: 100 + m_UpdateTriggerContactsPerJob: 100 + m_IslandSolverCostThreshold: 100 + m_IslandSolverBodyCostScale: 1 + m_IslandSolverContactCostScale: 10 + m_IslandSolverJointCostScale: 10 + m_IslandSolverBodiesPerJob: 50 + m_IslandSolverContactsPerJob: 50 + m_SimulationMode: 0 + m_QueriesHitTriggers: 1 + m_QueriesStartInColliders: 1 + m_CallbacksOnDisable: 1 + m_ReuseCollisionCallbacks: 1 + m_AutoSyncTransforms: 0 + m_GizmoOptions: 10 + m_LayerCollisionMatrix: ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff diff --git a/AxibugEmuOnline.Client/ProjectSettings/PresetManager.asset b/AxibugEmuOnline.Client/ProjectSettings/PresetManager.asset new file mode 100644 index 00000000..67a94dae --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/PresetManager.asset @@ -0,0 +1,7 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!1386491679 &1 +PresetManager: + m_ObjectHideFlags: 0 + serializedVersion: 2 + m_DefaultPresets: {} diff --git a/AxibugEmuOnline.Client/ProjectSettings/ProjectSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/ProjectSettings.asset new file mode 100644 index 00000000..129a36f7 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/ProjectSettings.asset @@ -0,0 +1,667 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!129 &1 +PlayerSettings: + m_ObjectHideFlags: 0 + serializedVersion: 26 + productGUID: eb1ab5b80e48ed5419399f517f98a577 + AndroidProfiler: 0 + AndroidFilterTouchesWhenObscured: 0 + AndroidEnableSustainedPerformanceMode: 0 + defaultScreenOrientation: 4 + targetDevice: 2 + useOnDemandResources: 0 + accelerometerFrequency: 60 + companyName: DefaultCompany + productName: AxibugEmuOnline.Client + defaultCursor: {fileID: 0} + cursorHotspot: {x: 0, y: 0} + m_SplashScreenBackgroundColor: {r: 0.12156863, g: 0.12156863, b: 0.1254902, a: 1} + m_ShowUnitySplashScreen: 1 + m_ShowUnitySplashLogo: 1 + m_SplashScreenOverlayOpacity: 1 + m_SplashScreenAnimation: 1 + m_SplashScreenLogoStyle: 1 + m_SplashScreenDrawMode: 0 + m_SplashScreenBackgroundAnimationZoom: 1 + m_SplashScreenLogoAnimationZoom: 1 + m_SplashScreenBackgroundLandscapeAspect: 1 + m_SplashScreenBackgroundPortraitAspect: 1 + m_SplashScreenBackgroundLandscapeUvs: + serializedVersion: 2 + x: 0 + y: 0 + width: 1 + height: 1 + m_SplashScreenBackgroundPortraitUvs: + serializedVersion: 2 + x: 0 + y: 0 + width: 1 + height: 1 + m_SplashScreenLogos: [] + m_VirtualRealitySplashScreen: {fileID: 0} + m_HolographicTrackingLossScreen: {fileID: 0} + defaultScreenWidth: 1920 + defaultScreenHeight: 1080 + defaultScreenWidthWeb: 960 + defaultScreenHeightWeb: 600 + m_StereoRenderingPath: 0 + m_ActiveColorSpace: 0 + unsupportedMSAAFallback: 0 + m_SpriteBatchVertexThreshold: 300 + m_MTRendering: 1 + mipStripping: 0 + numberOfMipsStripped: 0 + numberOfMipsStrippedPerMipmapLimitGroup: {} + m_StackTraceTypes: 010000000100000001000000010000000100000001000000 + iosShowActivityIndicatorOnLoading: -1 + androidShowActivityIndicatorOnLoading: -1 + iosUseCustomAppBackgroundBehavior: 0 + allowedAutorotateToPortrait: 1 + allowedAutorotateToPortraitUpsideDown: 1 + allowedAutorotateToLandscapeRight: 1 + allowedAutorotateToLandscapeLeft: 1 + useOSAutorotation: 1 + use32BitDisplayBuffer: 1 + preserveFramebufferAlpha: 0 + disableDepthAndStencilBuffers: 0 + androidStartInFullscreen: 1 + androidRenderOutsideSafeArea: 1 + androidUseSwappy: 1 + androidBlitType: 0 + androidResizableWindow: 0 + androidDefaultWindowWidth: 1920 + androidDefaultWindowHeight: 1080 + androidMinimumWindowWidth: 400 + androidMinimumWindowHeight: 300 + androidFullscreenMode: 1 + androidAutoRotationBehavior: 1 + defaultIsNativeResolution: 1 + macRetinaSupport: 1 + runInBackground: 0 + captureSingleScreen: 0 + muteOtherAudioSources: 0 + Prepare IOS For Recording: 0 + Force IOS Speakers When Recording: 0 + deferSystemGesturesMode: 0 + hideHomeButton: 0 + submitAnalytics: 1 + usePlayerLog: 1 + dedicatedServerOptimizations: 0 + bakeCollisionMeshes: 0 + forceSingleInstance: 0 + useFlipModelSwapchain: 1 + resizableWindow: 0 + useMacAppStoreValidation: 0 + macAppStoreCategory: public.app-category.games + gpuSkinning: 0 + xboxPIXTextureCapture: 0 + xboxEnableAvatar: 0 + xboxEnableKinect: 0 + xboxEnableKinectAutoTracking: 0 + xboxEnableFitness: 0 + visibleInBackground: 1 + allowFullscreenSwitch: 1 + fullscreenMode: 1 + xboxSpeechDB: 0 + xboxEnableHeadOrientation: 0 + xboxEnableGuest: 0 + xboxEnablePIXSampling: 0 + metalFramebufferOnly: 0 + xboxOneResolution: 0 + xboxOneSResolution: 0 + xboxOneXResolution: 3 + xboxOneMonoLoggingLevel: 0 + xboxOneLoggingLevel: 1 + xboxOneDisableEsram: 0 + xboxOneEnableTypeOptimization: 0 + xboxOnePresentImmediateThreshold: 0 + switchQueueCommandMemory: 1048576 + switchQueueControlMemory: 16384 + switchQueueComputeMemory: 262144 + switchNVNShaderPoolsGranularity: 33554432 + switchNVNDefaultPoolsGranularity: 16777216 + switchNVNOtherPoolsGranularity: 16777216 + switchGpuScratchPoolGranularity: 2097152 + switchAllowGpuScratchShrinking: 0 + switchNVNMaxPublicTextureIDCount: 0 + switchNVNMaxPublicSamplerIDCount: 0 + switchNVNGraphicsFirmwareMemory: 32 + switchMaxWorkerMultiple: 8 + stadiaPresentMode: 0 + stadiaTargetFramerate: 0 + vulkanNumSwapchainBuffers: 3 + vulkanEnableSetSRGBWrite: 0 + vulkanEnablePreTransform: 0 + vulkanEnableLateAcquireNextImage: 0 + vulkanEnableCommandBufferRecycling: 1 + loadStoreDebugModeEnabled: 0 + visionOSBundleVersion: 1.0 + tvOSBundleVersion: 1.0 + bundleVersion: 1.0 + preloadedAssets: [] + metroInputSource: 0 + wsaTransparentSwapchain: 0 + m_HolographicPauseOnTrackingLoss: 1 + xboxOneDisableKinectGpuReservation: 1 + xboxOneEnable7thCore: 1 + vrSettings: + enable360StereoCapture: 0 + isWsaHolographicRemotingEnabled: 0 + enableFrameTimingStats: 0 + enableOpenGLProfilerGPURecorders: 1 + allowHDRDisplaySupport: 0 + useHDRDisplay: 0 + hdrBitDepth: 0 + m_ColorGamuts: 00000000 + targetPixelDensity: 0 + resolutionScalingMode: 0 + resetResolutionOnWindowResize: 0 + androidSupportedAspectRatio: 1 + androidMaxAspectRatio: 2.1 + applicationIdentifier: {} + buildNumber: {} + overrideDefaultApplicationIdentifier: 0 + AndroidBundleVersionCode: 1 + AndroidMinSdkVersion: 22 + AndroidTargetSdkVersion: 0 + AndroidPreferredInstallLocation: 1 + aotOptions: + stripEngineCode: 1 + iPhoneStrippingLevel: 0 + iPhoneScriptCallOptimization: 0 + ForceInternetPermission: 0 + ForceSDCardPermission: 0 + CreateWallpaper: 0 + APKExpansionFiles: 0 + keepLoadedShadersAlive: 0 + StripUnusedMeshComponents: 0 + strictShaderVariantMatching: 0 + VertexChannelCompressionMask: 4054 + iPhoneSdkVersion: 988 + iOSTargetOSVersionString: + tvOSSdkVersion: 0 + tvOSRequireExtendedGameController: 0 + tvOSTargetOSVersionString: + VisionOSSdkVersion: 0 + VisionOSTargetOSVersionString: + uIPrerenderedIcon: 0 + uIRequiresPersistentWiFi: 0 + uIRequiresFullScreen: 1 + uIStatusBarHidden: 1 + uIExitOnSuspend: 0 + uIStatusBarStyle: 0 + appleTVSplashScreen: {fileID: 0} + appleTVSplashScreen2x: {fileID: 0} + tvOSSmallIconLayers: [] + tvOSSmallIconLayers2x: [] + tvOSLargeIconLayers: [] + tvOSLargeIconLayers2x: [] + tvOSTopShelfImageLayers: [] + tvOSTopShelfImageLayers2x: [] + tvOSTopShelfImageWideLayers: [] + tvOSTopShelfImageWideLayers2x: [] + iOSLaunchScreenType: 0 + iOSLaunchScreenPortrait: {fileID: 0} + iOSLaunchScreenLandscape: {fileID: 0} + iOSLaunchScreenBackgroundColor: + serializedVersion: 2 + rgba: 0 + iOSLaunchScreenFillPct: 100 + iOSLaunchScreenSize: 100 + iOSLaunchScreenCustomXibPath: + iOSLaunchScreeniPadType: 0 + iOSLaunchScreeniPadImage: {fileID: 0} + iOSLaunchScreeniPadBackgroundColor: + serializedVersion: 2 + rgba: 0 + iOSLaunchScreeniPadFillPct: 100 + iOSLaunchScreeniPadSize: 100 + iOSLaunchScreeniPadCustomXibPath: + iOSLaunchScreenCustomStoryboardPath: + iOSLaunchScreeniPadCustomStoryboardPath: + iOSDeviceRequirements: [] + iOSURLSchemes: [] + macOSURLSchemes: [] + iOSBackgroundModes: 0 + iOSMetalForceHardShadows: 0 + metalEditorSupport: 1 + metalAPIValidation: 1 + metalCompileShaderBinary: 0 + iOSRenderExtraFrameOnPause: 0 + iosCopyPluginsCodeInsteadOfSymlink: 0 + appleDeveloperTeamID: + iOSManualSigningProvisioningProfileID: + tvOSManualSigningProvisioningProfileID: + VisionOSManualSigningProvisioningProfileID: + iOSManualSigningProvisioningProfileType: 0 + tvOSManualSigningProvisioningProfileType: 0 + VisionOSManualSigningProvisioningProfileType: 0 + appleEnableAutomaticSigning: 0 + iOSRequireARKit: 0 + iOSAutomaticallyDetectAndAddCapabilities: 1 + appleEnableProMotion: 0 + shaderPrecisionModel: 0 + clonedFromGUID: 00000000000000000000000000000000 + templatePackageId: + templateDefaultScene: + useCustomMainManifest: 0 + useCustomLauncherManifest: 0 + useCustomMainGradleTemplate: 0 + useCustomLauncherGradleManifest: 0 + useCustomBaseGradleTemplate: 0 + useCustomGradlePropertiesTemplate: 0 + useCustomGradleSettingsTemplate: 0 + useCustomProguardFile: 0 + AndroidTargetArchitectures: 1 + AndroidTargetDevices: 0 + AndroidSplashScreenScale: 0 + androidSplashScreen: {fileID: 0} + AndroidKeystoreName: + AndroidKeyaliasName: + AndroidEnableArmv9SecurityFeatures: 0 + AndroidBuildApkPerCpuArchitecture: 0 + AndroidTVCompatibility: 0 + AndroidIsGame: 1 + AndroidEnableTango: 0 + androidEnableBanner: 1 + androidUseLowAccuracyLocation: 0 + androidUseCustomKeystore: 0 + m_AndroidBanners: + - width: 320 + height: 180 + banner: {fileID: 0} + androidGamepadSupportLevel: 0 + chromeosInputEmulation: 1 + AndroidMinifyRelease: 0 + AndroidMinifyDebug: 0 + AndroidValidateAppBundleSize: 1 + AndroidAppBundleSizeToValidate: 200 + m_BuildTargetIcons: [] + m_BuildTargetPlatformIcons: [] + m_BuildTargetBatching: [] + m_BuildTargetShaderSettings: [] + m_BuildTargetGraphicsJobs: [] + m_BuildTargetGraphicsJobMode: [] + m_BuildTargetGraphicsAPIs: [] + m_BuildTargetVRSettings: [] + m_DefaultShaderChunkSizeInMB: 16 + m_DefaultShaderChunkCount: 0 + openGLRequireES31: 0 + openGLRequireES31AEP: 0 + openGLRequireES32: 0 + m_TemplateCustomTags: {} + mobileMTRendering: + Android: 1 + VisionOS: 1 + iPhone: 1 + tvOS: 1 + m_BuildTargetGroupLightmapEncodingQuality: [] + m_BuildTargetGroupHDRCubemapEncodingQuality: [] + m_BuildTargetGroupLightmapSettings: [] + m_BuildTargetGroupLoadStoreDebugModeSettings: [] + m_BuildTargetNormalMapEncoding: [] + m_BuildTargetDefaultTextureCompressionFormat: [] + playModeTestRunnerEnabled: 0 + runPlayModeTestAsEditModeTest: 0 + actionOnDotNetUnhandledException: 1 + enableInternalProfiler: 0 + logObjCUncaughtExceptions: 1 + enableCrashReportAPI: 0 + cameraUsageDescription: + locationUsageDescription: + microphoneUsageDescription: + bluetoothUsageDescription: + macOSTargetOSVersion: + switchNMETAOverride: + switchNetLibKey: + switchSocketMemoryPoolSize: 6144 + switchSocketAllocatorPoolSize: 128 + switchSocketConcurrencyLimit: 14 + switchScreenResolutionBehavior: 2 + switchUseCPUProfiler: 0 + switchEnableFileSystemTrace: 0 + switchLTOSetting: 0 + switchApplicationID: 0x01004b9000490000 + switchNSODependencies: + switchCompilerFlags: + switchTitleNames_0: + switchTitleNames_1: + switchTitleNames_2: + switchTitleNames_3: + switchTitleNames_4: + switchTitleNames_5: + switchTitleNames_6: + switchTitleNames_7: + switchTitleNames_8: + switchTitleNames_9: + switchTitleNames_10: + switchTitleNames_11: + switchTitleNames_12: + switchTitleNames_13: + switchTitleNames_14: + switchTitleNames_15: + switchPublisherNames_0: + switchPublisherNames_1: + switchPublisherNames_2: + switchPublisherNames_3: + switchPublisherNames_4: + switchPublisherNames_5: + switchPublisherNames_6: + switchPublisherNames_7: + switchPublisherNames_8: + switchPublisherNames_9: + switchPublisherNames_10: + switchPublisherNames_11: + switchPublisherNames_12: + switchPublisherNames_13: + switchPublisherNames_14: + switchPublisherNames_15: + switchIcons_0: {fileID: 0} + switchIcons_1: {fileID: 0} + switchIcons_2: {fileID: 0} + switchIcons_3: {fileID: 0} + switchIcons_4: {fileID: 0} + switchIcons_5: {fileID: 0} + switchIcons_6: {fileID: 0} + switchIcons_7: {fileID: 0} + switchIcons_8: {fileID: 0} + switchIcons_9: {fileID: 0} + switchIcons_10: {fileID: 0} + switchIcons_11: {fileID: 0} + switchIcons_12: {fileID: 0} + switchIcons_13: {fileID: 0} + switchIcons_14: {fileID: 0} + switchIcons_15: {fileID: 0} + switchSmallIcons_0: {fileID: 0} + switchSmallIcons_1: {fileID: 0} + switchSmallIcons_2: {fileID: 0} + switchSmallIcons_3: {fileID: 0} + switchSmallIcons_4: {fileID: 0} + switchSmallIcons_5: {fileID: 0} + switchSmallIcons_6: {fileID: 0} + switchSmallIcons_7: {fileID: 0} + switchSmallIcons_8: {fileID: 0} + switchSmallIcons_9: {fileID: 0} + switchSmallIcons_10: {fileID: 0} + switchSmallIcons_11: {fileID: 0} + switchSmallIcons_12: {fileID: 0} + switchSmallIcons_13: {fileID: 0} + switchSmallIcons_14: {fileID: 0} + switchSmallIcons_15: {fileID: 0} + switchManualHTML: + switchAccessibleURLs: + switchLegalInformation: + switchMainThreadStackSize: 1048576 + switchPresenceGroupId: + switchLogoHandling: 0 + switchReleaseVersion: 0 + switchDisplayVersion: 1.0.0 + switchStartupUserAccount: 0 + switchSupportedLanguagesMask: 0 + switchLogoType: 0 + switchApplicationErrorCodeCategory: + switchUserAccountSaveDataSize: 0 + switchUserAccountSaveDataJournalSize: 0 + switchApplicationAttribute: 0 + switchCardSpecSize: -1 + switchCardSpecClock: -1 + switchRatingsMask: 0 + switchRatingsInt_0: 0 + switchRatingsInt_1: 0 + switchRatingsInt_2: 0 + switchRatingsInt_3: 0 + switchRatingsInt_4: 0 + switchRatingsInt_5: 0 + switchRatingsInt_6: 0 + switchRatingsInt_7: 0 + switchRatingsInt_8: 0 + switchRatingsInt_9: 0 + switchRatingsInt_10: 0 + switchRatingsInt_11: 0 + switchRatingsInt_12: 0 + switchLocalCommunicationIds_0: + switchLocalCommunicationIds_1: + switchLocalCommunicationIds_2: + switchLocalCommunicationIds_3: + switchLocalCommunicationIds_4: + switchLocalCommunicationIds_5: + switchLocalCommunicationIds_6: + switchLocalCommunicationIds_7: + switchParentalControl: 0 + switchAllowsScreenshot: 1 + switchAllowsVideoCapturing: 1 + switchAllowsRuntimeAddOnContentInstall: 0 + switchDataLossConfirmation: 0 + switchUserAccountLockEnabled: 0 + switchSystemResourceMemory: 16777216 + switchSupportedNpadStyles: 22 + switchNativeFsCacheSize: 32 + switchIsHoldTypeHorizontal: 1 + switchSupportedNpadCount: 8 + switchEnableTouchScreen: 1 + switchSocketConfigEnabled: 0 + switchTcpInitialSendBufferSize: 32 + switchTcpInitialReceiveBufferSize: 64 + switchTcpAutoSendBufferSizeMax: 256 + switchTcpAutoReceiveBufferSizeMax: 256 + switchUdpSendBufferSize: 9 + switchUdpReceiveBufferSize: 42 + switchSocketBufferEfficiency: 4 + switchSocketInitializeEnabled: 1 + switchNetworkInterfaceManagerInitializeEnabled: 1 + switchUseNewStyleFilepaths: 1 + switchUseLegacyFmodPriorities: 0 + switchUseMicroSleepForYield: 1 + switchEnableRamDiskSupport: 0 + switchMicroSleepForYieldTime: 25 + switchRamDiskSpaceSize: 12 + ps4NPAgeRating: 12 + ps4NPTitleSecret: + ps4NPTrophyPackPath: + ps4ParentalLevel: 11 + ps4ContentID: ED1633-NPXX51362_00-0000000000000000 + ps4Category: 0 + ps4MasterVersion: 01.00 + ps4AppVersion: 01.00 + ps4AppType: 0 + ps4ParamSfxPath: + ps4VideoOutPixelFormat: 0 + ps4VideoOutInitialWidth: 1920 + ps4VideoOutBaseModeInitialWidth: 1920 + ps4VideoOutReprojectionRate: 60 + ps4PronunciationXMLPath: + ps4PronunciationSIGPath: + ps4BackgroundImagePath: + ps4StartupImagePath: + ps4StartupImagesFolder: + ps4IconImagesFolder: + ps4SaveDataImagePath: + ps4SdkOverride: + ps4BGMPath: + ps4ShareFilePath: + ps4ShareOverlayImagePath: + ps4PrivacyGuardImagePath: + ps4ExtraSceSysFile: + ps4NPtitleDatPath: + ps4RemotePlayKeyAssignment: -1 + ps4RemotePlayKeyMappingDir: + ps4PlayTogetherPlayerCount: 0 + ps4EnterButtonAssignment: 2 + ps4ApplicationParam1: 0 + ps4ApplicationParam2: 0 + ps4ApplicationParam3: 0 + ps4ApplicationParam4: 0 + ps4DownloadDataSize: 0 + ps4GarlicHeapSize: 2048 + ps4ProGarlicHeapSize: 2560 + playerPrefsMaxSize: 32768 + ps4Passcode: frAQBc8Wsa1xVPfvJcrgRYwTiizs2trQ + ps4pnSessions: 1 + ps4pnPresence: 1 + ps4pnFriends: 1 + ps4pnGameCustomData: 1 + playerPrefsSupport: 0 + enableApplicationExit: 0 + resetTempFolder: 1 + restrictedAudioUsageRights: 0 + ps4UseResolutionFallback: 0 + ps4ReprojectionSupport: 0 + ps4UseAudio3dBackend: 0 + ps4UseLowGarlicFragmentationMode: 1 + ps4SocialScreenEnabled: 0 + ps4ScriptOptimizationLevel: 2 + ps4Audio3dVirtualSpeakerCount: 14 + ps4attribCpuUsage: 0 + ps4PatchPkgPath: + ps4PatchLatestPkgPath: + ps4PatchChangeinfoPath: + ps4PatchDayOne: 0 + ps4attribUserManagement: 0 + ps4attribMoveSupport: 0 + ps4attrib3DSupport: 0 + ps4attribShareSupport: 0 + ps4attribExclusiveVR: 0 + ps4disableAutoHideSplash: 0 + ps4videoRecordingFeaturesUsed: 0 + ps4contentSearchFeaturesUsed: 0 + ps4CompatibilityPS5: 0 + ps4AllowPS5Detection: 0 + ps4GPU800MHz: 1 + ps4attribEyeToEyeDistanceSettingVR: 0 + ps4IncludedModules: [] + ps4attribVROutputEnabled: 0 + monoEnv: + splashScreenBackgroundSourceLandscape: {fileID: 0} + splashScreenBackgroundSourcePortrait: {fileID: 0} + blurSplashScreenBackground: 1 + spritePackerPolicy: + webGLMemorySize: 32 + webGLExceptionSupport: 1 + webGLNameFilesAsHashes: 0 + webGLShowDiagnostics: 0 + webGLDataCaching: 1 + webGLDebugSymbols: 0 + webGLEmscriptenArgs: + webGLModulesDirectory: + webGLTemplate: APPLICATION:Default + webGLAnalyzeBuildSize: 0 + webGLUseEmbeddedResources: 0 + webGLCompressionFormat: 1 + webGLWasmArithmeticExceptions: 0 + webGLLinkerTarget: 1 + webGLThreadsSupport: 0 + webGLDecompressionFallback: 0 + webGLInitialMemorySize: 32 + webGLMaximumMemorySize: 2048 + webGLMemoryGrowthMode: 2 + webGLMemoryLinearGrowthStep: 16 + webGLMemoryGeometricGrowthStep: 0.2 + webGLMemoryGeometricGrowthCap: 96 + webGLPowerPreference: 2 + scriptingDefineSymbols: {} + additionalCompilerArguments: {} + platformArchitecture: {} + scriptingBackend: {} + il2cppCompilerConfiguration: {} + il2cppCodeGeneration: {} + managedStrippingLevel: {} + incrementalIl2cppBuild: {} + suppressCommonWarnings: 1 + allowUnsafeCode: 0 + useDeterministicCompilation: 1 + additionalIl2CppArgs: + scriptingRuntimeVersion: 1 + gcIncremental: 1 + gcWBarrierValidation: 0 + apiCompatibilityLevelPerPlatform: {} + m_RenderingPath: 1 + m_MobileRenderingPath: 1 + metroPackageName: AxibugEmuOnline.Client + metroPackageVersion: + metroCertificatePath: + metroCertificatePassword: + metroCertificateSubject: + metroCertificateIssuer: + metroCertificateNotAfter: 0000000000000000 + metroApplicationDescription: AxibugEmuOnline.Client + wsaImages: {} + metroTileShortName: + metroTileShowName: 0 + metroMediumTileShowName: 0 + metroLargeTileShowName: 0 + metroWideTileShowName: 0 + metroSupportStreamingInstall: 0 + metroLastRequiredScene: 0 + metroDefaultTileSize: 1 + metroTileForegroundText: 2 + metroTileBackgroundColor: {r: 0.13333334, g: 0.17254902, b: 0.21568628, a: 0} + metroSplashScreenBackgroundColor: {r: 0.12941177, g: 0.17254902, b: 0.21568628, a: 1} + metroSplashScreenUseBackgroundColor: 0 + syncCapabilities: 0 + platformCapabilities: {} + metroTargetDeviceFamilies: {} + metroFTAName: + metroFTAFileTypes: [] + metroProtocolName: + vcxProjDefaultLanguage: + XboxOneProductId: + XboxOneUpdateKey: + XboxOneSandboxId: + XboxOneContentId: + XboxOneTitleId: + XboxOneSCId: + XboxOneGameOsOverridePath: + XboxOnePackagingOverridePath: + XboxOneAppManifestOverridePath: + XboxOneVersion: 1.0.0.0 + XboxOnePackageEncryption: 0 + XboxOnePackageUpdateGranularity: 2 + XboxOneDescription: + XboxOneLanguage: + - enus + XboxOneCapability: [] + XboxOneGameRating: {} + XboxOneIsContentPackage: 0 + XboxOneEnhancedXboxCompatibilityMode: 0 + XboxOneEnableGPUVariability: 1 + XboxOneSockets: {} + XboxOneSplashScreen: {fileID: 0} + XboxOneAllowedProductIds: [] + XboxOnePersistentLocalStorageSize: 0 + XboxOneXTitleMemory: 8 + XboxOneOverrideIdentityName: + XboxOneOverrideIdentityPublisher: + vrEditorSettings: {} + cloudServicesEnabled: {} + luminIcon: + m_Name: + m_ModelFolderPath: + m_PortalFolderPath: + luminCert: + m_CertPath: + m_SignPackage: 1 + luminIsChannelApp: 0 + luminVersion: + m_VersionCode: 1 + m_VersionName: + hmiPlayerDataPath: + hmiForceSRGBBlit: 1 + embeddedLinuxEnableGamepadInput: 1 + hmiLogStartupTiming: 0 + hmiCpuConfiguration: + apiCompatibilityLevel: 6 + activeInputHandler: 0 + windowsGamepadBackendHint: 0 + cloudProjectId: + framebufferDepthMemorylessMode: 0 + qualitySettingsNames: [] + projectName: + organizationId: + cloudEnabled: 0 + legacyClampBlendShapeWeights: 0 + hmiLoadingImage: {fileID: 0} + platformRequiresReadableAssets: 0 + virtualTexturingSupportEnabled: 0 + insecureHttpOption: 0 diff --git a/AxibugEmuOnline.Client/ProjectSettings/ProjectVersion.txt b/AxibugEmuOnline.Client/ProjectSettings/ProjectVersion.txt new file mode 100644 index 00000000..0eb48a5a --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/ProjectVersion.txt @@ -0,0 +1,2 @@ +m_EditorVersion: 2022.3.30f1c1 +m_EditorVersionWithRevision: 2022.3.30f1c1 (a6dc00fbe362) diff --git a/AxibugEmuOnline.Client/ProjectSettings/QualitySettings.asset b/AxibugEmuOnline.Client/ProjectSettings/QualitySettings.asset new file mode 100644 index 00000000..bb69fbfb --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/QualitySettings.asset @@ -0,0 +1,322 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!47 &1 +QualitySettings: + m_ObjectHideFlags: 0 + serializedVersion: 5 + m_CurrentQuality: 5 + m_QualitySettings: + - serializedVersion: 3 + name: Very Low + pixelLightCount: 0 + shadows: 0 + shadowResolution: 0 + shadowProjection: 1 + shadowCascades: 1 + shadowDistance: 15 + shadowNearPlaneOffset: 3 + shadowCascade2Split: 0.33333334 + shadowCascade4Split: {x: 0.06666667, y: 0.2, z: 0.46666667} + shadowmaskMode: 0 + skinWeights: 1 + globalTextureMipmapLimit: 1 + textureMipmapLimitSettings: [] + anisotropicTextures: 0 + antiAliasing: 0 + softParticles: 0 + softVegetation: 0 + realtimeReflectionProbes: 0 + billboardsFaceCameraPosition: 0 + useLegacyDetailDistribution: 0 + vSyncCount: 0 + realtimeGICPUUsage: 25 + lodBias: 0.3 + maximumLODLevel: 0 + enableLODCrossFade: 1 + streamingMipmapsActive: 0 + streamingMipmapsAddAllCameras: 1 + streamingMipmapsMemoryBudget: 512 + streamingMipmapsRenderersPerFrame: 512 + streamingMipmapsMaxLevelReduction: 2 + streamingMipmapsMaxFileIORequests: 1024 + particleRaycastBudget: 4 + asyncUploadTimeSlice: 2 + asyncUploadBufferSize: 16 + asyncUploadPersistentBuffer: 1 + resolutionScalingFixedDPIFactor: 1 + customRenderPipeline: {fileID: 0} + terrainQualityOverrides: 0 + terrainPixelError: 1 + terrainDetailDensityScale: 1 + terrainBasemapDistance: 1000 + terrainDetailDistance: 80 + terrainTreeDistance: 5000 + terrainBillboardStart: 50 + terrainFadeLength: 5 + terrainMaxTrees: 50 + excludedTargetPlatforms: [] + - serializedVersion: 3 + name: Low + pixelLightCount: 0 + shadows: 0 + shadowResolution: 0 + shadowProjection: 1 + shadowCascades: 1 + shadowDistance: 20 + shadowNearPlaneOffset: 3 + shadowCascade2Split: 0.33333334 + shadowCascade4Split: {x: 0.06666667, y: 0.2, z: 0.46666667} + shadowmaskMode: 0 + skinWeights: 2 + globalTextureMipmapLimit: 0 + textureMipmapLimitSettings: [] + anisotropicTextures: 0 + antiAliasing: 0 + softParticles: 0 + softVegetation: 0 + realtimeReflectionProbes: 0 + billboardsFaceCameraPosition: 0 + useLegacyDetailDistribution: 0 + vSyncCount: 0 + realtimeGICPUUsage: 25 + lodBias: 0.4 + maximumLODLevel: 0 + enableLODCrossFade: 1 + streamingMipmapsActive: 0 + streamingMipmapsAddAllCameras: 1 + streamingMipmapsMemoryBudget: 512 + streamingMipmapsRenderersPerFrame: 512 + streamingMipmapsMaxLevelReduction: 2 + streamingMipmapsMaxFileIORequests: 1024 + particleRaycastBudget: 16 + asyncUploadTimeSlice: 2 + asyncUploadBufferSize: 16 + asyncUploadPersistentBuffer: 1 + resolutionScalingFixedDPIFactor: 1 + customRenderPipeline: {fileID: 0} + terrainQualityOverrides: 0 + terrainPixelError: 1 + terrainDetailDensityScale: 1 + terrainBasemapDistance: 1000 + terrainDetailDistance: 80 + terrainTreeDistance: 5000 + terrainBillboardStart: 50 + terrainFadeLength: 5 + terrainMaxTrees: 50 + excludedTargetPlatforms: [] + - serializedVersion: 3 + name: Medium + pixelLightCount: 1 + shadows: 1 + shadowResolution: 0 + shadowProjection: 1 + shadowCascades: 1 + shadowDistance: 20 + shadowNearPlaneOffset: 3 + shadowCascade2Split: 0.33333334 + shadowCascade4Split: {x: 0.06666667, y: 0.2, z: 0.46666667} + shadowmaskMode: 0 + skinWeights: 2 + globalTextureMipmapLimit: 0 + textureMipmapLimitSettings: [] + anisotropicTextures: 1 + antiAliasing: 0 + softParticles: 0 + softVegetation: 0 + realtimeReflectionProbes: 0 + billboardsFaceCameraPosition: 0 + useLegacyDetailDistribution: 0 + vSyncCount: 1 + realtimeGICPUUsage: 25 + lodBias: 0.7 + maximumLODLevel: 0 + enableLODCrossFade: 1 + streamingMipmapsActive: 0 + streamingMipmapsAddAllCameras: 1 + streamingMipmapsMemoryBudget: 512 + streamingMipmapsRenderersPerFrame: 512 + streamingMipmapsMaxLevelReduction: 2 + streamingMipmapsMaxFileIORequests: 1024 + particleRaycastBudget: 64 + asyncUploadTimeSlice: 2 + asyncUploadBufferSize: 16 + asyncUploadPersistentBuffer: 1 + resolutionScalingFixedDPIFactor: 1 + customRenderPipeline: {fileID: 0} + terrainQualityOverrides: 0 + terrainPixelError: 1 + terrainDetailDensityScale: 1 + terrainBasemapDistance: 1000 + terrainDetailDistance: 80 + terrainTreeDistance: 5000 + terrainBillboardStart: 50 + terrainFadeLength: 5 + terrainMaxTrees: 50 + excludedTargetPlatforms: [] + - serializedVersion: 3 + name: High + pixelLightCount: 2 + shadows: 2 + shadowResolution: 1 + shadowProjection: 1 + shadowCascades: 2 + shadowDistance: 40 + shadowNearPlaneOffset: 3 + shadowCascade2Split: 0.33333334 + shadowCascade4Split: {x: 0.06666667, y: 0.2, z: 0.46666667} + shadowmaskMode: 1 + skinWeights: 2 + globalTextureMipmapLimit: 0 + textureMipmapLimitSettings: [] + anisotropicTextures: 1 + antiAliasing: 0 + softParticles: 0 + softVegetation: 1 + realtimeReflectionProbes: 1 + billboardsFaceCameraPosition: 1 + useLegacyDetailDistribution: 0 + vSyncCount: 1 + realtimeGICPUUsage: 50 + lodBias: 1 + maximumLODLevel: 0 + enableLODCrossFade: 1 + streamingMipmapsActive: 0 + streamingMipmapsAddAllCameras: 1 + streamingMipmapsMemoryBudget: 512 + streamingMipmapsRenderersPerFrame: 512 + streamingMipmapsMaxLevelReduction: 2 + streamingMipmapsMaxFileIORequests: 1024 + particleRaycastBudget: 256 + asyncUploadTimeSlice: 2 + asyncUploadBufferSize: 16 + asyncUploadPersistentBuffer: 1 + resolutionScalingFixedDPIFactor: 1 + customRenderPipeline: {fileID: 0} + terrainQualityOverrides: 0 + terrainPixelError: 1 + terrainDetailDensityScale: 1 + terrainBasemapDistance: 1000 + terrainDetailDistance: 80 + terrainTreeDistance: 5000 + terrainBillboardStart: 50 + terrainFadeLength: 5 + terrainMaxTrees: 50 + excludedTargetPlatforms: [] + - serializedVersion: 3 + name: Very High + pixelLightCount: 3 + shadows: 2 + shadowResolution: 2 + shadowProjection: 1 + shadowCascades: 2 + shadowDistance: 70 + shadowNearPlaneOffset: 3 + shadowCascade2Split: 0.33333334 + shadowCascade4Split: {x: 0.06666667, y: 0.2, z: 0.46666667} + shadowmaskMode: 1 + skinWeights: 4 + globalTextureMipmapLimit: 0 + textureMipmapLimitSettings: [] + anisotropicTextures: 2 + antiAliasing: 2 + softParticles: 1 + softVegetation: 1 + realtimeReflectionProbes: 1 + billboardsFaceCameraPosition: 1 + useLegacyDetailDistribution: 0 + vSyncCount: 1 + realtimeGICPUUsage: 50 + lodBias: 1.5 + maximumLODLevel: 0 + enableLODCrossFade: 1 + streamingMipmapsActive: 0 + streamingMipmapsAddAllCameras: 1 + streamingMipmapsMemoryBudget: 512 + streamingMipmapsRenderersPerFrame: 512 + streamingMipmapsMaxLevelReduction: 2 + streamingMipmapsMaxFileIORequests: 1024 + particleRaycastBudget: 1024 + asyncUploadTimeSlice: 2 + asyncUploadBufferSize: 16 + asyncUploadPersistentBuffer: 1 + resolutionScalingFixedDPIFactor: 1 + customRenderPipeline: {fileID: 0} + terrainQualityOverrides: 0 + terrainPixelError: 1 + terrainDetailDensityScale: 1 + terrainBasemapDistance: 1000 + terrainDetailDistance: 80 + terrainTreeDistance: 5000 + terrainBillboardStart: 50 + terrainFadeLength: 5 + terrainMaxTrees: 50 + excludedTargetPlatforms: [] + - serializedVersion: 3 + name: Ultra + pixelLightCount: 4 + shadows: 2 + shadowResolution: 2 + shadowProjection: 1 + shadowCascades: 4 + shadowDistance: 150 + shadowNearPlaneOffset: 3 + shadowCascade2Split: 0.33333334 + shadowCascade4Split: {x: 0.06666667, y: 0.2, z: 0.46666667} + shadowmaskMode: 1 + skinWeights: 255 + globalTextureMipmapLimit: 0 + textureMipmapLimitSettings: [] + anisotropicTextures: 2 + antiAliasing: 2 + softParticles: 1 + softVegetation: 1 + realtimeReflectionProbes: 1 + billboardsFaceCameraPosition: 1 + useLegacyDetailDistribution: 0 + vSyncCount: 1 + realtimeGICPUUsage: 100 + lodBias: 2 + maximumLODLevel: 0 + enableLODCrossFade: 1 + streamingMipmapsActive: 0 + streamingMipmapsAddAllCameras: 1 + streamingMipmapsMemoryBudget: 512 + streamingMipmapsRenderersPerFrame: 512 + streamingMipmapsMaxLevelReduction: 2 + streamingMipmapsMaxFileIORequests: 1024 + particleRaycastBudget: 4096 + asyncUploadTimeSlice: 2 + asyncUploadBufferSize: 16 + asyncUploadPersistentBuffer: 1 + resolutionScalingFixedDPIFactor: 1 + customRenderPipeline: {fileID: 0} + terrainQualityOverrides: 0 + terrainPixelError: 1 + terrainDetailDensityScale: 1 + terrainBasemapDistance: 1000 + terrainDetailDistance: 80 + terrainTreeDistance: 5000 + terrainBillboardStart: 50 + terrainFadeLength: 5 + terrainMaxTrees: 50 + excludedTargetPlatforms: [] + m_TextureMipmapLimitGroupNames: [] + m_PerPlatformDefaultQuality: + Android: 2 + EmbeddedLinux: 5 + GameCoreScarlett: 5 + GameCoreXboxOne: 5 + LinuxHeadlessSimulation: 5 + Nintendo Switch: 5 + PS4: 5 + PS5: 5 + QNX: 5 + Server: 5 + Stadia: 5 + Standalone: 5 + VisionOS: 5 + WebGL: 3 + Windows Store Apps: 5 + XboxOne: 5 + iPhone: 2 + tvOS: 2 diff --git a/AxibugEmuOnline.Client/ProjectSettings/TagManager.asset b/AxibugEmuOnline.Client/ProjectSettings/TagManager.asset new file mode 100644 index 00000000..1c92a784 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/TagManager.asset @@ -0,0 +1,43 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!78 &1 +TagManager: + serializedVersion: 2 + tags: [] + layers: + - Default + - TransparentFX + - Ignore Raycast + - + - Water + - UI + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + - + m_SortingLayers: + - name: Default + uniqueID: 0 + locked: 0 diff --git a/AxibugEmuOnline.Client/ProjectSettings/TimeManager.asset b/AxibugEmuOnline.Client/ProjectSettings/TimeManager.asset new file mode 100644 index 00000000..558a017e --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/TimeManager.asset @@ -0,0 +1,9 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!5 &1 +TimeManager: + m_ObjectHideFlags: 0 + Fixed Timestep: 0.02 + Maximum Allowed Timestep: 0.33333334 + m_TimeScale: 1 + Maximum Particle Timestep: 0.03 diff --git a/AxibugEmuOnline.Client/ProjectSettings/UnityConnectSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/UnityConnectSettings.asset new file mode 100644 index 00000000..a27ab5f2 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/UnityConnectSettings.asset @@ -0,0 +1,38 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!310 &1 +UnityConnectSettings: + m_ObjectHideFlags: 0 + serializedVersion: 1 + m_Enabled: 0 + m_TestMode: 0 + m_EventOldUrl: https://api.uca.cloud.unity3d.com/v1/events + m_EventUrl: https://cdp.cloud.unity3d.com/v1/events + m_ConfigUrl: https://config.uca.cloud.unity3d.com + m_DashboardUrl: https://dashboard.unity3d.com + m_CNEventUrl: https://cdp.cloud.unity.cn/v1/events + m_CNConfigUrl: https://cdp.cloud.unity.cn/config + m_TestInitMode: 0 + CrashReportingSettings: + m_EventUrl: https://perf-events.cloud.unity.cn + m_Enabled: 0 + m_LogBufferSize: 10 + m_CaptureEditorExceptions: 1 + UnityPurchasingSettings: + m_Enabled: 0 + m_TestMode: 0 + UnityAnalyticsSettings: + m_Enabled: 0 + m_TestMode: 0 + m_InitializeOnStartup: 1 + m_PackageRequiringCoreStatsPresent: 0 + UnityAdsSettings: + m_Enabled: 0 + m_InitializeOnStartup: 1 + m_TestMode: 0 + m_IosGameId: + m_AndroidGameId: + m_GameIds: {} + m_GameId: + PerformanceReportingSettings: + m_Enabled: 0 diff --git a/AxibugEmuOnline.Client/ProjectSettings/VFXManager.asset b/AxibugEmuOnline.Client/ProjectSettings/VFXManager.asset new file mode 100644 index 00000000..852348bc --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/VFXManager.asset @@ -0,0 +1,18 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!937362698 &1 +VFXManager: + m_ObjectHideFlags: 0 + m_IndirectShader: {fileID: 0} + m_CopyBufferShader: {fileID: 0} + m_SortShader: {fileID: 0} + m_StripUpdateShader: {fileID: 0} + m_EmptyShader: {fileID: 0} + m_RenderPipeSettingsPath: + m_FixedTimeStep: 0.016666668 + m_MaxDeltaTime: 0.05 + m_MaxScrubTime: 30 + m_CompiledVersion: 0 + m_RuntimeVersion: 0 + m_RuntimeResources: {fileID: 0} + m_BatchEmptyLifetime: 300 diff --git a/AxibugEmuOnline.Client/ProjectSettings/VersionControlSettings.asset b/AxibugEmuOnline.Client/ProjectSettings/VersionControlSettings.asset new file mode 100644 index 00000000..dca28814 --- /dev/null +++ b/AxibugEmuOnline.Client/ProjectSettings/VersionControlSettings.asset @@ -0,0 +1,8 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!890905787 &1 +VersionControlSettings: + m_ObjectHideFlags: 0 + m_Mode: Visible Meta Files + m_CollabEditorSettings: + inProgressEnabled: 1 diff --git a/AxibugEmuOnline.Client/UserSettings/EditorUserSettings.asset b/AxibugEmuOnline.Client/UserSettings/EditorUserSettings.asset new file mode 100644 index 00000000..79d649d5 --- /dev/null +++ b/AxibugEmuOnline.Client/UserSettings/EditorUserSettings.asset @@ -0,0 +1,31 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!162 &1 +EditorUserSettings: + m_ObjectHideFlags: 0 + serializedVersion: 4 + m_ConfigSettings: + RecentlyUsedSceneGuid-0: + value: 06530405540d080d5c575d7545700744124e4d7b787f7e6975284960b2b5603a + flags: 0 + RecentlyUsedSceneGuid-1: + value: 5055560752040d595b59547547760644174e1a7f2e7f716078704f63b7e53760 + flags: 0 + vcSharedLogLevel: + value: 0d5e400f0650 + flags: 0 + m_VCAutomaticAdd: 1 + m_VCDebugCom: 0 + m_VCDebugCmd: 0 + m_VCDebugOut: 0 + m_SemanticMergeMode: 2 + m_DesiredImportWorkerCount: 4 + m_StandbyImportWorkerCount: 2 + m_IdleImportWorkerShutdownDelay: 60000 + m_VCShowFailedCheckout: 1 + m_VCOverwriteFailedCheckoutAssets: 1 + m_VCProjectOverlayIcons: 1 + m_VCHierarchyOverlayIcons: 1 + m_VCOtherOverlayIcons: 1 + m_VCAllowAsyncUpdate: 1 + m_ArtifactGarbageCollection: 1 diff --git a/AxibugEmuOnline.Client/UserSettings/Layouts/default-2022.dwlt b/AxibugEmuOnline.Client/UserSettings/Layouts/default-2022.dwlt new file mode 100644 index 00000000..0c6024b6 --- /dev/null +++ b/AxibugEmuOnline.Client/UserSettings/Layouts/default-2022.dwlt @@ -0,0 +1,1268 @@ +%YAML 1.1 +%TAG !u! tag:unity3d.com,2011: +--- !u!114 &1 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12004, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_PixelRect: + serializedVersion: 2 + x: 0 + y: 43 + width: 1920 + height: 989 + m_ShowMode: 4 + m_Title: Console + m_RootView: {fileID: 9} + m_MinSize: {x: 875, y: 300} + m_MaxSize: {x: 10000, y: 10000} + m_Maximized: 1 +--- !u!114 &2 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 12006, guid: 0000000000000000e000000000000000, type: 0} + m_Name: ConsoleWindow + m_EditorClassIdentifier: + m_Children: [] + m_Position: + serializedVersion: 2 + x: 0 + y: 580 + width: 1175 + height: 359 + m_MinSize: {x: 101, y: 121} + m_MaxSize: {x: 4001, y: 4021} + m_ActualView: {fileID: 19} + m_Panes: + - {fileID: 19} + - {fileID: 17} + m_Selected: 0 + m_LastSelected: 1 +--- !u!114 &3 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 12010, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: + - {fileID: 13} + - {fileID: 2} + m_Position: + serializedVersion: 2 + x: 0 + y: 0 + width: 1175 + height: 939 + m_MinSize: {x: 100, y: 100} + m_MaxSize: {x: 8096, y: 16192} + vertical: 1 + controlID: 16 +--- !u!114 &4 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 12006, guid: 0000000000000000e000000000000000, type: 0} + m_Name: ProjectBrowser + m_EditorClassIdentifier: + m_Children: [] + m_Position: + serializedVersion: 2 + x: 0 + y: 579 + width: 417 + height: 360 + m_MinSize: {x: 232, y: 271} + m_MaxSize: {x: 10002, y: 10021} + m_ActualView: {fileID: 14} + m_Panes: + - {fileID: 14} + m_Selected: 0 + m_LastSelected: 0 +--- !u!114 &5 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 12010, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: + - {fileID: 6} + - {fileID: 4} + m_Position: + serializedVersion: 2 + x: 1175 + y: 0 + width: 417 + height: 939 + m_MinSize: {x: 100, y: 100} + m_MaxSize: {x: 8096, y: 16192} + vertical: 1 + controlID: 77 +--- !u!114 &6 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 0 + m_Script: {fileID: 12006, guid: 0000000000000000e000000000000000, type: 0} + m_Name: SceneHierarchyWindow + m_EditorClassIdentifier: + m_Children: [] + m_Position: + serializedVersion: 2 + x: 0 + y: 0 + width: 417 + height: 579 + m_MinSize: {x: 200, y: 200} + m_MaxSize: {x: 4000, y: 4000} + m_ActualView: {fileID: 16} + m_Panes: + - {fileID: 16} + m_Selected: 0 + m_LastSelected: 0 +--- !u!114 &7 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12010, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: + - {fileID: 12} + - {fileID: 8} + m_Position: + serializedVersion: 2 + x: 0 + y: 30 + width: 1920 + height: 939 + m_MinSize: {x: 300, y: 100} + m_MaxSize: {x: 24288, y: 16192} + vertical: 0 + controlID: 140 +--- !u!114 &8 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12006, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: [] + m_Position: + serializedVersion: 2 + x: 1592 + y: 0 + width: 328 + height: 939 + m_MinSize: {x: 275, y: 50} + m_MaxSize: {x: 4000, y: 4000} + m_ActualView: {fileID: 15} + m_Panes: + - {fileID: 15} + m_Selected: 0 + m_LastSelected: 0 +--- !u!114 &9 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12008, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: + - {fileID: 10} + - {fileID: 7} + - {fileID: 11} + m_Position: + serializedVersion: 2 + x: 0 + y: 0 + width: 1920 + height: 989 + m_MinSize: {x: 875, y: 300} + m_MaxSize: {x: 10000, y: 10000} + m_UseTopView: 1 + m_TopViewHeight: 30 + m_UseBottomView: 1 + m_BottomViewHeight: 20 +--- !u!114 &10 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12011, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: [] + m_Position: + serializedVersion: 2 + x: 0 + y: 0 + width: 1920 + height: 30 + m_MinSize: {x: 0, y: 0} + m_MaxSize: {x: 0, y: 0} + m_LastLoadedLayoutName: +--- !u!114 &11 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12042, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: [] + m_Position: + serializedVersion: 2 + x: 0 + y: 969 + width: 1920 + height: 20 + m_MinSize: {x: 0, y: 0} + m_MaxSize: {x: 0, y: 0} +--- !u!114 &12 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12010, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_Children: + - {fileID: 3} + - {fileID: 5} + m_Position: + serializedVersion: 2 + x: 0 + y: 0 + width: 1592 + height: 939 + m_MinSize: {x: 200, y: 100} + m_MaxSize: {x: 16192, y: 16192} + vertical: 0 + controlID: 76 +--- !u!114 &13 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12006, guid: 0000000000000000e000000000000000, type: 0} + m_Name: GameView + m_EditorClassIdentifier: + m_Children: [] + m_Position: + serializedVersion: 2 + x: 0 + y: 0 + width: 1175 + height: 580 + m_MinSize: {x: 201, y: 221} + m_MaxSize: {x: 4001, y: 4021} + m_ActualView: {fileID: 18} + m_Panes: + - {fileID: 18} + m_Selected: 0 + m_LastSelected: 0 +--- !u!114 &14 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12014, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_MinSize: {x: 230, y: 250} + m_MaxSize: {x: 10000, y: 10000} + m_TitleContent: + m_Text: Project + m_Image: {fileID: -5467254957812901981, guid: 0000000000000000d000000000000000, type: 0} + m_Tooltip: + m_Pos: + serializedVersion: 2 + x: 1175 + y: 652 + width: 415 + height: 339 + m_SerializedDataModeController: + m_DataMode: 0 + m_PreferredDataMode: 0 + m_SupportedDataModes: + isAutomatic: 1 + m_ViewDataDictionary: {fileID: 0} + m_OverlayCanvas: + m_LastAppliedPresetName: Default + m_SaveData: [] + m_OverlaysVisible: 1 + m_SearchFilter: + m_NameFilter: + m_ClassNames: [] + m_AssetLabels: [] + m_AssetBundleNames: [] + m_ReferencingInstanceIDs: + m_SceneHandles: + m_ShowAllHits: 0 + m_SkipHidden: 0 + m_SearchArea: 1 + m_Folders: + - Assets + m_Globs: [] + m_OriginalText: + m_ImportLogFlags: 0 + m_FilterByTypeIntersection: 0 + m_ViewMode: 1 + m_StartGridSize: 16 + m_LastFolders: + - Assets + m_LastFoldersGridSize: 16 + m_LastProjectPath: E:\AxibugEmuOnline\AxibugEmuOnline.Client + m_LockTracker: + m_IsLocked: 0 + m_FolderTreeState: + scrollPos: {x: 0, y: 0} + m_SelectedIDs: 56410000 + m_LastClickedID: 16726 + m_ExpandedIDs: 00000000564100006c41000000ca9a3b + m_RenameOverlay: + m_UserAcceptedRename: 0 + m_Name: + m_OriginalName: + m_EditFieldRect: + serializedVersion: 2 + x: 0 + y: 0 + width: 0 + height: 0 + m_UserData: 0 + m_IsWaitingForDelay: 0 + m_IsRenaming: 0 + m_OriginalEventType: 11 + m_IsRenamingFilename: 1 + m_ClientGUIView: {fileID: 4} + m_SearchString: + m_CreateAssetUtility: + m_EndAction: {fileID: 0} + m_InstanceID: 0 + m_Path: + m_Icon: {fileID: 0} + m_ResourceFile: + m_AssetTreeState: + scrollPos: {x: 0, y: 0} + m_SelectedIDs: + m_LastClickedID: 0 + m_ExpandedIDs: 0000000056410000 + m_RenameOverlay: + m_UserAcceptedRename: 0 + m_Name: + m_OriginalName: + m_EditFieldRect: + serializedVersion: 2 + x: 0 + y: 0 + width: 0 + height: 0 + m_UserData: 0 + m_IsWaitingForDelay: 0 + m_IsRenaming: 0 + m_OriginalEventType: 11 + m_IsRenamingFilename: 1 + m_ClientGUIView: {fileID: 0} + m_SearchString: + m_CreateAssetUtility: + m_EndAction: {fileID: 0} + m_InstanceID: 0 + m_Path: + m_Icon: {fileID: 0} + m_ResourceFile: + m_ListAreaState: + m_SelectedInstanceIDs: + m_LastClickedInstanceID: 0 + m_HadKeyboardFocusLastEvent: 0 + m_ExpandedInstanceIDs: c6230000 + m_RenameOverlay: + m_UserAcceptedRename: 0 + m_Name: + m_OriginalName: + m_EditFieldRect: + serializedVersion: 2 + x: 0 + y: 0 + width: 0 + height: 0 + m_UserData: 0 + m_IsWaitingForDelay: 0 + m_IsRenaming: 0 + m_OriginalEventType: 11 + m_IsRenamingFilename: 1 + m_ClientGUIView: {fileID: 4} + m_CreateAssetUtility: + m_EndAction: {fileID: 0} + m_InstanceID: 0 + m_Path: + m_Icon: {fileID: 0} + m_ResourceFile: + m_NewAssetIndexInList: -1 + m_ScrollPosition: {x: 0, y: 0} + m_GridSize: 16 + m_SkipHiddenPackages: 0 + m_DirectoriesAreaWidth: 144 +--- !u!114 &15 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12019, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_MinSize: {x: 275, y: 50} + m_MaxSize: {x: 4000, y: 4000} + m_TitleContent: + m_Text: Inspector + m_Image: {fileID: -2667387946076563598, guid: 0000000000000000d000000000000000, type: 0} + m_Tooltip: + m_Pos: + serializedVersion: 2 + x: 1592 + y: 73 + width: 327 + height: 918 + m_SerializedDataModeController: + m_DataMode: 0 + m_PreferredDataMode: 0 + m_SupportedDataModes: + isAutomatic: 1 + m_ViewDataDictionary: {fileID: 0} + m_OverlayCanvas: + m_LastAppliedPresetName: Default + m_SaveData: [] + m_OverlaysVisible: 1 + m_ObjectsLockedBeforeSerialization: [] + m_InstanceIDsLockedBeforeSerialization: + m_PreviewResizer: + m_CachedPref: 160 + m_ControlHash: -371814159 + m_PrefName: Preview_InspectorPreview + m_LastInspectedObjectInstanceID: -1 + m_LastVerticalScrollValue: 0 + m_GlobalObjectId: + m_InspectorMode: 0 + m_LockTracker: + m_IsLocked: 0 + m_PreviewWindow: {fileID: 0} +--- !u!114 &16 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12061, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_MinSize: {x: 200, y: 200} + m_MaxSize: {x: 4000, y: 4000} + m_TitleContent: + m_Text: Hierarchy + m_Image: {fileID: 7966133145522015247, guid: 0000000000000000d000000000000000, type: 0} + m_Tooltip: + m_Pos: + serializedVersion: 2 + x: 1175 + y: 73 + width: 415 + height: 558 + m_SerializedDataModeController: + m_DataMode: 0 + m_PreferredDataMode: 0 + m_SupportedDataModes: + isAutomatic: 1 + m_ViewDataDictionary: {fileID: 0} + m_OverlayCanvas: + m_LastAppliedPresetName: Default + m_SaveData: [] + m_OverlaysVisible: 1 + m_SceneHierarchy: + m_TreeViewState: + scrollPos: {x: 0, y: 0} + m_SelectedIDs: + m_LastClickedID: 0 + m_ExpandedIDs: 40fbffff + m_RenameOverlay: + m_UserAcceptedRename: 0 + m_Name: + m_OriginalName: + m_EditFieldRect: + serializedVersion: 2 + x: 0 + y: 0 + width: 0 + height: 0 + m_UserData: 0 + m_IsWaitingForDelay: 0 + m_IsRenaming: 0 + m_OriginalEventType: 11 + m_IsRenamingFilename: 0 + m_ClientGUIView: {fileID: 6} + m_SearchString: + m_ExpandedScenes: [] + m_CurrenRootInstanceID: 0 + m_LockTracker: + m_IsLocked: 0 + m_CurrentSortingName: TransformSorting + m_WindowGUID: 4c969a2b90040154d917609493e03593 +--- !u!114 &17 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12013, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_MinSize: {x: 200, y: 200} + m_MaxSize: {x: 4000, y: 4000} + m_TitleContent: + m_Text: Scene + m_Image: {fileID: 2593428753322112591, guid: 0000000000000000d000000000000000, type: 0} + m_Tooltip: + m_Pos: + serializedVersion: 2 + x: 0 + y: 653 + width: 1172 + height: 338 + m_SerializedDataModeController: + m_DataMode: 0 + m_PreferredDataMode: 0 + m_SupportedDataModes: + isAutomatic: 1 + m_ViewDataDictionary: {fileID: 0} + m_OverlayCanvas: + m_LastAppliedPresetName: Default + m_SaveData: + - dockPosition: 0 + containerId: overlay-toolbar__top + floating: 0 + collapsed: 1 + displayed: 1 + snapOffset: {x: -24, y: -24} + snapOffsetDelta: {x: -33, y: -2} + snapCorner: 3 + id: Tool Settings + index: 1 + layout: 1 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-toolbar__top + floating: 0 + collapsed: 1 + displayed: 1 + snapOffset: {x: -141, y: -189} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 3 + id: unity-grid-and-snap-toolbar + index: 2 + layout: 1 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-toolbar__top + floating: 0 + collapsed: 0 + displayed: 1 + snapOffset: {x: 24, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: unity-scene-view-toolbar + index: 0 + layout: 1 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-toolbar__top + floating: 0 + collapsed: 1 + displayed: 1 + snapOffset: {x: -343, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 1 + id: unity-scene-view-camera-mode-toolbar + index: 4 + layout: 1 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-toolbar__top + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 1 + id: unity-search-toolbar + index: 1 + layout: 1 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-toolbar__top + floating: 0 + collapsed: 0 + displayed: 1 + snapOffset: {x: 114, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: unity-transform-toolbar + index: 0 + layout: 2 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-container--left + floating: 0 + collapsed: 0 + displayed: 1 + snapOffset: {x: 0, y: 197} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: unity-component-tools + index: 1 + layout: 2 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 1 + snapOffset: {x: 67.5, y: 86} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Orientation + index: 0 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Light Settings + index: 0 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Camera + index: 1 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Cloth Constraints + index: 1 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Cloth Collisions + index: 2 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Navmesh Display + index: 4 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Agent Display + index: 5 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Obstacle Display + index: 6 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Occlusion Culling + index: 3 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Physics Debugger + index: 4 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Scene Visibility + index: 5 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Particles + index: 6 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Tilemap + index: 11 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Tilemap Palette Helper + index: 12 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-toolbar__top + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 24, y: 0} + snapCorner: 0 + id: Brush Attributes + index: 3 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-toolbar__left + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 24, y: 0} + snapCorner: 0 + id: Terrain Tools + index: 0 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 0 + containerId: overlay-toolbar__left + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 24, y: 0} + snapCorner: 0 + id: Brush Masks + index: 1 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--left + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 24, y: 0} + snapCorner: 0 + id: Scene View/Lighting Visualization Colors + index: 0 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--left + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 24, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Overlays/OverlayMenu + index: 1 + layout: 1 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: APV Overlay + index: 7 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 24, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: SceneView/CamerasOverlay + index: 8 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 24, y: 0} + snapCorner: 0 + id: Scene View/PBR Validation Settings + index: 9 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 0, y: 0} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/TrailRenderer + index: 7 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 48, y: 48} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Open Tile Palette + index: 0 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 48, y: 48} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Tilemap Focus + index: 1 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + - dockPosition: 1 + containerId: overlay-container--right + floating: 0 + collapsed: 0 + displayed: 0 + snapOffset: {x: 48, y: 48} + snapOffsetDelta: {x: 0, y: 0} + snapCorner: 0 + id: Scene View/Path + index: 10 + layout: 4 + size: {x: 0, y: 0} + sizeOverriden: 0 + m_OverlaysVisible: 1 + m_WindowGUID: cc27987af1a868c49b0894db9c0f5429 + m_Gizmos: 1 + m_OverrideSceneCullingMask: 6917529027641081856 + m_SceneIsLit: 1 + m_SceneLighting: 1 + m_2DMode: 0 + m_isRotationLocked: 0 + m_PlayAudio: 0 + m_AudioPlay: 0 + m_Position: + m_Target: {x: 0, y: 0, z: 0} + speed: 2 + m_Value: {x: 0, y: 0, z: 0} + m_RenderMode: 0 + m_CameraMode: + drawMode: 0 + name: Shaded + section: Shading Mode + m_ValidateTrueMetals: 0 + m_DoValidateTrueMetals: 0 + m_SceneViewState: + m_AlwaysRefresh: 0 + showFog: 1 + showSkybox: 1 + showFlares: 1 + showImageEffects: 1 + showParticleSystems: 1 + showVisualEffectGraphs: 1 + m_FxEnabled: 1 + m_Grid: + xGrid: + m_Fade: + m_Target: 0 + speed: 2 + m_Value: 0 + m_Color: {r: 0.5, g: 0.5, b: 0.5, a: 0.4} + m_Pivot: {x: 0, y: 0, z: 0} + m_Size: {x: 0, y: 0} + yGrid: + m_Fade: + m_Target: 1 + speed: 2 + m_Value: 1 + m_Color: {r: 0.5, g: 0.5, b: 0.5, a: 0.4} + m_Pivot: {x: 0, y: 0, z: 0} + m_Size: {x: 1, y: 1} + zGrid: + m_Fade: + m_Target: 0 + speed: 2 + m_Value: 0 + m_Color: {r: 0.5, g: 0.5, b: 0.5, a: 0.4} + m_Pivot: {x: 0, y: 0, z: 0} + m_Size: {x: 1, y: 1} + m_ShowGrid: 1 + m_GridAxis: 1 + m_gridOpacity: 0.5 + m_Rotation: + m_Target: {x: -0.08717229, y: 0.89959055, z: -0.21045254, w: -0.3726226} + speed: 2 + m_Value: {x: -0.08717229, y: 0.89959055, z: -0.21045254, w: -0.3726226} + m_Size: + m_Target: 10 + speed: 2 + m_Value: 10 + m_Ortho: + m_Target: 0 + speed: 2 + m_Value: 0 + m_CameraSettings: + m_Speed: 1 + m_SpeedNormalized: 0.5 + m_SpeedMin: 0.001 + m_SpeedMax: 2 + m_EasingEnabled: 1 + m_EasingDuration: 0.4 + m_AccelerationEnabled: 1 + m_FieldOfViewHorizontalOrVertical: 60 + m_NearClip: 0.03 + m_FarClip: 10000 + m_DynamicClip: 1 + m_OcclusionCulling: 0 + m_LastSceneViewRotation: {x: -0.08717229, y: 0.89959055, z: -0.21045254, w: -0.3726226} + m_LastSceneViewOrtho: 0 + m_ReplacementShader: {fileID: 0} + m_ReplacementString: + m_SceneVisActive: 1 + m_LastLockedObject: {fileID: 0} + m_ViewIsLockedToObject: 0 +--- !u!114 &18 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12015, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_MinSize: {x: 200, y: 200} + m_MaxSize: {x: 4000, y: 4000} + m_TitleContent: + m_Text: Game + m_Image: {fileID: -6423792434712278376, guid: 0000000000000000d000000000000000, type: 0} + m_Tooltip: + m_Pos: + serializedVersion: 2 + x: 0 + y: 73 + width: 1174 + height: 559 + m_SerializedDataModeController: + m_DataMode: 0 + m_PreferredDataMode: 0 + m_SupportedDataModes: + isAutomatic: 1 + m_ViewDataDictionary: {fileID: 0} + m_OverlayCanvas: + m_LastAppliedPresetName: Default + m_SaveData: [] + m_OverlaysVisible: 1 + m_SerializedViewNames: [] + m_SerializedViewValues: [] + m_PlayModeViewName: GameView + m_ShowGizmos: 0 + m_TargetDisplay: 0 + m_ClearColor: {r: 0, g: 0, b: 0, a: 0} + m_TargetSize: {x: 1174, y: 538} + m_TextureFilterMode: 0 + m_TextureHideFlags: 61 + m_RenderIMGUI: 1 + m_EnterPlayModeBehavior: 0 + m_UseMipMap: 0 + m_VSyncEnabled: 0 + m_Gizmos: 0 + m_Stats: 0 + m_SelectedSizes: 00000000000000000000000012000000000000000000000000000000000000000000000000000000 + m_ZoomArea: + m_HRangeLocked: 0 + m_VRangeLocked: 0 + hZoomLockedByDefault: 0 + vZoomLockedByDefault: 0 + m_HBaseRangeMin: -587 + m_HBaseRangeMax: 587 + m_VBaseRangeMin: -269 + m_VBaseRangeMax: 269 + m_HAllowExceedBaseRangeMin: 1 + m_HAllowExceedBaseRangeMax: 1 + m_VAllowExceedBaseRangeMin: 1 + m_VAllowExceedBaseRangeMax: 1 + m_ScaleWithWindow: 0 + m_HSlider: 0 + m_VSlider: 0 + m_IgnoreScrollWheelUntilClicked: 0 + m_EnableMouseInput: 1 + m_EnableSliderZoomHorizontal: 0 + m_EnableSliderZoomVertical: 0 + m_UniformScale: 1 + m_UpDirection: 1 + m_DrawArea: + serializedVersion: 2 + x: 0 + y: 21 + width: 1174 + height: 538 + m_Scale: {x: 1, y: 1} + m_Translation: {x: 587, y: 269} + m_MarginLeft: 0 + m_MarginRight: 0 + m_MarginTop: 0 + m_MarginBottom: 0 + m_LastShownAreaInsideMargins: + serializedVersion: 2 + x: -587 + y: -269 + width: 1174 + height: 538 + m_MinimalGUI: 1 + m_defaultScale: 1 + m_LastWindowPixelSize: {x: 1174, y: 559} + m_ClearInEditMode: 1 + m_NoCameraWarning: 1 + m_LowResolutionForAspectRatios: 01000001000000000000 + m_XRRenderMode: 0 + m_RenderTexture: {fileID: 0} +--- !u!114 &19 +MonoBehaviour: + m_ObjectHideFlags: 52 + m_CorrespondingSourceObject: {fileID: 0} + m_PrefabInstance: {fileID: 0} + m_PrefabAsset: {fileID: 0} + m_GameObject: {fileID: 0} + m_Enabled: 1 + m_EditorHideFlags: 1 + m_Script: {fileID: 12003, guid: 0000000000000000e000000000000000, type: 0} + m_Name: + m_EditorClassIdentifier: + m_MinSize: {x: 100, y: 100} + m_MaxSize: {x: 4000, y: 4000} + m_TitleContent: + m_Text: Console + m_Image: {fileID: -4327648978806127646, guid: 0000000000000000d000000000000000, type: 0} + m_Tooltip: + m_Pos: + serializedVersion: 2 + x: 0 + y: 653 + width: 1174 + height: 338 + m_SerializedDataModeController: + m_DataMode: 0 + m_PreferredDataMode: 0 + m_SupportedDataModes: + isAutomatic: 1 + m_ViewDataDictionary: {fileID: 0} + m_OverlayCanvas: + m_LastAppliedPresetName: Default + m_SaveData: [] + m_OverlaysVisible: 1 diff --git a/AxibugEmuOnline.Client/UserSettings/Search.index b/AxibugEmuOnline.Client/UserSettings/Search.index new file mode 100644 index 00000000..dba62d53 --- /dev/null +++ b/AxibugEmuOnline.Client/UserSettings/Search.index @@ -0,0 +1,13 @@ +{ + "name": "Assets", + "roots": ["Assets"], + "includes": [], + "excludes": [], + "options": { + "types": true, + "properties": true, + "extended": false, + "dependencies": false + }, + "baseScore": 999 +} \ No newline at end of file diff --git a/AxibugEmuOnline.Client/UserSettings/Search.settings b/AxibugEmuOnline.Client/UserSettings/Search.settings new file mode 100644 index 00000000..74b054ba --- /dev/null +++ b/AxibugEmuOnline.Client/UserSettings/Search.settings @@ -0,0 +1,76 @@ +trackSelection = true +refreshSearchWindowsInPlayMode = false +pickerAdvancedUI = false +fetchPreview = true +defaultFlags = 0 +keepOpen = false +queryFolder = "Assets" +onBoardingDoNotAskAgain = true +showPackageIndexes = false +showStatusBar = false +scopes = { +} +providers = { + log = { + active = false + priority = 210 + defaultAction = null + } + find = { + active = true + priority = 25 + defaultAction = null + } + profilermarkers = { + active = false + priority = 100 + defaultAction = null + } + scene = { + active = true + priority = 50 + defaultAction = null + } + asset = { + active = true + priority = 25 + defaultAction = null + } + adb = { + active = false + priority = 2500 + defaultAction = null + } + packages = { + active = false + priority = 90 + defaultAction = null + } + store = { + active = false + priority = 100 + defaultAction = null + } + performance = { + active = false + priority = 100 + defaultAction = null + } +} +objectSelectors = { +} +recentSearches = [ +] +searchItemFavorites = [ +] +savedSearchesSortOrder = 0 +showSavedSearchPanel = false +hideTabs = false +expandedQueries = [ +] +queryBuilder = true +ignoredProperties = "id;name;classname;imagecontentshash" +helperWidgetCurrentArea = "all" +disabledIndexers = "" +minIndexVariations = 2 +findProviderIndexHelper = true \ No newline at end of file