2024-03-05 16:26:30 +08:00
|
|
|
|
#region
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2024-03-16 14:57:11 +08:00
|
|
|
|
using System.Runtime.CompilerServices;
|
|
|
|
|
using UnityEngine;
|
2024-03-05 16:26:30 +08:00
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
namespace nadena.dev.modular_avatar.core.armature_lock
|
|
|
|
|
{
|
2024-03-11 09:32:57 +08:00
|
|
|
|
internal interface ISegment : IDisposable
|
2024-03-05 16:26:30 +08:00
|
|
|
|
{
|
|
|
|
|
AllocationMap.DefragmentCallback Defragment { get; set; }
|
|
|
|
|
int Offset { get; }
|
|
|
|
|
int Length { get; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
internal class AllocationMap
|
|
|
|
|
{
|
|
|
|
|
public delegate void DefragmentCallback(int oldOffset, int newOffset, int length);
|
|
|
|
|
|
|
|
|
|
// Visible for unit tests
|
|
|
|
|
internal class Segment : ISegment
|
|
|
|
|
{
|
|
|
|
|
public int _offset;
|
|
|
|
|
public int _length;
|
|
|
|
|
public bool _inUse;
|
|
|
|
|
|
2024-03-11 09:32:57 +08:00
|
|
|
|
private Action<Segment> _onDispose;
|
|
|
|
|
public DefragmentCallback Defragment { get; set; }
|
2024-03-05 16:26:30 +08:00
|
|
|
|
public int Offset => _offset;
|
|
|
|
|
public int Length => _length;
|
|
|
|
|
|
2024-03-11 09:32:57 +08:00
|
|
|
|
internal Segment(Action<Segment> onDispose, int offset, int length, bool inUse)
|
2024-03-05 16:26:30 +08:00
|
|
|
|
{
|
2024-03-11 09:32:57 +08:00
|
|
|
|
_onDispose = onDispose;
|
2024-03-05 16:26:30 +08:00
|
|
|
|
_offset = offset;
|
|
|
|
|
_length = length;
|
|
|
|
|
_inUse = inUse;
|
|
|
|
|
}
|
2024-03-11 09:32:57 +08:00
|
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
_onDispose?.Invoke(this);
|
|
|
|
|
_onDispose = null;
|
|
|
|
|
}
|
2024-03-05 16:26:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// A list of allocated (and unallocated) segments.
|
|
|
|
|
///
|
|
|
|
|
/// Invariant: The last element (if any) is always inUse.
|
|
|
|
|
/// Invariant: No two consecutive elements are free (inUse = false).
|
|
|
|
|
///
|
|
|
|
|
/// </summary>
|
|
|
|
|
List<Segment> segments = new List<Segment>();
|
|
|
|
|
|
|
|
|
|
public ISegment Allocate(int requestedLength)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < segments.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
var segment = segments[i];
|
|
|
|
|
if (segment._inUse) continue;
|
|
|
|
|
|
|
|
|
|
if (segment._length == requestedLength)
|
|
|
|
|
{
|
|
|
|
|
segment._inUse = true;
|
|
|
|
|
return segment;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (segment._length > requestedLength)
|
|
|
|
|
{
|
|
|
|
|
var remaining = new Segment(
|
2024-03-11 09:32:57 +08:00
|
|
|
|
FreeSegment,
|
2024-03-05 16:26:30 +08:00
|
|
|
|
segment._offset + requestedLength,
|
|
|
|
|
segment._length - requestedLength,
|
|
|
|
|
false
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
segment._length = requestedLength;
|
|
|
|
|
segment._inUse = true;
|
|
|
|
|
segments.Insert(i + 1, remaining);
|
|
|
|
|
|
|
|
|
|
return segment;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Add a new in-use segment at the end
|
|
|
|
|
var newSegment = new Segment(
|
2024-03-11 09:32:57 +08:00
|
|
|
|
FreeSegment,
|
2024-03-05 16:26:30 +08:00
|
|
|
|
segments.Count == 0 ? 0 : segments[segments.Count - 1]._offset + segments[segments.Count - 1]._length,
|
|
|
|
|
requestedLength,
|
|
|
|
|
true
|
|
|
|
|
);
|
|
|
|
|
segments.Add(newSegment);
|
|
|
|
|
|
|
|
|
|
return newSegment;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void FreeSegment(ISegment inputSegment)
|
|
|
|
|
{
|
|
|
|
|
var s = inputSegment as Segment;
|
|
|
|
|
if (s == null) throw new ArgumentException("Passed a foreign segment???");
|
|
|
|
|
|
|
|
|
|
int index = segments.BinarySearch(s, Comparer<Segment>.Create((a, b) => a._offset.CompareTo(b._offset)));
|
2024-03-16 14:57:11 +08:00
|
|
|
|
if (index < 0 || segments[index] != s)
|
|
|
|
|
{
|
|
|
|
|
var segmentDump = string.Join("\n", segments.ConvertAll(seg => $"{seg._offset} {seg._length} {seg._inUse} id={RuntimeHelpers.GetHashCode(seg)}"));
|
|
|
|
|
segmentDump += "\n\nTarget segment " + s._offset + " " + s._length + " " + s._inUse + " id=" + RuntimeHelpers.GetHashCode(s);
|
|
|
|
|
|
|
|
|
|
throw new Exception("Segment not found in FreeSegment\nCurrent segments:\n" + segmentDump);
|
|
|
|
|
}
|
2024-03-05 16:26:30 +08:00
|
|
|
|
|
|
|
|
|
if (index == segments.Count - 1)
|
|
|
|
|
{
|
|
|
|
|
segments.RemoveAt(index);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (index + 1 < segments.Count)
|
|
|
|
|
{
|
|
|
|
|
var next = segments[index + 1];
|
|
|
|
|
if (!next._inUse)
|
|
|
|
|
{
|
|
|
|
|
next._offset = s._offset;
|
|
|
|
|
next._length += s._length;
|
|
|
|
|
segments.RemoveAt(index);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Replace with a fresh segment object to avoid any issues with leaking old references to the segment
|
2024-03-11 09:32:57 +08:00
|
|
|
|
segments[index] = new Segment(FreeSegment, s._offset, s._length, false);
|
2024-03-05 16:26:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Defragments all free space. When a segment is moved, the passed callback is called with the old and new offsets,
|
|
|
|
|
/// and then the callback associated with the segment (if any) is also invoked.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="callback"></param>
|
2024-03-11 09:32:57 +08:00
|
|
|
|
public void Defragment(DefragmentCallback callback)
|
2024-03-05 16:26:30 +08:00
|
|
|
|
{
|
|
|
|
|
int offset = 0;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < segments.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
var seg = segments[i];
|
|
|
|
|
if (!seg._inUse)
|
|
|
|
|
{
|
|
|
|
|
segments.RemoveAt(i);
|
|
|
|
|
i--;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (seg._offset != offset)
|
|
|
|
|
{
|
|
|
|
|
callback(seg._offset, offset, seg._length);
|
|
|
|
|
seg.Defragment?.Invoke(seg._offset, offset, seg._length);
|
|
|
|
|
seg._offset = offset;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
offset += seg.Length;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|