File: compmod\system\diagnostics\TraceSource.cs
Project: ndp\fx\src\System.csproj (System)
//------------------------------------------------------------------------------
// <copyright file="TraceSource.cs" company="Microsoft">
//     Copyright (c) Microsoft Corporation.  All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
 
using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Threading;
using System.Configuration;
using System.Security;
using System.Security.Permissions;
using System.Diagnostics.CodeAnalysis;
 
namespace System.Diagnostics {
    public class TraceSource {
        private static List<WeakReference> tracesources =  new List<WeakReference>();
        private static int s_LastCollectionCount;
 
        private volatile SourceSwitch internalSwitch;
        private volatile TraceListenerCollection listeners;
        private StringDictionary attributes;
        private SourceLevels switchLevel;
        private volatile string sourceName;
        internal volatile bool _initCalled = false;   // Whether we've called Initialize already.
        
        public TraceSource(string name) 
            : this(name, SourceLevels.Off) {
        }
        
        public TraceSource(string name, SourceLevels defaultLevel) {
            if (name == null)
                throw new ArgumentNullException("name");
            if (name.Length == 0)
                throw new ArgumentException("name");
 
            sourceName = name;
            switchLevel = defaultLevel;
 
            // Delay load config to avoid perf (and working set) issues in retail
 
            // Add a weakreference to this source and cleanup invalid references
            lock(tracesources) {
                _pruneCachedTraceSources();
                tracesources.Add(new WeakReference(this));
            }
        }
 
        private static void _pruneCachedTraceSources() {
            lock (tracesources) {
                if (s_LastCollectionCount != GC.CollectionCount(2)) {
                    List<WeakReference> buffer = new List<WeakReference>(tracesources.Count);
                    for (int i = 0; i < tracesources.Count; i++) {
                        TraceSource tracesource = ((TraceSource)tracesources[i].Target);
                        if (tracesource != null) {
                            buffer.Add(tracesources[i]);
                        }
                    }
                    if (buffer.Count < tracesources.Count) {
                        tracesources.Clear();
                        tracesources.AddRange(buffer);
                        tracesources.TrimExcess();
                    }
                    s_LastCollectionCount = GC.CollectionCount(2);
                }
            }
        }
 
        private void Initialize() {
            if (!_initCalled) {
                lock(this) {
                    if (_initCalled)
                        return;
                    
                    SourceElementsCollection sourceElements = DiagnosticsConfiguration.Sources;
    
                    if (sourceElements != null) {
                        SourceElement sourceElement = sourceElements[sourceName];
                        if (sourceElement != null) {
                            if (!String.IsNullOrEmpty(sourceElement.SwitchName)) {
                                CreateSwitch(sourceElement.SwitchType, sourceElement.SwitchName);
                            }
                            else {
                                CreateSwitch(sourceElement.SwitchType, sourceName);
    
                                if (!String.IsNullOrEmpty(sourceElement.SwitchValue)) 
                                    internalSwitch.Level = (SourceLevels) Enum.Parse(typeof(SourceLevels), sourceElement.SwitchValue);
                            }
 
                            listeners = sourceElement.Listeners.GetRuntimeObject();
    
                            attributes = new StringDictionary();
                            TraceUtils.VerifyAttributes(sourceElement.Attributes, GetSupportedAttributes(), this);
                            attributes.ReplaceHashtable(sourceElement.Attributes);
                        }
                        else
                            NoConfigInit();
                    }
                    else
                        NoConfigInit();
    
                    _initCalled = true;
                }
            }
        }
         
        private void NoConfigInit() {
            internalSwitch = new SourceSwitch(sourceName, switchLevel.ToString());
            listeners = new TraceListenerCollection();
            listeners.Add(new DefaultTraceListener());
            attributes = null;
        }
        
        [SecurityPermission(SecurityAction.LinkDemand, Flags=SecurityPermissionFlag.UnmanagedCode)]
        public void Close() {
            // No need to call Initialize()
            if (listeners != null) {
                // Use global lock
                lock (TraceInternal.critSec) {
                    foreach (TraceListener listener in listeners) {
                        listener.Close();
                    }
                }
            }
        }
 
        public void Flush() {
            // No need to call Initialize()
            if (listeners != null) {
                if (TraceInternal.UseGlobalLock) {
                    lock (TraceInternal.critSec) {
                        foreach (TraceListener listener in listeners) {
                            listener.Flush();
                        }
                    }
                }
                else {
                    foreach (TraceListener listener in listeners) {
                        if (!listener.IsThreadSafe) {
                            lock (listener) {
                                listener.Flush();
                            }
                        }
                        else {
                            listener.Flush();
                        }
                    }            
                }
            }
        }
 
        virtual protected internal string[] GetSupportedAttributes() {
            return null;
        }
 
        internal static void RefreshAll() {
            lock (tracesources) {
                _pruneCachedTraceSources();
                for (int i=0; i<tracesources.Count; i++) {
                    TraceSource tracesource = ((TraceSource) tracesources[i].Target);
                    if (tracesource != null) {
                        tracesource.Refresh();
                    }
                }
            }
        }
        
        internal void Refresh() {
            if (!_initCalled) {
                Initialize();
                return;
            }
 
            SourceElementsCollection sources = DiagnosticsConfiguration.Sources;
 
            if (sources != null) {
                SourceElement sourceElement = sources[Name];
                if (sourceElement != null) {
 
                    // first check if the type changed
                    if ((String.IsNullOrEmpty(sourceElement.SwitchType) && internalSwitch.GetType() != typeof(SourceSwitch)) ||
                         (sourceElement.SwitchType != internalSwitch.GetType().AssemblyQualifiedName)) {
 
                        if (!String.IsNullOrEmpty(sourceElement.SwitchName)) {
                            CreateSwitch(sourceElement.SwitchType, sourceElement.SwitchName);
                        }
                        else {
                            CreateSwitch(sourceElement.SwitchType, Name);
                        
                            if (!String.IsNullOrEmpty(sourceElement.SwitchValue)) 
                                internalSwitch.Level = (SourceLevels) Enum.Parse(typeof(SourceLevels), sourceElement.SwitchValue);
                        }
                    }
                    else if (!String.IsNullOrEmpty(sourceElement.SwitchName)) {
                        // create a new switch if the name changed, otherwise just refresh. 
                        if (sourceElement.SwitchName != internalSwitch.DisplayName)
                            CreateSwitch(sourceElement.SwitchType, sourceElement.SwitchName);
                        else
                            internalSwitch.Refresh();
                    }
                    else {
                        // the switchValue changed.  Just update our internalSwitch. 
                        if (!String.IsNullOrEmpty(sourceElement.SwitchValue)) 
                            internalSwitch.Level = (SourceLevels) Enum.Parse(typeof(SourceLevels), sourceElement.SwitchValue);
                        else
                            internalSwitch.Level = SourceLevels.Off;
                    }
 
                    TraceListenerCollection newListenerCollection = new TraceListenerCollection();
                    foreach (ListenerElement listenerElement in sourceElement.Listeners) {
                        TraceListener listener = listeners[listenerElement.Name];
                        if (listener != null) {
                            newListenerCollection.Add(listenerElement.RefreshRuntimeObject(listener));
                        }
                        else {
                            newListenerCollection.Add(listenerElement.GetRuntimeObject());
                        }
                    }
 
                    TraceUtils.VerifyAttributes(sourceElement.Attributes, GetSupportedAttributes(), this);
 
                    attributes  = new StringDictionary();
                    attributes.ReplaceHashtable(sourceElement.Attributes);
 
                    listeners = newListenerCollection;
                }
                else {
                    // there was no config, so clear whatever we have.  
                    internalSwitch.Level = switchLevel;
                    listeners.Clear();
                    attributes = null;
                }
            }
        }
        
