ikvm-fork/runtime/tracer.cs

221 строка
5.9 KiB
C#

/*
Copyright (C) 2004, 2005, 2006 Jeroen Frijters
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
Jeroen Frijters
jeroen@frijters.net
*/
using System;
using System.Threading;
using System.Diagnostics;
using System.Collections.Generic;
using System.Text.RegularExpressions;
using System.Configuration;
namespace IKVM.Internal
{
public static class Tracer
{
public readonly static TraceSwitch Compiler = new TraceSwitch("compiler", "Static Compiler");
public readonly static TraceSwitch FxBug = new TraceSwitch("fxbug", ".NET Framework bug related events");
public readonly static TraceSwitch ClassLoading = new TraceSwitch("classloading", "Class loading");
public readonly static TraceSwitch Verifier = new TraceSwitch("verifier", "Bytecode Verifier");
public readonly static TraceSwitch Runtime = new TraceSwitch("runtime", "Miscellaneous runtime events");
public readonly static TraceSwitch Jni = new TraceSwitch("jni", "JNI");
// public readonly static TraceSwitch Methods = new TraceSwitch("methods", "Method Trace");
private readonly static Dictionary<string, TraceSwitch> allTraceSwitches = new Dictionary<string, TraceSwitch>();
private readonly static List<string> methodtraces = new List<string>();
private class MyTextWriterTraceListener : TextWriterTraceListener
{
internal MyTextWriterTraceListener(System.IO.TextWriter tw)
: base(tw)
{
}
public override void Fail(string message)
{
this.WriteLine("Assert.Fail: " + message);
this.WriteLine(new StackTrace(true).ToString());
base.Fail(message);
}
public override void Fail(string message, string detailMessage)
{
this.WriteLine("Assert.Fail: " + message + ".\n" + detailMessage);
this.WriteLine(new StackTrace(true).ToString());
base.Fail(message, detailMessage);
}
}
static Tracer()
{
try
{
Init();
}
catch(System.Security.SecurityException)
{
}
}
private static void Init()
{
allTraceSwitches[Compiler.DisplayName] = Compiler;
allTraceSwitches[FxBug.DisplayName] = FxBug;
allTraceSwitches[ClassLoading.DisplayName] = ClassLoading;
allTraceSwitches[Verifier.DisplayName] = Verifier;
allTraceSwitches[Runtime.DisplayName] = Runtime;
allTraceSwitches[Jni.DisplayName] = Jni;
try
{
Trace.AutoFlush = true;
/* If the app config file gives some method trace - add it */
string trace = ConfigurationManager.AppSettings["Traced Methods"];
if(trace != null)
{
methodtraces.Add(trace);
}
}
catch(ConfigurationException)
{
// app.config is malformed, ignore
}
}
[Conditional("DEBUG")]
public static void EnableTraceForDebug()
{
SetTraceLevel("*", TraceLevel.Verbose);
}
public static void EnableTraceConsoleListener()
{
Trace.Listeners.Add(new MyTextWriterTraceListener(Console.Error));
}
public static void HandleMethodTrace(string name)
{
methodtraces.Add(name);
}
public static bool IsTracedMethod(string name)
{
if(methodtraces.Count == 0)
{
return false;
}
IEnumerator<string> e = methodtraces.GetEnumerator();
while(e.MoveNext())
{
try
{
if(Regex.IsMatch(name, e.Current))
{
return true;
}
}
catch(Exception x)
{
Tracer.Warning(Tracer.Compiler,"Regular Expression match failure: " + e.Current + ":" + x);
}
}
return false;
}
public static void SetTraceLevel(string name)
{
string[] trace = name.Split('=');
System.Diagnostics.TraceLevel level = System.Diagnostics.TraceLevel.Verbose;
if(trace.Length == 2)
{
level = (System.Diagnostics.TraceLevel)Enum.Parse(typeof(System.Diagnostics.TraceLevel), trace[1], true);
}
SetTraceLevel(trace[0], level);
}
public static void SetTraceLevel(string name, TraceLevel level)
{
if(name == "*")
{
foreach(TraceSwitch ts in allTraceSwitches.Values)
{
ts.Level = level;
}
}
else
{
TraceSwitch ts;
if(allTraceSwitches.TryGetValue(name, out ts))
{
ts.Level = level;
}
else
{
Console.Error.WriteLine("Warning: Invalid traceswitch name: {0}", name);
}
}
}
private static void WriteLine(string message, object[] p)
{
if(p.Length > 0)
{
message = string.Format(message, p);
}
Trace.WriteLine(string.Format("[{0:HH':'mm':'ss'.'fffff} {1}] {2}", DateTime.Now, Thread.CurrentThread.Name, message));
}
[Conditional("TRACE")]
public static void Info(TraceSwitch traceSwitch, string message, params object[] p)
{
if(traceSwitch.TraceInfo)
{
WriteLine(message, p);
}
}
[Conditional("TRACE")]
public static void MethodInfo(string message)
{
Trace.WriteLine(string.Format("[{0:HH':'mm':'ss'.'fffff} {1}] {2}", DateTime.Now, Thread.CurrentThread.Name, message));
}
[Conditional("TRACE")]
public static void Warning(TraceSwitch traceSwitch, string message, params object[] p)
{
if(traceSwitch.TraceWarning)
{
WriteLine(message, p);
}
}
[Conditional("TRACE")]
public static void Error(TraceSwitch traceSwitch, string message, params object[] p)
{
if(traceSwitch.TraceError)
{
WriteLine(message, p);
}
}
}
}