| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <array>
 | 
					
						
							|  |  |  | #include <cstdio>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | #   define WIN32_LEAN_AND_MEAN
 | 
					
						
							|  |  |  | #   include <Windows.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | #include "common/logging/backend.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "common/logging/text_formatter.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-02 12:55:31 -04:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_funcs.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-04 03:03:19 -02:00
										 |  |  | #include "common/string_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | namespace Log { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-04 03:03:19 -02:00
										 |  |  | // TODO(bunnei): This should be moved to a generic path manipulation library
 | 
					
						
							|  |  |  | const char* TrimSourcePath(const char* path, const char* root) { | 
					
						
							|  |  |  |     const char* p = path; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (*p != '\0') { | 
					
						
							|  |  |  |         const char* next_slash = p; | 
					
						
							|  |  |  |         while (*next_slash != '\0' && *next_slash != '/' && *next_slash != '\\') { | 
					
						
							|  |  |  |             ++next_slash; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         bool is_src = Common::ComparePartialString(p, next_slash, root); | 
					
						
							|  |  |  |         p = next_slash; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (*p != '\0') { | 
					
						
							|  |  |  |             ++p; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (is_src) { | 
					
						
							|  |  |  |             path = p; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return path; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | void FormatLogMessage(const Entry& entry, char* out_text, size_t text_len) { | 
					
						
							|  |  |  |     unsigned int time_seconds    = static_cast<unsigned int>(entry.timestamp.count() / 1000000); | 
					
						
							|  |  |  |     unsigned int time_fractional = static_cast<unsigned int>(entry.timestamp.count() % 1000000); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 02:19:44 -03:00
										 |  |  |     const char* class_name = GetLogClassName(entry.log_class); | 
					
						
							|  |  |  |     const char* level_name = GetLevelName(entry.log_level); | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     snprintf(out_text, text_len, "[%4u.%06u] %s <%s> %s: %s", | 
					
						
							|  |  |  |         time_seconds, time_fractional, class_name, level_name, | 
					
						
							| 
									
										
										
										
											2014-11-04 03:03:19 -02:00
										 |  |  |         TrimSourcePath(entry.location.c_str()), entry.message.c_str()); | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 02:09:56 -02:00
										 |  |  | void PrintMessage(const Entry& entry) { | 
					
						
							|  |  |  |     std::array<char, 4 * 1024> format_buffer; | 
					
						
							|  |  |  |     FormatLogMessage(entry, format_buffer.data(), format_buffer.size()); | 
					
						
							|  |  |  |     fputs(format_buffer.data(), stderr); | 
					
						
							|  |  |  |     fputc('\n', stderr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void PrintColoredMessage(const Entry& entry) { | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |     static HANDLE console_handle = GetStdHandle(STD_ERROR_HANDLE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 02:09:56 -02:00
										 |  |  |     CONSOLE_SCREEN_BUFFER_INFO original_info = {0}; | 
					
						
							|  |  |  |     GetConsoleScreenBufferInfo(console_handle, &original_info); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  |     WORD color = 0; | 
					
						
							| 
									
										
										
										
											2014-12-14 02:09:56 -02:00
										 |  |  |     switch (entry.log_level) { | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  |     case Level::Trace: // Grey
 | 
					
						
							|  |  |  |         color = FOREGROUND_INTENSITY; break; | 
					
						
							|  |  |  |     case Level::Debug: // Cyan
 | 
					
						
							|  |  |  |         color = FOREGROUND_GREEN | FOREGROUND_BLUE; break; | 
					
						
							|  |  |  |     case Level::Info: // Bright gray
 | 
					
						
							|  |  |  |         color = FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_BLUE; break; | 
					
						
							|  |  |  |     case Level::Warning: // Bright yellow
 | 
					
						
							|  |  |  |         color = FOREGROUND_RED | FOREGROUND_GREEN | FOREGROUND_INTENSITY; break; | 
					
						
							|  |  |  |     case Level::Error: // Bright red
 | 
					
						
							|  |  |  |         color = FOREGROUND_RED | FOREGROUND_INTENSITY; break; | 
					
						
							|  |  |  |     case Level::Critical: // Bright magenta
 | 
					
						
							|  |  |  |         color = FOREGROUND_RED | FOREGROUND_BLUE | FOREGROUND_INTENSITY; break; | 
					
						
							| 
									
										
										
										
											2015-08-02 12:55:31 -04:00
										 |  |  |     case Level::Count: | 
					
						
							|  |  |  |         ASSERT_MSG(false, "invalid log level"); break; | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SetConsoleTextAttribute(console_handle, color); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-12-14 02:09:56 -02:00
										 |  |  | #   define ESC "\x1b"
 | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  |     const char* color = ""; | 
					
						
							| 
									
										
										
										
											2014-12-14 02:09:56 -02:00
										 |  |  |     switch (entry.log_level) { | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  |     case Level::Trace: // Grey
 | 
					
						
							|  |  |  |         color = ESC "[1;30m"; break; | 
					
						
							|  |  |  |     case Level::Debug: // Cyan
 | 
					
						
							|  |  |  |         color = ESC "[0;36m"; break; | 
					
						
							|  |  |  |     case Level::Info: // Bright gray
 | 
					
						
							|  |  |  |         color = ESC "[0;37m"; break; | 
					
						
							|  |  |  |     case Level::Warning: // Bright yellow
 | 
					
						
							|  |  |  |         color = ESC "[1;33m"; break; | 
					
						
							|  |  |  |     case Level::Error: // Bright red
 | 
					
						
							|  |  |  |         color = ESC "[1;31m"; break; | 
					
						
							|  |  |  |     case Level::Critical: // Bright magenta
 | 
					
						
							|  |  |  |         color = ESC "[1;35m"; break; | 
					
						
							| 
									
										
										
										
											2015-08-02 12:55:31 -04:00
										 |  |  |     case Level::Count: | 
					
						
							|  |  |  |         ASSERT_MSG(false, "invalid log level"); break; | 
					
						
							| 
									
										
										
										
											2014-11-04 01:42:34 -02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fputs(color, stderr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-14 02:09:56 -02:00
										 |  |  |     PrintMessage(entry); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |     SetConsoleTextAttribute(console_handle, original_info.wAttributes); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     fputs(ESC "[0m", stderr); | 
					
						
							|  |  |  | #   undef ESC
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |