// Copyright (c) 2010 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Implementation of wrapper around common crash reporting. #include "chrome_frame/chrome_frame_reporting.h" #include "base/basictypes.h" #include "base/file_util.h" #include "base/file_version_info.h" #include "base/win/win_util.h" #include "chrome/installer/util/google_update_settings.h" #include "chrome/installer/util/install_util.h" #include "chrome_frame/crash_reporting/crash_report.h" #include "chrome_frame/exception_barrier.h" #include "chrome_frame/utils.h" extern "C" IMAGE_DOS_HEADER __ImageBase; namespace { // Well known SID for the system principal. const wchar_t kSystemPrincipalSid[] = L"S-1-5-18"; const wchar_t kChromePipeName[] = L"\\\\.\\pipe\\ChromeCrashServices"; // Returns the custom info structure based on the dll in parameter google_breakpad::CustomClientInfo* GetCustomInfo(const wchar_t* dll_path) { std::wstring product; std::wstring version; scoped_ptr<FileVersionInfo> version_info( FileVersionInfo::CreateFileVersionInfo(base::FilePath(dll_path))); if (version_info.get()) { version = version_info->product_version(); product = version_info->product_short_name(); } if (version.empty()) version = L"0.1.0.0"; if (product.empty()) product = L"ChromeFrame"; static google_breakpad::CustomInfoEntry ver_entry(L"ver", version.c_str()); static google_breakpad::CustomInfoEntry prod_entry(L"prod", product.c_str()); static google_breakpad::CustomInfoEntry plat_entry(L"plat", L"Win32"); static google_breakpad::CustomInfoEntry type_entry(L"ptype", L"chrome_frame"); static google_breakpad::CustomInfoEntry entries[] = { ver_entry, prod_entry, plat_entry, type_entry }; static google_breakpad::CustomClientInfo custom_info = { entries, arraysize(entries) }; return &custom_info; } bool InitializeCrashReporting() { // In headless mode we want crashes to be reported back. bool always_take_dump = IsHeadlessMode(); // We want to use the Google Update crash reporting. We need to check if the // user allows it first. if (!always_take_dump && !GoogleUpdateSettings::GetCollectStatsConsent()) return true; // If we got here, we want to report crashes, so make sure all // ExceptionBarrierBase instances do so. ExceptionBarrierConfig::set_enabled(true); // Get the alternate dump directory. We use the temp path. base::FilePath temp_directory; if (!base::GetTempDir(&temp_directory) || temp_directory.empty()) return false; wchar_t dll_path[MAX_PATH * 2] = {0}; GetModuleFileName(reinterpret_cast<HMODULE>(&__ImageBase), dll_path, arraysize(dll_path)); if (always_take_dump) { return InitializeVectoredCrashReportingWithPipeName( true, kChromePipeName, temp_directory.value(), GetCustomInfo(dll_path)); } // Build the pipe name. It can be either: // System-wide install: "NamedPipe\GoogleCrashServices\S-1-5-18" // Per-user install: "NamedPipe\GoogleCrashServices\<user SID>" std::wstring user_sid; if (InstallUtil::IsPerUserInstall(dll_path)) { if (!base::win::GetUserSidString(&user_sid)) return false; } else { user_sid = kSystemPrincipalSid; } return InitializeVectoredCrashReporting( false, user_sid.c_str(), temp_directory.value(), GetCustomInfo(dll_path)); } bool ShutdownCrashReporting() { ExceptionBarrierConfig::set_enabled(false); return ShutdownVectoredCrashReporting(); } } // namespace namespace chrome_frame { void CrashReportingTraits::Initialize() { InitializeCrashReporting(); } void CrashReportingTraits::Shutdown() { ShutdownCrashReporting(); } } // namespace chrome_frame