| Index: chrome_frame/crash_report.cc
|
| ===================================================================
|
| --- chrome_frame/crash_report.cc (revision 0)
|
| +++ chrome_frame/crash_report.cc (revision 0)
|
| @@ -0,0 +1,144 @@
|
| +// Copyright (c) 2009 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.
|
| +
|
| +// crash_report.cc : Implementation crash reporting.
|
| +#include "chrome_frame/crash_report.h"
|
| +
|
| +#include "base/file_util.h"
|
| +#include "base/logging.h"
|
| +#include "base/win_util.h"
|
| +#include "breakpad/src/client/windows/handler/exception_handler.h"
|
| +#include "chrome/installer/util/google_update_settings.h"
|
| +#include "chrome/installer/util/install_util.h"
|
| +#include "chrome_frame/vectored_handler.h"
|
| +#include "chrome_frame/vectored_handler-impl.h"
|
| +
|
| +namespace {
|
| +// TODO(joshia): factor out common code with chrome used for crash reporting
|
| +const wchar_t kGoogleUpdatePipeName[] = L"\\\\.\\pipe\\GoogleCrashServices\\";
|
| +const wchar_t kChromePipeName[] = L"\\\\.\\pipe\\ChromeCrashServices";
|
| +// Well known SID for the system principal.
|
| +const wchar_t kSystemPrincipalSid[] = L"S-1-5-18";
|
| +google_breakpad::ExceptionHandler* g_breakpad = NULL;
|
| +
|
| +// Returns the custom info structure based on the dll in parameter and the
|
| +// process type.
|
| +google_breakpad::CustomClientInfo* GetCustomInfo() {
|
| + // TODO(joshia): Grab these based on build.
|
| + static google_breakpad::CustomInfoEntry ver_entry(L"ver", L"0.1.0.0");
|
| + static google_breakpad::CustomInfoEntry prod_entry(L"prod", L"ChromeFrame");
|
| + 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;
|
| +}
|
| +
|
| +__declspec(naked)
|
| +static EXCEPTION_REGISTRATION_RECORD* InternalRtlpGetExceptionList() {
|
| + __asm {
|
| + mov eax, fs:0
|
| + ret
|
| + }
|
| +}
|
| +} // end of namespace
|
| +
|
| +// Class which methods simply forwards to Win32 API and uses breakpad to write
|
| +// a minidump. Used as template (external interface) of VectoredHandlerT<E>.
|
| +class Win32VEHTraits : public VEHTraitsBase {
|
| + public:
|
| + static inline void* Register(PVECTORED_EXCEPTION_HANDLER func,
|
| + const void* module_start, const void* module_end) {
|
| + VEHTraitsBase::SetModule(module_start, module_end);
|
| + return ::AddVectoredExceptionHandler(1, func);
|
| + }
|
| +
|
| + static inline ULONG Unregister(void* handle) {
|
| + return ::RemoveVectoredExceptionHandler(handle);
|
| + }
|
| +
|
| + static inline bool WriteDump(EXCEPTION_POINTERS* p) {
|
| + return g_breakpad->WriteMinidumpForException(p);
|
| + }
|
| +
|
| + static inline EXCEPTION_REGISTRATION_RECORD* RtlpGetExceptionList() {
|
| + return InternalRtlpGetExceptionList();
|
| + }
|
| +
|
| + static inline WORD RtlCaptureStackBackTrace(DWORD FramesToSkip,
|
| + DWORD FramesToCapture, void** BackTrace, DWORD* BackTraceHash) {
|
| + return ::RtlCaptureStackBackTrace(FramesToSkip, FramesToCapture,
|
| + BackTrace, BackTraceHash);
|
| + }
|
| +};
|
| +
|
| +extern "C" IMAGE_DOS_HEADER __ImageBase;
|
| +bool InitializeCrashReporting(bool use_crash_service, bool full_dump) {
|
| + if (g_breakpad)
|
| + return true;
|
| +
|
| + std::wstring pipe_name;
|
| + if (use_crash_service) {
|
| + // Crash reporting is done by crash_service.exe.
|
| + pipe_name = kChromePipeName;
|
| + } else {
|
| + // We want to use the Google Update crash reporting. We need to check if the
|
| + // user allows it first.
|
| + if (!GoogleUpdateSettings::GetCollectStatsConsent())
|
| + return true;
|
| +
|
| + // Build the pipe name. It can be either:
|
| + // System-wide install: "NamedPipe\GoogleCrashServices\S-1-5-18"
|
| + // Per-user install: "NamedPipe\GoogleCrashServices\<user SID>"
|
| + wchar_t dll_path[MAX_PATH * 2] = {0};
|
| + GetModuleFileName(reinterpret_cast<HMODULE>(&__ImageBase), dll_path,
|
| + arraysize(dll_path));
|
| +
|
| + std::wstring user_sid;
|
| + if (InstallUtil::IsPerUserInstall(dll_path)) {
|
| + if (!win_util::GetUserSidString(&user_sid)) {
|
| + return false;
|
| + }
|
| + } else {
|
| + user_sid = kSystemPrincipalSid;
|
| + }
|
| +
|
| + pipe_name = kGoogleUpdatePipeName;
|
| + pipe_name += user_sid;
|
| + }
|
| +
|
| + // Get the alternate dump directory. We use the temp path.
|
| + FilePath temp_directory;
|
| + if (!file_util::GetTempDir(&temp_directory) || temp_directory.empty()) {
|
| + return false;
|
| + }
|
| +
|
| + MINIDUMP_TYPE dump_type = full_dump ? MiniDumpWithFullMemory : MiniDumpNormal;
|
| + g_breakpad = new google_breakpad::ExceptionHandler(
|
| + temp_directory.value(), NULL, NULL, NULL,
|
| + google_breakpad::ExceptionHandler::HANDLER_INVALID_PARAMETER |
|
| + google_breakpad::ExceptionHandler::HANDLER_PURECALL, dump_type,
|
| + pipe_name.c_str(), GetCustomInfo());
|
| +
|
| + if (g_breakpad) {
|
| + // Find current module boundaries.
|
| + const void* start = &__ImageBase;
|
| + const char* s = reinterpret_cast<const char*>(start);
|
| + const IMAGE_NT_HEADERS32* nt = reinterpret_cast<const IMAGE_NT_HEADERS32*>
|
| + (s + __ImageBase.e_lfanew);
|
| + const void* end = s + nt->OptionalHeader.SizeOfImage;
|
| + VectoredHandler::Register(start, end);
|
| + }
|
| +
|
| + return g_breakpad != NULL;
|
| +}
|
| +
|
| +bool ShutdownCrashReporting() {
|
| + VectoredHandler::Unregister();
|
| + delete g_breakpad;
|
| + g_breakpad = NULL;
|
| + return true;
|
| +}
|
|
|