From 64ded912df3c7ed35e9ba4353df5cda3d5145466 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Wed, 2 Mar 2016 23:45:13 -0600 Subject: [PATCH 01/15] C64: Implement more of the VIA timers. --- .../Computers/Commodore64/C64.Motherboard.cs | 2 +- .../Commodore64/C64.MotherboardInterface.cs | 1 - .../Commodore64/MOS/Via.Registers.cs | 8 + .../Computers/Commodore64/MOS/Via.cs | 265 +++++++++++++----- 4 files changed, 205 insertions(+), 71 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 7c66ee50d1..d288c463de 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -156,7 +156,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public void Execute() { - _vicBank = (0x3 - (Cia1.EffectivePrA & 0x3)) << 14; + _vicBank = (0x3 - ((Cia1.PrA | ~Cia1.DdrA) & 0x3)) << 14; Vic.ExecutePhase(); Cassette.ExecutePhase(); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs index d48708c6b8..95f99362c2 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs @@ -10,7 +10,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private int _lastReadVicAddress = 0x3FFF; private int _lastReadVicData = 0xFF; private int _vicBank = 0xC000; - private int _tempCia1Cra; private bool CassPort_ReadDataOutput() { diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs index be694fc3d0..84119b3062 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs @@ -102,10 +102,18 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { case 0x0: _ifr &= 0xE7; + if (_pcrCb2Control == PCR_CONTROL_HANDSHAKE_OUTPUT || _pcrCb2Control == PCR_CONTROL_PULSE_OUTPUT) + { + _handshakeCb2NextClock = true; + } WriteRegister(addr, val); break; case 0x1: _ifr &= 0xFC; + if (_pcrCa2Control == PCR_CONTROL_HANDSHAKE_OUTPUT || _pcrCa2Control == PCR_CONTROL_PULSE_OUTPUT) + { + _handshakeCa2NextClock = true; + } WriteRegister(addr, val); break; case 0x4: diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs index b9cc6f4eb3..5110e3eb94 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs @@ -9,70 +9,118 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { public sealed partial class Via { - private const int PCR_INT_CONTROL_NEGATIVE_EDGE = 0x00; - private const int PCR_INT_CONTROL_POSITIVE_EDGE = 0x01; - private const int PCR_CONTROL_INPUT_NEGATIVE_ACTIVE_EDGE = 0x00; - private const int PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE = 0x02; - private const int PCR_CONTROL_INPUT_POSITIVE_ACTIVE_EDGE = 0x04; - private const int PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE = 0x06; - private const int PCR_CONTROL_HANDSHAKE_OUTPUT = 0x08; - private const int PCR_CONTROL_PULSE_OUTPUT = 0x0A; - private const int PCR_CONTROL_LOW_OUTPUT = 0x0C; - private const int PCR_CONTROL_HIGH_OUTPUT = 0x0E; - private const int ACR_SR_CONTROL_DISABLED = 0x00; - private const int ACR_SR_CONTROL_SHIFT_IN_T2_ONCE = 0x04; - private const int ACR_SR_CONTROL_SHIFT_IN_PHI2 = 0x08; - private const int ACR_SR_CONTROL_SHIFT_IN_CLOCK = 0x0C; - private const int ACR_SR_CONTROL_SHIFT_OUT_T2 = 0x10; - private const int ACR_SR_CONTROL_SHIFT_OUT_T2_ONCE = 0x14; - private const int ACR_SR_CONTROL_SHIFT_OUT_PHI2 = 0x18; - private const int ACR_SR_CONTROL_SHIFT_OUT_CLOCK = 0x1C; - private const int ACR_T2_CONTROL_TIMED = 0x00; - private const int ACR_T2_CONTROL_COUNT_ON_PB6 = 0x20; - private const int ACR_T1_CONTROL_INTERRUPT_ON_LOAD = 0x00; - private const int ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS = 0x40; - private const int ACR_T1_CONTROL_INTERRUPT_ON_LOAD_AND_ONESHOT_PB7 = 0x80; - private const int ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7 = 0xC0; + [SaveState.DoNotSave] private const int PCR_INT_CONTROL_NEGATIVE_EDGE = 0x00; + [SaveState.DoNotSave] private const int PCR_INT_CONTROL_POSITIVE_EDGE = 0x01; + [SaveState.DoNotSave] private const int PCR_CONTROL_INPUT_NEGATIVE_ACTIVE_EDGE = 0x00; + [SaveState.DoNotSave] private const int PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE = 0x02; + [SaveState.DoNotSave] private const int PCR_CONTROL_INPUT_POSITIVE_ACTIVE_EDGE = 0x04; + [SaveState.DoNotSave] private const int PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE = 0x06; + [SaveState.DoNotSave] private const int PCR_CONTROL_HANDSHAKE_OUTPUT = 0x08; + [SaveState.DoNotSave] private const int PCR_CONTROL_PULSE_OUTPUT = 0x0A; + [SaveState.DoNotSave] private const int PCR_CONTROL_LOW_OUTPUT = 0x0C; + [SaveState.DoNotSave] private const int PCR_CONTROL_HIGH_OUTPUT = 0x0E; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_DISABLED = 0x00; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_SHIFT_IN_T2_ONCE = 0x04; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_SHIFT_IN_PHI2 = 0x08; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_SHIFT_IN_CLOCK = 0x0C; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_SHIFT_OUT_T2 = 0x10; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_SHIFT_OUT_T2_ONCE = 0x14; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_SHIFT_OUT_PHI2 = 0x18; + [SaveState.DoNotSave] private const int ACR_SR_CONTROL_SHIFT_OUT_CLOCK = 0x1C; + [SaveState.DoNotSave] private const int ACR_T2_CONTROL_TIMED = 0x00; + [SaveState.DoNotSave] private const int ACR_T2_CONTROL_COUNT_ON_PB6 = 0x20; + [SaveState.DoNotSave] private const int ACR_T1_CONTROL_INTERRUPT_ON_LOAD = 0x00; + [SaveState.DoNotSave] private const int ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS = 0x40; + [SaveState.DoNotSave] private const int ACR_T1_CONTROL_INTERRUPT_ON_LOAD_AND_ONESHOT_PB7 = 0x80; + [SaveState.DoNotSave] private const int ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7 = 0xC0; + [SaveState.SaveWithName("PortOutputA")] private int _pra; + [SaveState.SaveWithName("PortDirectionA")] private int _ddra; + [SaveState.SaveWithName("PortOutputB")] private int _prb; + [SaveState.SaveWithName("PortDirectionB")] private int _ddrb; + [SaveState.SaveWithName("Timer1Counter")] private int _t1C; + [SaveState.SaveWithName("Timer1Latch")] private int _t1L; + [SaveState.SaveWithName("Timer2Counter")] private int _t2C; + [SaveState.SaveWithName("Timer2Latch")] private int _t2L; + [SaveState.SaveWithName("ShiftRegister")] private int _sr; + [SaveState.SaveWithName("AuxiliaryControlRegister")] private int _acr; + [SaveState.SaveWithName("PeripheralControlRegister")] private int _pcr; + [SaveState.SaveWithName("InterruptFlagRegister")] private int _ifr; + [SaveState.SaveWithName("InterruptEnableRegister")] private int _ier; + [SaveState.SaveWithName("Port")] private readonly Port _port; + [SaveState.SaveWithName("PortLatchA")] private int _paLatch; + [SaveState.SaveWithName("PortLatchB")] private int _pbLatch; + [SaveState.SaveWithName("CA1InterruptControl")] private int _pcrCa1IntControl; + [SaveState.SaveWithName("CA2Control")] private int _pcrCa2Control; + [SaveState.SaveWithName("CB1InterruptControl")] private int _pcrCb1IntControl; + [SaveState.SaveWithName("CB2Control")] private int _pcrCb2Control; + [SaveState.SaveWithName("PortLatchEnableA")] private bool _acrPaLatchEnable; + [SaveState.SaveWithName("PortLatchEnableB")] private bool _acrPbLatchEnable; + [SaveState.SaveWithName("ShiftRegisterControl")] private int _acrSrControl; + [SaveState.SaveWithName("Timer1Control")] private int _acrT1Control; + [SaveState.SaveWithName("Timer2Control")] private int _acrT2Control; + [SaveState.SaveWithName("PreviousCA1")] private bool _ca1L; + [SaveState.SaveWithName("PreviousCA2")] private bool _ca2L; + [SaveState.SaveWithName("PreviousCB1")] private bool _cb1L; + [SaveState.SaveWithName("PreviousCB2")] private bool _cb2L; + [SaveState.SaveWithName("PreviousPB6")] + private bool _pb6L; + [SaveState.SaveWithName("ResetCa2NextClock")] + private bool _resetCa2NextClock; + [SaveState.SaveWithName("ResetCb2NextClock")] + private bool _resetCb2NextClock; + + [SaveState.SaveWithName("HandshakeCa2NextClock")] + private bool _handshakeCa2NextClock; + [SaveState.SaveWithName("HandshakeCb2NextClock")] + private bool _handshakeCb2NextClock; + + [SaveState.SaveWithName("ShiftRegisterCounter")] private int _shiftCount; + [SaveState.SaveWithName("CA1")] public bool Ca1; + [SaveState.SaveWithName("CA2")] public bool Ca2; + [SaveState.SaveWithName("CB1")] public bool Cb1; + [SaveState.SaveWithName("CB2")] public bool Cb2; + [SaveState.SaveWithName("PB6")] + private bool _pb6; public Via() { @@ -128,78 +176,157 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS Ca2 = false; Cb1 = false; Cb2 = false; - } - private bool ProcessC2(bool c2, int control) - { - switch (control) - { - case PCR_CONTROL_INPUT_NEGATIVE_ACTIVE_EDGE: - return c2; - case PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE: - return c2; - case PCR_CONTROL_INPUT_POSITIVE_ACTIVE_EDGE: - return c2; - case PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE: - return c2; - case PCR_CONTROL_HANDSHAKE_OUTPUT: - return c2; - case PCR_CONTROL_PULSE_OUTPUT: - return c2; - case PCR_CONTROL_LOW_OUTPUT: - return false; - case PCR_CONTROL_HIGH_OUTPUT: - return true; - } - return c2; + _pb6L = false; + _pb6 = false; + _resetCa2NextClock = false; + _resetCb2NextClock = false; + _handshakeCa2NextClock = false; + _handshakeCb2NextClock = false; } public void ExecutePhase() { + // Process 'pulse' and 'handshake' outputs on CA2 and CB2 + + if (_resetCa2NextClock) + { + Ca2 = true; + _resetCa2NextClock = false; + } + else if (_handshakeCa2NextClock) + { + Ca2 = false; + _resetCa2NextClock = _pcrCa2Control == PCR_CONTROL_PULSE_OUTPUT; + _handshakeCa2NextClock = false; + } + + if (_resetCb2NextClock) + { + Cb2 = true; + _resetCb2NextClock = false; + } + else if (_handshakeCb2NextClock) + { + Cb2 = false; + _resetCb2NextClock = _pcrCb2Control == PCR_CONTROL_PULSE_OUTPUT; + _handshakeCb2NextClock = false; + } + + // Count timers + _t1C--; if (_t1C < 0) { - if (_acrT1Control == ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS || - _acrT1Control == ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7) + switch (_acrT1Control) { - _t1C = _t1L; + case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS: + _t1C = _t1L; + break; + case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7: + _t1C = _t1L; + _prb ^= 0x80; + break; + default: + _t1C = 0xFFFF; + break; } _ifr |= 0x40; } - if (_acrT2Control == ACR_T2_CONTROL_TIMED) + switch (_acrT2Control) { - _t2C--; - if (_t2C < 0) - { - _ifr |= 0x20; - _t2C = _t2L; - } + case ACR_T2_CONTROL_TIMED: + _t2C--; + if (_t2C < 0) + { + _ifr |= 0x20; + _t2C = 0xFFFF; + } + break; + case ACR_T2_CONTROL_COUNT_ON_PB6: + _pb6L = _pb6; + _pb6 = (_port.ReadExternalPrb() & 0x40) != 0; + if (!_pb6 && _pb6L) + { + _t2C--; + if (_t2C < 0) + { + _ifr |= 0x20; + _t2C = 0xFFFF; + } + } + break; } - Ca2 = ProcessC2(Ca2, _pcrCa2Control); - Cb2 = ProcessC2(Cb2, _pcrCb2Control); + // Process CA2 - // unknown behavior - - if (_acrT1Control != ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS && - _acrT1Control != ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7) + switch (_pcrCa2Control) { - // unknown ACR T1 control + case PCR_CONTROL_INPUT_NEGATIVE_ACTIVE_EDGE: + case PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE: + if (_ca2L && !Ca2) + _ifr |= 0x01; + break; + case PCR_CONTROL_INPUT_POSITIVE_ACTIVE_EDGE: + case PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE: + if (!_ca2L && Ca2) + _ifr |= 0x01; + break; + case PCR_CONTROL_HANDSHAKE_OUTPUT: + if (_ca1L && !Ca1) + { + Ca2 = true; + _ifr |= 0x01; + } + break; + case PCR_CONTROL_PULSE_OUTPUT: + break; + case PCR_CONTROL_LOW_OUTPUT: + Ca2 = false; + break; + case PCR_CONTROL_HIGH_OUTPUT: + Ca2 = true; + break; } - if (_acrT2Control != ACR_T2_CONTROL_TIMED) - { - // unknown ACR T2 control - } + // Process CB2 + switch (_pcrCb2Control) + { + case PCR_CONTROL_INPUT_NEGATIVE_ACTIVE_EDGE: + case PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE: + if (_cb2L && !Cb2) + _ifr |= 0x08; + break; + case PCR_CONTROL_INPUT_POSITIVE_ACTIVE_EDGE: + case PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE: + if (!_cb2L && Cb2) + _ifr |= 0x08; + break; + case PCR_CONTROL_HANDSHAKE_OUTPUT: + if (_cb1L && !Cb1) + { + Cb2 = true; + _ifr |= 0x08; + } + break; + case PCR_CONTROL_PULSE_OUTPUT: + break; + case PCR_CONTROL_LOW_OUTPUT: + Cb2 = false; + break; + case PCR_CONTROL_HIGH_OUTPUT: + Cb2 = true; + break; + } // interrupt generation if ((_pcrCb1IntControl == PCR_INT_CONTROL_POSITIVE_EDGE && Cb1 && !_cb1L) || (_pcrCb1IntControl == PCR_INT_CONTROL_NEGATIVE_EDGE && !Cb1 && _cb1L)) { - _ifr |= 0x01; + _ifr |= 0x10; if (_acrPbLatchEnable) { _pbLatch = _port.ReadExternalPrb(); From cff1708be6dc1a4b8833a7fca9f6875e24b9d05a Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Thu, 3 Mar 2016 00:13:55 -0600 Subject: [PATCH 02/15] C64: Perform sync check before shifting. --- .../Serial/Drive1541.FluxTransitions.cs | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs index cf1df7607b..6a393af42d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs @@ -101,16 +101,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial _diskSupplementaryCounter++; if ((_diskSupplementaryCounter & 0x3) == 0x2) { - _byteReady = false; _bitsRemainingInLatchedByte--; - if (_bitsRemainingInLatchedByte <= 0) - { - _bitsRemainingInLatchedByte = 8; - - // SOE (sync output enabled) - _byteReady = Via1.Ca2; - } - + _byteReady = false; _bitHistory = (_bitHistory << 1) | ((_diskSupplementaryCounter & 0xC) == 0x0 ? 1 : 0); _sync = false; if (Via1.Cb2 && (_bitHistory & 0x3FF) == 0x3FF) @@ -120,6 +112,14 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial _byteReady = false; } + if (_bitsRemainingInLatchedByte <= 0) + { + _bitsRemainingInLatchedByte = 8; + + // SOE (sync output enabled) + _byteReady = Via1.Ca2; + } + // negative transition activates SO pin on CPU _previousCa1 = Via1.Ca1; Via1.Ca1 = !_byteReady; From 28ced70f285b5012d088392ba48f743bc234e492 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Thu, 3 Mar 2016 15:43:43 -0600 Subject: [PATCH 03/15] C64: Populate half-tracks when loading images. --- .../Computers/Commodore64/Media/Disk.cs | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs index 30244cb8f4..a8a43e6f6b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs @@ -104,6 +104,17 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media private void FillMissingTracks() { + // Fill half tracks (should assist with EA "fat-track" protections) + for (var i = 1; i < _tracks.Length; i += 2) + { + if (_tracks[i] == null && _tracks[i - 1] != null) + { + _tracks[i] = new int[FluxEntriesPerTrack]; + Array.Copy(_tracks[i - 1], _tracks[i], FluxEntriesPerTrack); + } + } + + // Fill vacant tracks for (var i = 0; i < _tracks.Length; i++) { if (_tracks[i] == null) From a435cc912c78805cfee1994c4c4728b401faf42f Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Thu, 3 Mar 2016 18:03:48 -0600 Subject: [PATCH 04/15] C64: Use Chip6522 factory. --- .../BizHawk.Emulation.Cores.csproj | 1 + .../Computers/Commodore64/MOS/Chip6522.cs | 20 ++++ .../Commodore64/MOS/Via.Registers.cs | 10 +- .../Computers/Commodore64/MOS/Via.cs | 104 ++++++++++++------ .../Computers/Commodore64/Serial/Drive1541.cs | 4 +- .../Commodore64/Serial/SerialPortDevice.cs | 6 +- 6 files changed, 107 insertions(+), 38 deletions(-) create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6522.cs diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index 99f23f3401..6c21f9741d 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -228,6 +228,7 @@ + diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6522.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6522.cs new file mode 100644 index 0000000000..81bd05e946 --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6522.cs @@ -0,0 +1,20 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS +{ + public static class Chip6522 + { + public static Via Create(Func readPrA, Func readPrB) + { + return new Via(readPrA, readPrB); + } + + public static Via Create(Func readClock, Func readData, Func readAtn, int driveNumber) + { + return new Via(readClock, readData, readAtn, driveNumber); + } + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs index 84119b3062..2db8c5d966 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs @@ -124,9 +124,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _t1L = (_t1L & 0xFF) | ((val & 0xFF) << 8); _ifr &= 0xBF; _t1C = _t1L; + _t1CLoaded = true; + _t1Delayed = 1; break; case 0x7: _t1L = (_t1L & 0xFF) | ((val & 0xFF) << 8); + _ifr &= 0xBF; break; case 0x8: _t2L = (_t2L & 0xFF00) | (val & 0xFF); @@ -134,7 +137,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS case 0x9: _t2L = (_t2L & 0xFF) | ((val & 0xFF) << 8); _ifr &= 0xDF; - _t2C = _t2L; + if (_acrT2Control == ACR_T2_CONTROL_TIMED) + { + _t2C = _t2L; + _t2CLoaded = true; + } + _t2Delayed = 1; break; case 0xA: _ifr &= 0xFB; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs index 5110e3eb94..ad60babc6b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs @@ -122,6 +122,17 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS [SaveState.SaveWithName("PB6")] private bool _pb6; + [SaveState.SaveWithName("InterruptNextClock")] + private int _interruptNextClock; + [SaveState.SaveWithName("T1Loaded")] + private bool _t1CLoaded; + [SaveState.SaveWithName("T2Loaded")] + private bool _t2CLoaded; + [SaveState.SaveWithName("T1Delayed")] + private int _t1Delayed; + [SaveState.SaveWithName("T2Delayed")] + private int _t2Delayed; + public Via() { _port = new DisconnectedPort(); @@ -183,10 +194,17 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _resetCb2NextClock = false; _handshakeCa2NextClock = false; _handshakeCb2NextClock = false; + _interruptNextClock = 0; + _t1CLoaded = false; + _t2CLoaded = false; } public void ExecutePhase() { + // Process delayed interrupts + _ifr |= _interruptNextClock; + _interruptNextClock = 0; + // Process 'pulse' and 'handshake' outputs on CA2 and CB2 if (_resetCa2NextClock) @@ -215,48 +233,70 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // Count timers - _t1C--; - if (_t1C < 0) + if (_t1Delayed > 0) { - switch (_acrT1Control) + _t1Delayed--; + } + else + { + _t1C--; + if (_t1C < 0) { - case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS: - _t1C = _t1L; - break; - case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7: - _t1C = _t1L; - _prb ^= 0x80; - break; - default: - _t1C = 0xFFFF; - break; + if (_t1CLoaded) + { + _interruptNextClock |= 0x40; + _t1CLoaded = false; + } + switch (_acrT1Control) + { + case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS: + _t1C = _t1L; + _t1CLoaded = true; + break; + case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7: + _t1C = _t1L; + _prb ^= 0x80; + _t1CLoaded = true; + break; + } + _t1C &= 0xFFFF; } - _ifr |= 0x40; } - switch (_acrT2Control) + if (_t2Delayed > 0) { - case ACR_T2_CONTROL_TIMED: - _t2C--; - if (_t2C < 0) - { - _ifr |= 0x20; - _t2C = 0xFFFF; - } - break; - case ACR_T2_CONTROL_COUNT_ON_PB6: - _pb6L = _pb6; - _pb6 = (_port.ReadExternalPrb() & 0x40) != 0; - if (!_pb6 && _pb6L) - { + _t2Delayed--; + } + else + { + switch (_acrT2Control) + { + case ACR_T2_CONTROL_TIMED: _t2C--; if (_t2C < 0) { - _ifr |= 0x20; - _t2C = 0xFFFF; + if (_t2CLoaded) + { + _interruptNextClock |= 0x20; + _t2CLoaded = false; + } + _t2C = _t2L; } - } - break; + break; + case ACR_T2_CONTROL_COUNT_ON_PB6: + _pb6L = _pb6; + _pb6 = (_port.ReadExternalPrb() & 0x40) != 0; + if (!_pb6 && _pb6L) + { + _t2C--; + if (_t2C < 0) + { + _ifr |= 0x20; + _t2C = 0xFFFF; + } + } + break; + } } // Process CA2 diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs index c69b185509..82bba29bf1 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -71,8 +71,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial }; _ram = new int[0x800]; - Via0 = new Via(ViaReadClock, ViaReadData, ViaReadAtn, 8); - Via1 = new Via(ReadVia1PrA, ReadVia1PrB); + Via0 = Chip6522.Create(ViaReadClock, ViaReadData, ViaReadAtn, 8); + Via1 = Chip6522.Create(ReadVia1PrA, ReadVia1PrB); _cpuClockNum = clockNum; _driveCpuClockNum = clockDen*1000000; // 1mhz diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPortDevice.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPortDevice.cs index bee7d35688..1894373725 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPortDevice.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPortDevice.cs @@ -9,11 +9,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial public abstract class SerialPortDevice { [SaveState.DoNotSave] - public Func ReadMasterAtn; + public Func ReadMasterAtn = () => true; [SaveState.DoNotSave] - public Func ReadMasterClk; + public Func ReadMasterClk = () => true; [SaveState.DoNotSave] - public Func ReadMasterData; + public Func ReadMasterData = () => true; public virtual void ExecutePhase() { From 87200593b6e682d7434e4a5da983749c3e1ff017 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Thu, 3 Mar 2016 21:14:19 -0600 Subject: [PATCH 05/15] C64: Internal support for multiple files and Epyx Fastload support. --- .../BizHawk.Emulation.Cores.csproj | 2 +- .../Computers/Commodore64/C64.Motherboard.cs | 2 + .../Computers/Commodore64/C64.cs | 135 +++++++++--------- .../Commodore64/Cartridge/CartridgeDevice.cs | 11 +- .../Commodore64/Cartridge/CartridgePort.cs | 6 + .../Commodore64/Cartridge/Mapper000A.cs | 83 +++++++++++ .../Computers/Commodore64/InputFileInfo.cs | 8 -- .../Computers/Commodore64/Media/Disk.cs | 1 - .../Serial/Drive1541.FluxTransitions.cs | 20 ++- .../Computers/Commodore64/Serial/Drive1541.cs | 2 +- 10 files changed, 181 insertions(+), 89 deletions(-) create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs delete mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/InputFileInfo.cs diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index 6c21f9741d..50f0b3a832 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -209,6 +209,7 @@ + @@ -218,7 +219,6 @@ - diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index d288c463de..217b5aadff 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -159,6 +159,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 _vicBank = (0x3 - ((Cia1.PrA | ~Cia1.DdrA) & 0x3)) << 14; Vic.ExecutePhase(); + CartPort.ExecutePhase(); Cassette.ExecutePhase(); Serial.ExecutePhase(); Sid.ExecutePhase(); @@ -190,6 +191,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 Cassette.HardReset(); Serial.HardReset(); Cpu.HardReset(); + CartPort.HardReset(); } public void Init() diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 33778c5e99..b14a3787e0 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -22,7 +22,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public sealed partial class C64 : IEmulator, IRegionable { // framework - public C64(CoreComm comm, GameInfo game, byte[] rom, string romextension, object settings, object syncSettings) + public C64(CoreComm comm, GameInfo game, byte[] rom, string extension, object settings, object syncSettings) { PutSyncSettings((C64SyncSettings)syncSettings ?? new C64SyncSettings()); PutSettings((C64Settings)settings ?? new C64Settings()); @@ -30,14 +30,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 ServiceProvider = new BasicServiceProvider(this); InputCallbacks = new InputCallbackSystem(); - _inputFileInfo = new InputFileInfo - { - Data = rom, - Extension = romextension - }; - CoreComm = comm; - Init(SyncSettings.VicType, Settings.BorderType, SyncSettings.SidType, SyncSettings.TapeDriveType, SyncSettings.DiskDriveType); + Roms = new Dictionary { { "rom" + extension.ToUpper(), rom } }; + Init(SyncSettings.VicType, Settings.BorderType, SyncSettings.SidType, SyncSettings.TapeDriveType, SyncSettings.DiskDriveType); _cyclesPerFrame = _board.Vic.CyclesPerFrame; SetupMemoryDomains(_board.DiskDrive != null); _memoryCallbacks = new MemoryCallbackSystem(); @@ -62,7 +57,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 // internal variables private int _frame; [SaveState.DoNotSave] private readonly int _cyclesPerFrame; - [SaveState.DoNotSave] private InputFileInfo _inputFileInfo; private bool _driveLed; // bizhawk I/O @@ -94,6 +88,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 // controller [SaveState.DoNotSave] public ControllerDefinition ControllerDefinition { get { return C64ControllerDefinition; } } [SaveState.DoNotSave] public IController Controller { get { return _board.Controller; } set { _board.Controller = value; } } + [SaveState.DoNotSave] public IDictionary Roms { get; private set; } [SaveState.DoNotSave] private static readonly ControllerDefinition C64ControllerDefinition = new ControllerDefinition @@ -155,7 +150,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 // check to see if cpu PC is at the BASIC warm start vector if (_board.Cpu.Pc != 0 && _board.Cpu.Pc == ((_board.Ram.Peek(0x0303) << 8) | _board.Ram.Peek(0x0302))) { - Prg.Load(_board.Pla, _inputFileInfo.Data); + Prg.Load(_board.Pla, _prgFile); _loadPrg = false; } } @@ -182,6 +177,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private Motherboard _board; private bool _loadPrg; + [SaveState.DoNotSave] private byte[] _prgFile; private byte[] GetFirmware(int length, params string[] names) { @@ -194,29 +190,29 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private void Init(VicType initRegion, BorderType borderType, SidType sidType, TapeDriveType tapeDriveType, DiskDriveType diskDriveType) { // Force certain drive types to be available depending on ROM type - switch (_inputFileInfo.Extension.ToUpper()) + foreach (var rom in Roms) { - case @".D64": - case @".G64": - if (diskDriveType == DiskDriveType.None) - { - diskDriveType = DiskDriveType.Commodore1541; - } - break; - case @".TAP": - if (tapeDriveType == TapeDriveType.None) - { - tapeDriveType = TapeDriveType.Commodore1530; - } - break; - } + switch (Path.GetExtension(rom.Key).ToUpper()) + { + case @".D64": + case @".G64": + if (diskDriveType == DiskDriveType.None) + { + diskDriveType = DiskDriveType.Commodore1541; + } + break; + case @".TAP": + if (tapeDriveType == TapeDriveType.None) + { + tapeDriveType = TapeDriveType.Commodore1530; + } + break; + } + } _board = new Motherboard(this, initRegion, borderType, sidType, tapeDriveType, diskDriveType); InitRoms(diskDriveType); _board.Init(); - InitMedia(); - - // configure video CoreComm.VsyncDen = _board.Vic.CyclesPerFrame; @@ -225,42 +221,48 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private void InitMedia() { - switch (_inputFileInfo.Extension.ToUpper()) - { - case @".D64": - var d64 = D64.Read(_inputFileInfo.Data); - if (d64 != null) - { - _board.DiskDrive.InsertMedia(d64); - } - break; - case @".G64": - var g64 = G64.Read(_inputFileInfo.Data); - if (g64 != null) - { - _board.DiskDrive.InsertMedia(g64); - } - break; - case @".CRT": - var cart = CartridgeDevice.Load(_inputFileInfo.Data); - if (cart != null) - { - _board.CartPort.Connect(cart); - } - break; - case @".TAP": - var tape = Tape.Load(_inputFileInfo.Data); - if (tape != null) - { - _board.TapeDrive.Insert(tape); - } - break; - case @".PRG": - if (_inputFileInfo.Data.Length > 2) - _loadPrg = true; - break; - } - } + foreach (var rom in Roms) + { + switch (Path.GetExtension(rom.Key).ToUpper()) + { + case @".D64": + var d64 = D64.Read(rom.Value); + if (d64 != null) + { + _board.DiskDrive.InsertMedia(d64); + } + break; + case @".G64": + var g64 = G64.Read(rom.Value); + if (g64 != null) + { + _board.DiskDrive.InsertMedia(g64); + } + break; + case @".CRT": + var cart = CartridgeDevice.Load(rom.Value); + if (cart != null) + { + _board.CartPort.Connect(cart); + } + break; + case @".TAP": + var tape = Tape.Load(rom.Value); + if (tape != null) + { + _board.TapeDrive.Insert(tape); + } + break; + case @".PRG": + if (rom.Value.Length > 2) + { + _loadPrg = true; + _prgFile = rom.Value; + } + break; + } + } + } private void InitRoms(DiskDriveType diskDriveType) { @@ -274,7 +276,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 if (diskDriveType == DiskDriveType.Commodore1541) { - var diskRom = GetFirmware(0x4000, "Drive1541II", "Drive1541"); + var diskRom = GetFirmware(0x4000, "Drive1541", "Drive1541II"); _board.DiskDrive.DriveRom.Flash(diskRom); } } @@ -283,7 +285,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public void HardReset() { - _board.HardReset(); + InitMedia(); + _board.HardReset(); } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs index 40dd410c29..13c9d3dda6 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs @@ -74,7 +74,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge case 0x0005: result = new Mapper0005(chipAddress, chipBank, chipData); break; - case 0x000B: + case 0x000A: + result = new Mapper000A(chipAddress, chipBank, chipData); + break; + case 0x000B: result = new Mapper000B(chipAddress, chipData); break; case 0x000F: @@ -127,11 +130,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge [SaveState.DoNotSave] protected bool validCartridge; - public virtual void ExecutePhase1() - { - } - - public virtual void ExecutePhase2() + public virtual void ExecutePhase() { } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs index 8ea04893cb..c980d8ae85 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs @@ -89,6 +89,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge _connected = false; } + public void ExecutePhase() + { + if (_connected) + _cartridgeDevice.ExecutePhase(); + } + public void HardReset() { // note: this will not disconnect any attached media diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs new file mode 100644 index 0000000000..47b904d97f --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs @@ -0,0 +1,83 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge +{ + //792 + + // Epyx Fastload. Uppermost page is always visible at DFxx. + // They use a capacitor that is recharged by accesses to DExx + // to pull down EXROM. + + public abstract partial class CartridgeDevice + { + private class Mapper000A : CartridgeDevice + { + // This constant differs depending on whose research you reference. TODO: Verify. + [SaveState.DoNotSave] + private const int RESET_CAPACITOR_CYCLES = 512; + + [SaveState.SaveWithName("CapacitorCycles")] + private int _capacitorCycles; + [SaveState.DoNotSave] + private readonly int[] _rom; + + public Mapper000A(IList newAddresses, IList newBanks, IList newData) + { + _rom = new int[0x2000]; + Array.Copy(newData.First(), _rom, 0x2000); + pinGame = true; + } + + public override void ExecutePhase() + { + pinExRom = !(_capacitorCycles > 0); + if (!pinExRom) + { + _capacitorCycles--; + } + } + + public override void HardReset() + { + _capacitorCycles = RESET_CAPACITOR_CYCLES; + base.HardReset(); + } + + public override int Peek8000(int addr) + { + return _rom[addr & 0x1FFF]; + } + + public override int PeekDE00(int addr) + { + return 0x00; + } + + public override int PeekDF00(int addr) + { + return _rom[(addr & 0xFF) | 0x1F00]; + } + + public override int Read8000(int addr) + { + _capacitorCycles = RESET_CAPACITOR_CYCLES; + return _rom[addr & 0x1FFF]; + } + + public override int ReadDE00(int addr) + { + _capacitorCycles = RESET_CAPACITOR_CYCLES; + return 0x00; + } + + public override int ReadDF00(int addr) + { + return _rom[(addr & 0xFF) | 0x1F00]; + } + } + } + +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/InputFileInfo.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/InputFileInfo.cs deleted file mode 100644 index 464d647d6e..0000000000 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/InputFileInfo.cs +++ /dev/null @@ -1,8 +0,0 @@ -namespace BizHawk.Emulation.Cores.Computers.Commodore64 -{ - public struct InputFileInfo - { - public byte[] Data; - public string Extension; - } -} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs index a8a43e6f6b..339d7b215e 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs @@ -72,7 +72,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media paddedBytes[i] = 0xAA; } var result = new int[FluxEntriesPerTrack]; - var length = paddedLength; var lengthBits = (paddedLength * 8) - 7; var offsets = new List(); var remainingBits = lengthBits; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs index 6a393af42d..ae2b18361e 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs @@ -48,15 +48,23 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial // rotate disk if (_motorEnabled) { - if (_diskBitsLeft <= 0) + if (_disk == null) { - _diskByteOffset++; - if (_diskByteOffset == Disk.FluxEntriesPerTrack) + _diskBitsLeft = 1; + _diskBits = 0; + } + else + { + if (_diskBitsLeft <= 0) { - _diskByteOffset = 0; + _diskByteOffset++; + if (_diskByteOffset == Disk.FluxEntriesPerTrack) + { + _diskByteOffset = 0; + } + _diskBits = _trackImageData[_diskByteOffset]; + _diskBitsLeft = Disk.FluxBitsPerEntry; } - _diskBits = _trackImageData[_diskByteOffset]; - _diskBitsLeft = Disk.FluxBitsPerEntry; } if ((_diskBits & 1) != 0) { diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs index 82bba29bf1..306352c90d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -218,7 +218,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial public void RemoveMedia() { - _trackImageData = new int[1]; + _disk = null; } public int Peek(int addr) From f4f45f73a84daa2f62dc81bf5c90a3ce853c8249 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Fri, 4 Mar 2016 16:06:25 -0600 Subject: [PATCH 06/15] C64: Better lifecycle management and 1541-II support. --- .../BizHawk.Emulation.Cores.csproj | 1 + .../Computers/Commodore64/C64.ISettable.cs | 2 +- .../Computers/Commodore64/C64.Motherboard.cs | 1 + .../Computers/Commodore64/C64.cs | 36 ++- .../Commodore64/Cassette/TapeDrive.cs | 5 + .../Computers/Commodore64/Media/D64.cs | 226 +++++++++--------- .../Computers/Commodore64/Media/Disk.cs | 2 +- .../Computers/Commodore64/Media/G64.cs | 80 ++++--- .../Serial/Drive1541.FluxTransitions.cs | 17 +- .../Commodore64/Serial/Drive1541.Motor.cs | 12 - .../Commodore64/Serial/Drive1541.Registers.cs | 164 +++++++++++++ .../Computers/Commodore64/Serial/Drive1541.cs | 172 +------------ 12 files changed, 374 insertions(+), 344 deletions(-) create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index 50f0b3a832..728146771b 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -261,6 +261,7 @@ + diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs index 5ea9385b54..b0cf805828 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs @@ -115,7 +115,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public enum DiskDriveType { - None, Commodore1541 + None, Commodore1541, Commodore1541II } } } \ No newline at end of file diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 217b5aadff..071e3f2dff 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -130,6 +130,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 switch (diskDriveType) { case C64.DiskDriveType.Commodore1541: + case C64.DiskDriveType.Commodore1541II: DiskDrive = new Drive1541(ClockNumerator, ClockDenominator); Serial.Connect(DiskDrive); break; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index b14a3787e0..5f98b51a0a 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -117,7 +117,19 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public void Dispose() { - } + if (_board != null) + { + if (_board.TapeDrive != null) + { + _board.TapeDrive.RemoveMedia(); + } + if (_board.DiskDrive != null) + { + _board.DiskDrive.RemoveMedia(); + } + _board = null; + } + } private int _frameCycles; @@ -266,19 +278,19 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private void InitRoms(DiskDriveType diskDriveType) { - var basicRom = GetFirmware(0x2000, "Basic"); - var charRom = GetFirmware(0x1000, "Chargen"); - var kernalRom = GetFirmware(0x2000, "Kernal"); + _board.BasicRom.Flash(GetFirmware(0x2000, "Basic")); + _board.KernalRom.Flash(GetFirmware(0x1000, "Chargen")); + _board.CharRom.Flash(GetFirmware(0x2000, "Kernal")); - _board.BasicRom.Flash(basicRom); - _board.KernalRom.Flash(kernalRom); - _board.CharRom.Flash(charRom); - - if (diskDriveType == DiskDriveType.Commodore1541) + switch (diskDriveType) { - var diskRom = GetFirmware(0x4000, "Drive1541", "Drive1541II"); - _board.DiskDrive.DriveRom.Flash(diskRom); - } + case DiskDriveType.Commodore1541: + _board.DiskDrive.DriveRom.Flash(GetFirmware(0x4000, "Drive1541")); + break; + case DiskDriveType.Commodore1541II: + _board.DiskDrive.DriveRom.Flash(GetFirmware(0x4000, "Drive1541II")); + break; + } } // ------------------------------------ diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cassette/TapeDrive.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cassette/TapeDrive.cs index d91a8975f3..2687e2824b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cassette/TapeDrive.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cassette/TapeDrive.cs @@ -37,5 +37,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cassette { _tape = tape; } + + public void RemoveMedia() + { + _tape = null; + } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs index 5fd8bf9e2d..188f69acf8 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs @@ -76,30 +76,32 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media private static byte[] ConvertSectorToGcr(byte[] source, byte sectorNo, byte trackNo, byte formatA, byte formatB, out int bitsWritten) { - var mem = new MemoryStream(); - var writer = new BinaryWriter(mem); - var headerChecksum = (byte)(sectorNo ^ trackNo ^ formatA ^ formatB); + using (var mem = new MemoryStream()) + { + var writer = new BinaryWriter(mem); + var headerChecksum = (byte)(sectorNo ^ trackNo ^ formatA ^ formatB); - // assemble written data for GCR encoding - var writtenData = new byte[260]; - Array.Copy(source, 0, writtenData, 1, 256); - writtenData[0] = 0x07; - writtenData[0x101] = Checksum(source); - writtenData[0x102] = 0x00; - writtenData[0x103] = 0x00; + // assemble written data for GCR encoding + var writtenData = new byte[260]; + Array.Copy(source, 0, writtenData, 1, 256); + writtenData[0] = 0x07; + writtenData[0x101] = Checksum(source); + writtenData[0x102] = 0x00; + writtenData[0x103] = 0x00; - writer.Write(new byte[] { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }); // sync - writer.Write(EncodeGcr(new byte[] { 0x08, headerChecksum, sectorNo, trackNo, formatA, formatB, 0x0F, 0x0F })); // header - writer.Write(new byte[] { 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55 }); // gap - writer.Write(new byte[] { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }); // sync - writer.Write(EncodeGcr(writtenData)); // data - writer.Write(new byte[] { 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55 }); // gap + writer.Write(new byte[] { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }); // sync + writer.Write(EncodeGcr(new byte[] { 0x08, headerChecksum, sectorNo, trackNo, formatA, formatB, 0x0F, 0x0F })); // header + writer.Write(new byte[] { 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55 }); // gap + writer.Write(new byte[] { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }); // sync + writer.Write(EncodeGcr(writtenData)); // data + writer.Write(new byte[] { 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55, 0x55 }); // gap - bitsWritten = (int)mem.Length * 8; + bitsWritten = (int)mem.Length * 8; - writer.Flush(); - return mem.ToArray(); - } + writer.Flush(); + return mem.ToArray(); + } + } private static byte[] EncodeGcr(byte[] source) { @@ -107,113 +109,101 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media var gcr = new int[8]; var data = new byte[4]; var count = source.Length; - var mem = new MemoryStream(); - var writer = new BinaryWriter(mem); + using (var mem = new MemoryStream()) + { + var writer = new BinaryWriter(mem); - for (var i = 0; i < count; i += 4) - { - Array.Copy(source, i, data, 0, 4); - gcr[0] = gcrEncodeTable[data[0] >> 4]; - gcr[1] = gcrEncodeTable[data[0] & 0xF]; - gcr[2] = gcrEncodeTable[data[1] >> 4]; - gcr[3] = gcrEncodeTable[data[1] & 0xF]; - gcr[4] = gcrEncodeTable[data[2] >> 4]; - gcr[5] = gcrEncodeTable[data[2] & 0xF]; - gcr[6] = gcrEncodeTable[data[3] >> 4]; - gcr[7] = gcrEncodeTable[data[3] & 0xF]; + for (var i = 0; i < count; i += 4) + { + Array.Copy(source, i, data, 0, 4); + gcr[0] = gcrEncodeTable[data[0] >> 4]; + gcr[1] = gcrEncodeTable[data[0] & 0xF]; + gcr[2] = gcrEncodeTable[data[1] >> 4]; + gcr[3] = gcrEncodeTable[data[1] & 0xF]; + gcr[4] = gcrEncodeTable[data[2] >> 4]; + gcr[5] = gcrEncodeTable[data[2] & 0xF]; + gcr[6] = gcrEncodeTable[data[3] >> 4]; + gcr[7] = gcrEncodeTable[data[3] & 0xF]; - // -------- -------- -------- -------- -------- - // 00000111 11222223 33334444 45555566 66677777 + // -------- -------- -------- -------- -------- + // 00000111 11222223 33334444 45555566 66677777 - var outputValue = (gcr[0] << 3) | (gcr[1] >> 2); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[1] << 6) | (gcr[2] << 1) | (gcr[3] >> 4); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[3] << 4) | (gcr[4] >> 1); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[4] << 7) | (gcr[5] << 2) | (gcr[6] >> 3); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[6] << 5) | (gcr[7]); - writer.Write((byte)(outputValue & 0xFF)); - - /* - // -------- -------- -------- -------- -------- - // 11100000 32222211 44443333 66555554 77777666 - - var outputValue = (gcr[0]) | (gcr[1] << 5); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[1] >> 3) | (gcr[2] << 2) | (gcr[3] << 7); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[3] >> 1) | (gcr[4] << 4); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[4] >> 4) | (gcr[5] << 1) | (gcr[6] << 6); - writer.Write((byte)(outputValue & 0xFF)); - outputValue = (gcr[6] >> 2) | (gcr[7] << 3); - writer.Write((byte)(outputValue & 0xFF)); - */ + var outputValue = (gcr[0] << 3) | (gcr[1] >> 2); + writer.Write((byte)(outputValue & 0xFF)); + outputValue = (gcr[1] << 6) | (gcr[2] << 1) | (gcr[3] >> 4); + writer.Write((byte)(outputValue & 0xFF)); + outputValue = (gcr[3] << 4) | (gcr[4] >> 1); + writer.Write((byte)(outputValue & 0xFF)); + outputValue = (gcr[4] << 7) | (gcr[5] << 2) | (gcr[6] >> 3); + writer.Write((byte)(outputValue & 0xFF)); + outputValue = (gcr[6] << 5) | (gcr[7]); + writer.Write((byte)(outputValue & 0xFF)); + } + writer.Flush(); + return mem.ToArray(); } - writer.Flush(); - return mem.ToArray(); - } + } public static Disk Read(byte[] source) { - var mem = new MemoryStream(source); - var reader = new BinaryReader(mem); - var trackDatas = new List(); - var trackLengths = new List(); - var trackNumbers = new List(); - var trackDensities = new List(); - int trackCount; + using (var mem = new MemoryStream(source)) + { + var reader = new BinaryReader(mem); + var trackDatas = new List(); + var trackLengths = new List(); + var trackNumbers = new List(); + var trackDensities = new List(); + int trackCount; - switch (source.Length) - { - case 174848: // 35 tracks no errors - trackCount = 35; - break; - case 175531: // 35 tracks with errors - trackCount = 35; - break; - case 196608: // 40 tracks no errors - trackCount = 40; - break; - case 197376: // 40 tracks with errors - trackCount = 40; - break; - default: - throw new Exception("Not able to identify capacity of the D64 file."); - } - - for (var i = 0; i < trackCount; i++) - { - var sectors = sectorsPerTrack[i]; - var trackLengthBits = 0; - using (var trackMem = new MemoryStream()) - { - for (var j = 0; j < sectors; j++) - { - int bitsWritten; - var sectorData = reader.ReadBytes(256); - var diskData = ConvertSectorToGcr(sectorData, (byte)j, (byte)(i + 1), 0xA0, 0xA0, out bitsWritten); - trackMem.Write(diskData, 0, diskData.Length); - trackLengthBits += bitsWritten; - } - var density = densityTable[i]; - - // we pad the tracks with extra gap bytes to meet MNIB standards - while (trackMem.Length < standardTrackLengthBytes[density]) - { - trackMem.WriteByte(0x55); - } - - trackDatas.Add(trackMem.ToArray()); - trackLengths.Add(trackLengthBits); - trackNumbers.Add(i * 2); - trackDensities.Add(densityTable[i]); + switch (source.Length) + { + case 174848: // 35 tracks no errors + trackCount = 35; + break; + case 175531: // 35 tracks with errors + trackCount = 35; + break; + case 196608: // 40 tracks no errors + trackCount = 40; + break; + case 197376: // 40 tracks with errors + trackCount = 40; + break; + default: + throw new Exception("Not able to identify capacity of the D64 file."); } - } - return new Disk(trackDatas, trackNumbers, trackDensities, trackLengths, 84); - } + for (var i = 0; i < trackCount; i++) + { + var sectors = sectorsPerTrack[i]; + var trackLengthBits = 0; + using (var trackMem = new MemoryStream()) + { + for (var j = 0; j < sectors; j++) + { + int bitsWritten; + var sectorData = reader.ReadBytes(256); + var diskData = ConvertSectorToGcr(sectorData, (byte)j, (byte)(i + 1), 0xA0, 0xA0, out bitsWritten); + trackMem.Write(diskData, 0, diskData.Length); + trackLengthBits += bitsWritten; + } + var density = densityTable[i]; + + // we pad the tracks with extra gap bytes to meet MNIB standards + while (trackMem.Length < standardTrackLengthBytes[density]) + { + trackMem.WriteByte(0x55); + } + + trackDatas.Add(trackMem.ToArray()); + trackLengths.Add(trackLengthBits); + trackNumbers.Add(i * 2); + trackDensities.Add(densityTable[i]); + } + } + + return new Disk(trackDatas, trackNumbers, trackDensities, trackLengths, 84); + } + } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs index 339d7b215e..00175d06bb 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs @@ -23,7 +23,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media FillMissingTracks(); _originalMedia = SerializeTracks(_tracks); Valid = true; - } + } /// /// Create an expanded representation of a magnetic disk. diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs index 1d22f557ef..2ddac42728 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs @@ -9,54 +9,56 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { public static Disk Read(byte[] source) { - var mem = new MemoryStream(source); - var reader = new BinaryReader(mem); - var id = new string(reader.ReadChars(8)); - var trackDatas = new List(); - var trackLengths = new List(); - var trackNumbers = new List(); - var trackDensities = new List(); + using (var mem = new MemoryStream(source)) + { + var reader = new BinaryReader(mem); + var id = new string(reader.ReadChars(8)); + var trackDatas = new List(); + var trackLengths = new List(); + var trackNumbers = new List(); + var trackDensities = new List(); - if (id == @"GCR-1541") - { + if (id == @"GCR-1541") + { - reader.ReadByte(); //version - int trackCount = reader.ReadByte(); - reader.ReadInt16(); //max track size in bytes + reader.ReadByte(); //version + int trackCount = reader.ReadByte(); + reader.ReadInt16(); //max track size in bytes - var trackOffsetTable = new int[trackCount]; - var trackSpeedTable = new int[trackCount]; + var trackOffsetTable = new int[trackCount]; + var trackSpeedTable = new int[trackCount]; - for (var i = 0; i < trackCount; i++) - trackOffsetTable[i] = reader.ReadInt32(); + for (var i = 0; i < trackCount; i++) + trackOffsetTable[i] = reader.ReadInt32(); - for (var i = 0; i < trackCount; i++) - trackSpeedTable[i] = reader.ReadInt32(); + for (var i = 0; i < trackCount; i++) + trackSpeedTable[i] = reader.ReadInt32(); - for (var i = 0; i < trackCount; i++) - { - if (trackOffsetTable[i] > 0) - { - mem.Position = trackOffsetTable[i]; - int trackLength = reader.ReadInt16(); - var trackData = reader.ReadBytes(trackLength); + for (var i = 0; i < trackCount; i++) + { + if (trackOffsetTable[i] > 0) + { + mem.Position = trackOffsetTable[i]; + int trackLength = reader.ReadInt16(); + var trackData = reader.ReadBytes(trackLength); - trackDatas.Add(trackData); - trackLengths.Add(trackLength * 8); - trackDensities.Add(trackSpeedTable[i]); - trackNumbers.Add(i); - } - } + trackDatas.Add(trackData); + trackLengths.Add(trackLength * 8); + trackDensities.Add(trackSpeedTable[i]); + trackNumbers.Add(i); + } + } - if (trackSpeedTable.Any(ts => ts > 3 || ts < 0)) - { - throw new Exception("Byte-level speeds are not yet supported in the G64 loader."); - } + if (trackSpeedTable.Any(ts => ts > 3 || ts < 0)) + { + throw new Exception("Byte-level speeds are not yet supported in the G64 loader."); + } - return new Disk(trackDatas, trackNumbers, trackDensities, trackLengths, 84); - } + return new Disk(trackDatas, trackNumbers, trackDensities, trackLengths, 84); + } - return new Disk(84); - } + return new Disk(84); + } + } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs index ae2b18361e..903472aa28 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs @@ -32,6 +32,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial private int _countsBeforeRandomTransition; [SaveState.SaveWithName("CurrentRNG")] private int _rngCurrent; + [SaveState.SaveWithName("Clocks")] + private int _clocks; + [SaveState.SaveWithName("CpuClocks")] + private int _cpuClocks; // Lehmer RNG private void AdvanceRng() @@ -43,8 +47,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial private void ExecuteFlux() { - for (_diskCycle = 0; _diskCycle < 16; _diskCycle++) + // This actually executes the main 16mhz clock + while (_clocks > 0) { + _clocks--; + // rotate disk if (_motorEnabled) { @@ -144,7 +151,15 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial _diskSupplementaryCounter = 0; } + _cpuClocks--; + if (_cpuClocks <= 0) + { + ExecuteSystem(); + _cpuClocks = 16; + } + _diskDensityCounter++; + _diskCycle = (_diskCycle + 1) & 0xF; } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Motor.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Motor.cs index 65dfca8b0f..b1b4087334 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Motor.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Motor.cs @@ -11,18 +11,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial [SaveState.DoNotSave] private int _tempStep; [SaveState.DoNotSave] private int _tempPrB1; - private int _overflowFlagDelaySr; - - private int ReadVia1PrA() - { - return _bitHistory & 0xFF; - } - - private int ReadVia1PrB() - { - return (_motorStep & 0x03) | (_motorEnabled ? 0x04 : 0x00) | (_sync ? 0x00 : 0x80); - } - private void ExecuteMotor() { _tempPrB1 = Via1.EffectivePrB; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs new file mode 100644 index 0000000000..9f0fb5a890 --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs @@ -0,0 +1,164 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial +{ + public sealed partial class Drive1541 + { + [SaveState.SaveWithName("OverflowFlagDelayShiftRegister")] + private int _overflowFlagDelaySr; + + private byte CpuPeek(ushort addr) + { + return unchecked((byte)Peek(addr)); + } + + private byte CpuRead(ushort addr) + { + return unchecked((byte)Read(addr)); + } + + private void CpuWrite(ushort addr, byte val) + { + Write(addr, val); + } + + private bool ViaReadClock() + { + var inputClock = ReadMasterClk(); + var outputClock = ReadDeviceClk(); + return !(inputClock && outputClock); + } + + private bool ViaReadData() + { + var inputData = ReadMasterData(); + var outputData = ReadDeviceData(); + return !(inputData && outputData); + } + + private bool ViaReadAtn() + { + var inputAtn = ReadMasterAtn(); + return !inputAtn; + } + + private int ReadVia1PrA() + { + return _bitHistory & 0xFF; + } + + private int ReadVia1PrB() + { + return (_motorStep & 0x03) | (_motorEnabled ? 0x04 : 0x00) | (_sync ? 0x00 : 0x80); + } + + public int Peek(int addr) + { + switch (addr & 0xFC00) + { + case 0x1800: + return Via0.Peek(addr); + case 0x1C00: + return Via1.Peek(addr); + } + if ((addr & 0x8000) != 0) + return DriveRom.Peek(addr & 0x3FFF); + if ((addr & 0x1F00) < 0x800) + return _ram[addr & 0x7FF]; + return (addr >> 8) & 0xFF; + } + + public int PeekVia0(int addr) + { + return Via0.Peek(addr); + } + + public int PeekVia1(int addr) + { + return Via1.Peek(addr); + } + + public void Poke(int addr, int val) + { + switch (addr & 0xFC00) + { + case 0x1800: + Via0.Poke(addr, val); + break; + case 0x1C00: + Via1.Poke(addr, val); + break; + default: + if ((addr & 0x8000) == 0 && (addr & 0x1F00) < 0x800) + _ram[addr & 0x7FF] = val & 0xFF; + break; + } + } + + public void PokeVia0(int addr, int val) + { + Via0.Poke(addr, val); + } + + public void PokeVia1(int addr, int val) + { + Via1.Poke(addr, val); + } + + public int Read(int addr) + { + switch (addr & 0xFC00) + { + case 0x1800: + return Via0.Read(addr); + case 0x1C00: + return Via1.Read(addr); + } + if ((addr & 0x8000) != 0) + return DriveRom.Read(addr & 0x3FFF); + if ((addr & 0x1F00) < 0x800) + return _ram[addr & 0x7FF]; + return (addr >> 8) & 0xFF; + } + + public void Write(int addr, int val) + { + switch (addr & 0xFC00) + { + case 0x1800: + Via0.Write(addr, val); + break; + case 0x1C00: + Via1.Write(addr, val); + break; + default: + if ((addr & 0x8000) == 0 && (addr & 0x1F00) < 0x800) + _ram[addr & 0x7FF] = val & 0xFF; + break; + } + } + + public override bool ReadDeviceClk() + { + var viaOutputClock = (Via0.DdrB & 0x08) != 0 && (Via0.PrB & 0x08) != 0; + return !viaOutputClock; + } + + public override bool ReadDeviceData() + { + var viaOutputData = (Via0.DdrB & 0x02) != 0 && (Via0.PrB & 0x02) != 0; + var viaInputAtn = ViaReadAtn(); + var viaOutputAtna = (Via0.DdrB & 0x10) != 0 && (Via0.PrB & 0x10) != 0; + + return !(viaOutputAtna ^ viaInputAtn) && !viaOutputData; + } + + public override bool ReadDeviceLight() + { + return _driveLightOffTime > 0; + } + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs index 306352c90d..2ce00085d4 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -25,6 +25,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial private bool _byteReady; [SaveState.SaveWithName("DriveCpuClockNumerator")] private readonly int _driveCpuClockNum; + [SaveState.SaveWithName("DriveCpuClockDenominator")] + private readonly int _driveCpuClockDen; [SaveState.SaveWithName("TrackNumber")] private int _trackNumber; [SaveState.SaveWithName("MotorEnabled")] @@ -75,74 +77,30 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial Via1 = Chip6522.Create(ReadVia1PrA, ReadVia1PrB); _cpuClockNum = clockNum; - _driveCpuClockNum = clockDen*1000000; // 1mhz - } - - private byte CpuPeek(ushort addr) - { - return unchecked((byte)Peek(addr)); - } - - private byte CpuRead(ushort addr) - { - return unchecked((byte) Read(addr)); - } - - private void CpuWrite(ushort addr, byte val) - { - Write(addr, val); - } - - private bool ViaReadClock() - { - var inputClock = ReadMasterClk(); - var outputClock = ReadDeviceClk(); - return !(inputClock && outputClock); - } - - private bool ViaReadData() - { - var inputData = ReadMasterData(); - var outputData = ReadDeviceData(); - return !(inputData && outputData); - } - - private bool ViaReadAtn() - { - var inputAtn = ReadMasterAtn(); - return !inputAtn; + _driveCpuClockNum = clockDen*16000000; // 16mhz } public override void ExecutePhase() { - if (_cpuClockNum > _driveCpuClockNum) + _ratioDifference += _driveCpuClockNum; + while (_ratioDifference > _cpuClockNum) { - _ratioDifference += _cpuClockNum - _driveCpuClockNum; - if (_ratioDifference > _cpuClockNum) - { - _ratioDifference -= _cpuClockNum; - return; - } - } - else if (_cpuClockNum <= _driveCpuClockNum) - { - _ratioDifference += _driveCpuClockNum - _cpuClockNum; - while (_ratioDifference > _driveCpuClockNum) - { - _ratioDifference -= _driveCpuClockNum; - ExecutePhaseInternal(); - } + _ratioDifference -= _cpuClockNum; + _clocks++; } ExecutePhaseInternal(); } private void ExecutePhaseInternal() { - Via0.Ca1 = ViaReadAtn(); - // clock output from 325572-01 drives CPU clock (phi0) ExecuteMotor(); ExecuteFlux(); + } + + private void ExecuteSystem() + { + Via0.Ca1 = ViaReadAtn(); Via0.ExecutePhase(); Via1.ExecutePhase(); @@ -220,111 +178,5 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial { _disk = null; } - - public int Peek(int addr) - { - switch (addr & 0xFC00) - { - case 0x1800: - return Via0.Peek(addr); - case 0x1C00: - return Via1.Peek(addr); - } - if ((addr & 0x8000) != 0) - return DriveRom.Peek(addr & 0x3FFF); - if ((addr & 0x1F00) < 0x800) - return _ram[addr & 0x7FF]; - return (addr >> 8) & 0xFF; - } - - public int PeekVia0(int addr) - { - return Via0.Peek(addr); - } - - public int PeekVia1(int addr) - { - return Via1.Peek(addr); - } - - public void Poke(int addr, int val) - { - switch (addr & 0xFC00) - { - case 0x1800: - Via0.Poke(addr, val); - break; - case 0x1C00: - Via1.Poke(addr, val); - break; - default: - if ((addr & 0x8000) == 0 && (addr & 0x1F00) < 0x800) - _ram[addr & 0x7FF] = val & 0xFF; - break; - } - } - - public void PokeVia0(int addr, int val) - { - Via0.Poke(addr, val); - } - - public void PokeVia1(int addr, int val) - { - Via1.Poke(addr, val); - } - - public int Read(int addr) - { - switch (addr & 0xFC00) - { - case 0x1800: - return Via0.Read(addr); - case 0x1C00: - return Via1.Read(addr); - } - if ((addr & 0x8000) != 0) - return DriveRom.Read(addr & 0x3FFF); - if ((addr & 0x1F00) < 0x800) - return _ram[addr & 0x7FF]; - return (addr >> 8) & 0xFF; - } - - public void Write(int addr, int val) - { - switch (addr & 0xFC00) - { - case 0x1800: - Via0.Write(addr, val); - break; - case 0x1C00: - Via1.Write(addr, val); - break; - default: - if ((addr & 0x8000) == 0 && (addr & 0x1F00) < 0x800) - _ram[addr & 0x7FF] = val & 0xFF; - break; - } - } - - public override bool ReadDeviceClk() - { - var viaOutputClock = (Via0.DdrB & 0x08) != 0 && (Via0.PrB & 0x08) != 0; - return !viaOutputClock; - } - - public override bool ReadDeviceData() - { - var viaOutputData = (Via0.DdrB & 0x02) != 0 && (Via0.PrB & 0x02) != 0; - var viaInputAtn = ViaReadAtn(); - var viaOutputAtna = (Via0.DdrB & 0x10) != 0 && (Via0.PrB & 0x10) != 0; - - return !(viaOutputAtna ^ viaInputAtn) && !viaOutputData; - } - - public override bool ReadDeviceLight() - { - return _driveLightOffTime > 0; - } } } From b0e4e18ebc2f5fd598fa9bb431a7798e53ea777b Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Fri, 4 Mar 2016 16:15:01 -0600 Subject: [PATCH 07/15] C64: Manage memory for Cartridge, Write protect on media. --- .../Commodore64/Cartridge/CartridgeDevice.cs | 164 +++++++++--------- .../Computers/Commodore64/Media/Disk.cs | 5 +- .../Computers/Commodore64/Serial/Drive1541.cs | 2 + 3 files changed, 89 insertions(+), 82 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs index 13c9d3dda6..2c7ef4fddc 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs @@ -10,98 +10,100 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { public static CartridgeDevice Load(byte[] crtFile) { - var mem = new MemoryStream(crtFile); - var reader = new BinaryReader(mem); - - if (new string(reader.ReadChars(16)) != "C64 CARTRIDGE ") + using (var mem = new MemoryStream(crtFile)) { - return null; - } + var reader = new BinaryReader(mem); - var chipAddress = new List(); - var chipBank = new List(); - var chipData = new List(); - var chipType = new List(); + if (new string(reader.ReadChars(16)) != "C64 CARTRIDGE ") + { + return null; + } - var headerLength = ReadCRTInt(reader); - var version = ReadCRTShort(reader); - var mapper = ReadCRTShort(reader); - var exrom = reader.ReadByte() != 0; - var game = reader.ReadByte() != 0; + var chipAddress = new List(); + var chipBank = new List(); + var chipData = new List(); + var chipType = new List(); - // reserved - reader.ReadBytes(6); + var headerLength = ReadCRTInt(reader); + var version = ReadCRTShort(reader); + var mapper = ReadCRTShort(reader); + var exrom = reader.ReadByte() != 0; + var game = reader.ReadByte() != 0; - // cartridge name - reader.ReadBytes(0x20); + // reserved + reader.ReadBytes(6); - // skip extra header bytes - if (headerLength > 0x40) - { - reader.ReadBytes(headerLength - 0x40); - } + // cartridge name + reader.ReadBytes(0x20); - // read chips - while (reader.PeekChar() >= 0) - { - if (new string(reader.ReadChars(4)) != "CHIP") - { - break; - } + // skip extra header bytes + if (headerLength > 0x40) + { + reader.ReadBytes(headerLength - 0x40); + } - var chipLength = ReadCRTInt(reader); - chipType.Add(ReadCRTShort(reader)); - chipBank.Add(ReadCRTShort(reader)); - chipAddress.Add(ReadCRTShort(reader)); - var chipDataLength = ReadCRTShort(reader); - chipData.Add(reader.ReadBytes(chipDataLength).Select(x => (int)x).ToArray()); - chipLength -= chipDataLength + 0x10; - if (chipLength > 0) - reader.ReadBytes(chipLength); - } + // read chips + while (reader.PeekChar() >= 0) + { + if (new string(reader.ReadChars(4)) != "CHIP") + { + break; + } - if (chipData.Count <= 0) - { - return null; - } + var chipLength = ReadCRTInt(reader); + chipType.Add(ReadCRTShort(reader)); + chipBank.Add(ReadCRTShort(reader)); + chipAddress.Add(ReadCRTShort(reader)); + var chipDataLength = ReadCRTShort(reader); + chipData.Add(reader.ReadBytes(chipDataLength).Select(x => (int)x).ToArray()); + chipLength -= chipDataLength + 0x10; + if (chipLength > 0) + reader.ReadBytes(chipLength); + } - CartridgeDevice result; - switch (mapper) - { - case 0x0000: - result = new Mapper0000(chipAddress, chipData, game, exrom); - break; - case 0x0005: - result = new Mapper0005(chipAddress, chipBank, chipData); - break; - case 0x000A: - result = new Mapper000A(chipAddress, chipBank, chipData); - break; - case 0x000B: - result = new Mapper000B(chipAddress, chipData); - break; - case 0x000F: - result = new Mapper000F(chipAddress, chipBank, chipData); - break; - case 0x0011: - result = new Mapper0011(chipAddress, chipBank, chipData); - break; - case 0x0012: - result = new Mapper0012(chipAddress, chipBank, chipData); - break; - case 0x0013: - result = new Mapper0013(chipAddress, chipBank, chipData); - break; - case 0x0020: - result = new Mapper0020(chipAddress, chipBank, chipData); - break; - default: - throw new Exception("This cartridge file uses an unrecognized mapper: " + mapper); - } - result.HardReset(); + if (chipData.Count <= 0) + { + return null; + } - return result; - } + CartridgeDevice result; + switch (mapper) + { + case 0x0000: + result = new Mapper0000(chipAddress, chipData, game, exrom); + break; + case 0x0005: + result = new Mapper0005(chipAddress, chipBank, chipData); + break; + case 0x000A: + result = new Mapper000A(chipAddress, chipBank, chipData); + break; + case 0x000B: + result = new Mapper000B(chipAddress, chipData); + break; + case 0x000F: + result = new Mapper000F(chipAddress, chipBank, chipData); + break; + case 0x0011: + result = new Mapper0011(chipAddress, chipBank, chipData); + break; + case 0x0012: + result = new Mapper0012(chipAddress, chipBank, chipData); + break; + case 0x0013: + result = new Mapper0013(chipAddress, chipBank, chipData); + break; + case 0x0020: + result = new Mapper0020(chipAddress, chipBank, chipData); + break; + default: + throw new Exception("This cartridge file uses an unrecognized mapper: " + mapper); + } + result.HardReset(); + + return result; + } + } private static int ReadCRTShort(BinaryReader reader) { diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs index 00175d06bb..189066f5ee 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs @@ -13,12 +13,14 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media [SaveState.DoNotSave] private int[][] _tracks; [SaveState.DoNotSave] private readonly int[] _originalMedia; [SaveState.DoNotSave] public bool Valid; + [SaveState.SaveWithName("DiskIsWriteProtected")] public bool WriteProtected; /// /// Create a blank, unformatted disk. /// public Disk(int trackCapacity) - { + { + WriteProtected = false; _tracks = new int[trackCapacity][]; FillMissingTracks(); _originalMedia = SerializeTracks(_tracks); @@ -35,6 +37,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media /// Total number of tracks on the media. public Disk(IList trackData, IList trackNumbers, IList trackDensities, IList trackLengths, int trackCapacity) { + WriteProtected = true; _tracks = new int[trackCapacity][]; for (var i = 0; i < trackData.Count; i++) { diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs index 2ce00085d4..c31635e03c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -177,6 +177,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial public void RemoveMedia() { _disk = null; + _trackImageData = null; + _diskBits = 0; } } } From 57676608d0ed8b8b1a1292c1ae5f084f8296a9d2 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Fri, 4 Mar 2016 17:29:47 -0600 Subject: [PATCH 08/15] C64: Fix ROM load order. --- BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 5f98b51a0a..5b80726e40 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -279,8 +279,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private void InitRoms(DiskDriveType diskDriveType) { _board.BasicRom.Flash(GetFirmware(0x2000, "Basic")); - _board.KernalRom.Flash(GetFirmware(0x1000, "Chargen")); - _board.CharRom.Flash(GetFirmware(0x2000, "Kernal")); + _board.KernalRom.Flash(GetFirmware(0x2000, "Kernal")); + _board.CharRom.Flash(GetFirmware(0x1000, "Chargen")); switch (diskDriveType) { From 8fd552274ea421b36137aef11d098ce5425ce6c7 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Sat, 5 Mar 2016 15:23:22 -0600 Subject: [PATCH 09/15] C64: Discover file formats heuristically instead of by filename. --- BizHawk.Client.Common/RomLoader.cs | 2 +- .../BizHawk.Emulation.Cores.csproj | 2 + .../Computers/Commodore64/C64.ISettable.cs | 2 +- .../Computers/Commodore64/C64.cs | 51 +++++++++++-------- .../Computers/Commodore64/C64Format.cs | 24 +++++++++ .../Computers/Commodore64/C64FormatFinder.cs | 47 +++++++++++++++++ .../Computers/Commodore64/Media/D64.cs | 2 +- .../Computers/Commodore64/Media/Disk.cs | 2 +- .../Computers/Commodore64/Media/G64.cs | 2 +- .../Computers/Commodore64/Serial/Drive1541.cs | 2 +- 10 files changed, 108 insertions(+), 28 deletions(-) create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/C64Format.cs create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs diff --git a/BizHawk.Client.Common/RomLoader.cs b/BizHawk.Client.Common/RomLoader.cs index 677f153f25..b8e00b5874 100644 --- a/BizHawk.Client.Common/RomLoader.cs +++ b/BizHawk.Client.Common/RomLoader.cs @@ -776,7 +776,7 @@ namespace BizHawk.Client.Common nextEmulator = new Atari7800(nextComm, game, rom.RomData, gamedbpath); break; case "C64": - var c64 = new C64(nextComm, game, rom.RomData, rom.Extension, GetCoreSettings(), GetCoreSyncSettings()); + var c64 = new C64(nextComm, game, rom.RomData, GetCoreSettings(), GetCoreSyncSettings()); nextEmulator = c64; break; case "GBA": diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index 728146771b..b8961e5eb6 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -205,6 +205,7 @@ + @@ -218,6 +219,7 @@ + diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs index b0cf805828..a34ff6f602 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs @@ -115,7 +115,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public enum DiskDriveType { - None, Commodore1541, Commodore1541II + None, Commodore1541, Commodore1541II, Commodore1571 } } } \ No newline at end of file diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 5b80726e40..632e72073d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -22,7 +22,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public sealed partial class C64 : IEmulator, IRegionable { // framework - public C64(CoreComm comm, GameInfo game, byte[] rom, string extension, object settings, object syncSettings) + public C64(CoreComm comm, GameInfo game, byte[] rom, object settings, object syncSettings) { PutSyncSettings((C64SyncSettings)syncSettings ?? new C64SyncSettings()); PutSettings((C64Settings)settings ?? new C64Settings()); @@ -31,7 +31,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 InputCallbacks = new InputCallbackSystem(); CoreComm = comm; - Roms = new Dictionary { { "rom" + extension.ToUpper(), rom } }; + Roms = new List { rom }; Init(SyncSettings.VicType, Settings.BorderType, SyncSettings.SidType, SyncSettings.TapeDriveType, SyncSettings.DiskDriveType); _cyclesPerFrame = _board.Vic.CyclesPerFrame; SetupMemoryDomains(_board.DiskDrive != null); @@ -88,7 +88,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 // controller [SaveState.DoNotSave] public ControllerDefinition ControllerDefinition { get { return C64ControllerDefinition; } } [SaveState.DoNotSave] public IController Controller { get { return _board.Controller; } set { _board.Controller = value; } } - [SaveState.DoNotSave] public IDictionary Roms { get; private set; } + [SaveState.DoNotSave] public IEnumerable Roms { get; private set; } [SaveState.DoNotSave] private static readonly ControllerDefinition C64ControllerDefinition = new ControllerDefinition @@ -204,16 +204,20 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 // Force certain drive types to be available depending on ROM type foreach (var rom in Roms) { - switch (Path.GetExtension(rom.Key).ToUpper()) + switch (C64FormatFinder.GetFormat(rom)) { - case @".D64": - case @".G64": + case C64Format.D64: + case C64Format.G64: + case C64Format.X64: if (diskDriveType == DiskDriveType.None) - { diskDriveType = DiskDriveType.Commodore1541; - } break; - case @".TAP": + case C64Format.D71: + if (diskDriveType == DiskDriveType.None) + diskDriveType = DiskDriveType.Commodore1571; + break; + case C64Format.T64: + case C64Format.TAP: if (tapeDriveType == TapeDriveType.None) { tapeDriveType = TapeDriveType.Commodore1530; @@ -235,41 +239,41 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 { foreach (var rom in Roms) { - switch (Path.GetExtension(rom.Key).ToUpper()) + switch (C64FormatFinder.GetFormat(rom)) { - case @".D64": - var d64 = D64.Read(rom.Value); + case C64Format.D64: + var d64 = D64.Read(rom); if (d64 != null) { _board.DiskDrive.InsertMedia(d64); } break; - case @".G64": - var g64 = G64.Read(rom.Value); + case C64Format.G64: + var g64 = G64.Read(rom); if (g64 != null) { _board.DiskDrive.InsertMedia(g64); } break; - case @".CRT": - var cart = CartridgeDevice.Load(rom.Value); + case C64Format.CRT: + var cart = CartridgeDevice.Load(rom); if (cart != null) { _board.CartPort.Connect(cart); } break; - case @".TAP": - var tape = Tape.Load(rom.Value); + case C64Format.TAP: + var tape = Tape.Load(rom); if (tape != null) { _board.TapeDrive.Insert(tape); } break; - case @".PRG": - if (rom.Value.Length > 2) + default: + if (rom.Length > 2) { _loadPrg = true; - _prgFile = rom.Value; + _prgFile = rom; } break; } @@ -290,8 +294,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 case DiskDriveType.Commodore1541II: _board.DiskDrive.DriveRom.Flash(GetFirmware(0x4000, "Drive1541II")); break; + case DiskDriveType.Commodore1571: + _board.DiskDrive.DriveRom.Flash(GetFirmware(0x8000, "Drive1571")); + break; } - } + } // ------------------------------------ diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64Format.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64Format.cs new file mode 100644 index 0000000000..b730b49d05 --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64Format.cs @@ -0,0 +1,24 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64 +{ + public enum C64Format + { + Unknown, + D64, + D71, + D81, + X64, + G64, + T64, + TAP, + CRT, + P64, + P00, + D82, + D80, + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs new file mode 100644 index 0000000000..36126f7c5d --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs @@ -0,0 +1,47 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64 +{ + public static class C64FormatFinder + { + public static C64Format GetFormat(byte[] data) + { + if (data == null || data.Length < 0x10) + return C64Format.Unknown; + using (var mem = new MemoryStream(data)) + { + var reader = new BinaryReader(mem); + var header = Encoding.GetEncoding(437).GetString(reader.ReadBytes(0x10)); + if (header.StartsWith("C64 CARTRIDGE ")) + return C64Format.CRT; + if (header.StartsWith("GCR-1541")) + return C64Format.G64; + if (header.StartsWith("C64S tape image ")) + return C64Format.T64; + if (header.StartsWith("C64-TAPE-RAW")) + return C64Format.TAP; + if (header.StartsWith("C64File")) + return C64Format.P00; + if (header.StartsWith("P64-1541")) + return C64Format.P64; + if (data[0] == 0x43 && data[1] == 0x15 && data[2] == 0x41 && data[3] == 0x64) + return C64Format.X64; + if (data.Length == 174848 || data.Length == 175531 || data.Length == 196608 || data.Length == 197376) + return C64Format.D64; + if (data.Length == 349696 || data.Length == 351062) + return C64Format.D71; + if (data.Length == 533248) + return C64Format.D80; + if (data.Length == 819200 || data.Length == 822400) + return C64Format.D81; + if (data.Length == 1066496) + return C64Format.D82; + } + return C64Format.Unknown; + } + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs index 188f69acf8..e1f27c3ed0 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs @@ -202,7 +202,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media } } - return new Disk(trackDatas, trackNumbers, trackDensities, trackLengths, 84); + return new Disk(trackDatas, trackNumbers, trackDensities, 84); } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs index 189066f5ee..e9874d481e 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Disk.cs @@ -35,7 +35,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media /// Density zones for the raw bit data. /// Length, in bits, of each raw bit data. /// Total number of tracks on the media. - public Disk(IList trackData, IList trackNumbers, IList trackDensities, IList trackLengths, int trackCapacity) + public Disk(IList trackData, IList trackNumbers, IList trackDensities, int trackCapacity) { WriteProtected = true; _tracks = new int[trackCapacity][]; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs index 2ddac42728..26b24523b3 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs @@ -54,7 +54,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media throw new Exception("Byte-level speeds are not yet supported in the G64 loader."); } - return new Disk(trackDatas, trackNumbers, trackDensities, trackLengths, 84); + return new Disk(trackDatas, trackNumbers, trackDensities, 84); } return new Disk(84); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs index c31635e03c..e7afd30f56 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -119,7 +119,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial if (_ledEnabled) { - _driveLightOffTime = 1000000; + _driveLightOffTime = 25000; } else if (_driveLightOffTime > 0) { From 178457309a36a8a195f9050fa41cbb9716a518ef Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Sat, 5 Mar 2016 16:00:19 -0600 Subject: [PATCH 10/15] C64: Drive light reflects both cartridge LED and disk drive now. --- .../Computers/Commodore64/C64.IDriveLight.cs | 4 ++-- BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs | 4 ++-- .../Computers/Commodore64/Cartridge/CartridgeDevice.cs | 6 +++++- .../Computers/Commodore64/Cartridge/CartridgePort.cs | 6 +++++- .../Computers/Commodore64/Cartridge/Mapper0020.cs | 3 +++ .../Computers/Commodore64/Serial/Drive1541.cs | 7 ------- .../Computers/Commodore64/Serial/SerialPort.cs | 2 ++ 7 files changed, 19 insertions(+), 13 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IDriveLight.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IDriveLight.cs index 7a1834497d..d782403dea 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IDriveLight.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IDriveLight.cs @@ -4,7 +4,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 { public partial class C64 : IDriveLight { - public bool DriveLightEnabled { get { return true; } } - public bool DriveLightOn { get; private set; } + public bool DriveLightEnabled { get { return _board != null && (_board.CartPort.DriveLightEnabled || _board.Serial.DriveLightEnabled); } } + public bool DriveLightOn { get { return _board != null && (_board.CartPort.DriveLightOn || _board.Serial.DriveLightOn);} } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 632e72073d..6c4a061b85 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -20,7 +20,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 )] [ServiceNotApplicable(typeof(ISettable<,>))] public sealed partial class C64 : IEmulator, IRegionable - { + { // framework public C64(CoreComm comm, GameInfo game, byte[] rom, object settings, object syncSettings) { @@ -51,7 +51,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 } ((BasicServiceProvider) ServiceProvider).Register(_board.Vic); - ((BasicServiceProvider) ServiceProvider).Register(_board.Serial); + ((BasicServiceProvider) ServiceProvider).Register(this); } // internal variables diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs index 2c7ef4fddc..7fed92029f 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs @@ -3,10 +3,11 @@ using System.Collections.Generic; using System.IO; using System.Linq; using BizHawk.Common; +using BizHawk.Emulation.Common; namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { - public abstract partial class CartridgeDevice + public abstract partial class CartridgeDevice : IDriveLight { public static CartridgeDevice Load(byte[] crtFile) { @@ -273,5 +274,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge public virtual void WriteDF00(int addr, int val) { } + + public bool DriveLightEnabled { get; protected set; } + public bool DriveLightOn { get; protected set; } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs index c980d8ae85..9ca7f2112b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs @@ -1,8 +1,9 @@ using BizHawk.Common; +using BizHawk.Emulation.Common; namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { - public sealed class CartridgePort + public sealed class CartridgePort : IDriveLight { private CartridgeDevice _cartridgeDevice; private bool _connected; @@ -126,5 +127,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { SaveState.SyncObject(ser, this); } + + public bool DriveLightEnabled { get { return _connected && _cartridgeDevice.DriveLightEnabled; } } + public bool DriveLightOn { get { return _connected && _cartridgeDevice.DriveLightOn; } } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0020.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0020.cs index ffb5906798..3bac7c0e69 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0020.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0020.cs @@ -53,6 +53,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge public Mapper0020(IList newAddresses, IList newBanks, IList newData) { + DriveLightEnabled = true; var count = newAddresses.Count; // force ultimax mode (the cart SHOULD set this @@ -194,6 +195,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge pinExRom = (val & 0x02) == 0; _boardLed = (val & 0x80) != 0; _internalRomState = 0; + DriveLightOn = _boardLed; } public override void Write8000(int addr, int val) @@ -289,6 +291,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge SaveState.SyncDelta("MediaStateA", ser, _originalMediaA, ref _banksA); SaveState.SyncDelta("MediaStateB", ser, _originalMediaB, ref _banksB); base.SyncState(ser); + DriveLightOn = _boardLed; } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs index e7afd30f56..1397450b1c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -25,8 +25,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial private bool _byteReady; [SaveState.SaveWithName("DriveCpuClockNumerator")] private readonly int _driveCpuClockNum; - [SaveState.SaveWithName("DriveCpuClockDenominator")] - private readonly int _driveCpuClockDen; [SaveState.SaveWithName("TrackNumber")] private int _trackNumber; [SaveState.SaveWithName("MotorEnabled")] @@ -35,8 +33,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial private bool _ledEnabled; [SaveState.SaveWithName("MotorStep")] private int _motorStep; - [SaveState.DoNotSave] - private int _via0PortBtemp; [SaveState.SaveWithName("CPU")] private readonly MOS6502X _cpu; [SaveState.SaveWithName("RAM")] @@ -114,9 +110,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial _cpu.IRQ = !(Via0.Irq && Via1.Irq); // active low IRQ line _cpu.ExecuteOne(); - _via0PortBtemp = Via0.EffectivePrB; - _ledEnabled = (_via0PortBtemp & 0x08) != 0; - if (_ledEnabled) { _driveLightOffTime = 25000; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPort.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPort.cs index 9e16296ad2..25b9debb42 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPort.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/SerialPort.cs @@ -84,5 +84,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial public bool DriveLightEnabled { get { return true; } } [SaveState.DoNotSave] public bool DriveLightOn { get { return ReadDeviceLight(); } } + [SaveState.DoNotSave] + public bool IsConnected { get { return _connected; } } } } From aaa0da85fded3c812e7500c6b2f9c4ac8c7bbc7f Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Sun, 6 Mar 2016 16:31:29 -0600 Subject: [PATCH 11/15] C64: Prophet64 and Action Replay mapper (incomplete) added. --- .../BizHawk.Emulation.Cores.csproj | 2 + .../Computers/Commodore64/C64.Motherboard.cs | 4 +- .../Commodore64/C64.MotherboardInterface.cs | 5 + .../Commodore64/Cartridge/CartridgeDevice.cs | 44 +++-- .../Commodore64/Cartridge/CartridgePort.cs | 6 +- .../Commodore64/Cartridge/Mapper0001.cs | 154 ++++++++++++++++++ .../Commodore64/Cartridge/Mapper000A.cs | 10 +- .../Commodore64/Cartridge/Mapper002B.cs | 80 +++++++++ 8 files changed, 280 insertions(+), 25 deletions(-) create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0001.cs create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper002B.cs diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index b8961e5eb6..cdbc943f47 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -209,6 +209,7 @@ + @@ -217,6 +218,7 @@ + diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 071e3f2dff..641692ffda 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -196,7 +196,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 } public void Init() - { + { + CartPort.ReadOpenBus = ReadOpenBus; + Cassette.ReadDataOutput = CassPort_ReadDataOutput; Cassette.ReadMotor = CassPort_ReadMotor; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs index 95f99362c2..14375eceb6 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs @@ -172,5 +172,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 _lastReadVicData = Pla.VicRead(_lastReadVicAddress); return _lastReadVicData; } + + private int ReadOpenBus() + { + return _lastReadVicData; + } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs index 7fed92029f..3636f28b85 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgeDevice.cs @@ -9,6 +9,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { public abstract partial class CartridgeDevice : IDriveLight { + public Func ReadOpenBus; + public static CartridgeDevice Load(byte[] crtFile) { using (var mem = new MemoryStream(crtFile)) @@ -70,33 +72,39 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge CartridgeDevice result; switch (mapper) { - case 0x0000: + case 0x0000: // Standard Cartridge result = new Mapper0000(chipAddress, chipData, game, exrom); break; - case 0x0005: + case 0x0001: // Action Replay (4.2 and up) + result = new Mapper0001(chipAddress, chipBank, chipData); + break; + case 0x0005: // Ocean result = new Mapper0005(chipAddress, chipBank, chipData); break; - case 0x000A: - result = new Mapper000A(chipAddress, chipBank, chipData); + case 0x000A: // Epyx FastLoad + result = new Mapper000A(chipData); break; - case 0x000B: + case 0x000B: // Westermann Learning result = new Mapper000B(chipAddress, chipData); break; - case 0x000F: + case 0x000F: // C64 Game System / System 3 result = new Mapper000F(chipAddress, chipBank, chipData); break; - case 0x0011: + case 0x0011: // Dinamic result = new Mapper0011(chipAddress, chipBank, chipData); break; - case 0x0012: + case 0x0012: // Zaxxon / Super Zaxxon result = new Mapper0012(chipAddress, chipBank, chipData); break; - case 0x0013: + case 0x0013: // Domark result = new Mapper0013(chipAddress, chipBank, chipData); break; - case 0x0020: + case 0x0020: // EasyFlash result = new Mapper0020(chipAddress, chipBank, chipData); break; + case 0x002B: // Prophet 64 + result = new Mapper002B(chipAddress, chipBank, chipData); + break; default: throw new Exception("This cartridge file uses an unrecognized mapper: " + mapper); } @@ -182,22 +190,22 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge public virtual int Peek8000(int addr) { - return 0xFF; + return ReadOpenBus(); } public virtual int PeekA000(int addr) { - return 0xFF; + return ReadOpenBus(); } public virtual int PeekDE00(int addr) { - return 0xFF; + return ReadOpenBus(); } public virtual int PeekDF00(int addr) { - return 0xFF; + return ReadOpenBus(); } public virtual void Poke8000(int addr, int val) @@ -218,22 +226,22 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge public virtual int Read8000(int addr) { - return 0xFF; + return ReadOpenBus(); } public virtual int ReadA000(int addr) { - return 0xFF; + return ReadOpenBus(); } public virtual int ReadDE00(int addr) { - return 0xFF; + return ReadOpenBus(); } public virtual int ReadDF00(int addr) { - return 0xFF; + return ReadOpenBus(); } [SaveState.DoNotSave] diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs index 9ca7f2112b..88c65a1a47 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/CartridgePort.cs @@ -1,10 +1,13 @@ -using BizHawk.Common; +using System; +using BizHawk.Common; using BizHawk.Emulation.Common; namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { public sealed class CartridgePort : IDriveLight { + public Func ReadOpenBus; + private CartridgeDevice _cartridgeDevice; private bool _connected; @@ -82,6 +85,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { _connected = true; _cartridgeDevice = newCartridgeDevice; + newCartridgeDevice.ReadOpenBus = ReadOpenBus; } public void Disconnect() diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0001.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0001.cs new file mode 100644 index 0000000000..d2dd29fbec --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper0001.cs @@ -0,0 +1,154 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge +{ + public abstract partial class CartridgeDevice + { + private sealed class Mapper0001 : CartridgeDevice + { + [SaveState.SaveWithName("RAM")] private readonly int[] _ram = new int[0x2000]; + [SaveState.SaveWithName("RAMEnabled")] private bool _ramEnabled; + [SaveState.DoNotSave] private readonly int[] _rom = new int[0x8000]; + [SaveState.SaveWithName("ROMOffset")] private int _romOffset; + [SaveState.SaveWithName("CartEnabled")] private bool _cartEnabled; + + public Mapper0001(IList newAddresses, IList newBanks, IList newData) + { + pinExRom = false; + pinGame = false; + for (var i = 0; i < newData.Count; i++) + { + if (newAddresses[i] == 0x8000) + Array.Copy(newData[i], 0, _rom, 0x2000 * newBanks[i], 0x2000); + } + _romOffset = 0; + _cartEnabled = true; + } + + public override void HardReset() + { + base.HardReset(); + pinExRom = false; + pinGame = false; + for (var i = 0; i < 0x2000; i++) + _ram[i] = 0x00; + _romOffset = 0; + _cartEnabled = true; + } + + public override int Peek8000(int addr) + { + return GetLoRom(addr); + } + + public override int PeekA000(int addr) + { + return Peek8000(addr); + } + + public override int PeekDF00(int addr) + { + return GetIo2(addr); + } + + public override void Poke8000(int addr, int val) + { + SetLoRom(addr, val); + } + + public override void PokeA000(int addr, int val) + { + Poke8000(addr, val); + } + + public override void PokeDE00(int addr, int val) + { + SetState(val); + } + + public override void PokeDF00(int addr, int val) + { + SetIo2(addr, val); + } + + public override int Read8000(int addr) + { + return GetLoRom(addr); + } + + public override int ReadA000(int addr) + { + return GetHiRom(addr); + } + + public override int ReadDF00(int addr) + { + return GetIo2(addr); + } + + public override void Write8000(int addr, int val) + { + SetLoRom(addr, val); + } + + public override void WriteA000(int addr, int val) + { + SetLoRom(addr, val); + } + + public override void WriteDE00(int addr, int val) + { + SetState(val); + } + + public override void WriteDF00(int addr, int val) + { + SetIo2(addr, val); + } + + private void SetState(int val) + { + pinGame = (val & 0x01) == 0; + pinExRom = (val & 0x02) != 0; + _cartEnabled = (val & 0x04) == 0; + _romOffset = (val & 0x18) << 10; + _ramEnabled = (val & 0x20) == 0; + } + + private int GetLoRom(int addr) + { + return _ramEnabled + ? _ram[addr & 0x1FFF] + : _rom[(addr & 0x1FFF) | _romOffset]; + } + + private int GetHiRom(int addr) + { + return _rom[(addr & 0x1FFF) | _romOffset]; + } + + private void SetLoRom(int addr, int val) + { + _ram[addr & 0x1FFF] = val; + } + + private int GetIo2(int addr) + { + if (!_cartEnabled) + return ReadOpenBus(); + + return _ramEnabled + ? _ram[(addr & 0xFF) | 0x1F00] + : _rom[(addr & 0xFF) | _romOffset | 0x1F00]; + } + + private void SetIo2(int addr, int val) + { + _ram[addr & 0x1FFF] = val & 0xFF; + } + } + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs index 47b904d97f..e3e1250e91 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper000A.cs @@ -5,11 +5,11 @@ using System.Text; namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge { - //792 - // Epyx Fastload. Uppermost page is always visible at DFxx. - // They use a capacitor that is recharged by accesses to DExx - // to pull down EXROM. + // They use a capacitor that is discharged by accesses to DExx + // to pull down EXROM. Also, accesses to LOROM while it is active + // discharge the capacitor. + // Thanks to VICE team for the info: http://vice-emu.sourceforge.net/vice_15.html public abstract partial class CartridgeDevice { @@ -24,7 +24,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge [SaveState.DoNotSave] private readonly int[] _rom; - public Mapper000A(IList newAddresses, IList newBanks, IList newData) + public Mapper000A(IList newData) { _rom = new int[0x2000]; Array.Copy(newData.First(), _rom, 0x2000); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper002B.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper002B.cs new file mode 100644 index 0000000000..8e14566357 --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Cartridge/Mapper002B.cs @@ -0,0 +1,80 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cartridge +{ + // Prophet 64 cartridge. Because we can. + // 32 banks of 8KB. + // DFxx = status register, xxABBBBB. A=enable cart, B=bank + // Thanks to VICE team for the info: http://vice-emu.sourceforge.net/vice_15.html + + public abstract partial class CartridgeDevice + { + private class Mapper002B : CartridgeDevice + { + [SaveState.DoNotSave] + private readonly int[] _rom; + [SaveState.SaveWithName("RomOffset")] + private int _romOffset; + [SaveState.SaveWithName("RomEnabled")] + private bool _romEnabled; + + public Mapper002B(IList newAddresses, IList newBanks, IList newData) + { + pinExRom = false; + pinGame = true; + _rom = new int[0x40000]; + Array.Copy(newData.First(), _rom, 0x2000); + pinGame = true; + for (var i = 0; i < newData.Count; i++) + { + if (newAddresses[i] == 0x8000) + { + Array.Copy(newData[i], 0, _rom, newBanks[i] * 0x2000, 0x2000); + } + } + } + + public override void HardReset() + { + _romEnabled = true; + _romOffset = 0; + } + + public override int Peek8000(int addr) + { + return _romOffset | (addr & 0x1FFF); + } + + public override int PeekDF00(int addr) + { + // For debugging only. The processor does not see this. + return ((_romOffset >> 13) & 0x1F) | (_romEnabled ? 0x20 : 0x00); + } + + public override void PokeDF00(int addr, int val) + { + _romOffset = (val & 0x1F) << 13; + _romEnabled = (val & 0x20) != 0; + } + + public override int Read8000(int addr) + { + return _romOffset | (addr & 0x1FFF); + } + + public override int ReadDF00(int addr) + { + return 0x00; + } + + public override void WriteDF00(int addr, int val) + { + _romOffset = (val & 0x1F) << 13; + _romEnabled = (val & 0x20) != 0; + } + } + } +} From 9557a253014ca5eeafd8d5d0c0718dcad5bb0686 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Wed, 9 Mar 2016 19:53:02 -0600 Subject: [PATCH 12/15] C64: Improved disk support. --- .../Computers/Commodore64/C64.ISettable.cs | 240 +++++++------- .../Computers/Commodore64/C64.cs | 55 ++-- .../Commodore64/MOS/Cia.Registers.cs | 16 + .../Computers/Commodore64/MOS/Cia.Tod.cs | 138 ++++---- .../Computers/Commodore64/MOS/Cia.cs | 12 +- .../Computers/Commodore64/MOS/LatchedPort.cs | 170 +++++----- .../Computers/Commodore64/MOS/Via.cs | 3 - .../Computers/Commodore64/Media/D64.cs | 34 +- .../Commodore64/Media/DiskBuilder.cs | 307 ++++++++++++++++++ .../Commodore64/User/UserPortDevice.cs | 98 +++--- 10 files changed, 702 insertions(+), 371 deletions(-) create mode 100644 BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs index a34ff6f602..d68e1f8503 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.ISettable.cs @@ -1,121 +1,121 @@ -using BizHawk.Emulation.Common; - -using Newtonsoft.Json; - -using System; -using System.ComponentModel; -using System.Drawing; - - -namespace BizHawk.Emulation.Cores.Computers.Commodore64 -{ - // adelikat: changing settings to default object until there are actually settings, as the ui depends on it to know if there are any settings avaialable - public partial class C64 : ISettable - { - public C64Settings GetSettings() - { - return Settings.Clone(); - } - - public C64SyncSettings GetSyncSettings() - { - return SyncSettings.Clone(); - } - - public bool PutSettings(C64Settings o) - { - Settings = o; - return false; - } - - public bool PutSyncSettings(C64SyncSettings o) - { - SyncSettings = o; - return false; - } - - internal C64Settings Settings { get; private set; } - internal C64SyncSettings SyncSettings { get; private set; } - - public class C64Settings - { - [DisplayName("Border type")] - [Description("Select how to show the border area")] - [DefaultValue(BorderType.SmallProportional)] - public BorderType BorderType { get; set; } - - public C64Settings Clone() - { - return (C64Settings)MemberwiseClone(); - } - - public C64Settings() - { - BizHawk.Common.SettingsUtil.SetDefaultValues(this); - } - } - - public class C64SyncSettings - { - [DisplayName("VIC type")] - [Description("Set the type of video chip to use")] - [DefaultValue(VicType.Pal)] - public VicType VicType { get; set; } - - [DisplayName("SID type")] - [Description("Set the type of sound chip to use")] - [DefaultValue(SidType.OldR2)] - public SidType SidType { get; set; } - - [DisplayName("Tape drive type")] - [Description("Set the type of tape drive attached")] - [DefaultValue(TapeDriveType.None)] - public TapeDriveType TapeDriveType { get; set; } - - [DisplayName("Disk drive type")] - [Description("Set the type of disk drive attached")] - [DefaultValue(DiskDriveType.None)] - public DiskDriveType DiskDriveType { get; set; } - - public C64SyncSettings Clone() - { - return (C64SyncSettings)MemberwiseClone(); - } - - public C64SyncSettings() - { - BizHawk.Common.SettingsUtil.SetDefaultValues(this); - } - } - - public enum VicType - { - Pal, Ntsc, NtscOld, Drean - } - - public enum CiaType - { - Pal, Ntsc, PalRevA, NtscRevA - } - - public enum BorderType - { - SmallProportional, SmallFixed, Normal, Full - } - - public enum SidType - { - OldR2, OldR3, OldR4AR, NewR5 - } - - public enum TapeDriveType - { - None, Commodore1530 - } - - public enum DiskDriveType - { - None, Commodore1541, Commodore1541II, Commodore1571 - } - } +using BizHawk.Emulation.Common; + +using Newtonsoft.Json; + +using System; +using System.ComponentModel; +using System.Drawing; + + +namespace BizHawk.Emulation.Cores.Computers.Commodore64 +{ + // adelikat: changing settings to default object until there are actually settings, as the ui depends on it to know if there are any settings avaialable + public partial class C64 : ISettable + { + public C64Settings GetSettings() + { + return Settings.Clone(); + } + + public C64SyncSettings GetSyncSettings() + { + return SyncSettings.Clone(); + } + + public bool PutSettings(C64Settings o) + { + Settings = o; + return false; + } + + public bool PutSyncSettings(C64SyncSettings o) + { + SyncSettings = o; + return false; + } + + internal C64Settings Settings { get; private set; } + internal C64SyncSettings SyncSettings { get; private set; } + + public class C64Settings + { + [DisplayName("Border type")] + [Description("Select how to show the border area")] + [DefaultValue(BorderType.SmallProportional)] + public BorderType BorderType { get; set; } + + public C64Settings Clone() + { + return (C64Settings)MemberwiseClone(); + } + + public C64Settings() + { + BizHawk.Common.SettingsUtil.SetDefaultValues(this); + } + } + + public class C64SyncSettings + { + [DisplayName("VIC type")] + [Description("Set the type of video chip to use")] + [DefaultValue(VicType.Pal)] + public VicType VicType { get; set; } + + [DisplayName("SID type")] + [Description("Set the type of sound chip to use")] + [DefaultValue(SidType.OldR2)] + public SidType SidType { get; set; } + + [DisplayName("Tape drive type")] + [Description("Set the type of tape drive attached")] + [DefaultValue(TapeDriveType.None)] + public TapeDriveType TapeDriveType { get; set; } + + [DisplayName("Disk drive type")] + [Description("Set the type of disk drive attached")] + [DefaultValue(DiskDriveType.None)] + public DiskDriveType DiskDriveType { get; set; } + + public C64SyncSettings Clone() + { + return (C64SyncSettings)MemberwiseClone(); + } + + public C64SyncSettings() + { + BizHawk.Common.SettingsUtil.SetDefaultValues(this); + } + } + + public enum VicType + { + Pal, Ntsc, NtscOld, Drean + } + + public enum CiaType + { + Pal, Ntsc, PalRevA, NtscRevA + } + + public enum BorderType + { + SmallProportional, SmallFixed, Normal, Full + } + + public enum SidType + { + OldR2, OldR3, OldR4AR, NewR5 + } + + public enum TapeDriveType + { + None, Commodore1530 + } + + public enum DiskDriveType + { + None, Commodore1541, Commodore1541II + } + } } \ No newline at end of file diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 6c4a061b85..6276a45612 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -156,17 +156,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 _board.Execute(); _frameCycles++; - // load PRG file if needed - if (_loadPrg) - { - // check to see if cpu PC is at the BASIC warm start vector - if (_board.Cpu.Pc != 0 && _board.Cpu.Pc == ((_board.Ram.Peek(0x0303) << 8) | _board.Ram.Peek(0x0302))) - { - Prg.Load(_board.Pla, _prgFile); - _loadPrg = false; - } - } - if (_frameCycles != _cyclesPerFrame) { return; @@ -188,8 +177,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 } private Motherboard _board; - private bool _loadPrg; - [SaveState.DoNotSave] private byte[] _prgFile; private byte[] GetFirmware(int length, params string[] names) { @@ -212,10 +199,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 if (diskDriveType == DiskDriveType.None) diskDriveType = DiskDriveType.Commodore1541; break; - case C64Format.D71: - if (diskDriveType == DiskDriveType.None) - diskDriveType = DiskDriveType.Commodore1571; - break; case C64Format.T64: case C64Format.TAP: if (tapeDriveType == TapeDriveType.None) @@ -223,6 +206,19 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 tapeDriveType = TapeDriveType.Commodore1530; } break; + case C64Format.CRT: + // Nothing required. + break; + case C64Format.Unknown: + if (rom.Length >= 0xFE00) + { + throw new Exception("The image format is not known, and too large to be used as a PRG."); + } + if (diskDriveType == DiskDriveType.None) + diskDriveType = DiskDriveType.Commodore1541; + break; + default: + throw new Exception("The image format is not yet supported by the Commodore 64 core."); } } @@ -269,11 +265,25 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 _board.TapeDrive.Insert(tape); } break; - default: - if (rom.Length > 2) + case C64Format.Unknown: + var prgDisk = new DiskBuilder { - _loadPrg = true; - _prgFile = rom; + Entries = new List + { + new DiskBuilder.Entry + { + Closed = true, + Data = rom, + Locked = false, + Name = "PRG", + RecordLength = 0, + Type = DiskBuilder.FileType.Program + } + } + }.Build(); + if (prgDisk != null) + { + _board.DiskDrive.InsertMedia(prgDisk); } break; } @@ -294,9 +304,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 case DiskDriveType.Commodore1541II: _board.DiskDrive.DriveRom.Flash(GetFirmware(0x4000, "Drive1541II")); break; - case DiskDriveType.Commodore1571: - _board.DiskDrive.DriveRom.Flash(GetFirmware(0x8000, "Drive1571")); - break; } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs index 6f7061cbea..ae08655914 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs @@ -173,6 +173,22 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _icr |= 0x80; } break; + case 0xE: + var oldCra = _cra; + WriteRegister(addr, val); + + // Toggle output begins high when timer starts. + if ((_cra & 0x05) == 0x05 && (oldCra & 0x01) == 0) + _prb |= 0x40; + break; + case 0xF: + var oldCrb = _crb; + WriteRegister(addr, val); + + // Toggle output begins high when timer starts. + if ((_crb & 0x05) == 0x05 && (oldCrb & 0x01) == 0) + _prb |= 0x80; + break; default: WriteRegister(addr, val); break; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Tod.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Tod.cs index f0512eddc2..e06b1c1ab2 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Tod.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Tod.cs @@ -1,69 +1,69 @@ -namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS -{ - public sealed partial class Cia - { - private void CountTod() - { - if (_todCounter > 0) - { - _todCounter -= _todDen; - return; - } - - _todCounter += _todNum * ((_cra & 0x80) != 0 ? 6 : 5); - _tod10Ths++; - if (_tod10Ths > 9) - { - _tod10Ths = 0; - _todlo = (_todSec & 0x0F) + 1; - _todhi = (_todSec >> 4); - if (_todlo > 9) - { - _todlo = 0; - _todhi++; - } - if (_todhi > 5) - { - _todSec = 0; - _todlo = (_todMin & 0x0F) + 1; - _todhi = (_todMin >> 4); - if (_todlo > 9) - { - _todlo = 0; - _todhi++; - } - if (_todhi > 5) - { - _todMin = 0; - _todlo = (_todHr & 0x0F) + 1; - _todhi = (_todHr >> 4); - _todHr &= 0x80; - if (_todlo > 9) - { - _todlo = 0; - _todhi++; - } - _todHr |= (_todhi << 4) | _todlo; - if ((_todHr & 0x1F) > 0x11) - { - _todHr &= 0x80 ^ 0x80; - } - } - else - { - _todMin = (_todhi << 4) | _todlo; - } - } - else - { - _todSec = (_todhi << 4) | _todlo; - } - } - - if (_tod10Ths == _alm10Ths && _todSec == _almSec && _todMin == _almMin && _todHr == _almHr) - { - TriggerInterrupt(4); - } - } - } -} +namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS +{ + public sealed partial class Cia + { + private void CountTod() + { + if (_todCounter > 0) + { + _todCounter -= _todDen; + return; + } + + _todCounter += _todNum * ((_cra & 0x80) != 0 ? 6 : 5); + _tod10Ths++; + if (_tod10Ths > 9) + { + _tod10Ths = 0; + _todlo = (_todSec & 0x0F) + 1; + _todhi = (_todSec >> 4); + if (_todlo > 9) + { + _todlo = 0; + _todhi++; + } + if (_todhi > 5) + { + _todSec = 0; + _todlo = (_todMin & 0x0F) + 1; + _todhi = (_todMin >> 4); + if (_todlo > 9) + { + _todlo = 0; + _todhi++; + } + if (_todhi > 5) + { + _todMin = 0; + _todlo = (_todHr & 0x0F) + 1; + _todhi = (_todHr >> 4); + _todHr &= 0x80; + if (_todlo > 9) + { + _todlo = 0; + _todhi++; + } + _todHr |= (_todhi << 4) | _todlo; + if ((_todHr & 0x1F) > 0x11) + { + _todHr &= 0x80 ^ 0x80; + } + } + else + { + _todMin = (_todhi << 4) | _todlo; + } + } + else + { + _todSec = (_todhi << 4) | _todlo; + } + } + + if (_tod10Ths == _alm10Ths && _todSec == _almSec && _todMin == _almMin && _todHr == _almHr) + { + TriggerInterrupt(4); + } + } + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs index 9d54a8fbea..3606d6424f 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs @@ -164,7 +164,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _prb &= 0x7F; _tbPrb7NegativeNextCycle = false; } - + switch (_taState) { @@ -268,6 +268,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { CheckIrqs(); } + + if ((_cra & 0x02) != 0) + _ddra |= 0x40; + if ((_crb & 0x02) != 0) + _ddrb |= 0x80; } private void Ta_Count() @@ -412,9 +417,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _tbState = TimerState.LoadThenCount; } - if ((_cra & 0x02) != 0) + if ((_crb & 0x02) != 0) { - if ((_cra & 0x04) != 0) + if ((_crb & 0x04) != 0) { _tbPrb7NegativeNextCycle = true; _prb |= 0x80; @@ -423,7 +428,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { _prb ^= 0x80; } - _ddrb |= 0x80; } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/LatchedPort.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/LatchedPort.cs index 2fd7e3bd3c..d4931fe541 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/LatchedPort.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/LatchedPort.cs @@ -1,85 +1,85 @@ -using System; -using BizHawk.Common; - -namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS -{ - public sealed class LatchedPort - { - public int Direction; - public int Latch; - - public LatchedPort() - { - Direction = 0x00; - Latch = 0x00; - } - - // data works like this in these types of systems: - // - // directionA directionB result - // 0 0 1 - // 1 0 latchA - // 0 1 latchB - // 1 1 latchA && latchB - // - // however because this uses transistor logic, there are cases where wired-ands - // cause the pull-up resistors not to be enough to keep the bus bit set to 1 when - // both the direction and latch are 1 (the keyboard and joystick port 2 can do this.) - // the class does not handle this case as it must be handled differently in every occurrence. - - public int ReadInput(int bus) - { - return (Latch & Direction) | ((Direction ^ 0xFF) & bus); - } - - public int ReadOutput() - { - return (Latch & Direction) | (Direction ^ 0xFF); - } - - public void SyncState(Serializer ser) - { - SaveState.SyncObject(ser, this); - } - } - - public sealed class LatchedBooleanPort - { - public bool Direction; - public bool Latch; - - public LatchedBooleanPort() - { - Direction = false; - Latch = false; - } - - // data dir bus out - // 0 0 0 0 - // 0 0 1 1 - - // 0 1 0 0 - // 0 1 1 0 - - // 1 0 0 0 - // 1 0 1 1 - - // 1 1 0 1 - // 1 1 1 1 - - public bool ReadInput(bool bus) - { - return (Direction && Latch) || (!Direction && bus); - } - - public bool ReadOutput() - { - return (Latch || !Direction); - } - - public void SyncState(Serializer ser) - { - SaveState.SyncObject(ser, this); - } - } -} +using System; +using BizHawk.Common; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS +{ + public sealed class LatchedPort + { + public int Direction; + public int Latch; + + public LatchedPort() + { + Direction = 0x00; + Latch = 0x00; + } + + // data works like this in these types of systems: + // + // directionA directionB result + // 0 0 1 + // 1 0 latchA + // 0 1 latchB + // 1 1 latchA && latchB + // + // however because this uses transistor logic, there are cases where wired-ands + // cause the pull-up resistors not to be enough to keep the bus bit set to 1 when + // both the direction and latch are 1 (the keyboard and joystick port 2 can do this.) + // the class does not handle this case as it must be handled differently in every occurrence. + + public int ReadInput(int bus) + { + return (Latch & Direction) | ((Direction ^ 0xFF) & bus); + } + + public int ReadOutput() + { + return (Latch & Direction) | (Direction ^ 0xFF); + } + + public void SyncState(Serializer ser) + { + SaveState.SyncObject(ser, this); + } + } + + public sealed class LatchedBooleanPort + { + public bool Direction; + public bool Latch; + + public LatchedBooleanPort() + { + Direction = false; + Latch = false; + } + + // data dir bus out + // 0 0 0 0 + // 0 0 1 1 + + // 0 1 0 0 + // 0 1 1 0 + + // 1 0 0 0 + // 1 0 1 1 + + // 1 1 0 1 + // 1 1 1 1 + + public bool ReadInput(bool bus) + { + return (Direction && Latch) || (!Direction && bus); + } + + public bool ReadOutput() + { + return (Latch || !Direction); + } + + public void SyncState(Serializer ser) + { + SaveState.SyncObject(ser, this); + } + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs index ad60babc6b..39be668838 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs @@ -108,9 +108,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS [SaveState.SaveWithName("HandshakeCb2NextClock")] private bool _handshakeCb2NextClock; - [SaveState.SaveWithName("ShiftRegisterCounter")] - private int _shiftCount; - [SaveState.SaveWithName("CA1")] public bool Ca1; [SaveState.SaveWithName("CA2")] diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs index e1f27c3ed0..dc46956c18 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs @@ -6,7 +6,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { public static class D64 { - private static readonly int[] densityTable = + private static readonly int[] DensityTable = { 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, @@ -18,7 +18,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media 0, 0, 0, 0, 0 }; - private static readonly int[] gcrDecodeTable = + private static readonly int[] GcrDecodeTable = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, //00xxx 0xFF, 0x08, 0x00, 0x01, 0xFF, 0x0C, 0x04, 0x05, //01xxx @@ -26,7 +26,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media 0xFF, 0x09, 0x0A, 0x0B, 0xFF, 0x0D, 0x0E, 0xFF //11xxx }; - private static readonly int[] gcrEncodeTable = + private static readonly int[] GcrEncodeTable = { Convert.ToByte("01010", 2), // 0 Convert.ToByte("01011", 2), // 1 @@ -46,7 +46,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media Convert.ToByte("10101", 2) // F }; - private static readonly int[] sectorsPerTrack = + private static readonly int[] SectorsPerTrack = { 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, @@ -58,7 +58,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media 17, 17, 17, 17, 17 }; - private static readonly int[] standardTrackLengthBytes = + private static readonly int[] StandardTrackLengthBytes = { 6250, 6666, 7142, 7692 }; @@ -116,14 +116,14 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media for (var i = 0; i < count; i += 4) { Array.Copy(source, i, data, 0, 4); - gcr[0] = gcrEncodeTable[data[0] >> 4]; - gcr[1] = gcrEncodeTable[data[0] & 0xF]; - gcr[2] = gcrEncodeTable[data[1] >> 4]; - gcr[3] = gcrEncodeTable[data[1] & 0xF]; - gcr[4] = gcrEncodeTable[data[2] >> 4]; - gcr[5] = gcrEncodeTable[data[2] & 0xF]; - gcr[6] = gcrEncodeTable[data[3] >> 4]; - gcr[7] = gcrEncodeTable[data[3] & 0xF]; + gcr[0] = GcrEncodeTable[data[0] >> 4]; + gcr[1] = GcrEncodeTable[data[0] & 0xF]; + gcr[2] = GcrEncodeTable[data[1] >> 4]; + gcr[3] = GcrEncodeTable[data[1] & 0xF]; + gcr[4] = GcrEncodeTable[data[2] >> 4]; + gcr[5] = GcrEncodeTable[data[2] & 0xF]; + gcr[6] = GcrEncodeTable[data[3] >> 4]; + gcr[7] = GcrEncodeTable[data[3] & 0xF]; // -------- -------- -------- -------- -------- // 00000111 11222223 33334444 45555566 66677777 @@ -175,7 +175,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media for (var i = 0; i < trackCount; i++) { - var sectors = sectorsPerTrack[i]; + var sectors = SectorsPerTrack[i]; var trackLengthBits = 0; using (var trackMem = new MemoryStream()) { @@ -187,10 +187,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media trackMem.Write(diskData, 0, diskData.Length); trackLengthBits += bitsWritten; } - var density = densityTable[i]; + var density = DensityTable[i]; // we pad the tracks with extra gap bytes to meet MNIB standards - while (trackMem.Length < standardTrackLengthBytes[density]) + while (trackMem.Length < StandardTrackLengthBytes[density]) { trackMem.WriteByte(0x55); } @@ -198,7 +198,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media trackDatas.Add(trackMem.ToArray()); trackLengths.Add(trackLengthBits); trackNumbers.Add(i * 2); - trackDensities.Add(densityTable[i]); + trackDensities.Add(DensityTable[i]); } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs new file mode 100644 index 0000000000..cd0f5d2a62 --- /dev/null +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs @@ -0,0 +1,307 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Linq.Expressions; +using System.Text; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media +{ + public class DiskBuilder + { + public enum FileType + { + Deleted = 0, + Sequential = 1, + Program = 2, + User = 3, + Relative = 4 + } + + protected class BamEntry + { + public int Data { get; private set; } + public int Sectors { get; private set; } + + public BamEntry(int sectors) + { + Data = 0; + for (var i = 0; i < sectors; i++) + { + Data >>= 1; + Data |= 0x800000; + } + Data |= (sectors << 24); + Sectors = sectors; + } + + private int GetBit(int sector) + { + if (sector < 0 || sector >= Sectors) + { + return 0; + } + return 0x800000 >> sector; + } + + public void Allocate(int sector) + { + var bit = GetBit(sector); + if (bit != 0 && (Data & bit) != 0) + { + Data &= ~bit; + Data -= 0x1000000; + } + } + + public void Free(int sector) + { + var bit = GetBit(sector); + if (bit != 0 && (Data & bit) == 0) + { + Data |= bit; + Data += 0x1000000; + } + } + + public int SectorsRemaining + { + get { return (Data >> 24) & 0xFF; } + } + + public bool this[int sector] + { + get { return (Data & (1 << sector)) != 0; } + set + { + if (value) + Free(sector); + else + Allocate(sector); + } + } + + public byte[] GetBytes() + { + return GetBytesEnumerable().ToArray(); + } + + private IEnumerable GetBytesEnumerable() + { + yield return unchecked((byte)(Data >> 24)); + yield return unchecked((byte)(Data >> 16)); + yield return unchecked((byte)(Data >> 8)); + yield return unchecked((byte)Data); + } + + public IEnumerable Entries + { + get + { + var d = Data; + for (var i = 0; i < Sectors; i++) + { + d <<= 1; + yield return (d & 0x1000000) != 0; + } + } + } + } + + protected class LocatedEntry + { + public Entry Entry { get; set; } + public int DirectoryTrack { get; set; } + public int DirectorySector { get; set; } + public int Track { get; set; } + public int Sector { get; set; } + public int SideTrack { get; set; } + public int SideSector { get; set; } + public int LengthInSectors { get; set; } + } + + public class Entry + { + public FileType Type { get; set; } + public bool Locked { get; set; } + public bool Closed { get; set; } + public string Name { get; set; } + public int RecordLength { get; set; } + public byte[] Data { get; set; } + } + + private static readonly int[] SectorsPerTrack = + { + 21, 21, 21, 21, 21, + 21, 21, 21, 21, 21, + 21, 21, 21, 21, 21, + 21, 21, 19, 19, 19, + 19, 19, 19, 19, 18, + 18, 18, 18, 18, 18, + 17, 17, 17, 17, 17, + 17, 17, 17, 17, 17 + }; + + public List Entries { get; set; } + public int VersionType { get; set; } + public string Title { get; set; } + + public DiskBuilder() + { + Entries = new List(); + VersionType = 0x41; + } + + public Disk Build() + { + const int tracks = 35; + var trackByteOffsets = new int[tracks]; + var bam = new BamEntry[tracks]; + var diskFull = false; + + for (var i = 0; i < tracks; i++) + { + bam[i] = new BamEntry(SectorsPerTrack[i]); + if (i > 0) + { + trackByteOffsets[i] = trackByteOffsets[i - 1] + (SectorsPerTrack[i - 1] * 256); + } + } + var bytes = new byte[trackByteOffsets[tracks - 1] + (SectorsPerTrack[tracks - 1] *256)]; + + var currentTrack = 16; + var currentSector = 0; + var interleaveStart = 0; + var sectorInterleave = 3; + var directory = new List(); + + Func GetOutputOffset = (t, s) => trackByteOffsets[t] + (s*256); + + foreach (var entry in Entries) + { + var sourceOffset = 0; + var dataLength = entry.Data == null ? 0 : entry.Data.Length; + var lengthInSectors = dataLength / 254; + var dataRemaining = dataLength; + var directoryEntry = new LocatedEntry + { + Entry = entry, + LengthInSectors = lengthInSectors + 1, + Track = currentTrack, + Sector = currentSector + }; + directory.Add(directoryEntry); + + while (!diskFull) + { + var outputOffset = GetOutputOffset(currentTrack, currentSector); + + if (dataRemaining > 254) + { + Array.Copy(entry.Data, sourceOffset, bytes, outputOffset + 2, 254); + dataRemaining -= 254; + } + else + { + if (dataRemaining > 0) + { + Array.Copy(entry.Data, sourceOffset, bytes, outputOffset + 2, dataRemaining); + bytes[outputOffset + 0] = 0; + bytes[outputOffset + 1] = (byte) dataRemaining; + dataRemaining = 0; + } + } + + bam[currentTrack].Allocate(currentSector); + currentSector += sectorInterleave; + if (currentSector >= SectorsPerTrack[currentTrack]) + { + interleaveStart++; + if (interleaveStart >= sectorInterleave) + { + interleaveStart = 0; + if (currentTrack >= 17) + { + currentTrack++; + if (currentTrack >= 35) + { + diskFull = true; + break; + } + } + else + { + currentTrack--; + if (currentTrack < 0) + currentTrack = 18; + } + } + currentSector = interleaveStart; + } + + if (dataRemaining <= 0) + break; + + bytes[outputOffset + 0] = (byte)(currentTrack + 1); + bytes[outputOffset + 1] = (byte) currentSector; + } + + if (diskFull) + break; + } + + // write Directory + var directoryOffset = -(0x20); + currentTrack = 17; + currentSector = 1; + var directoryOutputOffset = GetOutputOffset(currentTrack, currentSector); + var fileIndex = 0; + bam[currentTrack].Allocate(currentSector); + foreach (var entry in directory) + { + directoryOffset += 0x20; + if (directoryOffset == 0x100) + { + directoryOffset = 0; + currentSector += 3; + bytes[directoryOutputOffset] = (byte) currentTrack; + bytes[directoryOutputOffset + 1] = (byte) currentSector; + directoryOutputOffset = GetOutputOffset(currentTrack, currentSector); + bam[currentTrack].Allocate(currentSector); + } + bytes[directoryOutputOffset + directoryOffset + 0x00] = 0x00; + bytes[directoryOutputOffset + directoryOffset + 0x01] = 0x00; + bytes[directoryOutputOffset + directoryOffset + 0x02] = (byte)((int)entry.Entry.Type | (entry.Entry.Locked ? 0x40 : 0x00) | (entry.Entry.Closed ? 0x80 : 0x00)); + bytes[directoryOutputOffset + directoryOffset + 0x03] = (byte)(entry.Track + 1); + bytes[directoryOutputOffset + directoryOffset + 0x04] = (byte)entry.Sector; + for (var i = 0x05; i <= 0x14; i++) + bytes[directoryOutputOffset + directoryOffset + i] = 0xA0; + var fileNameBytes = Encoding.ASCII.GetBytes(entry.Entry.Name ?? string.Format("FILE{0:D3}", fileIndex)); + Array.Copy(fileNameBytes, 0, bytes, directoryOutputOffset + directoryOffset + 0x05, Math.Min(fileNameBytes.Length, 0x10)); + bytes[directoryOutputOffset + directoryOffset + 0x1E] = (byte)(entry.LengthInSectors & 0xFF); + bytes[directoryOutputOffset + directoryOffset + 0x1F] = (byte)((entry.LengthInSectors >> 8) & 0xFF); + fileIndex++; + } + bytes[directoryOutputOffset + 0x00] = 0x00; + bytes[directoryOutputOffset + 0x01] = 0xFF; + + // write BAM + var bamOutputOffset = GetOutputOffset(17, 0); + bytes[bamOutputOffset + 0x00] = 18; + bytes[bamOutputOffset + 0x01] = 1; + bytes[bamOutputOffset + 0x02] = (byte)VersionType; + for (var i = 0; i < 35; i++) + { + Array.Copy(bam[i].GetBytes(), 0, bytes, bamOutputOffset + 4 + (i * 4), 4); + } + for (var i = 0x90; i <= 0xAA; i++) + { + bytes[bamOutputOffset + i] = 0xA0; + } + var titleBytes = Encoding.ASCII.GetBytes(Title ?? "UNTITLED"); + Array.Copy(titleBytes, 0, bytes, bamOutputOffset + 0x90, Math.Min(titleBytes.Length, 0x10)); + + return D64.Read(bytes); + } + } +} diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/User/UserPortDevice.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/User/UserPortDevice.cs index b4e215f3bc..108e4d2674 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/User/UserPortDevice.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/User/UserPortDevice.cs @@ -1,49 +1,49 @@ -using System; -using BizHawk.Common; - -namespace BizHawk.Emulation.Cores.Computers.Commodore64.User -{ - public abstract class UserPortDevice - { - public Func ReadCounter1; - public Func ReadCounter2; - public Func ReadHandshake; - public Func ReadSerial1; - public Func ReadSerial2; - - public virtual void HardReset() - { - // note: this will not disconnect any attached media - } - - public virtual bool ReadAtn() - { - return true; - } - - public virtual int ReadData() - { - return 0xFF; - } - - public virtual bool ReadFlag2() - { - return true; - } - - public virtual bool ReadPa2() - { - return true; - } - - public virtual bool ReadReset() - { - return true; - } - - public void SyncState(Serializer ser) - { - SaveState.SyncObject(ser, this); - } - } -} +using System; +using BizHawk.Common; + +namespace BizHawk.Emulation.Cores.Computers.Commodore64.User +{ + public abstract class UserPortDevice + { + public Func ReadCounter1; + public Func ReadCounter2; + public Func ReadHandshake; + public Func ReadSerial1; + public Func ReadSerial2; + + public virtual void HardReset() + { + // note: this will not disconnect any attached media + } + + public virtual bool ReadAtn() + { + return true; + } + + public virtual int ReadData() + { + return 0xFF; + } + + public virtual bool ReadFlag2() + { + return true; + } + + public virtual bool ReadPa2() + { + return true; + } + + public virtual bool ReadReset() + { + return true; + } + + public void SyncState(Serializer ser) + { + SaveState.SyncObject(ser, this); + } + } +} From bd7e10af6be7ffa05b7472faf231314cc8a996b2 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Wed, 9 Mar 2016 19:53:41 -0600 Subject: [PATCH 13/15] C64: Add disk builder to project. --- BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj | 1 + 1 file changed, 1 insertion(+) diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index cdbc943f47..3a3b97c08a 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -222,6 +222,7 @@ + From aa83c33c58685e40172e60e0ec1466cdb9eb3af8 Mon Sep 17 00:00:00 2001 From: Anthony Konzel Date: Wed, 9 Mar 2016 20:12:49 -0600 Subject: [PATCH 14/15] 6502: All non-writes are affected by RDY. --- .../CPUs/MOS 6502X/Execute.cs | 150 ++++++++++-------- 1 file changed, 85 insertions(+), 65 deletions(-) diff --git a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs index acb5f97f3b..32565b7cad 100644 --- a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs +++ b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs @@ -534,7 +534,7 @@ namespace BizHawk.Emulation.Cores.Components.M6502 void FetchDummy() { - DummyReadMemory(PC); + DummyReadMemory(PC); } public void Execute(int cycles) @@ -1152,45 +1152,59 @@ namespace BizHawk.Emulation.Cores.Components.M6502 } void RelBranch_Stage3() { - FetchDummy(); - alu_temp = (byte)PC + (int)(sbyte)opcode2; - PC &= 0xFF00; - PC |= (ushort)((alu_temp & 0xFF)); - if (alu_temp.Bit(8)) - { - //we need to carry the add, and then we'll be ready to fetch the next instruction - opcode = VOP_RelativeStuff2; - mi = -1; - } - else - { - //to pass cpu_interrupts_v2/5-branch_delays_irq we need to handle a quirk here - //if we decide to interrupt in the next cycle, this condition will cause it to get deferred by one instruction - if (!interrupt_pending) - branch_irq_hack = true; - } - - } + rdy_freeze = !RDY; + if (RDY) + { + FetchDummy(); + alu_temp = (byte)PC + (int)(sbyte)opcode2; + PC &= 0xFF00; + PC |= (ushort)((alu_temp & 0xFF)); + if (alu_temp.Bit(8)) + { + //we need to carry the add, and then we'll be ready to fetch the next instruction + opcode = VOP_RelativeStuff2; + mi = -1; + } + else + { + //to pass cpu_interrupts_v2/5-branch_delays_irq we need to handle a quirk here + //if we decide to interrupt in the next cycle, this condition will cause it to get deferred by one instruction + if (!interrupt_pending) + branch_irq_hack = true; + } + } + } void RelBranch_Stage4() { - FetchDummy(); - if (alu_temp.Bit(31)) - PC = (ushort)(PC - 0x100); - else PC = (ushort)(PC + 0x100); - - - } + rdy_freeze = !RDY; + if (RDY) + { + FetchDummy(); + if (alu_temp.Bit(31)) + PC = (ushort)(PC - 0x100); + else PC = (ushort)(PC + 0x100); + } + } void NOP() { - } - void DecS() + rdy_freeze = !RDY; + } + void DecS() { - S--; - } + rdy_freeze = !RDY; + if (RDY) + { + S--; + } + } void IncS() { - S++; - } + rdy_freeze = !RDY; + if (RDY) + { + S++; + } + } void JSR() { rdy_freeze = !RDY; @@ -2041,35 +2055,50 @@ namespace BizHawk.Emulation.Cores.Components.M6502 } void Imp_ASL_A() { - FetchDummy(); - FlagC = (A & 0x80) != 0; - A = (byte)(A << 1); - NZ_A(); - } + rdy_freeze = !RDY; + if (RDY) + { + FetchDummy(); + FlagC = (A & 0x80) != 0; + A = (byte)(A << 1); + NZ_A(); + } + } void Imp_ROL_A() { - FetchDummy(); - temp8 = A; - A = (byte)((A << 1) | (P & 1)); - FlagC = (temp8 & 0x80) != 0; - NZ_A(); - } + rdy_freeze = !RDY; + if (RDY) + { + FetchDummy(); + temp8 = A; + A = (byte)((A << 1) | (P & 1)); + FlagC = (temp8 & 0x80) != 0; + NZ_A(); + } + } void Imp_ROR_A() { - FetchDummy(); - temp8 = A; - A = (byte)((A >> 1) | ((P & 1) << 7)); - FlagC = (temp8 & 1) != 0; - NZ_A(); - } + rdy_freeze = !RDY; + if (RDY) + { + FetchDummy(); + temp8 = A; + A = (byte)((A >> 1) | ((P & 1) << 7)); + FlagC = (temp8 & 1) != 0; + NZ_A(); + } + } void Imp_LSR_A() { - FetchDummy(); - FlagC = (A & 1) != 0; - A = (byte)(A >> 1); - NZ_A(); - - } + rdy_freeze = !RDY; + if (RDY) + { + FetchDummy(); + FlagC = (A & 1) != 0; + A = (byte)(A >> 1); + NZ_A(); + } + } void JMP_abs() { rdy_freeze = !RDY; @@ -2082,8 +2111,6 @@ namespace BizHawk.Emulation.Cores.Components.M6502 void IncPC() { PC++; - - } void ZP_RMW_Stage3() { @@ -2097,21 +2124,18 @@ namespace BizHawk.Emulation.Cores.Components.M6502 void ZP_RMW_Stage5() { WriteMemory(opcode2, (byte)alu_temp); - } void ZP_RMW_INC() { WriteMemory(opcode2, (byte)alu_temp); alu_temp = (byte)((alu_temp + 1) & 0xFF); P = (byte)((P & 0x7D) | TableNZ[alu_temp]); - } void ZP_RMW_DEC() { WriteMemory(opcode2, (byte)alu_temp); alu_temp = (byte)((alu_temp - 1) & 0xFF); P = (byte)((P & 0x7D) | TableNZ[alu_temp]); - } void ZP_RMW_ASL() { @@ -2120,7 +2144,6 @@ namespace BizHawk.Emulation.Cores.Components.M6502 FlagC = (value8 & 0x80) != 0; alu_temp = value8 = (byte)(value8 << 1); P = (byte)((P & 0x7D) | TableNZ[value8]); - } void ZP_RMW_SRE() { @@ -2163,7 +2186,6 @@ namespace BizHawk.Emulation.Cores.Components.M6502 alu_temp = value8 = (byte)((value8 >> 1) | ((P & 1) << 7)); FlagC = (temp8 & 1) != 0; P = (byte)((P & 0x7D) | TableNZ[value8]); - } void ZP_RMW_ROL() { @@ -2172,7 +2194,6 @@ namespace BizHawk.Emulation.Cores.Components.M6502 alu_temp = value8 = (byte)((value8 << 1) | (P & 1)); FlagC = (temp8 & 0x80) != 0; P = (byte)((P & 0x7D) | TableNZ[value8]); - } void ZP_RMW_SLO() { @@ -2198,7 +2219,6 @@ namespace BizHawk.Emulation.Cores.Components.M6502 FlagC = (temp8 & 0x80) != 0; A &= value8; NZ_A(); - } void AbsIdx_Stage3_Y() { From 3ce0264fd3a5e60aa9961a028141c68a79746375 Mon Sep 17 00:00:00 2001 From: Tony Konzel Date: Wed, 9 Mar 2016 20:35:56 -0600 Subject: [PATCH 15/15] C64: Actually use the correct data and length when building PRGs onto disk images. --- .../Computers/Commodore64/Media/DiskBuilder.cs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs index cd0f5d2a62..b987b44ff8 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/DiskBuilder.cs @@ -200,6 +200,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { Array.Copy(entry.Data, sourceOffset, bytes, outputOffset + 2, 254); dataRemaining -= 254; + sourceOffset += 254; } else { @@ -207,7 +208,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { Array.Copy(entry.Data, sourceOffset, bytes, outputOffset + 2, dataRemaining); bytes[outputOffset + 0] = 0; - bytes[outputOffset + 1] = (byte) dataRemaining; + bytes[outputOffset + 1] = (byte)(dataRemaining + 1); dataRemaining = 0; } }