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/debug/activity_tracker.h" | 10 #include "base/debug/activity_tracker.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> |
55 | 56 |
56 #include "base/base_switches.h" | 57 #include "base/base_switches.h" |
57 #include "base/command_line.h" | 58 #include "base/command_line.h" |
58 #include "base/debug/alias.h" | 59 #include "base/debug/alias.h" |
59 #include "base/debug/debugger.h" | 60 #include "base/debug/debugger.h" |
60 #include "base/debug/stack_trace.h" | 61 #include "base/debug/stack_trace.h" |
| 62 #include "base/lazy_instance.h" |
61 #include "base/posix/eintr_wrapper.h" | 63 #include "base/posix/eintr_wrapper.h" |
62 #include "base/strings/string_piece.h" | 64 #include "base/strings/string_piece.h" |
63 #include "base/strings/string_util.h" | 65 #include "base/strings/string_util.h" |
64 #include "base/strings/stringprintf.h" | 66 #include "base/strings/stringprintf.h" |
65 #include "base/strings/sys_string_conversions.h" | 67 #include "base/strings/sys_string_conversions.h" |
66 #include "base/strings/utf_string_conversions.h" | 68 #include "base/strings/utf_string_conversions.h" |
67 #include "base/synchronization/lock_impl.h" | 69 #include "base/synchronization/lock_impl.h" |
| 70 #include "base/synchronization/read_write_lock.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) |
71 #include "base/posix/safe_strerror.h" | 74 #include "base/posix/safe_strerror.h" |
72 #endif | 75 #endif |
73 | 76 |
74 #if defined(OS_ANDROID) | 77 #if defined(OS_ANDROID) |
75 #include <android/log.h> | 78 #include <android/log.h> |
76 #endif | 79 #endif |
77 | 80 |
(...skipping 38 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 listeners that get notified of every log message we process. |
127 LogMessageHandlerFunction log_message_handler = nullptr; | 130 base::LazyInstance<std::deque<LogMessageListener*>>::Leaky |
| 131 log_message_listeners = LAZY_INSTANCE_INITIALIZER; |
| 132 base::LazyInstance<base::subtle::ReadWriteLock>::Leaky |
| 133 log_message_listener_lock = LAZY_INSTANCE_INITIALIZER; |
128 | 134 |
129 // Helper functions to wrap platform differences. | 135 // Helper functions to wrap platform differences. |
130 | 136 |
131 int32_t CurrentProcessId() { | 137 int32_t CurrentProcessId() { |
132 #if defined(OS_WIN) | 138 #if defined(OS_WIN) |
133 return GetCurrentProcessId(); | 139 return GetCurrentProcessId(); |
134 #elif defined(OS_POSIX) | 140 #elif defined(OS_POSIX) |
135 return getpid(); | 141 return getpid(); |
136 #endif | 142 #endif |
137 } | 143 } |
(...skipping 197 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
335 void CloseLogFileUnlocked() { | 341 void CloseLogFileUnlocked() { |
336 if (!g_log_file) | 342 if (!g_log_file) |
337 return; | 343 return; |
338 | 344 |
339 CloseFile(g_log_file); | 345 CloseFile(g_log_file); |
340 g_log_file = nullptr; | 346 g_log_file = nullptr; |
341 } | 347 } |
342 | 348 |
343 } // namespace | 349 } // namespace |
344 | 350 |
| 351 LogMessageListener::LogMessageListener() { |
| 352 base::subtle::AutoWriteLock lock(log_message_listener_lock.Get()); |
| 353 log_message_listeners.Get().push_front(this); |
| 354 } |
| 355 |
| 356 LogMessageListener::~LogMessageListener() { |
| 357 base::subtle::AutoWriteLock lock(log_message_listener_lock.Get()); |
| 358 auto& listeners = log_message_listeners.Get(); |
| 359 size_t count = listeners.size(); |
| 360 listeners.erase(std::remove(listeners.begin(), listeners.end(), this), |
| 361 listeners.end()); |
| 362 DCHECK_EQ(count - 1, listeners.size()); |
| 363 } |
| 364 |
| 365 size_t LogMessageListenerCountForTesting() { |
| 366 base::subtle::AutoReadLock lock(log_message_listener_lock.Get()); |
| 367 return log_message_listeners.Get().size(); |
| 368 } |
| 369 |
345 // This is never instantiated, it's just used for EAT_STREAM_PARAMETERS to have | 370 // This is never instantiated, it's just used for EAT_STREAM_PARAMETERS to have |
346 // an object of the correct type on the LHS of the unused part of the ternary | 371 // an object of the correct type on the LHS of the unused part of the ternary |
347 // operator. | 372 // operator. |
348 std::ostream* g_swallow_stream; | 373 std::ostream* g_swallow_stream; |
349 | 374 |
350 LoggingSettings::LoggingSettings() | 375 LoggingSettings::LoggingSettings() |
351 : logging_dest(LOG_DEFAULT), | 376 : logging_dest(LOG_DEFAULT), |
352 log_file(nullptr), | 377 log_file(nullptr), |
353 lock_log(LOCK_LOG_FILE), | 378 lock_log(LOCK_LOG_FILE), |
354 delete_old(APPEND_TO_OLD_LOG_FILE) {} | 379 delete_old(APPEND_TO_OLD_LOG_FILE) {} |
(...skipping 52 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
407 return g_min_log_level; | 432 return g_min_log_level; |
408 } | 433 } |
409 | 434 |
410 bool ShouldCreateLogMessage(int severity) { | 435 bool ShouldCreateLogMessage(int severity) { |
411 if (severity < g_min_log_level) | 436 if (severity < g_min_log_level) |
412 return false; | 437 return false; |
413 | 438 |
414 // Return true here unless we know ~LogMessage won't do anything. Note that | 439 // Return true here unless we know ~LogMessage won't do anything. Note that |
415 // ~LogMessage writes to stderr if severity_ >= kAlwaysPrintErrorLevel, even | 440 // ~LogMessage writes to stderr if severity_ >= kAlwaysPrintErrorLevel, even |
416 // when g_logging_destination is LOG_NONE. | 441 // when g_logging_destination is LOG_NONE. |
417 return g_logging_destination != LOG_NONE || log_message_handler || | 442 bool listeners_empty; |
| 443 { |
| 444 base::subtle::AutoReadLock lock(log_message_listener_lock.Get()); |
| 445 listeners_empty = log_message_listeners.Get().empty(); |
| 446 } |
| 447 return g_logging_destination != LOG_NONE || !listeners_empty || |
418 severity >= kAlwaysPrintErrorLevel; | 448 severity >= kAlwaysPrintErrorLevel; |
419 } | 449 } |
420 | 450 |
421 int GetVlogVerbosity() { | 451 int GetVlogVerbosity() { |
422 return std::max(-1, LOG_INFO - GetMinLogLevel()); | 452 return std::max(-1, LOG_INFO - GetMinLogLevel()); |
423 } | 453 } |
424 | 454 |
425 int GetVlogLevelHelper(const char* file, size_t N) { | 455 int GetVlogLevelHelper(const char* file, size_t N) { |
426 DCHECK_GT(N, 0U); | 456 DCHECK_GT(N, 0U); |
427 // Note: |g_vlog_info| may change on a different thread during startup | 457 // Note: |g_vlog_info| may change on a different thread during startup |
(...skipping 13 matching lines...) Expand all Loading... |
441 } | 471 } |
442 | 472 |
443 void SetShowErrorDialogs(bool enable_dialogs) { | 473 void SetShowErrorDialogs(bool enable_dialogs) { |
444 show_error_dialogs = enable_dialogs; | 474 show_error_dialogs = enable_dialogs; |
445 } | 475 } |
446 | 476 |
447 void SetLogAssertHandler(LogAssertHandlerFunction handler) { | 477 void SetLogAssertHandler(LogAssertHandlerFunction handler) { |
448 log_assert_handler = handler; | 478 log_assert_handler = handler; |
449 } | 479 } |
450 | 480 |
451 void SetLogMessageHandler(LogMessageHandlerFunction handler) { | |
452 log_message_handler = handler; | |
453 } | |
454 | |
455 LogMessageHandlerFunction GetLogMessageHandler() { | |
456 return log_message_handler; | |
457 } | |
458 | |
459 // Explicit instantiations for commonly used comparisons. | 481 // Explicit instantiations for commonly used comparisons. |
460 template std::string* MakeCheckOpString<int, int>( | 482 template std::string* MakeCheckOpString<int, int>( |
461 const int&, const int&, const char* names); | 483 const int&, const int&, const char* names); |
462 template std::string* MakeCheckOpString<unsigned long, unsigned long>( | 484 template std::string* MakeCheckOpString<unsigned long, unsigned long>( |
463 const unsigned long&, const unsigned long&, const char* names); | 485 const unsigned long&, const unsigned long&, const char* names); |
464 template std::string* MakeCheckOpString<unsigned long, unsigned int>( | 486 template std::string* MakeCheckOpString<unsigned long, unsigned int>( |
465 const unsigned long&, const unsigned int&, const char* names); | 487 const unsigned long&, const unsigned int&, const char* names); |
466 template std::string* MakeCheckOpString<unsigned int, unsigned long>( | 488 template std::string* MakeCheckOpString<unsigned int, unsigned long>( |
467 const unsigned int&, const unsigned long&, const char* names); | 489 const unsigned int&, const unsigned long&, const char* names); |
468 template std::string* MakeCheckOpString<std::string, std::string>( | 490 template std::string* MakeCheckOpString<std::string, std::string>( |
(...skipping 66 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
535 if (severity_ == LOG_FATAL && !base::debug::BeingDebugged()) { | 557 if (severity_ == LOG_FATAL && !base::debug::BeingDebugged()) { |
536 // Include a stack trace on a fatal, unless a debugger is attached. | 558 // Include a stack trace on a fatal, unless a debugger is attached. |
537 base::debug::StackTrace trace; | 559 base::debug::StackTrace trace; |
538 stream_ << std::endl; // Newline to separate from log message. | 560 stream_ << std::endl; // Newline to separate from log message. |
539 trace.OutputToStream(&stream_); | 561 trace.OutputToStream(&stream_); |
540 } | 562 } |
541 #endif | 563 #endif |
542 stream_ << std::endl; | 564 stream_ << std::endl; |
543 std::string str_newline(stream_.str()); | 565 std::string str_newline(stream_.str()); |
544 | 566 |
545 // Give any log message handler first dibs on the message. | 567 { |
546 if (log_message_handler && | 568 base::subtle::AutoReadLock lock(log_message_listener_lock.Get()); |
547 log_message_handler(severity_, file_, line_, | 569 // Broadcast to log message listeners first. |
548 message_start_, str_newline)) { | 570 for (auto* listener : log_message_listeners.Get()) { |
549 // The handler took care of it, no further processing. | 571 listener->OnMessage(severity_, file_, line_, message_start_, str_newline); |
550 return; | 572 } |
551 } | 573 } |
552 | 574 |
553 if ((g_logging_destination & LOG_TO_SYSTEM_DEBUG_LOG) != 0) { | 575 if ((g_logging_destination & LOG_TO_SYSTEM_DEBUG_LOG) != 0) { |
554 #if defined(OS_WIN) | 576 #if defined(OS_WIN) |
555 OutputDebugStringA(str_newline.c_str()); | 577 OutputDebugStringA(str_newline.c_str()); |
556 #elif defined(OS_MACOSX) | 578 #elif defined(OS_MACOSX) |
557 // In LOG_TO_SYSTEM_DEBUG_LOG mode, log messages are always written to | 579 // In LOG_TO_SYSTEM_DEBUG_LOG mode, log messages are always written to |
558 // stderr. If stderr is /dev/null, also log via ASL (Apple System Log). If | 580 // stderr. If stderr is /dev/null, also log via ASL (Apple System Log). If |
559 // there's something weird about stderr, assume that log messages are going | 581 // there's something weird about stderr, assume that log messages are going |
560 // nowhere and log via ASL too. Messages logged via ASL show up in | 582 // nowhere and log via ASL too. Messages logged via ASL show up in |
(...skipping 386 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
947 BASE_EXPORT void LogErrorNotReached(const char* file, int line) { | 969 BASE_EXPORT void LogErrorNotReached(const char* file, int line) { |
948 LogMessage(file, line, LOG_ERROR).stream() | 970 LogMessage(file, line, LOG_ERROR).stream() |
949 << "NOTREACHED() hit."; | 971 << "NOTREACHED() hit."; |
950 } | 972 } |
951 | 973 |
952 } // namespace logging | 974 } // namespace logging |
953 | 975 |
954 std::ostream& std::operator<<(std::ostream& out, const wchar_t* wstr) { | 976 std::ostream& std::operator<<(std::ostream& out, const wchar_t* wstr) { |
955 return out << (wstr ? base::WideToUTF8(wstr) : std::string()); | 977 return out << (wstr ? base::WideToUTF8(wstr) : std::string()); |
956 } | 978 } |
OLD | NEW |