/Nes/APU/ChannelNoi.cs

http://mynes.codeplex.com · C# · 155 lines · 121 code · 14 blank · 20 comment · 18 complexity · c8058aced2f271bda6443ae2269269ab 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. using System;
  22. namespace MyNes.Nes
  23. {
  24. public class ChannelNoi : NesApuChannel
  25. {
  26. private static readonly int[] FrequencyTable =
  27. {
  28. 0x0004, 0x0008, 0x0010, 0x0020, 0x0040, 0x0060, 0x0080, 0x00A0,
  29. 0x00CA, 0x00FE, 0x017C, 0x01FC, 0x02FA, 0x03F8, 0x07F2, 0x0FE4,
  30. };
  31. private NesApuDuration duration;
  32. private NesApuEnvelope envelope;
  33. private bool active;
  34. private int enabled;
  35. private int shiftBit = 8;
  36. private int shiftReg = 1;
  37. private float delay = FrequencyTable[0];
  38. private float timer;
  39. public bool Enabled
  40. {
  41. get { return duration.Counter != 0; }
  42. set
  43. {
  44. enabled = value ? ~0 : 0;
  45. duration.Counter &= enabled;
  46. active = CanOutput();
  47. }
  48. }
  49. public ChannelNoi(NesSystem nes)
  50. : base(nes) { }
  51. private bool CanOutput()
  52. {
  53. return duration.Counter != 0 && envelope.Sound != 0;
  54. }
  55. public override void ClockHalf()
  56. {
  57. duration.Clock();
  58. active = CanOutput();
  59. }
  60. public override void ClockQuad()
  61. {
  62. envelope.Clock();
  63. active = CanOutput();
  64. }
  65. public override void Poke1(int addr, byte data)
  66. {
  67. duration.Enabled = (data & 0x20) == 0;
  68. envelope.Looping = (data & 0x20) != 0;
  69. envelope.Enabled = (data & 0x10) != 0;
  70. envelope.Sound = (data & 0x0F);
  71. }
  72. public override void Poke2(int addr, byte data) { }
  73. public override void Poke3(int addr, byte data)
  74. {
  75. delay = FrequencyTable[data & 0x0F];
  76. shiftBit = (data & 0x80) != 0 ? 8 : 13;
  77. }
  78. public override void Poke4(int addr, byte data)
  79. {
  80. duration.Counter = DurationTable[data >> 3] & enabled;
  81. envelope.Refresh = true;
  82. active = CanOutput();
  83. }
  84. public override int RenderSample(float rate)
  85. {
  86. float sum = timer;
  87. timer -= rate;
  88. if (active)
  89. {
  90. if (timer >= 0)
  91. {
  92. if ((shiftReg & 0x01) == 0)
  93. return envelope.Sound;
  94. }
  95. else
  96. {
  97. if ((shiftReg & 0x01) != 0)
  98. sum = 0;
  99. do
  100. {
  101. shiftReg = (shiftReg >> 1) | ((shiftReg << 14 ^ shiftReg << shiftBit) & 0x4000);
  102. if ((shiftReg & 0x01) == 0)
  103. sum += Math.Min(-timer, delay);
  104. timer += delay;
  105. }
  106. while (timer < 0);
  107. return (int)((sum * envelope.Sound) / rate);
  108. }
  109. }
  110. else
  111. {
  112. while (timer < 0)
  113. {
  114. shiftReg = (shiftReg >> 1) | ((shiftReg << 14 ^ shiftReg << shiftBit) & 0x4000);
  115. timer += delay;
  116. }
  117. }
  118. return 0;
  119. }
  120. public override void SaveState(StateStream stateStream)
  121. {
  122. duration.SaveState(stateStream);
  123. envelope.SaveState(stateStream);
  124. stateStream.Write(shiftBit);
  125. stateStream.Write(shiftReg);
  126. stateStream.Write(delay);
  127. stateStream.Write(timer);
  128. base.SaveState(stateStream);
  129. }
  130. public override void LoadState(StateStream stateStream)
  131. {
  132. duration.LoadState(stateStream);
  133. envelope.LoadState(stateStream);
  134. shiftBit = stateStream.ReadInt32();
  135. shiftReg = stateStream.ReadInt32();
  136. delay = stateStream.ReadFloat();
  137. timer = stateStream.ReadFloat();
  138. base.LoadState(stateStream);
  139. }
  140. }
  141. }