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 "build/build_config.h" 6 7 // Need to include this before most other files because it defines 8 // IPC_MESSAGE_LOG_ENABLED. We need to use it to define 9 // IPC_MESSAGE_MACROS_LOG_ENABLED so render_messages.h will generate the 10 // ViewMsgLog et al. functions. 11 #include "ipc/ipc_message.h" 12 13 // On Windows, the about:ipc dialog shows IPCs; on POSIX, we hook up a 14 // logger in this file. (We implement about:ipc on Mac but implement 15 // the loggers here anyway). We need to do this real early to be sure 16 // IPC_MESSAGE_MACROS_LOG_ENABLED doesn't get undefined. 17 #if defined(OS_POSIX) && defined(IPC_MESSAGE_LOG_ENABLED) 18 #define IPC_MESSAGE_MACROS_LOG_ENABLED 19 #include "content/public/common/content_ipc_logging.h" 20 #define IPC_LOG_TABLE_ADD_ENTRY(msg_id, logger) \ 21 content::RegisterIPCLogger(msg_id, logger) 22 #include "chrome/common/all_messages.h" 23 #endif 24 25 #if defined(OS_WIN) 26 #include <windows.h> 27 #endif 28 29 #include <fstream> 30 31 #include "chrome/common/logging_chrome.h" 32 33 #include "base/command_line.h" 34 #include "base/compiler_specific.h" 35 #include "base/debug/debugger.h" 36 #include "base/debug/dump_without_crashing.h" 37 #include "base/environment.h" 38 #include "base/files/file_path.h" 39 #include "base/files/file_util.h" 40 #include "base/logging.h" 41 #include "base/path_service.h" 42 #include "base/strings/string_number_conversions.h" 43 #include "base/strings/string_util.h" 44 #include "base/strings/stringprintf.h" 45 #include "base/strings/utf_string_conversions.h" 46 #include "base/threading/thread_restrictions.h" 47 #include "base/time/time.h" 48 #include "chrome/common/chrome_constants.h" 49 #include "chrome/common/chrome_paths.h" 50 #include "chrome/common/chrome_switches.h" 51 #include "chrome/common/env_vars.h" 52 #include "ipc/ipc_logging.h" 53 54 #if defined(OS_CHROMEOS) 55 #include "chromeos/chromeos_switches.h" 56 #endif 57 58 #if defined(OS_WIN) 59 #include <initguid.h> 60 #include "base/logging_win.h" 61 #endif 62 63 namespace { 64 65 // When true, this means that error dialogs should not be shown. 66 bool dialogs_are_suppressed_ = false; 67 68 // This should be true for exactly the period between the end of 69 // InitChromeLogging() and the beginning of CleanupChromeLogging(). 70 bool chrome_logging_initialized_ = false; 71 72 // Set if we called InitChromeLogging() but failed to initialize. 73 bool chrome_logging_failed_ = false; 74 75 // This should be true for exactly the period between the end of 76 // InitChromeLogging() and the beginning of CleanupChromeLogging(). 77 bool chrome_logging_redirected_ = false; 78 79 #if defined(OS_WIN) 80 // {7FE69228-633E-4f06-80C1-527FEA23E3A7} 81 const GUID kChromeTraceProviderName = { 82 0x7fe69228, 0x633e, 0x4f06, 83 { 0x80, 0xc1, 0x52, 0x7f, 0xea, 0x23, 0xe3, 0xa7 } }; 84 #endif 85 86 // Assertion handler for logging errors that occur when dialogs are 87 // silenced. To record a new error, pass the log string associated 88 // with that error in the str parameter. 89 MSVC_DISABLE_OPTIMIZE(); 90 void SilentRuntimeAssertHandler(const std::string& str) { 91 base::debug::BreakDebugger(); 92 } 93 MSVC_ENABLE_OPTIMIZE(); 94 95 // Suppresses error/assertion dialogs and enables the logging of 96 // those errors into silenced_errors_. 97 void SuppressDialogs() { 98 if (dialogs_are_suppressed_) 99 return; 100 101 logging::SetLogAssertHandler(SilentRuntimeAssertHandler); 102 103 #if defined(OS_WIN) 104 UINT new_flags = SEM_FAILCRITICALERRORS | 105 SEM_NOGPFAULTERRORBOX | 106 SEM_NOOPENFILEERRORBOX; 107 108 // Preserve existing error mode, as discussed at http://t/dmea 109 UINT existing_flags = SetErrorMode(new_flags); 110 SetErrorMode(existing_flags | new_flags); 111 #endif 112 113 dialogs_are_suppressed_ = true; 114 } 115 116 } // anonymous namespace 117 118 namespace logging { 119 120 LoggingDestination DetermineLogMode(const CommandLine& command_line) { 121 // only use OutputDebugString in debug mode 122 #ifdef NDEBUG 123 bool enable_logging = false; 124 const char *kInvertLoggingSwitch = switches::kEnableLogging; 125 const logging::LoggingDestination kDefaultLoggingMode = logging::LOG_TO_FILE; 126 #else 127 bool enable_logging = true; 128 const char *kInvertLoggingSwitch = switches::kDisableLogging; 129 const logging::LoggingDestination kDefaultLoggingMode = logging::LOG_TO_ALL; 130 #endif 131 132 if (command_line.HasSwitch(kInvertLoggingSwitch)) 133 enable_logging = !enable_logging; 134 135 logging::LoggingDestination log_mode; 136 if (enable_logging) { 137 // Let --enable-logging=stderr force only stderr, particularly useful for 138 // non-debug builds where otherwise you can't get logs to stderr at all. 139 if (command_line.GetSwitchValueASCII(switches::kEnableLogging) == "stderr") 140 log_mode = logging::LOG_TO_SYSTEM_DEBUG_LOG; 141 else 142 log_mode = kDefaultLoggingMode; 143 } else { 144 log_mode = logging::LOG_NONE; 145 } 146 return log_mode; 147 } 148 149 #if defined(OS_CHROMEOS) 150 namespace { 151 base::FilePath SetUpSymlinkIfNeeded(const base::FilePath& symlink_path, 152 bool new_log) { 153 DCHECK(!symlink_path.empty()); 154 155 // If not starting a new log, then just log through the existing 156 // symlink, but if the symlink doesn't exist, create it. If 157 // starting a new log, then delete the old symlink and make a new 158 // one to a fresh log file. 159 base::FilePath target_path; 160 bool symlink_exists = base::PathExists(symlink_path); 161 if (new_log || !symlink_exists) { 162 target_path = GenerateTimestampedName(symlink_path, base::Time::Now()); 163 164 // We don't care if the unlink fails; we're going to continue anyway. 165 if (::unlink(symlink_path.value().c_str()) == -1) { 166 if (symlink_exists) // only warn if we might expect it to succeed. 167 DPLOG(WARNING) << "Unable to unlink " << symlink_path.value(); 168 } 169 if (!base::CreateSymbolicLink(target_path, symlink_path)) { 170 DPLOG(ERROR) << "Unable to create symlink " << symlink_path.value() 171 << " pointing at " << target_path.value(); 172 } 173 } else { 174 if (!base::ReadSymbolicLink(symlink_path, &target_path)) 175 DPLOG(ERROR) << "Unable to read symlink " << symlink_path.value(); 176 } 177 return target_path; 178 } 179 180 void RemoveSymlinkAndLog(const base::FilePath& link_path, 181 const base::FilePath& target_path) { 182 if (::unlink(link_path.value().c_str()) == -1) 183 DPLOG(WARNING) << "Unable to unlink symlink " << link_path.value(); 184 if (::unlink(target_path.value().c_str()) == -1) 185 DPLOG(WARNING) << "Unable to unlink log file " << target_path.value(); 186 } 187 188 } // anonymous namespace 189 190 base::FilePath GetSessionLogDir(const CommandLine& command_line) { 191 base::FilePath log_dir; 192 std::string log_dir_str; 193 scoped_ptr<base::Environment> env(base::Environment::Create()); 194 if (env->GetVar(env_vars::kSessionLogDir, &log_dir_str) && 195 !log_dir_str.empty()) { 196 log_dir = base::FilePath(log_dir_str); 197 } else if (command_line.HasSwitch(chromeos::switches::kLoginProfile)) { 198 PathService::Get(chrome::DIR_USER_DATA, &log_dir); 199 base::FilePath profile_dir; 200 std::string login_profile_value = 201 command_line.GetSwitchValueASCII(chromeos::switches::kLoginProfile); 202 if (login_profile_value == chrome::kLegacyProfileDir || 203 login_profile_value == chrome::kTestUserProfileDir) { 204 profile_dir = base::FilePath(login_profile_value); 205 } else { 206 // We could not use g_browser_process > profile_helper() here. 207 std::string profile_dir_str = chrome::kProfileDirPrefix; 208 profile_dir_str.append(login_profile_value); 209 profile_dir = base::FilePath(profile_dir_str); 210 } 211 log_dir = log_dir.Append(profile_dir); 212 } 213 return log_dir; 214 } 215 216 base::FilePath GetSessionLogFile(const CommandLine& command_line) { 217 return GetSessionLogDir(command_line).Append(GetLogFileName().BaseName()); 218 } 219 220 void RedirectChromeLogging(const CommandLine& command_line) { 221 if (chrome_logging_redirected_) { 222 // TODO(nkostylev): Support multiple active users. http://crbug.com/230345 223 LOG(WARNING) << "NOT redirecting logging for multi-profiles case."; 224 return; 225 } 226 227 DCHECK(!chrome_logging_redirected_) << 228 "Attempted to redirect logging when it was already initialized."; 229 230 // Redirect logs to the session log directory, if set. Otherwise 231 // defaults to the profile dir. 232 base::FilePath log_path = GetSessionLogFile(command_line); 233 234 // Creating symlink causes us to do blocking IO on UI thread. 235 // Temporarily allow it until we fix http://crbug.com/61143 236 base::ThreadRestrictions::ScopedAllowIO allow_io; 237 // Always force a new symlink when redirecting. 238 base::FilePath target_path = SetUpSymlinkIfNeeded(log_path, true); 239 240 // ChromeOS always logs through the symlink, so it shouldn't be 241 // deleted if it already exists. 242 logging::LoggingSettings settings; 243 settings.logging_dest = DetermineLogMode(command_line); 244 settings.log_file = log_path.value().c_str(); 245 if (!logging::InitLogging(settings)) { 246 DLOG(ERROR) << "Unable to initialize logging to " << log_path.value(); 247 RemoveSymlinkAndLog(log_path, target_path); 248 } else { 249 chrome_logging_redirected_ = true; 250 } 251 } 252 253 #endif // OS_CHROMEOS 254 255 void InitChromeLogging(const CommandLine& command_line, 256 OldFileDeletionState delete_old_log_file) { 257 DCHECK(!chrome_logging_initialized_) << 258 "Attempted to initialize logging when it was already initialized."; 259 260 LoggingDestination logging_dest = DetermineLogMode(command_line); 261 LogLockingState log_locking_state = LOCK_LOG_FILE; 262 base::FilePath log_path; 263 #if defined(OS_CHROMEOS) 264 base::FilePath target_path; 265 #endif 266 267 // Don't resolve the log path unless we need to. Otherwise we leave an open 268 // ALPC handle after sandbox lockdown on Windows. 269 if ((logging_dest & LOG_TO_FILE) != 0) { 270 log_path = GetLogFileName(); 271 272 #if defined(OS_CHROMEOS) 273 // For BWSI (Incognito) logins, we want to put the logs in the user 274 // profile directory that is created for the temporary session instead 275 // of in the system log directory, for privacy reasons. 276 if (command_line.HasSwitch(chromeos::switches::kGuestSession)) 277 log_path = GetSessionLogFile(command_line); 278 279 // On ChromeOS we log to the symlink. We force creation of a new 280 // symlink if we've been asked to delete the old log, since that 281 // indicates the start of a new session. 282 target_path = SetUpSymlinkIfNeeded( 283 log_path, delete_old_log_file == logging::DELETE_OLD_LOG_FILE); 284 285 // Because ChromeOS manages the move to a new session by redirecting 286 // the link, it shouldn't remove the old file in the logging code, 287 // since that will remove the newly created link instead. 288 delete_old_log_file = logging::APPEND_TO_OLD_LOG_FILE; 289 #endif 290 } else { 291 log_locking_state = DONT_LOCK_LOG_FILE; 292 } 293 294 logging::LoggingSettings settings; 295 settings.logging_dest = logging_dest; 296 settings.log_file = log_path.value().c_str(); 297 settings.lock_log = log_locking_state; 298 settings.delete_old = delete_old_log_file; 299 bool success = logging::InitLogging(settings); 300 301 #if defined(OS_CHROMEOS) 302 if (!success) { 303 DPLOG(ERROR) << "Unable to initialize logging to " << log_path.value() 304 << " (which should be a link to " << target_path.value() << ")"; 305 RemoveSymlinkAndLog(log_path, target_path); 306 chrome_logging_failed_ = true; 307 return; 308 } 309 #else 310 if (!success) { 311 DPLOG(ERROR) << "Unable to initialize logging to " << log_path.value(); 312 chrome_logging_failed_ = true; 313 return; 314 } 315 #endif 316 317 // Default to showing error dialogs. 318 if (!CommandLine::ForCurrentProcess()->HasSwitch(switches::kNoErrorDialogs)) 319 logging::SetShowErrorDialogs(true); 320 321 // we want process and thread IDs because we have a lot of things running 322 logging::SetLogItems(true, // enable_process_id 323 true, // enable_thread_id 324 true, // enable_timestamp 325 false); // enable_tickcount 326 327 // We call running in unattended mode "headless", and allow 328 // headless mode to be configured either by the Environment 329 // Variable or by the Command Line Switch. This is for 330 // automated test purposes. 331 scoped_ptr<base::Environment> env(base::Environment::Create()); 332 if (env->HasVar(env_vars::kHeadless) || 333 command_line.HasSwitch(switches::kNoErrorDialogs)) 334 SuppressDialogs(); 335 336 // Use a minimum log level if the command line asks for one, 337 // otherwise leave it at the default level (INFO). 338 if (command_line.HasSwitch(switches::kLoggingLevel)) { 339 std::string log_level = command_line.GetSwitchValueASCII( 340 switches::kLoggingLevel); 341 int level = 0; 342 if (base::StringToInt(log_level, &level) && 343 level >= 0 && level < LOG_NUM_SEVERITIES) { 344 logging::SetMinLogLevel(level); 345 } else { 346 DLOG(WARNING) << "Bad log level: " << log_level; 347 } 348 } 349 350 #if defined(OS_WIN) 351 // Enable trace control and transport through event tracing for Windows. 352 logging::LogEventProvider::Initialize(kChromeTraceProviderName); 353 #endif 354 355 chrome_logging_initialized_ = true; 356 } 357 358 // This is a no-op, but we'll keep it around in case 359 // we need to do more cleanup in the future. 360 void CleanupChromeLogging() { 361 if (chrome_logging_failed_) 362 return; // We failed to initiailize logging, no cleanup. 363 364 DCHECK(chrome_logging_initialized_) << 365 "Attempted to clean up logging when it wasn't initialized."; 366 367 CloseLogFile(); 368 369 chrome_logging_initialized_ = false; 370 chrome_logging_redirected_ = false; 371 } 372 373 base::FilePath GetLogFileName() { 374 std::string filename; 375 scoped_ptr<base::Environment> env(base::Environment::Create()); 376 if (env->GetVar(env_vars::kLogFileName, &filename) && !filename.empty()) 377 return base::FilePath::FromUTF8Unsafe(filename); 378 379 const base::FilePath log_filename(FILE_PATH_LITERAL("chrome_debug.log")); 380 base::FilePath log_path; 381 382 if (PathService::Get(chrome::DIR_LOGS, &log_path)) { 383 log_path = log_path.Append(log_filename); 384 return log_path; 385 } else { 386 // error with path service, just use some default file somewhere 387 return log_filename; 388 } 389 } 390 391 bool DialogsAreSuppressed() { 392 return dialogs_are_suppressed_; 393 } 394 395 size_t GetFatalAssertions(AssertionList* assertions) { 396 // In this function, we don't assume that assertions is non-null, so 397 // that if you just want an assertion count, you can pass in NULL. 398 if (assertions) 399 assertions->clear(); 400 size_t assertion_count = 0; 401 402 std::ifstream log_file; 403 log_file.open(GetLogFileName().value().c_str()); 404 if (!log_file.is_open()) 405 return 0; 406 407 std::string utf8_line; 408 std::wstring wide_line; 409 while (!log_file.eof()) { 410 getline(log_file, utf8_line); 411 if (utf8_line.find(":FATAL:") != std::string::npos) { 412 wide_line = base::UTF8ToWide(utf8_line); 413 if (assertions) 414 assertions->push_back(wide_line); 415 ++assertion_count; 416 } 417 } 418 log_file.close(); 419 420 return assertion_count; 421 } 422 423 base::FilePath GenerateTimestampedName(const base::FilePath& base_path, 424 base::Time timestamp) { 425 base::Time::Exploded time_deets; 426 timestamp.LocalExplode(&time_deets); 427 std::string suffix = base::StringPrintf("_%02d%02d%02d-%02d%02d%02d", 428 time_deets.year, 429 time_deets.month, 430 time_deets.day_of_month, 431 time_deets.hour, 432 time_deets.minute, 433 time_deets.second); 434 return base_path.InsertBeforeExtensionASCII(suffix); 435 } 436 437 } // namespace logging 438