2022-11-16 00:33:23 +01:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.IO.MemoryMappedFiles;
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
|
using System.Runtime.Serialization.Formatters.Binary;
|
|
|
|
|
|
using System.Text;
|
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
|
|
|
|
namespace CommonHelpers
|
|
|
|
|
|
{
|
2022-11-25 21:28:43 +01:00
|
|
|
|
public class SharedData<T> : IDisposable where T : struct
|
2022-11-16 00:33:23 +01:00
|
|
|
|
{
|
2022-11-25 21:28:43 +01:00
|
|
|
|
const int MMF_MAX_SIZE = 16384;
|
|
|
|
|
|
const int MMF_ALIGN_SIZE = 256;
|
2022-11-16 00:33:23 +01:00
|
|
|
|
|
|
|
|
|
|
private MemoryMappedFile mmf;
|
2022-11-25 21:28:43 +01:00
|
|
|
|
private int size;
|
2022-11-16 00:33:23 +01:00
|
|
|
|
|
2022-11-25 21:28:43 +01:00
|
|
|
|
private SharedData(int size)
|
|
|
|
|
|
{
|
|
|
|
|
|
this.size = size;
|
|
|
|
|
|
}
|
2022-11-16 00:33:23 +01:00
|
|
|
|
|
|
|
|
|
|
public T NewValue()
|
|
|
|
|
|
{
|
|
|
|
|
|
return default(T);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public bool GetValue(out T value)
|
|
|
|
|
|
{
|
|
|
|
|
|
using (MemoryMappedViewStream mmvStream = mmf.CreateViewStream())
|
|
|
|
|
|
{
|
|
|
|
|
|
value = default(T);
|
|
|
|
|
|
|
|
|
|
|
|
if (!mmvStream.CanRead)
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
2022-11-25 21:28:43 +01:00
|
|
|
|
byte[] buffer = new byte[size];
|
2022-11-16 00:33:23 +01:00
|
|
|
|
mmvStream.Read(buffer, 0, buffer.Length);
|
|
|
|
|
|
|
|
|
|
|
|
var handle = GCHandle.Alloc(buffer, GCHandleType.Pinned);
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
var output = Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(T));
|
|
|
|
|
|
if (output is null)
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
value = (T)output;
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch
|
|
|
|
|
|
{
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
finally
|
|
|
|
|
|
{
|
|
|
|
|
|
handle.Free();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public bool SetValue(T value)
|
|
|
|
|
|
{
|
|
|
|
|
|
using (MemoryMappedViewStream mmvStream = mmf.CreateViewStream())
|
|
|
|
|
|
{
|
|
|
|
|
|
if (!mmvStream.CanWrite)
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
2022-11-25 21:28:43 +01:00
|
|
|
|
byte[] buffer = new byte[size];
|
2022-11-16 00:33:23 +01:00
|
|
|
|
var handle = GCHandle.Alloc(buffer, GCHandleType.Pinned);
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
Marshal.StructureToPtr(value, handle.AddrOfPinnedObject(), false);
|
|
|
|
|
|
}
|
|
|
|
|
|
catch
|
|
|
|
|
|
{
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
finally
|
|
|
|
|
|
{
|
|
|
|
|
|
handle.Free();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
mmvStream.Write(buffer, 0, buffer.Length);
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static bool GetExistingValue(out T value)
|
|
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
using (var shared = OpenExisting())
|
|
|
|
|
|
{
|
|
|
|
|
|
if (shared.GetValue(out value))
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
catch
|
|
|
|
|
|
{
|
|
|
|
|
|
value = default(T);
|
|
|
|
|
|
}
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static bool SetExistingValue(T value)
|
|
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
using (var shared = OpenExisting())
|
|
|
|
|
|
{
|
|
|
|
|
|
if (shared.SetValue(value))
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
catch
|
|
|
|
|
|
{
|
|
|
|
|
|
}
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
|
|
{
|
|
|
|
|
|
mmf.Dispose();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static String GetUniqueName()
|
|
|
|
|
|
{
|
|
|
|
|
|
return String.Format("Global_{0}_Setting", typeof(T).Name);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2022-11-25 21:28:43 +01:00
|
|
|
|
private static int AlignedSize()
|
|
|
|
|
|
{
|
|
|
|
|
|
int size = Marshal.SizeOf<T>();
|
|
|
|
|
|
size = (size + MMF_ALIGN_SIZE - 1) / MMF_ALIGN_SIZE * MMF_ALIGN_SIZE;
|
|
|
|
|
|
if (size > MMF_MAX_SIZE)
|
|
|
|
|
|
throw new ArgumentException();
|
|
|
|
|
|
return size;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2022-11-16 00:33:23 +01:00
|
|
|
|
public static SharedData<T> CreateNew(String? name = null)
|
|
|
|
|
|
{
|
2022-11-25 21:28:43 +01:00
|
|
|
|
int size = AlignedSize();
|
|
|
|
|
|
|
|
|
|
|
|
return new SharedData<T>(size)
|
2022-11-16 00:33:23 +01:00
|
|
|
|
{
|
2022-11-25 21:28:43 +01:00
|
|
|
|
mmf = MemoryMappedFile.CreateOrOpen(name ?? GetUniqueName(), size)
|
2022-11-16 00:33:23 +01:00
|
|
|
|
};
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static SharedData<T> OpenExisting(String? name = null)
|
|
|
|
|
|
{
|
2022-11-25 21:28:43 +01:00
|
|
|
|
int size = AlignedSize();
|
|
|
|
|
|
|
|
|
|
|
|
return new SharedData<T>(size)
|
2022-11-16 00:33:23 +01:00
|
|
|
|
{
|
|
|
|
|
|
mmf = MemoryMappedFile.OpenExisting(name ?? GetUniqueName())
|
|
|
|
|
|
};
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|