Logging.cs 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. using Ryujinx.Core.OsHle.Ipc;
  2. using System;
  3. using System.Diagnostics;
  4. using System.IO;
  5. using System.Runtime.CompilerServices;
  6. using System.Text;
  7. namespace Ryujinx.Core
  8. {
  9. public static class Logging
  10. {
  11. private static Stopwatch ExecutionTime;
  12. private const string LogFileName = "Ryujinx.log";
  13. private static bool EnableInfo = Config.LoggingEnableInfo;
  14. private static bool EnableTrace = Config.LoggingEnableTrace;
  15. private static bool EnableDebug = Config.LoggingEnableDebug;
  16. private static bool EnableWarn = Config.LoggingEnableWarn;
  17. private static bool EnableError = Config.LoggingEnableError;
  18. private static bool EnableFatal = Config.LoggingEnableFatal;
  19. private static bool EnableIpc = Config.LoggingEnableIpc;
  20. private static bool EnableLogFile = Config.LoggingEnableLogFile;
  21. private enum LogLevel
  22. {
  23. Debug = 1,
  24. Error = 2,
  25. Fatal = 3,
  26. Info = 4,
  27. Trace = 5,
  28. Warn = 6
  29. }
  30. static Logging()
  31. {
  32. if (File.Exists(LogFileName)) File.Delete(LogFileName);
  33. ExecutionTime = new Stopwatch();
  34. ExecutionTime.Start();
  35. }
  36. public static string GetExecutionTime() => ExecutionTime.ElapsedMilliseconds.ToString().PadLeft(8, '0') + "ms";
  37. private static void LogMessage(LogEntry LogEntry)
  38. {
  39. ConsoleColor consoleColor = ConsoleColor.White;
  40. switch (LogEntry.LogLevel)
  41. {
  42. case LogLevel.Debug:
  43. consoleColor = ConsoleColor.Gray;
  44. break;
  45. case LogLevel.Error:
  46. consoleColor = ConsoleColor.Red;
  47. break;
  48. case LogLevel.Fatal:
  49. consoleColor = ConsoleColor.Magenta;
  50. break;
  51. case LogLevel.Info:
  52. consoleColor = ConsoleColor.White;
  53. break;
  54. case LogLevel.Trace:
  55. consoleColor = ConsoleColor.DarkGray;
  56. break;
  57. case LogLevel.Warn:
  58. consoleColor = ConsoleColor.Yellow;
  59. break;
  60. }
  61. string Text = $"{LogEntry.ExecutionTime} | {LogEntry.LogLevel.ToString()} > " +
  62. $"{LogEntry.CallingMember}:{LogEntry.CallingLineNumber} > {LogEntry.Message}";
  63. Console.ForegroundColor = consoleColor;
  64. Console.WriteLine(Text.PadLeft(Text.Length + 1, ' '));
  65. Console.ResetColor();
  66. LogFile(Text);
  67. }
  68. private static void LogFile(string Message)
  69. {
  70. if (EnableLogFile)
  71. {
  72. using (StreamWriter Writer = File.AppendText(LogFileName))
  73. {
  74. Writer.WriteLine(Message);
  75. }
  76. }
  77. }
  78. public static void Info(string Message,
  79. [CallerMemberName] string CallingMember = "",
  80. [CallerLineNumber] int CallingLineNumber = 0)
  81. {
  82. if (EnableInfo)
  83. {
  84. LogMessage(new LogEntry
  85. {
  86. CallingLineNumber = CallingLineNumber,
  87. CallingMember = CallingMember,
  88. LogLevel = LogLevel.Info,
  89. Message = Message,
  90. ExecutionTime = GetExecutionTime()
  91. });
  92. }
  93. }
  94. public static void Trace(string Message,
  95. [CallerMemberName] string CallingMember = "",
  96. [CallerLineNumber] int CallingLineNumber = 0)
  97. {
  98. if (EnableTrace)
  99. {
  100. LogMessage(new LogEntry
  101. {
  102. CallingLineNumber = CallingLineNumber,
  103. CallingMember = CallingMember,
  104. LogLevel = LogLevel.Trace,
  105. Message = Message,
  106. ExecutionTime = GetExecutionTime()
  107. });
  108. }
  109. }
  110. public static void Debug(string Message,
  111. [CallerMemberName] string CallingMember = "",
  112. [CallerLineNumber] int CallingLineNumber = 0)
  113. {
  114. if (EnableDebug)
  115. {
  116. LogMessage(new LogEntry
  117. {
  118. CallingLineNumber = CallingLineNumber,
  119. CallingMember = CallingMember,
  120. LogLevel = LogLevel.Debug,
  121. Message = Message,
  122. ExecutionTime = GetExecutionTime()
  123. });
  124. }
  125. }
  126. public static void Warn(string Message,
  127. [CallerMemberName] string CallingMember = "",
  128. [CallerLineNumber] int CallingLineNumber = 0)
  129. {
  130. if (EnableWarn)
  131. {
  132. LogMessage(new LogEntry
  133. {
  134. CallingLineNumber = CallingLineNumber,
  135. CallingMember = CallingMember,
  136. LogLevel = LogLevel.Warn,
  137. Message = Message,
  138. ExecutionTime = GetExecutionTime()
  139. });
  140. }
  141. }
  142. public static void Error(string Message,
  143. [CallerMemberName] string CallingMember = "",
  144. [CallerLineNumber] int CallingLineNumber = 0)
  145. {
  146. if (EnableError)
  147. {
  148. LogMessage(new LogEntry
  149. {
  150. CallingLineNumber = CallingLineNumber,
  151. CallingMember = CallingMember,
  152. LogLevel = LogLevel.Error,
  153. Message = Message,
  154. ExecutionTime = GetExecutionTime()
  155. });
  156. }
  157. }
  158. public static void Fatal(string Message,
  159. [CallerMemberName] string CallingMember = "",
  160. [CallerLineNumber] int CallingLineNumber = 0)
  161. {
  162. if (EnableFatal)
  163. {
  164. LogMessage(new LogEntry
  165. {
  166. CallingLineNumber = CallingLineNumber,
  167. CallingMember = CallingMember,
  168. LogLevel = LogLevel.Fatal,
  169. Message = Message,
  170. ExecutionTime = GetExecutionTime()
  171. });
  172. }
  173. }
  174. public static void Ipc(byte[] Data, long CmdPtr, bool Domain)
  175. {
  176. if (EnableIpc)
  177. {
  178. Console.ForegroundColor = ConsoleColor.Cyan;
  179. Console.WriteLine(IpcLog.Message(Data, CmdPtr, Domain));
  180. Console.ResetColor();
  181. }
  182. }
  183. //https://www.codeproject.com/Articles/36747/Quick-and-Dirty-HexDump-of-a-Byte-Array
  184. public static string HexDump(byte[] bytes, int bytesPerLine = 16)
  185. {
  186. if (bytes == null) return "<null>";
  187. int bytesLength = bytes.Length;
  188. char[] HexChars = "0123456789ABCDEF".ToCharArray();
  189. int firstHexColumn =
  190. 8 // 8 characters for the address
  191. + 3; // 3 spaces
  192. int firstCharColumn = firstHexColumn
  193. + bytesPerLine * 3 // - 2 digit for the hexadecimal value and 1 space
  194. + (bytesPerLine - 1) / 8 // - 1 extra space every 8 characters from the 9th
  195. + 2; // 2 spaces
  196. int lineLength = firstCharColumn
  197. + bytesPerLine // - characters to show the ascii value
  198. + Environment.NewLine.Length; // Carriage return and line feed (should normally be 2)
  199. char[] line = (new String(' ', lineLength - Environment.NewLine.Length) + Environment.NewLine).ToCharArray();
  200. int expectedLines = (bytesLength + bytesPerLine - 1) / bytesPerLine;
  201. StringBuilder result = new StringBuilder(expectedLines * lineLength);
  202. for (int i = 0; i < bytesLength; i += bytesPerLine)
  203. {
  204. line[0] = HexChars[(i >> 28) & 0xF];
  205. line[1] = HexChars[(i >> 24) & 0xF];
  206. line[2] = HexChars[(i >> 20) & 0xF];
  207. line[3] = HexChars[(i >> 16) & 0xF];
  208. line[4] = HexChars[(i >> 12) & 0xF];
  209. line[5] = HexChars[(i >> 8) & 0xF];
  210. line[6] = HexChars[(i >> 4) & 0xF];
  211. line[7] = HexChars[(i >> 0) & 0xF];
  212. int hexColumn = firstHexColumn;
  213. int charColumn = firstCharColumn;
  214. for (int j = 0; j < bytesPerLine; j++)
  215. {
  216. if (j > 0 && (j & 7) == 0) hexColumn++;
  217. if (i + j >= bytesLength)
  218. {
  219. line[hexColumn] = ' ';
  220. line[hexColumn + 1] = ' ';
  221. line[charColumn] = ' ';
  222. }
  223. else
  224. {
  225. byte b = bytes[i + j];
  226. line[hexColumn] = HexChars[(b >> 4) & 0xF];
  227. line[hexColumn + 1] = HexChars[b & 0xF];
  228. line[charColumn] = (b < 32 ? '·' : (char)b);
  229. }
  230. hexColumn += 3;
  231. charColumn++;
  232. }
  233. result.Append(line);
  234. }
  235. return result.ToString();
  236. }
  237. private struct LogEntry
  238. {
  239. public string CallingMember;
  240. public string ExecutionTime;
  241. public string Message;
  242. public int CallingLineNumber;
  243. public LogLevel LogLevel;
  244. }
  245. }
  246. }