-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPinned.cs
69 lines (62 loc) · 1.91 KB
/
Pinned.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
using System;
using System.Runtime.InteropServices;
namespace GSSAPI.Utility
{
/// <summary>
/// Memory pinned object
/// </summary>
public static class Pinned
{
/// <summary>
/// Create memory pinned object from <paramref name="value"/>
/// </summary>
/// <typeparam name="T">Any class type</typeparam>
/// <param name="value">Value to pin</param>
/// <returns>Pinned value</returns>
public static Pinned<T> From<T>(T value) where T : class => new Pinned<T>(value);
}
/// <summary>
/// Memory pinned object
/// </summary>
/// <typeparam name="T">Any class type</typeparam>
public sealed class Pinned<T> : IDisposable where T : class
{
/// <summary>
/// Original object value, can be used with <code>ref</code>
/// </summary>
public T Value;
/// <summary>
/// In memory address of the object
/// </summary>
public IntPtr Addr { get; }
private GCHandle _handle;
/// <summary>
/// Create memory pinned object from <paramref name="value"/>
/// </summary>
/// <param name="value">Value to pin</param>
public Pinned(T value)
{
Value = value;
_handle = GCHandle.Alloc(value, GCHandleType.Pinned);
Addr = _handle.AddrOfPinnedObject();
}
/// <summary>
/// Returns address of object in memory
/// </summary>
public static implicit operator IntPtr(Pinned<T> p)
{
return p.Addr;
}
/// <summary>
/// Returns original object value
/// </summary>
public static implicit operator T(Pinned<T> p)
{
return p.Value;
}
public void Dispose()
{
_handle.Free();
}
}
}