From b1296dd9bb0f37534d00d6387f4d3eb91e8234bf Mon Sep 17 00:00:00 2001 From: adelikat Date: Sat, 15 Jun 2019 15:13:08 -0500 Subject: [PATCH] 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