/Nes/Mappers/Mapper189.cs

http://mynes.codeplex.com · C# · 224 lines · 198 code · 6 blank · 20 comment · 22 complexity · b592aad03eb9bfdb753d284102c4190e MD5 · raw file

  1. /*********************************************************************\
  2. *This file is part of My Nes *
  3. *A Nintendo Entertainment System Emulator. *
  4. * *
  5. *Copyright Š Ala Hadid 2009 - 2011 *
  6. *E-mail: mailto:ahdsoftwares@hotmail.com *
  7. * *
  8. *My Nes is free software: you can redistribute it and/or modify *
  9. *it under the terms of the GNU General Public License as published by *
  10. *the Free Software Foundation, either version 3 of the License, or *
  11. *(at your option) any later version. *
  12. * *
  13. *My Nes is distributed in the hope that it will be useful, *
  14. *but WITHOUT ANY WARRANTY; without even the implied warranty of *
  15. *MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
  16. *GNU General Public License for more details. *
  17. * *
  18. *You should have received a copy of the GNU General Public License *
  19. *along with this program. If not, see <http://www.gnu.org/licenses/>.*
  20. \*********************************************************************/
  21. namespace MyNes.Nes
  22. {
  23. class Mapper189 : Mapper
  24. {
  25. byte[] reg = new byte[2];
  26. byte chr01 = 0;
  27. byte chr23 = 2;
  28. byte chr4 = 4;
  29. byte chr5 = 5;
  30. byte chr6 = 6;
  31. byte chr7 = 7;
  32. byte irq_enable = 0;
  33. byte irq_counter = 0;
  34. byte irq_latch = 0;
  35. byte[] protect_dat = new byte[4];
  36. public Mapper189(NesSystem nesSystem)
  37. : base(nesSystem) { }
  38. public override void Poke(int Address, byte data)
  39. {
  40. if (Address < 0x8000)
  41. {
  42. if ((Address & 0xFF00) == 0x4100)
  43. {
  44. cpuMemory.Switch32kPrgRom(((data & 0x30) >> 4) * 8);
  45. }
  46. else if ((Address & 0xFF00) == 0x6100)
  47. {
  48. cpuMemory.Switch32kPrgRom((data & 0x30) * 8);
  49. }
  50. }
  51. else
  52. {
  53. switch (Address & 0xE001)
  54. {
  55. case 0x8000:
  56. reg[0] = data;
  57. SetBank_PPU();
  58. break;
  59. case 0x8001:
  60. reg[1] = data;
  61. SetBank_PPU();
  62. switch (reg[0] & 0x07)
  63. {
  64. case 0x00:
  65. chr01 = (byte)(data & 0xFE);
  66. SetBank_PPU();
  67. break;
  68. case 0x01:
  69. chr23 = (byte)(data & 0xFE);
  70. SetBank_PPU();
  71. break;
  72. case 0x02:
  73. chr4 = data;
  74. SetBank_PPU();
  75. break;
  76. case 0x03:
  77. chr5 = data;
  78. SetBank_PPU();
  79. break;
  80. case 0x04:
  81. chr6 = data;
  82. SetBank_PPU();
  83. break;
  84. case 0x05:
  85. chr7 = data;
  86. SetBank_PPU();
  87. break;
  88. }
  89. break;
  90. case 0xA000:
  91. if ((data & 0x01) == 0x01)
  92. cartridge.Mirroring = Mirroring.ModeHorz;
  93. else
  94. cartridge.Mirroring = Mirroring.ModeVert;
  95. break;
  96. case 0xC000:
  97. irq_counter = data;
  98. break;
  99. case 0xC001:
  100. irq_latch = data;
  101. break;
  102. case 0xE000:
  103. irq_enable = 0;
  104. cpu.Interrupt(NesCpu.IsrType.External, false);
  105. break;
  106. case 0xE001:
  107. irq_enable = 0xFF;
  108. break;
  109. }
  110. }
  111. }
  112. protected override void Initialize(bool initializing)
  113. {
  114. cpuMemory.Map(0x4018, 0x7FFF, Poke);
  115. cpuMemory.Switch16kPrgRom((cartridge.PrgPages - 2) * 4, 0);
  116. cpuMemory.Switch16kPrgRom((cartridge.PrgPages - 1) * 4, 1);
  117. if (cartridge.HasCharRam)
  118. cpuMemory.FillChr(16);
  119. SetBank_PPU();
  120. if (cartridge.PrgPages == 1)
  121. {
  122. cpuMemory.Switch16kPrgRom(0, 1);
  123. }
  124. }
  125. void SetBank_PPU()
  126. {
  127. if (!cartridge.HasCharRam)
  128. {
  129. if ((reg[0] & 0x80) == 0x80)
  130. {
  131. cpuMemory.Switch1kChrRom(chr4, 0);
  132. cpuMemory.Switch1kChrRom(chr5, 1);
  133. cpuMemory.Switch1kChrRom(chr6, 2);
  134. cpuMemory.Switch1kChrRom(chr7, 3);
  135. cpuMemory.Switch1kChrRom(chr01, 4);
  136. cpuMemory.Switch1kChrRom(chr01 + 1, 5);
  137. cpuMemory.Switch1kChrRom(chr23, 6);
  138. cpuMemory.Switch1kChrRom(chr23 + 1, 7);
  139. }
  140. else
  141. {
  142. cpuMemory.Switch1kChrRom(chr01, 0);
  143. cpuMemory.Switch1kChrRom(chr01 + 1, 1);
  144. cpuMemory.Switch1kChrRom(chr23, 2);
  145. cpuMemory.Switch1kChrRom(chr23 + 1, 3);
  146. cpuMemory.Switch1kChrRom(chr4, 4);
  147. cpuMemory.Switch1kChrRom(chr5, 5);
  148. cpuMemory.Switch1kChrRom(chr6, 6);
  149. cpuMemory.Switch1kChrRom(chr7, 7);
  150. }
  151. }
  152. else
  153. {
  154. if ((reg[0] & 0x80) == 0x80)
  155. {
  156. cpuMemory.Switch1kChrRom((chr01 + 0) & 0x07, 4);
  157. cpuMemory.Switch1kChrRom((chr01 + 1) & 0x07, 5);
  158. cpuMemory.Switch1kChrRom((chr23 + 0) & 0x07, 6);
  159. cpuMemory.Switch1kChrRom((chr23 + 1) & 0x07, 7);
  160. cpuMemory.Switch1kChrRom(chr4 & 0x07, 0);
  161. cpuMemory.Switch1kChrRom(chr5 & 0x07, 1);
  162. cpuMemory.Switch1kChrRom(chr6 & 0x07, 2);
  163. cpuMemory.Switch1kChrRom(chr7 & 0x07, 3);
  164. }
  165. else
  166. {
  167. cpuMemory.Switch1kChrRom((chr01 + 0) & 0x07, 0);
  168. cpuMemory.Switch1kChrRom((chr01 + 1) & 0x07, 1);
  169. cpuMemory.Switch1kChrRom((chr23 + 0) & 0x07, 2);
  170. cpuMemory.Switch1kChrRom((chr23 + 1) & 0x07, 3);
  171. cpuMemory.Switch1kChrRom(chr4 & 0x07, 4);
  172. cpuMemory.Switch1kChrRom(chr5 & 0x07, 5);
  173. cpuMemory.Switch1kChrRom(chr6 & 0x07, 6);
  174. cpuMemory.Switch1kChrRom(chr7 & 0x07, 7);
  175. }
  176. }
  177. }
  178. public override void TickScanlineTimer()
  179. {
  180. if (irq_enable != 0)
  181. {
  182. if ((--irq_counter) == 0)
  183. {
  184. irq_counter = irq_latch;
  185. cpu.Interrupt(NesCpu.IsrType.External, true);
  186. }
  187. }
  188. }
  189. public override void SaveState(StateStream stateStream)
  190. {
  191. stateStream.Write(reg);
  192. stateStream.Write(chr01);
  193. stateStream.Write(chr23);
  194. stateStream.Write(chr4);
  195. stateStream.Write(chr5);
  196. stateStream.Write(chr6);
  197. stateStream.Write(chr7);
  198. stateStream.Write(irq_enable);
  199. stateStream.Write(irq_latch);
  200. stateStream.Write(protect_dat);
  201. base.SaveState(stateStream);
  202. }
  203. public override void LoadState(StateStream stateStream)
  204. {
  205. stateStream.Read(reg);
  206. chr01 = stateStream.ReadByte();
  207. chr23 = stateStream.ReadByte();
  208. chr4 = stateStream.ReadByte();
  209. chr5 = stateStream.ReadByte();
  210. chr6 = stateStream.ReadByte();
  211. chr7 = stateStream.ReadByte();
  212. irq_enable = stateStream.ReadByte();
  213. irq_latch = stateStream.ReadByte();
  214. stateStream.Read(protect_dat);
  215. base.LoadState(stateStream);
  216. }
  217. }
  218. }