OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "base/logging.h" | 5 #include "base/logging.h" |
6 | 6 |
7 #include <limits.h> | 7 #include <limits.h> |
8 #include <stdint.h> | 8 #include <stdint.h> |
9 | 9 |
10 #include "base/macros.h" | 10 #include "base/macros.h" |
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
42 #include <sys/stat.h> | 42 #include <sys/stat.h> |
43 #include <unistd.h> | 43 #include <unistd.h> |
44 #define MAX_PATH PATH_MAX | 44 #define MAX_PATH PATH_MAX |
45 typedef FILE* FileHandle; | 45 typedef FILE* FileHandle; |
46 typedef pthread_mutex_t* MutexHandle; | 46 typedef pthread_mutex_t* MutexHandle; |
47 #endif | 47 #endif |
48 | 48 |
49 #include <algorithm> | 49 #include <algorithm> |
50 #include <cstring> | 50 #include <cstring> |
51 #include <ctime> | 51 #include <ctime> |
52 #include <deque> | |
52 #include <iomanip> | 53 #include <iomanip> |
53 #include <ostream> | 54 #include <ostream> |
54 #include <string> | 55 #include <string> |
56 #include <unordered_set> | |
55 | 57 |
56 #include "base/base_switches.h" | 58 #include "base/base_switches.h" |
57 #include "base/command_line.h" | 59 #include "base/command_line.h" |
58 #include "base/debug/alias.h" | 60 #include "base/debug/alias.h" |
59 #include "base/debug/debugger.h" | 61 #include "base/debug/debugger.h" |
60 #include "base/debug/stack_trace.h" | 62 #include "base/debug/stack_trace.h" |
63 #include "base/lazy_instance.h" | |
61 #include "base/posix/eintr_wrapper.h" | 64 #include "base/posix/eintr_wrapper.h" |
62 #include "base/strings/string_piece.h" | 65 #include "base/strings/string_piece.h" |
63 #include "base/strings/string_util.h" | 66 #include "base/strings/string_util.h" |
64 #include "base/strings/stringprintf.h" | 67 #include "base/strings/stringprintf.h" |
65 #include "base/strings/sys_string_conversions.h" | 68 #include "base/strings/sys_string_conversions.h" |
66 #include "base/strings/utf_string_conversions.h" | 69 #include "base/strings/utf_string_conversions.h" |
67 #include "base/synchronization/lock_impl.h" | 70 #include "base/synchronization/lock_impl.h" |
68 #include "base/threading/platform_thread.h" | 71 #include "base/threading/platform_thread.h" |
69 #include "base/vlog.h" | 72 #include "base/vlog.h" |
70 #if defined(OS_POSIX) | 73 #if defined(OS_POSIX) |
(...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
116 bool g_log_thread_id = false; | 119 bool g_log_thread_id = false; |
117 bool g_log_timestamp = true; | 120 bool g_log_timestamp = true; |
118 bool g_log_tickcount = false; | 121 bool g_log_tickcount = false; |
119 | 122 |
120 // Should we pop up fatal debug messages in a dialog? | 123 // Should we pop up fatal debug messages in a dialog? |
121 bool show_error_dialogs = false; | 124 bool show_error_dialogs = false; |
122 | 125 |
123 // An assert handler override specified by the client to be called instead of | 126 // An assert handler override specified by the client to be called instead of |
124 // the debug message dialog and process termination. | 127 // the debug message dialog and process termination. |
125 LogAssertHandlerFunction log_assert_handler = nullptr; | 128 LogAssertHandlerFunction log_assert_handler = nullptr; |
126 // A log message handler that gets notified of every log message we process. | 129 // Log message handlers that get notified of every log message we process. |
127 LogMessageHandlerFunction log_message_handler = nullptr; | 130 base::LazyInstance<std::deque<LogMessageHandler*>>::Leaky |
131 log_message_handlers = LAZY_INSTANCE_INITIALIZER; | |
132 // Log message listeners that get notified of every log message we process | |
133 // before log message handlers. | |
134 base::LazyInstance<std::unordered_set<LogMessageListener*>>::Leaky | |
grt (UTC plus 2)
2016/07/27 20:38:20
why an unordered set? are there really so many lis
wychen
2016/08/01 16:12:25
The number of listeners should be very low practic
grt (UTC plus 2)
2016/08/01 21:17:44
Ouch. Apologies for the complexity goof. I haven't
wychen
2016/08/01 23:11:03
Unordered set does use a hash table internally. Do
grt (UTC plus 2)
2016/08/02 07:25:46
I'm a bit divided. On the one hand, I prefer callb
wychen
2016/08/04 03:53:55
I meant keeping the single-method interface as is,
| |
135 log_message_listeners = LAZY_INSTANCE_INITIALIZER; | |
128 | 136 |
129 // Helper functions to wrap platform differences. | 137 // Helper functions to wrap platform differences. |
130 | 138 |
131 int32_t CurrentProcessId() { | 139 int32_t CurrentProcessId() { |
132 #if defined(OS_WIN) | 140 #if defined(OS_WIN) |
133 return GetCurrentProcessId(); | 141 return GetCurrentProcessId(); |
134 #elif defined(OS_POSIX) | 142 #elif defined(OS_POSIX) |
135 return getpid(); | 143 return getpid(); |
136 #endif | 144 #endif |
137 } | 145 } |
(...skipping 197 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
335 void CloseLogFileUnlocked() { | 343 void CloseLogFileUnlocked() { |
336 if (!g_log_file) | 344 if (!g_log_file) |
337 return; | 345 return; |
338 | 346 |
339 CloseFile(g_log_file); | 347 CloseFile(g_log_file); |
340 g_log_file = nullptr; | 348 g_log_file = nullptr; |
341 } | 349 } |
342 | 350 |
343 } // namespace | 351 } // namespace |
344 | 352 |
353 LogMessageHandler::LogMessageHandler() { | |
354 log_message_handlers.Get().push_front(this); | |
355 } | |
356 | |
357 LogMessageHandler::~LogMessageHandler() { | |
358 auto& handlers = log_message_handlers.Get(); | |
359 unsigned count = handlers.size(); | |
grt (UTC plus 2)
2016/07/27 20:38:20
size_t
wychen
2016/08/01 16:12:25
Done.
| |
360 handlers.erase(std::remove(handlers.begin(), handlers.end(), this), | |
361 handlers.end()); | |
362 DCHECK_EQ(count - 1, handlers.size()); | |
363 } | |
364 | |
365 unsigned LogMessageHandlerCountForTesting() { | |
366 return log_message_handlers.Get().size(); | |
367 } | |
368 | |
369 LogMessageListener::LogMessageListener() { | |
370 log_message_listeners.Get().insert(this); | |
371 } | |
372 | |
373 LogMessageListener::~LogMessageListener() { | |
374 DCHECK_EQ(1u, log_message_listeners.Get().erase(this)); | |
375 } | |
376 | |
377 unsigned LogMessageListenerCountForTesting() { | |
378 return log_message_listeners.Get().size(); | |
379 } | |
380 | |
345 LoggingSettings::LoggingSettings() | 381 LoggingSettings::LoggingSettings() |
346 : logging_dest(LOG_DEFAULT), | 382 : logging_dest(LOG_DEFAULT), |
347 log_file(nullptr), | 383 log_file(nullptr), |
348 lock_log(LOCK_LOG_FILE), | 384 lock_log(LOCK_LOG_FILE), |
349 delete_old(APPEND_TO_OLD_LOG_FILE) {} | 385 delete_old(APPEND_TO_OLD_LOG_FILE) {} |
350 | 386 |
351 bool BaseInitLoggingImpl(const LoggingSettings& settings) { | 387 bool BaseInitLoggingImpl(const LoggingSettings& settings) { |
352 #if defined(OS_NACL) | 388 #if defined(OS_NACL) |
353 // Can log only to the system debug log. | 389 // Can log only to the system debug log. |
354 CHECK_EQ(settings.logging_dest & ~LOG_TO_SYSTEM_DEBUG_LOG, 0); | 390 CHECK_EQ(settings.logging_dest & ~LOG_TO_SYSTEM_DEBUG_LOG, 0); |
(...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
402 return g_min_log_level; | 438 return g_min_log_level; |
403 } | 439 } |
404 | 440 |
405 bool ShouldCreateLogMessage(int severity) { | 441 bool ShouldCreateLogMessage(int severity) { |
406 if (severity < g_min_log_level) | 442 if (severity < g_min_log_level) |
407 return false; | 443 return false; |
408 | 444 |
409 // Return true here unless we know ~LogMessage won't do anything. Note that | 445 // Return true here unless we know ~LogMessage won't do anything. Note that |
410 // ~LogMessage writes to stderr if severity_ >= kAlwaysPrintErrorLevel, even | 446 // ~LogMessage writes to stderr if severity_ >= kAlwaysPrintErrorLevel, even |
411 // when g_logging_destination is LOG_NONE. | 447 // when g_logging_destination is LOG_NONE. |
412 return g_logging_destination != LOG_NONE || log_message_handler || | 448 return g_logging_destination != LOG_NONE || |
449 !log_message_handlers.Get().empty() || | |
450 !log_message_listeners.Get().empty() || | |
413 severity >= kAlwaysPrintErrorLevel; | 451 severity >= kAlwaysPrintErrorLevel; |
414 } | 452 } |
415 | 453 |
416 int GetVlogVerbosity() { | 454 int GetVlogVerbosity() { |
417 return std::max(-1, LOG_INFO - GetMinLogLevel()); | 455 return std::max(-1, LOG_INFO - GetMinLogLevel()); |
418 } | 456 } |
419 | 457 |
420 int GetVlogLevelHelper(const char* file, size_t N) { | 458 int GetVlogLevelHelper(const char* file, size_t N) { |
421 DCHECK_GT(N, 0U); | 459 DCHECK_GT(N, 0U); |
422 // Note: |g_vlog_info| may change on a different thread during startup | 460 // Note: |g_vlog_info| may change on a different thread during startup |
(...skipping 13 matching lines...) Expand all Loading... | |
436 } | 474 } |
437 | 475 |
438 void SetShowErrorDialogs(bool enable_dialogs) { | 476 void SetShowErrorDialogs(bool enable_dialogs) { |
439 show_error_dialogs = enable_dialogs; | 477 show_error_dialogs = enable_dialogs; |
440 } | 478 } |
441 | 479 |
442 void SetLogAssertHandler(LogAssertHandlerFunction handler) { | 480 void SetLogAssertHandler(LogAssertHandlerFunction handler) { |
443 log_assert_handler = handler; | 481 log_assert_handler = handler; |
444 } | 482 } |
445 | 483 |
446 void SetLogMessageHandler(LogMessageHandlerFunction handler) { | |
447 log_message_handler = handler; | |
448 } | |
449 | |
450 LogMessageHandlerFunction GetLogMessageHandler() { | |
451 return log_message_handler; | |
452 } | |
453 | |
454 // Explicit instantiations for commonly used comparisons. | 484 // Explicit instantiations for commonly used comparisons. |
455 template std::string* MakeCheckOpString<int, int>( | 485 template std::string* MakeCheckOpString<int, int>( |
456 const int&, const int&, const char* names); | 486 const int&, const int&, const char* names); |
457 template std::string* MakeCheckOpString<unsigned long, unsigned long>( | 487 template std::string* MakeCheckOpString<unsigned long, unsigned long>( |
458 const unsigned long&, const unsigned long&, const char* names); | 488 const unsigned long&, const unsigned long&, const char* names); |
459 template std::string* MakeCheckOpString<unsigned long, unsigned int>( | 489 template std::string* MakeCheckOpString<unsigned long, unsigned int>( |
460 const unsigned long&, const unsigned int&, const char* names); | 490 const unsigned long&, const unsigned int&, const char* names); |
461 template std::string* MakeCheckOpString<unsigned int, unsigned long>( | 491 template std::string* MakeCheckOpString<unsigned int, unsigned long>( |
462 const unsigned int&, const unsigned long&, const char* names); | 492 const unsigned int&, const unsigned long&, const char* names); |
463 template std::string* MakeCheckOpString<std::string, std::string>( | 493 template std::string* MakeCheckOpString<std::string, std::string>( |
(...skipping 66 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
530 if (severity_ == LOG_FATAL && !base::debug::BeingDebugged()) { | 560 if (severity_ == LOG_FATAL && !base::debug::BeingDebugged()) { |
531 // Include a stack trace on a fatal, unless a debugger is attached. | 561 // Include a stack trace on a fatal, unless a debugger is attached. |
532 base::debug::StackTrace trace; | 562 base::debug::StackTrace trace; |
533 stream_ << std::endl; // Newline to separate from log message. | 563 stream_ << std::endl; // Newline to separate from log message. |
534 trace.OutputToStream(&stream_); | 564 trace.OutputToStream(&stream_); |
535 } | 565 } |
536 #endif | 566 #endif |
537 stream_ << std::endl; | 567 stream_ << std::endl; |
538 std::string str_newline(stream_.str()); | 568 std::string str_newline(stream_.str()); |
539 | 569 |
540 // Give any log message handler first dibs on the message. | 570 // Broadcast to log message listeners first. |
541 if (log_message_handler && | 571 for (auto* listeners : log_message_listeners.Get()) { |
grt (UTC plus 2)
2016/07/27 20:38:20
listeners -> listener
grt (UTC plus 2)
2016/07/28 06:28:47
since logging may happen on any thread at any time
wychen
2016/08/01 16:12:25
I've added Locks.
wychen
2016/08/01 16:12:25
Done.
| |
542 log_message_handler(severity_, file_, line_, | 572 listeners->OnMessage(severity_, file_, line_, message_start_, str_newline); |
543 message_start_, str_newline)) { | 573 } |
544 // The handler took care of it, no further processing. | 574 |
545 return; | 575 // Give log message handlers first dibs on the message. |
576 for (auto* handler : log_message_handlers.Get()) { | |
577 if (handler->OnMessage(severity_, file_, line_, message_start_, | |
578 str_newline)) { | |
579 // The handler took care of it, no further processing. | |
580 return; | |
581 } | |
546 } | 582 } |
547 | 583 |
548 if ((g_logging_destination & LOG_TO_SYSTEM_DEBUG_LOG) != 0) { | 584 if ((g_logging_destination & LOG_TO_SYSTEM_DEBUG_LOG) != 0) { |
549 #if defined(OS_WIN) | 585 #if defined(OS_WIN) |
550 OutputDebugStringA(str_newline.c_str()); | 586 OutputDebugStringA(str_newline.c_str()); |
551 #elif defined(OS_MACOSX) | 587 #elif defined(OS_MACOSX) |
552 // In LOG_TO_SYSTEM_DEBUG_LOG mode, log messages are always written to | 588 // In LOG_TO_SYSTEM_DEBUG_LOG mode, log messages are always written to |
553 // stderr. If stderr is /dev/null, also log via ASL (Apple System Log). If | 589 // stderr. If stderr is /dev/null, also log via ASL (Apple System Log). If |
554 // there's something weird about stderr, assume that log messages are going | 590 // there's something weird about stderr, assume that log messages are going |
555 // nowhere and log via ASL too. Messages logged via ASL show up in | 591 // nowhere and log via ASL too. Messages logged via ASL show up in |
(...skipping 365 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
921 BASE_EXPORT void LogErrorNotReached(const char* file, int line) { | 957 BASE_EXPORT void LogErrorNotReached(const char* file, int line) { |
922 LogMessage(file, line, LOG_ERROR).stream() | 958 LogMessage(file, line, LOG_ERROR).stream() |
923 << "NOTREACHED() hit."; | 959 << "NOTREACHED() hit."; |
924 } | 960 } |
925 | 961 |
926 } // namespace logging | 962 } // namespace logging |
927 | 963 |
928 std::ostream& std::operator<<(std::ostream& out, const wchar_t* wstr) { | 964 std::ostream& std::operator<<(std::ostream& out, const wchar_t* wstr) { |
929 return out << (wstr ? base::WideToUTF8(wstr) : std::string()); | 965 return out << (wstr ? base::WideToUTF8(wstr) : std::string()); |
930 } | 966 } |
OLD | NEW |