-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCacheTable.cs
151 lines (127 loc) · 5.4 KB
/
CacheTable.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
#region Copyright (c) Pixeval.Caching/Pixeval.Caching
// GPL v3 License
//
// Pixeval.Caching/Pixeval.Caching
// Copyright (c) 2024 Pixeval.Caching/CacheTable.cs
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
#endregion
using System.Reflection;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
namespace Pixeval.Caching;
public class CacheTable<TKey, THeader, TProtocol>(
TProtocol protocol,
CacheToken token)
where THeader : unmanaged
where TKey : IEquatable<TKey>
where TProtocol : ICacheProtocol<TKey, THeader>
{
public MemoryMappedFileMemoryManager MemoryManager { get; } = new(token);
private Dictionary<TKey, (nint ptr, int allocatedLength)> _cacheTable = [];
private PriorityQueue<TKey, int> _lruCacheIndex = new(Comparer<int>.Create((x, y) => y.CompareTo(x)));
// ReSharper disable once InconsistentNaming
public int CacheLRUFactor { get; set; } = 2;
private TProtocol _protocol = protocol;
/// <summary>
/// While calling this method, all cache operations should be halted.
/// </summary>
private unsafe void PurgeCompact()
{
var retain = _lruCacheIndex.Count / CacheLRUFactor;
var newPriorityQueue = new PriorityQueue<TKey, int>();
for (var i = 0; i < retain; i++)
{
_lruCacheIndex.TryDequeue(out var element, out var priority);
newPriorityQueue.Enqueue(element!, priority);
}
var garbage = new Dictionary<nint, int>();
foreach (var key in _lruCacheIndex.UnorderedItems.Select(x => x.Element))
{
if (TryReadCache0(key, out var span, true))
{
var pointer = (byte*) Unsafe.AsPointer(ref MemoryMarshal.GetReference(span)) - TProtocol.GetHeaderLength();
garbage[(nint) pointer] = span.Length;
}
}
var grouped = _cacheTable.Values.ToDictionary().GroupBy(
tuple => MemoryManager.BumpPointerAllocators.First(pair => pair.Value.GetBlock((byte*) tuple.Key) != default).Value,
tuple => tuple);
foreach (var group in grouped)
{
var replacement = group.Key.Compact(group.ToDictionary(tuple => tuple.Key, tuple => tuple.Value), garbage.Keys.ToHashSet());
// forward reference
_cacheTable = _cacheTable.SelectMany(pair =>
{
return replacement.TryGetValue(pair.Value.ptr, out var newPointer)
? new[] { KeyValuePair.Create(pair.Key, (newPointer, pair.Value.allocatedLength)) }
: new[] { pair };
}).ToDictionary();
}
_lruCacheIndex = newPriorityQueue;
}
public unsafe AllocatorState TryCache(TKey key, Span<byte> span)
{
return TryCache0(key, span, false);
}
private unsafe AllocatorState TryCache0(TKey key, Span<byte> span, bool collected)
{
if (_cacheTable.ContainsKey(key))
{
return AllocatorState.AllocationSuccess;
}
var header = _protocol.SerializeHeader(_protocol.GetHeader(key));
var result = MemoryManager.DominantAllocator.TryAllocate(header.Length + _protocol.GetDataLength(_protocol.GetHeader(key)), out var cacheArea);
if (result is AllocatorState.AllocationSuccess)
{
header.CopyTo(cacheArea);
span.CopyTo(cacheArea[header.Length..]);
_cacheTable[key] = ((nint)Unsafe.AsPointer(ref cacheArea.GetPinnableReference()), cacheArea.Length);
_lruCacheIndex.Enqueue(key, 0);
return AllocatorState.AllocationSuccess;
}
if (result is AllocatorState.OutOfMemory)
{
if (collected) return result;
PurgeCompact();
// ReSharper disable once TailRecursiveCall
return TryCache0(key, span, true);
}
return result;
}
public bool TryReadCache(TKey key, out Span<byte> span)
{
return TryReadCache0(key, out span, false);
}
private unsafe bool TryReadCache0(TKey key, out Span<byte> span, bool transparent)
{
if (_cacheTable.TryGetValue(key, out var tuple))
{
var headerLength = TProtocol.GetHeaderLength();
var header = new Span<byte>((byte*) tuple.ptr, headerLength);
var headerStruct = _protocol.DeserializeHeader(header);
var dataLength = _protocol.GetDataLength(headerStruct);
var totalSpan = new Span<byte>((void*) tuple.ptr, tuple.allocatedLength);
span = totalSpan[headerLength..(headerLength + dataLength)];
if (!transparent)
{
_lruCacheIndex.Remove(key, out _, out var oldPriority);
_lruCacheIndex.Enqueue(key, oldPriority + 1);
}
return true;
}
span = Span<byte>.Empty;
return false;
}
}