From cb48104d7af9cb0e3441f34474d6841c50dfa320 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Wed, 3 Jul 2019 23:49:55 -0500 Subject: [PATCH 01/53] 6502X: Fix ADC with decimal mode enabled. --- .../CPUs/MOS 6502X/Execute.cs | 46 +++++++++---------- 1 file changed, 22 insertions(+), 24 deletions(-) diff --git a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs index 5d9282cae4..4d9a8edfad 100644 --- a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs +++ b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs @@ -1659,34 +1659,32 @@ namespace BizHawk.Emulation.Cores.Components.M6502 } void _Adc() { + //TODO - an extra cycle penalty on 65C02 only + value8 = (byte)alu_temp; + if (FlagD && BCD_Enabled) { - //TODO - an extra cycle penalty? - value8 = (byte)alu_temp; - if (FlagD && BCD_Enabled) - { - lo = (A & 0x0F) + (value8 & 0x0F) + (FlagC ? 1 : 0); - hi = (A & 0xF0) + (value8 & 0xF0); - if (lo > 0x09) - { - hi += 0x10; - lo += 0x06; - } - if (hi > 0x90) hi += 0x60; - FlagV = (~(A ^ value8) & (A ^ hi) & 0x80) != 0; - FlagC = hi > 0xFF; - A = (byte)((lo & 0x0F) | (hi & 0xF0)); - } - else - { - tempint = value8 + A + (FlagC ? 1 : 0); - FlagV = (~(A ^ value8) & (A ^ tempint) & 0x80) != 0; - FlagC = tempint > 0xFF; - A = (byte)tempint; - } + tempint = (A & 0x0F) + (value8 & 0x0F) + (FlagC ? 0x01 : 0x00); + if (tempint > 0x09) + tempint += 0x06; + tempint = (tempint & 0x0F) + (A & 0xF0) + (value8 & 0xF0) + (tempint > 0x0F ? 0x10 : 0x00); + FlagV = (~(A ^ value8) & (A ^ tempint) & 0x80) != 0; + FlagZ = ((A + value8 + (FlagC ? 1 : 0)) & 0xFF) == 0; + FlagN = (tempint & 0x80) != 0; + if ((tempint & 0x1F0) > 0x090) + tempint += 0x060; + FlagC = tempint > 0xFF; + A = (byte)(tempint & 0xFF); + } + else + { + tempint = value8 + A + (FlagC ? 1 : 0); + FlagV = (~(A ^ value8) & (A ^ tempint) & 0x80) != 0; + FlagC = tempint > 0xFF; + A = (byte)tempint; NZ_A(); } - } + void Unsupported() { From e74dfe15a8ea3c773a65ddc8ecd18210ae21009b Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:00:05 -0500 Subject: [PATCH 02/53] C64: VIA PB7 output timing adjusted to match datasheet. --- .../Commodore64/C64.MotherboardInterface.cs | 35 ++----------------- .../Commodore64/MOS/Via.Registers.cs | 1 + .../Computers/Commodore64/MOS/Via.cs | 35 +++++++++++++++++-- 3 files changed, 35 insertions(+), 36 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs index 37b606fb74..3c67940b25 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs @@ -18,43 +18,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 return (Cpu.PortData & 0x20) != 0; } - /* - private bool Cia0_ReadCnt() - { - return User.ReadCounter1() && Cia0.ReadCntBuffer(); - } - - private int Cia0_ReadPortA() - { - return cia0InputLatchA; - } - - private int Cia0_ReadPortB() - { - return cia0InputLatchB; - } - - private bool Cia0_ReadSP() - { - return User.ReadSerial1() && Cia0.ReadSpBuffer(); - } - - private bool Cia1_ReadSP() - { - return User.ReadSerial2() && Cia1.ReadSpBuffer(); - } - - private bool Cia1_ReadCnt() - { - return User.ReadCounter2() && Cia1.ReadCntBuffer(); - } - */ - private int Cia1_ReadPortA() { // the low bits are actually the VIC memory address. return (SerPort_ReadDataOut() && Serial.ReadDeviceData() ? 0x80 : 0x00) | - (SerPort_ReadClockOut() && Serial.ReadDeviceClock() ? 0x40 : 0x00); + (SerPort_ReadClockOut() && Serial.ReadDeviceClock() ? 0x40 : 0x00) | + 0x3F; } private int Cia1_ReadPortB() diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs index aa76b80b65..8e354c89e3 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs @@ -124,6 +124,7 @@ _t1C = _t1L; _t1CLoaded = true; _t1Delayed = 1; + _resetPb7NextClock = _acrT1Control == ACR_T1_CONTROL_INTERRUPT_ON_LOAD_AND_PULSE_PB7; break; case 0x7: _t1L = (_t1L & 0xFF) | ((val & 0xFF) << 8); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs index f7faccb431..a2729101e2 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs @@ -29,6 +29,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS 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; + private const int ACR_T1_CONTROL_INTERRUPT_ON_LOAD_AND_PULSE_PB7 = 0x80; private int _pra; private int _ddra; @@ -66,6 +67,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private bool _resetCa2NextClock; private bool _resetCb2NextClock; + private bool _resetPb7NextClock; + private bool _setPb7NextClock; private bool _handshakeCa2NextClock; private bool _handshakeCb2NextClock; @@ -142,6 +145,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _interruptNextClock = 0; _t1CLoaded = false; _t2CLoaded = false; + _resetPb7NextClock = false; + _setPb7NextClock = false; } public void ExecutePhase() @@ -150,7 +155,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _ifr |= _interruptNextClock; _interruptNextClock = 0; - // Process 'pulse' and 'handshake' outputs on CA2 and CB2 + // Process 'pulse' and 'handshake' outputs on PB7, CA2 and CB2 if (_resetCa2NextClock) { Ca2 = true; @@ -174,6 +179,17 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _resetCb2NextClock = _pcrCb2Control == PCR_CONTROL_PULSE_OUTPUT; _handshakeCb2NextClock = false; } + + if (_resetPb7NextClock) + { + _prb &= 0x7F; + _resetPb7NextClock = false; + } + else if (_setPb7NextClock) + { + _prb |= 0x80; + _setPb7NextClock = false; + } // Count timers if (_t1Delayed > 0) @@ -183,7 +199,19 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS else { _t1C--; - if (_t1C < 0) + if (_t1C == 0) + { + switch (_acrT1Control) + { + case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7: + _prb ^= 0x80; + break; + case ACR_T1_CONTROL_INTERRUPT_ON_LOAD_AND_PULSE_PB7: + _prb |= 0x80; + break; + } + } + else if (_t1C < 0) { if (_t1CLoaded) { @@ -199,7 +227,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS break; case ACR_T1_CONTROL_CONTINUOUS_INTERRUPTS_AND_OUTPUT_ON_PB7: _t1C = _t1L; - _prb ^= 0x80; _t1CLoaded = true; break; } @@ -399,6 +426,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync("T2Loaded", ref _t2CLoaded); ser.Sync("T1Delayed", ref _t1Delayed); ser.Sync("T2Delayed", ref _t2Delayed); + ser.Sync("ResetPb7NextClock", ref _resetPb7NextClock); + ser.Sync("SetPb7NextClock", ref _setPb7NextClock); } } } From ceb13384598274f0cfd389fdc670a14ddbdce3a2 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:02:28 -0500 Subject: [PATCH 03/53] C64: Use proper sector gaps based on density when converting from D64. --- .../Computers/Commodore64/Media/D64.cs | 12 ++- .../Computers/Commodore64/Media/G64.cs | 75 +++++++++++++++++++ 2 files changed, 84 insertions(+), 3 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs index 87d0a60668..c84a505e5e 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs @@ -1,6 +1,7 @@ using System; using System.Collections.Generic; using System.IO; +using System.Linq; namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { @@ -63,6 +64,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media 6250, 6666, 7142, 7692 }; + private static readonly int[] StandardSectorGapLength = + { + 9, 19, 13, 10 + }; + private static byte Checksum(byte[] source) { var count = source.Length; @@ -76,7 +82,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media return result; } - private static byte[] ConvertSectorToGcr(byte[] source, byte sectorNo, byte trackNo, byte formatA, byte formatB, out int bitsWritten) + private static byte[] ConvertSectorToGcr(byte[] source, byte sectorNo, byte trackNo, byte formatA, byte formatB, int gapLength, out int bitsWritten) { using (var mem = new MemoryStream()) { @@ -96,7 +102,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media 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(Enumerable.Repeat((byte)0x55, gapLength).ToArray()); // gap bitsWritten = (int)mem.Length * 8; @@ -185,7 +191,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { int bitsWritten; var sectorData = reader.ReadBytes(256); - var diskData = ConvertSectorToGcr(sectorData, (byte)j, (byte)(i + 1), 0xA0, 0xA0, out bitsWritten); + var diskData = ConvertSectorToGcr(sectorData, (byte)j, (byte)(i + 1), 0xA0, 0xA0, StandardSectorGapLength[DensityTable[i]], out bitsWritten); trackMem.Write(diskData, 0, diskData.Length); trackLengthBits += bitsWritten; } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs index fc54893e89..bceb344add 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs @@ -63,5 +63,80 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media return new Disk(84); } } + + public static byte[] Write(IList trackData, IList trackNumbers, IList trackDensities) + { + const byte version = 0; + const byte trackCount = 84; + const int headerLength = 0xC; + const byte dataFillerValue = 0xFF; + + var trackMaxLength = (ushort)Math.Max(7928, trackData.Max(d => d.Length)); + + using (var mem = new MemoryStream()) + { + var writer = new BinaryWriter(mem); + + // header ID + writer.Write("GCR-1541".ToCharArray()); + + // version # + writer.Write(version); + + // tracks in the image + writer.Write(trackCount); + + // maximum track size in bytes + writer.Write(trackMaxLength); + + // combine track data + var offsets = new List(); + var densities = new List(); + using (var trackMem = new MemoryStream()) + { + var trackMemWriter = new BinaryWriter(trackMem); + for (var i = 0; i < trackCount; i++) + { + if (trackNumbers.Contains(i)) + { + var trackIndex = trackNumbers.IndexOf(i); + offsets.Add((int)trackMem.Length); + densities.Add(trackDensities[trackIndex]); + + var data = trackData[trackIndex]; + var buffer = Enumerable.Repeat(dataFillerValue, trackMaxLength).ToArray(); + var dataBytes = data.Select(d => unchecked((byte)d)).ToArray(); + Array.Copy(dataBytes, buffer, dataBytes.Length); + trackMemWriter.Write((ushort)dataBytes.Length); + trackMemWriter.Write(buffer); + } + else + { + offsets.Add(-1); + densities.Add(0); + } + } + trackMemWriter.Flush(); + + // offset table + foreach (var offset in offsets.Select(o => o >= 0 ? o + headerLength + trackCount * 8 : 0)) + { + writer.Write(offset); + } + + // speed zone data + foreach (var density in densities) + { + writer.Write(density); + } + + // track data + writer.Write(trackMem.ToArray()); + } + + writer.Flush(); + return mem.ToArray(); + } + } } } From 0cdb28fc8f5150dc03ec69de2ec1a51092cacf0e Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:03:50 -0500 Subject: [PATCH 04/53] C64: Format D64 sector headers with directory ID instead of A0/A0. --- BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs index c84a505e5e..fd58006c5d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs @@ -7,6 +7,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { public static class D64 { + const int D64_DISK_ID_OFFSET = 0x165A2; // track 18, sector 0, 0xA2 + private static readonly int[] DensityTable = { 3, 3, 3, 3, 3, @@ -154,6 +156,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media public static Disk Read(byte[] source) { + var formatB = source[D64_DISK_ID_OFFSET + 0x00]; + var formatA = source[D64_DISK_ID_OFFSET + 0x01]; + using (var mem = new MemoryStream(source)) { var reader = new BinaryReader(mem); @@ -191,7 +196,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { int bitsWritten; var sectorData = reader.ReadBytes(256); - var diskData = ConvertSectorToGcr(sectorData, (byte)j, (byte)(i + 1), 0xA0, 0xA0, StandardSectorGapLength[DensityTable[i]], out bitsWritten); + var diskData = ConvertSectorToGcr(sectorData, (byte)j, (byte)(i + 1), formatA, formatB, StandardSectorGapLength[DensityTable[i]], out bitsWritten); trackMem.Write(diskData, 0, diskData.Length); trackLengthBits += bitsWritten; } From 4e1892d0947a4b30b7d5bb44849e6c80fa70e1b1 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:11:03 -0500 Subject: [PATCH 05/53] C64: Allow writing to disk. --- .../Computers/Commodore64/MOS/Via.cs | 80 +++--- .../Computers/Commodore64/Media/D64.cs | 162 ++++++----- .../Serial/Drive1541.FluxTransitions.cs | 259 ++++++++++-------- .../Commodore64/Serial/Drive1541.Registers.cs | 2 +- .../Computers/Commodore64/Serial/Drive1541.cs | 5 + 5 files changed, 304 insertions(+), 204 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs index a2729101e2..147449bf9c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs @@ -129,15 +129,15 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _acrSrControl = 0; _acrT1Control = 0; _acrT2Control = 0; - _ca1L = false; - _cb1L = false; - Ca1 = false; - Ca2 = false; - Cb1 = false; - Cb2 = false; + _ca1L = true; + _cb1L = true; + Ca1 = true; + Ca2 = true; + Cb1 = true; + Cb2 = true; - _pb6L = false; - _pb6 = false; + _pb6L = true; + _pb6 = true; _resetCa2NextClock = false; _resetCb2NextClock = false; _handshakeCa2NextClock = false; @@ -222,9 +222,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS 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; _t1CLoaded = true; @@ -261,11 +258,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS if (!_pb6 && _pb6L) { _t2C--; - if (_t2C < 0) + if (_t2C == 0) { _ifr |= 0x20; - _t2C = 0xFFFF; } + _t2C &= 0xFFFF; } break; } @@ -331,26 +328,45 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS break; } - // interrupt generation - if ((_pcrCb1IntControl == PCR_INT_CONTROL_POSITIVE_EDGE && Cb1 && !_cb1L) || - (_pcrCb1IntControl == PCR_INT_CONTROL_NEGATIVE_EDGE && !Cb1 && _cb1L)) - { - _ifr |= 0x10; - if (_acrPbLatchEnable) - { - _pbLatch = _port.ReadExternalPrb(); - } - } + // interrupt generation - if ((_pcrCa1IntControl == PCR_INT_CONTROL_POSITIVE_EDGE && Ca1 && !_ca1L) || - (_pcrCa1IntControl == PCR_INT_CONTROL_NEGATIVE_EDGE && !Ca1 && _ca1L)) - { - _ifr |= 0x02; - if (_acrPaLatchEnable) - { - _paLatch = _port.ReadExternalPra(); - } - } + /* + As long as the CA1 interrupt flag is set, the data on the peripheral pins can change + without affecting the data in the latches. This input latching can be used with any of the CA2 + input or output modes. + It is important to note that on the PA port, the processor always reads the data on the + peripheral pins (as reflected in the latches). For output pins, the processor still reads the + latches. This may or may not reflect the data currently in the ORA. Proper system operation + requires careful planning on the part of the system designer if input latching is combined + with output pins on the peripheral ports. + */ + + if ((_pcrCa1IntControl == PCR_INT_CONTROL_POSITIVE_EDGE && Ca1 && !_ca1L) || + (_pcrCa1IntControl == PCR_INT_CONTROL_NEGATIVE_EDGE && !Ca1 && _ca1L)) + { + if (_acrPaLatchEnable && (_ifr & 0x02) == 0) + { + _paLatch = _port.ReadExternalPra(); + } + _ifr |= 0x02; + } + + /* + Input latching on the PB port is controlled in the same manner as that described for the PA port. + However, with the peripheral B port the input latch will store either the voltage on the pin or the contents + of the Output Register (ORB) depending on whether the pin is programmed to act as an input or an + output. As with the PA port, the processor always reads the input latches. + */ + + if ((_pcrCb1IntControl == PCR_INT_CONTROL_POSITIVE_EDGE && Cb1 && !_cb1L) || + (_pcrCb1IntControl == PCR_INT_CONTROL_NEGATIVE_EDGE && !Cb1 && _cb1L)) + { + if (_acrPbLatchEnable && (_ifr & 0x10) == 0) + { + _pbLatch = _port.ReadPrb(_prb, _ddrb); + } + _ifr |= 0x10; + } switch (_acrSrControl) { diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs index fd58006c5d..082fc32b0a 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs @@ -9,6 +9,22 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media { const int D64_DISK_ID_OFFSET = 0x165A2; // track 18, sector 0, 0xA2 + private enum ErrorType + { + NoError = 0x01, + HeaderNotFound = 0x02, + NoSyncSequence = 0x03, + DataNotFound = 0x04, + DataChecksumError = 0x05, + WriteVerifyFormatError = 0x06, + WriteVerifyError = 0x07, + WriteProtectOn = 0x08, + HeaderChecksumError = 0x09, + WriteError = 0x0A, + IdMismatch = 0x0B, + DriveNotReady = 0x0F + } + private static readonly int[] DensityTable = { 3, 3, 3, 3, 3, @@ -84,25 +100,34 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media return result; } - private static byte[] ConvertSectorToGcr(byte[] source, byte sectorNo, byte trackNo, byte formatA, byte formatB, int gapLength, out int bitsWritten) + private static byte[] ConvertSectorToGcr(byte[] source, byte sectorNo, byte trackNo, byte formatA, byte formatB, int gapLength, ErrorType errorType, out int bitsWritten) { using (var mem = new MemoryStream()) { var writer = new BinaryWriter(mem); - var headerChecksum = (byte)(sectorNo ^ trackNo ^ formatA ^ formatB); + + if (errorType == ErrorType.IdMismatch) + { + formatA ^= 0xFF; + formatB ^= 0xFF; + } + + var headerChecksum = (byte)(sectorNo ^ trackNo ^ formatA ^ formatB ^ (errorType == ErrorType.HeaderChecksumError ? 0xFF : 0x00)); // assemble written data for GCR encoding var writtenData = new byte[260]; + var syncBytes40 = Enumerable.Repeat((byte) (errorType == ErrorType.NoSyncSequence ? 0x00 : 0xFF), 5).ToArray(); + Array.Copy(source, 0, writtenData, 1, 256); - writtenData[0] = 0x07; - writtenData[0x101] = Checksum(source); + writtenData[0] = (byte)(errorType == ErrorType.HeaderNotFound ? 0x00 : 0x07); + writtenData[0x101] = (byte)(Checksum(source) ^ (errorType == ErrorType.DataChecksumError ? 0xFF : 0x00)); 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(syncBytes40); // sync + writer.Write(EncodeGcr(new byte[] { (byte)(errorType == ErrorType.DataNotFound ? 0x00 : 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(syncBytes40); // sync writer.Write(EncodeGcr(writtenData)); // data writer.Write(Enumerable.Repeat((byte)0x55, gapLength).ToArray()); // gap @@ -156,67 +181,76 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media public static Disk Read(byte[] source) { - var formatB = source[D64_DISK_ID_OFFSET + 0x00]; - var formatA = source[D64_DISK_ID_OFFSET + 0x01]; + var formatB = source[D64_DISK_ID_OFFSET + 0x00]; + var formatA = source[D64_DISK_ID_OFFSET + 0x01]; - 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; + 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(); + var errorType = ErrorType.NoError; + int trackCount; + int errorOffset = -1; - 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."); - } + switch (source.Length) + { + case 174848: // 35 tracks no errors + trackCount = 35; + break; + case 175531: // 35 tracks with errors + trackCount = 35; + errorOffset = 174848; + break; + case 196608: // 40 tracks no errors + trackCount = 40; + break; + case 197376: // 40 tracks with errors + trackCount = 40; + errorOffset = 196608; + 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), formatA, formatB, StandardSectorGapLength[DensityTable[i]], out bitsWritten); - trackMem.Write(diskData, 0, diskData.Length); - trackLengthBits += bitsWritten; - } - var density = DensityTable[i]; + for (var i = 0; i < trackCount; i++) + { + if (errorOffset >= 0) + { + errorType = (ErrorType) source[errorOffset]; + errorOffset++; + } + 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), formatA, formatB, StandardSectorGapLength[DensityTable[i]], errorType, 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); - } + // 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, 84); - } - } + trackDatas.Add(trackMem.ToArray()); + trackLengths.Add(trackLengthBits); + trackNumbers.Add(i * 2); + trackDensities.Add(DensityTable[i]); + } + } + + return new Disk(trackDatas, trackNumbers, trackDensities, 84); + } + } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs index 1252524211..2cdd555caf 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs @@ -19,6 +19,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial private int _rngCurrent; private int _clocks; private int _cpuClocks; + private int _diskWriteBitsRemaining; + private bool _diskWriteEnabled; + private int _diskWriteLatch; + private int _diskOutputBits; + private bool _diskWriteProtected; // Lehmer RNG private void AdvanceRng() @@ -31,127 +36,167 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial _rngCurrent = (int)(_rngCurrent * LEHMER_RNG_PRIME % int.MaxValue); } - private void ExecuteFlux() - { - // This actually executes the main 16mhz clock - while (_clocks > 0) - { - _clocks--; + private void ExecuteFlux() + { + // This actually executes the main 16mhz clock + while (_clocks > 0) + { + _clocks--; - // rotate disk - if (_motorEnabled) - { - if (_disk == null) - { - _diskBitsLeft = 1; - _diskBits = 0; - } - else - { - if (_diskBitsLeft <= 0) - { - _diskByteOffset++; - if (_diskByteOffset == Disk.FluxEntriesPerTrack) - { - _diskByteOffset = 0; - } + // rotate disk + if (_motorEnabled) + { + if (_disk == null) + { + _diskBitsLeft = 1; + _diskBits = 0; + } + else + { + if (_diskBitsLeft <= 0) + { + if (_diskWriteEnabled) + _trackImageData[_diskByteOffset] = _diskOutputBits; - _diskBits = _trackImageData[_diskByteOffset]; - _diskBitsLeft = Disk.FluxBitsPerEntry; - } - } + _diskByteOffset++; - if ((_diskBits & 1) != 0) - { - _countsBeforeRandomTransition = 0; - _diskFluxReversalDetected = true; - } + if (_diskByteOffset == Disk.FluxEntriesPerTrack) + _diskByteOffset = 0; - _diskBits >>= 1; - _diskBitsLeft--; - } + if (!_diskWriteEnabled) + _diskBits = _trackImageData[_diskByteOffset]; - // random flux transition readings for unformatted data - if (_countsBeforeRandomTransition > 0) - { - _countsBeforeRandomTransition--; - if (_countsBeforeRandomTransition == 0) - { - _diskFluxReversalDetected = true; - AdvanceRng(); + _diskOutputBits = 0; + _diskBitsLeft = Disk.FluxBitsPerEntry; + } + } + _diskOutputBits >>= 1; - // This constant is what VICE uses. TODO: Determine accuracy. - _countsBeforeRandomTransition = (_rngCurrent % 367) + 33; - } - } + if (_diskWriteEnabled) + _countsBeforeRandomTransition = 0; - // flux transition circuitry - if (_diskFluxReversalDetected) - { - _diskDensityCounter = _diskDensity; - _diskSupplementaryCounter = 0; - _diskFluxReversalDetected = false; - if (_countsBeforeRandomTransition == 0) - { - AdvanceRng(); + if ((_diskBits & 1) != 0) + { + _countsBeforeRandomTransition = 0; + _diskFluxReversalDetected = true; + _diskOutputBits |= int.MinValue; // set bit 31 + } + else + { + _diskOutputBits &= int.MaxValue; // clear bit 31 + } - // This constant is what VICE uses. TODO: Determine accuracy. - _countsBeforeRandomTransition = (_rngCurrent & 0x1F) + 289; - } - } + _diskBits >>= 1; + _diskBitsLeft--; + } - // counter circuitry - if (_diskDensityCounter >= 16) - { - _diskDensityCounter = _diskDensity; - _diskSupplementaryCounter++; - if ((_diskSupplementaryCounter & 0x3) == 0x2) - { - _bitsRemainingInLatchedByte--; - _byteReady = false; - _bitHistory = (_bitHistory << 1) | ((_diskSupplementaryCounter & 0xC) == 0x0 ? 1 : 0); - _sync = false; - if (Via1.Cb2 && (_bitHistory & 0x3FF) == 0x3FF) - { - _sync = true; - _bitsRemainingInLatchedByte = 8; - _byteReady = false; - } + // random flux transition readings for unformatted data + if (_countsBeforeRandomTransition > 0) + { + _countsBeforeRandomTransition--; + if (_countsBeforeRandomTransition == 0) + { + _diskFluxReversalDetected = true; + AdvanceRng(); + // This constant is what VICE uses. TODO: Determine accuracy. + _countsBeforeRandomTransition = (_rngCurrent % 367) + 33; + } + } - if (_bitsRemainingInLatchedByte <= 0) - { - _bitsRemainingInLatchedByte = 8; + // flux transition circuitry + if (_diskFluxReversalDetected) + { + if (!_diskWriteEnabled) + { + _diskDensityCounter = _diskDensity; + _diskSupplementaryCounter = 0; + } + _diskFluxReversalDetected = false; + if (_countsBeforeRandomTransition == 0) + { + AdvanceRng(); + // This constant is what VICE uses. TODO: Determine accuracy. + _countsBeforeRandomTransition = (_rngCurrent & 0x1F) + 289; + } + } - // SOE (sync output enabled) - _byteReady = Via1.Ca2; - } + // counter circuitry + if (_diskDensityCounter >= 16) + { + _diskDensityCounter = _diskDensity; + _diskSupplementaryCounter++; - // negative transition activates SO pin on CPU - _previousCa1 = Via1.Ca1; - Via1.Ca1 = !_byteReady; - if (_previousCa1 && !Via1.Ca1) - { - // cycle 6 is roughly 400ns - _overflowFlagDelaySr |= _diskCycle > 6 ? 4 : 2; - } - } - } + if ((_diskSupplementaryCounter & 0x3) == 0x2) + { + if (!_diskWriteEnabled) + _diskWriteBitsRemaining = 0; + _diskWriteEnabled = !Via1.Cb2; - if (_diskSupplementaryCounter >= 16) - { - _diskSupplementaryCounter = 0; - } + _diskWriteBitsRemaining--; + if (_diskWriteEnabled) + { + _countsBeforeRandomTransition = 0; + _byteReady = false; + if (_diskWriteBitsRemaining <= 0) + { + _diskWriteLatch = Via1.EffectivePrA; + _diskWriteBitsRemaining = 8; + _byteReady = Via1.Ca2; + } + if ((_diskWriteLatch & 0x80) != 0) + { + _diskOutputBits |= int.MinValue; // set bit 31 + } + _diskWriteLatch <<= 1; + } + else + { + _bitsRemainingInLatchedByte--; + _byteReady = false; + _bitHistory = (_bitHistory << 1) | ((_diskSupplementaryCounter & 0xC) == 0x0 ? 1 : 0); + _sync = false; + if (!_diskWriteEnabled && (_bitHistory & 0x3FF) == 0x3FF) + { + _sync = true; + _bitsRemainingInLatchedByte = 8; + _byteReady = false; + } - _cpuClocks--; - if (_cpuClocks <= 0) - { - ExecuteSystem(); - _cpuClocks = 16; - } + if (_bitsRemainingInLatchedByte <= 0) + { + _bitsRemainingInLatchedByte = 8; - _diskDensityCounter++; - _diskCycle = (_diskCycle + 1) & 0xF; - } - } + // SOE (SO/Byte Ready enabled) + _byteReady = Via1.Ca2; + } + } + } + + // negative transition activates SO pin on CPU + _previousCa1 = Via1.Ca1; + Via1.Ca1 = !_byteReady; + if (_previousCa1 && !Via1.Ca1) + { + // cycle 6 is roughly 400ns + _overflowFlagDelaySr |= _diskCycle > 6 ? 4 : 2; + } + } + + if (_diskSupplementaryCounter >= 16) + { + _diskSupplementaryCounter = 0; + } + + _cpuClocks--; + if (_cpuClocks <= 0) + { + ExecuteSystem(); + _cpuClocks = 16; + } + + _diskDensityCounter++; + _diskCycle = (_diskCycle + 1) & 0xF; + } + } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs index 84e54cef51..02257c2253 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs @@ -46,7 +46,7 @@ private int ReadVia1PrB() { - return (_motorStep & 0x03) | (_motorEnabled ? 0x04 : 0x00) | (_sync ? 0x00 : 0x80); + return (_motorStep & 0x03) | (_motorEnabled ? 0x04 : 0x00) | (_sync ? 0x00 : 0x80) | (_diskWriteProtected ? 0x00 : 0x10); } public int Peek(int addr) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs index d6556af21e..afe069f236 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -117,6 +117,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial ser.Sync("Clocks", ref _clocks); ser.Sync("CpuClocks", ref _cpuClocks); ser.Sync("OverflowFlagDelayShiftRegister", ref _overflowFlagDelaySr); + ser.Sync("DiskWriteBitsRemaining", ref _diskWriteBitsRemaining); + ser.Sync("DiskWriteEnabled", ref _diskWriteEnabled); + ser.Sync("DiskWriteLatch", ref _diskWriteLatch); + ser.Sync("DiskOutputBits", ref _diskOutputBits); + ser.Sync("DiskWriteProtected", ref _diskWriteProtected); } public override void ExecutePhase() From 7fbccb7a464ee7eddb7fe9d00ab2fdad81b4df4c Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:14:21 -0500 Subject: [PATCH 06/53] C64: Use write protection on G64 images (which are often copy protected), and disable it on D64 images. --- .../Computers/Commodore64/MOS/Via.Registers.cs | 6 +++--- BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs | 6 +++--- BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs | 4 ++-- .../Commodore64/Serial/Drive1541.FluxTransitions.cs | 2 +- .../Computers/Commodore64/Serial/Drive1541.cs | 5 +++++ 5 files changed, 14 insertions(+), 9 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs index 8e354c89e3..1215121257 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs @@ -58,7 +58,7 @@ case 0x0: return _port.ReadPrb(_prb, _ddrb); case 0x1: - return _port.ReadPra(_pra, _ddra); + return _port.ReadExternalPra(); case 0x2: return _ddrb; case 0x3: @@ -86,7 +86,7 @@ case 0xE: return _ier | 0x80; case 0xF: - return _port.ReadPra(_pra, _ddra); + return _port.ReadExternalPra(); } return 0xFF; @@ -238,7 +238,7 @@ public int EffectivePrB => _prb | ~_ddrb; - public int ActualPrA => _acrPaLatchEnable ? _paLatch : _port.ReadPra(_pra, _ddra); + public int ActualPrA => _acrPaLatchEnable ? _paLatch : _port.ReadExternalPra(); public int ActualPrB => _acrPbLatchEnable ? _pbLatch : _port.ReadPrb(_prb, _ddrb); } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs index 082fc32b0a..f086bb3b21 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/D64.cs @@ -248,9 +248,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media trackDensities.Add(DensityTable[i]); } } - - return new Disk(trackDatas, trackNumbers, trackDensities, 84); - } + + return new Disk(trackDatas, trackNumbers, trackDensities, 84) {WriteProtected = false}; + } } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs index bceb344add..74f3fd2045 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/G64.cs @@ -57,10 +57,10 @@ 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, 84); + return new Disk(trackDatas, trackNumbers, trackDensities, 84) {WriteProtected = true}; } - return new Disk(84); + return new Disk(84) {WriteProtected = false}; } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs index 2cdd555caf..0f9a3782c5 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs @@ -72,7 +72,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial } _diskOutputBits >>= 1; - if (_diskWriteEnabled) + if (_diskWriteEnabled && !_diskWriteProtected) _countsBeforeRandomTransition = 0; 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 afe069f236..50fb387ecb 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.cs @@ -214,6 +214,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial { _trackImageData = _disk.GetDataForTrack(_trackNumber); _diskBits = _trackImageData[_diskByteOffset] >> (Disk.FluxBitsPerEntry - _diskBitsLeft); + _diskWriteProtected = _disk.WriteProtected; + } + else + { + _diskWriteProtected = true; } } From 32d59e8514d98630a16fd373b1a874aa458e22f2 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:23:11 -0500 Subject: [PATCH 07/53] C64: Implement more VIA features. --- .../Computers/Commodore64/MOS/Via.Port.cs | 4 +- .../Commodore64/MOS/Via.Registers.cs | 32 +++++------ .../Computers/Commodore64/MOS/Via.cs | 57 ++++++++++--------- 3 files changed, 45 insertions(+), 48 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Port.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Port.cs index 173314dfb5..14d89e0c43 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Port.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Port.cs @@ -56,7 +56,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public int ReadPra(int pra, int ddra) { - return _readPrA(); + return (pra | ~ddra) & ReadExternalPra(); } public int ReadPrb(int prb, int ddrb) @@ -98,7 +98,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public int ReadPra(int pra, int ddra) { - return (pra | ~ddra) & 0xFF; + return (pra | ~ddra) & ReadExternalPra(); } public int ReadPrb(int prb, int ddrb) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs index 1215121257..a5b122724b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.Registers.cs @@ -18,18 +18,16 @@ switch (addr) { case 0x0: - _ifr &= 0xE7; + if (_pcrCb2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE && _pcrCb2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE) + _ifr &= 0xE7; if (_acrPbLatchEnable) - { return _pbLatch; - } break; case 0x1: - _ifr &= 0xFC; + if (_pcrCa2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE && _pcrCa2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE) + _ifr &= 0xFC; if (_acrPaLatchEnable) - { return _paLatch; - } break; case 0x4: _ifr &= 0xBF; @@ -39,6 +37,7 @@ break; case 0xA: _ifr &= 0xFB; + _srCount = 8; break; case 0xF: if (_acrPaLatchEnable) @@ -58,6 +57,7 @@ case 0x0: return _port.ReadPrb(_prb, _ddrb); case 0x1: + case 0xF: return _port.ReadExternalPra(); case 0x2: return _ddrb; @@ -85,8 +85,6 @@ return _ifr; case 0xE: return _ier | 0x80; - case 0xF: - return _port.ReadExternalPra(); } return 0xFF; @@ -98,20 +96,17 @@ switch (addr) { case 0x0: - _ifr &= 0xE7; - if (_pcrCb2Control == PCR_CONTROL_HANDSHAKE_OUTPUT || _pcrCb2Control == PCR_CONTROL_PULSE_OUTPUT) - { + if (_pcrCb2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE && _pcrCb2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE) + _ifr &= 0xE7; + if (_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) - { + if (_pcrCa2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_NEGATIVE_EDGE && _pcrCa2Control != PCR_CONTROL_INDEPENDENT_INTERRUPT_INPUT_POSITIVE_EDGE) + _ifr &= 0xFC; + if (_pcrCa2Control == PCR_CONTROL_PULSE_OUTPUT) _handshakeCa2NextClock = true; - } - WriteRegister(addr, val); break; case 0x4: @@ -146,6 +141,7 @@ break; case 0xA: _ifr &= 0xFB; + _srCount = 8; WriteRegister(addr, val); break; case 0xD: @@ -238,7 +234,7 @@ public int EffectivePrB => _prb | ~_ddrb; - public int ActualPrA => _acrPaLatchEnable ? _paLatch : _port.ReadExternalPra(); + public int ActualPrA => _acrPaLatchEnable ? _paLatch : _port.ReadPra(_pra, _ddra); public int ActualPrB => _acrPbLatchEnable ? _pbLatch : _port.ReadPrb(_prb, _ddrb); } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs index 147449bf9c..9c3ed0ed3d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs @@ -58,6 +58,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _acrSrControl; private int _acrT1Control; private int _acrT2Control; + private int _srCount; private bool _ca1L; private bool _ca2L; @@ -135,6 +136,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS Ca2 = true; Cb1 = true; Cb2 = true; + _srCount = 0; _pb6L = true; _pb6 = true; @@ -151,6 +153,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public void ExecutePhase() { + var _shiftIn = false; + var _shiftOut = false; + // Process delayed interrupts _ifr |= _interruptNextClock; _interruptNextClock = 0; @@ -259,9 +264,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { _t2C--; if (_t2C == 0) - { _ifr |= 0x20; - } _t2C &= 0xFFFF; } break; @@ -330,24 +333,28 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // interrupt generation - /* - As long as the CA1 interrupt flag is set, the data on the peripheral pins can change - without affecting the data in the latches. This input latching can be used with any of the CA2 - input or output modes. - It is important to note that on the PA port, the processor always reads the data on the - peripheral pins (as reflected in the latches). For output pins, the processor still reads the - latches. This may or may not reflect the data currently in the ORA. Proper system operation - requires careful planning on the part of the system designer if input latching is combined - with output pins on the peripheral ports. - */ + if (_acrSrControl == ACR_SR_CONTROL_DISABLED) + { + _ifr &= 0xFB; + _srCount = 0; + } + + /* + As long as the CA1 interrupt flag is set, the data on the peripheral pins can change + without affecting the data in the latches. This input latching can be used with any of the CA2 + input or output modes. + It is important to note that on the PA port, the processor always reads the data on the + peripheral pins (as reflected in the latches). For output pins, the processor still reads the + latches. This may or may not reflect the data currently in the ORA. Proper system operation + requires careful planning on the part of the system designer if input latching is combined + with output pins on the peripheral ports. + */ if ((_pcrCa1IntControl == PCR_INT_CONTROL_POSITIVE_EDGE && Ca1 && !_ca1L) || (_pcrCa1IntControl == PCR_INT_CONTROL_NEGATIVE_EDGE && !Ca1 && _ca1L)) { if (_acrPaLatchEnable && (_ifr & 0x02) == 0) - { - _paLatch = _port.ReadExternalPra(); - } + _paLatch = _port.ReadExternalPra(); _ifr |= 0x02; } @@ -362,29 +369,22 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS (_pcrCb1IntControl == PCR_INT_CONTROL_NEGATIVE_EDGE && !Cb1 && _cb1L)) { if (_acrPbLatchEnable && (_ifr & 0x10) == 0) - { - _pbLatch = _port.ReadPrb(_prb, _ddrb); - } + _pbLatch = _port.ReadPrb(_prb, _ddrb); + if (_acrSrControl == ACR_SR_CONTROL_DISABLED) + _shiftIn = true; _ifr |= 0x10; } - switch (_acrSrControl) + if (_shiftIn) { - case ACR_SR_CONTROL_DISABLED: - _ifr &= 0xFB; - break; - default: - break; + _sr <<= 1; + _sr |= Cb2 ? 1 : 0; } if ((_ifr & _ier & 0x7F) != 0) - { _ifr |= 0x80; - } else - { _ifr &= 0x7F; - } _ca1L = Ca1; _ca2L = Ca2; @@ -444,6 +444,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync("T2Delayed", ref _t2Delayed); ser.Sync("ResetPb7NextClock", ref _resetPb7NextClock); ser.Sync("SetPb7NextClock", ref _setPb7NextClock); + ser.Sync("ShiftRegisterCount", ref _srCount); } } } From 2dd80eb0f4c42f5a87e9839d9d9c84457de20dc9 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:31:48 -0500 Subject: [PATCH 08/53] C64: Implement more CIA features and CIA/VIA defaults. --- .../Commodore64/C64.MotherboardInterface.cs | 6 +- .../Computers/Commodore64/MOS/Chip6526.cs | 10 +- .../Computers/Commodore64/MOS/Cia.Port.cs | 6 +- .../Commodore64/MOS/Cia.Registers.cs | 11 +- .../Computers/Commodore64/MOS/Cia.cs | 798 +++++++++--------- .../Computers/Commodore64/MOS/Via.cs | 10 +- .../Commodore64/Serial/Drive1541.Registers.cs | 3 + 7 files changed, 420 insertions(+), 424 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs index 3c67940b25..6ea70757ac 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs @@ -28,7 +28,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private int Cia1_ReadPortB() { - return 0xFF; + // Ordinarily these are connected to the userport. + return 0x00; } private int Cpu_ReadPort() @@ -111,16 +112,19 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private bool SerPort_ReadAtnOut() { + // inverted PA3 (input NOT pulled up) return !((Cia1.DdrA & 0x08) != 0 && (Cia1.PrA & 0x08) != 0); } private bool SerPort_ReadClockOut() { + // inverted PA4 (input NOT pulled up) return !((Cia1.DdrA & 0x10) != 0 && (Cia1.PrA & 0x10) != 0); } private bool SerPort_ReadDataOut() { + // inverted PA5 (input NOT pulled up) return !((Cia1.DdrA & 0x20) != 0 && (Cia1.PrA & 0x20) != 0); } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs index 43d8db3944..bddb02b6f1 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs @@ -9,27 +9,27 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // * A low RES pin is emulated via HardReset(). public static class Chip6526 { - public static Cia Create(C64.CiaType type, Func readIec) + public static Cia Create(C64.CiaType type, Func readIec, Func readUserPort) { switch (type) { case C64.CiaType.Ntsc: - return new Cia(14318181, 14 * 60, readIec) + return new Cia(14318181, 14 * 60, readIec, readUserPort) { DelayedInterrupts = true }; case C64.CiaType.NtscRevA: - return new Cia(14318181, 14 * 60, readIec) + return new Cia(14318181, 14 * 60, readIec, readUserPort) { DelayedInterrupts = false }; case C64.CiaType.Pal: - return new Cia(17734472, 18 * 50, readIec) + return new Cia(17734472, 18 * 50, readIec, readUserPort) { DelayedInterrupts = true }; case C64.CiaType.PalRevA: - return new Cia(17734472, 18 * 50, readIec) + return new Cia(17734472, 18 * 50, readIec, readUserPort) { DelayedInterrupts = false }; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Port.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Port.cs index 32cfba5d9a..f4af0f5146 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Port.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Port.cs @@ -131,10 +131,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private sealed class IecPort : IPort { private readonly Func _readIec; + private readonly Func _readUserPort; - public IecPort(Func readIec) + public IecPort(Func readIec, Func readUserPort) { _readIec = readIec; + _readUserPort = readUserPort; } public int ReadPra(int pra, int ddra, int prb, int ddrb) @@ -144,7 +146,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public int ReadPrb(int pra, int ddra, int prb, int ddrb) { - return (prb | ~ddrb) & 0xFF; + return (prb | ~ddrb) | (~ddrb & _readUserPort()); } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs index 0ba58e1ceb..af5ed166fc 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.Registers.cs @@ -252,15 +252,16 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS case 0xE: _hasNewCra = true; _newCra = val; - _taCntPhi2 = ((val & 0x20) == 0); - _taCntCnt = ((val & 0x20) == 0x20); + _taCntPhi2 = (val & 0x20) == 0; + _taCntCnt = (val & 0x20) == 0x20; break; case 0xF: _hasNewCrb = true; _newCrb = val; - _tbCntPhi2 = ((val & 0x60) == 0); - _tbCntTa = ((val & 0x40) == 0x40); - _tbCntCnt = ((val & 0x20) == 0x20); + _tbCntPhi2 = (val & 0x60) == 0; + _tbCntCnt = (val & 0x60) == 0x20; + _tbCntTa = (val & 0x60) == 0x40; + _tbCntTaCnt = (val & 0x60) == 0x60; break; } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs index 113c0fcfa6..92d193737b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs @@ -33,6 +33,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } public Func ReadFlag = () => true; + public Func ReadCnt = () => true; + public Func ReadSp = () => true; public bool DelayedInterrupts = true; private int _pra; @@ -79,6 +81,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private bool _flagLatch; private bool _flagInput; private bool _taUnderflow; + private bool _lastCnt; + private bool _thisCnt; + private bool _tbCntTaCnt; private readonly IPort _port; private int _todlo; @@ -98,438 +103,417 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _port = new JoystickKeyboardPort(joysticks, keyboard); } - public Cia(int todNum, int todDen, Func readIec) : this(todNum, todDen) + public Cia(int todNum, int todDen, Func readIec, Func readUserPort) : this(todNum, todDen) { - _port = new IecPort(readIec); + _port = new IecPort(readIec, readUserPort); } - public void HardReset() - { - _pra = 0; - _prb = 0; - _ddra = 0; - _ddrb = 0; - _ta = 0xFFFF; - _tb = 0xFFFF; - _latcha = 1; - _latchb = 1; - _tod10Ths = 0; - _todSec = 0; - _todMin = 0; - _todHr = 0; - _alm10Ths = 0; - _almSec = 0; - _almMin = 0; - _almHr = 0; - _sdr = 0; - _icr = 0; - _cra = 0; - _crb = 0; - _intMask = 0; - _todLatch = false; - _taCntPhi2 = false; - _taCntCnt = false; - _tbCntPhi2 = false; - _tbCntTa = false; - _tbCntCnt = false; - _taIrqNextCycle = false; - _tbIrqNextCycle = false; - _taState = TimerState.Stop; - _tbState = TimerState.Stop; - } + public void HardReset() + { + _pra = 0; + _prb = 0; + _ddra = 0; + _ddrb = 0; + _ta = 0xFFFF; + _tb = 0xFFFF; + _latcha = 1; + _latchb = 1; + _tod10Ths = 0; + _todSec = 0; + _todMin = 0; + _todHr = 0; + _alm10Ths = 0; + _almSec = 0; + _almMin = 0; + _almHr = 0; + _sdr = 0; + _icr = 0; + _cra = 0; + _crb = 0; + _intMask = 0; + _todLatch = false; + _taCntPhi2 = false; + _taCntCnt = false; + _tbCntPhi2 = false; + _tbCntTa = false; + _tbCntCnt = false; + _taIrqNextCycle = false; + _tbIrqNextCycle = false; + _taState = TimerState.Stop; + _tbState = TimerState.Stop; + _lastCnt = true; + } - private void CheckIrqs() - { - if (_taIrqNextCycle) - { - _taIrqNextCycle = false; - TriggerInterrupt(1); - } + private void CheckIrqs() + { + if (_taIrqNextCycle) + { + _taIrqNextCycle = false; + TriggerInterrupt(1); + } + if (_tbIrqNextCycle) + { + _tbIrqNextCycle = false; + TriggerInterrupt(2); + } + } - if (_tbIrqNextCycle) - { - _tbIrqNextCycle = false; - TriggerInterrupt(2); - } - } + public void ExecutePhase() + { + _thisCnt = ReadCnt(); + _taUnderflow = false; + if (DelayedInterrupts) + { + CheckIrqs(); + } - public void ExecutePhase() - { - _taUnderflow = false; - if (DelayedInterrupts) - { - CheckIrqs(); - } + if (_taPrb6NegativeNextCycle) + { + _prb &= 0xBF; + _taPrb6NegativeNextCycle = false; + } + if (_tbPrb7NegativeNextCycle) + { + _prb &= 0x7F; + _tbPrb7NegativeNextCycle = false; + } - if (_taPrb6NegativeNextCycle) - { - _prb &= 0xBF; - _taPrb6NegativeNextCycle = false; - } - if (_tbPrb7NegativeNextCycle) - { - _prb &= 0x7F; - _tbPrb7NegativeNextCycle = false; - } + switch (_taState) + { + case TimerState.WaitThenCount: + _taState = TimerState.Count; + Ta_Idle(); + break; + case TimerState.Stop: + Ta_Idle(); + break; + case TimerState.LoadThenStop: + _taState = TimerState.Stop; + _ta = _latcha; + Ta_Idle(); + break; + case TimerState.LoadThenCount: + _taState = TimerState.Count; + _ta = _latcha; + Ta_Idle(); + break; + case TimerState.LoadThenWaitThenCount: + _taState = TimerState.WaitThenCount; + if (_ta == 1) + { + Ta_Interrupt(); + _taUnderflow = true; + } + else + { + _ta = _latcha; + } + Ta_Idle(); + break; + case TimerState.Count: + Ta_Count(); + break; + case TimerState.CountThenStop: + _taState = TimerState.Stop; + Ta_Count(); + break; + } - switch (_taState) - { - case TimerState.WaitThenCount: - _taState = TimerState.Count; - Ta_Idle(); - break; - case TimerState.Stop: - Ta_Idle(); - break; - case TimerState.LoadThenStop: - _taState = TimerState.Stop; - _ta = _latcha; - Ta_Idle(); - break; - case TimerState.LoadThenCount: - _taState = TimerState.Count; - _ta = _latcha; - Ta_Idle(); - break; - case TimerState.LoadThenWaitThenCount: - _taState = TimerState.WaitThenCount; - if (_ta == 1) - { - Ta_Interrupt(); - _taUnderflow = true; - } - else - { - _ta = _latcha; - } + switch (_tbState) + { + case TimerState.WaitThenCount: + _tbState = TimerState.Count; + Tb_Idle(); + break; + case TimerState.Stop: + Tb_Idle(); + break; + case TimerState.LoadThenStop: + _tbState = TimerState.Stop; + _tb = _latchb; + Tb_Idle(); + break; + case TimerState.LoadThenCount: + _tbState = TimerState.Count; + _tb = _latchb; + Tb_Idle(); + break; + case TimerState.LoadThenWaitThenCount: + _tbState = TimerState.WaitThenCount; + if (_tb == 1) + { + Tb_Interrupt(); + } + else + { + _tb = _latchb; + } + Tb_Idle(); + break; + case TimerState.Count: + Tb_Count(); + break; + case TimerState.CountThenStop: + _tbState = TimerState.Stop; + Tb_Count(); + break; + } - Ta_Idle(); - break; - case TimerState.Count: - Ta_Count(); - break; - case TimerState.CountThenStop: - _taState = TimerState.Stop; - Ta_Count(); - break; - } + CountTod(); - switch (_tbState) - { - case TimerState.WaitThenCount: - _tbState = TimerState.Count; - Tb_Idle(); - break; - case TimerState.Stop: - Tb_Idle(); - break; - case TimerState.LoadThenStop: - _tbState = TimerState.Stop; - _tb = _latchb; - Tb_Idle(); - break; - case TimerState.LoadThenCount: - _tbState = TimerState.Count; - _tb = _latchb; - Tb_Idle(); - break; - case TimerState.LoadThenWaitThenCount: - _tbState = TimerState.WaitThenCount; - if (_tb == 1) - { - Tb_Interrupt(); - } - else - { - _tb = _latchb; - } + if (!_todLatch) + { + _latch10Ths = _tod10Ths; + _latchSec = _todSec; + _latchMin = _todMin; + _latchHr = _todHr; + } - Tb_Idle(); - break; - case TimerState.Count: - Tb_Count(); - break; - case TimerState.CountThenStop: - _tbState = TimerState.Stop; - Tb_Count(); - break; - } + _flagInput = ReadFlag(); + if (!_flagInput && _flagLatch) + { + TriggerInterrupt(16); + } + _flagLatch = _flagInput; - CountTod(); + if (!DelayedInterrupts) + { + CheckIrqs(); + } - if (!_todLatch) - { - _latch10Ths = _tod10Ths; - _latchSec = _todSec; - _latchMin = _todMin; - _latchHr = _todHr; - } + if ((_cra & 0x02) != 0) + _ddra |= 0x40; + if ((_crb & 0x02) != 0) + _ddrb |= 0x80; - _flagInput = ReadFlag(); - if (!_flagInput && _flagLatch) - { - TriggerInterrupt(16); - } + _lastCnt = _thisCnt; + } - _flagLatch = _flagInput; + private void Ta_Count() + { + if (_taCntPhi2 || (_taCntCnt && !_lastCnt && _thisCnt)) + { + if (_ta <= 0 || --_ta == 0) + { + if (_taState != TimerState.Stop) + { + Ta_Interrupt(); + } + _taUnderflow = true; + } + } + Ta_Idle(); + } - if (!DelayedInterrupts) - { - CheckIrqs(); - } + private void Ta_Interrupt() + { + _ta = _latcha; + _taIrqNextCycle = true; + _icr |= 1; - if ((_cra & 0x02) != 0) - { - _ddra |= 0x40; - } + if ((_cra & 0x08) != 0) + { + _cra &= 0xFE; + _newCra &= 0xFE; + _taState = TimerState.LoadThenStop; + } + else + { + _taState = TimerState.LoadThenCount; + } - if ((_crb & 0x02) != 0) - { - _ddrb |= 0x80; - } - } + if ((_cra & 0x02) != 0) + { + if ((_cra & 0x04) != 0) + { + _taPrb6NegativeNextCycle = true; + _prb |= 0x40; + } + else + { + _prb ^= 0x40; + } + _ddrb |= 0x40; + } + } - private void Ta_Count() - { - if (_taCntPhi2) - { - if (_ta <= 0 || --_ta == 0) - { - if (_taState != TimerState.Stop) - { - Ta_Interrupt(); - } + private void Ta_Idle() + { + if (_hasNewCra) + { + switch (_taState) + { + case TimerState.Stop: + case TimerState.LoadThenStop: + if ((_newCra & 0x01) != 0) + { + _taState = (_newCra & 0x10) != 0 + ? TimerState.LoadThenWaitThenCount + : TimerState.WaitThenCount; + } + else + { + if ((_newCra & 0x10) != 0) + { + _taState = TimerState.LoadThenStop; + } + } + break; + case TimerState.Count: + if ((_newCra & 0x01) != 0) + { + if ((_newCra & 0x10) != 0) + { + _taState = TimerState.LoadThenWaitThenCount; + } + } + else + { + _taState = (_newCra & 0x10) != 0 + ? TimerState.LoadThenStop + : TimerState.CountThenStop; + } + break; + case TimerState.LoadThenCount: + case TimerState.WaitThenCount: + if ((_newCra & 0x01) != 0) + { + if ((_newCra & 0x08) != 0) + { + _newCra &= 0xFE; + _taState = TimerState.Stop; + } + else if ((_newCra & 0x10) != 0) + { + _taState = TimerState.LoadThenWaitThenCount; + } + } + else + { + _taState = TimerState.Stop; + } + break; + } + _cra = _newCra & 0xEF; + _hasNewCra = false; + } + } - _taUnderflow = true; - } - } + private void Tb_Count() + { + if (_tbCntPhi2 || (_tbCntTa && _taUnderflow) || (_tbCntTaCnt && _taUnderflow && _thisCnt) || (_tbCntCnt && !_lastCnt && _thisCnt)) + { + if (_tb <= 0 || --_tb == 0) + { + if (_tbState != TimerState.Stop) + { + Tb_Interrupt(); + } + } + } + Tb_Idle(); + } - Ta_Idle(); - } + private void Tb_Interrupt() + { + _tb = _latchb; + _tbIrqNextCycle = true; + _icr |= 2; - private void Ta_Interrupt() - { - _ta = _latcha; - _taIrqNextCycle = true; - _icr |= 1; + if ((_crb & 0x08) != 0) + { + _crb &= 0xFE; + _newCrb &= 0xFE; + _tbState = TimerState.LoadThenStop; + } + else + { + _tbState = TimerState.LoadThenCount; + } - if ((_cra & 0x08) != 0) - { - _cra &= 0xFE; - _newCra &= 0xFE; - _taState = TimerState.LoadThenStop; - } - else - { - _taState = TimerState.LoadThenCount; - } + if ((_crb & 0x02) != 0) + { + if ((_crb & 0x04) != 0) + { + _tbPrb7NegativeNextCycle = true; + _prb |= 0x80; + } + else + { + _prb ^= 0x80; + } + } + } - if ((_cra & 0x02) != 0) - { - if ((_cra & 0x04) != 0) - { - _taPrb6NegativeNextCycle = true; - _prb |= 0x40; - } - else - { - _prb ^= 0x40; - } + private void Tb_Idle() + { + if (_hasNewCrb) + { + switch (_tbState) + { + case TimerState.Stop: + case TimerState.LoadThenStop: + if ((_newCrb & 0x01) != 0) + { + _tbState = (_newCrb & 0x10) != 0 + ? TimerState.LoadThenWaitThenCount + : TimerState.WaitThenCount; + } + else + { + if ((_newCrb & 0x10) != 0) + { + _tbState = TimerState.LoadThenStop; + } + } + break; + case TimerState.Count: + if ((_newCrb & 0x01) != 0) + { + if ((_newCrb & 0x10) != 0) + { + _tbState = TimerState.LoadThenWaitThenCount; + } + } + else + { + _tbState = (_newCrb & 0x10) != 0 + ? TimerState.LoadThenStop + : TimerState.CountThenStop; + } + break; + case TimerState.LoadThenCount: + case TimerState.WaitThenCount: + if ((_newCrb & 0x01) != 0) + { + if ((_newCrb & 0x08) != 0) + { + _newCrb &= 0xFE; + _tbState = TimerState.Stop; + } + else if ((_newCrb & 0x10) != 0) + { + _tbState = TimerState.LoadThenWaitThenCount; + } + } + else + { + _tbState = TimerState.Stop; + } + break; + } + _crb = _newCrb & 0xEF; + _hasNewCrb = false; + } + } - _ddrb |= 0x40; - } - } - - private void Ta_Idle() - { - if (_hasNewCra) - { - switch (_taState) - { - case TimerState.Stop: - case TimerState.LoadThenStop: - if ((_newCra & 0x01) != 0) - { - _taState = (_newCra & 0x10) != 0 - ? TimerState.LoadThenWaitThenCount - : TimerState.WaitThenCount; - } - else - { - if ((_newCra & 0x10) != 0) - { - _taState = TimerState.LoadThenStop; - } - } - - break; - case TimerState.Count: - if ((_newCra & 0x01) != 0) - { - if ((_newCra & 0x10) != 0) - { - _taState = TimerState.LoadThenWaitThenCount; - } - } - else - { - _taState = (_newCra & 0x10) != 0 - ? TimerState.LoadThenStop - : TimerState.CountThenStop; - } - - break; - case TimerState.LoadThenCount: - case TimerState.WaitThenCount: - if ((_newCra & 0x01) != 0) - { - if ((_newCra & 0x08) != 0) - { - _newCra &= 0xFE; - _taState = TimerState.Stop; - } - else if ((_newCra & 0x10) != 0) - { - _taState = TimerState.LoadThenWaitThenCount; - } - } - else - { - _taState = TimerState.Stop; - } - - break; - } - - _cra = _newCra & 0xEF; - _hasNewCra = false; - } - } - - private void Tb_Count() - { - if (_tbCntPhi2 || (_tbCntTa && _taUnderflow)) - { - if (_tb <= 0 || --_tb == 0) - { - if (_tbState != TimerState.Stop) - { - Tb_Interrupt(); - } - } - } - - Tb_Idle(); - } - - private void Tb_Interrupt() - { - _tb = _latchb; - _tbIrqNextCycle = true; - _icr |= 2; - - if ((_crb & 0x08) != 0) - { - _crb &= 0xFE; - _newCrb &= 0xFE; - _tbState = TimerState.LoadThenStop; - } - else - { - _tbState = TimerState.LoadThenCount; - } - - if ((_crb & 0x02) != 0) - { - if ((_crb & 0x04) != 0) - { - _tbPrb7NegativeNextCycle = true; - _prb |= 0x80; - } - else - { - _prb ^= 0x80; - } - } - } - - private void Tb_Idle() - { - if (_hasNewCrb) - { - switch (_tbState) - { - case TimerState.Stop: - case TimerState.LoadThenStop: - if ((_newCrb & 0x01) != 0) - { - _tbState = (_newCrb & 0x10) != 0 - ? TimerState.LoadThenWaitThenCount - : TimerState.WaitThenCount; - } - else - { - if ((_newCrb & 0x10) != 0) - { - _tbState = TimerState.LoadThenStop; - } - } - - break; - case TimerState.Count: - if ((_newCrb & 0x01) != 0) - { - if ((_newCrb & 0x10) != 0) - { - _tbState = TimerState.LoadThenWaitThenCount; - } - } - else - { - _tbState = (_newCrb & 0x10) != 0 - ? TimerState.LoadThenStop - : TimerState.CountThenStop; - } - - break; - case TimerState.LoadThenCount: - case TimerState.WaitThenCount: - if ((_newCrb & 0x01) != 0) - { - if ((_newCrb & 0x08) != 0) - { - _newCrb &= 0xFE; - _tbState = TimerState.Stop; - } - else if ((_newCrb & 0x10) != 0) - { - _tbState = TimerState.LoadThenWaitThenCount; - } - } - else - { - _tbState = TimerState.Stop; - } - - break; - } - - _crb = _newCrb & 0xEF; - _hasNewCrb = false; - } - } - - private void TriggerInterrupt(int bit) - { - _icr |= bit; - if ((_intMask & bit) == 0) - { - return; - } - - _icr |= 0x80; - } + private void TriggerInterrupt(int bit) + { + _icr |= bit; + if ((_intMask & bit) == 0) return; + _icr |= 0x80; + } public void SyncState(Serializer ser) { @@ -578,6 +562,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_flagLatch), ref _flagLatch); ser.Sync(nameof(_todCounter), ref _todCounter); + ser.Sync(nameof(_lastCnt), ref _lastCnt); + ser.Sync(nameof(_thisCnt), ref _thisCnt); } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs index 9c3ed0ed3d..cd57404c71 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Via.cs @@ -110,11 +110,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _prb = 0; _ddra = 0; _ddrb = 0; - _t1C = 0; - _t1L = 0; - _t2C = 0; - _t2L = 0; - _sr = 0; + _t1C = 0xFFFF; + _t1L = 0xFFFF; + _t2C = 0xFFFF; + _t2L = 0xFFFF; + _sr = 0xFF; _acr = 0; _pcr = 0; _ifr = 0; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs index 02257c2253..44259d3899 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.Registers.cs @@ -163,8 +163,11 @@ public override bool ReadDeviceData() { + // PB1 (input not pulled up) var viaOutputData = (Via0.DdrB & 0x02) != 0 && (Via0.PrB & 0x02) != 0; + // inverted from c64, input, not pulled up to PB7/CA1 var viaInputAtn = ViaReadAtn(); + // PB4 (input not pulled up) var viaOutputAtna = (Via0.DdrB & 0x10) != 0 && (Via0.PrB & 0x10) != 0; return !(viaOutputAtna ^ viaInputAtn) && !viaOutputData; From 2c804cab3414261ccc656262c7b5634b9b290436 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 00:51:19 -0500 Subject: [PATCH 09/53] C64: Fix a function ambiguity in the CIA class. --- .../Computers/Commodore64/C64.Motherboard.cs | 16 ++++++++-------- .../Computers/Commodore64/MOS/Chip6526.cs | 4 ++-- 2 files changed, 10 insertions(+), 10 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index d37a27590f..605277c2ad 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -100,23 +100,23 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 { case C64.VicType.Ntsc: Vic = Chip6567R8.Create(borderType); - Cia0 = Chip6526.Create(C64.CiaType.Ntsc, Input_ReadKeyboard, Input_ReadJoysticks); - Cia1 = Chip6526.Create(C64.CiaType.Ntsc, Cia1_ReadPortA); + Cia0 = Chip6526.CreateCia0(C64.CiaType.Ntsc, Input_ReadKeyboard, Input_ReadJoysticks); + Cia1 = Chip6526.CreateCia1(C64.CiaType.Ntsc, Cia1_ReadPortA, () => 0xFF); break; case C64.VicType.Pal: Vic = Chip6569.Create(borderType); - Cia0 = Chip6526.Create(C64.CiaType.Pal, Input_ReadKeyboard, Input_ReadJoysticks); - Cia1 = Chip6526.Create(C64.CiaType.Pal, Cia1_ReadPortA); + Cia0 = Chip6526.CreateCia0(C64.CiaType.Pal, Input_ReadKeyboard, Input_ReadJoysticks); + Cia1 = Chip6526.CreateCia1(C64.CiaType.Pal, Cia1_ReadPortA, () => 0xFF); break; case C64.VicType.NtscOld: Vic = Chip6567R56A.Create(borderType); - Cia0 = Chip6526.Create(C64.CiaType.NtscRevA, Input_ReadKeyboard, Input_ReadJoysticks); - Cia1 = Chip6526.Create(C64.CiaType.NtscRevA, Cia1_ReadPortA); + Cia0 = Chip6526.CreateCia0(C64.CiaType.NtscRevA, Input_ReadKeyboard, Input_ReadJoysticks); + Cia1 = Chip6526.CreateCia1(C64.CiaType.NtscRevA, Cia1_ReadPortA, () => 0xFF); break; case C64.VicType.Drean: Vic = Chip6572.Create(borderType); - Cia0 = Chip6526.Create(C64.CiaType.Pal, Input_ReadKeyboard, Input_ReadJoysticks); - Cia1 = Chip6526.Create(C64.CiaType.Pal, Cia1_ReadPortA); + Cia0 = Chip6526.CreateCia0(C64.CiaType.Pal, Input_ReadKeyboard, Input_ReadJoysticks); + Cia1 = Chip6526.CreateCia1(C64.CiaType.Pal, Cia1_ReadPortA, () => 0xFF); break; } User = new UserPort(); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs index bddb02b6f1..f7fb0665f1 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6526.cs @@ -9,7 +9,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // * A low RES pin is emulated via HardReset(). public static class Chip6526 { - public static Cia Create(C64.CiaType type, Func readIec, Func readUserPort) + public static Cia CreateCia1(C64.CiaType type, Func readIec, Func readUserPort) { switch (type) { @@ -38,7 +38,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } } - public static Cia Create(C64.CiaType type, Func keyboard, Func joysticks) + public static Cia CreateCia0(C64.CiaType type, Func keyboard, Func joysticks) { switch (type) { From 55145ff7ba376352d4bd757ed5170ddd84c630ea Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 12:46:28 -0500 Subject: [PATCH 10/53] C64: The T64 format was never supported, but at least make the core aware of it --- .../Computers/Commodore64/C64FormatFinder.cs | 2 +- BizHawk.Emulation.Cores/Computers/Commodore64/Media/Tape.cs | 5 +++++ 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs index f8d975c5c7..ac3df6e62f 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64FormatFinder.cs @@ -27,7 +27,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 return C64Format.G64; } - if (header.StartsWith("C64S tape image ")) + if (header.StartsWith("C64S tape image ") || header.StartsWith("C64 tape image f")) { return C64Format.T64; } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Tape.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Tape.cs index 9113eeb59a..529b893b09 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Tape.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Media/Tape.cs @@ -97,6 +97,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media result = new Tape(version, tapeFile, 20, tapeFile.Length); } + else if (Encoding.ASCII.GetString(tapeFile, 0, 0x12) == "C64 tape image file") + { + throw new Exception("The T64 format is not yet supported."); + } + return result; } From 5c9445fb96226e8a12ff0a1088a1ab50e29ac934 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 12:47:09 -0500 Subject: [PATCH 11/53] C64: Reuse some local memory in the sprite renderer. --- .../Computers/Commodore64/MOS/Vic.Render.cs | 61 +++++++++---------- 1 file changed, 30 insertions(+), 31 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs index ce4b29ec76..3962d16305 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs @@ -9,6 +9,7 @@ private int _pixelCounter; private int _pixelData; private int _pixelOwner; + private Sprite _spr; private int _sprData; private int _sprIndex; private int _sprPixel; @@ -43,8 +44,8 @@ _hblank = true; _renderEnabled = !_hblank && !_vblank; - _pixelCounter = -1; - while (_pixelCounter++ < 3) + _pixelCounter = 4; + while (--_pixelCounter >= 0) { if ((_srColorSync & SrColorMask) != 0) @@ -173,53 +174,53 @@ #region SPRITES // render sprites _pixelOwner = -1; - _sprIndex = 0; - foreach (var spr in _sprites) + for (_sprIndex = 0; _sprIndex < 8; _sprIndex++) { + _spr = _sprites[_sprIndex]; _sprData = 0; _sprPixel = _pixel; - if (spr.X == _rasterX) + if (_spr.X == _rasterX) { - spr.ShiftEnable = spr.Display; - spr.XCrunch = !spr.XExpand; - spr.MulticolorCrunch = false; + _spr.ShiftEnable = _spr.Display; + _spr.XCrunch = !_spr.XExpand; + _spr.MulticolorCrunch = false; } else { - spr.XCrunch |= !spr.XExpand; + _spr.XCrunch |= !_spr.XExpand; } - if (spr.ShiftEnable) // sprite rule 6 + if (_spr.ShiftEnable) // sprite rule 6 { - if (spr.Multicolor) + if (_spr.Multicolor) { - _sprData = spr.Sr & SrSpriteMaskMc; - if (spr.MulticolorCrunch && spr.XCrunch && !_rasterXHold) + _sprData = _spr.Sr & SrSpriteMaskMc; + if (_spr.MulticolorCrunch && _spr.XCrunch && !_rasterXHold) { - if (spr.Loaded == 0) + if (_spr.Loaded == 0) { - spr.ShiftEnable = false; + _spr.ShiftEnable = false; } - spr.Sr <<= 2; - spr.Loaded >>= 2; + _spr.Sr <<= 2; + _spr.Loaded >>= 2; } - spr.MulticolorCrunch ^= spr.XCrunch; + _spr.MulticolorCrunch ^= _spr.XCrunch; } else { - _sprData = spr.Sr & SrSpriteMask; - if (spr.XCrunch && !_rasterXHold) + _sprData = _spr.Sr & SrSpriteMask; + if (_spr.XCrunch && !_rasterXHold) { - if (spr.Loaded == 0) + if (_spr.Loaded == 0) { - spr.ShiftEnable = false; + _spr.ShiftEnable = false; } - spr.Sr <<= 1; - spr.Loaded >>= 1; + _spr.Sr <<= 1; + _spr.Loaded >>= 1; } } - spr.XCrunch ^= spr.XExpand; + _spr.XCrunch ^= _spr.XExpand; if (_sprData != 0) { @@ -232,7 +233,7 @@ _sprPixel = _spriteMulticolor0; break; case SrSpriteMask2: - _sprPixel = spr.Color; + _sprPixel = _spr.Color; break; case SrSpriteMask3: _sprPixel = _spriteMulticolor1; @@ -244,7 +245,7 @@ { if (!_borderOnVertical) { - spr.CollideSprite = true; + _spr.CollideSprite = true; _sprites[_pixelOwner].CollideSprite = true; } } @@ -252,11 +253,11 @@ // sprite-data collision if (!_borderOnVertical && (_pixelData >= SrMask2)) { - spr.CollideData = true; + _spr.CollideData = true; } // sprite priority logic - if (spr.Priority) + if (_spr.Priority) { _pixel = _pixelData >= SrMask2 ? _pixel : _sprPixel; } @@ -266,8 +267,6 @@ } } } - - _sprIndex++; } #endregion From 36ac592193db46d35cf133f4fbd3cd3fd2c28a94 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 13:59:41 -0500 Subject: [PATCH 12/53] C64: Individual IRQ flags for S/S or S/D collisions are always set even if not eligible to assert IRQ externally --- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs index 3962d16305..685a191ef5 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs @@ -247,6 +247,7 @@ { _spr.CollideSprite = true; _sprites[_pixelOwner].CollideSprite = true; + _intSpriteCollision = true; } } @@ -254,6 +255,7 @@ if (!_borderOnVertical && (_pixelData >= SrMask2)) { _spr.CollideData = true; + _intSpriteDataCollision = true; } // sprite priority logic From 691577499fadb0484b2075b241fcbdcb43f8e2fc Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 14:12:46 -0500 Subject: [PATCH 13/53] C64: When a sprite is eligible for display, initialize it with the correct crunch state based on Y expansion --- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 2d727659fa..fe55e794f1 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -181,7 +181,7 @@ { spr.Dma = true; spr.Mcbase = 0; - spr.YCrunch = true; + spr.YCrunch = !spr.YExpand; } } } From 579ffe5c25c6126f2884b2fd343bef9d47bd46a4 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 14:20:12 -0500 Subject: [PATCH 14/53] C64: Had the flag with the wrong polarity. Thanks, C64Anabalt. --- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index fe55e794f1..7e33d1eae8 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -181,7 +181,7 @@ { spr.Dma = true; spr.Mcbase = 0; - spr.YCrunch = !spr.YExpand; + spr.YCrunch = spr.YExpand; } } } From f45e934fec72673084d973b34cb465753aacfaee Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 17:24:22 -0500 Subject: [PATCH 15/53] C64: Reset the SID filter on hard reset. --- .../Commodore64/MOS/Sid.SoundProvider.cs | 10 ++-- .../Computers/Commodore64/MOS/Sid.cs | 55 +++++++++++-------- 2 files changed, 37 insertions(+), 28 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.SoundProvider.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.SoundProvider.cs index d5d4d337dd..1cc3e97650 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.SoundProvider.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.SoundProvider.cs @@ -47,10 +47,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _outputBuffer = new short[_outputBufferIndex * 2]; for (int i = 0; i < _outputBufferIndex; i++) { - _mixer = _outputBuffer_not_filtered[i] + _outputBuffer_filtered[i]; + _mixer = _outputBufferNotFiltered[i] + _outputBufferFiltered[i]; _mixer = _mixer >> 7; - _mixer = (_mixer * _volume_at_sample_time[i]) >> 4; - _mixer -= _volume_at_sample_time[i] << 8; + _mixer = (_mixer * _volumeAtSampleTime[i]) >> 4; + _mixer -= _volumeAtSampleTime[i] << 8; if (_mixer > 0x7FFF) { @@ -69,9 +69,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS samples = _outputBuffer; nsamp = _outputBufferIndex; - last_filtered_value = _outputBuffer_filtered[_outputBufferIndex - 1]; + _lastFilteredValue = _outputBufferFiltered[_outputBufferIndex - 1]; _outputBufferIndex = 0; - filter_index = 0; + _filterIndex = 0; } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs index 87744b909b..1ffb2e1985 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs @@ -39,12 +39,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private bool _filterSelectHiPass; private int _mixer; private short[] _outputBuffer; - private int[] _outputBuffer_filtered; - private int[] _outputBuffer_not_filtered; - private int[] _volume_at_sample_time; + private readonly int[] _outputBufferFiltered; + private readonly int[] _outputBufferNotFiltered; + private readonly int[] _volumeAtSampleTime; private int _outputBufferIndex; - private int filter_index; - private int last_filtered_value; + private int _filterIndex; + private int _lastFilteredValue; private int _potCounter; private int _potX; private int _potY; @@ -92,9 +92,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS for (var i = 0; i < 3; i++) _filterEnable[i] = false; - _outputBuffer_filtered = new int[sampleRate]; - _outputBuffer_not_filtered = new int[sampleRate]; - _volume_at_sample_time = new int[sampleRate]; + _outputBufferFiltered = new int[sampleRate]; + _outputBufferNotFiltered = new int[sampleRate]; + _volumeAtSampleTime = new int[sampleRate]; } // ------------------------------------ @@ -109,6 +109,15 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _potCounter = 0; _potX = 0; _potY = 0; + _filterEnable[0] = false; + _filterEnable[1] = false; + _filterEnable[2] = false; + _filterFrequency = 0; + _filterSelectBandPass = false; + _filterSelectHiPass = false; + _filterSelectLoPass = false; + _filterResonance = 0; + _volume = 0; } // ------------------------------------ @@ -190,9 +199,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS if (_outputBufferIndex < _sampleRate) { - _outputBuffer_not_filtered[_outputBufferIndex] = temp_not_filtered; - _outputBuffer_filtered[_outputBufferIndex] = temp_filtered; - _volume_at_sample_time[_outputBufferIndex] = _volume; + _outputBufferNotFiltered[_outputBufferIndex] = temp_not_filtered; + _outputBufferFiltered[_outputBufferIndex] = temp_filtered; + _volumeAtSampleTime[_outputBufferIndex] = _volume; _outputBufferIndex++; } } @@ -202,7 +211,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { if (_filterEnable[0] | _filterEnable[1] | _filterEnable[2]) { - if ((_outputBufferIndex - filter_index) >= 16) + if ((_outputBufferIndex - _filterIndex) >= 16) { filter_operator(); } @@ -210,22 +219,22 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { // the length is too short for the FFT to reliably act on the output // instead, clamp it to the previous output. - for (int i = filter_index; i < _outputBufferIndex; i++) + for (int i = _filterIndex; i < _outputBufferIndex; i++) { - _outputBuffer_filtered[i] = last_filtered_value; + _outputBufferFiltered[i] = _lastFilteredValue; } } } - filter_index = _outputBufferIndex; + _filterIndex = _outputBufferIndex; if (_outputBufferIndex>0) - last_filtered_value = _outputBuffer_filtered[_outputBufferIndex - 1]; + _lastFilteredValue = _outputBufferFiltered[_outputBufferIndex - 1]; } // if the filter is off, keep updating the filter index to the most recent Flush if (!(_filterEnable[0] | _filterEnable[1] | _filterEnable[2])) { - filter_index = _outputBufferIndex; + _filterIndex = _outputBufferIndex; } } @@ -236,7 +245,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS double attenuation; - int nsamp = _outputBufferIndex - filter_index; + int nsamp = _outputBufferIndex - _filterIndex; // pass the list of filtered samples to the FFT // but needs to be a power of 2, so find the next highest power of 2 and re-sample @@ -258,7 +267,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // linearly interpolate the original sample set into the new denser sample set for (double i = 0; i < nsamp_2; i++) { - temp_buffer[(int)i] = _outputBuffer_filtered[(int)Math.Floor((i / (nsamp_2-1) * (nsamp - 1))) + filter_index]; + temp_buffer[(int)i] = _outputBufferFiltered[(int)Math.Floor((i / (nsamp_2-1) * (nsamp - 1))) + _filterIndex]; } // now we have everything we need to perform the FFT @@ -312,11 +321,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS //re-sample back down to the original number of samples for (double i = 0; i < nsamp; i++) { - _outputBuffer_filtered[(int)i + filter_index] = (int)(temp_buffer[(int)Math.Ceiling((i / (nsamp - 1) * (nsamp_2 - 1)))]/(nsamp_2/2)); + _outputBufferFiltered[(int)i + _filterIndex] = (int)(temp_buffer[(int)Math.Ceiling((i / (nsamp - 1) * (nsamp_2 - 1)))]/(nsamp_2/2)); - if (_outputBuffer_filtered[(int)i + filter_index] < 0) + if (_outputBufferFiltered[(int)i + _filterIndex] < 0) { - _outputBuffer_filtered[(int)i + filter_index] = 0; + _outputBufferFiltered[(int)i + _filterIndex] = 0; } // the FFT is only an approximate model and fails at low sample rates @@ -324,7 +333,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // thus smoothing out the FT samples if (i<16) - _outputBuffer_filtered[(int)i + filter_index] = (int)((last_filtered_value * Math.Pow(15 - i,1) + _outputBuffer_filtered[(int)i + filter_index] * Math.Pow(i,1))/ Math.Pow(15,1)); + _outputBufferFiltered[(int)i + _filterIndex] = (int)((_lastFilteredValue * Math.Pow(15 - i,1) + _outputBufferFiltered[(int)i + _filterIndex] * Math.Pow(i,1))/ Math.Pow(15,1)); } } // ---------------------------------- From 49b613962e5d9bc4366b86138e158d10c4c259a9 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Thu, 4 Jul 2019 17:32:35 -0500 Subject: [PATCH 16/53] C64: Fix a typo disabling voice 3 when high pass filter is set --- .../Computers/Commodore64/MOS/Sid.Registers.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.Registers.cs index 71f65cf74e..3140be4fef 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.Registers.cs @@ -222,7 +222,7 @@ _filterSelectLoPass = (val & 0x10) != 0; _filterSelectBandPass = (val & 0x20) != 0; _filterSelectHiPass = (val & 0x40) != 0; - _disableVoice3 = (val & 0x40) != 0; + _disableVoice3 = (val & 0x80) != 0; break; case 0x19: _potX = val; From 69f8b143a363ef046290d20903ccdd5b32db0735 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Fri, 5 Jul 2019 21:05:38 -0500 Subject: [PATCH 17/53] C64: Foreground pixels are black when VIC is in idle state. --- .../Computers/Commodore64/MOS/Vic.Render.cs | 24 +++++++++---------- 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs index 685a191ef5..8636bf799d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs @@ -93,13 +93,13 @@ _pixel = _backgroundColor0; break; case SrMask1: - _pixel = _backgroundColor1; + _pixel = _idle ? 0 : _backgroundColor1; break; case SrMask2: - _pixel = _backgroundColor2; + _pixel = _idle ? 0 :_backgroundColor2; break; default: - _pixel = (_displayC & 0x700) >> 8; + _pixel = _idle ? 0 : (_displayC & 0x700) >> 8; break; } } @@ -107,12 +107,12 @@ { // standard 001 _pixelData = _sr & SrMask2; - _pixel = _pixelData != 0 ? _displayC >> 8 : _backgroundColor0; + _pixel = _pixelData != 0 ? (_idle ? 0 : _displayC >> 8) : _backgroundColor0; } break; case VideoMode010: _pixelData = _sr & SrMask2; - _pixel = _pixelData != 0 ? _displayC >> 4 : _displayC; + _pixel = _idle ? 0 : _pixelData != 0 ? _displayC >> 4 : _displayC; break; case VideoMode011: if ((_srSync & SrMask2) != 0) @@ -124,13 +124,13 @@ _pixel = _backgroundColor0; break; case SrMask1: - _pixel = _displayC >> 4; + _pixel = _idle ? 0 : _displayC >> 4; break; case SrMask2: - _pixel = _displayC; + _pixel = _idle ? 0 : _displayC; break; default: - _pixel = _displayC >> 8; + _pixel = _idle ? 0 : _displayC >> 8; break; } break; @@ -138,7 +138,7 @@ _pixelData = _sr & SrMask2; if (_pixelData != 0) { - _pixel = _displayC >> 8; + _pixel = _idle ? 0 : _displayC >> 8; } else { @@ -149,13 +149,13 @@ _pixel = _backgroundColor0; break; case 1: - _pixel = _backgroundColor1; + _pixel = _idle ? 0 : _backgroundColor1; break; case 2: - _pixel = _backgroundColor2; + _pixel = _idle ? 0 : _backgroundColor2; break; default: - _pixel = _backgroundColor3; + _pixel = _idle ? 0 : _backgroundColor3; break; } } From 400b04b690b5be8a2cf9a13de3607c1e3f0afe84 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Fri, 5 Jul 2019 23:59:01 -0500 Subject: [PATCH 18/53] C64: CIA was sometimes delaying too long to fire interrupts by 1 cycle. - This could have implications for existing TASes (!) --- .../Computers/Commodore64/MOS/Cia.cs | 49 +++++++++---------- 1 file changed, 24 insertions(+), 25 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs index 92d193737b..974bead03d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Cia.cs @@ -144,34 +144,29 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _lastCnt = true; } - private void CheckIrqs() - { - if (_taIrqNextCycle) - { - _taIrqNextCycle = false; - TriggerInterrupt(1); - } - if (_tbIrqNextCycle) - { - _tbIrqNextCycle = false; - TriggerInterrupt(2); - } - } - public void ExecutePhase() { _thisCnt = ReadCnt(); _taUnderflow = false; - if (DelayedInterrupts) - { - CheckIrqs(); - } + + if (_taIrqNextCycle) + { + _taIrqNextCycle = false; + TriggerInterrupt(1); + } + + if (_tbIrqNextCycle) + { + _tbIrqNextCycle = false; + TriggerInterrupt(2); + } if (_taPrb6NegativeNextCycle) { _prb &= 0xBF; _taPrb6NegativeNextCycle = false; } + if (_tbPrb7NegativeNextCycle) { _prb &= 0x7F; @@ -277,11 +272,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } _flagLatch = _flagInput; - if (!DelayedInterrupts) - { - CheckIrqs(); - } - if ((_cra & 0x02) != 0) _ddra |= 0x40; if ((_crb & 0x02) != 0) @@ -309,7 +299,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private void Ta_Interrupt() { _ta = _latcha; - _taIrqNextCycle = true; + + if (DelayedInterrupts) + _taIrqNextCycle = true; + else + TriggerInterrupt(1); + _icr |= 1; if ((_cra & 0x08) != 0) @@ -418,7 +413,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private void Tb_Interrupt() { _tb = _latchb; - _tbIrqNextCycle = true; + if (DelayedInterrupts) + _tbIrqNextCycle = true; + else + TriggerInterrupt(2); + _icr |= 2; if ((_crb & 0x08) != 0) From d48964b642c072cbba1571981e8308227f910305 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 6 Jul 2019 00:00:51 -0500 Subject: [PATCH 19/53] 6502X: According to the datasheet, RDY must be high in order for interrupts to trigger - this has implications for C64, as it may cause VIC interrupts to fire quite later than they currently do --- BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs index 4d9a8edfad..8efa5c3fb0 100644 --- a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs +++ b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs @@ -528,7 +528,7 @@ namespace BizHawk.Emulation.Cores.Components.M6502 { get { - return NMI || (IRQ && !FlagI); + return RDY && (NMI || (IRQ && !FlagI)); } } @@ -560,6 +560,10 @@ namespace BizHawk.Emulation.Cores.Components.M6502 void Fetch1() { + rdy_freeze = !RDY; + if (!RDY) + return; + my_iflag = FlagI; FlagI = iflag_pending; if (!branch_irq_hack) From 6ed11de85b1acc7dfcd3ca51934b6abfc2e625df Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 6 Jul 2019 01:19:58 -0500 Subject: [PATCH 20/53] C64: Soft/Hard reset: it's about time --- .../Computers/Commodore64/C64.IEmulator.cs | 26 +++++++++++++++++++ .../Computers/Commodore64/C64.Motherboard.cs | 18 +++++++++++++ .../Computers/Commodore64/C64.cs | 14 ++++++++-- .../Computers/Commodore64/MOS/Chip6510.cs | 7 +++++ .../Computers/Commodore64/MOS/Vic.State.cs | 1 + 5 files changed, 64 insertions(+), 2 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IEmulator.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IEmulator.cs index 29aea5b261..b0f5b1a55c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IEmulator.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.IEmulator.cs @@ -21,6 +21,32 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 _board.Cpu.TraceCallback = null; } + if (controller.IsPressed("Power")) + { + if (!_powerPressed) + { + _powerPressed = true; + HardReset(); + } + } + else + { + _powerPressed = false; + } + + if (controller.IsPressed("Reset")) + { + if (!_resetPressed) + { + _resetPressed = true; + SoftReset(); + } + } + else + { + _resetPressed = false; + } + if (controller.IsPressed("Next Disk") && !_nextPressed) { _nextPressed = true; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 605277c2ad..a2b72bb9a3 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -191,6 +191,24 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 CartPort.HardReset(); } + public void SoftReset() + { + // equivalent to a hard reset EXCEPT cpu, color ram, memory + Bus = 0xFF; + InputRead = false; + + Cia0.HardReset(); + Cia1.HardReset(); + Serial.HardReset(); + Sid.HardReset(); + Vic.HardReset(); + User.HardReset(); + Cassette.HardReset(); + Serial.HardReset(); + Cpu.SoftReset(); + CartPort.HardReset(); + } + public void Init() { CartPort.ReadOpenBus = ReadOpenBus; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 0e17385256..926942bbe4 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -34,6 +34,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 _cyclesPerFrame = _board.Vic.CyclesPerFrame; _memoryCallbacks = new MemoryCallbackSystem(new[] { "System Bus" }); + InitMedia(_roms[_currentDisk]); HardReset(); switch (SyncSettings.VicType) @@ -137,7 +138,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 "Key Commodore", "Key Left Shift", "Key Z", "Key X", "Key C", "Key V", "Key B", "Key N", "Key M", "Key Comma", "Key Period", "Key Slash", "Key Right Shift", "Key Cursor Up/Down", "Key Cursor Left/Right", "Key Space", "Key F1", "Key F3", "Key F5", "Key F7", - "Previous Disk", "Next Disk" + "Previous Disk", "Next Disk", + "Power", "Reset" } }; @@ -147,6 +149,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private int _frame; private readonly ITraceable _tracer; + + // Power stuff + private bool _powerPressed; + private bool _resetPressed; // Disk stuff private bool _nextPressed; @@ -358,8 +364,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private void HardReset() { - InitMedia(_roms[_currentDisk]); _board.HardReset(); } + + private void SoftReset() + { + _board.SoftReset(); + } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index dc99da4183..ba2ba3bf53 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -97,6 +97,13 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _pinNmiLast = true; } + public void SoftReset() + { + _cpu.NESSoftReset(); + _port.Direction = 0x00; + _port.Latch = 0xFF; + } + // ------------------------------------ public void ExecutePhase() { diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index aae4938e95..aa6976c443 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -137,6 +137,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _xScroll = 0; _yScroll = 0; _cycle = 0; + _cycleIndex = 0; // reset sprites for (var i = 0; i < 8; i++) From 3bf37f1c17648cc3a6920cf75fb1bb4a39bf9019 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 6 Jul 2019 16:29:14 -0500 Subject: [PATCH 21/53] C64: No need for LagCycles anymore. --- BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs | 1 - .../Computers/Commodore64/MOS/Chip6510.cs | 7 ------- 2 files changed, 8 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 926942bbe4..1592c35a51 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -207,7 +207,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 { _board.InputRead = false; _board.PollInput(); - _board.Cpu.LagCycles = 0; } _board.Execute(); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index ba2ba3bf53..f3c9ad60ec 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -117,14 +117,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _cpu.NMI |= _pinNmiLast && !_thisNmi; _cpu.ExecuteOne(); } - else - { - LagCycles++; - } } - public int LagCycles; - internal bool AtInstructionStart() { return _cpu.AtInstructionStart(); @@ -233,7 +227,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.EndSection(); ser.Sync(nameof(_thisNmi), ref _thisNmi); - ser.Sync(nameof(LagCycles), ref LagCycles); } public void Write(int addr, int val) From d36e02045b86d64181f9ce2d07a2c21bf2e9ed2a Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 6 Jul 2019 16:32:21 -0500 Subject: [PATCH 22/53] C64: Optimize the RNG for 1541 flux transitions. (same output) --- .../Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs index 0f9a3782c5..569a8ef855 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/Serial/Drive1541.FluxTransitions.cs @@ -33,7 +33,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.Serial _rngCurrent = 1; } - _rngCurrent = (int)(_rngCurrent * LEHMER_RNG_PRIME % int.MaxValue); + _rngCurrent = unchecked((int) ((_rngCurrent * LEHMER_RNG_PRIME) & int.MaxValue)); } private void ExecuteFlux() From 3efea1503892470f1157a5b7976e62757058c429 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 05:24:47 -0500 Subject: [PATCH 23/53] 6502X: When !RDY is asserted, still do other operations. Plus, do dummy reads on stack ops --- .../CPUs/MOS 6502X/Execute.cs | 51 ++++++++++--------- 1 file changed, 27 insertions(+), 24 deletions(-) diff --git a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs index 8efa5c3fb0..db65556c49 100644 --- a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs +++ b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs @@ -928,19 +928,19 @@ namespace BizHawk.Emulation.Cores.Components.M6502 } void IndIdx_READ_Stage5() { - rdy_freeze = !RDY; - if (RDY) + if (!alu_temp.Bit(8)) { - if (!alu_temp.Bit(8)) + mi++; + ExecuteOneRetry(); + return; + } + else + { + rdy_freeze = !RDY; + if (RDY) { - mi++; - ExecuteOneRetry(); - return; - } - else - { - _link.ReadMemory((ushort)ea); - ea = (ushort)(ea + 0x100); + _link.ReadMemory((ushort) ea); + ea = (ushort) (ea + 0x100); } } } @@ -1196,13 +1196,17 @@ namespace BizHawk.Emulation.Cores.Components.M6502 void NOP() { rdy_freeze = !RDY; + if (RDY) + { + FetchDummy(); + } } void DecS() { rdy_freeze = !RDY; if (RDY) { - S--; + _link.DummyReadMemory((ushort) (0x100 | --S)); } } void IncS() @@ -1210,7 +1214,7 @@ namespace BizHawk.Emulation.Cores.Components.M6502 rdy_freeze = !RDY; if (RDY) { - S++; + _link.DummyReadMemory((ushort) (0x100 | S++)); } } void JSR() @@ -2256,19 +2260,18 @@ namespace BizHawk.Emulation.Cores.Components.M6502 } void AbsIdx_READ_Stage4() { - rdy_freeze = !RDY; - if (RDY) + if (!alu_temp.Bit(8)) { - if (!alu_temp.Bit(8)) + mi++; + ExecuteOneRetry(); + } + else + { + rdy_freeze = !RDY; + if (RDY) { - mi++; - ExecuteOneRetry(); - return; - } - else - { - alu_temp = _link.ReadMemory((ushort)ea); - ea = (ushort)(ea + 0x100); + alu_temp = _link.ReadMemory((ushort) ea); + ea = (ushort) (ea + 0x100); } } From 9f733d3e7a3650a10a13091d6c3ba654354a6db7 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 05:26:26 -0500 Subject: [PATCH 24/53] VIC: More accurate pixel pipeline --- .../Computers/Commodore64/MOS/Vic.Parse.cs | 113 +++++++++++++- .../Computers/Commodore64/MOS/Vic.Render.cs | 141 +++--------------- .../Computers/Commodore64/MOS/Vic.State.cs | 14 +- .../Computers/Commodore64/MOS/Vic.cs | 5 - 4 files changed, 131 insertions(+), 142 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 7e33d1eae8..c2aa36b04b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -42,6 +42,10 @@ private int _parseBa; private int _parseAct; private bool _parseIsSprCrunch; + private int _parsePixelData; + private int _parsePixelDataIndex; + private int _parseSrData0; + private int _parseSrData1; private void ParseCycle() { @@ -80,8 +84,6 @@ _dataC = 0; _bufferC[_vmli] = _dataC; } - - _srColorSync |= 0x01 << (7 - _xScroll); break; case FetchTypeGraphics: // fetch G @@ -96,14 +98,114 @@ if (_extraColorModeBuffer) _parseAddr &= AddressMaskEc; _dataG = ReadMemory(_parseAddr); - _sr |= _dataG << (7 - _xScroll); - _srSync |= 0xAA << (7 - _xScroll); + if (!_idle) { - _bufferG[_vmli] = _dataG; _vmli = (_vmli + 1) & 0x3F; _vc = (_vc + 1) & 0x3FF; } + + // graphics data shift register + _srData1 &= ~(0xFF << (7 - _xScroll)); + _srActive |= 0xFF << (7 - _xScroll); + + if (_multicolorMode && (_bitmapMode || (_dataC & 0x800) != 0)) + { + _parseSrData0 = (_dataG & 0x55) | ((_dataG & 0x55) << 1); + _parseSrData1 = (_dataG & 0xAA) | ((_dataG & 0xAA) >> 1); + } + else + { + _parseSrData0 = _bitmapMode ? 0 : _dataG; + _parseSrData1 = _dataG; + } + _srData1 |= _parseSrData1 << (7 - _xScroll); + + // graphics color shift register + _srColor0 &= ~(0xFF << (7 - _xScroll)); + _srColor1 &= ~(0xFF << (7 - _xScroll)); + _srColor2 &= ~(0xFF << (7 - _xScroll)); + _srColor3 &= ~(0xFF << (7 - _xScroll)); + for (_parsePixelDataIndex = 7; _parsePixelDataIndex >= 0; _parsePixelDataIndex--) + { + _parsePixelData = ((_parseSrData0 & 0x80) >> 7) | ((_parseSrData1 & 0x80) >> 6); + switch (_videoMode) + { + case VideoMode000: + case VideoMode001: + switch (_parsePixelData) + { + case 0: + _pixel = _backgroundColor0; + break; + case 1: + _pixel = _idle ? 0 : _backgroundColor1; + break; + case 2: + _pixel = _idle ? 0 :_backgroundColor2; + break; + default: + _pixel = _idle ? 0 : (_multicolorMode ? _dataC & 0x700 : _dataC) >> 8; + break; + } + break; + case VideoMode010: + case VideoMode011: + switch (_parsePixelData) + { + case 0: + _pixel = _backgroundColor0; + break; + case 1: + _pixel = _idle ? 0 : _dataC >> 4; + break; + case 2: + _pixel = _idle ? 0 : _dataC; + break; + default: + _pixel = _idle ? 0 : _dataC >> 8; + break; + } + break; + case VideoMode100: + if (_parsePixelData != 0) + { + _pixel = _idle ? 0 : _dataC >> 8; + } + else + { + _pixel = (_dataC & 0xC0) >> 6; + switch (_pixel) + { + case 0: + _pixel = _backgroundColor0; + break; + case 1: + _pixel = _idle ? 0 : _backgroundColor1; + break; + case 2: + _pixel = _idle ? 0 : _backgroundColor2; + break; + default: + _pixel = _idle ? 0 : _backgroundColor3; + break; + } + } + break; + default: + _parsePixelData = 0; + _pixel = 0; + break; + } + + _parseSrData0 <<= 1; + _parseSrData1 <<= 1; + _srColor0 |= (_pixel & 1) << (7 - _xScroll + _parsePixelDataIndex); + _srColor1 |= ((_pixel >> 1) & 1) << (7 - _xScroll + _parsePixelDataIndex); + _srColor2 |= ((_pixel >> 2) & 1) << (7 - _xScroll + _parsePixelDataIndex); + _srColor3 |= ((_pixel >> 3) & 1) << (7 - _xScroll + _parsePixelDataIndex); + } + break; case FetchTypeNone: // fetch none @@ -221,7 +323,6 @@ if ((_parseAct & PipelineUpdateVc) != 0) // VC/RC rule 2 { _vc = _vcbase; - _srColorIndexLatch = 0; _vmli = 0; if (_badline) { diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs index 8636bf799d..b1f2b12b38 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs @@ -4,18 +4,19 @@ { private int _borderPixel; private int _bufferPixel; - private int _ecmPixel; private int _pixel; private int _pixelCounter; - private int _pixelData; private int _pixelOwner; private Sprite _spr; private int _sprData; private int _sprIndex; private int _sprPixel; - private int _srSync; - private int _srColorSync; - private int _srColorIndexLatch; + private int _srColor0; + private int _srColor1; + private int _srColor2; + private int _srColor3; + private int _srData1; + private int _srActive; private int _videoMode; private int _borderOnShiftReg; @@ -26,10 +27,7 @@ private const int VideoMode100 = 4; private const int VideoModeInvalid = -1; - private const int SrMask1 = 0x20000; - private const int SrMask2 = SrMask1 << 1; - private const int SrMask3 = SrMask1 | SrMask2; - private const int SrColorMask = 0x8000; + private const int SrMask1 = 0x40000; private const int SrSpriteMask = SrSpriteMask2; private const int SrSpriteMask1 = 0x400000; private const int SrSpriteMask2 = SrSpriteMask1 << 1; @@ -47,13 +45,6 @@ _pixelCounter = 4; while (--_pixelCounter >= 0) { - - if ((_srColorSync & SrColorMask) != 0) - { - _displayC = _bufferC[_srColorIndexLatch]; - _srColorIndexLatch = (_srColorIndexLatch + 1) & 0x3F; - } - #region PRE-RENDER BORDER // check left border @@ -73,105 +64,14 @@ #endregion - #region CHARACTER GRAPHICS - switch (_videoMode) - { - case VideoMode000: - _pixelData = _sr & SrMask2; - _pixel = _pixelData != 0 ? _displayC >> 8 : _backgroundColor0; - break; - case VideoMode001: - if ((_displayC & 0x800) != 0) - { - // multicolor 001 - if ((_srSync & SrMask2) != 0) - _pixelData = _sr & SrMask3; + // render graphics + _pixel = (_srActive & SrMask1) != 0 + ? ((_srColor0 & SrMask1) >> 18) | + ((_srColor1 & SrMask1) >> 17) | + ((_srColor2 & SrMask1) >> 16) | + ((_srColor3 & SrMask1) >> 15) + : _backgroundColor0; - switch (_pixelData) - { - case 0: - _pixel = _backgroundColor0; - break; - case SrMask1: - _pixel = _idle ? 0 : _backgroundColor1; - break; - case SrMask2: - _pixel = _idle ? 0 :_backgroundColor2; - break; - default: - _pixel = _idle ? 0 : (_displayC & 0x700) >> 8; - break; - } - } - else - { - // standard 001 - _pixelData = _sr & SrMask2; - _pixel = _pixelData != 0 ? (_idle ? 0 : _displayC >> 8) : _backgroundColor0; - } - break; - case VideoMode010: - _pixelData = _sr & SrMask2; - _pixel = _idle ? 0 : _pixelData != 0 ? _displayC >> 4 : _displayC; - break; - case VideoMode011: - if ((_srSync & SrMask2) != 0) - _pixelData = _sr & SrMask3; - - switch (_pixelData) - { - case 0: - _pixel = _backgroundColor0; - break; - case SrMask1: - _pixel = _idle ? 0 : _displayC >> 4; - break; - case SrMask2: - _pixel = _idle ? 0 : _displayC; - break; - default: - _pixel = _idle ? 0 : _displayC >> 8; - break; - } - break; - case VideoMode100: - _pixelData = _sr & SrMask2; - if (_pixelData != 0) - { - _pixel = _idle ? 0 : _displayC >> 8; - } - else - { - _ecmPixel = (_displayC & 0xC0) >> 6; - switch (_ecmPixel) - { - case 0: - _pixel = _backgroundColor0; - break; - case 1: - _pixel = _idle ? 0 : _backgroundColor1; - break; - case 2: - _pixel = _idle ? 0 : _backgroundColor2; - break; - default: - _pixel = _idle ? 0 : _backgroundColor3; - break; - } - } - break; - default: - _pixelData = 0; - _pixel = 0; - break; - } - _pixel &= 0xF; - _sr <<= 1; - _srSync <<= 1; - _srColorSync <<= 1; - #endregion - - #region SPRITES // render sprites _pixelOwner = -1; for (_sprIndex = 0; _sprIndex < 8; _sprIndex++) @@ -252,7 +152,7 @@ } // sprite-data collision - if (!_borderOnVertical && (_pixelData >= SrMask2)) + if (!_borderOnVertical && (_srData1 & SrMask1) != 0) { _spr.CollideData = true; _intSpriteDataCollision = true; @@ -261,7 +161,7 @@ // sprite priority logic if (_spr.Priority) { - _pixel = _pixelData >= SrMask2 ? _pixel : _sprPixel; + _pixel = (_srData1 & SrMask1) != 0 ? _pixel : _sprPixel; } else { @@ -271,8 +171,6 @@ } } - #endregion - #region POST-RENDER BORDER // border doesn't work with the background buffer @@ -298,6 +196,13 @@ if (!_rasterXHold) _rasterX++; + + _srColor0 <<= 1; + _srColor1 <<= 1; + _srColor2 <<= 1; + _srColor3 <<= 1; + _srData1 <<= 1; + _srActive <<= 1; } if (_pixBufferBorderIndex >= PixBorderBufferSize) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index aa6976c443..adda9d09ae 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -22,14 +22,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _borderR; private int _borderT; private int[] _bufferC; - private int[] _bufferG; private int _cycle; private int _cycleIndex; private bool _columnSelect; private int _dataC; private int _dataG; private bool _displayEnable; - private int _displayC; private bool _enableIntLightPen; private bool _enableIntRaster; private bool _enableIntSpriteCollision; @@ -72,7 +70,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private readonly Sprite _sprite6; private readonly Sprite _sprite7; private readonly Sprite[] _sprites; - private int _sr; private bool _vblank; private int _vblankEnd; private int _vblankStart; @@ -130,7 +127,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _spriteSpriteCollisionClearPending = false; _spriteMulticolor0 = 0; _spriteMulticolor1 = 0; - _sr = 0; _vc = 0; _vcbase = 0; _vmli = 0; @@ -149,7 +145,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS for (var i = 0; i < 40; i++) { _bufferC[i] = 0; - _bufferG[i] = 0; } _pixBuffer = new int[PixBufferSize]; @@ -161,11 +156,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public void SyncState(Serializer ser) { - ser.Sync(nameof(_cyclesExecuted), ref _cyclesExecuted); ser.Sync(nameof(_parseIsSprCrunch), ref _parseIsSprCrunch); - ser.Sync(nameof(_srSync), ref _srSync); - ser.Sync(nameof(_srColorSync), ref _srColorSync); - ser.Sync(nameof(_srColorIndexLatch), ref _srColorIndexLatch); ser.Sync(nameof(_videoMode), ref _videoMode); ser.Sync(nameof(_borderOnShiftReg), ref _borderOnShiftReg); ser.Sync(nameof(_backgroundColor0), ref _backgroundColor0); @@ -186,14 +177,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_borderR), ref _borderR); ser.Sync(nameof(_borderT), ref _borderT); ser.Sync(nameof(_bufferC), ref _bufferC, useNull: false); - ser.Sync(nameof(_bufferG), ref _bufferG, useNull: false); ser.Sync(nameof(_cycle), ref _cycle); ser.Sync(nameof(_cycleIndex), ref _cycleIndex); ser.Sync(nameof(_columnSelect), ref _columnSelect); ser.Sync(nameof(_dataC), ref _dataC); ser.Sync(nameof(_dataG), ref _dataG); ser.Sync(nameof(_displayEnable), ref _displayEnable); - ser.Sync(nameof(_displayC), ref _displayC); ser.Sync(nameof(_enableIntLightPen), ref _enableIntLightPen); ser.Sync(nameof(_enableIntRaster), ref _enableIntRaster); ser.Sync(nameof(_enableIntSpriteCollision), ref _enableIntSpriteCollision); @@ -234,7 +223,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.EndSection(); } - ser.Sync(nameof(_sr), ref _sr); ser.Sync(nameof(_vc), ref _vc); ser.Sync(nameof(_vcbase), ref _vcbase); ser.Sync(nameof(_vmli), ref _vmli); @@ -245,7 +233,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_pixBufferIndex), ref _pixBufferIndex); ser.Sync(nameof(_pixBorderBuffer), ref _pixBorderBuffer, useNull: false); ser.Sync(nameof(_pixBufferBorderIndex), ref _pixBufferBorderIndex); - + if (ser.IsReader) { UpdateBorder(); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 621a0ea59b..fa0a8ec95c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -36,7 +36,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private readonly int _totalCycles; private readonly int _totalLines; - private int _cyclesExecuted; private int _hblankStartCheckXRaster; private int _hblankEndCheckXRaster; @@ -81,7 +80,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _sprite7 = _sprites[7]; _bufferC = new int[40]; - _bufferG = new int[40]; } private void ConfigureBlanking(int lines, int hblankStart, int hblankEnd, int vblankStart, int vblankEnd, @@ -228,7 +226,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _vc = 0; _badlineEnable = false; _refreshCounter = 0xFF; - _cyclesExecuted = 0; } } @@ -327,8 +324,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // must always come last UpdatePins(); - - _cyclesExecuted++; } private void UpdateBorder() From 89fa1534771cfa35830df50f9c1d29cdfb7a3ab5 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 06:55:55 -0500 Subject: [PATCH 25/53] VIC: Resolve background color registers separately to color matrix memory --- .../Computers/Commodore64/MOS/Vic.Parse.cs | 67 ++++++++----------- .../Computers/Commodore64/MOS/Vic.Render.cs | 33 +++++++-- .../Computers/Commodore64/MOS/Vic.State.cs | 1 + 3 files changed, 55 insertions(+), 46 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index c2aa36b04b..fc83f9257c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -46,6 +46,8 @@ private int _parsePixelDataIndex; private int _parseSrData0; private int _parseSrData1; + private int _parseSrShift; + private bool _parseSrColorEnable; private void ParseCycle() { @@ -106,8 +108,9 @@ } // graphics data shift register - _srData1 &= ~(0xFF << (7 - _xScroll)); - _srActive |= 0xFF << (7 - _xScroll); + _parseSrShift = 7 - _xScroll; + _srData1 &= ~(0xFF << _parseSrShift); + _srActive |= 0xFF << _parseSrShift; if (_multicolorMode && (_bitmapMode || (_dataC & 0x800) != 0)) { @@ -119,13 +122,13 @@ _parseSrData0 = _bitmapMode ? 0 : _dataG; _parseSrData1 = _dataG; } - _srData1 |= _parseSrData1 << (7 - _xScroll); - + _srData1 |= _parseSrData1 << _parseSrShift; + // graphics color shift register - _srColor0 &= ~(0xFF << (7 - _xScroll)); - _srColor1 &= ~(0xFF << (7 - _xScroll)); - _srColor2 &= ~(0xFF << (7 - _xScroll)); - _srColor3 &= ~(0xFF << (7 - _xScroll)); + _srColor0 &= ~(0xFF << _parseSrShift); + _srColor1 &= ~(0xFF << _parseSrShift); + _srColor2 &= ~(0xFF << _parseSrShift); + _srColor3 &= ~(0xFF << _parseSrShift); for (_parsePixelDataIndex = 7; _parsePixelDataIndex >= 0; _parsePixelDataIndex--) { _parsePixelData = ((_parseSrData0 & 0x80) >> 7) | ((_parseSrData1 & 0x80) >> 6); @@ -133,24 +136,20 @@ { case VideoMode000: case VideoMode001: - switch (_parsePixelData) + if (_parsePixelData == 3) { - case 0: - _pixel = _backgroundColor0; - break; - case 1: - _pixel = _idle ? 0 : _backgroundColor1; - break; - case 2: - _pixel = _idle ? 0 :_backgroundColor2; - break; - default: - _pixel = _idle ? 0 : (_multicolorMode ? _dataC & 0x700 : _dataC) >> 8; - break; + _pixel = _idle ? 0 : (_multicolorMode ? _dataC & 0x700 : _dataC) >> 8; + _parseSrColorEnable = true; + } + else + { + _pixel = _parsePixelData; + _parseSrColorEnable = false; } break; case VideoMode010: case VideoMode011: + _parseSrColorEnable = _parsePixelData != 0; switch (_parsePixelData) { case 0: @@ -171,25 +170,12 @@ if (_parsePixelData != 0) { _pixel = _idle ? 0 : _dataC >> 8; + _parseSrColorEnable = true; } else { _pixel = (_dataC & 0xC0) >> 6; - switch (_pixel) - { - case 0: - _pixel = _backgroundColor0; - break; - case 1: - _pixel = _idle ? 0 : _backgroundColor1; - break; - case 2: - _pixel = _idle ? 0 : _backgroundColor2; - break; - default: - _pixel = _idle ? 0 : _backgroundColor3; - break; - } + _parseSrColorEnable = false; } break; default: @@ -200,10 +186,11 @@ _parseSrData0 <<= 1; _parseSrData1 <<= 1; - _srColor0 |= (_pixel & 1) << (7 - _xScroll + _parsePixelDataIndex); - _srColor1 |= ((_pixel >> 1) & 1) << (7 - _xScroll + _parsePixelDataIndex); - _srColor2 |= ((_pixel >> 2) & 1) << (7 - _xScroll + _parsePixelDataIndex); - _srColor3 |= ((_pixel >> 3) & 1) << (7 - _xScroll + _parsePixelDataIndex); + _srColor0 |= (_pixel & 1) << (_parseSrShift + _parsePixelDataIndex); + _srColor1 |= ((_pixel >> 1) & 1) << (_parseSrShift + _parsePixelDataIndex); + _srColor2 |= ((_pixel >> 2) & 1) << (_parseSrShift + _parsePixelDataIndex); + _srColor3 |= ((_pixel >> 3) & 1) << (_parseSrShift + _parsePixelDataIndex); + _srColorEnable |= (_parseSrColorEnable ? 1 : 0) << (_parseSrShift + _parsePixelDataIndex); } break; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs index b1f2b12b38..57b911f1a6 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs @@ -17,6 +17,7 @@ private int _srColor3; private int _srData1; private int _srActive; + private int _srColorEnable; private int _videoMode; private int _borderOnShiftReg; @@ -65,12 +66,31 @@ #endregion // render graphics - _pixel = (_srActive & SrMask1) != 0 - ? ((_srColor0 & SrMask1) >> 18) | - ((_srColor1 & SrMask1) >> 17) | - ((_srColor2 & SrMask1) >> 16) | - ((_srColor3 & SrMask1) >> 15) - : _backgroundColor0; + if ((_srColorEnable & SrMask1) != 0) + { + _pixel = ((_srColor0 & SrMask1) >> 18) | + ((_srColor1 & SrMask1) >> 17) | + ((_srColor2 & SrMask1) >> 16) | + ((_srColor3 & SrMask1) >> 15); + } + else + { + switch (((_srColor0 & SrMask1) >> 18) | ((_srColor1 & SrMask1) >> 17)) + { + case 1: + _pixel = _backgroundColor1; + break; + case 2: + _pixel = _backgroundColor2; + break; + case 3: + _pixel = _backgroundColor3; + break; + default: + _pixel = _backgroundColor0; + break; + } + } // render sprites _pixelOwner = -1; @@ -203,6 +223,7 @@ _srColor3 <<= 1; _srData1 <<= 1; _srActive <<= 1; + _srColorEnable <<= 1; } if (_pixBufferBorderIndex >= PixBorderBufferSize) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index adda9d09ae..ecef548c29 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -51,6 +51,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _rasterInterruptLine; private bool _rasterInterruptTriggered; private int _rasterLine; + private int _rasterLineInterruptCompare; private int _rasterX; private bool _rasterXHold; private int _rc; From 83b6553749f1bcc7ddb89e9beb649146c5b343d1 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 06:58:13 -0500 Subject: [PATCH 26/53] VIC: Respect idle state background color registers, plus black in undocumented gfx mode --- .../Computers/Commodore64/MOS/Vic.Parse.cs | 1 + .../Computers/Commodore64/MOS/Vic.Render.cs | 6 +++--- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index fc83f9257c..af86811337 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -181,6 +181,7 @@ default: _parsePixelData = 0; _pixel = 0; + _parseSrColorEnable = true; break; } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs index 57b911f1a6..8c7f07d927 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs @@ -78,13 +78,13 @@ switch (((_srColor0 & SrMask1) >> 18) | ((_srColor1 & SrMask1) >> 17)) { case 1: - _pixel = _backgroundColor1; + _pixel = _idle ? 0 : _backgroundColor1; break; case 2: - _pixel = _backgroundColor2; + _pixel = _idle ? 0 : _backgroundColor2; break; case 3: - _pixel = _backgroundColor3; + _pixel = _idle ? 0 : _backgroundColor3; break; default: _pixel = _backgroundColor0; From a8fd85157c2df9d71ee5a70ef9b30fba0a0980cb Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 08:02:55 -0500 Subject: [PATCH 27/53] VIC: Use correct color mapping for non-multicolor bitmap mode --- .../Computers/Commodore64/MOS/Vic.Parse.cs | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index af86811337..68639f8f10 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -119,11 +119,13 @@ } else { - _parseSrData0 = _bitmapMode ? 0 : _dataG; - _parseSrData1 = _dataG; + _parseSrData0 = _parseSrData1 = _dataG; } _srData1 |= _parseSrData1 << _parseSrShift; + if (_bitmapMode && !_multicolorMode) + _parseSrData1 ^= 0xFF; + // graphics color shift register _srColor0 &= ~(0xFF << _parseSrShift); _srColor1 &= ~(0xFF << _parseSrShift); From 9758efe604337f3612910d607f1d850efa33d11c Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 19:46:33 -0500 Subject: [PATCH 28/53] 6502X: CPU does a read or write regardless if the result is trashed, even during reset and dummy pushes --- .../CPUs/MOS 6502X/Execute.cs | 33 ++++++++----------- 1 file changed, 14 insertions(+), 19 deletions(-) diff --git a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs index db65556c49..859ebc7ef5 100644 --- a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs +++ b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs @@ -524,13 +524,7 @@ namespace BizHawk.Emulation.Cores.Components.M6502 bool interrupt_pending; bool branch_irq_hack; //see Uop.RelBranch_Stage3 for more details - bool Interrupted - { - get - { - return RDY && (NMI || (IRQ && !FlagI)); - } - } + bool Interrupted => RDY && (NMI || (IRQ && !FlagI)); void FetchDummy() { @@ -560,10 +554,6 @@ namespace BizHawk.Emulation.Cores.Components.M6502 void Fetch1() { - rdy_freeze = !RDY; - if (!RDY) - return; - my_iflag = FlagI; FlagI = iflag_pending; if (!branch_irq_hack) @@ -672,15 +662,21 @@ namespace BizHawk.Emulation.Cores.Components.M6502 } void PushP_Reset() { - ea = ResetVector; - S--; - FlagI = true; - + rdy_freeze = !RDY; + if (RDY) + { + ea = ResetVector; + _link.DummyReadMemory((ushort)(S-- + 0x100)); + FlagI = true; + } } void PushDummy() { - S--; - + rdy_freeze = !RDY; + if (RDY) + { + _link.DummyReadMemory((ushort)(S-- + 0x100)); + } } void FetchPCLVector() { @@ -2702,7 +2698,6 @@ namespace BizHawk.Emulation.Cores.Components.M6502 mi = 0; iflag_pending = FlagI; ExecuteOneRetry(); - return; } void End_BranchSpecial() { @@ -2973,7 +2968,7 @@ namespace BizHawk.Emulation.Cores.Components.M6502 public void ExecuteOne() { - // total cycles now incraments every time a cycle is called to accurately count during RDY + // total cycles now increments every time a cycle is called to accurately count during RDY TotalExecutedCycles++; if (!rdy_freeze) { From 2abe832289dbea29593ada13ed94ab8f229160b8 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 20:52:51 -0500 Subject: [PATCH 29/53] C64: AEC does not prohibit the CPU from functioning, only BA (RDY) does --- .../Computers/Commodore64/C64.cs | 1 - .../Commodore64/MOS/Chip6510.IDebuggable.cs | 6 +-- .../MOS/Chip6510.IDisassemblable.cs | 2 +- .../Computers/Commodore64/MOS/Chip6510.cs | 46 ++++--------------- 4 files changed, 14 insertions(+), 41 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs index 926942bbe4..1592c35a51 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.cs @@ -207,7 +207,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 { _board.InputRead = false; _board.PollInput(); - _board.Cpu.LagCycles = 0; } _board.Execute(); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDebuggable.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDebuggable.cs index 67ba9a45c1..7dbe83fd05 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDebuggable.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDebuggable.cs @@ -97,7 +97,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private void StepOver() { - var instruction = CpuPeek(_cpu.PC); + var instruction = Peek(_cpu.PC); if (instruction == Jsr) { @@ -116,13 +116,13 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private void StepOut() { var instructionsBeforeBailout = 1000000; - var instr = CpuPeek(_cpu.PC); + var instr = Peek(_cpu.PC); _jsrCount = instr == Jsr ? 1 : 0; while (--instructionsBeforeBailout > 0) { StepInto(); - instr = CpuPeek(_cpu.PC); + instr = Peek(_cpu.PC); if (instr == Jsr) { _jsrCount++; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDisassemblable.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDisassemblable.cs index e5a9459c25..ccacc83988 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDisassemblable.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.IDisassemblable.cs @@ -27,7 +27,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public string Disassemble(MemoryDomain m, uint addr, out int length) { - return MOS6502X.Disassemble((ushort)addr, out length, CpuPeek); + return MOS6502X.Disassemble((ushort) addr, out length, a => unchecked((byte) Peek(a))); } } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index ba2ba3bf53..5d01e5edcf 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -67,25 +67,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS set { _cpu.TraceCallback = value; } } - public void SetOverflow() - { - } - - 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); - } - public void HardReset() { _cpu.NESSoftReset(); @@ -109,22 +90,15 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { _cpu.RDY = ReadRdy(); - if (ReadAec()) - { - _cpu.IRQ = !ReadIrq(); - _pinNmiLast = _thisNmi; - _thisNmi = ReadNmi(); - _cpu.NMI |= _pinNmiLast && !_thisNmi; - _cpu.ExecuteOne(); - } - else - { - LagCycles++; - } +// if (!ReadAec()) +// return; + _cpu.IRQ = !ReadIrq(); + _pinNmiLast = _thisNmi; + _thisNmi = ReadNmi(); + _cpu.NMI |= _pinNmiLast && !_thisNmi; + _cpu.ExecuteOne(); } - public int LagCycles; - internal bool AtInstructionStart() { return _cpu.AtInstructionStart(); @@ -216,7 +190,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS case 0x0001: return PortData; default: - return ReadMemory(addr); + return ReadAec() ? ReadMemory(addr) : 0xFF; } } @@ -233,7 +207,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.EndSection(); ser.Sync(nameof(_thisNmi), ref _thisNmi); - ser.Sync(nameof(LagCycles), ref LagCycles); } public void Write(int addr, int val) @@ -249,7 +222,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS WriteMemoryPort(addr, val); break; default: - WriteMemory(addr, val); + if (ReadAec()) + WriteMemory(addr, val); break; } } From b471fdc6926ee79c02a1c20fff6bc37fcd9f22da Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 20:53:54 -0500 Subject: [PATCH 30/53] C64: The CPU can trigger VIC badlines on its own (needed for VSP) --- .../Commodore64/MOS/Vic.Registers.cs | 21 +++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index 0e63f4c16f..be8b1293af 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -292,6 +292,27 @@ _extraColorMode = (val & 0x40) != 0; _rasterInterruptLine &= 0xFF; _rasterInterruptLine |= (val & 0x80) << 1; + + if (_rasterLine == FirstDmaLine) + _badlineEnable |= _displayEnable; + + if (_badlineEnable) + { + if ((_rasterLine & 0x7) == _yScroll) + { + _badline = true; + _idle = false; + } + else + { + _badline = false; + } + } + else + { + _badline = false; + } + UpdateBorder(); UpdateVideoMode(); break; From e63d10b608cdea5c0bf2a330220b7e81848bee0f Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 20:55:14 -0500 Subject: [PATCH 31/53] C64: Interrupts generated in phase 2 by the VIC won't trigger for the CPU until next cycle, also buffer BA --- .../Computers/Commodore64/MOS/Vic.Parse.cs | 8 ++++---- .../Computers/Commodore64/MOS/Vic.State.cs | 3 +++ .../Computers/Commodore64/MOS/Vic.cs | 17 ++++++++++------- 3 files changed, 17 insertions(+), 11 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 68639f8f10..33c5f29a48 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -2,7 +2,7 @@ { public sealed partial class Vic { - private const int BaResetCounter = 4; + private const int BaResetCounter = 6; private const int PipelineUpdateVc = 0x00000001; // vc/rc rule 2 private const int PipelineSpriteCrunch = 0x00000002; private const int PipelineUpdateMcBase = 0x00000004; @@ -336,13 +336,13 @@ } // perform BA flag manipulation - _pinBa = true; + _ba = true; switch (_parseBa) { case BaTypeNone: break; case BaTypeCharacter: - _pinBa = !_badline; + _ba = !_badline; break; default: _parseCycleBaSprite0 = _parseBa & BaTypeMaskSprite0; @@ -351,7 +351,7 @@ if ((_parseCycleBaSprite0 < 8 && _sprites[_parseCycleBaSprite0].Dma) || (_parseCycleBaSprite1 < 8 && _sprites[_parseCycleBaSprite1].Dma) || (_parseCycleBaSprite2 < 8 && _sprites[_parseCycleBaSprite2].Dma)) - _pinBa = false; + _ba = false; break; } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index ecef548c29..f30bde782b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -8,6 +8,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _backgroundColor1; private int _backgroundColor2; private int _backgroundColor3; + private bool _ba; private int _baCount; private bool _badline; private bool _badlineEnable; @@ -85,6 +86,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _pinAec = true; _pinBa = true; _pinIrq = true; + _ba = true; _bufOffset = 0; @@ -160,6 +162,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_parseIsSprCrunch), ref _parseIsSprCrunch); ser.Sync(nameof(_videoMode), ref _videoMode); ser.Sync(nameof(_borderOnShiftReg), ref _borderOnShiftReg); + ser.Sync(nameof(_ba), ref _ba); ser.Sync(nameof(_backgroundColor0), ref _backgroundColor0); ser.Sync(nameof(_backgroundColor1), ref _backgroundColor1); ser.Sync(nameof(_backgroundColor2), ref _backgroundColor2); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index fa0a8ec95c..0096a75089 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -310,20 +310,21 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // render ParseCycle(); + UpdateBa(); Render(); ParseCycle(); + UpdateBa(); + UpdatePins(); Render(); _extraColorModeBuffer = _extraColorMode; + } - // if the BA counter is nonzero, allow CPU bus access - if (_pinBa) + private void UpdateBa() + { + if (_ba) _baCount = BaResetCounter; - else if (_baCount > 0) + else if (_baCount >= 0) _baCount--; - _pinAec = _pinBa || _baCount > 0; - - // must always come last - UpdatePins(); } private void UpdateBorder() @@ -343,6 +344,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS (_enableIntLightPen & _intLightPen)); _pinIrq = irqTemp; + _pinAec = _ba || _baCount >= 0; + _pinBa = _ba; } private void UpdateVideoMode() From 3a135c7c2646e567f5ea0042445e0f252f87290a Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 21:40:03 -0500 Subject: [PATCH 32/53] C64: Raster interrupt bit can be set even if not enabled, just won't actually assert IRQ --- .../Computers/Commodore64/MOS/Vic.Registers.cs | 6 +----- .../Computers/Commodore64/MOS/Vic.State.cs | 1 - BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs | 8 +------- 3 files changed, 2 insertions(+), 13 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index be8b1293af..f8b8d78508 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -99,7 +99,7 @@ return 0x01 | ((_pointerVm & 0x3C00) >> 6) | ((_pointerCb & 0x7) << 1); case 0x19: - return 0x70 | (_rasterInterruptTriggered ? 0x01 : 0x00) | + return 0x70 | (_intRaster ? 0x01 : 0x00) | (_intSpriteDataCollision ? 0x02 : 0x00) | (_intSpriteCollision ? 0x04 : 0x00) | (_intLightPen ? 0x08 : 0x00) | @@ -207,11 +207,7 @@ case 0x19: // interrupts are cleared by writing a 1 if ((val & 0x01) != 0) - { _intRaster = false; - _rasterInterruptTriggered = false; - } - if ((val & 0x02) != 0) _intSpriteDataCollision = false; if ((val & 0x04) != 0) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index f30bde782b..69509feb91 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -207,7 +207,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_pointerCb), ref _pointerCb); ser.Sync(nameof(_pointerVm), ref _pointerVm); ser.Sync(nameof(_rasterInterruptLine), ref _rasterInterruptLine); - ser.Sync(nameof(_rasterInterruptTriggered), ref _rasterInterruptTriggered); ser.Sync(nameof(_rasterLine), ref _rasterLine); ser.Sync(nameof(_rasterX), ref _rasterX); ser.Sync(nameof(_rasterXHold), ref _rasterXHold); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 0096a75089..18c2ab846d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -262,13 +262,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // IRQ compares are done here if (_rasterLine == _rasterInterruptLine) { - _rasterInterruptTriggered = true; - - // interrupt needs to be enabled to be set to true - if (_enableIntRaster) - { - _intRaster = true; - } + _intRaster = true; } } From 0a7dc52aa08f10ec47dc2a25865835088af36846 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Tue, 9 Jul 2019 22:41:12 -0500 Subject: [PATCH 33/53] C64: BA and raster IRQ cleanup --- .../Computers/Commodore64/MOS/Vic.Parse.cs | 7 +++---- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs | 4 +--- 2 files changed, 4 insertions(+), 7 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 33c5f29a48..6337dc022f 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -336,10 +336,10 @@ } // perform BA flag manipulation - _ba = true; switch (_parseBa) { case BaTypeNone: + _ba = true; break; case BaTypeCharacter: _ba = !_badline; @@ -348,10 +348,9 @@ _parseCycleBaSprite0 = _parseBa & BaTypeMaskSprite0; _parseCycleBaSprite1 = (_parseBa & BaTypeMaskSprite1) >> 4; _parseCycleBaSprite2 = (_parseBa & BaTypeMaskSprite2) >> 8; - if ((_parseCycleBaSprite0 < 8 && _sprites[_parseCycleBaSprite0].Dma) || + _ba = !((_parseCycleBaSprite0 < 8 && _sprites[_parseCycleBaSprite0].Dma) || (_parseCycleBaSprite1 < 8 && _sprites[_parseCycleBaSprite1].Dma) || - (_parseCycleBaSprite2 < 8 && _sprites[_parseCycleBaSprite2].Dma)) - _ba = false; + (_parseCycleBaSprite2 < 8 && _sprites[_parseCycleBaSprite2].Dma)); break; } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 18c2ab846d..b1a27fe757 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -254,12 +254,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // start of rasterline if ((_cycle == RasterIrqLineXCycle && _rasterLine > 0) || (_cycle == RasterIrqLine0Cycle && _rasterLine == 0)) { - //_rasterInterruptTriggered = false; - if (_rasterLine == LastDmaLine) _badlineEnable = false; - // IRQ compares are done here + // raster compares are done here if (_rasterLine == _rasterInterruptLine) { _intRaster = true; From dbf6b39e7f69ab6479c08a02ddab0b1b9b6c711e Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Fri, 12 Jul 2019 23:51:55 -0500 Subject: [PATCH 34/53] C64: Split out VIC phase1/phase2 --- .../Computers/Commodore64/C64.Motherboard.cs | 3 ++- .../Computers/Commodore64/MOS/Vic.Parse.cs | 4 ++-- .../Computers/Commodore64/MOS/Vic.cs | 17 +++++++++++------ 3 files changed, 15 insertions(+), 9 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index a2b72bb9a3..6e518bb7e9 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 { _vicBank = (0x3 - ((Cia1.PrA | ~Cia1.DdrA) & 0x3)) << 14; - Vic.ExecutePhase(); + Vic.ExecutePhase1(); CartPort.ExecutePhase(); Cassette.ExecutePhase(); Serial.ExecutePhase(); @@ -164,6 +164,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 Cia0.ExecutePhase(); Cia1.ExecutePhase(); Cpu.ExecutePhase(); + Vic.ExecutePhase2(); } public void Flush() diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 6337dc022f..951d988b67 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -2,7 +2,7 @@ { public sealed partial class Vic { - private const int BaResetCounter = 6; + private const int BaResetCounter = 3; private const int PipelineUpdateVc = 0x00000001; // vc/rc rule 2 private const int PipelineSpriteCrunch = 0x00000002; private const int PipelineUpdateMcBase = 0x00000004; @@ -227,7 +227,7 @@ { _parseAddr = spr.Mc | (spr.Pointer << 6); spr.Sr |= ReadMemory(_parseAddr) << ((0x30 - (_parseFetch & 0x30)) >> 1); - spr.Mc++; + spr.Mc = (spr.Mc + 1) & 0x3F; spr.Loaded |= 0x800000; } else if ((_parseFetch & 0xF0) == 0x20) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index b1a27fe757..b5db242f94 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -188,7 +188,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public int CyclesPerSecond => _cyclesPerSec; - public void ExecutePhase() + public void ExecutePhase1() { // phi1 @@ -248,7 +248,16 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } _spriteSpriteCollisionClearPending = false; } + + // render + ParseCycle(); + UpdateBa(); + UpdatePins(); + Render(); + } + public void ExecutePhase2() + { // phi2 // start of rasterline @@ -302,15 +311,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // render ParseCycle(); - UpdateBa(); - Render(); - ParseCycle(); - UpdateBa(); UpdatePins(); Render(); _extraColorModeBuffer = _extraColorMode; } - + private void UpdateBa() { if (_ba) From e6871b2cc3046e84b9479cbefb7f439c48f9b035 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 00:27:08 -0500 Subject: [PATCH 35/53] C64: Move VIC raster IRQ to phase 1 --- .../Computers/Commodore64/MOS/Vic.cs | 22 +++++++++---------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index b5db242f94..c9cbc09747 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -249,17 +249,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _spriteSpriteCollisionClearPending = false; } - // render - ParseCycle(); - UpdateBa(); - UpdatePins(); - Render(); - } - - public void ExecutePhase2() - { - // phi2 - // start of rasterline if ((_cycle == RasterIrqLineXCycle && _rasterLine > 0) || (_cycle == RasterIrqLine0Cycle && _rasterLine == 0)) { @@ -273,6 +262,17 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } } + // render + ParseCycle(); + UpdateBa(); + UpdatePins(); + Render(); + } + + public void ExecutePhase2() + { + // phi2 + // check top and bottom border if (_rasterLine == _borderB) { From d62f2ac3fe0897bef9f0f0ae4979d28cadea4207 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 01:41:58 -0500 Subject: [PATCH 36/53] C64: 0F7 is a badline eligible raster (fixes 26-line text demo in Frodo test suite) --- .../Computers/Commodore64/MOS/Vic.Registers.cs | 2 +- .../Computers/Commodore64/MOS/Vic.TimingBuilder.cs | 4 ++-- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs | 4 ++-- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index f8b8d78508..98985bc44a 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -289,7 +289,7 @@ _rasterInterruptLine &= 0xFF; _rasterInterruptLine |= (val & 0x80) << 1; - if (_rasterLine == FirstDmaLine) + if (_rasterLine == BadLineEnableRaster) _badlineEnable |= _displayEnable; if (_badlineEnable) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.TimingBuilder.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.TimingBuilder.cs index 4a3715b4d1..9a3109db89 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.TimingBuilder.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.TimingBuilder.cs @@ -13,8 +13,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private const int BorderTop24 = 0x037; private const int BorderBottom25 = 0x0FB; private const int BorderBottom24 = 0x0F7; - private const int FirstDmaLine = 0x030; - private const int LastDmaLine = 0x0F7; + private const int BadLineEnableRaster = 0x030; + private const int BadLineDisableRaster = 0x0F8; // The special actions taken by the Vic are in the same order and interval on all chips, just different offsets. private static readonly int[] TimingBuilderCycle14Act = diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index c9cbc09747..4c69d69863 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -252,7 +252,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // start of rasterline if ((_cycle == RasterIrqLineXCycle && _rasterLine > 0) || (_cycle == RasterIrqLine0Cycle && _rasterLine == 0)) { - if (_rasterLine == LastDmaLine) + if (_rasterLine == BadLineDisableRaster) _badlineEnable = false; // raster compares are done here @@ -284,7 +284,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } // display enable compare - if (_rasterLine == FirstDmaLine) + if (_rasterLine == BadLineEnableRaster) { _badlineEnable |= _displayEnable; } From bf2cba0e2353cf21ac50aec51bd469fa1f3836af Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 11:38:03 -0500 Subject: [PATCH 37/53] 6502X: remove a comment (this is indeed a dummy fetch) --- BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs index 859ebc7ef5..2d57892446 100644 --- a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs +++ b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs @@ -1852,7 +1852,7 @@ namespace BizHawk.Emulation.Cores.Components.M6502 rdy_freeze = !RDY; if (RDY) { - _link.ReadMemory(opcode2); //dummy? + _link.DummyReadMemory(opcode2); alu_temp = (opcode2 + X) & 0xFF; } From d39f3e2e61ee6c9be40c52e9f1d8d8fde3bdde84 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 12:31:09 -0500 Subject: [PATCH 38/53] 6502X: pending IRQs are not delayed when !RDY is asserted --- BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs | 15 +++++++++------ .../Computers/Commodore64/MOS/Chip6510.cs | 3 --- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs index 2d57892446..d0cd3cab16 100644 --- a/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs +++ b/BizHawk.Emulation.Cores/CPUs/MOS 6502X/Execute.cs @@ -803,18 +803,24 @@ namespace BizHawk.Emulation.Cores.Components.M6502 } void Imp_SEI() { + // not affected by RDY + iflag_pending = true; + rdy_freeze = !RDY; if (RDY) { - FetchDummy(); iflag_pending = true; + FetchDummy(); } } void Imp_CLI() { + // not affected by RDY + iflag_pending = false; + rdy_freeze = !RDY; if (RDY) { - FetchDummy(); iflag_pending = false; + FetchDummy(); } } void Imp_SEC() @@ -2970,10 +2976,7 @@ namespace BizHawk.Emulation.Cores.Components.M6502 { // total cycles now increments every time a cycle is called to accurately count during RDY TotalExecutedCycles++; - if (!rdy_freeze) - { - interrupt_pending |= Interrupted; - } + interrupt_pending |= Interrupted; rdy_freeze = false; //i tried making ExecuteOneRetry not re-entrant by having it set a flag instead, then exit from the call below, check the flag, and GOTO if it was flagged, but it wasnt faster diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index 5d01e5edcf..9d4954a2cb 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -89,9 +89,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public void ExecutePhase() { _cpu.RDY = ReadRdy(); - -// if (!ReadAec()) -// return; _cpu.IRQ = !ReadIrq(); _pinNmiLast = _thisNmi; _thisNmi = ReadNmi(); From 3369dbf43f62a135574764eb4793d665564edd79 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 12:51:39 -0500 Subject: [PATCH 39/53] C64: IRQ is implemented as a delay line; no delay added (yet) --- .../Computers/Commodore64/MOS/Vic.Registers.cs | 2 +- .../Computers/Commodore64/MOS/Vic.State.cs | 5 ++--- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs | 8 +++++--- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index 98985bc44a..85fa2e8fbc 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -103,7 +103,7 @@ (_intSpriteDataCollision ? 0x02 : 0x00) | (_intSpriteCollision ? 0x04 : 0x00) | (_intLightPen ? 0x08 : 0x00) | - (_pinIrq ? 0x00 : 0x80); + ((_irqBuffer & 1) << 7); case 0x1A: return 0xF0 | (_enableIntRaster ? 0x01 : 0x00) | (_enableIntSpriteDataCollision ? 0x02 : 0x00) | diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index 69509feb91..a75f02e3db 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -46,7 +46,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private bool _multicolorMode; private bool _pinAec = true; private bool _pinBa = true; - private bool _pinIrq = true; private int _pointerCb; private int _pointerVm; private int _rasterInterruptLine; @@ -85,7 +84,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { _pinAec = true; _pinBa = true; - _pinIrq = true; + _irqBuffer = 0; _ba = true; _bufOffset = 0; @@ -203,7 +202,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_multicolorMode), ref _multicolorMode); ser.Sync(nameof(_pinAec), ref _pinAec); ser.Sync(nameof(_pinBa), ref _pinBa); - ser.Sync(nameof(_pinIrq), ref _pinIrq); + ser.Sync(nameof(_irqBuffer), ref _irqBuffer); ser.Sync(nameof(_pointerCb), ref _pointerCb); ser.Sync(nameof(_pointerVm), ref _pointerVm); ser.Sync(nameof(_rasterInterruptLine), ref _rasterInterruptLine); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 4c69d69863..85c3230826 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -26,7 +26,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public bool ReadAec() { return _pinAec; } public bool ReadBa() { return _pinBa; } - public bool ReadIrq() { return _pinIrq; } + public bool ReadIrq() { return (_irqBuffer & 1) != 0; } private readonly int _cyclesPerSec; private readonly int[] _rasterXPipeline; @@ -35,6 +35,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private readonly int[] _actPipeline; private readonly int _totalCycles; private readonly int _totalLines; + private int _irqBuffer; private int _hblankStartCheckXRaster; private int _hblankEndCheckXRaster; @@ -224,7 +225,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _rasterLine = 0; _vcbase = 0; _vc = 0; - _badlineEnable = false; _refreshCounter = 0xFF; } } @@ -340,7 +340,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS (_enableIntSpriteCollision & _intSpriteCollision) | (_enableIntLightPen & _intLightPen)); - _pinIrq = irqTemp; + // IRQ buffer is treated as a delay line + _irqBuffer >>= 1; + _irqBuffer |= irqTemp ? 0x1 : 0; _pinAec = _ba || _baCount >= 0; _pinBa = _ba; } From cae3340946164cc52badc6861697190b456a9156 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 12:53:34 -0500 Subject: [PATCH 40/53] C64: No need to expose these with the CPU link in place --- .../Computers/Commodore64/MOS/Chip6510.cs | 51 ------------------- 1 file changed, 51 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index 9d4954a2cb..e5bd61ff76 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -96,57 +96,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _cpu.ExecuteOne(); } - internal bool AtInstructionStart() - { - return _cpu.AtInstructionStart(); - } - - // ------------------------------------ - public ushort Pc - { - get - { - return _cpu.PC; - } - set - { - _cpu.PC = value; - } - } - - public int A - { - get { return _cpu.A; } - set { _cpu.A = unchecked((byte)value); } - } - - public int X - { - get { return _cpu.X; } - set { _cpu.X = unchecked((byte)value); } - } - - public int Y - { - get { return _cpu.Y; } - set { _cpu.Y = unchecked((byte)value); } - } - - public int S - { - get { return _cpu.S; } - set { _cpu.S = unchecked((byte)value); } - } - - public bool FlagC => _cpu.FlagC; - public bool FlagZ => _cpu.FlagZ; - public bool FlagI => _cpu.FlagI; - public bool FlagD => _cpu.FlagD; - public bool FlagB => _cpu.FlagB; - public bool FlagV => _cpu.FlagV; - public bool FlagN => _cpu.FlagN; - public bool FlagT => _cpu.FlagT; - public int Peek(int addr) { switch (addr) From f18e7c8833f1aaf5535de74ee55e602632bf6285 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 13:15:50 -0500 Subject: [PATCH 41/53] C64: Make the system debuggable for once --- .../Computers/Commodore64/C64.Motherboard.cs | 3 +++ 1 file changed, 3 insertions(+) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 6e518bb7e9..15e05954b7 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -79,6 +79,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 Pla = new Chip90611401(); Ram = new Chip4864(); Serial = new SerialPort(); + + Cpu.DebuggerStep = Execute; + DiskDrive.DebuggerStep = Execute; switch (sidType) { From bd20b355f07093d912e9ec03d7f71f10fbf7aa62 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 14:06:23 -0500 Subject: [PATCH 42/53] C64: Writing to CPU port writes open bus data to 00/01 --- .../Computers/Commodore64/C64.Motherboard.cs | 17 +++++++++-------- .../Commodore64/C64.MotherboardInterface.cs | 6 +++--- .../Computers/Commodore64/MOS/Chip6510.cs | 6 +++--- 3 files changed, 15 insertions(+), 14 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 15e05954b7..9151154a5a 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -38,8 +38,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public readonly Drive1541 DiskDrive; // state - //public int address; - public int Bus; public bool InputRead; public bool Irq; public bool Nmi; @@ -79,9 +77,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 Pla = new Chip90611401(); Ram = new Chip4864(); Serial = new SerialPort(); - - Cpu.DebuggerStep = Execute; - DiskDrive.DebuggerStep = Execute; switch (sidType) { @@ -149,6 +144,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 BasicRom = new Chip23128(); CharRom = new Chip23128(); KernalRom = new Chip23128(); + + if (Cpu != null) + Cpu.DebuggerStep = Execute; + if (DiskDrive != null) + DiskDrive.DebuggerStep = Execute; } public int ClockNumerator { get; } @@ -178,7 +178,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 // ----------------------------------------- public void HardReset() { - Bus = 0xFF; + _lastReadVicAddress = 0x3FFF; + _lastReadVicData = 0xFF; InputRead = false; Cia0.HardReset(); @@ -198,7 +199,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 public void SoftReset() { // equivalent to a hard reset EXCEPT cpu, color ram, memory - Bus = 0xFF; + _lastReadVicAddress = 0x3FFF; + _lastReadVicData = 0xFF; InputRead = false; Cia0.HardReset(); @@ -365,7 +367,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 ser.EndSection(); } - ser.Sync(nameof(Bus), ref Bus); ser.Sync(nameof(InputRead), ref InputRead); ser.Sync(nameof(Irq), ref Irq); ser.Sync(nameof(Nmi), ref Nmi); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs index 6ea70757ac..cfb9b5534d 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.MotherboardInterface.cs @@ -43,9 +43,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 return data; } - private void Cpu_WriteMemoryPort(int addr, int val) + private void Cpu_WriteMemoryPort(int addr) { - Pla.WriteMemory(addr, Bus); + Pla.WriteMemory(addr, ReadOpenBus()); } private bool Glue_ReadIRQ() @@ -84,7 +84,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 private int Pla_ReadColorRam(int addr) { - var result = Bus; + var result = ReadOpenBus(); result &= 0xF0; result |= ColorRam.Read(addr); return result; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index e5bd61ff76..a79632d865 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -44,7 +44,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public Func ReadMemory; public Func ReadPort; public Action WriteMemory; - public Action WriteMemoryPort; + public Action WriteMemoryPort; public Action DebuggerStep; @@ -161,11 +161,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { case 0x0000: _port.Direction = val; - WriteMemoryPort(addr, val); + WriteMemoryPort(addr); break; case 0x0001: _port.Latch = val; - WriteMemoryPort(addr, val); + WriteMemoryPort(addr); break; default: if (ReadAec()) From 76679bc8bcad031802ba3c3264f67879ae548a48 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 15:01:11 -0500 Subject: [PATCH 43/53] C64: Use the correct background color in bitmapped modes for 0 --- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 951d988b67..a348f499f9 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -111,6 +111,7 @@ _parseSrShift = 7 - _xScroll; _srData1 &= ~(0xFF << _parseSrShift); _srActive |= 0xFF << _parseSrShift; + _srColorEnable &= ~(0xFF << _parseSrShift); if (_multicolorMode && (_bitmapMode || (_dataC & 0x800) != 0)) { @@ -155,7 +156,7 @@ switch (_parsePixelData) { case 0: - _pixel = _backgroundColor0; + _pixel = 0; break; case 1: _pixel = _idle ? 0 : _dataC >> 4; From 894adbb61076efe70fcc9e85a477fb002aadc627 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 15:02:18 -0500 Subject: [PATCH 44/53] C64: Remove an unused variable --- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs | 1 - BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs | 2 -- 2 files changed, 3 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index a348f499f9..9d8ba3ce4f 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -110,7 +110,6 @@ // graphics data shift register _parseSrShift = 7 - _xScroll; _srData1 &= ~(0xFF << _parseSrShift); - _srActive |= 0xFF << _parseSrShift; _srColorEnable &= ~(0xFF << _parseSrShift); if (_multicolorMode && (_bitmapMode || (_dataC & 0x800) != 0)) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs index 8c7f07d927..4964ff4450 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Render.cs @@ -16,7 +16,6 @@ private int _srColor2; private int _srColor3; private int _srData1; - private int _srActive; private int _srColorEnable; private int _videoMode; private int _borderOnShiftReg; @@ -222,7 +221,6 @@ _srColor2 <<= 1; _srColor3 <<= 1; _srData1 <<= 1; - _srActive <<= 1; _srColorEnable <<= 1; } From f22c9b7abd61bb0dca111e5dea02e2679707184d Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 15:25:40 -0500 Subject: [PATCH 45/53] C64: CPU reads open bus when !AEC is asserted --- .../Computers/Commodore64/C64.Motherboard.cs | 1 + BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 9151154a5a..8e951246dc 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -234,6 +234,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 Cpu.ReadMemory = Pla.Read; Cpu.WriteMemory = Pla.Write; Cpu.WriteMemoryPort = Cpu_WriteMemoryPort; + Cpu.ReadBus = ReadOpenBus; Pla.PeekBasicRom = BasicRom.Peek; Pla.PeekCartridgeHi = CartPort.PeekHiRom; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index a79632d865..a4343f44fc 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -41,6 +41,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public Func ReadIrq; public Func ReadNmi; public Func ReadRdy; + public Func ReadBus; public Func ReadMemory; public Func ReadPort; public Action WriteMemory; @@ -136,7 +137,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS case 0x0001: return PortData; default: - return ReadAec() ? ReadMemory(addr) : 0xFF; + return ReadAec() ? ReadMemory(addr) : ReadBus(); } } From db38d5e65bc399dd1495633cf459f659d6aa9d38 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 15:28:57 -0500 Subject: [PATCH 46/53] C64: Try counting IRQ and BA correctly --- .../Computers/Commodore64/MOS/Vic.Registers.cs | 3 --- .../Computers/Commodore64/MOS/Vic.cs | 12 ++++++------ 2 files changed, 6 insertions(+), 9 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index 85fa2e8fbc..50424f36b5 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -214,7 +214,6 @@ _intSpriteCollision = false; if ((val & 0x08) != 0) _intLightPen = false; - UpdatePins(); break; case 0x1E: case 0x1F: @@ -358,14 +357,12 @@ _intSpriteDataCollision = (val & 0x02) != 0; _intSpriteCollision = (val & 0x04) != 0; _intLightPen = (val & 0x08) != 0; - UpdatePins(); break; case 0x1A: _enableIntRaster = (val & 0x01) != 0; _enableIntSpriteDataCollision = (val & 0x02) != 0; _enableIntSpriteCollision = (val & 0x04) != 0; _enableIntLightPen = (val & 0x08) != 0; - UpdatePins(); break; case 0x1B: _sprite0.Priority = (val & 0x01) != 0; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 85c3230826..fe987dd613 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -26,7 +26,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public bool ReadAec() { return _pinAec; } public bool ReadBa() { return _pinBa; } - public bool ReadIrq() { return (_irqBuffer & 1) != 0; } + public bool ReadIrq() { return (_irqBuffer & 1) == 0; } private readonly int _cyclesPerSec; private readonly int[] _rasterXPipeline; @@ -320,7 +320,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { if (_ba) _baCount = BaResetCounter; - else if (_baCount >= 0) + else if (_baCount > 0) _baCount--; } @@ -334,16 +334,16 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private void UpdatePins() { - var irqTemp = !( + var irqTemp = (_enableIntRaster & _intRaster) | (_enableIntSpriteDataCollision & _intSpriteDataCollision) | (_enableIntSpriteCollision & _intSpriteCollision) | - (_enableIntLightPen & _intLightPen)); + (_enableIntLightPen & _intLightPen); // IRQ buffer is treated as a delay line _irqBuffer >>= 1; - _irqBuffer |= irqTemp ? 0x1 : 0; - _pinAec = _ba || _baCount >= 0; + _irqBuffer |= irqTemp ? 0x8 : 0; + _pinAec = _ba || _baCount > 0; _pinBa = _ba; } From 154eefd2ad293de126f4c9ecc8b2fa522066d15d Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 16:51:30 -0500 Subject: [PATCH 47/53] C64: Give BA/IRQ counting another go, seems to resolve many issues --- .../Computers/Commodore64/MOS/Chip6510.cs | 5 ++++- .../Computers/Commodore64/MOS/Vic.Parse.cs | 2 +- BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs | 4 ++-- 3 files changed, 7 insertions(+), 4 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index a4343f44fc..87825957b9 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -137,7 +137,10 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS case 0x0001: return PortData; default: - return ReadAec() ? ReadMemory(addr) : ReadBus(); + if (ReadAec()) + return ReadMemory(addr); + else + return ReadBus(); } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 9d8ba3ce4f..c593082dad 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -72,7 +72,7 @@ if (_badline) { _parseAddr = _pointerVm | _vc; - _dataC = ReadMemory(_parseAddr); + _dataC = _baCount >= 0 ? 0xFF : ReadMemory(_parseAddr); _dataC |= (ReadColorRam(_parseAddr) & 0xF) << 8; _bufferC[_vmli] = _dataC; } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index fe987dd613..4ba40d3fc1 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -320,7 +320,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { if (_ba) _baCount = BaResetCounter; - else if (_baCount > 0) + else if (_baCount >= 0) _baCount--; } @@ -343,7 +343,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // IRQ buffer is treated as a delay line _irqBuffer >>= 1; _irqBuffer |= irqTemp ? 0x8 : 0; - _pinAec = _ba || _baCount > 0; + _pinAec = _ba || _baCount >= 0; _pinBa = _ba; } From 3bbfb98fc29ea67ed8a1a5c35e36c7a4c88c10ce Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sat, 13 Jul 2019 19:28:44 -0500 Subject: [PATCH 48/53] C64: Split out VIC IRQ delays --- .../Computers/Commodore64/MOS/Vic.cs | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 4ba40d3fc1..4a8640429b 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -334,15 +334,15 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private void UpdatePins() { - var irqTemp = - (_enableIntRaster & _intRaster) | - (_enableIntSpriteDataCollision & _intSpriteDataCollision) | - (_enableIntSpriteCollision & _intSpriteCollision) | - (_enableIntLightPen & _intLightPen); + // IRQ is treated as a delay line + + var intIrq = (_enableIntRaster && _intRaster) ? 0x0008 : 0x0000; + var sdIrq = (_enableIntSpriteDataCollision & _intSpriteDataCollision) ? 0x0001 : 0x0000; + var ssIrq = (_enableIntSpriteCollision & _intSpriteCollision) ? 0x0001 : 0x0000; + var lpIrq = (_enableIntLightPen & _intLightPen) ? 0x0001 : 0x0000; - // IRQ buffer is treated as a delay line _irqBuffer >>= 1; - _irqBuffer |= irqTemp ? 0x8 : 0; + _irqBuffer |= intIrq | sdIrq | ssIrq | lpIrq; _pinAec = _ba || _baCount >= 0; _pinBa = _ba; } From e8902b829a1e912573099a555e673cd30836f702 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sun, 14 Jul 2019 10:43:52 -0500 Subject: [PATCH 49/53] C64: Apparently the 6502X core needs interrupts delayed by a cycle, do that with IRQ and NMI --- .../Computers/Commodore64/MOS/Chip6510.cs | 22 +++++++++---------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index 87825957b9..5773a8885e 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -11,9 +11,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { // ------------------------------------ private readonly MOS6502X _cpu; - private bool _pinNmiLast; private LatchedPort _port; - private bool _thisNmi; + private int _irqDelay; + private int _nmiDelay; private struct CpuLink : IMOS6502XLink { @@ -76,7 +76,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS Direction = 0x00, Latch = 0xFF }; - _pinNmiLast = true; } public void SoftReset() @@ -89,11 +88,13 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // ------------------------------------ public void ExecutePhase() { + _irqDelay >>= 1; + _nmiDelay >>= 1; + _irqDelay |= ReadIrq() ? 0x0 : 0x2; + _nmiDelay |= ReadNmi() ? 0x0 : 0x2; _cpu.RDY = ReadRdy(); - _cpu.IRQ = !ReadIrq(); - _pinNmiLast = _thisNmi; - _thisNmi = ReadNmi(); - _cpu.NMI |= _pinNmiLast && !_thisNmi; + _cpu.IRQ = (_irqDelay & 1) != 0; + _cpu.NMI |= (_nmiDelay & 3) == 2; _cpu.ExecuteOne(); } @@ -150,13 +151,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _cpu.SyncState(ser); ser.EndSection(); - ser.Sync(nameof(_pinNmiLast), ref _pinNmiLast); - ser.BeginSection(nameof(_port)); _port.SyncState(ser); ser.EndSection(); - - ser.Sync(nameof(_thisNmi), ref _thisNmi); + + ser.Sync(nameof(_irqDelay), ref _irqDelay); + ser.Sync(nameof(_nmiDelay), ref _nmiDelay); } public void Write(int addr, int val) From 8e8d3a6a1bf588f9368ad9ec190b6af4e1250cd6 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sun, 14 Jul 2019 10:44:56 -0500 Subject: [PATCH 50/53] C64: Writes to some registers on the VIC in phase 2 by the CPU should only take effect on the following cycle --- .../Commodore64/MOS/Vic.Registers.cs | 25 ++----------------- .../Computers/Commodore64/MOS/Vic.State.cs | 8 ++++-- .../Computers/Commodore64/MOS/Vic.cs | 4 ++- 3 files changed, 11 insertions(+), 26 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index 50424f36b5..428fd854ce 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -280,34 +280,13 @@ _sprite7.X = (_sprite7.X & 0xFF) | ((val & 0x80) << 1); break; case 0x11: - _yScroll = val & 0x07; + _yScrollNext = val & 0x07; _rowSelect = (val & 0x08) != 0; - _displayEnable = (val & 0x10) != 0; + _displayEnableNext = (val & 0x10) != 0; _bitmapMode = (val & 0x20) != 0; _extraColorMode = (val & 0x40) != 0; _rasterInterruptLine &= 0xFF; _rasterInterruptLine |= (val & 0x80) << 1; - - if (_rasterLine == BadLineEnableRaster) - _badlineEnable |= _displayEnable; - - if (_badlineEnable) - { - if ((_rasterLine & 0x7) == _yScroll) - { - _badline = true; - _idle = false; - } - else - { - _badline = false; - } - } - else - { - _badline = false; - } - UpdateBorder(); UpdateVideoMode(); break; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index a75f02e3db..ffdea63faa 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -49,9 +49,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _pointerCb; private int _pointerVm; private int _rasterInterruptLine; - private bool _rasterInterruptTriggered; private int _rasterLine; - private int _rasterLineInterruptCompare; private int _rasterX; private bool _rasterXHold; private int _rc; @@ -79,6 +77,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _vmli; private int _xScroll; private int _yScroll; + private int _yScrollNext; + private bool _displayEnableNext; public void HardReset() { @@ -104,6 +104,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _borderOnVertical = true; _columnSelect = false; _displayEnable = false; + _displayEnableNext = false; _enableIntLightPen = false; _enableIntRaster = false; _enableIntSpriteCollision = false; @@ -134,6 +135,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _vmli = 0; _xScroll = 0; _yScroll = 0; + _yScrollNext = 0; _cycle = 0; _cycleIndex = 0; @@ -186,6 +188,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_dataC), ref _dataC); ser.Sync(nameof(_dataG), ref _dataG); ser.Sync(nameof(_displayEnable), ref _displayEnable); + ser.Sync(nameof(_displayEnableNext), ref _displayEnableNext); ser.Sync(nameof(_enableIntLightPen), ref _enableIntLightPen); ser.Sync(nameof(_enableIntRaster), ref _enableIntRaster); ser.Sync(nameof(_enableIntSpriteCollision), ref _enableIntSpriteCollision); @@ -230,6 +233,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_vmli), ref _vmli); ser.Sync(nameof(_xScroll), ref _xScroll); ser.Sync(nameof(_yScroll), ref _yScroll); + ser.Sync(nameof(_yScrollNext), ref _yScrollNext); ser.Sync(nameof(_bufOffset), ref _bufOffset); ser.Sync(nameof(_pixBuffer), ref _pixBuffer, useNull: false); ser.Sync(nameof(_pixBufferIndex), ref _pixBufferIndex); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 4a8640429b..4bd3212e7c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -314,6 +314,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS UpdatePins(); Render(); _extraColorModeBuffer = _extraColorMode; + _yScroll = _yScrollNext; + _displayEnable = _displayEnableNext; } private void UpdateBa() @@ -336,7 +338,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS { // IRQ is treated as a delay line - var intIrq = (_enableIntRaster && _intRaster) ? 0x0008 : 0x0000; + var intIrq = (_enableIntRaster && _intRaster) ? 0x0002 : 0x0000; var sdIrq = (_enableIntSpriteDataCollision & _intSpriteDataCollision) ? 0x0001 : 0x0000; var ssIrq = (_enableIntSpriteCollision & _intSpriteCollision) ? 0x0001 : 0x0000; var lpIrq = (_enableIntLightPen & _intLightPen) ? 0x0001 : 0x0000; From 4d6ed8d6c80e97164290a9b4ba7dd3b7eed78138 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sun, 14 Jul 2019 16:32:53 -0500 Subject: [PATCH 51/53] C64: Savestate should include the new variables --- .../Computers/Commodore64/C64.Motherboard.cs | 2 - .../Computers/Commodore64/MOS/Chip6510.cs | 3 - .../Computers/Commodore64/MOS/Chip90611401.cs | 2 - .../Computers/Commodore64/MOS/Vic.Parse.cs | 2 +- .../Commodore64/MOS/Vic.Registers.cs | 29 +++++---- .../Computers/Commodore64/MOS/Vic.Sprite.cs | 6 ++ .../Computers/Commodore64/MOS/Vic.State.cs | 62 +++++++++---------- .../Computers/Commodore64/MOS/Vic.cs | 16 ++--- 8 files changed, 63 insertions(+), 59 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs index 8e951246dc..e519e1a0c3 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Motherboard.cs @@ -259,8 +259,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64 Pla.PokeMemory = Ram.Poke; Pla.PokeSid = Sid.Poke; Pla.PokeVic = Vic.Poke; - Pla.ReadAec = Vic.ReadAec; - Pla.ReadBa = Vic.ReadBa; Pla.ReadBasicRom = BasicRom.Read; Pla.ReadCartridgeHi = CartPort.ReadHiRom; Pla.ReadCartridgeLo = CartPort.ReadLoRom; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs index 5773a8885e..278c052e1e 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip6510.cs @@ -49,8 +49,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public Action DebuggerStep; - // ------------------------------------ - public Chip6510() { // configure cpu r/w @@ -85,7 +83,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _port.Latch = 0xFF; } - // ------------------------------------ public void ExecutePhase() { _irqDelay >>= 1; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip90611401.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip90611401.cs index d01280b092..75af8f84ac 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip90611401.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Chip90611401.cs @@ -31,8 +31,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public Action PokeMemory; public Action PokeSid; public Action PokeVic; - public Func ReadAec; - public Func ReadBa; public Func ReadBasicRom; public Func ReadCartridgeLo; public Func ReadCartridgeHi; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index c593082dad..4fdbd80f2c 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -97,7 +97,7 @@ _parseAddr = _rc | ((_dataC & 0xFF) << 3) | (_pointerCb << 11); } - if (_extraColorModeBuffer) + if (_extraColorMode) _parseAddr &= AddressMaskEc; _dataG = ReadMemory(_parseAddr); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index 428fd854ce..7ddd859c00 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -181,17 +181,22 @@ return 0xFF; } } - + public void Write(int addr, int val) { - addr &= 0x3F; - switch (addr) + _writtenData = val & 0xFF; + _writtenRegister = addr & 0x3F; + } + + private void DoWrite() + { + switch (_writtenRegister) { case 0x17: // vic-ii addendum rule 7 foreach (var spr in _sprites) { - if ((val & 1) == 0 && !spr.YCrunch) + if ((_writtenData & 1) == 0 && !spr.YCrunch) { if (_parseIsSprCrunch) { @@ -202,17 +207,17 @@ } } - WriteRegister(addr, val); + WriteRegister(_writtenRegister, _writtenData); break; case 0x19: // interrupts are cleared by writing a 1 - if ((val & 0x01) != 0) + if ((_writtenData & 0x01) != 0) _intRaster = false; - if ((val & 0x02) != 0) + if ((_writtenData & 0x02) != 0) _intSpriteDataCollision = false; - if ((val & 0x04) != 0) + if ((_writtenData & 0x04) != 0) _intSpriteCollision = false; - if ((val & 0x08) != 0) + if ((_writtenData & 0x08) != 0) _intLightPen = false; break; case 0x1E: @@ -239,7 +244,7 @@ // not connected break; default: - WriteRegister(addr, val); + WriteRegister(_writtenRegister, _writtenData); break; } } @@ -280,9 +285,9 @@ _sprite7.X = (_sprite7.X & 0xFF) | ((val & 0x80) << 1); break; case 0x11: - _yScrollNext = val & 0x07; + _yScroll = val & 0x07; _rowSelect = (val & 0x08) != 0; - _displayEnableNext = (val & 0x10) != 0; + _displayEnable = (val & 0x10) != 0; _bitmapMode = (val & 0x20) != 0; _extraColorMode = (val & 0x40) != 0; _rasterInterruptLine &= 0xFF; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Sprite.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Sprite.cs index 7fd16863de..50b88516af 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Sprite.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Sprite.cs @@ -12,6 +12,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public bool Display; public bool Dma; public bool Enable; + public int Index; public int Loaded; public int Mc; public int Mcbase; @@ -28,6 +29,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public bool YCrunch; public bool YExpand; + public Sprite(int index) + { + Index = index; + } + public void HardReset() { CollideData = false; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index ffdea63faa..3a1d153783 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -34,7 +34,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private bool _enableIntSpriteCollision; private bool _enableIntSpriteDataCollision; private bool _extraColorMode; - private bool _extraColorModeBuffer; private bool _hblank; private bool _idle; private bool _intLightPen; @@ -77,22 +76,17 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _vmli; private int _xScroll; private int _yScroll; - private int _yScrollNext; - private bool _displayEnableNext; + + private int _writtenRegister; + private int _writtenData; public void HardReset() { - _pinAec = true; - _pinBa = true; - _irqBuffer = 0; - _ba = true; - - _bufOffset = 0; - _backgroundColor0 = 0; _backgroundColor1 = 0; _backgroundColor2 = 0; _backgroundColor3 = 0; + _ba = true; _baCount = BaResetCounter; _badline = false; _badlineEnable = false; @@ -102,9 +96,13 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _borderColor = 0; _borderOnMain = true; _borderOnVertical = true; + _bufOffset = 0; _columnSelect = false; + _cycle = 0; + _cycleIndex = 0; + _dataC = 0; + _dataG = 0; _displayEnable = false; - _displayEnableNext = false; _enableIntLightPen = false; _enableIntRaster = false; _enableIntSpriteCollision = false; @@ -115,14 +113,18 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _intRaster = false; _intSpriteCollision = false; _intSpriteDataCollision = false; + _irqBuffer = 0; _lightPenX = 0; _lightPenY = 0; _multicolorMode = false; + _pinAec = true; + _pinBa = true; _pointerCb = 0; _pointerVm = 0; _rasterInterruptLine = 0; _rasterLine = 0; _rasterX = 0; + _rasterXHold = false; _rc = 7; _refreshCounter = 0xFF; _rowSelect = false; @@ -135,9 +137,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _vmli = 0; _xScroll = 0; _yScroll = 0; - _yScrollNext = 0; - _cycle = 0; - _cycleIndex = 0; + + _writtenRegister = -1; + _writtenData = 0; // reset sprites for (var i = 0; i < 8; i++) @@ -151,8 +153,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _bufferC[i] = 0; } - _pixBuffer = new int[PixBufferSize]; - _pixBorderBuffer = new int[PixBorderBufferSize]; _pixBufferIndex = 0; _pixBufferBorderIndex = 0; UpdateBorder(); @@ -160,9 +160,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public void SyncState(Serializer ser) { - ser.Sync(nameof(_parseIsSprCrunch), ref _parseIsSprCrunch); - ser.Sync(nameof(_videoMode), ref _videoMode); - ser.Sync(nameof(_borderOnShiftReg), ref _borderOnShiftReg); ser.Sync(nameof(_ba), ref _ba); ser.Sync(nameof(_backgroundColor0), ref _backgroundColor0); ser.Sync(nameof(_backgroundColor1), ref _backgroundColor1); @@ -178,34 +175,39 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_borderColor), ref _borderColor); ser.Sync(nameof(_borderL), ref _borderL); ser.Sync(nameof(_borderOnMain), ref _borderOnMain); + ser.Sync(nameof(_borderOnShiftReg), ref _borderOnShiftReg); ser.Sync(nameof(_borderOnVertical), ref _borderOnVertical); ser.Sync(nameof(_borderR), ref _borderR); ser.Sync(nameof(_borderT), ref _borderT); ser.Sync(nameof(_bufferC), ref _bufferC, useNull: false); + ser.Sync(nameof(_bufOffset), ref _bufOffset); ser.Sync(nameof(_cycle), ref _cycle); ser.Sync(nameof(_cycleIndex), ref _cycleIndex); ser.Sync(nameof(_columnSelect), ref _columnSelect); ser.Sync(nameof(_dataC), ref _dataC); ser.Sync(nameof(_dataG), ref _dataG); ser.Sync(nameof(_displayEnable), ref _displayEnable); - ser.Sync(nameof(_displayEnableNext), ref _displayEnableNext); ser.Sync(nameof(_enableIntLightPen), ref _enableIntLightPen); ser.Sync(nameof(_enableIntRaster), ref _enableIntRaster); ser.Sync(nameof(_enableIntSpriteCollision), ref _enableIntSpriteCollision); ser.Sync(nameof(_enableIntSpriteDataCollision), ref _enableIntSpriteDataCollision); ser.Sync(nameof(_extraColorMode), ref _extraColorMode); - ser.Sync(nameof(_extraColorModeBuffer), ref _extraColorModeBuffer); ser.Sync(nameof(_idle), ref _idle); ser.Sync(nameof(_intLightPen), ref _intLightPen); ser.Sync(nameof(_intRaster), ref _intRaster); ser.Sync(nameof(_intSpriteCollision), ref _intSpriteCollision); ser.Sync(nameof(_intSpriteDataCollision), ref _intSpriteDataCollision); + ser.Sync(nameof(_irqBuffer), ref _irqBuffer); ser.Sync(nameof(_lightPenX), ref _lightPenX); ser.Sync(nameof(_lightPenY), ref _lightPenY); ser.Sync(nameof(_multicolorMode), ref _multicolorMode); ser.Sync(nameof(_pinAec), ref _pinAec); ser.Sync(nameof(_pinBa), ref _pinBa); - ser.Sync(nameof(_irqBuffer), ref _irqBuffer); + ser.Sync(nameof(_parseIsSprCrunch), ref _parseIsSprCrunch); + ser.Sync(nameof(_pixBorderBuffer), ref _pixBorderBuffer, useNull: false); + ser.Sync(nameof(_pixBufferBorderIndex), ref _pixBufferBorderIndex); + ser.Sync(nameof(_pixBuffer), ref _pixBuffer, useNull: false); + ser.Sync(nameof(_pixBufferIndex), ref _pixBufferIndex); ser.Sync(nameof(_pointerCb), ref _pointerCb); ser.Sync(nameof(_pointerVm), ref _pointerVm); ser.Sync(nameof(_rasterInterruptLine), ref _rasterInterruptLine); @@ -221,29 +223,25 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_spriteMulticolor0), ref _spriteMulticolor0); ser.Sync(nameof(_spriteMulticolor1), ref _spriteMulticolor1); - for (int i = 0; i < _sprites.Length; i++) + foreach (var sprite in _sprites) { - ser.BeginSection("Sprite" + i); - _sprites[i].SyncState(ser); + ser.BeginSection($"Sprite{sprite.Index}"); + sprite.SyncState(ser); ser.EndSection(); } ser.Sync(nameof(_vc), ref _vc); ser.Sync(nameof(_vcbase), ref _vcbase); + ser.Sync(nameof(_videoMode), ref _videoMode); ser.Sync(nameof(_vmli), ref _vmli); + ser.Sync(nameof(_writtenData), ref _writtenData); + ser.Sync(nameof(_writtenRegister), ref _writtenRegister); ser.Sync(nameof(_xScroll), ref _xScroll); ser.Sync(nameof(_yScroll), ref _yScroll); - ser.Sync(nameof(_yScrollNext), ref _yScrollNext); - ser.Sync(nameof(_bufOffset), ref _bufOffset); - ser.Sync(nameof(_pixBuffer), ref _pixBuffer, useNull: false); - ser.Sync(nameof(_pixBufferIndex), ref _pixBufferIndex); - ser.Sync(nameof(_pixBorderBuffer), ref _pixBorderBuffer, useNull: false); - ser.Sync(nameof(_pixBufferBorderIndex), ref _pixBufferBorderIndex); if (ser.IsReader) { UpdateBorder(); - UpdatePins(); UpdateVideoMode(); } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 4bd3212e7c..2cef75eac6 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -67,9 +67,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _sprites = new Sprite[8]; for (var i = 0; i < 8; i++) - { - _sprites[i] = new Sprite(); - } + _sprites[i] = new Sprite(i); _sprite0 = _sprites[0]; _sprite1 = _sprites[1]; @@ -79,8 +77,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _sprite5 = _sprites[5]; _sprite6 = _sprites[6]; _sprite7 = _sprites[7]; - _bufferC = new int[40]; + _pixBuffer = new int[PixBufferSize]; + _pixBorderBuffer = new int[PixBorderBufferSize]; } private void ConfigureBlanking(int lines, int hblankStart, int hblankEnd, int vblankStart, int vblankEnd, @@ -313,9 +312,12 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ParseCycle(); UpdatePins(); Render(); - _extraColorModeBuffer = _extraColorMode; - _yScroll = _yScrollNext; - _displayEnable = _displayEnableNext; + + if (_writtenRegister >= 0) + { + DoWrite(); + _writtenRegister = -1; + } } private void UpdateBa() From a119420c792fa7499bdb792f491989221072acdf Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Sun, 14 Jul 2019 20:22:07 -0500 Subject: [PATCH 52/53] C64: VC count enable seems to need to be delayed by 1 cycle after badline - which doesn't affect normal operation - which DOES affect VSP --- .../Computers/Commodore64/MOS/Vic.Parse.cs | 2 +- .../Commodore64/MOS/Vic.Registers.cs | 23 ++++++++----------- .../Computers/Commodore64/MOS/Vic.State.cs | 11 +++------ .../Computers/Commodore64/MOS/Vic.cs | 7 +----- 4 files changed, 15 insertions(+), 28 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs index 4fdbd80f2c..d514c320a8 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Parse.cs @@ -101,7 +101,7 @@ _parseAddr &= AddressMaskEc; _dataG = ReadMemory(_parseAddr); - if (!_idle) + if (!_idle && _vcEnable) { _vmli = (_vmli + 1) & 0x3F; _vc = (_vc + 1) & 0x3FF; diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs index 7ddd859c00..d6de85ab6a 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.Registers.cs @@ -184,19 +184,16 @@ public void Write(int addr, int val) { - _writtenData = val & 0xFF; - _writtenRegister = addr & 0x3F; - } + addr &= 0x3F; + val &= 0xFF; - private void DoWrite() - { - switch (_writtenRegister) + switch (addr) { case 0x17: // vic-ii addendum rule 7 foreach (var spr in _sprites) { - if ((_writtenData & 1) == 0 && !spr.YCrunch) + if ((val & 1) == 0 && !spr.YCrunch) { if (_parseIsSprCrunch) { @@ -207,17 +204,17 @@ } } - WriteRegister(_writtenRegister, _writtenData); + WriteRegister(addr, val); break; case 0x19: // interrupts are cleared by writing a 1 - if ((_writtenData & 0x01) != 0) + if ((val & 0x01) != 0) _intRaster = false; - if ((_writtenData & 0x02) != 0) + if ((val & 0x02) != 0) _intSpriteDataCollision = false; - if ((_writtenData & 0x04) != 0) + if ((val & 0x04) != 0) _intSpriteCollision = false; - if ((_writtenData & 0x08) != 0) + if ((val & 0x08) != 0) _intLightPen = false; break; case 0x1E: @@ -244,7 +241,7 @@ // not connected break; default: - WriteRegister(_writtenRegister, _writtenData); + WriteRegister(addr, val); break; } } diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs index 3a1d153783..cca8532705 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.State.cs @@ -73,13 +73,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS private int _vblankStart; private int _vc; private int _vcbase; + private bool _vcEnable; private int _vmli; private int _xScroll; private int _yScroll; - private int _writtenRegister; - private int _writtenData; - public void HardReset() { _backgroundColor0 = 0; @@ -134,13 +132,11 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS _spriteMulticolor1 = 0; _vc = 0; _vcbase = 0; + _vcEnable = false; _vmli = 0; _xScroll = 0; _yScroll = 0; - _writtenRegister = -1; - _writtenData = 0; - // reset sprites for (var i = 0; i < 8; i++) { @@ -232,10 +228,9 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ser.Sync(nameof(_vc), ref _vc); ser.Sync(nameof(_vcbase), ref _vcbase); + ser.Sync(nameof(_vcEnable), ref _vcEnable); ser.Sync(nameof(_videoMode), ref _videoMode); ser.Sync(nameof(_vmli), ref _vmli); - ser.Sync(nameof(_writtenData), ref _writtenData); - ser.Sync(nameof(_writtenRegister), ref _writtenRegister); ser.Sync(nameof(_xScroll), ref _xScroll); ser.Sync(nameof(_yScroll), ref _yScroll); diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs index 2cef75eac6..c0fc6740bb 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Vic.cs @@ -289,6 +289,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } // badline compare + _vcEnable = !_idle; if (_badlineEnable) { if ((_rasterLine & 0x7) == _yScroll) @@ -312,12 +313,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS ParseCycle(); UpdatePins(); Render(); - - if (_writtenRegister >= 0) - { - DoWrite(); - _writtenRegister = -1; - } } private void UpdateBa() From 1e5fe55f3027c10dbdff66efb7b9f2b5e40178c1 Mon Sep 17 00:00:00 2001 From: SaxxonPike Date: Fri, 19 Jul 2019 19:09:08 -0500 Subject: [PATCH 53/53] C64: Don't reallocate the SID filter buffer every time (purely perf) --- .../Computers/Commodore64/MOS/Sid.cs | 25 +++++++++++-------- 1 file changed, 14 insertions(+), 11 deletions(-) diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs index 1ffb2e1985..b96573b995 100644 --- a/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs +++ b/BizHawk.Emulation.Cores/Computers/Commodore64/MOS/Sid.cs @@ -61,7 +61,8 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS public Func ReadPotX; public Func ReadPotY; - public RealFFT fft; + private RealFFT _fft; + private double[] _fftBuffer = new double[0]; private readonly int _cpuCyclesNum; private int _sampleCyclesNum; @@ -260,18 +261,20 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS } } - fft = new RealFFT(nsamp_2); + _fft = new RealFFT(nsamp_2); - double[] temp_buffer = new double[nsamp_2]; + // eventually this will settle on a single buffer size and stop reallocating + if (_fftBuffer.Length < nsamp_2) + Array.Resize(ref _fftBuffer, nsamp_2); // linearly interpolate the original sample set into the new denser sample set for (double i = 0; i < nsamp_2; i++) { - temp_buffer[(int)i] = _outputBufferFiltered[(int)Math.Floor((i / (nsamp_2-1) * (nsamp - 1))) + _filterIndex]; + _fftBuffer[(int)i] = _outputBufferFiltered[(int)Math.Floor((i / (nsamp_2-1) * (nsamp - 1))) + _filterIndex]; } // now we have everything we need to perform the FFT - fft.ComputeForward(temp_buffer); + _fft.ComputeForward(_fftBuffer); // for each element in the frequency list, attenuate it according to the specs for (int i = 1; i < nsamp_2; i++) @@ -282,7 +285,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS // let's assume that frequencies near the peak are doubled in strength at max resonance if ((1.2 > freq / loc_filterFrequency) && (freq / loc_filterFrequency > 0.8 )) { - temp_buffer[i] = temp_buffer[i] * (1 + (double)_filterResonance/15); + _fftBuffer[i] = _fftBuffer[i] * (1 + (double)_filterResonance/15); } // low pass filter @@ -291,7 +294,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS //attenuated at 12db per octave attenuation = Math.Log(freq / loc_filterFrequency, 2); attenuation = 12 * attenuation; - temp_buffer[i] = temp_buffer[i] * Math.Pow(2, -attenuation / 10); + _fftBuffer[i] = _fftBuffer[i] * Math.Pow(2, -attenuation / 10); } // High pass filter @@ -300,7 +303,7 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS //attenuated at 12db per octave attenuation = Math.Log(loc_filterFrequency / freq, 2); attenuation = 12 * attenuation; - temp_buffer[i] = temp_buffer[i] * Math.Pow(2, -attenuation / 10); + _fftBuffer[i] = _fftBuffer[i] * Math.Pow(2, -attenuation / 10); } // Band pass filter @@ -309,19 +312,19 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64.MOS //attenuated at 6db per octave attenuation = Math.Log(freq / loc_filterFrequency, 2); attenuation = 6 * attenuation; - temp_buffer[i] = temp_buffer[i] * Math.Pow(2, -Math.Abs(attenuation) / 10); + _fftBuffer[i] = _fftBuffer[i] * Math.Pow(2, -Math.Abs(attenuation) / 10); } } // now transform back into time space and reassemble the attenuated frequency components - fft.ComputeReverse(temp_buffer); + _fft.ComputeReverse(_fftBuffer); int temp = nsamp - 1; //re-sample back down to the original number of samples for (double i = 0; i < nsamp; i++) { - _outputBufferFiltered[(int)i + _filterIndex] = (int)(temp_buffer[(int)Math.Ceiling((i / (nsamp - 1) * (nsamp_2 - 1)))]/(nsamp_2/2)); + _outputBufferFiltered[(int)i + _filterIndex] = (int)(_fftBuffer[(int)Math.Ceiling((i / (nsamp - 1) * (nsamp_2 - 1)))]/(nsamp_2/2)); if (_outputBufferFiltered[(int)i + _filterIndex] < 0) {