From 3b2c00cf39e3839cec900d31ac8fcf426a471f13 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 17 Jan 2015 21:02:59 +0000 Subject: [PATCH] NES - remove a bit more partial class as namespace abuse --- .../Consoles/Nintendo/NES/BisqAPU.cs | 1180 ++++++++--------- .../Consoles/Nintendo/NES/BoardSystem.cs | 32 +- .../Consoles/Nintendo/NES/Core.cs | 36 +- .../Nintendo/NES/NES.IMemoryDomains.cs | 32 +- .../Nintendo/NES/NES.INESPPUViewable.cs | 12 +- .../Consoles/Nintendo/NES/NES.ISaveRam.cs | 22 +- .../Consoles/Nintendo/NES/NES.IStatable.cs | 4 +- .../Consoles/Nintendo/NES/NES.cs | 48 +- .../Consoles/Nintendo/NES/PPU.cs | 12 +- .../Consoles/Nintendo/NES/PPU.regs.cs | 6 +- 10 files changed, 689 insertions(+), 695 deletions(-) diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BisqAPU.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BisqAPU.cs index 01ec5629d0..4f4715e926 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BisqAPU.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BisqAPU.cs @@ -13,643 +13,637 @@ using BizHawk.Emulation.Common; namespace BizHawk.Emulation.Cores.Nintendo.NES { - partial class NES + public class BisqAPU : ISoundProvider { - public class BisqAPU : ISoundProvider + class RegBitSetStore { - class RegBitSetStore + public RegBitSetStore(int numUints) { - public RegBitSetStore(int numUints) - { - data = new uint[numUints]; - } - public uint[] data; + data = new uint[numUints]; + } + public uint[] data; + } + + class RegBitSet + { + RegBitSetStore store; + int mask; + int bitno, nbits, dim, index; + public RegBitSet(RegBitSetStore store, int bitno, int nbits, int dim = 1, int index = 0) + { + this.store = store; + this.bitno = bitno; + this.nbits = nbits; + this.dim = dim; + this.index = index; + mask = ((1 << (nbits / 2)) << (nbits - (nbits / 2))) - 1; } - class RegBitSet + public static implicit operator int(RegBitSet rhs) { return (int)rhs.get(); } + public static implicit operator uint(RegBitSet rhs) { return rhs.get(); } + public static explicit operator bool(RegBitSet rhs) { return rhs.get() != 0; } + + public uint Value { get { return get(); } set { set(value); } } + + public uint this[int offset] { - RegBitSetStore store; - int mask; - int bitno, nbits, dim, index; - public RegBitSet(RegBitSetStore store, int bitno, int nbits, int dim = 1, int index = 0) + get { - this.store = store; - this.bitno = bitno; - this.nbits = nbits; - this.dim = dim; - this.index = index; - mask = ((1 << (nbits / 2)) << (nbits - (nbits / 2))) - 1; + return get(offset); } - - public static implicit operator int(RegBitSet rhs) { return (int)rhs.get(); } - public static implicit operator uint(RegBitSet rhs) { return rhs.get(); } - public static explicit operator bool(RegBitSet rhs) { return rhs.get() != 0; } - - public uint Value { get { return get(); } set { set(value); } } - - public uint this[int offset] + set { - get - { - return get(offset); - } - set - { - set(value, offset); - } + set(value, offset); } - - public uint get(int offset = 0) - { - return (uint)((store.data[index + offset] >> bitno) & mask); - } - - public void set(uint val, int offset = 0) - { - long temp = (store.data[index + offset] & ~(mask << bitno)) - | ((nbits > 1 ? val & mask : ((val != 0) ? 1 : 0)) << bitno); - store.data[index + offset] = (uint)temp; - } - - public static uint operator ^(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() ^ rhs)); } - public static uint operator |(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() | rhs)); } - public static uint operator &(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() & rhs)); } - public static uint operator +(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() + rhs)); } - public static uint operator -(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() - rhs)); } - }; - - class SignedRegBitSet - { - RegBitSetStore store; - int mask; - int bitno, nbits, dim, index; - public SignedRegBitSet(RegBitSetStore store, int bitno, int nbits, int dim = 1, int index = 0) - { - this.store = store; - this.bitno = bitno; - this.nbits = nbits; - this.dim = dim; - this.index = index; - mask = ((1 << (nbits / 2)) << (nbits - (nbits / 2))) - 1; - } - - public static implicit operator int(SignedRegBitSet rhs) { return rhs.get(); } - public static implicit operator uint(SignedRegBitSet rhs) { return (uint)rhs.get(); } - public static explicit operator bool(SignedRegBitSet rhs) { return rhs.get() != 0; } - - public int Value { get { return get(); } set { set(value); } } - - public int this[int offset] - { - get - { - return get(offset); - } - set - { - set(value, offset); - } - } - - public int get(int offset = 0) - { - return (int)((store.data[index + offset] >> bitno) & mask); - } - - public void set(int val, int offset = 0) - { - long temp = ((int)store.data[index + offset] & (int)~(mask << bitno)) - | ((nbits > 1 ? val & mask : ((val != 0) ? 1 : 0)) << bitno); - store.data[index + offset] = (uint)temp; - } - - public static int operator ^(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() ^ rhs)); } - public static int operator |(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() | rhs)); } - public static int operator &(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() & rhs)); } - public static int operator +(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() + rhs)); } - public static int operator -(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() - rhs)); } - - }; - - - public static bool CFG_USE_METASPU = true; - - class APUdata - { - public APUdata() - { - bs = new RegBitSetStore(5); - ChannelsEnabled = new RegBitSet(bs, 15, 1, 5, 0); - frame_delay = new RegBitSet(bs, 0, 15, 5, 0); - frame = new RegBitSet(bs, 0, 8, 5, 1); - FiveCycleDivider = new RegBitSet(bs, 8, 1, 5, 1); - IRQdisable = new RegBitSet(bs, 9, 1, 5, 1); - DMC_CycleCost = new RegBitSet(bs, 10, 3, 5, 1); - IRQ_delay = new RegBitSet(bs, 0, 15, 5, 2); - } - public RegBitSetStore bs; - public RegBitSet ChannelsEnabled; - public RegBitSet frame_delay; - public RegBitSet frame; - public RegBitSet FiveCycleDivider; - public RegBitSet IRQdisable; - public RegBitSet DMC_CycleCost; - public RegBitSet IRQ_delay; - } - APUdata data = new APUdata(); - - class APUchannel - { - public APUchannel() - { - bs = new RegBitSetStore(11); - - reg0 = new RegBitSet(bs,0, 8, 11, 0); - DutyCycle = new RegBitSet(bs,6, 2, 11, 0); - EnvDecayDisable = new RegBitSet(bs,4, 1, 11, 0); - EnvDecayRate = new RegBitSet(bs,0, 4, 11, 0); - EnvDecayLoopEnable = new RegBitSet(bs,5, 1, 11, 0); - FixedVolume = new RegBitSet(bs,0, 4, 11, 0); - LengthCounterDisable = new RegBitSet(bs,5, 1, 11, 0); - LinearCounterInit = new RegBitSet(bs,0, 7, 11, 0); - LinearCounterDisable = new RegBitSet(bs,7, 1, 11, 0); - - reg1 = new RegBitSet(bs,8, 8, 11, 0); - SweepShift = new RegBitSet(bs,8, 3, 11, 0); - SweepDecrease = new RegBitSet(bs,11, 1, 11, 0); - SweepRate = new RegBitSet(bs,12, 3, 11, 0); - SweepEnable = new RegBitSet(bs,15, 1, 11, 0); - PCMlength = new RegBitSet(bs,8, 8, 11, 0); - - reg2 = new RegBitSet(bs,16, 8, 11, 0); - NoiseFreq = new RegBitSet(bs,16, 4, 11, 0); - NoiseType = new RegBitSet(bs,23, 1, 11, 0); - WaveLength = new RegBitSet(bs,16, 11, 11, 0); - - reg3 = new RegBitSet(bs,24, 8, 11, 0); - LengthCounterInit = new RegBitSet(bs,27, 5, 11, 0); - LoopEnabled = new RegBitSet(bs,30, 1, 11, 0); - IRQenable = new RegBitSet(bs,31, 1, 11, 0); - - length_counter = new SignedRegBitSet(bs,0, 32, 11, 1); - linear_counter = new SignedRegBitSet(bs,0, 32, 11, 2); - address = new SignedRegBitSet(bs,0, 32, 11, 3); - envelope = new SignedRegBitSet(bs,0, 32, 11, 4); - sweep_delay = new SignedRegBitSet(bs,0, 32, 11, 5); - env_delay = new SignedRegBitSet(bs,0, 32, 11, 6); - wave_counter = new SignedRegBitSet(bs,0, 32, 11, 7); - hold = new SignedRegBitSet(bs,0, 32, 11, 8); - phase = new SignedRegBitSet(bs,0, 32, 11, 9); - level = new SignedRegBitSet(bs,0, 32, 11, 10); - } - public RegBitSetStore bs; - // 4000, 4004, 400C, 4012: - public RegBitSet reg0, DutyCycle, EnvDecayDisable, EnvDecayRate, EnvDecayLoopEnable, FixedVolume, LengthCounterDisable, LinearCounterInit, LinearCounterDisable; - // 4001, 4005, 4013: - public RegBitSet reg1, SweepShift, SweepDecrease, SweepRate, SweepEnable, PCMlength; - // 4002, 4006, 400A, 400E: - public RegBitSet reg2, NoiseFreq, NoiseType, WaveLength; - // 4003, 4007, 400B, 400F, 4010: - public RegBitSet reg3, LengthCounterInit, LoopEnabled, IRQenable; - // Internals: - public SignedRegBitSet length_counter, linear_counter, address, envelope, sweep_delay, env_delay, wave_counter, hold, phase, level; } - APUchannel[] channels = new APUchannel[5] { new APUchannel(), new APUchannel(), new APUchannel(), new APUchannel(), new APUchannel() }; - - static readonly byte[] LengthCounters = 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 }; - - static readonly ushort[] NoisePeriods = new ushort[16] - { 2,4,8,16,32,48,64,80,101,127,190,254,381,508,1017,2034 }; - - static readonly ushort[] DMCperiods = new ushort[16] - { 428,380,340,320, 286,254,226,214, 190,160,142,128, 106,84,72,54 }; - /* - For PAL: - static const u16 DMCperiods[16] = - { 0x18E,0x162,0x13C,0x12A, 0x114,0x0EC,0x0D2,0x0C6, - 0x0B0,0x094,0x084,0x076, 0x062,0x04E,0x042,0x032 }; - */ - - const int frame_period = 7458; - - // Utility function for sound - bool count(ref int v, int reset) + public uint get(int offset = 0) { - if (--v < 0) - { - v = reset; - return true; - } - else return false; + return (uint)((store.data[index + offset] >> bitno) & mask); } - bool count(ref int v, int reset, int n_at_time) + public void set(uint val, int offset = 0) { - if ((v -= n_at_time) < 0) - { - v += reset; - return true; - } - else return false; + long temp = (store.data[index + offset] & ~(mask << bitno)) + | ((nbits > 1 ? val & mask : ((val != 0) ? 1 : 0)) << bitno); + store.data[index + offset] = (uint)temp; } + public static uint operator ^(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() ^ rhs)); } + public static uint operator |(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() | rhs)); } + public static uint operator &(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() & rhs)); } + public static uint operator +(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() + rhs)); } + public static uint operator -(RegBitSet lhs, uint rhs) { return ((uint)(lhs.get() - rhs)); } + }; - int tick(int c) + class SignedRegBitSet + { + RegBitSetStore store; + int mask; + int bitno, nbits, dim, index; + public SignedRegBitSet(RegBitSetStore store, int bitno, int nbits, int dim = 1, int index = 0) { - APUchannel ch = channels[c]; - int wl = ch.WaveLength; - if (c != 4) ++wl; - if (c < 2) wl *= 2; + this.store = store; + this.bitno = bitno; + this.nbits = nbits; + this.dim = dim; + this.index = index; + mask = ((1 << (nbits / 2)) << (nbits - (nbits / 2))) - 1; + } - if (c == 3) wl = NoisePeriods[ch.NoiseFreq]; + public static implicit operator int(SignedRegBitSet rhs) { return rhs.get(); } + public static implicit operator uint(SignedRegBitSet rhs) { return (uint)rhs.get(); } + public static explicit operator bool(SignedRegBitSet rhs) { return rhs.get() != 0; } - //if(c != 4) wl = wl * (IO::UISpeed); - // ^ Match to the UI speed (but don't for DPCM, because it would skew the timings) + public int Value { get { return get(); } set { set(value); } } - int volume = (bool)ch.length_counter ? (bool)ch.EnvDecayDisable ? (int)ch.FixedVolume : ch.envelope : 0; - // Sample may change at wavelen intervals. - int ref_S = ch.level; - int ref_ch_wave_counter = ch.wave_counter; - if (!(data.ChannelsEnabled[c] != 0) - || !count(ref ref_ch_wave_counter, wl)) + public int this[int offset] + { + get { - ch.wave_counter.Value = ref_ch_wave_counter; - return ref_S; + return get(offset); } + set + { + set(value, offset); + } + } + + public int get(int offset = 0) + { + return (int)((store.data[index + offset] >> bitno) & mask); + } + + public void set(int val, int offset = 0) + { + long temp = ((int)store.data[index + offset] & (int)~(mask << bitno)) + | ((nbits > 1 ? val & mask : ((val != 0) ? 1 : 0)) << bitno); + store.data[index + offset] = (uint)temp; + } + + public static int operator ^(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() ^ rhs)); } + public static int operator |(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() | rhs)); } + public static int operator &(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() & rhs)); } + public static int operator +(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() + rhs)); } + public static int operator -(SignedRegBitSet lhs, int rhs) { return ((int)(lhs.get() - rhs)); } + + }; + + + public static bool CFG_USE_METASPU = true; + + class APUdata + { + public APUdata() + { + bs = new RegBitSetStore(5); + ChannelsEnabled = new RegBitSet(bs, 15, 1, 5, 0); + frame_delay = new RegBitSet(bs, 0, 15, 5, 0); + frame = new RegBitSet(bs, 0, 8, 5, 1); + FiveCycleDivider = new RegBitSet(bs, 8, 1, 5, 1); + IRQdisable = new RegBitSet(bs, 9, 1, 5, 1); + DMC_CycleCost = new RegBitSet(bs, 10, 3, 5, 1); + IRQ_delay = new RegBitSet(bs, 0, 15, 5, 2); + } + public RegBitSetStore bs; + public RegBitSet ChannelsEnabled; + public RegBitSet frame_delay; + public RegBitSet frame; + public RegBitSet FiveCycleDivider; + public RegBitSet IRQdisable; + public RegBitSet DMC_CycleCost; + public RegBitSet IRQ_delay; + } + APUdata data = new APUdata(); + + class APUchannel + { + public APUchannel() + { + bs = new RegBitSetStore(11); + + reg0 = new RegBitSet(bs,0, 8, 11, 0); + DutyCycle = new RegBitSet(bs,6, 2, 11, 0); + EnvDecayDisable = new RegBitSet(bs,4, 1, 11, 0); + EnvDecayRate = new RegBitSet(bs,0, 4, 11, 0); + EnvDecayLoopEnable = new RegBitSet(bs,5, 1, 11, 0); + FixedVolume = new RegBitSet(bs,0, 4, 11, 0); + LengthCounterDisable = new RegBitSet(bs,5, 1, 11, 0); + LinearCounterInit = new RegBitSet(bs,0, 7, 11, 0); + LinearCounterDisable = new RegBitSet(bs,7, 1, 11, 0); + + reg1 = new RegBitSet(bs,8, 8, 11, 0); + SweepShift = new RegBitSet(bs,8, 3, 11, 0); + SweepDecrease = new RegBitSet(bs,11, 1, 11, 0); + SweepRate = new RegBitSet(bs,12, 3, 11, 0); + SweepEnable = new RegBitSet(bs,15, 1, 11, 0); + PCMlength = new RegBitSet(bs,8, 8, 11, 0); + + reg2 = new RegBitSet(bs,16, 8, 11, 0); + NoiseFreq = new RegBitSet(bs,16, 4, 11, 0); + NoiseType = new RegBitSet(bs,23, 1, 11, 0); + WaveLength = new RegBitSet(bs,16, 11, 11, 0); + + reg3 = new RegBitSet(bs,24, 8, 11, 0); + LengthCounterInit = new RegBitSet(bs,27, 5, 11, 0); + LoopEnabled = new RegBitSet(bs,30, 1, 11, 0); + IRQenable = new RegBitSet(bs,31, 1, 11, 0); + + length_counter = new SignedRegBitSet(bs,0, 32, 11, 1); + linear_counter = new SignedRegBitSet(bs,0, 32, 11, 2); + address = new SignedRegBitSet(bs,0, 32, 11, 3); + envelope = new SignedRegBitSet(bs,0, 32, 11, 4); + sweep_delay = new SignedRegBitSet(bs,0, 32, 11, 5); + env_delay = new SignedRegBitSet(bs,0, 32, 11, 6); + wave_counter = new SignedRegBitSet(bs,0, 32, 11, 7); + hold = new SignedRegBitSet(bs,0, 32, 11, 8); + phase = new SignedRegBitSet(bs,0, 32, 11, 9); + level = new SignedRegBitSet(bs,0, 32, 11, 10); + } + public RegBitSetStore bs; + // 4000, 4004, 400C, 4012: + public RegBitSet reg0, DutyCycle, EnvDecayDisable, EnvDecayRate, EnvDecayLoopEnable, FixedVolume, LengthCounterDisable, LinearCounterInit, LinearCounterDisable; + // 4001, 4005, 4013: + public RegBitSet reg1, SweepShift, SweepDecrease, SweepRate, SweepEnable, PCMlength; + // 4002, 4006, 400A, 400E: + public RegBitSet reg2, NoiseFreq, NoiseType, WaveLength; + // 4003, 4007, 400B, 400F, 4010: + public RegBitSet reg3, LengthCounterInit, LoopEnabled, IRQenable; + // Internals: + public SignedRegBitSet length_counter, linear_counter, address, envelope, sweep_delay, env_delay, wave_counter, hold, phase, level; + } + + APUchannel[] channels = new APUchannel[5] { new APUchannel(), new APUchannel(), new APUchannel(), new APUchannel(), new APUchannel() }; + + static readonly byte[] LengthCounters = 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 }; + + static readonly ushort[] NoisePeriods = new ushort[16] + { 2,4,8,16,32,48,64,80,101,127,190,254,381,508,1017,2034 }; + + static readonly ushort[] DMCperiods = new ushort[16] + { 428,380,340,320, 286,254,226,214, 190,160,142,128, 106,84,72,54 }; + /* + For PAL: + static const u16 DMCperiods[16] = + { 0x18E,0x162,0x13C,0x12A, 0x114,0x0EC,0x0D2,0x0C6, + 0x0B0,0x094,0x084,0x076, 0x062,0x04E,0x042,0x032 }; + */ + + const int frame_period = 7458; + + // Utility function for sound + bool count(ref int v, int reset) + { + if (--v < 0) + { + v = reset; + return true; + } + else return false; + } + + bool count(ref int v, int reset, int n_at_time) + { + if ((v -= n_at_time) < 0) + { + v += reset; + return true; + } + else return false; + } + + + int tick(int c) + { + APUchannel ch = channels[c]; + int wl = ch.WaveLength; + if (c != 4) ++wl; + if (c < 2) wl *= 2; + + if (c == 3) wl = NoisePeriods[ch.NoiseFreq]; + + //if(c != 4) wl = wl * (IO::UISpeed); + // ^ Match to the UI speed (but don't for DPCM, because it would skew the timings) + + int volume = (bool)ch.length_counter ? (bool)ch.EnvDecayDisable ? (int)ch.FixedVolume : ch.envelope : 0; + // Sample may change at wavelen intervals. + int ref_S = ch.level; + int ref_ch_wave_counter = ch.wave_counter; + if (!(data.ChannelsEnabled[c] != 0) + || !count(ref ref_ch_wave_counter, wl)) + { ch.wave_counter.Value = ref_ch_wave_counter; - switch (c) + return ref_S; + } + ch.wave_counter.Value = ref_ch_wave_counter; + switch (c) + { + case 0: + default: + case 1: // Square wave. With four different 8-step binary waveforms (32 bits of data total). + ch.phase.Value++; + if (wl < 8) return ref_S; + if ((bool)ch.SweepEnable && !(bool)ch.SweepDecrease) + if (wl + (wl >> ch.SweepShift) >= 0x800) + return ref_S; + return ref_S = ch.level.Value = (0xF33C0C04u & (1u << (ch.phase % 8 + ch.DutyCycle * 8))) != 0 ? volume : 0; + case 2: // Triangle wave + if ((bool)ch.length_counter && (bool)ch.linear_counter && wl >= 3) ++ch.phase.Value; + return ref_S = ch.level.Value = (ch.phase & 15) ^ (((ch.phase & 16) != 0) ? 15 : 0); + + case 3: // Noise: Linear feedback shift register + if (!(bool)ch.hold) ch.hold.Value = 1; + ch.hold.Value = (ch.hold >> 1) + | (((ch.hold ^ (ch.hold >> ((bool)ch.NoiseType ? 6 : 1))) & 1) << 14); + return ref_S = ch.level.Value = ((ch.hold & 1) != 0) ? 0 : volume; + + case 4: // Delta modulation channel (DMC) + // hold = 8 bit value + // phase = number of bits buffered + // length_counter = + if (wl == 0) return ref_S; + if (ch.phase == 0) // Nothing in sample buffer? + { + if (!(bool)ch.length_counter && (bool)ch.LoopEnabled) // Loop? + { + ch.length_counter.Value = ch.PCMlength * 16 + 1; + ch.address.Value = (int)((ch.reg0 | 0x300) << 6); + } + if (ch.length_counter > 0) // Load next 8 bits if available + { + //==========================TODO============== DMC COST===================== + //for(unsigned t = data.DMC_CycleCost; t > 1; --t) + // CPU::RB(u16(ch.address) | 0x8000); // timing + ch.hold.Value = nes.ReadMemory((ushort)((ch.address.Value++) | 0x8000)); // Fetch byte + ch.phase.Value = 8; + --ch.length_counter.Value; + } + else // Otherwise, disable channel or issue IRQ + { + if ((bool)ch.IRQenable) + { + //CPU::reg.APU_DMC_IRQ = true; + dmc_irq = true; + SyncIRQ(); + } + + data.ChannelsEnabled[4] = 0; + } + } + if (ch.phase != 0) // Update the signal if sample buffer nonempty + { + int v = ch.linear_counter; + if (((ch.hold << --ch.phase.Value) & 0x80)!=0) v += 2; else v -= 2; + if (v >= 0 && v <= 0x7F) ch.linear_counter.Value = v; + } + return ref_S = ch.level = ch.linear_counter; + } + } + + bool dmc_irq; + bool sequencer_irq; + public bool irq_pending; + void SyncIRQ() + { + irq_pending = sequencer_irq | dmc_irq; + } + + NES nes; + public BisqAPU(NES nes) + { + this.nes = nes; + } + + public void RunOne() + { + if (data.IRQ_delay != 0x7FFF) + { + if (data.IRQ_delay > 0) --data.IRQ_delay.Value; + else { sequencer_irq = true; SyncIRQ(); data.IRQ_delay.Value = 0x7FFF; } + } + if (data.frame_delay > 0) + --data.frame_delay.Value; + else + { + bool Do240 = true, Do120 = false; + data.frame_delay.Value += frame_period; + switch (data.frame.Value++) { case 0: - default: - case 1: // Square wave. With four different 8-step binary waveforms (32 bits of data total). - ch.phase.Value++; - if (wl < 8) return ref_S; - if ((bool)ch.SweepEnable && !(bool)ch.SweepDecrease) - if (wl + (wl >> ch.SweepShift) >= 0x800) - return ref_S; - return ref_S = ch.level.Value = (0xF33C0C04u & (1u << (ch.phase % 8 + ch.DutyCycle * 8))) != 0 ? volume : 0; - case 2: // Triangle wave - if ((bool)ch.length_counter && (bool)ch.linear_counter && wl >= 3) ++ch.phase.Value; - return ref_S = ch.level.Value = (ch.phase & 15) ^ (((ch.phase & 16) != 0) ? 15 : 0); - - case 3: // Noise: Linear feedback shift register - if (!(bool)ch.hold) ch.hold.Value = 1; - ch.hold.Value = (ch.hold >> 1) - | (((ch.hold ^ (ch.hold >> ((bool)ch.NoiseType ? 6 : 1))) & 1) << 14); - return ref_S = ch.level.Value = ((ch.hold & 1) != 0) ? 0 : volume; - - case 4: // Delta modulation channel (DMC) - // hold = 8 bit value - // phase = number of bits buffered - // length_counter = - if (wl == 0) return ref_S; - if (ch.phase == 0) // Nothing in sample buffer? - { - if (!(bool)ch.length_counter && (bool)ch.LoopEnabled) // Loop? - { - ch.length_counter.Value = ch.PCMlength * 16 + 1; - ch.address.Value = (int)((ch.reg0 | 0x300) << 6); - } - if (ch.length_counter > 0) // Load next 8 bits if available - { - //==========================TODO============== DMC COST===================== - //for(unsigned t = data.DMC_CycleCost; t > 1; --t) - // CPU::RB(u16(ch.address) | 0x8000); // timing - ch.hold.Value = nes.ReadMemory((ushort)((ch.address.Value++) | 0x8000)); // Fetch byte - ch.phase.Value = 8; - --ch.length_counter.Value; - } - else // Otherwise, disable channel or issue IRQ - { - if ((bool)ch.IRQenable) - { - //CPU::reg.APU_DMC_IRQ = true; - dmc_irq = true; - SyncIRQ(); - } - - data.ChannelsEnabled[4] = 0; - } - } - if (ch.phase != 0) // Update the signal if sample buffer nonempty - { - int v = ch.linear_counter; - if (((ch.hold << --ch.phase.Value) & 0x80)!=0) v += 2; else v -= 2; - if (v >= 0 && v <= 0x7F) ch.linear_counter.Value = v; - } - return ref_S = ch.level = ch.linear_counter; - } - } - - bool dmc_irq; - bool sequencer_irq; - public bool irq_pending; - void SyncIRQ() - { - irq_pending = sequencer_irq | dmc_irq; - } - - NES nes; - public BisqAPU(NES nes) - { - this.nes = nes; - } - - public void RunOne() - { - if (data.IRQ_delay != 0x7FFF) - { - if (data.IRQ_delay > 0) --data.IRQ_delay.Value; - else { sequencer_irq = true; SyncIRQ(); data.IRQ_delay.Value = 0x7FFF; } - } - if (data.frame_delay > 0) - --data.frame_delay.Value; - else - { - bool Do240 = true, Do120 = false; - data.frame_delay.Value += frame_period; - switch (data.frame.Value++) - { - case 0: - if (!(bool)data.IRQdisable && !(bool)data.FiveCycleDivider) - data.IRQ_delay.Value = frame_period * 4 + 2; - // passthru - goto case 2; - case 2: - Do120 = true; - break; - case 1: - data.frame_delay.Value -= 2; - break; - case 3: - data.frame.Value = 0; - if ((bool)data.FiveCycleDivider) - data.frame_delay.Value += frame_period - 6; - break; - } - // Some events are invoked at 96 Hz or 120 Hz rate. Others, 192 Hz or 240 Hz. - for (int c = 0; c < 4; ++c) - { - APUchannel ch = channels[c]; - int wl = ch.WaveLength; - - // 96/120 Hz events: - if (Do120) - { - // Length tick (all channels except DMC, but different disable bit for triangle wave) - if ((bool)ch.length_counter - && !(c == 2 ? (bool)ch.LinearCounterDisable : (bool)ch.LengthCounterDisable)) - ch.length_counter.Value -= 1; - - // Sweep tick (square waves only) - int ref_ch_sweep_delay = ch.sweep_delay; - if (c < 2 && count(ref ref_ch_sweep_delay, ch.SweepRate)) - if (wl >= 8 && (bool)ch.SweepEnable && (bool)ch.SweepShift) - { - int s = wl >> ch.SweepShift; - wl += ((bool)ch.SweepDecrease ? ((c != 0) ? -s : ~s) : s); - if (wl < 0x800) ch.WaveLength.Value = (uint)wl; - } - - ch.sweep_delay.Value = ref_ch_sweep_delay; - } - - // 240/192 Hz events: - if (Do240) - { - // Linear tick (triangle wave only) (all ticks) - if (c == 2) - ch.linear_counter.Value = (bool)ch.LinearCounterDisable - ? ch.LinearCounterInit - : (ch.linear_counter > 0 ? ch.linear_counter - 1 : 0); - - // Envelope tick (square and noise channels) (all ticks) - int ref_ch_env_delay = ch.env_delay; - if (c != 2 && count(ref ref_ch_env_delay, ch.EnvDecayRate)) - if (ch.envelope > 0 || (bool)ch.EnvDecayLoopEnable) - ch.envelope.Value = (ch.envelope - 1) & 15; - ch.env_delay.Value = ref_ch_env_delay; - } - - } - } - - // Mix the audio: Get the momentary sample from each channel and mix them. - // #define s(c) tick() - //v = [](float m,float n, float d) { return n!=0.f ? m/n : d; }; - Func v = (float m,float n, float d) => ( n!=0.0f ? m/n : d ); - - short sample = (short)(30000 * - ( - // Square 0 and 1 - v(95.88f, (100.0f + v(8128.0f, tick(0) + tick(1), -100.0f)), 0.0f) - // Triangle, noise, DMC - + v(159.79f, (100.0f + v(1.0f, tick(2)/8227.0f + tick(3)/12241.0f + tick(4)/22638.0f, -1000.0f)), 0.0f) - // GamePak audio (these volume values are bogus, but sound acceptable) - + v(95.88f, (100.0f + v(32512.0f, /*GamePak::ExtAudio()*/ 0, -100.0f)), 0.0f) - - 0.5f - )); - - EmitSample(sample); - - ////this (and the similar line below) is a crude hack - ////we should be generating logic to suppress the $4015 clear when the assert signal is set instead - ////be sure to test "apu_test" if you mess with this - //sequencer_irq |= sequencer_irq_assert; - } - - void WriteC(int chno, int index, byte value) - { - APUchannel ch = channels[chno]; - switch (index) - { - case 0: - if ((bool)ch.LinearCounterDisable) ch.linear_counter.Value = value & 0x7F; - ch.reg0.Value = value; + if (!(bool)data.IRQdisable && !(bool)data.FiveCycleDivider) + data.IRQ_delay.Value = frame_period * 4 + 2; + // passthru + goto case 2; + case 2: + Do120 = true; break; case 1: - ch.reg1.Value = value; - ch.sweep_delay.Value = ch.SweepRate; - break; - case 2: - ch.reg2.Value = value; + data.frame_delay.Value -= 2; break; case 3: - ch.reg3.Value = value; - if (data.ChannelsEnabled[chno]!=0) - ch.length_counter.Value = LengthCounters[ch.LengthCounterInit]; - ch.linear_counter.Value = ch.LinearCounterInit; - ch.env_delay.Value = ch.EnvDecayRate; - ch.envelope.Value = 15; - if (index < 8) ch.phase.Value = 0; - break; - case 0x12: - ch.reg0.Value = value; - ch.address.Value = ((int)ch.reg0 | 0x300) << 6; - break; - case 0x10: - ch.reg3.Value = value; - ch.WaveLength.Value = (uint)(DMCperiods[value & 0x0F] - 1); - if (!(bool)ch.IRQenable) { dmc_irq = false; SyncIRQ(); } - break; - case 0x13: // sample length - ch.reg1.Value = value; - if (ch.length_counter == 0) - ch.length_counter.Value = ch.PCMlength * 16 + 1; - break; - case 0x11: // dac value - ch.linear_counter.Value = value & 0x7F; - break; - case 0x15: - for (int c = 0; c < 5; ++c) - data.ChannelsEnabled[c] = (uint)((value >> c) & 1); //noteworthy tweak - for (int c = 0; c < 5; ++c) - if (data.ChannelsEnabled[c]==0) - channels[c].length_counter.Value = 0; - else if (c == 4 && channels[c].length_counter == 0) - { - APUchannel chh = channels[c]; - chh.length_counter.Value = chh.PCMlength * 16 + 1; - chh.address.Value = ((int)chh.reg0 | 0x300) << 6; - chh.phase.Value = 0; - } - //CPU::reg.APU_DMC_IRQ = false; - dmc_irq = false; SyncIRQ(); - break; - case 0x17: - data.IRQdisable.Value = (uint)(value & 0x40); - data.FiveCycleDivider.Value = (uint)(value & 0x80); - // apu_test 1-len_ctr: Writing $80 to $4017 should clock length immediately - // But Writing $00 to $4017 shouldn't clock length immediately - data.frame_delay.Value &= 1; data.frame.Value = 0; - data.IRQ_delay.Value = 0x7FFF; - if ((bool)data.IRQdisable) { sequencer_irq = false; SyncIRQ(); } - if (!(bool)data.FiveCycleDivider) - { - data.frame.Value = 1; - data.frame_delay.Value += frame_period; - - if (!(bool)data.IRQdisable) - { - data.IRQ_delay.Value = data.frame_delay + frame_period * 3 + 1 - 3; - // ^ "- 3" makes apu_test "4-jitter" not complain - // that "Frame irq is set too late" - } - } + if ((bool)data.FiveCycleDivider) + data.frame_delay.Value += frame_period - 6; break; } - } - - public byte ReadReg(int addr) - { - byte res = 0; - for (int c = 0; c < 5; ++c) res |= (byte)(((bool)channels[c].length_counter ? 1 << c : 0)); - if (sequencer_irq) res |= 0x40; sequencer_irq = false; SyncIRQ(); - if (dmc_irq) res |= 0x80; - return res; - } - - public void WriteReg(int addr, byte val) - { - int index = addr - 0x4000; - WriteC((index / 4) % 5, index < 0x10 ? index % 4 : index, val); - } - - public void NESSoftReset() - { - } - - public void DiscardSamples() - { - metaspu.buffer.clear(); - } - - public void SyncState(Serializer ser) - { - } - - double accumulate; - double timer; - Queue squeue = new Queue(); - int last_hwsamp; - int panic_sample, panic_count; - void EmitSample(int samp) - { - //kill the annoying hum that is a consequence of the shitty code below - if (samp == panic_sample) - panic_count++; - else panic_count = 0; - if (panic_count > 178977) - samp = 0; - else - panic_sample = samp; - - int this_samp = samp; - const double kMixRate = 44100.0 / 1789772.0; - const double kInvMixRate = (1 / kMixRate); - timer += kMixRate; - accumulate += samp; - if (timer <= 1) - return; - - accumulate -= samp; - timer -= 1; - double ratio = (timer / kMixRate); - double fractional = (this_samp - last_hwsamp) * ratio; - double factional_remainder = (this_samp - last_hwsamp) * (1 - ratio); - accumulate += fractional; - - accumulate *= 436; //32768/(15*4) -- adjust later for other sound channels - int outsamp = (int)(accumulate / kInvMixRate); - if (CFG_USE_METASPU) - metaspu.buffer.enqueue_sample((short)outsamp, (short)outsamp); - else squeue.Enqueue(outsamp); - accumulate = factional_remainder; - - last_hwsamp = this_samp; - } - - MetaspuSoundProvider metaspu = new MetaspuSoundProvider(ESynchMethod.ESynchMethod_V); - public int MaxVolume { get; set; } // not supported - - void ISoundProvider.GetSamples(short[] samples) - { - if (CFG_USE_METASPU) + // Some events are invoked at 96 Hz or 120 Hz rate. Others, 192 Hz or 240 Hz. + for (int c = 0; c < 4; ++c) { - metaspu.GetSamples(samples); - //foreach(short sample in samples) bw.Write((short)sample); - } - else - MyGetSamples(samples); + APUchannel ch = channels[c]; + int wl = ch.WaveLength; - //mix in the cart's extra sound circuit - nes.board.ApplyCustomAudio(samples); - } + // 96/120 Hz events: + if (Do120) + { + // Length tick (all channels except DMC, but different disable bit for triangle wave) + if ((bool)ch.length_counter + && !(c == 2 ? (bool)ch.LinearCounterDisable : (bool)ch.LengthCounterDisable)) + ch.length_counter.Value -= 1; - //static BinaryWriter bw = new BinaryWriter(new FileStream("d:\\out.raw",FileMode.Create,FileAccess.Write,FileShare.Read)); - void MyGetSamples(short[] samples) - { - //Console.WriteLine("a: {0} with todo: {1}",squeue.Count,samples.Length/2); + // Sweep tick (square waves only) + int ref_ch_sweep_delay = ch.sweep_delay; + if (c < 2 && count(ref ref_ch_sweep_delay, ch.SweepRate)) + if (wl >= 8 && (bool)ch.SweepEnable && (bool)ch.SweepShift) + { + int s = wl >> ch.SweepShift; + wl += ((bool)ch.SweepDecrease ? ((c != 0) ? -s : ~s) : s); + if (wl < 0x800) ch.WaveLength.Value = (uint)wl; + } - for (int i = 0; i < samples.Length / 2; i++) - { - int samp = 0; - if (squeue.Count != 0) - samp = squeue.Dequeue(); + ch.sweep_delay.Value = ref_ch_sweep_delay; + } + + // 240/192 Hz events: + if (Do240) + { + // Linear tick (triangle wave only) (all ticks) + if (c == 2) + ch.linear_counter.Value = (bool)ch.LinearCounterDisable + ? ch.LinearCounterInit + : (ch.linear_counter > 0 ? ch.linear_counter - 1 : 0); + + // Envelope tick (square and noise channels) (all ticks) + int ref_ch_env_delay = ch.env_delay; + if (c != 2 && count(ref ref_ch_env_delay, ch.EnvDecayRate)) + if (ch.envelope > 0 || (bool)ch.EnvDecayLoopEnable) + ch.envelope.Value = (ch.envelope - 1) & 15; + ch.env_delay.Value = ref_ch_env_delay; + } - samples[i * 2 + 0] = (short)(samp); - samples[i * 2 + 1] = (short)(samp); - //bw.Write((short)samp); } } - } //class BisqAPU + // Mix the audio: Get the momentary sample from each channel and mix them. + // #define s(c) tick() + //v = [](float m,float n, float d) { return n!=0.f ? m/n : d; }; + Func v = (float m,float n, float d) => ( n!=0.0f ? m/n : d ); + short sample = (short)(30000 * + ( + // Square 0 and 1 + v(95.88f, (100.0f + v(8128.0f, tick(0) + tick(1), -100.0f)), 0.0f) + // Triangle, noise, DMC + + v(159.79f, (100.0f + v(1.0f, tick(2)/8227.0f + tick(3)/12241.0f + tick(4)/22638.0f, -1000.0f)), 0.0f) + // GamePak audio (these volume values are bogus, but sound acceptable) + + v(95.88f, (100.0f + v(32512.0f, /*GamePak::ExtAudio()*/ 0, -100.0f)), 0.0f) + - 0.5f + )); - } + EmitSample(sample); + ////this (and the similar line below) is a crude hack + ////we should be generating logic to suppress the $4015 clear when the assert signal is set instead + ////be sure to test "apu_test" if you mess with this + //sequencer_irq |= sequencer_irq_assert; + } + + void WriteC(int chno, int index, byte value) + { + APUchannel ch = channels[chno]; + switch (index) + { + case 0: + if ((bool)ch.LinearCounterDisable) ch.linear_counter.Value = value & 0x7F; + ch.reg0.Value = value; + break; + case 1: + ch.reg1.Value = value; + ch.sweep_delay.Value = ch.SweepRate; + break; + case 2: + ch.reg2.Value = value; + break; + case 3: + ch.reg3.Value = value; + if (data.ChannelsEnabled[chno]!=0) + ch.length_counter.Value = LengthCounters[ch.LengthCounterInit]; + ch.linear_counter.Value = ch.LinearCounterInit; + ch.env_delay.Value = ch.EnvDecayRate; + ch.envelope.Value = 15; + if (index < 8) ch.phase.Value = 0; + break; + case 0x12: + ch.reg0.Value = value; + ch.address.Value = ((int)ch.reg0 | 0x300) << 6; + break; + case 0x10: + ch.reg3.Value = value; + ch.WaveLength.Value = (uint)(DMCperiods[value & 0x0F] - 1); + if (!(bool)ch.IRQenable) { dmc_irq = false; SyncIRQ(); } + break; + case 0x13: // sample length + ch.reg1.Value = value; + if (ch.length_counter == 0) + ch.length_counter.Value = ch.PCMlength * 16 + 1; + break; + case 0x11: // dac value + ch.linear_counter.Value = value & 0x7F; + break; + case 0x15: + for (int c = 0; c < 5; ++c) + data.ChannelsEnabled[c] = (uint)((value >> c) & 1); //noteworthy tweak + for (int c = 0; c < 5; ++c) + if (data.ChannelsEnabled[c]==0) + channels[c].length_counter.Value = 0; + else if (c == 4 && channels[c].length_counter == 0) + { + APUchannel chh = channels[c]; + chh.length_counter.Value = chh.PCMlength * 16 + 1; + chh.address.Value = ((int)chh.reg0 | 0x300) << 6; + chh.phase.Value = 0; + } + //CPU::reg.APU_DMC_IRQ = false; + dmc_irq = false; SyncIRQ(); + break; + case 0x17: + data.IRQdisable.Value = (uint)(value & 0x40); + data.FiveCycleDivider.Value = (uint)(value & 0x80); + // apu_test 1-len_ctr: Writing $80 to $4017 should clock length immediately + // But Writing $00 to $4017 shouldn't clock length immediately + data.frame_delay.Value &= 1; + data.frame.Value = 0; + data.IRQ_delay.Value = 0x7FFF; + if ((bool)data.IRQdisable) { sequencer_irq = false; SyncIRQ(); } + if (!(bool)data.FiveCycleDivider) + { + data.frame.Value = 1; + data.frame_delay.Value += frame_period; + + if (!(bool)data.IRQdisable) + { + data.IRQ_delay.Value = data.frame_delay + frame_period * 3 + 1 - 3; + // ^ "- 3" makes apu_test "4-jitter" not complain + // that "Frame irq is set too late" + } + } + break; + } + } + + public byte ReadReg(int addr) + { + byte res = 0; + for (int c = 0; c < 5; ++c) res |= (byte)(((bool)channels[c].length_counter ? 1 << c : 0)); + if (sequencer_irq) res |= 0x40; sequencer_irq = false; SyncIRQ(); + if (dmc_irq) res |= 0x80; + return res; + } + + public void WriteReg(int addr, byte val) + { + int index = addr - 0x4000; + WriteC((index / 4) % 5, index < 0x10 ? index % 4 : index, val); + } + + public void NESSoftReset() + { + } + + public void DiscardSamples() + { + metaspu.buffer.clear(); + } + + public void SyncState(Serializer ser) + { + } + + double accumulate; + double timer; + Queue squeue = new Queue(); + int last_hwsamp; + int panic_sample, panic_count; + void EmitSample(int samp) + { + //kill the annoying hum that is a consequence of the shitty code below + if (samp == panic_sample) + panic_count++; + else panic_count = 0; + if (panic_count > 178977) + samp = 0; + else + panic_sample = samp; + + int this_samp = samp; + const double kMixRate = 44100.0 / 1789772.0; + const double kInvMixRate = (1 / kMixRate); + timer += kMixRate; + accumulate += samp; + if (timer <= 1) + return; + + accumulate -= samp; + timer -= 1; + double ratio = (timer / kMixRate); + double fractional = (this_samp - last_hwsamp) * ratio; + double factional_remainder = (this_samp - last_hwsamp) * (1 - ratio); + accumulate += fractional; + + accumulate *= 436; //32768/(15*4) -- adjust later for other sound channels + int outsamp = (int)(accumulate / kInvMixRate); + if (CFG_USE_METASPU) + metaspu.buffer.enqueue_sample((short)outsamp, (short)outsamp); + else squeue.Enqueue(outsamp); + accumulate = factional_remainder; + + last_hwsamp = this_samp; + } + + MetaspuSoundProvider metaspu = new MetaspuSoundProvider(ESynchMethod.ESynchMethod_V); + public int MaxVolume { get; set; } // not supported + + void ISoundProvider.GetSamples(short[] samples) + { + if (CFG_USE_METASPU) + { + metaspu.GetSamples(samples); + //foreach(short sample in samples) bw.Write((short)sample); + } + else + MyGetSamples(samples); + + //mix in the cart's extra sound circuit + nes.Board.ApplyCustomAudio(samples); + } + + //static BinaryWriter bw = new BinaryWriter(new FileStream("d:\\out.raw",FileMode.Create,FileAccess.Write,FileShare.Read)); + void MyGetSamples(short[] samples) + { + //Console.WriteLine("a: {0} with todo: {1}",squeue.Count,samples.Length/2); + + for (int i = 0; i < samples.Length / 2; i++) + { + int samp = 0; + if (squeue.Count != 0) + samp = squeue.Dequeue(); + + samples[i * 2 + 0] = (short)(samp); + samples[i * 2 + 1] = (short)(samp); + //bw.Write((short)samp); + } + } + + } //class BisqAPU } \ No newline at end of file diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BoardSystem.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BoardSystem.cs index 1dea4e399a..9f178bdcc5 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BoardSystem.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/BoardSystem.cs @@ -333,50 +333,50 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES return board; } - public string BoardName { get { return board.GetType().Name; } } + public string BoardName { get { return Board.GetType().Name; } } void BoardSystemHardReset() { INESBoard newboard; // fds has a unique activation setup - if (board is FDS) + if (Board is FDS) { var newfds = new FDS(); - var oldfds = board as FDS; + var oldfds = Board as FDS; newfds.biosrom = oldfds.biosrom; newfds.SetDiskImage(oldfds.GetDiskImage()); newboard = newfds; } else { - newboard = CreateBoardInstance(board.GetType()); + newboard = CreateBoardInstance(Board.GetType()); } newboard.Create(this); // i suppose the old board could have changed its initial register values, although it really shouldn't // you can't use SyncSettings.BoardProperties here because they very well might be different than before // in case the user actually changed something in the UI - newboard.InitialRegisterValues = board.InitialRegisterValues; + newboard.InitialRegisterValues = Board.InitialRegisterValues; newboard.Configure(origin); - newboard.ROM = board.ROM; - newboard.VROM = board.VROM; - if (board.WRAM != null) - newboard.WRAM = new byte[board.WRAM.Length]; - if (board.VRAM != null) - newboard.VRAM = new byte[board.VRAM.Length]; + newboard.ROM = Board.ROM; + newboard.VROM = Board.VROM; + if (Board.WRAM != null) + newboard.WRAM = new byte[Board.WRAM.Length]; + if (Board.VRAM != null) + newboard.VRAM = new byte[Board.VRAM.Length]; newboard.PostConfigure(); // the old board's sram must be restored if (newboard is FDS) { var newfds = newboard as FDS; - var oldfds = board as FDS; + var oldfds = Board as FDS; newfds.StoreSaveRam(oldfds.ReadSaveRam()); } - else if (board.SaveRam != null) + else if (Board.SaveRam != null) { - Buffer.BlockCopy(board.SaveRam, 0, newboard.SaveRam, 0, board.SaveRam.Length); + Buffer.BlockCopy(Board.SaveRam, 0, newboard.SaveRam, 0, Board.SaveRam.Length); } - board.Dispose(); - board = newboard; + Board.Dispose(); + Board = newboard; } diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Core.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Core.cs index 8fab5a042e..e5931a67b1 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Core.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/Core.cs @@ -22,7 +22,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public byte[] CIRAM; //AKA nametables string game_name = string.Empty; //friendly name exposed to user and used as filename base CartInfo cart; //the current cart prototype. should be moved into the board, perhaps - INESBoard board; //the board hardware that is currently driving things + internal INESBoard Board; //the board hardware that is currently driving things EDetectionOrigin origin = EDetectionOrigin.None; int sprdma_countdown; bool _irq_apu; //various irq signals that get merged to the cpu irq pin @@ -56,7 +56,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES /// public INESBoard GetBoard() { - return board; + return Board; } public void Dispose() @@ -106,7 +106,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES samples[i + 1] = samples[i]; //mix in the cart's extra sound circuit - nes.board.ApplyCustomAudio(samples); + nes.Board.ApplyCustomAudio(samples); } public void GetSamples(out short[] samples, out int nsamp) @@ -126,7 +126,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES for (int i = 0; i < nsamp * 2; i += 2) samples[i + 1] = samples[i]; - nes.board.ApplyCustomAudio(samples); + nes.Board.ApplyCustomAudio(samples); } public void DiscardSamples() @@ -170,9 +170,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES // controls other than the deck ControllerDefinition.BoolButtons.Add("Power"); ControllerDefinition.BoolButtons.Add("Reset"); - if (board is FDS) + if (Board is FDS) { - var b = board as FDS; + var b = Board as FDS; ControllerDefinition.BoolButtons.Add("FDS Eject"); for (int i = 0; i < b.NumSides; i++) ControllerDefinition.BoolButtons.Add("FDS Insert " + i); @@ -244,7 +244,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES lagged = true; if (resetSignal) { - board.NESSoftReset(); + Board.NESSoftReset(); cpu.NESSoftReset(); apu.NESSoftReset(); ppu.NESSoftReset(); @@ -261,9 +261,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES resetSignal = Controller["Reset"]; hardResetSignal = Controller["Power"]; - if (board is FDS) + if (Board is FDS) { - var b = board as FDS; + var b = Board as FDS; if (Controller["FDS Eject"]) b.Eject(); for (int i = 0; i < b.NumSides; i++) @@ -320,12 +320,12 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES cpu_deadcounter--; else { - cpu.IRQ = _irq_apu || board.IRQSignal; + cpu.IRQ = _irq_apu || Board.IRQSignal; cpu.ExecuteOne(); } apu.RunOne(); - board.ClockCPU(); + Board.ClockCPU(); ppu.PostCpuInstructionOne(); } } @@ -493,7 +493,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES if (addr >= 0x4020) { - ret = board.PeekCart(addr); //easy optimization, since rom reads are so common, move this up (reordering the rest of these elseifs is not easy) + ret = Board.PeekCart(addr); //easy optimization, since rom reads are so common, move this up (reordering the rest of these elseifs is not easy) } else if (addr < 0x0800) { @@ -534,7 +534,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES if (addr >= 0x8000) { - ret = board.ReadPRG(addr - 0x8000); //easy optimization, since rom reads are so common, move this up (reordering the rest of these elseifs is not easy) + ret = Board.ReadPRG(addr - 0x8000); //easy optimization, since rom reads are so common, move this up (reordering the rest of these elseifs is not easy) } else if (addr < 0x0800) { @@ -554,11 +554,11 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } else if (addr < 0x6000) { - ret = board.ReadEXP(addr - 0x4000); + ret = Board.ReadEXP(addr - 0x4000); } else { - ret = board.ReadWRAM(addr - 0x6000); + ret = Board.ReadWRAM(addr - 0x6000); } //handle breakpoints and stuff. @@ -613,15 +613,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES } else if (addr < 0x6000) { - board.WriteEXP(addr - 0x4000, value); + Board.WriteEXP(addr - 0x4000, value); } else if (addr < 0x8000) { - board.WriteWRAM(addr - 0x6000, value); + Board.WriteWRAM(addr - 0x6000, value); } else { - board.WritePRG(addr - 0x8000, value); + Board.WritePRG(addr - 0x8000, value); } MemoryCallbacks.CallWrites(addr); diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IMemoryDomains.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IMemoryDomains.cs index d036adf676..95fa877e9c 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IMemoryDomains.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IMemoryDomains.cs @@ -30,46 +30,46 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES domains.Add(CIRAMdomain); domains.Add(OAMdoman); - if (!(board is FDS) && board.SaveRam != null) + if (!(Board is FDS) && Board.SaveRam != null) { - var BatteryRam = new MemoryDomain("Battery RAM", board.SaveRam.Length, MemoryDomain.Endian.Little, - addr => board.SaveRam[addr], (addr, value) => board.SaveRam[addr] = value); + var BatteryRam = new MemoryDomain("Battery RAM", Board.SaveRam.Length, MemoryDomain.Endian.Little, + addr => Board.SaveRam[addr], (addr, value) => Board.SaveRam[addr] = value); domains.Add(BatteryRam); } var PRGROM = new MemoryDomain("PRG ROM", cart.prg_size * 1024, MemoryDomain.Endian.Little, - addr => board.ROM[addr], (addr, value) => board.ROM[addr] = value); + addr => Board.ROM[addr], (addr, value) => Board.ROM[addr] = value); domains.Add(PRGROM); - if (board.VROM != null) + if (Board.VROM != null) { var CHRROM = new MemoryDomain("CHR VROM", cart.chr_size * 1024, MemoryDomain.Endian.Little, - addr => board.VROM[addr], (addr, value) => board.VROM[addr] = value); + addr => Board.VROM[addr], (addr, value) => Board.VROM[addr] = value); domains.Add(CHRROM); } - if (board.VRAM != null) + if (Board.VRAM != null) { - var VRAM = new MemoryDomain("VRAM", board.VRAM.Length, MemoryDomain.Endian.Little, - addr => board.VRAM[addr], (addr, value) => board.VRAM[addr] = value); + var VRAM = new MemoryDomain("VRAM", Board.VRAM.Length, MemoryDomain.Endian.Little, + addr => Board.VRAM[addr], (addr, value) => Board.VRAM[addr] = value); domains.Add(VRAM); } - if (board.WRAM != null) + if (Board.WRAM != null) { - var WRAM = new MemoryDomain("WRAM", board.WRAM.Length, MemoryDomain.Endian.Little, - addr => board.WRAM[addr], (addr, value) => board.WRAM[addr] = value); + var WRAM = new MemoryDomain("WRAM", Board.WRAM.Length, MemoryDomain.Endian.Little, + addr => Board.WRAM[addr], (addr, value) => Board.WRAM[addr] = value); domains.Add(WRAM); } // if there were more boards with special ram sets, we'd want to do something more general - if (board is FDS) + if (Board is FDS) { - domains.Add((board as FDS).GetDiskPeeker()); + domains.Add((Board as FDS).GetDiskPeeker()); } - else if (board is ExROM) + else if (Board is ExROM) { - domains.Add((board as ExROM).GetExRAM()); + domains.Add((Board as ExROM).GetExRAM()); } _memoryDomains = new MemoryDomainList(domains); diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs index f11d1ef447..cc9748ed15 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.INESPPUViewable.cs @@ -50,14 +50,14 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public byte PeekPPU(int addr) { - return board.PeekPPU(addr); + return Board.PeekPPU(addr); } public byte[] GetExTiles() { - if (board is ExROM) + if (Board is ExROM) { - return board.VROM ?? board.VRAM; + return Board.VROM ?? Board.VRAM; } else { @@ -67,14 +67,14 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES public bool ExActive { - get { return board is ExROM && (board as ExROM).ExAttrActive; } + get { return Board is ExROM && (Board as ExROM).ExAttrActive; } } public byte[] GetExRam() { - if (board is ExROM) + if (Board is ExROM) { - return (board as ExROM).GetExRAMArray(); + return (Board as ExROM).GetExRAMArray(); } else { diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.ISaveRam.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.ISaveRam.cs index 5d8d063622..1b8306ac7a 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.ISaveRam.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.ISaveRam.cs @@ -12,34 +12,34 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES { get { - if (board == null) return false; - if (board is FDS) return true; - if (board.SaveRam == null) return false; + if (Board == null) return false; + if (Board is FDS) return true; + if (Board.SaveRam == null) return false; return true; } } public byte[] CloneSaveRam() { - if (board is FDS) - return (board as FDS).ReadSaveRam(); + if (Board is FDS) + return (Board as FDS).ReadSaveRam(); - if (board == null || board.SaveRam == null) + if (Board == null || Board.SaveRam == null) return null; - return (byte[])board.SaveRam.Clone(); + return (byte[])Board.SaveRam.Clone(); } public void StoreSaveRam(byte[] data) { - if (board is FDS) + if (Board is FDS) { - (board as FDS).StoreSaveRam(data); + (Board as FDS).StoreSaveRam(data); return; } - if (board == null || board.SaveRam == null) + if (Board == null || Board.SaveRam == null) return; - Array.Copy(data, board.SaveRam, data.Length); + Array.Copy(data, Board.SaveRam, data.Length); } } } diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IStatable.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IStatable.cs index 49b049a75c..695ad16f69 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IStatable.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.IStatable.cs @@ -62,8 +62,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES ser.Sync("cpu_stepcounter", ref cpu_stepcounter); ser.Sync("cpu_deadcounter", ref cpu_deadcounter); ser.BeginSection("Board"); - board.SyncState(ser); - if (board is NESBoardBase && !((NESBoardBase)board).SyncStateFlag) + Board.SyncState(ser); + if (Board is NESBoardBase && !((NESBoardBase)Board).SyncStateFlag) throw new InvalidOperationException("the current NES mapper didnt call base.SyncState"); ser.EndSection(); ppu.SyncState(ser); diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs index 75972813c2..bca6de5ca1 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/NES.cs @@ -46,10 +46,10 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES BootGodDB.Initialize(); videoProvider = new MyVideoProvider(this); Init(game, rom, fdsbios); - if (board is FDS) + if (Board is FDS) { DriveLightEnabled = true; - (board as FDS).SetDriveLightCallback((val) => DriveLightOn = val); + (Board as FDS).SetDriveLightCallback((val) => DriveLightOn = val); } PutSettings((NESSettings)Settings ?? new NESSettings()); @@ -60,9 +60,9 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES ser.Register(Tracer); ser.Register(videoProvider); - if (board is BANDAI_FCG_1) + if (Board is BANDAI_FCG_1) { - var reader = (board as BANDAI_FCG_1).reader; + var reader = (Board as BANDAI_FCG_1).reader; // not all BANDAI FCG 1 boards have a barcode reader if (reader != null) ser.Register(reader); @@ -426,15 +426,15 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES fdsboard.InitialRegisterValues = InitialMapperRegisterValues; fdsboard.Configure(origin); - board = fdsboard; + Board = fdsboard; //create the vram and wram if necessary if (cart.wram_size != 0) - board.WRAM = new byte[cart.wram_size * 1024]; + Board.WRAM = new byte[cart.wram_size * 1024]; if (cart.vram_size != 0) - board.VRAM = new byte[cart.vram_size * 1024]; + Board.VRAM = new byte[cart.vram_size * 1024]; - board.PostConfigure(); + Board.PostConfigure(); Console.WriteLine("Using NTSC display type for FDS disk image"); _display_type = Common.DisplayType.NTSC; @@ -608,12 +608,12 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES LoadWriteLine("END NES rom analysis"); LoadWriteLine("------"); - board = CreateBoardInstance(boardType); + Board = CreateBoardInstance(boardType); cart = choice; - board.Create(this); - board.InitialRegisterValues = InitialMapperRegisterValues; - board.Configure(origin); + Board.Create(this); + Board.InitialRegisterValues = InitialMapperRegisterValues; + Board.Configure(origin); if (origin == EDetectionOrigin.BootGodDB) { @@ -647,26 +647,26 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES if (iNesHeaderInfo != null) { //pluck the necessary bytes out of the file - board.ROM = new byte[choice.prg_size * 1024]; - Array.Copy(file, 16, board.ROM, 0, board.ROM.Length); + Board.ROM = new byte[choice.prg_size * 1024]; + Array.Copy(file, 16, Board.ROM, 0, Board.ROM.Length); if (choice.chr_size > 0) { - board.VROM = new byte[choice.chr_size * 1024]; + Board.VROM = new byte[choice.chr_size * 1024]; int vrom_offset = iNesHeaderInfo.prg_size * 1024; // if file isn't long enough for VROM, truncate - int vrom_copy_size = Math.Min(board.VROM.Length, file.Length - 16 - vrom_offset); - Array.Copy(file, 16 + vrom_offset, board.VROM, 0, vrom_copy_size); - if (vrom_copy_size < board.VROM.Length) - LoadWriteLine("Less than the expected VROM was found in the file: {0} < {1}", vrom_copy_size, board.VROM.Length); + int vrom_copy_size = Math.Min(Board.VROM.Length, file.Length - 16 - vrom_offset); + Array.Copy(file, 16 + vrom_offset, Board.VROM, 0, vrom_copy_size); + if (vrom_copy_size < Board.VROM.Length) + LoadWriteLine("Less than the expected VROM was found in the file: {0} < {1}", vrom_copy_size, Board.VROM.Length); } if (choice.prg_size != iNesHeaderInfo.prg_size || choice.chr_size != iNesHeaderInfo.chr_size) LoadWriteLine("Warning: Detected choice has different filesizes than the INES header!"); } else { - board.ROM = unif.PRG; - board.VROM = unif.CHR; + Board.ROM = unif.PRG; + Board.VROM = unif.CHR; } LoadReport.Flush(); @@ -676,11 +676,11 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES //create the vram and wram if necessary if (cart.wram_size != 0) - board.WRAM = new byte[cart.wram_size * 1024]; + Board.WRAM = new byte[cart.wram_size * 1024]; if (cart.vram_size != 0) - board.VRAM = new byte[cart.vram_size * 1024]; + Board.VRAM = new byte[cart.vram_size * 1024]; - board.PostConfigure(); + Board.PostConfigure(); // set up display type diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.cs index 5196ded5ee..ec9fc98d7b 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.cs @@ -81,8 +81,8 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES //when the ppu issues a write it goes through here and into the game board public void ppubus_write(int addr, byte value) { - nes.board.AddressPPU(addr); - nes.board.WritePPU(addr, value); + nes.Board.AddressPPU(addr); + nes.Board.WritePPU(addr, value); } //when the ppu issues a read it goes through here and into the game board @@ -92,14 +92,14 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES if (!reg_2001.PPUON && ppu) return 0xFF; - nes.board.AddressPPU(addr); - return nes.board.ReadPPU(addr); + nes.Board.AddressPPU(addr); + return nes.Board.ReadPPU(addr); } //debug tools peek into the ppu through this public byte ppubus_peek(int addr) { - return nes.board.PeekPPU(addr); + return nes.Board.PeekPPU(addr); } public enum PPUPHASE @@ -229,7 +229,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES Reg2002_vblank_clear_pending = false; } - nes.board.ClockPPU(); + nes.Board.ClockPPU(); } } diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.regs.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.regs.cs index 69ed8276ce..463387340a 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.regs.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NES/PPU.regs.cs @@ -443,7 +443,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES //normally the address isnt observed by the board till it gets clocked by a read or write. //but maybe thats just because a ppu read/write shoves it on the address bus //apparently this shoves it on the address bus, too, or else blargg's mmc3 tests dont pass - nes.board.AddressPPU(ppur.get_2007access()); + nes.Board.AddressPPU(ppur.get_2007access()); } vtoggle ^= true; @@ -480,7 +480,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES ppur.increment2007(ppur.status.rendering && reg_2001.PPUON, reg_2000.vram_incr32 != 0); //see comments in $2006 - nes.board.AddressPPU(ppur.get_2007access()); + nes.Board.AddressPPU(ppur.get_2007access()); } byte read_2007() { @@ -502,7 +502,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES ppur.increment2007(ppur.status.rendering && reg_2001.PPUON, reg_2000.vram_incr32 != 0); //see comments in $2006 - nes.board.AddressPPU(ppur.get_2007access()); + nes.Board.AddressPPU(ppur.get_2007access()); return ret; }