Debug.cs 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Diagnostics;
  4. using System.Reflection;
  5. using System.Runtime.CompilerServices;
  6. using System.Text;
  7. namespace BansheeEngine
  8. {
  9. /// <summary>
  10. /// Possible types of debug messages.
  11. /// </summary>
  12. public enum DebugMessageType // Note: Must match C++ enum DebugChannel
  13. {
  14. Info, Warning, Error
  15. }
  16. /// <summary>
  17. /// Utility class providing various debug functionality.
  18. /// </summary>
  19. public sealed class Debug
  20. {
  21. /// <summary>
  22. /// Triggered when a new message is added to the debug log.
  23. /// </summary>
  24. public static Action<DebugMessageType, string> OnAdded;
  25. /// <summary>
  26. /// Logs a new informative message to the global debug log.
  27. /// </summary>
  28. /// <param name="message">Message to log.</param>
  29. public static void Log(object message)
  30. {
  31. StringBuilder sb = new StringBuilder();
  32. sb.AppendLine(message.ToString());
  33. sb.Append(GetStackTrace());
  34. Internal_Log(sb.ToString());
  35. }
  36. /// <summary>
  37. /// Logs a new warning message to the global debug log.
  38. /// </summary>
  39. /// <param name="message">Message to log.</param>
  40. public static void LogWarning(object message)
  41. {
  42. StringBuilder sb = new StringBuilder();
  43. sb.AppendLine(message.ToString());
  44. sb.Append(GetStackTrace());
  45. Internal_LogWarning(sb.ToString());
  46. }
  47. /// <summary>
  48. /// Logs a new error message to the global debug log.
  49. /// </summary>
  50. /// <param name="message">Message to log.</param>
  51. public static void LogError(object message)
  52. {
  53. StringBuilder sb = new StringBuilder();
  54. sb.AppendLine(message.ToString());
  55. sb.Append(GetStackTrace());
  56. Internal_LogError(sb.ToString());
  57. }
  58. /// <summary>
  59. /// Returns the stack trace of the current point in code.
  60. /// </summary>
  61. /// <returns>String containing the stack trace.</returns>
  62. public static string GetStackTrace()
  63. {
  64. StackTrace stackTrace = new StackTrace(1, true);
  65. StackFrame[] frames = stackTrace.GetFrames();
  66. if (frames == null)
  67. return "";
  68. StringBuilder sb = new StringBuilder();
  69. foreach (var frame in frames)
  70. {
  71. MethodBase method = frame.GetMethod();
  72. if (method == null)
  73. continue;
  74. Type parentType = method.DeclaringType;
  75. if (parentType == null)
  76. continue;
  77. sb.Append("\tat " + parentType.Name + "." + method.Name + "(");
  78. ParameterInfo[] methodParams = method.GetParameters();
  79. for(int i = 0; i < methodParams.Length; i++)
  80. {
  81. if (i > 0)
  82. sb.Append(", ");
  83. sb.Append(methodParams[i].ParameterType.Name);
  84. }
  85. sb.Append(")");
  86. string ns = parentType.Namespace;
  87. string fileName = frame.GetFileName();
  88. if (!string.IsNullOrEmpty(fileName))
  89. {
  90. int line = frame.GetFileLineNumber();
  91. int column = frame.GetFileColumnNumber();
  92. sb.Append(" in " + fileName + ", line " + line + ", column " + column + ", namespace " + ns);
  93. }
  94. else
  95. {
  96. if (!string.IsNullOrEmpty(ns))
  97. sb.Append(" in namespace " + ns);
  98. }
  99. sb.AppendLine();
  100. }
  101. return sb.ToString();
  102. }
  103. /// <summary>
  104. /// Triggered by the runtime when a new message is added to the debug log.
  105. /// </summary>
  106. /// <param name="type">Type of the message that was added.</param>
  107. /// <param name="message">Text of the message.</param>
  108. private static void Internal_OnAdded(DebugMessageType type, string message)
  109. {
  110. if (OnAdded != null)
  111. OnAdded(type, message);
  112. }
  113. [MethodImpl(MethodImplOptions.InternalCall)]
  114. internal static extern void Internal_Log(string message);
  115. [MethodImpl(MethodImplOptions.InternalCall)]
  116. internal static extern void Internal_LogWarning(string message);
  117. [MethodImpl(MethodImplOptions.InternalCall)]
  118. internal static extern void Internal_LogError(string message);
  119. }
  120. }