From 3c27a332bbd2cdf2ab3478983bb911bc75a39da8 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 14:31:06 -0500 Subject: [PATCH 01/27] create IStateManager interface and use that instead of TasStateManager --- .../BizHawk.Client.Common.csproj | 1 + .../movie/tasproj/IStateManager.cs | 62 +++++++++++++++++++ .../movie/tasproj/StateManagerDecay.cs | 4 +- .../movie/tasproj/StateManagerState.cs | 6 +- .../movie/tasproj/TasMovie.cs | 4 +- .../movie/tasproj/TasStateManager.cs | 4 +- 6 files changed, 72 insertions(+), 9 deletions(-) create mode 100644 BizHawk.Client.Common/movie/tasproj/IStateManager.cs diff --git a/BizHawk.Client.Common/BizHawk.Client.Common.csproj b/BizHawk.Client.Common/BizHawk.Client.Common.csproj index ec805f3bfe..99e67e062f 100644 --- a/BizHawk.Client.Common/BizHawk.Client.Common.csproj +++ b/BizHawk.Client.Common/BizHawk.Client.Common.csproj @@ -186,6 +186,7 @@ + diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs new file mode 100644 index 0000000000..b0bafb28d3 --- /dev/null +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -0,0 +1,62 @@ +using System; + +namespace BizHawk.Client.Common +{ + using System.Collections.Generic; + using System.IO; + + using BizHawk.Common; + + public interface IStateManager : IDisposable + { + // byte[] this[int frame] { get; } // TODO: I had it refactored to this back in the day + KeyValuePair this[int frame] { get; } + + TasStateManagerSettings Settings { get; set; } + + Action InvalidateCallback { set; } + + void Capture(bool force = false); + + bool HasState(int frame); + + bool Invalidate(int frame); + + // TODO: rename to Clear() + // TODO: consider it passing a bool if anything was cleared, and the .Any() could go away + void ClearStateHistory(); + + void Save(BinaryWriter bw); + + void Load(BinaryReader br); + + KeyValuePair GetStateClosestToFrame(int frame); + + bool Any(); + + // TODO: rename to Last + int LastStatedFrame { get; } + + // ********* Delete these ********** + void MountWriteAccess(); + + // TODO: delete me, I don't work + NDBDatabase NdbDatabase { get; } + + // *********** Reconsider these ************/ + void LimitStateCount(); + + void UpdateStateFrequency(); + + bool RemoveState(int frame); + + int LastEditedFrame { get; } + bool StateIsMarker(int frame); + + int StateCount { get; } + + int GetStateIndexByFrame(int frame); + + int GetStateFrameByIndex(int index); + } +} diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index f7b3130881..6f06f9f850 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -44,7 +44,7 @@ namespace BizHawk.Client.Common { internal class StateManagerDecay { - private TasStateManager _tsm; // access tsm methods to make life easier + private readonly IStateManager _tsm; // access tsm methods to make life easier private List _zeros; // amount of least significant zeros in bitwise view (also max pattern step) private int _bits; // size of _zeros is 2 raised to the power of _bits private int _mask; // for remainder calculation using bitwise instead of division @@ -53,7 +53,7 @@ namespace BizHawk.Client.Common private int _step; // initial memory state gap private bool _align; // extra care about fine alignment. TODO: do we want it? - public StateManagerDecay(TasStateManager tsm) + public StateManagerDecay(IStateManager tsm) { _tsm = tsm; _align = false; diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs index ae5255c80c..f045aa957b 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs @@ -9,14 +9,14 @@ namespace BizHawk.Client.Common internal class StateManagerState : IDisposable { private static long _stateId; - private readonly TasStateManager _manager; + private readonly IStateManager _manager; private readonly long _id; private byte[] _state; public int Frame { get; } - public static StateManagerState Read(BinaryReader r, TasStateManager m) + public static StateManagerState Read(BinaryReader r, IStateManager m) { int frame = r.ReadInt32(); byte[] data = r.ReadBytes(r.ReadInt32()); @@ -59,7 +59,7 @@ namespace BizHawk.Client.Common public bool IsOnDisk => _state == null; - public StateManagerState(TasStateManager manager, byte[] state, int frame) + public StateManagerState(IStateManager manager, byte[] state, int frame) { _manager = manager; _state = state; diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs index 3814fcdb11..08e8023487 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs @@ -13,7 +13,7 @@ namespace BizHawk.Client.Common public sealed partial class TasMovie : Bk2Movie, INotifyPropertyChanged { private readonly Bk2MnemonicConstants _mnemonics = new Bk2MnemonicConstants(); - private readonly TasStateManager _stateManager; + private readonly IStateManager _stateManager; private readonly TasLagLog _lagLog = new TasLagLog(); private readonly Dictionary _inputStateCache = new Dictionary(); private BackgroundWorker _progressReportWorker; @@ -37,7 +37,7 @@ namespace BizHawk.Client.Common public int BranchCount => Branches.Count; public int LastStatedFrame => _stateManager.LastStatedFrame; public override string PreferredExtension => Extension; - public TasStateManager TasStateManager => _stateManager; + public IStateManager TasStateManager => _stateManager; public TasMovieRecord this[int index] => new TasMovieRecord { diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 38b961c6c1..4ece710660 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -15,7 +15,7 @@ namespace BizHawk.Client.Common /// Captures savestates and manages the logic of adding, retrieving, /// invalidating/clearing of states. Also does memory management and limiting of states /// - public class TasStateManager : IDisposable + public class TasStateManager : IStateManager { // TODO: pass this in, and find a solution to a stale reference (this is instantiated BEFORE a new core instance is made, making this one stale if it is simply set in the constructor private IStatable Core => Global.Emulator.AsStatable(); @@ -27,7 +27,7 @@ namespace BizHawk.Client.Common InvalidateCallback?.Invoke(index); } - internal NDBDatabase NdbDatabase { get; set; } + public NDBDatabase NdbDatabase { get; private set; } private Guid _guid = Guid.NewGuid(); private SortedList _states = new SortedList(); From 8f116a84280cdbba00b5b4699df7af531ac98c78 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 14:35:31 -0500 Subject: [PATCH 02/27] Pass TasMovie into StateManagerDecay and have it get the last edited frame instead of from IStateManager --- BizHawk.Client.Common/movie/tasproj/IStateManager.cs | 1 - .../movie/tasproj/StateManagerDecay.cs | 11 +++++++---- .../movie/tasproj/TasStateManager.cs | 3 +-- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index b0bafb28d3..d63313f015 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -50,7 +50,6 @@ namespace BizHawk.Client.Common bool RemoveState(int frame); - int LastEditedFrame { get; } bool StateIsMarker(int frame); int StateCount { get; } diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index 6f06f9f850..4479e8c60e 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -44,7 +44,9 @@ namespace BizHawk.Client.Common { internal class StateManagerDecay { - private readonly IStateManager _tsm; // access tsm methods to make life easier + private readonly TasMovie _movie; + private readonly IStateManager _tsm; + private List _zeros; // amount of least significant zeros in bitwise view (also max pattern step) private int _bits; // size of _zeros is 2 raised to the power of _bits private int _mask; // for remainder calculation using bitwise instead of division @@ -53,8 +55,9 @@ namespace BizHawk.Client.Common private int _step; // initial memory state gap private bool _align; // extra care about fine alignment. TODO: do we want it? - public StateManagerDecay(IStateManager tsm) + public StateManagerDecay(TasMovie movie, IStateManager tsm) { + _movie = movie; _tsm = tsm; _align = false; } @@ -79,7 +82,7 @@ namespace BizHawk.Client.Common { continue; } - else if (currentFrame + 1 == _tsm.LastEditedFrame) + else if (currentFrame + 1 == _movie.LastEditedFrame) { continue; } @@ -124,7 +127,7 @@ namespace BizHawk.Client.Common { continue; } - else if ((currentFrame % _step > 0) && (currentFrame + 1 != _tsm.LastEditedFrame)) + else if ((currentFrame % _step > 0) && (currentFrame + 1 != _movie.LastEditedFrame)) { // ignore the pattern if the state doesn't belong already, drop it blindly and skip everything if (_tsm.RemoveState(currentFrame)) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 4ece710660..be8cd25dc0 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -62,7 +62,7 @@ namespace BizHawk.Client.Common SetState(0, _movie.BinarySavestate); } - _decay = new StateManagerDecay(this); + _decay = new StateManagerDecay(_movie, this); } public void Dispose() @@ -511,7 +511,6 @@ namespace BizHawk.Client.Common } public int StateCount => _states.Count; - public int LastEditedFrame => _movie.LastEditedFrame; public bool Any() { From cb4e71ebf7dfb147cb623ad9f49a0701c9cd31bd Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 14:40:18 -0500 Subject: [PATCH 03/27] rename a StateManager method, and delete some unused methods --- .../movie/tasproj/IStateManager.cs | 4 +-- .../movie/tasproj/StateManagerDecay.cs | 4 +-- .../movie/tasproj/TasStateManager.cs | 36 ++++--------------- 3 files changed, 11 insertions(+), 33 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index d63313f015..2524c33c32 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -37,6 +37,8 @@ namespace BizHawk.Client.Common // TODO: rename to Last int LastStatedFrame { get; } + bool IsMarkerState(int frame); + // ********* Delete these ********** void MountWriteAccess(); @@ -50,8 +52,6 @@ namespace BizHawk.Client.Common bool RemoveState(int frame); - bool StateIsMarker(int frame); - int StateCount { get; } int GetStateIndexByFrame(int frame); diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index 4479e8c60e..8769b1277b 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -78,7 +78,7 @@ namespace BizHawk.Client.Common { int currentFrame = _tsm.GetStateFrameByIndex(currentStateIndex); - if (_tsm.StateIsMarker(currentFrame)) + if (_tsm.IsMarkerState(currentFrame)) { continue; } @@ -123,7 +123,7 @@ namespace BizHawk.Client.Common { int currentFrame = _tsm.GetStateFrameByIndex(currentStateIndex); - if (_tsm.StateIsMarker(currentFrame)) + if (_tsm.IsMarkerState(currentFrame)) { continue; } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index be8cd25dc0..6de8df4211 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -150,7 +150,7 @@ namespace BizHawk.Client.Common /// /// Requests that the current emulator state be captured - /// Unless force is true, the state may or may not be captured depending on the logic employed by "greenzone" management + /// Unless force is true, the state may or may not be captured depending on the logic employed by "green-zone" management /// public void Capture(bool force = false) { @@ -165,13 +165,13 @@ namespace BizHawk.Client.Common { shouldCapture = force; } - else if (frame == 0) // For now, long term, TasMovie should have a .StartState property, and a tasproj file for the start state in non-savestate anchored movies + else if (frame == 0) // For now, long term, TasMovie should have a .StartState property, and a .tasproj file for the start state in non-savestate anchored movies { shouldCapture = true; } - else if (StateIsMarker(frame)) + else if (IsMarkerState(frame)) { - shouldCapture = true; // Markers shoudl always get priority + shouldCapture = true; // Markers should always get priority } else { @@ -184,18 +184,6 @@ namespace BizHawk.Client.Common } } - private void MoveStateToDisk(int index) - { - Used -= (ulong)_states[index].Length; - _states[index].MoveToDisk(); - } - - private void MoveStateToMemory(int index) - { - _states[index].MoveToRAM(); - Used += (ulong)_states[index].Length; - } - internal void SetState(int frame, byte[] state, bool skipRemoval = true) { if (!skipRemoval) // skipRemoval: false only when capturing new states @@ -252,7 +240,7 @@ namespace BizHawk.Client.Common return anyInvalidated; } - public bool StateIsMarker(int frame) + public bool IsMarkerState(int frame) { if (frame == -1) { @@ -310,7 +298,7 @@ namespace BizHawk.Client.Common { int frame = GetStateFrameByIndex(i); - if (StateIsMarker(frame) || frame % _fileStateGap < _stateFrequency) + if (IsMarkerState(frame) || frame % _fileStateGap < _stateFrequency) { continue; } @@ -339,7 +327,7 @@ namespace BizHawk.Client.Common break; } } - while (StateIsMarker(GetStateFrameByIndex(index))); + while (IsMarkerState(GetStateFrameByIndex(index))); if (index >= _states.Count) { @@ -547,15 +535,5 @@ namespace BizHawk.Client.Common return 0; } } - - private int FindState(StateManagerState s) - { - if (!_states.ContainsValue(s)) - { - return -1; - } - - return s.Frame; - } } } From e0a51eaab1298b603cd6fb78ddb824cd4a58eda0 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 14:49:43 -0500 Subject: [PATCH 04/27] Rename IStateManager.StateCount -> IStateManager.Count --- BizHawk.Client.Common/movie/tasproj/IStateManager.cs | 4 ++-- .../movie/tasproj/StateManagerDecay.cs | 4 ++-- .../movie/tasproj/TasStateManager.cs | 11 ++++++----- BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs | 2 +- 4 files changed, 11 insertions(+), 10 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index 2524c33c32..eb723f06a4 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -34,6 +34,8 @@ namespace BizHawk.Client.Common bool Any(); + int Count { get; } + // TODO: rename to Last int LastStatedFrame { get; } @@ -52,8 +54,6 @@ namespace BizHawk.Client.Common bool RemoveState(int frame); - int StateCount { get; } - int GetStateIndexByFrame(int frame); int GetStateFrameByIndex(int index); diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index 8769b1277b..e7b638d662 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -65,7 +65,7 @@ namespace BizHawk.Client.Common // todo: go through all states once, remove as many as we need. refactor to not need goto public void Trigger(int decayStates) { - for (; decayStates > 0 && _tsm.StateCount > 1;) + for (; decayStates > 0 && _tsm.Count > 1;) { int baseStateIndex = _tsm.GetStateIndexByFrame(Global.Emulator.Frame); int baseStateFrame = _tsm.GetStateFrameByIndex(baseStateIndex) / _step; // reduce right away @@ -119,7 +119,7 @@ namespace BizHawk.Client.Common } } - for (int currentStateIndex = _tsm.StateCount - 1; currentStateIndex > baseStateIndex; currentStateIndex--) + for (int currentStateIndex = _tsm.Count - 1; currentStateIndex > baseStateIndex; currentStateIndex--) { int currentFrame = _tsm.GetStateFrameByIndex(currentStateIndex); diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 6de8df4211..59c1065ef2 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -22,6 +22,7 @@ namespace BizHawk.Client.Common public Action InvalidateCallback { get; set; } + private void CallInvalidateCallback(int index) { InvalidateCallback?.Invoke(index); @@ -135,7 +136,7 @@ namespace BizHawk.Client.Common } } - public byte[] InitialState + private byte[] InitialState { get { @@ -281,9 +282,9 @@ namespace BizHawk.Client.Common /// public void LimitStateCount() { - if (StateCount + 1 > _maxStates || DiskUsed > (ulong)Settings.DiskCapacitymb * 1024 * 1024) + if (Count + 1 > _maxStates || DiskUsed > (ulong)Settings.DiskCapacitymb * 1024 * 1024) { - _decay.Trigger(StateCount + 1 - _maxStates); + _decay.Trigger(Count + 1 - _maxStates); } } @@ -498,7 +499,7 @@ namespace BizHawk.Client.Common } } - public int StateCount => _states.Count; + public int Count => _states.Count; public bool Any() { @@ -527,7 +528,7 @@ namespace BizHawk.Client.Common { get { - if (StateCount > 0) + if (Count > 0) { return LastKey; } diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs index c9a866b822..6f9a438201 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs @@ -995,7 +995,7 @@ namespace BizHawk.Client.EmuHawk private void SetSplicer() { // TODO: columns selected? - var temp = $"Selected: {TasView.SelectedRows.Count()} {(TasView.SelectedRows.Count() == 1 ? "frame" : "frames")}, States: {CurrentTasMovie.TasStateManager.StateCount}"; + var temp = $"Selected: {TasView.SelectedRows.Count()} {(TasView.SelectedRows.Count() == 1 ? "frame" : "frames")}, States: {CurrentTasMovie.TasStateManager.Count}"; if (_tasClipboard.Any()) temp += $", Clipboard: {_tasClipboard.Count} {(_tasClipboard.Count == 1 ? "frame" : "frames")}"; SplicerStatusLabel.Text = temp; } From 81ee00a1b6efdfc8196d26c0dab06f4ed5f2d393 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 14:56:03 -0500 Subject: [PATCH 05/27] cleanup --- .../movie/tasproj/StateManagerDecay.cs | 1 + .../movie/tasproj/TasStateManager.cs | 32 +++++++------------ BizHawk.sln.DotSettings | 5 ++- 3 files changed, 17 insertions(+), 21 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index e7b638d662..1d1792333f 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -42,6 +42,7 @@ using System.Collections.Generic; namespace BizHawk.Client.Common { + // TODO: interface me internal class StateManagerDecay { private readonly TasMovie _movie; diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 59c1065ef2..b62ebca844 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -2,7 +2,6 @@ using System.Collections.Generic; using System.IO; using System.Linq; -using System.Drawing; using BizHawk.Common; using BizHawk.Common.NumberExtensions; @@ -19,10 +18,10 @@ namespace BizHawk.Client.Common { // TODO: pass this in, and find a solution to a stale reference (this is instantiated BEFORE a new core instance is made, making this one stale if it is simply set in the constructor private IStatable Core => Global.Emulator.AsStatable(); + private readonly StateManagerDecay _decay; public Action InvalidateCallback { get; set; } - private void CallInvalidateCallback(int index) { InvalidateCallback?.Invoke(index); @@ -44,14 +43,13 @@ namespace BizHawk.Client.Common private bool _isMountedForWrite; private readonly TasMovie _movie; - private StateManagerDecay _decay; private ulong _expectedStateSize; private int _stateFrequency; private readonly int _minFrequency = 1; private readonly int _maxFrequency = 16; - private int _maxStates => (int)(Settings.Cap / _expectedStateSize) + + private int MaxStates => (int)(Settings.Cap / _expectedStateSize) + (int)((ulong)Settings.DiskCapacitymb * 1024 * 1024 / _expectedStateSize); - private int _fileStateGap => 1 << Settings.FileStateGap; + private int FileStateGap => 1 << Settings.FileStateGap; public TasStateManager(TasMovie movie) { @@ -78,7 +76,7 @@ namespace BizHawk.Client.Common ((int)_expectedStateSize / Settings.MemStateGapDivider / 1024), _minFrequency, _maxFrequency); - _decay.UpdateSettings(_maxStates, _stateFrequency, 4); + _decay.UpdateSettings(MaxStates, _stateFrequency, 4); } /// @@ -98,7 +96,7 @@ namespace BizHawk.Client.Common if (_expectedStateSize > 0) { - limit = _maxStates; + limit = MaxStates; } _states = new SortedList(limit); @@ -282,9 +280,9 @@ namespace BizHawk.Client.Common /// public void LimitStateCount() { - if (Count + 1 > _maxStates || DiskUsed > (ulong)Settings.DiskCapacitymb * 1024 * 1024) + if (Count + 1 > MaxStates || DiskUsed > (ulong)Settings.DiskCapacitymb * 1024 * 1024) { - _decay.Trigger(Count + 1 - _maxStates); + _decay.Trigger(Count + 1 - MaxStates); } } @@ -299,7 +297,7 @@ namespace BizHawk.Client.Common { int frame = GetStateFrameByIndex(i); - if (IsMarkerState(frame) || frame % _fileStateGap < _stateFrequency) + if (IsMarkerState(frame) || frame % FileStateGap < _stateFrequency) { continue; } @@ -373,16 +371,10 @@ namespace BizHawk.Client.Common { if (_states.Any()) { - var temp_state = _states.Values; - StateManagerState power = null; - if (temp_state[0].Frame==0) - { - power = _states.Values.First(s => s.Frame == 0); - } - else - { - power = _states.Values[0]; - } + var tempState = _states.Values; + var power = tempState[0].Frame == 0 + ? _states.Values.First(s => s.Frame == 0) + : _states.Values[0]; _states.Clear(); SetState(0, power.State); diff --git a/BizHawk.sln.DotSettings b/BizHawk.sln.DotSettings index a365a0b643..799d05463e 100644 --- a/BizHawk.sln.DotSettings +++ b/BizHawk.sln.DotSettings @@ -173,5 +173,8 @@ True True True - True + True + True + True + True \ No newline at end of file From ed85ce806b1c5f37b352ab4bd6f7355ac3c2389d Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 14:57:26 -0500 Subject: [PATCH 06/27] more cleanup --- BizHawk.Client.Common/movie/tasproj/TasStateManager.cs | 9 +++------ 1 file changed, 3 insertions(+), 6 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index b62ebca844..c3da5cf061 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -72,9 +72,8 @@ namespace BizHawk.Client.Common public void UpdateStateFrequency() { - _stateFrequency = NumberExtensions.Clamp( - ((int)_expectedStateSize / Settings.MemStateGapDivider / 1024), - _minFrequency, _maxFrequency); + _stateFrequency = ((int)_expectedStateSize / Settings.MemStateGapDivider / 1024) + .Clamp(_minFrequency, _maxFrequency); _decay.UpdateSettings(MaxStates, _stateFrequency, 4); } @@ -162,7 +161,7 @@ namespace BizHawk.Client.Common } else if (force) { - shouldCapture = force; + shouldCapture = true; } else if (frame == 0) // For now, long term, TasMovie should have a .StartState property, and a .tasproj file for the start state in non-savestate anchored movies { @@ -451,8 +450,6 @@ namespace BizHawk.Client.Common /// public int GetStateFrameByIndex(int index) { - // feos: this is called super often by decay - // this method is hundred times faster than _states.ElementAt(index).Key return _states.Keys[index]; } From b1296dd9bb0f37534d00d6387f4d3eb91e8234bf Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:13:08 -0500 Subject: [PATCH 07/27] rip out NDBDatabase --- .../movie/tasproj/IStateManager.cs | 3 - .../movie/tasproj/StateManagerState.cs | 88 +------- .../movie/tasproj/TasStateManager.cs | 74 +------ BizHawk.Common/BizHawk.Common.csproj | 1 - BizHawk.Common/NDBDatabase.cs | 200 ------------------ 5 files changed, 11 insertions(+), 355 deletions(-) delete mode 100644 BizHawk.Common/NDBDatabase.cs diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index eb723f06a4..f0a259fdb2 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -44,9 +44,6 @@ namespace BizHawk.Client.Common // ********* Delete these ********** void MountWriteAccess(); - // TODO: delete me, I don't work - NDBDatabase NdbDatabase { get; } - // *********** Reconsider these ************/ void LimitStateCount(); diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs index f045aa957b..44c1cb7970 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs @@ -8,102 +8,20 @@ namespace BizHawk.Client.Common /// internal class StateManagerState : IDisposable { - private static long _stateId; - private readonly IStateManager _manager; - private readonly long _id; - - private byte[] _state; - public int Frame { get; } - public static StateManagerState Read(BinaryReader r, IStateManager m) - { - int frame = r.ReadInt32(); - byte[] data = r.ReadBytes(r.ReadInt32()); - return new StateManagerState(m, data, frame); - } - - public void Write(BinaryWriter w) - { - w.Write(Frame); - w.Write(_state.Length); - w.Write(_state); - } - - public byte[] State - { - get - { - if (_state != null) - { - return _state; - } - - return _manager.NdbDatabase.FetchAll(_id.ToString()); - } - - set - { - if (_state != null) - { - _state = value; - } - else - { - throw new Exception("Attempted to set a state to null."); - } - } - } + public byte[] State { get; set; } public int Length => State.Length; - public bool IsOnDisk => _state == null; - - public StateManagerState(IStateManager manager, byte[] state, int frame) + public StateManagerState(byte[] state, int frame) { - _manager = manager; - _state = state; + State = state; Frame = frame; - - if (_stateId > long.MaxValue - 100) - { - throw new InvalidOperationException(); - } - - _id = System.Threading.Interlocked.Increment(ref _stateId); - } - - public void MoveToDisk() - { - if (IsOnDisk) - { - return; - } - - _manager.NdbDatabase.Store(_id.ToString(), _state, 0, _state.Length); - _state = null; - } - - public void MoveToRAM() - { - if (!IsOnDisk) - { - return; - } - - string key = _id.ToString(); - _state = _manager.NdbDatabase.FetchAll(key); - _manager.NdbDatabase.Release(key); } public void Dispose() { - if (!IsOnDisk) - { - return; - } - - _manager.NdbDatabase.Release(_id.ToString()); } } } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index c3da5cf061..432746ffe5 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -27,19 +27,8 @@ namespace BizHawk.Client.Common InvalidateCallback?.Invoke(index); } - public NDBDatabase NdbDatabase { get; private set; } - private Guid _guid = Guid.NewGuid(); private SortedList _states = new SortedList(); - private string StatePath - { - get - { - var basePath = PathManager.MakeAbsolutePath(Global.Config.PathEntries["Global", "TAStudio states"].Path, null); - return Path.Combine(basePath, _guid.ToString()); - } - } - private bool _isMountedForWrite; private readonly TasMovie _movie; @@ -66,8 +55,6 @@ namespace BizHawk.Client.Common public void Dispose() { - // States and BranchStates don't need cleaning because they would only contain an ndbdatabase entry which was demolished by the below - NdbDatabase?.Dispose(); } public void UpdateStateFrequency() @@ -104,8 +91,6 @@ namespace BizHawk.Client.Common { throw new InvalidOperationException(); } - - NdbDatabase = new NDBDatabase(StatePath, Settings.DiskCapacitymb * 1024 * 1024, (int)_expectedStateSize); } public TasStateManagerSettings Settings { get; set; } @@ -196,7 +181,7 @@ namespace BizHawk.Client.Common else { Used += (ulong)state.Length; - _states.Add(frame, new StateManagerState(this, state, frame)); + _states.Add(frame, new StateManagerState(state, frame)); } } @@ -257,16 +242,9 @@ namespace BizHawk.Client.Common return false; } - StateManagerState state = _states.Values[index]; + StateManagerState state = _states.Values[index]; // TODO: remove .Values here? - if (state.IsOnDisk) - { - state.Dispose(); - } - else - { - Used -= (ulong)state.Length; - } + Used -= (ulong)state.Length; _states.RemoveAt(index); @@ -279,7 +257,7 @@ namespace BizHawk.Client.Common /// public void LimitStateCount() { - if (Count + 1 > MaxStates || DiskUsed > (ulong)Settings.DiskCapacitymb * 1024 * 1024) + if (Count + 1 > MaxStates) { _decay.Trigger(Count + 1 - MaxStates); } @@ -288,7 +266,7 @@ namespace BizHawk.Client.Common private List ExcludeStates() { List ret = new List(); - ulong saveUsed = Used + DiskUsed; + ulong saveUsed = Used; // respect state gap no matter how small the resulting size will be // still leave marker states @@ -303,14 +281,7 @@ namespace BizHawk.Client.Common ret.Add(i); - if (_states.Values[i].IsOnDisk) - { - saveUsed -= _expectedStateSize; - } - else - { - saveUsed -= (ulong)_states.Values[i].Length; - } + saveUsed -= (ulong)_states.Values[i].Length; } // if the size is still too big, exclude states form the beginning @@ -333,15 +304,7 @@ namespace BizHawk.Client.Common } ret.Add(index); - - if (_states.Values[index].IsOnDisk) - { - saveUsed -= _expectedStateSize; - } - else - { - saveUsed -= (ulong)_states.Values[index].Length; - } + saveUsed -= (ulong)_states.Values[index].Length; } // if there are enough markers to still be over the limit, remove marker frames @@ -353,14 +316,7 @@ namespace BizHawk.Client.Common ret.Add(index); } - if (_states.Values[index].IsOnDisk) - { - saveUsed -= _expectedStateSize; - } - else - { - saveUsed -= (ulong)_states.Values[index].Length; - } + saveUsed -= (ulong)_states.Values[index].Length; } return ret; @@ -378,7 +334,6 @@ namespace BizHawk.Client.Common _states.Clear(); SetState(0, power.State); Used = (ulong)power.State.Length; - NdbDatabase?.Clear(); } } @@ -475,19 +430,6 @@ namespace BizHawk.Client.Common } } - private ulong DiskUsed - { - get - { - if (NdbDatabase == null) - { - return 0; - } - - return (ulong)NdbDatabase.Consumed; - } - } - public int Count => _states.Count; public bool Any() diff --git a/BizHawk.Common/BizHawk.Common.csproj b/BizHawk.Common/BizHawk.Common.csproj index 57821a4430..2a93d8a56f 100644 --- a/BizHawk.Common/BizHawk.Common.csproj +++ b/BizHawk.Common/BizHawk.Common.csproj @@ -84,7 +84,6 @@ - diff --git a/BizHawk.Common/NDBDatabase.cs b/BizHawk.Common/NDBDatabase.cs deleted file mode 100644 index cefeffccc6..0000000000 --- a/BizHawk.Common/NDBDatabase.cs +++ /dev/null @@ -1,200 +0,0 @@ -using System; -using System.IO; -using System.Collections.Generic; -using System.Linq; - -namespace BizHawk.Common -{ - /// - /// Non-consecutive Disk Block Database - /// Opens a file and stores blocks in it. - /// Blocks can be differently sized than the basic block size. Wastage will occur. - /// TODO: Mount on memory as well? - /// - public class NDBDatabase : IDisposable - { - readonly int BlockSize; - readonly long BlockCount; - - Dictionary Items = new Dictionary(); - LinkedList FreeList = new LinkedList(); - long FreeWatermark; - FileStream Stream; - - class Block - { - public long Number; - } - - class Item - { - public LinkedList Blocks = new LinkedList(); - public long Size; - } - - Block AllocBlock() - { - if (FreeList.Count != 0) - { - var blocknode = FreeList.First; - FreeList.RemoveFirst(); - Consumed += BlockSize; - return blocknode.Value; - } - - if (FreeWatermark == BlockCount) - throw new OutOfMemoryException($"{nameof(NDBDatabase)} out of reserved space"); - - var b = new Block() { Number = FreeWatermark }; - FreeWatermark++; - Consumed += BlockSize; - - return b; - } - - long GetOffsetForBlock(Block b) - { - return b.Number * BlockSize; - } - - /// - /// Creates a new instance around a DeleteOnClose file of the provided path - /// - public NDBDatabase(string path, long size, int blocksize) - { - Capacity = size; - Consumed = 0; - BlockSize = blocksize; - BlockCount = size / BlockSize; - Directory.CreateDirectory(Path.GetDirectoryName(path)); - Stream = new FileStream(path, FileMode.Create, System.Security.AccessControl.FileSystemRights.FullControl, FileShare.None, 4 * 1024, FileOptions.DeleteOnClose); - } - - /// - /// Clears the state of the datastructure to its original condition - /// - public void Clear() - { - Consumed = 0; - Items.Clear(); - FreeList.Clear(); - FreeWatermark = 0; - } - - public void Dispose() - { - Stream.Dispose(); - } - - /// - /// Total reserved storage capacity. You may nto be able to fit that much data in here though (due to blockiness) - /// - public readonly long Capacity; - - /// - /// The amount of bytes of storage consumed. Not necessarily equal to the total amount of data stored (due to blockiness) - /// - public long Consumed { get; private set; } - - /// - /// The amount of bytes of storage available. Store operations <= Remain will always succeed - /// - public long Remain { get { return Capacity - Consumed; } } - - /// - /// Stores an item with the given key - /// - public void Store(string name, byte[] buf, int offset, int length) - { - if (Items.ContainsKey(name)) - throw new InvalidOperationException($"Can't add already existing key of name {name}"); - - if (length > Remain) - throw new OutOfMemoryException($"Insufficient storage reserved for {length} bytes"); - - long todo = length; - int src = offset; - Item item = new Item { Size = length }; - Items[name] = item; - while (todo > 0) - { - var b = AllocBlock(); - item.Blocks.AddLast(b); - - long tocopy = todo; - if (tocopy > BlockSize) - tocopy = BlockSize; - - Stream.Position = GetOffsetForBlock(b); - Stream.Write(buf, src, (int)tocopy); - - todo -= tocopy; - src += (int)tocopy; - } - } - - /// - /// Fetches an item with the given key - /// - public byte[] FetchAll(string name) - { - var buf = new byte[GetSize(name)]; - Fetch(name, buf, 0); - return buf; - } - - /// - /// Fetches an item with the given key - /// - public void Fetch(string name, byte[] buf, int offset) - { - Item item; - if (!Items.TryGetValue(name, out item)) - throw new KeyNotFoundException(); - - long todo = item.Size; - var curr = item.Blocks.First; - while (todo > 0) - { - long tocopy = todo; - if (tocopy > BlockSize) - tocopy = BlockSize; - Stream.Position = GetOffsetForBlock(curr.Value); - Stream.Read(buf, offset, (int)tocopy); - - todo -= tocopy; - offset += (int)tocopy; - - curr = curr.Next; - } - - System.Diagnostics.Debug.Assert(curr == null); - } - - /// - /// Releases the item with the given key. - /// Removing a non-existent item is benign, I guess - /// - public void Release(string name) - { - Item item; - if (!Items.TryGetValue(name, out item)) - return; - Items.Remove(name); - var blocks = item.Blocks.ToArray(); - item.Blocks.Clear(); - foreach (var block in blocks) - FreeList.AddLast(block); - Consumed -= blocks.Length * BlockSize; - } - - /// - /// Gets the size of the item with the given key - /// - public long GetSize(string name) - { - return Items[name].Size; - } - } - -} \ No newline at end of file From 0ab1b1887eee03dd93d26d01a0b433fdaa9ed662 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:15:14 -0500 Subject: [PATCH 08/27] IStateManager - remove IDisposable, this isn't a terrible idea, but let's put it back if/when we need it. Remove IDisposable from StateManagerState --- BizHawk.Client.Common/movie/tasproj/StateManagerState.cs | 6 +----- BizHawk.Client.Common/movie/tasproj/TasStateManager.cs | 5 ----- 2 files changed, 1 insertion(+), 10 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs index 44c1cb7970..03214e8a0c 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs @@ -6,7 +6,7 @@ namespace BizHawk.Client.Common /// /// Represents a savestate in the /// - internal class StateManagerState : IDisposable + internal class StateManagerState { public int Frame { get; } @@ -19,9 +19,5 @@ namespace BizHawk.Client.Common State = state; Frame = frame; } - - public void Dispose() - { - } } } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 432746ffe5..78656b5c44 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -3,7 +3,6 @@ using System.Collections.Generic; using System.IO; using System.Linq; -using BizHawk.Common; using BizHawk.Common.NumberExtensions; using BizHawk.Emulation.Common; using BizHawk.Emulation.Common.IEmulatorExtensions; @@ -53,10 +52,6 @@ namespace BizHawk.Client.Common _decay = new StateManagerDecay(_movie, this); } - public void Dispose() - { - } - public void UpdateStateFrequency() { _stateFrequency = ((int)_expectedStateSize / Settings.MemStateGapDivider / 1024) From 6004f30d3abe9e4df70722bf011d8309c5e45b6f Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:19:18 -0500 Subject: [PATCH 09/27] rename IStateManager.ClearStateHistory -> Clear, LastStatedFrame -> Last --- .../movie/conversions/MovieConversionExtensions.cs | 4 ++-- BizHawk.Client.Common/movie/tasproj/IStateManager.cs | 8 +++----- BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs | 2 +- BizHawk.Client.Common/movie/tasproj/TasMovie.cs | 4 ++-- BizHawk.Client.Common/movie/tasproj/TasStateManager.cs | 4 ++-- 5 files changed, 10 insertions(+), 12 deletions(-) diff --git a/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs b/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs index ac88cba786..a15f250e0e 100644 --- a/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs +++ b/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs @@ -168,7 +168,7 @@ namespace BizHawk.Client.Common.MovieConversionExtensions // States can't be easily moved over, because they contain the frame number. // TODO? I'm not sure how this would be done. tas.TasStateManager.MountWriteAccess(); - old.TasStateManager.ClearStateHistory(); + old.TasStateManager.Clear(); // Lag Log tas.TasLagLog.FromLagLog(old.TasLagLog); @@ -242,7 +242,7 @@ namespace BizHawk.Client.Common.MovieConversionExtensions } var tas = new TasMovie(newFilename, true) { SaveRam = saveRam }; - tas.TasStateManager.ClearStateHistory(); + tas.TasStateManager.Clear(); tas.ClearLagLog(); var entries = old.GetLogEntries(); diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index f0a259fdb2..e3fe175b82 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -7,7 +7,7 @@ namespace BizHawk.Client.Common using BizHawk.Common; - public interface IStateManager : IDisposable + public interface IStateManager { // byte[] this[int frame] { get; } // TODO: I had it refactored to this back in the day KeyValuePair this[int frame] { get; } @@ -22,9 +22,8 @@ namespace BizHawk.Client.Common bool Invalidate(int frame); - // TODO: rename to Clear() // TODO: consider it passing a bool if anything was cleared, and the .Any() could go away - void ClearStateHistory(); + void Clear(); void Save(BinaryWriter bw); @@ -36,8 +35,7 @@ namespace BizHawk.Client.Common int Count { get; } - // TODO: rename to Last - int LastStatedFrame { get; } + int Last { get; } bool IsMarkerState(int frame); diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs index 221d235b89..cd99ea0766 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs @@ -272,7 +272,7 @@ namespace BizHawk.Client.Common private void ClearTasprojExtras() { ClearLagLog(); - _stateManager.ClearStateHistory(); + _stateManager.Clear(); Markers.Clear(); ChangeLog.ClearLog(); } diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs index 08e8023487..9b26f14e53 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs @@ -35,7 +35,7 @@ namespace BizHawk.Client.Common public TasLagLog TasLagLog => _lagLog; public IStringLog InputLog => Log; public int BranchCount => Branches.Count; - public int LastStatedFrame => _stateManager.LastStatedFrame; + public int LastStatedFrame => _stateManager.Last; public override string PreferredExtension => Extension; public IStateManager TasStateManager => _stateManager; @@ -218,7 +218,7 @@ namespace BizHawk.Client.Common { if (_stateManager.Any()) { - _stateManager.ClearStateHistory(); + _stateManager.Clear(); Changes = true; } } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 78656b5c44..717516a1a1 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -317,7 +317,7 @@ namespace BizHawk.Client.Common return ret; } - public void ClearStateHistory() + public void Clear() { if (_states.Any()) { @@ -450,7 +450,7 @@ namespace BizHawk.Client.Common } } - public int LastStatedFrame + public int Last { get { From 21c164abcfc33a1ff2832e39318d8e1e2ab2d6a8 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:22:12 -0500 Subject: [PATCH 10/27] simplify Last implementation --- .../movie/tasproj/TasStateManager.cs | 28 ++----------------- 1 file changed, 3 insertions(+), 25 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 717516a1a1..7e401f1d26 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -437,30 +437,8 @@ namespace BizHawk.Client.Common return _states.Count > 1; } - public int LastKey - { - get - { - if (_states.Count == 0) - { - return 0; - } - - return _states.Last().Key; - } - } - - public int Last - { - get - { - if (Count > 0) - { - return LastKey; - } - - return 0; - } - } + public int Last => _states.Count > 0 + ? _states.Last().Key + : 0; } } From d998a9587f06d68f364117ad801dd36dd5d46882 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:28:10 -0500 Subject: [PATCH 11/27] remove a todo --- BizHawk.Client.Common/movie/tasproj/IStateManager.cs | 1 - 1 file changed, 1 deletion(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index e3fe175b82..327e108288 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -22,7 +22,6 @@ namespace BizHawk.Client.Common bool Invalidate(int frame); - // TODO: consider it passing a bool if anything was cleared, and the .Any() could go away void Clear(); void Save(BinaryWriter bw); From 7f43fb09af7ce321f100563d884ad5b1dd7c68b4 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:29:25 -0500 Subject: [PATCH 12/27] delete an absurd check --- .../movie/tasproj/TasStateManager.cs | 28 +++---------------- 1 file changed, 4 insertions(+), 24 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 7e401f1d26..cc13f44c28 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -175,7 +175,7 @@ namespace BizHawk.Client.Common } else { - Used += (ulong)state.Length; + _used += (ulong)state.Length; _states.Add(frame, new StateManagerState(state, frame)); } } @@ -239,7 +239,7 @@ namespace BizHawk.Client.Common StateManagerState state = _states.Values[index]; // TODO: remove .Values here? - Used -= (ulong)state.Length; + _used -= (ulong)state.Length; _states.RemoveAt(index); @@ -261,7 +261,7 @@ namespace BizHawk.Client.Common private List ExcludeStates() { List ret = new List(); - ulong saveUsed = Used; + ulong saveUsed = _used; // respect state gap no matter how small the resulting size will be // still leave marker states @@ -328,7 +328,7 @@ namespace BizHawk.Client.Common _states.Clear(); SetState(0, power.State); - Used = (ulong)power.State.Length; + _used = (ulong)power.State.Length; } } @@ -404,26 +404,6 @@ namespace BizHawk.Client.Common } private ulong _used; - private ulong Used - { - get - { - return _used; - } - - set - { - // TODO: Shouldn't we throw an exception? Debug.Fail only runs in debug mode? - if (value > 0xf000000000000000) - { - System.Diagnostics.Debug.Fail("ulong Used underfow!"); - } - else - { - _used = value; - } - } - } public int Count => _states.Count; From 2a8b4504cd5934d4b51827ffa8b5c3c11c3ff74d Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:43:22 -0500 Subject: [PATCH 13/27] Simplify --- .../movie/tasproj/IStateManager.cs | 14 ++++++-------- .../movie/tasproj/TasStateManager.cs | 9 ++++----- .../tools/TAStudio/TAStudio.MenuItems.cs | 1 - 3 files changed, 10 insertions(+), 14 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index 327e108288..e45cd078ce 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -38,18 +38,16 @@ namespace BizHawk.Client.Common bool IsMarkerState(int frame); + int GetStateIndexByFrame(int frame); + + int GetStateFrameByIndex(int index); + + void UpdateStateFrequency(); + // ********* Delete these ********** void MountWriteAccess(); // *********** Reconsider these ************/ - void LimitStateCount(); - - void UpdateStateFrequency(); - bool RemoveState(int frame); - - int GetStateIndexByFrame(int frame); - - int GetStateFrameByIndex(int index); } } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index cc13f44c28..e35f67d839 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -58,6 +58,7 @@ namespace BizHawk.Client.Common .Clamp(_minFrequency, _maxFrequency); _decay.UpdateSettings(MaxStates, _stateFrequency, 4); + LimitStateCount(); } /// @@ -246,11 +247,9 @@ namespace BizHawk.Client.Common return true; } - /// - /// Deletes states to follow the state storage size limits. - /// Used after changing the settings too. - /// - public void LimitStateCount() + // Deletes states to follow the state storage size limits. + // Used after changing the settings too. + private void LimitStateCount() { if (Count + 1 > MaxStates) { diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs index 3695e8d996..d7743fc18f 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs @@ -1080,7 +1080,6 @@ namespace BizHawk.Client.EmuHawk Statable = this.StatableEmulator }.ShowDialog(); CurrentTasMovie.TasStateManager.UpdateStateFrequency(); - CurrentTasMovie.TasStateManager.LimitStateCount(); UpdateChangesIndicator(); } From d4432db7f998f86949054351c405ff1dabe08151 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:44:34 -0500 Subject: [PATCH 14/27] simplify --- .../movie/tasproj/IStateManager.cs | 7 +++---- .../movie/tasproj/StateManagerDecay.cs | 14 +++++++------- .../movie/tasproj/TasStateManager.cs | 4 ++-- 3 files changed, 12 insertions(+), 13 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index e45cd078ce..1aa258bbe5 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -38,16 +38,15 @@ namespace BizHawk.Client.Common bool IsMarkerState(int frame); + void UpdateStateFrequency(); + int GetStateIndexByFrame(int frame); int GetStateFrameByIndex(int index); - void UpdateStateFrequency(); + bool Remove(int frame); // ********* Delete these ********** void MountWriteAccess(); - - // *********** Reconsider these ************/ - bool RemoveState(int frame); } } diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index 1d1792333f..88a47ff934 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -90,7 +90,7 @@ namespace BizHawk.Client.Common else if (currentFrame % _step > 0) { // ignore the pattern if the state doesn't belong already, drop it blindly and skip everything - if (_tsm.RemoveState(currentFrame)) + if (_tsm.Remove(currentFrame)) { // decrementing this if no state was removed is BAD decayStates--; @@ -131,7 +131,7 @@ namespace BizHawk.Client.Common else if ((currentFrame % _step > 0) && (currentFrame + 1 != _movie.LastEditedFrame)) { // ignore the pattern if the state doesn't belong already, drop it blindly and skip everything - if (_tsm.RemoveState(currentFrame)) + if (_tsm.Remove(currentFrame)) { // decrementing this if no state was removed is BAD decayStates--; @@ -167,7 +167,7 @@ namespace BizHawk.Client.Common { if (baseStateFrame - forwardFrame > backwardFrame - baseStateFrame) { - if (_tsm.RemoveState(forwardFrame * _step)) + if (_tsm.Remove(forwardFrame * _step)) { // decrementing this if no state was removed is BAD decayStates--; @@ -175,7 +175,7 @@ namespace BizHawk.Client.Common } else { - if (_tsm.RemoveState(backwardFrame * _step)) + if (_tsm.Remove(backwardFrame * _step)) { // decrementing this if no state was removed is BAD decayStates--; @@ -184,7 +184,7 @@ namespace BizHawk.Client.Common } else if (forwardFrame > -1) { - if (_tsm.RemoveState(forwardFrame * _step)) + if (_tsm.Remove(forwardFrame * _step)) { // decrementing this if no state was removed is BAD decayStates--; @@ -192,7 +192,7 @@ namespace BizHawk.Client.Common } else if (backwardFrame > -1) { - if (_tsm.RemoveState(backwardFrame * _step)) + if (_tsm.Remove(backwardFrame * _step)) { // decrementing this if no state was removed is BAD decayStates--; @@ -203,7 +203,7 @@ namespace BizHawk.Client.Common // this shouldn't happen, but if we don't do it here, nothing good will happen either if (decayStatesLast == decayStates) { - if (_tsm.RemoveState(_tsm.GetStateFrameByIndex(1))) + if (_tsm.Remove(_tsm.GetStateFrameByIndex(1))) { // decrementing this if no state was removed is BAD decayStates--; diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index e35f67d839..0530a929db 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -210,7 +210,7 @@ namespace BizHawk.Client.Common foreach (var state in statesToRemove) { - RemoveState(state.Key); + Remove(state.Key); } CallInvalidateCallback(frame); @@ -229,7 +229,7 @@ namespace BizHawk.Client.Common return _movie.Markers.IsMarker(frame + 1); } - public bool RemoveState(int frame) + public bool Remove(int frame) { int index = _states.IndexOfKey(frame); From c87c1d8328bbd9c242f6d961bc87cc3c71e1d8e8 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:51:56 -0500 Subject: [PATCH 15/27] simplify --- BizHawk.Client.Common/movie/tasproj/TasStateManager.cs | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 0530a929db..84353f2d24 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -21,11 +21,6 @@ namespace BizHawk.Client.Common public Action InvalidateCallback { get; set; } - private void CallInvalidateCallback(int index) - { - InvalidateCallback?.Invoke(index); - } - private SortedList _states = new SortedList(); private bool _isMountedForWrite; @@ -213,7 +208,7 @@ namespace BizHawk.Client.Common Remove(state.Key); } - CallInvalidateCallback(frame); + InvalidateCallback?.Invoke(frame); } return anyInvalidated; From eaef33649289639811631f67da54bab7d04dc353 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:06:35 -0500 Subject: [PATCH 16/27] rip out IStateManager.MountForWriteAccess() and simplify --- .../conversions/MovieConversionExtensions.cs | 2 - .../movie/tasproj/IStateManager.cs | 10 +--- .../movie/tasproj/TasStateManager.cs | 50 ++++++------------- BizHawk.Client.EmuHawk/MainForm.Movie.cs | 7 --- .../tools/TAStudio/TAStudio.cs | 1 - 5 files changed, 17 insertions(+), 53 deletions(-) diff --git a/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs b/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs index a15f250e0e..a1450bf56a 100644 --- a/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs +++ b/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs @@ -37,7 +37,6 @@ namespace BizHawk.Client.Common.MovieConversionExtensions } var tas = new TasMovie(newFilename, old.StartsFromSavestate); - tas.TasStateManager.MountWriteAccess(); for (var i = 0; i < old.InputLogLength; i++) { @@ -167,7 +166,6 @@ namespace BizHawk.Client.Common.MovieConversionExtensions // States can't be easily moved over, because they contain the frame number. // TODO? I'm not sure how this would be done. - tas.TasStateManager.MountWriteAccess(); old.TasStateManager.Clear(); // Lag Log diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index 1aa258bbe5..06ca643f8f 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -1,12 +1,9 @@ using System; +using System.Collections.Generic; +using System.IO; namespace BizHawk.Client.Common { - using System.Collections.Generic; - using System.IO; - - using BizHawk.Common; - public interface IStateManager { // byte[] this[int frame] { get; } // TODO: I had it refactored to this back in the day @@ -45,8 +42,5 @@ namespace BizHawk.Client.Common int GetStateFrameByIndex(int index); bool Remove(int frame); - - // ********* Delete these ********** - void MountWriteAccess(); } } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 84353f2d24..b4fcd3daf0 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -18,15 +18,11 @@ namespace BizHawk.Client.Common // TODO: pass this in, and find a solution to a stale reference (this is instantiated BEFORE a new core instance is made, making this one stale if it is simply set in the constructor private IStatable Core => Global.Emulator.AsStatable(); private readonly StateManagerDecay _decay; - - public Action InvalidateCallback { get; set; } - - private SortedList _states = new SortedList(); - - private bool _isMountedForWrite; private readonly TasMovie _movie; - private ulong _expectedStateSize; + private readonly SortedList _states; + private readonly ulong _expectedStateSize; + private int _stateFrequency; private readonly int _minFrequency = 1; private readonly int _maxFrequency = 16; @@ -45,8 +41,20 @@ namespace BizHawk.Client.Common } _decay = new StateManagerDecay(_movie, this); + + _expectedStateSize = (ulong)Core.SaveStateBinary().Length; + if (_expectedStateSize == 0) + { + throw new InvalidOperationException("Savestate size can not be zero!"); + } + + _states = new SortedList(MaxStates); + + UpdateStateFrequency(); } + public Action InvalidateCallback { get; set; } + public void UpdateStateFrequency() { _stateFrequency = ((int)_expectedStateSize / Settings.MemStateGapDivider / 1024) @@ -56,34 +64,6 @@ namespace BizHawk.Client.Common LimitStateCount(); } - /// - /// Mounts this instance for write access. Prior to that it's read-only - /// - public void MountWriteAccess() - { - if (_isMountedForWrite) - { - return; - } - - int limit = 0; - _isMountedForWrite = true; - _expectedStateSize = (ulong)Core.SaveStateBinary().Length; - UpdateStateFrequency(); - - if (_expectedStateSize > 0) - { - limit = MaxStates; - } - - _states = new SortedList(limit); - - if (_expectedStateSize > int.MaxValue) - { - throw new InvalidOperationException(); - } - } - public TasStateManagerSettings Settings { get; set; } /// diff --git a/BizHawk.Client.EmuHawk/MainForm.Movie.cs b/BizHawk.Client.EmuHawk/MainForm.Movie.cs index b9bc4db335..d7094c8893 100644 --- a/BizHawk.Client.EmuHawk/MainForm.Movie.cs +++ b/BizHawk.Client.EmuHawk/MainForm.Movie.cs @@ -5,10 +5,6 @@ using System.Windows.Forms; using BizHawk.Client.Common; using BizHawk.Emulation.Common; using BizHawk.Emulation.Common.IEmulatorExtensions; -using BizHawk.Emulation.Cores.Consoles.Nintendo.QuickNES; -using BizHawk.Emulation.Cores.Nintendo.NES; -using BizHawk.Emulation.Cores.Nintendo.SNES9X; -using BizHawk.Emulation.Cores.Nintendo.SNES; namespace BizHawk.Client.EmuHawk { @@ -30,9 +26,6 @@ namespace BizHawk.Client.EmuHawk try { - var tasmovie = (movie as TasMovie); - if (tasmovie != null) - tasmovie.TasStateManager.MountWriteAccess(); Global.MovieSession.QueueNewMovie(movie, record, Emulator); } catch (MoviePlatformMismatchException ex) diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs index 6f9a438201..6a48406b51 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs @@ -642,7 +642,6 @@ namespace BizHawk.Client.EmuHawk Global.MovieSession.Movie = new TasMovie(false, _seekBackgroundWorker); var stateManager = ((TasMovie)Global.MovieSession.Movie).TasStateManager; - stateManager.MountWriteAccess(); stateManager.InvalidateCallback = GreenzoneInvalidated; BookMarkControl.LoadedCallback = BranchLoaded; From e89bb6c276aa1dc76619f389266ce4f453f61866 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:13:16 -0500 Subject: [PATCH 17/27] reorg TasStateManager file --- .../movie/tasproj/IStateManager.cs | 19 ++ .../movie/tasproj/TasStateManager.cs | 263 ++++++++---------- 2 files changed, 142 insertions(+), 140 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index 06ca643f8f..265c15e342 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -7,16 +7,29 @@ namespace BizHawk.Client.Common public interface IStateManager { // byte[] this[int frame] { get; } // TODO: I had it refactored to this back in the day + + /// + /// Retrieves the savestate for the given frame, + /// If this frame does not have a state currently, will return an empty array + /// + /// A savestate for the given frame or an empty array if there isn't one KeyValuePair this[int frame] { get; } TasStateManagerSettings Settings { get; set; } Action InvalidateCallback { set; } + /// + /// Requests that the current emulator state be captured + /// Unless force is true, the state may or may not be captured depending on the logic employed by "green-zone" management + /// void Capture(bool force = false); bool HasState(int frame); + /// + /// Clears out all savestates after the given frame number + /// bool Invalidate(int frame); void Clear(); @@ -37,8 +50,14 @@ namespace BizHawk.Client.Common void UpdateStateFrequency(); + /// + /// Returns index of the state right above the given frame + /// int GetStateIndexByFrame(int frame); + /// + /// Returns frame of the state at the given index + /// int GetStateFrameByIndex(int index); bool Remove(int frame); diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index b4fcd3daf0..cbd3cdb54a 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -15,6 +15,9 @@ namespace BizHawk.Client.Common /// public class TasStateManager : IStateManager { + private const int MinFrequency = 1; + private const int MaxFrequency = 16; + // TODO: pass this in, and find a solution to a stale reference (this is instantiated BEFORE a new core instance is made, making this one stale if it is simply set in the constructor private IStatable Core => Global.Emulator.AsStatable(); private readonly StateManagerDecay _decay; @@ -23,9 +26,9 @@ namespace BizHawk.Client.Common private readonly SortedList _states; private readonly ulong _expectedStateSize; + private ulong _used; private int _stateFrequency; - private readonly int _minFrequency = 1; - private readonly int _maxFrequency = 16; + private int MaxStates => (int)(Settings.Cap / _expectedStateSize) + (int)((ulong)Settings.DiskCapacitymb * 1024 * 1024 / _expectedStateSize); private int FileStateGap => 1 << Settings.FileStateGap; @@ -55,22 +58,8 @@ namespace BizHawk.Client.Common public Action InvalidateCallback { get; set; } - public void UpdateStateFrequency() - { - _stateFrequency = ((int)_expectedStateSize / Settings.MemStateGapDivider / 1024) - .Clamp(_minFrequency, _maxFrequency); - - _decay.UpdateSettings(MaxStates, _stateFrequency, 4); - LimitStateCount(); - } - public TasStateManagerSettings Settings { get; set; } - /// - /// Retrieves the savestate for the given frame, - /// If this frame does not have a state currently, will return an empty array - /// - /// A savestate for the given frame or an empty array if there isn't one public KeyValuePair this[int frame] { get @@ -89,6 +78,12 @@ namespace BizHawk.Client.Common } } + public int Count => _states.Count; + + public int Last => _states.Count > 0 + ? _states.Last().Key + : 0; + private byte[] InitialState { get @@ -102,10 +97,25 @@ namespace BizHawk.Client.Common } } - /// - /// Requests that the current emulator state be captured - /// Unless force is true, the state may or may not be captured depending on the logic employed by "green-zone" management - /// + public bool Any() + { + if (_movie.StartsFromSavestate) + { + return _states.Count > 0; + } + + return _states.Count > 1; + } + + public void UpdateStateFrequency() + { + _stateFrequency = ((int)_expectedStateSize / Settings.MemStateGapDivider / 1024) + .Clamp(MinFrequency, MaxFrequency); + + _decay.UpdateSettings(MaxStates, _stateFrequency, 4); + LimitStateCount(); + } + public void Capture(bool force = false) { bool shouldCapture; @@ -138,21 +148,18 @@ namespace BizHawk.Client.Common } } - internal void SetState(int frame, byte[] state, bool skipRemoval = true) + public void Clear() { - if (!skipRemoval) // skipRemoval: false only when capturing new states + if (_states.Any()) { - LimitStateCount(); // Remove before adding so this state won't be removed. - } - - if (_states.ContainsKey(frame)) - { - _states[frame].State = state; - } - else - { - _used += (ulong)state.Length; - _states.Add(frame, new StateManagerState(state, frame)); + var tempState = _states.Values; + var power = tempState[0].Frame == 0 + ? _states.Values.First(s => s.Frame == 0) + : _states.Values[0]; + + _states.Clear(); + SetState(0, power.State); + _used = (ulong)power.State.Length; } } @@ -166,9 +173,6 @@ namespace BizHawk.Client.Common return _states.ContainsKey(frame); } - /// - /// Clears out all savestates after the given frame number - /// public bool Invalidate(int frame) { bool anyInvalidated = false; @@ -222,6 +226,89 @@ namespace BizHawk.Client.Common return true; } + // Map: + // 4 bytes - total savestate count + // [Foreach state] + // 4 bytes - frame + // 4 bytes - length of savestate + // 0 - n savestate + public void Save(BinaryWriter bw) + { + List noSave = ExcludeStates(); + bw.Write(_states.Count - noSave.Count); + + for (int i = 0; i < _states.Count; i++) + { + if (noSave.Contains(i)) + { + continue; + } + + bw.Write(_states.Keys[i]); + bw.Write(_states.Values[i].Length); + bw.Write(_states.Values[i].State); + } + } + + public void Load(BinaryReader br) + { + _states.Clear(); + + try + { + int nstates = br.ReadInt32(); + + for (int i = 0; i < nstates; i++) + { + int frame = br.ReadInt32(); + int len = br.ReadInt32(); + byte[] data = br.ReadBytes(len); + + // whether we should allow state removal check here is an interesting question + // nothing was edited yet, so it might make sense to show the project untouched first + SetState(frame, data); + } + } + catch (EndOfStreamException) + { + } + } + + public KeyValuePair GetStateClosestToFrame(int frame) + { + var s = _states.LastOrDefault(state => state.Key < frame); + + return this[s.Key]; + } + + public int GetStateIndexByFrame(int frame) + { + return _states.IndexOfKey(GetStateClosestToFrame(frame).Key); + } + + public int GetStateFrameByIndex(int index) + { + return _states.Keys[index]; + } + + private void SetState(int frame, byte[] state, bool skipRemoval = true) + { + if (!skipRemoval) // skipRemoval: false only when capturing new states + { + LimitStateCount(); // Remove before adding so this state won't be removed. + } + + if (_states.ContainsKey(frame)) + { + _states[frame].State = state; + } + else + { + _used += (ulong)state.Length; + _states.Add(frame, new StateManagerState(state, frame)); + } + } + // Deletes states to follow the state storage size limits. // Used after changing the settings too. private void LimitStateCount() @@ -290,109 +377,5 @@ namespace BizHawk.Client.Common return ret; } - - public void Clear() - { - if (_states.Any()) - { - var tempState = _states.Values; - var power = tempState[0].Frame == 0 - ? _states.Values.First(s => s.Frame == 0) - : _states.Values[0]; - - _states.Clear(); - SetState(0, power.State); - _used = (ulong)power.State.Length; - } - } - - // Map: - // 4 bytes - total savestate count - // [Foreach state] - // 4 bytes - frame - // 4 bytes - length of savestate - // 0 - n savestate - public void Save(BinaryWriter bw) - { - List noSave = ExcludeStates(); - bw.Write(_states.Count - noSave.Count); - - for (int i = 0; i < _states.Count; i++) - { - if (noSave.Contains(i)) - { - continue; - } - - bw.Write(_states.Keys[i]); - bw.Write(_states.Values[i].Length); - bw.Write(_states.Values[i].State); - } - } - - public void Load(BinaryReader br) - { - _states.Clear(); - - try - { - int nstates = br.ReadInt32(); - - for (int i = 0; i < nstates; i++) - { - int frame = br.ReadInt32(); - int len = br.ReadInt32(); - byte[] data = br.ReadBytes(len); - - // whether we should allow state removal check here is an interesting question - // nothing was edited yet, so it might make sense to show the project untouched first - SetState(frame, data); - } - } - catch (EndOfStreamException) - { - } - } - - public KeyValuePair GetStateClosestToFrame(int frame) - { - var s = _states.LastOrDefault(state => state.Key < frame); - - return this[s.Key]; - } - - /// - /// Returns index of the state right above the given frame - /// - public int GetStateIndexByFrame(int frame) - { - return _states.IndexOfKey(GetStateClosestToFrame(frame).Key); - } - - /// - /// Returns frame of the state at the given index - /// - public int GetStateFrameByIndex(int index) - { - return _states.Keys[index]; - } - - private ulong _used; - - public int Count => _states.Count; - - public bool Any() - { - if (_movie.StartsFromSavestate) - { - return _states.Count > 0; - } - - return _states.Count > 1; - } - - public int Last => _states.Count > 0 - ? _states.Last().Key - : 0; } } From d79a00f035bf5a62cd807229505a8b8932e75ac5 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:17:49 -0500 Subject: [PATCH 18/27] remove IStateManager.IsMarkerState --- .../movie/tasproj/IStateManager.cs | 6 ++++-- .../movie/tasproj/StateManagerDecay.cs | 13 +++++++----- .../movie/tasproj/TasStateManager.cs | 20 +++++++++---------- 3 files changed, 22 insertions(+), 17 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index 265c15e342..c1d430e5bd 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -46,8 +46,6 @@ namespace BizHawk.Client.Common int Last { get; } - bool IsMarkerState(int frame); - void UpdateStateFrequency(); /// @@ -60,6 +58,10 @@ namespace BizHawk.Client.Common /// int GetStateFrameByIndex(int index); + /// + /// Directly remove a state from the given frame, if it exists + /// Should only be called by pruning operations + /// bool Remove(int frame); } } diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index 88a47ff934..fb4f28e861 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -79,15 +79,17 @@ namespace BizHawk.Client.Common { int currentFrame = _tsm.GetStateFrameByIndex(currentStateIndex); - if (_tsm.IsMarkerState(currentFrame)) + if (_movie.Markers.IsMarker(currentFrame + 1)) { continue; } - else if (currentFrame + 1 == _movie.LastEditedFrame) + + if (currentFrame + 1 == _movie.LastEditedFrame) { continue; } - else if (currentFrame % _step > 0) + + if (currentFrame % _step > 0) { // ignore the pattern if the state doesn't belong already, drop it blindly and skip everything if (_tsm.Remove(currentFrame)) @@ -124,11 +126,12 @@ namespace BizHawk.Client.Common { int currentFrame = _tsm.GetStateFrameByIndex(currentStateIndex); - if (_tsm.IsMarkerState(currentFrame)) + if (_movie.Markers.IsMarker(currentFrame + 1)) { continue; } - else if ((currentFrame % _step > 0) && (currentFrame + 1 != _movie.LastEditedFrame)) + + if ((currentFrame % _step > 0) && (currentFrame + 1 != _movie.LastEditedFrame)) { // ignore the pattern if the state doesn't belong already, drop it blindly and skip everything if (_tsm.Remove(currentFrame)) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index cbd3cdb54a..8568b20d8c 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -198,16 +198,6 @@ namespace BizHawk.Client.Common return anyInvalidated; } - public bool IsMarkerState(int frame) - { - if (frame == -1) - { - return false; - } - - return _movie.Markers.IsMarker(frame + 1); - } - public bool Remove(int frame) { int index = _states.IndexOfKey(frame); @@ -291,6 +281,16 @@ namespace BizHawk.Client.Common return _states.Keys[index]; } + private bool IsMarkerState(int frame) + { + if (frame == -1) + { + return false; + } + + return _movie.Markers.IsMarker(frame + 1); + } + private void SetState(int frame, byte[] state, bool skipRemoval = true) { if (!skipRemoval) // skipRemoval: false only when capturing new states From 5366b854a66064c1394de9ac53939403604c784b Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:20:57 -0500 Subject: [PATCH 19/27] remove unnecessary savestate clone --- BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs index d7743fc18f..3cb1c13867 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs @@ -814,7 +814,7 @@ namespace BizHawk.Client.EmuHawk if (CurrentTasMovie.TasStateManager.HasState(Emulator.Frame)) { - byte[] state = (byte[])StatableEmulator.SaveStateBinary().Clone(); // Why is this cloning it? + byte[] state = StatableEmulator.SaveStateBinary(); byte[] greenzone = CurrentTasMovie.TasStateManager[Emulator.Frame].Value; if (!state.SequenceEqual(greenzone)) From cd0c237e31042c17d91eeff22602886f60f90b0b Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:26:16 -0500 Subject: [PATCH 20/27] simplify some tastudio logic --- .../tools/TAStudio/TAStudio.Navigation.cs | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.Navigation.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.Navigation.cs index e6ec6e7615..4ea57477ed 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.Navigation.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.Navigation.cs @@ -57,10 +57,12 @@ namespace BizHawk.Client.EmuHawk { TastudioPlayMode(); - int lastState = CurrentTasMovie.TasStateManager.GetStateClosestToFrame(frame).Key; // Simply getting the last state doesn't work if that state is the frame. [dispaly isn't saved in the state, need to emulate to frame] - if (lastState > Emulator.Frame) + // Simply getting the last state doesn't work if that state is the frame. + // display isn't saved in the state, need to emulate to frame + var lastState = CurrentTasMovie.TasStateManager.GetStateClosestToFrame(frame); + if (lastState.Key > Emulator.Frame) { - LoadState(CurrentTasMovie.TasStateManager[lastState]); // STATE ACCESS + LoadState(lastState); } StartSeeking(frame); From 394a6d86b52dee7f53ff8537709ed4da60a0ca09 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:38:33 -0500 Subject: [PATCH 21/27] According to the TSM api, the indexer will never return null, check length instead --- BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs index 6a48406b51..c7a25d82c0 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs @@ -900,7 +900,7 @@ namespace BizHawk.Client.EmuHawk _unpauseAfterSeeking = (fromRewinding || WasRecording) && !Mainform.EmulatorPaused; TastudioPlayMode(); KeyValuePair closestState = CurrentTasMovie.TasStateManager.GetStateClosestToFrame(frame); - if (closestState.Value != null && (frame < Emulator.Frame || closestState.Key > Emulator.Frame)) + if (closestState.Value.Length > 0 && (frame < Emulator.Frame || closestState.Key > Emulator.Frame)) { LoadState(closestState); } From 3ec2ec325ad224af8da4ca8e13025405786b546f Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:43:21 -0500 Subject: [PATCH 22/27] IStateManager - refactor indexer to not return a keyvalue pair, since it by design will try to return the exact frame, returning the frame is redundant --- .../movie/tasproj/IStateManager.cs | 4 +--- .../movie/tasproj/TasStateManager.cs | 14 +++++++++----- .../tools/TAStudio/TAStudio.MenuItems.cs | 4 ++-- 3 files changed, 12 insertions(+), 10 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs index c1d430e5bd..95a678ad6a 100644 --- a/BizHawk.Client.Common/movie/tasproj/IStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/IStateManager.cs @@ -6,14 +6,12 @@ namespace BizHawk.Client.Common { public interface IStateManager { - // byte[] this[int frame] { get; } // TODO: I had it refactored to this back in the day - /// /// Retrieves the savestate for the given frame, /// If this frame does not have a state currently, will return an empty array /// /// A savestate for the given frame or an empty array if there isn't one - KeyValuePair this[int frame] { get; } + byte[] this[int frame] { get; } TasStateManagerSettings Settings { get; set; } diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 8568b20d8c..bd556bbe33 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -60,21 +60,21 @@ namespace BizHawk.Client.Common public TasStateManagerSettings Settings { get; set; } - public KeyValuePair this[int frame] + public byte[] this[int frame] { get { if (frame == 0) { - return new KeyValuePair(0, InitialState); + return InitialState; } if (_states.ContainsKey(frame)) { - return new KeyValuePair(frame, _states[frame].State); + return _states[frame].State; } - return new KeyValuePair(-1, new byte[0]); + return new byte[0]; } } @@ -267,8 +267,12 @@ namespace BizHawk.Client.Common public KeyValuePair GetStateClosestToFrame(int frame) { var s = _states.LastOrDefault(state => state.Key < frame); + if (s.Key > 0) + { + return new KeyValuePair(s.Key, s.Value.State); + } - return this[s.Key]; + return new KeyValuePair(0, InitialState); } public int GetStateIndexByFrame(int frame) diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs index 3cb1c13867..3aa06f75b6 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.MenuItems.cs @@ -815,9 +815,9 @@ namespace BizHawk.Client.EmuHawk if (CurrentTasMovie.TasStateManager.HasState(Emulator.Frame)) { byte[] state = StatableEmulator.SaveStateBinary(); - byte[] greenzone = CurrentTasMovie.TasStateManager[Emulator.Frame].Value; + byte[] greenZone = CurrentTasMovie.TasStateManager[Emulator.Frame]; - if (!state.SequenceEqual(greenzone)) + if (!state.SequenceEqual(greenZone)) { MessageBox.Show($"Bad data between frames {lastState} and {Emulator.Frame}"); return; From ea55217c3f4a495f2dddc89b1824f2fac268914b Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 16:49:08 -0500 Subject: [PATCH 23/27] I'm convinced this is a worthless check, -1 was never being passed in, and shouldn't be --- BizHawk.Client.Common/movie/tasproj/TasStateManager.cs | 5 ----- 1 file changed, 5 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index bd556bbe33..d58584a625 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -287,11 +287,6 @@ namespace BizHawk.Client.Common private bool IsMarkerState(int frame) { - if (frame == -1) - { - return false; - } - return _movie.Markers.IsMarker(frame + 1); } From c9ae8e3db315c01c1c65a6db622c5e76eb5a1f03 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 17:06:13 -0500 Subject: [PATCH 24/27] StateManagerState begone! --- .../BizHawk.Client.Common.csproj | 1 - .../movie/tasproj/StateManagerState.cs | 23 ----------- .../movie/tasproj/TasStateManager.cs | 40 +++++++++++-------- 3 files changed, 23 insertions(+), 41 deletions(-) delete mode 100644 BizHawk.Client.Common/movie/tasproj/StateManagerState.cs diff --git a/BizHawk.Client.Common/BizHawk.Client.Common.csproj b/BizHawk.Client.Common/BizHawk.Client.Common.csproj index 99e67e062f..c866a9a85c 100644 --- a/BizHawk.Client.Common/BizHawk.Client.Common.csproj +++ b/BizHawk.Client.Common/BizHawk.Client.Common.csproj @@ -188,7 +188,6 @@ - diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs deleted file mode 100644 index 03214e8a0c..0000000000 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerState.cs +++ /dev/null @@ -1,23 +0,0 @@ -using System; -using System.IO; - -namespace BizHawk.Client.Common -{ - /// - /// Represents a savestate in the - /// - internal class StateManagerState - { - public int Frame { get; } - - public byte[] State { get; set; } - - public int Length => State.Length; - - public StateManagerState(byte[] state, int frame) - { - State = state; - Frame = frame; - } - } -} diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index d58584a625..57f5a4ba33 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -23,7 +23,7 @@ namespace BizHawk.Client.Common private readonly StateManagerDecay _decay; private readonly TasMovie _movie; - private readonly SortedList _states; + private readonly SortedList _states; private readonly ulong _expectedStateSize; private ulong _used; @@ -51,7 +51,7 @@ namespace BizHawk.Client.Common throw new InvalidOperationException("Savestate size can not be zero!"); } - _states = new SortedList(MaxStates); + _states = new SortedList(MaxStates); UpdateStateFrequency(); } @@ -71,7 +71,7 @@ namespace BizHawk.Client.Common if (_states.ContainsKey(frame)) { - return _states[frame].State; + return _states[frame]; } return new byte[0]; @@ -93,7 +93,7 @@ namespace BizHawk.Client.Common return _movie.BinarySavestate; } - return _states[0].State; + return _states[0]; } } @@ -152,14 +152,20 @@ namespace BizHawk.Client.Common { if (_states.Any()) { - var tempState = _states.Values; - var power = tempState[0].Frame == 0 - ? _states.Values.First(s => s.Frame == 0) - : _states.Values[0]; - + // For power-on movies, we can't lose frame 0; + byte[] power = null; + if (!_movie.StartsFromSavestate) + { + power = _states[0]; + } + _states.Clear(); - SetState(0, power.State); - _used = (ulong)power.State.Length; + + if (power != null) + { + SetState(0, power); + _used = (ulong)power.Length; + } } } @@ -184,7 +190,7 @@ namespace BizHawk.Client.Common frame = 1; } - List> statesToRemove = _states.Where(s => s.Key >= frame).ToList(); + List> statesToRemove = _states.Where(s => s.Key >= frame).ToList(); anyInvalidated = statesToRemove.Any(); foreach (var state in statesToRemove) @@ -207,7 +213,7 @@ namespace BizHawk.Client.Common return false; } - StateManagerState state = _states.Values[index]; // TODO: remove .Values here? + var state = _states.Values[index]; // TODO: remove .Values here and use frame? _used -= (ulong)state.Length; @@ -236,7 +242,7 @@ namespace BizHawk.Client.Common bw.Write(_states.Keys[i]); bw.Write(_states.Values[i].Length); - bw.Write(_states.Values[i].State); + bw.Write(_states.Values[i]); } } @@ -269,7 +275,7 @@ namespace BizHawk.Client.Common var s = _states.LastOrDefault(state => state.Key < frame); if (s.Key > 0) { - return new KeyValuePair(s.Key, s.Value.State); + return s; } return new KeyValuePair(0, InitialState); @@ -299,12 +305,12 @@ namespace BizHawk.Client.Common if (_states.ContainsKey(frame)) { - _states[frame].State = state; + _states[frame] = state; } else { _used += (ulong)state.Length; - _states.Add(frame, new StateManagerState(state, frame)); + _states.Add(frame, state); } } From 26c190cd32b183d50bf5f85bacb15791b623a766 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 18:26:59 -0500 Subject: [PATCH 25/27] simplify --- BizHawk.Client.Common/movie/tasproj/TasStateManager.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs index 57f5a4ba33..08891bb478 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasStateManager.cs @@ -213,7 +213,7 @@ namespace BizHawk.Client.Common return false; } - var state = _states.Values[index]; // TODO: remove .Values here and use frame? + var state = _states[frame]; _used -= (ulong)state.Length; From cb4b7d7089840ddc36b59d273715b242a6f6fee8 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 18:31:46 -0500 Subject: [PATCH 26/27] some cleanup --- .../movie/tasproj/StateManagerDecay.cs | 15 +++++++-------- BizHawk.sln.DotSettings | 1 + 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs index fb4f28e861..28fa07e735 100644 --- a/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs +++ b/BizHawk.Client.Common/movie/tasproj/StateManagerDecay.cs @@ -8,7 +8,7 @@ First element is always assumed to be 16, which has all 4 bits set to 0. Each right zero means that we lower the priority of a state that goes at that index. Priority changes depending on current frame and amount of states. States with biggest priority get erased - first. With a 4-bit battern and no initial gap between states, total frame coverage is + first. With a 4-bit pattern and no initial gap between states, total frame coverage is about 5 times state count. Initial state gap can screw up our patterns, so do all the calculations like the gap @@ -18,7 +18,7 @@ don't drop it) or appears inside the state gap (in which case we forcibly drop it). This step doesn't involve numbers reduction. - _zeros values are essentialy the values of rshiftby here: + _zeros values are essentially the values of rshiftby here: bitwise view frame rshiftby priority 00010000 0 4 1 00000001 1 0 15 @@ -108,11 +108,11 @@ namespace BizHawk.Client.Common } int zeroCount = _zeros[currentFrame & _mask]; - int priority = ((baseStateFrame - currentFrame) >> zeroCount); + int priority = (baseStateFrame - currentFrame) >> zeroCount; if (_align) { - priority -= ((_base * ((1 << zeroCount) * 2 - 1)) >> zeroCount); + priority -= (_base * ((1 << zeroCount) * 2 - 1)) >> zeroCount; } if (priority > forwardPriority) @@ -150,7 +150,7 @@ namespace BizHawk.Client.Common } int zeroCount = _zeros[currentFrame & _mask]; - int priority = ((currentFrame - baseStateFrame) >> zeroCount); + int priority = (currentFrame - baseStateFrame) >> zeroCount; if (_align) { @@ -214,7 +214,7 @@ namespace BizHawk.Client.Common } // this is the kind of highly complex loops that might justify goto - next_state:; + next_state: ; } } @@ -225,8 +225,7 @@ namespace BizHawk.Client.Common _bits = bits; _mask = (1 << _bits) - 1; _base = (_capacity + _bits / 2) / (_bits + 1); - _zeros = new List(); - _zeros.Add(_bits); + _zeros = new List { _bits }; for (int i = 1; i < (1 << _bits); i++) { diff --git a/BizHawk.sln.DotSettings b/BizHawk.sln.DotSettings index 799d05463e..2f5c4a46d9 100644 --- a/BizHawk.sln.DotSettings +++ b/BizHawk.sln.DotSettings @@ -174,6 +174,7 @@ True True True + True True True True From 5075160355eaaf74fe3884f398afccf17c9dbec0 Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 18:38:56 -0500 Subject: [PATCH 27/27] some cleanups --- BizHawk.Client.Common/movie/tasproj/TasMovie.cs | 11 +---------- BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs | 7 ++++--- BizHawk.sln.DotSettings | 1 + 3 files changed, 6 insertions(+), 13 deletions(-) diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs index 9b26f14e53..ee482e958f 100644 --- a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs +++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs @@ -41,7 +41,6 @@ namespace BizHawk.Client.Common public TasMovieRecord this[int index] => new TasMovieRecord { - // State = StateManager[index], HasState = _stateManager.HasState(index), LogEntry = GetInputLogEntry(index), Lagged = _lagLog[index + 1], @@ -246,14 +245,6 @@ namespace BizHawk.Client.Common _lagLog.Clear(); } - public void DeleteLogBefore(int frame) - { - if (frame < Log.Count) - { - Log.RemoveRange(0, frame); - } - } - public void CopyLog(IEnumerable log) { Log.Clear(); @@ -276,7 +267,7 @@ namespace BizHawk.Client.Common return Log; } - private int? _timelineBranchFrame = null; + private int? _timelineBranchFrame; // TODO: this is 99% copy pasting of bad code public override bool ExtractInputLog(TextReader reader, out string errorMessage) diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs index c7a25d82c0..e7afee0d36 100644 --- a/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs +++ b/BizHawk.Client.EmuHawk/tools/TAStudio/TAStudio.cs @@ -739,7 +739,7 @@ namespace BizHawk.Client.EmuHawk return; } - MovieZone loadZone = new MovieZone(path) + var loadZone = new MovieZone(path) { Start = TasView.FirstSelectedIndex.Value }; @@ -864,7 +864,6 @@ namespace BizHawk.Client.EmuHawk TasView.Refresh(); - //SetSplicer(); CurrentTasMovie.FlushInputCache(); CurrentTasMovie.UseInputCache = false; @@ -895,11 +894,13 @@ namespace BizHawk.Client.EmuHawk private void StartAtNearestFrameAndEmulate(int frame, bool fromLua, bool fromRewinding) { if (frame == Emulator.Frame) + { return; + } _unpauseAfterSeeking = (fromRewinding || WasRecording) && !Mainform.EmulatorPaused; TastudioPlayMode(); - KeyValuePair closestState = CurrentTasMovie.TasStateManager.GetStateClosestToFrame(frame); + var closestState = CurrentTasMovie.TasStateManager.GetStateClosestToFrame(frame); if (closestState.Value.Length > 0 && (frame < Emulator.Frame || closestState.Key > Emulator.Frame)) { LoadState(closestState); diff --git a/BizHawk.sln.DotSettings b/BizHawk.sln.DotSettings index 2f5c4a46d9..f62ceeecdc 100644 --- a/BizHawk.sln.DotSettings +++ b/BizHawk.sln.DotSettings @@ -174,6 +174,7 @@ True True True + True True True True