##// END OF EJS Templates
Working on Implab.Diagnostics
Working on Implab.Diagnostics

File last commit:

r286:67ebcfd7d1c8 v3
r286:67ebcfd7d1c8 v3
Show More
TraceRegistry.cs
82 lines | 2.7 KiB | text/x-csharp | CSharpLexer
using System;
using System.Collections.Generic;
using System.Diagnostics;
using Implab.Parallels;
namespace Implab.Diagnostics {
public class TraceRegistry {
class Subscription : IDisposable {
readonly WeakReference<TraceRegistry> m_registry;
readonly Action<object> m_unsubscribe;
public Subscription(TraceRegistry registry) {
m_registry = new WeakReference<TraceRegistry>(registry);
}
public void Dispose() {
TraceRegistry t;
if (m_registry.TryGetTarget(out t))
t.RemoveSubscription(this);
}
}
public static TraceRegistry Global { get; } = new TraceRegistry();
readonly object m_lock = new object();
readonly Dictionary<object, Action<TraceChannel>> m_subscriptions = new Dictionary<object, Action<TraceChannel>>();
readonly SimpleAsyncQueue<TraceChannel> m_channels = new SimpleAsyncQueue<TraceChannel>();
internal void Register(TraceChannel channel) {
// notifications can run in parallel
Action<TraceChannel>[] handlers = null;
lock(m_lock) {
m_channels.Enqueue(channel);
if (m_subscriptions.Count > 0) {
handlers = new Action<TraceChannel>[m_subscriptions.Count];
m_subscriptions.Values.CopyTo(handlers, 0);
}
}
if (handlers != null)
foreach(var h in handlers)
h(channel);
}
/// <summary>
/// Subscribes the specified handler to notifications about new trace
/// channels
/// </summary>
/// <param name="handler"></param>
/// <returns></returns>
public IDisposable Subscribe(Action<TraceChannel> handler, bool existing) {
Safe.ArgumentNotNull(handler, nameof(handler));
var cookie = new Subscription(this);
IEnumerable<TraceChannel> snap;
// lock to ensure that no new channels will be added
// while the subscription is added
lock(m_lock) {
m_subscriptions.Add(cookie, handler);
snap = m_channels.Snapshot();
}
// announce previously declared channels if required
if (existing) {
foreach(var c in snap)
handler(c);
}
// return the subscription
return cookie;
}
void RemoveSubscription(object cookie) {
lock(m_lock)
m_subscriptions.Remove(cookie);
}
}
}