Index: components/crash/app/crash_keys_win.cc |
diff --git a/components/crash/app/crash_keys_win.cc b/components/crash/app/crash_keys_win.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..eda4340d5df8737837a0dfcd0b38872f2df262cc |
--- /dev/null |
+++ b/components/crash/app/crash_keys_win.cc |
@@ -0,0 +1,192 @@ |
+// Copyright 2014 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. |
+ |
+#include "components/crash/app/crash_keys_win.h" |
+ |
+#include <algorithm> |
+ |
+#include "base/base_switches.h" |
+#include "base/command_line.h" |
+#include "base/files/file_path.h" |
+#include "base/logging.h" |
+#include "base/strings/string_number_conversions.h" |
+#include "base/strings/stringprintf.h" |
+#include "components/crash/app/crash_reporter_client.h" |
+ |
+namespace breakpad { |
+ |
+using crash_reporter::CrashReporterClient; |
+ |
+namespace { |
+ |
+const size_t kMaxPluginPathLength = 256; |
+const size_t kMaxDynamicEntries = 256; |
+ |
+} // namespace |
+ |
+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; |
+} |
+ |
+// Appends the plugin path to |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. |
+ SetPluginPath(path.substr(path.size() - kMaxPluginPathLength)); |
+ return; |
+ } |
+ |
+ // The chunk size without terminator. |
+ const size_t kChunkSize = static_cast<size_t>( |
+ google_breakpad::CustomInfoEntry::kValueMaxLength - 1); |
+ |
+ int chunk_index = 0; |
+ size_t chunk_start = 0; // Current position inside |path| |
+ |
+ for (chunk_start = 0; chunk_start < path.size(); chunk_index++) { |
+ size_t chunk_length = std::min(kChunkSize, path.size() - chunk_start); |
+ |
+ 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())); |
+ |
+ chunk_start += chunk_length; |
+ } |
+} |
+ |
+// Appends the breakpad dump path to |g_custom_entries|. |
+void CrashKeysWin::SetBreakpadDumpPath(CrashReporterClient* crash_client) { |
+ base::FilePath crash_dumps_dir_path; |
+ if (crash_client->GetAlternativeCrashDumpLocation(&crash_dumps_dir_path)) { |
+ custom_entries_.push_back(google_breakpad::CustomInfoEntry( |
+ L"breakpad-dump-location", crash_dumps_dir_path.value().c_str())); |
+ } |
+} |
+ |
+// Returns the custom info structure based on the dll in parameter and the |
+// process type. |
+google_breakpad::CustomClientInfo* |
+CrashKeysWin::GetCustomInfo(const std::wstring& exe_path, |
+ const std::wstring& type, |
+ const std::wstring& profile_type, |
+ base::CommandLine* cmd_line, |
+ CrashReporterClient* crash_client) { |
+ base::string16 version, product; |
+ base::string16 special_build; |
+ base::string16 channel_name; |
+ |
+ crash_client->GetProductNameAndVersion( |
+ base::FilePath(exe_path), |
+ &product, |
+ &version, |
+ &special_build, |
+ &channel_name); |
+ |
+ // We only expect this method to be called once per process. |
+ // Common enties |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"ver", version.c_str())); |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"prod", product.c_str())); |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"plat", L"Win32")); |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"ptype", type.c_str())); |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry( |
+ L"pid", base::IntToString16(::GetCurrentProcessId()).c_str())); |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"channel", channel_name.c_str())); |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"profile-type", profile_type.c_str())); |
+ |
+ if (!special_build.empty()) { |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"special", special_build.c_str())); |
+ } |
+ |
+ if (type == L"plugin" || type == L"ppapi") { |
+ std::wstring plugin_path = cmd_line->GetSwitchValueNative("plugin-path"); |
+ if (!plugin_path.empty()) |
+ SetPluginPath(plugin_path); |
+ } |
+ |
+ // Check whether configuration management controls crash reporting. |
+ bool crash_reporting_enabled = true; |
+ bool controlled_by_policy = crash_client->ReportingIsEnforcedByPolicy( |
+ &crash_reporting_enabled); |
+ bool use_crash_service = !controlled_by_policy && |
+ (cmd_line->HasSwitch(switches::kNoErrorDialogs) || |
+ crash_client->IsRunningUnattended()); |
+ if (use_crash_service) |
+ SetBreakpadDumpPath(crash_client); |
+ |
+ // Create space for dynamic ad-hoc keys. The names and values are set using |
+ // the API defined in base/debug/crash_logging.h. |
+ 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. |
+ custom_entries_.push_back( |
+ google_breakpad::CustomInfoEntry(L"unspecified-crash-key", L"")); |
+ } |
+ |
+ static google_breakpad::CustomClientInfo custom_client_info; |
+ custom_client_info.entries = &custom_entries_.front(); |
+ custom_client_info.count = custom_entries_.size(); |
+ |
+ return &custom_client_info; |
+} |
+ |
+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. |
+ std::wstring safe_key(std::wstring(key).substr( |
+ 0, google_breakpad::CustomInfoEntry::kNameMaxLength - 1)); |
+ std::wstring safe_value(std::wstring(value).substr( |
+ 0, google_breakpad::CustomInfoEntry::kValueMaxLength - 1)); |
+ |
+ // 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. |
+ base::AutoLock lock(lock_); |
+ |
+ DynamicEntriesMap::iterator it = dynamic_entries_.find(safe_key); |
+ google_breakpad::CustomInfoEntry* entry = NULL; |
+ if (it == dynamic_entries_.end()) { |
+ if (dynamic_entries_.size() >= kMaxDynamicEntries) |
+ return; |
+ entry = &custom_entries_[dynamic_keys_offset_++]; |
+ dynamic_entries_.insert(std::make_pair(safe_key, entry)); |
+ } else { |
+ entry = it->second; |
+ } |
+ |
+ entry->set(safe_key.data(), safe_value.data()); |
+} |
+ |
+void CrashKeysWin::ClearCrashKeyValue(const std::wstring& key) { |
+ base::AutoLock lock(lock_); |
+ |
+ std::wstring key_string(key); |
+ DynamicEntriesMap::iterator it = dynamic_entries_.find(key_string); |
+ if (it == dynamic_entries_.end()) |
+ return; |
+ |
+ it->second->set_value(NULL); |
+} |
+ |
+} // namespace breakpad |