diff --git a/src/BizHawk.Client.Common/rewind/ZwinderBuffer.cs b/src/BizHawk.Client.Common/rewind/ZwinderBuffer.cs index 3d5c32a04a..38fd29fb75 100644 --- a/src/BizHawk.Client.Common/rewind/ZwinderBuffer.cs +++ b/src/BizHawk.Client.Common/rewind/ZwinderBuffer.cs @@ -27,19 +27,19 @@ namespace BizHawk.Client.Common _sizeMask = Size - 1; _buffer = new byte[Size]; _targetFrameLength = settings.TargetFrameLength; - _states = new StateInfo[StateMask + 1]; + _states = new StateInfo[STATEMASK + 1]; _useCompression = settings.UseCompression; } /// /// Number of states that could be in the state ringbuffer, Mask for the state ringbuffer /// - private const int StateMask = 16383; + private const int STATEMASK = 16383; /// /// How many states are actually in the state ringbuffer /// - public int Count => (_nextStateIndex - _firstStateIndex) & StateMask; + public int Count => (_nextStateIndex - _firstStateIndex) & STATEMASK; /// /// total number of bytes used @@ -73,7 +73,7 @@ namespace BizHawk.Client.Common private readonly StateInfo[] _states; private int _firstStateIndex; private int _nextStateIndex; - private int HeadStateIndex => (_nextStateIndex - 1) & StateMask; + private int HeadStateIndex => (_nextStateIndex - 1) & STATEMASK; private readonly bool _useCompression; @@ -136,7 +136,7 @@ namespace BizHawk.Client.Common if (Count == 0) throw new IOException("A single state must not be larger than the buffer"); indexInvalidated?.Invoke(0); - _firstStateIndex = (_firstStateIndex + 1) & StateMask; + _firstStateIndex = (_firstStateIndex + 1) & STATEMASK; return Count > 0 ? (_states[_firstStateIndex].Start - start) & _sizeMask : Size; @@ -156,7 +156,7 @@ namespace BizHawk.Client.Common _states[_nextStateIndex].Frame = frame; _states[_nextStateIndex].Start = start; _states[_nextStateIndex].Size = (int)stream.Length; - _nextStateIndex = (_nextStateIndex + 1) & StateMask; + _nextStateIndex = (_nextStateIndex + 1) & STATEMASK; Util.DebugWriteLine($"Size: {Size >> 20}MiB, Used: {Used >> 20}MiB, States: {Count}"); } @@ -196,7 +196,7 @@ namespace BizHawk.Client.Common { if ((uint)index >= (uint)Count) throw new IndexOutOfRangeException(); - return new StateInformation(this, (index + _firstStateIndex) & StateMask); + return new StateInformation(this, (index + _firstStateIndex) & STATEMASK); } /// @@ -207,7 +207,7 @@ namespace BizHawk.Client.Common { if ((uint)index > (uint)Count) throw new IndexOutOfRangeException(); - _nextStateIndex = (index + _firstStateIndex) & StateMask; + _nextStateIndex = (index + _firstStateIndex) & STATEMASK; Util.DebugWriteLine($"Size: {Size >> 20}MiB, Used: {Used >> 20}MiB, States: {Count}"); } @@ -218,42 +218,43 @@ namespace BizHawk.Client.Common writer.Write(_targetFrameLength); writer.Write(_useCompression); - writer.Write(_buffer); - foreach (var s in _states) - { - writer.Write(s.Start); - writer.Write(s.Size); - writer.Write(s.Frame); - } - writer.Write(_firstStateIndex); - writer.Write(_nextStateIndex); + SaveStateBodyBinary(writer); } - // public void LoadStateBinary(BinaryReader reader) - // { - // if (reader.ReadInt64() != Size) - // throw new InvalidOperationException("Bad format"); - // if (reader.ReadInt64() != _sizeMask) - // throw new InvalidOperationException("Bad format"); - // if (reader.ReadInt32() != _targetFrameLength) - // throw new InvalidOperationException("Bad format"); - // if (reader.ReadBoolean() != _useCompression) - // throw new InvalidOperationException("Bad format"); - - // LoadStateBodyBinary(reader); - // } + private void SaveStateBodyBinary(BinaryWriter writer) + { + writer.Write(Count); + for (var i = _firstStateIndex; i != _nextStateIndex; i = (i + 1) & STATEMASK) + { + writer.Write(_states[i].Frame); + writer.Write(_states[i].Size); + } + if (Count != 0) + { + var startByte = _states[_firstStateIndex].Start; + var endByte = _states[HeadStateIndex].Start + _states[HeadStateIndex].Size; + if (startByte > endByte) + { + writer.BaseStream.Write(_buffer, (int)startByte, (int)(Size - startByte)); + startByte = 0; + } + writer.BaseStream.Write(_buffer, (int)startByte, (int)(endByte - startByte)); + } + } private void LoadStateBodyBinary(BinaryReader reader) { - reader.Read(_buffer, 0, _buffer.Length); - for (var i = 0; i < _states.Length; i++) - { - _states[i].Start = reader.ReadInt64(); - _states[i].Size = reader.ReadInt32(); - _states[i].Frame = reader.ReadInt32(); - } - _firstStateIndex = reader.ReadInt32(); + _firstStateIndex = 0; _nextStateIndex = reader.ReadInt32(); + long nextByte = 0; + for (var i = 0; i < _nextStateIndex; i++) + { + _states[i].Frame = reader.ReadInt32(); + _states[i].Size = reader.ReadInt32(); + _states[i].Start = nextByte; + nextByte += _states[i].Size; + } + reader.Read(_buffer, 0, (int)nextByte); } public static ZwinderBuffer Create(BinaryReader reader) diff --git a/src/BizHawk.Common/SpanStream.cs b/src/BizHawk.Common/SpanStream.cs index 8223613aac..a05bb35721 100644 --- a/src/BizHawk.Common/SpanStream.cs +++ b/src/BizHawk.Common/SpanStream.cs @@ -9,8 +9,8 @@ namespace BizHawk.Common /// public interface ISpanStream { - void Write (ReadOnlySpan buffer); - int Read (Span buffer); + void Write(ReadOnlySpan buffer); + int Read(Span buffer); } public static class SpanStream { diff --git a/src/BizHawk.Tests/Client.Common/Movie/ZwinderStateManagerTests.cs b/src/BizHawk.Tests/Client.Common/Movie/ZwinderStateManagerTests.cs index 658cd9ace6..ad7b0e3f17 100644 --- a/src/BizHawk.Tests/Client.Common/Movie/ZwinderStateManagerTests.cs +++ b/src/BizHawk.Tests/Client.Common/Movie/ZwinderStateManagerTests.cs @@ -25,6 +25,42 @@ namespace BizHawk.Tests.Client.Common.Movie Assert.AreEqual(zw.Settings.RecentBufferSize, zw2.Settings.RecentBufferSize); } + [TestMethod] + public void SaveCreateBufferRoundTrip() + { + var buff = new ZwinderBuffer(new RewindConfig + { + UseCompression = false, + BufferSize = 1, + TargetFrameLength = 10 + }); + var ss = new StateSource { PaddingData = new byte[500] }; + for (var frame = 0; frame < 2090; frame++) + { + ss.Frame = frame; + buff.Capture(frame, (s) => ss.SaveStateBinary(new BinaryWriter(s))); + } + // states are 504 bytes large, buffer is 1048576 bytes large + Assert.AreEqual(buff.Count, 2080); + Assert.AreEqual(buff.GetState(0).Frame, 10); + Assert.AreEqual(buff.GetState(2079).Frame, 2089); + Assert.AreEqual(StateSource.GetFrameNumberInState(buff.GetState(0).GetReadStream()), 10); + Assert.AreEqual(StateSource.GetFrameNumberInState(buff.GetState(2079).GetReadStream()), 2089); + + var ms = new MemoryStream(); + buff.SaveStateBinary(new BinaryWriter(ms)); + ms.Position = 0; + var buff2 = ZwinderBuffer.Create(new BinaryReader(ms)); + + Assert.AreEqual(buff.Size, buff2.Size); + Assert.AreEqual(buff.Used, buff2.Used); + Assert.AreEqual(buff2.Count, 2080); + Assert.AreEqual(buff2.GetState(0).Frame, 10); + Assert.AreEqual(buff2.GetState(2079).Frame, 2089); + Assert.AreEqual(StateSource.GetFrameNumberInState(buff2.GetState(0).GetReadStream()), 10); + Assert.AreEqual(StateSource.GetFrameNumberInState(buff2.GetState(2079).GetReadStream()), 2089); + } + [TestMethod] public void SomethingSomething() {