        [Conditional("TRACE")]
        public void TraceEvent(TraceEventType eventType, int id) {
            // Ensure that config is loaded 
            Initialize();
 
            TraceEventCache manager = new TraceEventCache();
 
            if (internalSwitch.ShouldTrace(eventType) && listeners != null) {
                if (TraceInternal.UseGlobalLock) {
                    // we lock on the same object that Trace does because we're writing to the same Listeners.
                    lock (TraceInternal.critSec) {
                        for (int i=0; i<listeners.Count; i++) {
                            TraceListener listener = listeners[i];
                            listener.TraceEvent(manager, Name, eventType, id);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
                else {
                    for (int i=0; i<listeners.Count; i++) {
                        TraceListener listener = listeners[i];
                        if (!listener.IsThreadSafe) {
                            lock (listener) {
                                listener.TraceEvent(manager, Name, eventType, id);
                                if (Trace.AutoFlush) listener.Flush();
                            }
                        }
                        else {
                            listener.TraceEvent(manager, Name, eventType, id);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
            }
        }
 
        [Conditional("TRACE")]
        public void TraceEvent(TraceEventType eventType, int id, string message) {
            // Ensure that config is loaded 
            Initialize();
 
            TraceEventCache manager = new TraceEventCache();            
 
            if (internalSwitch.ShouldTrace(eventType) && listeners != null) {
                if (TraceInternal.UseGlobalLock) {
                    // we lock on the same object that Trace does because we're writing to the same Listeners.
                    lock (TraceInternal.critSec) {
                        for (int i=0; i<listeners.Count; i++) {
                            TraceListener listener = listeners[i];
                            listener.TraceEvent(manager, Name, eventType, id, message);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
                else {
                    for (int i=0; i<listeners.Count; i++) {
                        TraceListener listener = listeners[i];
                        if (!listener.IsThreadSafe) {
                            lock (listener) {
                                listener.TraceEvent(manager, Name, eventType, id, message);
                                if (Trace.AutoFlush) listener.Flush();
                            }
                        }
                        else {
                            listener.TraceEvent(manager, Name, eventType, id, message);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
            }
        }
 
        [Conditional("TRACE")]
        public void TraceEvent(TraceEventType eventType, int id, string format, params object[] args) {
            // Ensure that config is loaded 
            Initialize();
 
            TraceEventCache manager = new TraceEventCache();
 
            if (internalSwitch.ShouldTrace(eventType) && listeners != null) {
                if (TraceInternal.UseGlobalLock) {
                    // we lock on the same object that Trace does because we're writing to the same Listeners.
                    lock (TraceInternal.critSec) {
                        for (int i=0; i<listeners.Count; i++) {
                            TraceListener listener = listeners[i];
                            listener.TraceEvent(manager, Name, eventType, id, format, args);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
                else {
                    for (int i=0; i<listeners.Count; i++) {
                        TraceListener listener = listeners[i];
                        if (!listener.IsThreadSafe) {
                            lock (listener) {
                                listener.TraceEvent(manager, Name, eventType, id, format, args);
                                if (Trace.AutoFlush) listener.Flush();
                            }
                        }
                        else {
                            listener.TraceEvent(manager, Name, eventType, id, format, args);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
            }
        }
 
        [Conditional("TRACE")]
        public void TraceData(TraceEventType eventType, int id, object data) {
            // Ensure that config is loaded 
            Initialize();
 
            TraceEventCache manager = new TraceEventCache();
             
            if (internalSwitch.ShouldTrace(eventType) && listeners != null) {
                if (TraceInternal.UseGlobalLock) {
                    // we lock on the same object that Trace does because we're writing to the same Listeners.
                    lock (TraceInternal.critSec) {
                        for (int i=0; i<listeners.Count; i++) {
                            TraceListener listener = listeners[i];
                            listener.TraceData(manager, Name, eventType, id, data);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
                else {
                    for (int i=0; i<listeners.Count; i++) {
                        TraceListener listener = listeners[i];
                        if (!listener.IsThreadSafe) {
                            lock (listener) {
                                listener.TraceData(manager, Name, eventType, id, data);
                                if (Trace.AutoFlush) listener.Flush();
                            }
                        }
                        else {
                            listener.TraceData(manager, Name, eventType, id, data);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
            }
        }
 
        [Conditional("TRACE")]
        public void TraceData(TraceEventType eventType, int id, params object[] data) {
            // Ensure that config is loaded 
            Initialize();
 
            TraceEventCache manager = new TraceEventCache();
             
            if (internalSwitch.ShouldTrace(eventType) && listeners != null) {
                if (TraceInternal.UseGlobalLock) {
                    // we lock on the same object that Trace does because we're writing to the same Listeners.
                    lock (TraceInternal.critSec) {
                        for (int i=0; i<listeners.Count; i++) {
                            TraceListener listener = listeners[i];
                            listener.TraceData(manager, Name, eventType, id, data);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
                else {
                    for (int i=0; i<listeners.Count; i++) {
                        TraceListener listener = listeners[i];
                        if (!listener.IsThreadSafe) {
                            lock (listener) {
                                listener.TraceData(manager, Name, eventType, id, data);
                                if (Trace.AutoFlush) listener.Flush();
                            }
                        }
                        else {
                            listener.TraceData(manager, Name, eventType, id, data);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
            }
        }
 
        [Conditional("TRACE")]
        public void TraceInformation(string message) { // eventType= TraceEventType.Info, id=0
            // No need to call Initialize()
            TraceEvent(TraceEventType.Information, 0, message, null);
        }
 
        [Conditional("TRACE")]
        public void TraceInformation(string format, params object[] args) {
            // No need to call Initialize()
            TraceEvent(TraceEventType.Information, 0, format, args);
        }
 
        [Conditional("TRACE")]
        public void TraceTransfer(int id, string message, Guid relatedActivityId) {
            // Ensure that config is loaded 
            Initialize();
 
            TraceEventCache manager = new TraceEventCache();
 
            if (internalSwitch.ShouldTrace(TraceEventType.Transfer) && listeners != null) {
                if (TraceInternal.UseGlobalLock) {
                    // we lock on the same object that Trace does because we're writing to the same Listeners.
                    lock (TraceInternal.critSec) {
                        for (int i=0; i<listeners.Count; i++) {
                            TraceListener listener = listeners[i];
                            listener.TraceTransfer(manager, Name, id, message, relatedActivityId);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
                else {
                    for (int i=0; i<listeners.Count; i++) {
                        TraceListener listener = listeners[i];
                        if (!listener.IsThreadSafe) {
                            lock (listener) {
                                listener.TraceTransfer(manager, Name, id, message, relatedActivityId);
                                if (Trace.AutoFlush) listener.Flush();
                            }
                        }
                        else {
                            listener.TraceTransfer(manager, Name, id, message, relatedActivityId);
                            if (Trace.AutoFlush) listener.Flush();
                        }
                    }
                }
            }
        }
 
        private void CreateSwitch(string typename, string name) {
            if (!String.IsNullOrEmpty(typename))
                internalSwitch = (SourceSwitch) TraceUtils.GetRuntimeObject(typename, typeof(SourceSwitch), name);
            else
                internalSwitch = new SourceSwitch(name, switchLevel.ToString());
        }
        
        public StringDictionary Attributes {
            get {
                // Ensure that config is loaded 
                Initialize();
 
                if (attributes == null)
                    attributes  = new StringDictionary();
 
                return attributes;
            }
        }
 
        public string Name {
            get {
                return sourceName;
            }
        }
 
        public TraceListenerCollection Listeners {
            [SecurityPermission(SecurityAction.LinkDemand, Flags=SecurityPermissionFlag.UnmanagedCode)]
            get {
                // Ensure that config is loaded 
                Initialize();
 
                return listeners;
            }
        }
 
        public SourceSwitch Switch {
            // No need for security demand here. SourceSwitch.set_Level is protected already.
            get {
                // Ensure that config is loaded 
                Initialize();
 
                return internalSwitch;
            }
            [SecurityPermission(SecurityAction.LinkDemand, Flags=SecurityPermissionFlag.UnmanagedCode)]
            set {
                if (value == null)
                    throw new ArgumentNullException("Switch");
 
                // Ensure that config is loaded 
                Initialize();
 
                internalSwitch = value;
            }
        }
    }
}