forked from 47-studio-org/PostSharp.Samples
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathWeakEventAttribute.cs
218 lines (177 loc) · 5.29 KB
/
WeakEventAttribute.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
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
using PostSharp.Aspects;
using PostSharp.Serialization;
using System;
using System.Collections.Immutable;
using System.Reflection;
using System.Threading;
namespace PostSharp.Samples.WeakEvent
{
/// <summary>
/// Aspect that, when applied to an event, prevents the target event from holding a strong reference to event handlers.
/// Therefore, the aspect prevents the event to prevent clients to be garbage collected.
/// </summary>
[PSerializable]
[LinesOfCodeAvoided(6)]
public sealed class WeakEventAttribute : EventInterceptionAspect, IInstanceScopedAspect
{
[PNonSerialized] private volatile int cleanUpCounter;
[PNonSerialized] private ImmutableArray<object> handlers;
[PNonSerialized] private bool initialized;
[PNonSerialized] private SpinLock spinLock;
/// <summary>
/// Creates an instance of the aspect for a specific instance of the target class (when the target event is not
/// static). A call of this method is followed by a call of RuntimeInitializeInstance.
/// </summary>
object IInstanceScopedAspect.CreateInstance(AdviceArgs adviceArgs)
{
return MemberwiseClone();
}
/// <summary>
/// Initializes the current instance of the aspect (when the target event is not static).
/// </summary>
void IInstanceScopedAspect.RuntimeInitializeInstance()
{
if (!initialized)
{
cleanUpCounter = 0;
initialized = true;
handlers = ImmutableArray<object>.Empty;
}
}
/// <summary>
/// Initializes the current instance of the aspect (whether target event is static or not).
/// </summary>
public override void RuntimeInitialize(EventInfo eventInfo)
{
if (eventInfo.AddMethod.IsStatic)
{
if (!initialized)
{
cleanUpCounter = 0;
initialized = true;
handlers = ImmutableArray<object>.Empty;
}
}
}
#region Add
/// <summary>
/// Method invoked when (instead of) a new handler is added to the target event.
/// </summary>
/// <param name="args">Context information.</param>
public override void OnAddHandler(EventInterceptionArgs args)
{
// Add the handler to our own list.
if (AddHandler(args.Handler))
{
args.AddHandler(null);
}
// Register the handler to the client to prevent garbage collection of the handler.
DelegateReferenceKeeper.AddReference(args.Handler);
}
private bool AddHandler(Delegate handler)
{
var lockTaken = false;
try
{
spinLock.Enter(ref lockTaken);
handlers = handlers.Add(new WeakReference(handler));
return handlers.Length == 1;
}
finally
{
if (lockTaken)
{
spinLock.Exit();
}
}
}
#endregion
#region Remove
/// <summary>
/// Method invoked when (instead of) a new handler is removed from the target event.
/// </summary>
/// <param name="args">Context information.</param>
public override void OnRemoveHandler(EventInterceptionArgs args)
{
// Remove the handler from our own list.
if (RemoveHandler(args.Handler))
{
args.RemoveHandler(null);
}
// Remove the handler from the client.
DelegateReferenceKeeper.RemoveReference(args.Handler);
}
private bool RemoveHandler(Delegate handler)
{
var lockTaken = false;
try
{
spinLock.Enter(ref lockTaken);
handlers =
handlers.RemoveAll(
o => ((WeakReference) o).Target?.Equals(handler) ?? false);
return handlers.IsEmpty;
}
finally
{
if (lockTaken)
{
spinLock.Exit();
}
}
}
#endregion
#region Invoke
/// <summary>
/// Method invoked when (instead of) a the target event is raised.
/// </summary>
/// <param name="args">Context information.</param>
public override void OnInvokeHandler(EventInterceptionArgs args)
{
// Note that args.Handler == null because it's the value we added in OnAddHandler, but it really does not matter.
InvokeHandler(args.Arguments.ToArray());
}
private void InvokeHandler(object[] args)
{
var lastCleanUpCounter = -1;
// Take a snapshot of the handlers list.
var invocationList = handlers;
var needCleanUp = false;
foreach (var obj in invocationList)
{
var handler = (Delegate) ((WeakReference) obj).Target;
if (handler == null)
{
if (!needCleanUp)
{
needCleanUp = true;
lastCleanUpCounter = cleanUpCounter;
}
continue;
}
handler.DynamicInvoke(args);
}
if (needCleanUp && lastCleanUpCounter == cleanUpCounter)
{
if (lastCleanUpCounter == cleanUpCounter)
{
var lockTaken = false;
try
{
spinLock.Enter(ref lockTaken);
handlers = handlers.RemoveAll(w => !((WeakReference) w).IsAlive);
Interlocked.Increment(ref cleanUpCounter);
}
finally
{
if (lockTaken)
{
spinLock.Exit();
}
}
}
}
}
#endregion
}
}