Index: components/breakpad/app/breakpad_win.cc |
diff --git a/components/breakpad/app/breakpad_win.cc b/components/breakpad/app/breakpad_win.cc |
index 4ac93ea5e838a2a3a086084e9fb8e3d5e27cabf3..ca3ae8cf009a947a605dcb021fe0a33a6f6db609 100644 |
--- a/components/breakpad/app/breakpad_win.cc |
+++ b/components/breakpad/app/breakpad_win.cc |
@@ -11,6 +11,7 @@ |
#include <winnt.h> |
#include <algorithm> |
+#include <map> |
#include <vector> |
#include "base/base_switches.h" |
@@ -45,8 +46,54 @@ |
namespace breakpad { |
-std::vector<google_breakpad::CustomInfoEntry>* g_custom_entries = NULL; |
-bool g_deferred_crash_uploads = false; |
+// Manages the breakpad key/value pair stash, there may only be one instance |
+// of this class per process at one time. |
+class CrashKeysWin { |
+ public: |
+ CrashKeysWin(); |
+ ~CrashKeysWin(); |
+ |
+ // May only be called once. |
+ google_breakpad::CustomClientInfo* |
+ GetCustomInfo(const std::wstring& exe_path, const std::wstring& type); |
+ |
+ void SetCrashKeyValue(const std::wstring& key, const std::wstring& value); |
+ void ClearCrashKeyValue(const std::wstring& key); |
+ |
+ static CrashKeysWin* keeper() { return keeper_; } |
+ |
+ private: |
+ // One-time initialization of private key/value pairs. |
+ void SetPluginPath(const std::wstring& path); |
+ void SetBreakpadDumpPath(); |
+ |
+ // Must not be resized after GetCustomInfo is invoked. |
+ std::vector<google_breakpad::CustomInfoEntry> custom_entries_; |
+ |
+ typedef std::map<std::wstring, google_breakpad::CustomInfoEntry*> |
+ DynamicEntriesMap; |
+ base::Lock lock_; |
+ // Keeps track of the next index for a new dynamic entry. |
+ size_t dynamic_keys_offset_; // Under lock_. |
+ // Maintains key->entry information for dynamic key/value entries |
+ // in custom_entries_. |
+ DynamicEntriesMap dynamic_entries_; // Under lock_. |
+ |
+ // Stores the sole instance of this class allowed per process. |
+ static CrashKeysWin* keeper_; |
+}; |
+ |
+CrashKeysWin* CrashKeysWin::keeper_; |
+ |
+CrashKeysWin::CrashKeysWin() : dynamic_keys_offset_(0) { |
+ DCHECK_EQ(static_cast<CrashKeysWin*>(NULL), keeper_); |
+ keeper_ = this; |
+} |
+ |
+CrashKeysWin::~CrashKeysWin() { |
+ DCHECK_EQ(this, keeper_); |
+ keeper_ = NULL; |
+} |
namespace { |
@@ -87,13 +134,6 @@ typedef NTSTATUS (WINAPI* NtTerminateProcessPtr)(HANDLE ProcessHandle, |
NTSTATUS ExitStatus); |
char* g_real_terminate_process_stub = NULL; |
-base::Lock* g_dynamic_entries_lock = NULL; |
-// Under *g_dynamic_entries_lock. |
-size_t g_dynamic_keys_offset = 0; |
-typedef std::map<std::wstring, google_breakpad::CustomInfoEntry*> |
- DynamicEntriesMap; |
-// Under *g_dynamic_entries_lock. |
-DynamicEntriesMap* g_dynamic_entries = NULL; |
// Allow for 256 dynamic entries in addition to the fixed set of entries |
// set up in this file. |
@@ -102,6 +142,8 @@ const size_t kMaxDynamicEntries = 256; |
// Maximum length for plugin path to include in plugin crash reports. |
const size_t kMaxPluginPathLength = 256; |
+} // namespace |
+ |
// Dumps the current process memory. |
extern "C" void __declspec(dllexport) __cdecl DumpProcess() { |
if (g_breakpad) { |
@@ -116,6 +158,8 @@ extern "C" void __declspec(dllexport) __cdecl DumpProcessWithoutCrash() { |
} |
} |
+namespace { |
+ |
// We need to prevent ICF from folding DumpForHangDebuggingThread() and |
// DumpProcessWithoutCrashThread() together, since that makes them |
// indistinguishable in crash dumps. We do this by making the function |
@@ -142,6 +186,8 @@ DWORD WINAPI DumpForHangDebuggingThread(void*) { |
MSVC_POP_WARNING() |
MSVC_ENABLE_OPTIMIZE() |
+} // namespace |
+ |
// Injects a thread into a remote process to dump state when there is no crash. |
extern "C" HANDLE __declspec(dllexport) __cdecl |
InjectDumpProcessWithoutCrash(HANDLE process) { |
@@ -156,9 +202,7 @@ InjectDumpForHangDebugging(HANDLE process) { |
} |
// Appends the plugin path to |g_custom_entries|. |
-void SetPluginPath(const std::wstring& path) { |
- DCHECK(g_custom_entries); |
- |
+void CrashKeysWin::SetPluginPath(const std::wstring& path) { |
if (path.size() > kMaxPluginPathLength) { |
// If the path is too long, truncate from the start rather than the end, |
// since we want to be able to recover the DLL name. |
@@ -176,7 +220,7 @@ void SetPluginPath(const std::wstring& path) { |
for (chunk_start = 0; chunk_start < path.size(); chunk_index++) { |
size_t chunk_length = std::min(kChunkSize, path.size() - chunk_start); |
- g_custom_entries->push_back(google_breakpad::CustomInfoEntry( |
+ custom_entries_.push_back(google_breakpad::CustomInfoEntry( |
base::StringPrintf(L"plugin-path-chunk-%i", chunk_index + 1).c_str(), |
path.substr(chunk_start, chunk_length).c_str())); |
@@ -185,12 +229,11 @@ void SetPluginPath(const std::wstring& path) { |
} |
// Appends the breakpad dump path to |g_custom_entries|. |
-void SetBreakpadDumpPath() { |
- DCHECK(g_custom_entries); |
+void CrashKeysWin::SetBreakpadDumpPath() { |
base::FilePath crash_dumps_dir_path; |
if (GetBreakpadClient()->GetAlternativeCrashDumpLocation( |
&crash_dumps_dir_path)) { |
- g_custom_entries->push_back(google_breakpad::CustomInfoEntry( |
+ custom_entries_.push_back(google_breakpad::CustomInfoEntry( |
L"breakpad-dump-location", crash_dumps_dir_path.value().c_str())); |
} |
} |
@@ -226,8 +269,9 @@ std::wstring GetProfileType() { |
// Returns the custom info structure based on the dll in parameter and the |
// process type. |
-google_breakpad::CustomClientInfo* GetCustomInfo(const std::wstring& exe_path, |
- const std::wstring& type) { |
+google_breakpad::CustomClientInfo* |
+CrashKeysWin::GetCustomInfo(const std::wstring& exe_path, |
+ const std::wstring& type) { |
base::string16 version, product; |
base::string16 special_build; |
base::string16 channel_name; |
@@ -239,33 +283,26 @@ google_breakpad::CustomClientInfo* GetCustomInfo(const std::wstring& exe_path, |
&channel_name); |
// We only expect this method to be called once per process. |
- DCHECK(!g_custom_entries); |
- g_custom_entries = new std::vector<google_breakpad::CustomInfoEntry>; |
- |
- // Common g_custom_entries. |
- g_custom_entries->push_back( |
+ // Common enties |
+ custom_entries_.push_back( |
google_breakpad::CustomInfoEntry(L"ver", |
base::UTF16ToWide(version).c_str())); |
- g_custom_entries->push_back( |
+ custom_entries_.push_back( |
google_breakpad::CustomInfoEntry(L"prod", |
base::UTF16ToWide(product).c_str())); |
- g_custom_entries->push_back( |
+ custom_entries_.push_back( |
google_breakpad::CustomInfoEntry(L"plat", L"Win32")); |
- g_custom_entries->push_back( |
+ custom_entries_.push_back( |
google_breakpad::CustomInfoEntry(L"ptype", type.c_str())); |
- g_custom_entries->push_back(google_breakpad::CustomInfoEntry( |
+ custom_entries_.push_back(google_breakpad::CustomInfoEntry( |
L"pid", base::StringPrintf(L"%d", ::GetCurrentProcessId()).c_str())); |
- g_custom_entries->push_back(google_breakpad::CustomInfoEntry( |
+ custom_entries_.push_back(google_breakpad::CustomInfoEntry( |
L"channel", base::UTF16ToWide(channel_name).c_str())); |
- g_custom_entries->push_back(google_breakpad::CustomInfoEntry( |
+ custom_entries_.push_back(google_breakpad::CustomInfoEntry( |
L"profile-type", GetProfileType().c_str())); |
- if (g_deferred_crash_uploads) |
- g_custom_entries->push_back( |
- google_breakpad::CustomInfoEntry(L"deferred-upload", L"true")); |
- |
if (!special_build.empty()) |
- g_custom_entries->push_back(google_breakpad::CustomInfoEntry( |
+ custom_entries_.push_back(google_breakpad::CustomInfoEntry( |
L"special", base::UTF16ToWide(special_build).c_str())); |
if (type == L"plugin" || type == L"ppapi") { |
@@ -288,25 +325,24 @@ google_breakpad::CustomClientInfo* GetCustomInfo(const std::wstring& exe_path, |
// Create space for dynamic ad-hoc keys. The names and values are set using |
// the API defined in base/debug/crash_logging.h. |
- g_dynamic_keys_offset = g_custom_entries->size(); |
+ dynamic_keys_offset_ = custom_entries_.size(); |
for (size_t i = 0; i < kMaxDynamicEntries; ++i) { |
// The names will be mutated as they are set. Un-numbered since these are |
// merely placeholders. The name cannot be empty because Breakpad's |
// HTTPUpload will interpret that as an invalid parameter. |
- g_custom_entries->push_back( |
+ custom_entries_.push_back( |
google_breakpad::CustomInfoEntry(L"unspecified-crash-key", L"")); |
} |
- g_dynamic_entries_lock = new base::Lock; |
- g_dynamic_entries = new DynamicEntriesMap; |
- |
static google_breakpad::CustomClientInfo custom_client_info; |
- custom_client_info.entries = &g_custom_entries->front(); |
- custom_client_info.count = g_custom_entries->size(); |
+ custom_client_info.entries = &custom_entries_.front(); |
+ custom_client_info.count = custom_entries_.size(); |
return &custom_client_info; |
} |
+namespace { |
+ |
// This callback is used when we want to get a dump without crashing the |
// process. |
bool DumpDoneCallbackWhenNoCrash(const wchar_t*, const wchar_t*, void*, |
@@ -396,15 +432,10 @@ long WINAPI ServiceExceptionFilter(EXCEPTION_POINTERS* info) { |
return EXCEPTION_EXECUTE_HANDLER; |
} |
-// NOTE: This function is used by SyzyASAN to annotate crash reports. If you |
-// change the name or signature of this function you will break SyzyASAN |
-// instrumented releases of Chrome. Please contact syzygy-team@chromium.org |
-// before doing so! |
-extern "C" void __declspec(dllexport) __cdecl SetCrashKeyValueImpl( |
- const wchar_t* key, const wchar_t* value) { |
- if (!g_dynamic_entries) |
- return; |
+} // namespace |
+void CrashKeysWin::SetCrashKeyValue( |
+ const std::wstring& key, const std::wstring& value) { |
// CustomInfoEntry limits the length of key and value. If they exceed |
// their maximum length the underlying string handling functions raise |
// an exception and prematurely trigger a crash. Truncate here. |
@@ -416,16 +447,15 @@ extern "C" void __declspec(dllexport) __cdecl SetCrashKeyValueImpl( |
// If we already have a value for this key, update it; otherwise, insert |
// the new value if we have not exhausted the pre-allocated slots for dynamic |
// entries. |
- DCHECK(g_dynamic_entries_lock); |
- base::AutoLock lock(*g_dynamic_entries_lock); |
+ base::AutoLock lock(lock_); |
- DynamicEntriesMap::iterator it = g_dynamic_entries->find(safe_key); |
+ DynamicEntriesMap::iterator it = dynamic_entries_.find(safe_key); |
google_breakpad::CustomInfoEntry* entry = NULL; |
- if (it == g_dynamic_entries->end()) { |
- if (g_dynamic_entries->size() >= kMaxDynamicEntries) |
+ if (it == dynamic_entries_.end()) { |
+ if (dynamic_entries_.size() >= kMaxDynamicEntries) |
return; |
- entry = &(*g_custom_entries)[g_dynamic_keys_offset++]; |
- g_dynamic_entries->insert(std::make_pair(safe_key, entry)); |
+ entry = &custom_entries_[dynamic_keys_offset_++]; |
+ dynamic_entries_.insert(std::make_pair(safe_key, entry)); |
} else { |
entry = it->second; |
} |
@@ -433,23 +463,42 @@ extern "C" void __declspec(dllexport) __cdecl SetCrashKeyValueImpl( |
entry->set(safe_key.data(), safe_value.data()); |
} |
-extern "C" void __declspec(dllexport) __cdecl ClearCrashKeyValueImpl( |
- const wchar_t* key) { |
- if (!g_dynamic_entries) |
+// NOTE: This function is used by SyzyASAN to annotate crash reports. If you |
+// change the name or signature of this function you will break SyzyASAN |
+// instrumented releases of Chrome. Please contact syzygy-team@chromium.org |
+// before doing so! |
+extern "C" void __declspec(dllexport) __cdecl SetCrashKeyValueImpl( |
+ const wchar_t* key, const wchar_t* value) { |
+ CrashKeysWin* keeper = CrashKeysWin::keeper(); |
+ if (!keeper) |
return; |
- DCHECK(g_dynamic_entries_lock); |
- base::AutoLock lock(*g_dynamic_entries_lock); |
+ // TODO(siggi): This doesn't look quite right - there's NULL deref potential |
+ // here, and an implicit std::wstring conversion. Fixme. |
+ keeper->SetCrashKeyValue(key, value); |
+} |
+ |
+void CrashKeysWin::ClearCrashKeyValue(const std::wstring& key) { |
+ base::AutoLock lock(lock_); |
std::wstring key_string(key); |
- DynamicEntriesMap::iterator it = g_dynamic_entries->find(key_string); |
- if (it == g_dynamic_entries->end()) |
+ DynamicEntriesMap::iterator it = dynamic_entries_.find(key_string); |
+ if (it == dynamic_entries_.end()) |
return; |
it->second->set_value(NULL); |
} |
-} // namespace |
+extern "C" void __declspec(dllexport) __cdecl ClearCrashKeyValueImpl( |
+ const wchar_t* key) { |
+ CrashKeysWin* keeper = CrashKeysWin::keeper(); |
+ if (!keeper) |
+ return; |
+ |
+ // TODO(siggi): This doesn't look quite right - there's NULL deref potential |
+ // here, and an implicit std::wstring conversion. Fixme. |
+ keeper->ClearCrashKeyValue(key); |
+} |
static bool WrapMessageBoxWithSEH(const wchar_t* text, const wchar_t* caption, |
UINT flags, bool* exit_now) { |
@@ -660,8 +709,11 @@ void InitCrashReporter(const std::string& process_type_switch) { |
bool is_per_user_install = |
GetBreakpadClient()->GetIsPerUserInstall(base::FilePath(exe_path)); |
+ // This is intentionally leaked. |
+ CrashKeysWin* keeper = new CrashKeysWin(); |
+ |
google_breakpad::CustomClientInfo* custom_info = |
- GetCustomInfo(exe_path, process_type); |
+ keeper->GetCustomInfo(exe_path, process_type); |
google_breakpad::ExceptionHandler::MinidumpCallback callback = NULL; |
LPTOP_LEVEL_EXCEPTION_FILTER default_filter = NULL; |