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
3 // found in the LICENSE file.
4
5 #include "base/logging.h"
6
7 #include <sys/syscall.h>
8 #include <time.h>
9 #include <errno.h>
10 #include <pthread.h>
11 #include <stdio.h>
12 #include <stdlib.h>
13 #include <string.h>
14 #include <sys/stat.h>
15 #include <unistd.h>
16
17 #include <algorithm>
18 #include <cstring>
19 #include <ctime>
20 #include <iomanip>
21 #include <ostream>
22 #include <string>
23
24 #include "base/posix/eintr_wrapper.h"
25 #include "base/strings/string_piece.h"
26 #include "base/strings/string_util.h"
27 #include "base/strings/stringprintf.h"
28 #include "base/strings/utf_string_conversion_utils.h"
29
30 namespace logging {
31
32 namespace {
33
34 const char* const log_severity_names[LOG_NUM_SEVERITIES] = {
35 "INFO", "WARNING", "ERROR", "FATAL" };
36
log_severity_name(int severity)37 const char* log_severity_name(int severity) {
38 if (severity >= 0 && severity < LOG_NUM_SEVERITIES)
39 return log_severity_names[severity];
40 return "UNKNOWN";
41 }
42
43 int g_min_log_level = 0;
44
45 LoggingDestination g_logging_destination = LOG_DEFAULT;
46
47 // For LOG_ERROR and above, always print to stderr.
48 const int kAlwaysPrintErrorLevel = LOG_ERROR;
49
50 // What should be prepended to each message?
51 bool g_log_timestamp = true;
52
53 // Should we pop up fatal debug messages in a dialog?
54 bool show_error_dialogs = false;
55
56 // An assert handler override specified by the client to be called instead of
57 // the debug message dialog and process termination.
58 LogAssertHandlerFunction log_assert_handler = nullptr;
59 // A log message handler that gets notified of every log message we process.
60 LogMessageHandlerFunction log_message_handler = nullptr;
61
62 // Helper functions to wrap platform differences.
63
64 } // namespace
65
LoggingSettings()66 LoggingSettings::LoggingSettings()
67 : logging_dest(LOG_DEFAULT) {}
68
BaseInitLoggingImpl(const LoggingSettings & settings)69 bool BaseInitLoggingImpl(const LoggingSettings& settings) {
70 g_logging_destination = settings.logging_dest;
71
72 return true;
73 }
74
SetMinLogLevel(int level)75 void SetMinLogLevel(int level) {
76 g_min_log_level = std::min(LOG_FATAL, level);
77 }
78
GetMinLogLevel()79 int GetMinLogLevel() {
80 return g_min_log_level;
81 }
82
ShouldCreateLogMessage(int severity)83 bool ShouldCreateLogMessage(int severity) {
84 if (severity < g_min_log_level)
85 return false;
86
87 // Return true here unless we know ~LogMessage won't do anything. Note that
88 // ~LogMessage writes to stderr if severity_ >= kAlwaysPrintErrorLevel, even
89 // when g_logging_destination is LOG_NONE.
90 return g_logging_destination != LOG_NONE || log_message_handler ||
91 severity >= kAlwaysPrintErrorLevel;
92 }
93
GetVlogVerbosity()94 int GetVlogVerbosity() {
95 return std::max(-1, LOG_INFO - GetMinLogLevel());
96 }
97
SetLogItems(bool enable_process_id,bool enable_thread_id,bool enable_timestamp,bool enable_tickcount)98 void SetLogItems(bool enable_process_id, bool enable_thread_id,
99 bool enable_timestamp, bool enable_tickcount) {
100 g_log_timestamp = enable_timestamp;
101 }
102
SetShowErrorDialogs(bool enable_dialogs)103 void SetShowErrorDialogs(bool enable_dialogs) {
104 show_error_dialogs = enable_dialogs;
105 }
106
SetLogAssertHandler(LogAssertHandlerFunction handler)107 void SetLogAssertHandler(LogAssertHandlerFunction handler) {
108 log_assert_handler = handler;
109 }
110
SetLogMessageHandler(LogMessageHandlerFunction handler)111 void SetLogMessageHandler(LogMessageHandlerFunction handler) {
112 log_message_handler = handler;
113 }
114
GetLogMessageHandler()115 LogMessageHandlerFunction GetLogMessageHandler() {
116 return log_message_handler;
117 }
118
119 // Explicit instantiations for commonly used comparisons.
120 template std::string* MakeCheckOpString<int, int>(
121 const int&, const int&, const char* names);
122 template std::string* MakeCheckOpString<unsigned long, unsigned long>(
123 const unsigned long&, const unsigned long&, const char* names);
124 template std::string* MakeCheckOpString<unsigned long, unsigned int>(
125 const unsigned long&, const unsigned int&, const char* names);
126 template std::string* MakeCheckOpString<unsigned int, unsigned long>(
127 const unsigned int&, const unsigned long&, const char* names);
128 template std::string* MakeCheckOpString<std::string, std::string>(
129 const std::string&, const std::string&, const char* name);
130
LogMessage(const char * file,int line,LogSeverity severity)131 LogMessage::LogMessage(const char* file, int line, LogSeverity severity)
132 : severity_(severity), file_(file), line_(line) {
133 Init(file, line);
134 }
135
LogMessage(const char * file,int line,const char * condition)136 LogMessage::LogMessage(const char* file, int line, const char* condition)
137 : severity_(LOG_FATAL), file_(file), line_(line) {
138 Init(file, line);
139 stream_ << "Check failed: " << condition << ". ";
140 }
141
LogMessage(const char * file,int line,std::string * result)142 LogMessage::LogMessage(const char* file, int line, std::string* result)
143 : severity_(LOG_FATAL), file_(file), line_(line) {
144 Init(file, line);
145 stream_ << "Check failed: " << *result;
146 delete result;
147 }
148
LogMessage(const char * file,int line,LogSeverity severity,std::string * result)149 LogMessage::LogMessage(const char* file, int line, LogSeverity severity,
150 std::string* result)
151 : severity_(severity), file_(file), line_(line) {
152 Init(file, line);
153 stream_ << "Check failed: " << *result;
154 delete result;
155 }
156
~LogMessage()157 LogMessage::~LogMessage() {
158 stream_ << std::endl;
159 std::string str_newline(stream_.str());
160
161 // Give any log message handler first dibs on the message.
162 if (log_message_handler &&
163 log_message_handler(severity_, file_, line_,
164 message_start_, str_newline)) {
165 // The handler took care of it, no further processing.
166 return;
167 }
168
169 if ((g_logging_destination & LOG_TO_SYSTEM_DEBUG_LOG) != 0) {
170 ignore_result(fwrite(str_newline.data(), str_newline.size(), 1, stderr));
171 fflush(stderr);
172 } else if (severity_ >= kAlwaysPrintErrorLevel) {
173 // When we're only outputting to a log file, above a certain log level, we
174 // should still output to stderr so that we can better detect and diagnose
175 // problems with unit tests, especially on the buildbots.
176 ignore_result(fwrite(str_newline.data(), str_newline.size(), 1, stderr));
177 fflush(stderr);
178 }
179
180 if (severity_ == LOG_FATAL) {
181 // Ensure the first characters of the string are on the stack so they
182 // are contained in minidumps for diagnostic purposes.
183 char str_stack[1024];
184 str_newline.copy(str_stack, arraysize(str_stack));
185
186 if (log_assert_handler) {
187 // Make a copy of the string for the handler out of paranoia.
188 log_assert_handler(std::string(stream_.str()));
189 } else {
190 // Crash the process to generate a dump.
191 abort();
192 }
193 }
194 }
195
196 // writes the common header info to the stream
Init(const char * file,int line)197 void LogMessage::Init(const char* file, int line) {
198 base::StringPiece filename(file);
199 size_t last_slash_pos = filename.find_last_of("\\/");
200 if (last_slash_pos != base::StringPiece::npos)
201 filename.remove_prefix(last_slash_pos + 1);
202
203 // TODO(darin): It might be nice if the columns were fixed width.
204
205 stream_ << '[';
206 if (g_log_timestamp) {
207 time_t t = time(nullptr);
208 struct tm local_time;
209 memset(&local_time, 0, sizeof(local_time));
210 #ifdef _MSC_VER
211 localtime_s(&local_time, &t);
212 #else
213 localtime_r(&t, &local_time);
214 #endif
215 struct tm* tm_time = &local_time;
216 stream_ << std::setfill('0')
217 << std::setw(2) << 1 + tm_time->tm_mon
218 << std::setw(2) << tm_time->tm_mday
219 << '/'
220 << std::setw(2) << tm_time->tm_hour
221 << std::setw(2) << tm_time->tm_min
222 << std::setw(2) << tm_time->tm_sec
223 << ':';
224 }
225 if (severity_ >= 0)
226 stream_ << log_severity_name(severity_);
227 else
228 stream_ << "VERBOSE" << -severity_;
229
230 stream_ << ":" << filename << "(" << line << ")] ";
231
232 message_start_ = stream_.str().length();
233 }
234
RawLog(int level,const char * message)235 void RawLog(int level, const char* message) {
236 if (level >= g_min_log_level) {
237 size_t bytes_written = 0;
238 const size_t message_len = strlen(message);
239 int rv;
240 while (bytes_written < message_len) {
241 rv = HANDLE_EINTR(
242 write(STDERR_FILENO, message + bytes_written,
243 message_len - bytes_written));
244 if (rv < 0) {
245 // Give up, nothing we can do now.
246 break;
247 }
248 bytes_written += rv;
249 }
250
251 if (message_len > 0 && message[message_len - 1] != '\n') {
252 do {
253 rv = HANDLE_EINTR(write(STDERR_FILENO, "\n", 1));
254 if (rv < 0) {
255 // Give up, nothing we can do now.
256 break;
257 }
258 } while (rv != 1);
259 }
260 }
261
262 if (level == LOG_FATAL)
263 abort();
264 }
265
266 // This was defined at the beginning of this file.
267 #undef write
268
LogErrorNotReached(const char * file,int line)269 void LogErrorNotReached(const char* file, int line) {
270 LogMessage(file, line, LOG_ERROR).stream()
271 << "NOTREACHED() hit.";
272 }
273
274 } // namespace logging
275