summaryrefslogtreecommitdiffstats
path: root/chrome/app/breakpad_win.cc
diff options
context:
space:
mode:
authoreroman@chromium.org <eroman@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-09-14 01:36:11 +0000
committereroman@chromium.org <eroman@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-09-14 01:36:11 +0000
commit783f06f30e8f4e9e6635d5e0ea49b258e8274cc0 (patch)
tree7fdc3cce535b96377edf93dff43d05f384110c0e /chrome/app/breakpad_win.cc
parent3227bd548c98f90ecabfcd4e3ccc3785ee9409d8 (diff)
downloadchromium_src-783f06f30e8f4e9e6635d5e0ea49b258e8274cc0.zip
chromium_src-783f06f30e8f4e9e6635d5e0ea49b258e8274cc0.tar.gz
chromium_src-783f06f30e8f4e9e6635d5e0ea49b258e8274cc0.tar.bz2
[Windows] Include the about:flags experiments in crash reports.
* Increases the number of command line switches included in dumps from 2 to 15. * Saves the command line flags for all process types (before it was just for browser). * Includes the "fake" command line flags that were added by about:flags experiments. This change will make it possible to cluster crashes with enabled experiments from our server-side analysis. BUG=60992 Review URL: http://codereview.chromium.org/7866033 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@101009 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome/app/breakpad_win.cc')
-rw-r--r--chrome/app/breakpad_win.cc88
1 files changed, 59 insertions, 29 deletions
diff --git a/chrome/app/breakpad_win.cc b/chrome/app/breakpad_win.cc
index d24bcda..63014d5 100644
--- a/chrome/app/breakpad_win.cc
+++ b/chrome/app/breakpad_win.cc
@@ -72,6 +72,13 @@ static size_t g_extension_ids_offset;
static size_t g_client_id_offset;
static size_t g_gpu_info_offset;
static size_t g_num_of_views_offset;
+static size_t g_num_switches_offset;
+static size_t g_switches_offset;
+
+// The maximum number of command line switches to include in the crash
+// report's metadata. Note that the mini-dump itself will also contain the
+// (original) command line arguments within the PEB.
+const size_t kMaxSwitches = 15;
// Dumps the current process memory.
extern "C" void __declspec(dllexport) __cdecl DumpProcess() {
@@ -88,6 +95,41 @@ std::wstring TrimToBreakpadMax(const std::wstring& str) {
google_breakpad::CustomInfoEntry::kValueMaxLength - 1);
}
+static void SetIntegerValue(size_t offset, int value) {
+ if (!g_custom_entries)
+ return;
+
+ wcscpy_s((*g_custom_entries)[offset].value,
+ google_breakpad::CustomInfoEntry::kValueMaxLength,
+ base::StringPrintf(L"%d", value).c_str());
+}
+
+extern "C" void __declspec(dllexport) __cdecl SetCommandLine(
+ const CommandLine* command_line) {
+ if (!g_custom_entries)
+ return;
+
+ const CommandLine::StringVector& argv = command_line->argv();
+
+ // Copy up to the kMaxSwitches arguments into the custom entries array. Skip
+ // past the first argument, as it is just the executable path.
+ size_t argv_i = 1;
+ size_t num_added = 0;
+
+ for (; argv_i < argv.size() && num_added < kMaxSwitches;
+ ++argv_i, ++num_added) {
+ // TODO(eroman): Filter out flags which aren't useful and just add bloat
+ // to the report.
+ wcsncpy((*g_custom_entries)[g_switches_offset + num_added].value,
+ argv[argv_i].c_str(),
+ google_breakpad::CustomInfoEntry::kValueMaxLength);
+ }
+
+ // Make note of the total number of switches. This is useful in case we have
+ // truncated at kMaxSwitches, to see how many were unaccounted for.
+ SetIntegerValue(g_num_switches_offset, static_cast<int>(argv.size()) - 1);
+}
+
// Returns the custom info structure based on the dll in parameter and the
// process type.
google_breakpad::CustomClientInfo* GetCustomInfo(const std::wstring& dll_path,
@@ -168,6 +210,23 @@ google_breakpad::CustomClientInfo* GetCustomInfo(const std::wstring& dll_path,
g_custom_entries->push_back(
google_breakpad::CustomInfoEntry(L"guid", guid.c_str()));
+ // Add empty values for the command line switches. We will fill them with
+ // actual values as part of SetCommandLine().
+ g_num_switches_offset = g_custom_entries->size();
+ g_custom_entries->push_back(
+ google_breakpad::CustomInfoEntry(L"num-switches", L""));
+
+ g_switches_offset = g_custom_entries->size();
+ for (int i = 0; i < kMaxSwitches; ++i) {
+ g_custom_entries->push_back(google_breakpad::CustomInfoEntry(
+ base::StringPrintf(L"switch-%i", i + 1).c_str(), L""));
+ }
+
+ // Fill in the command line arguments using CommandLine::ForCurrentProcess().
+ // The browser process may call SetCommandLine() again later on with a command
+ // line that has been augmented with the about:flags experiments.
+ SetCommandLine(CommandLine::ForCurrentProcess());
+
if (type == L"renderer" || type == L"plugin" || type == L"gpu-process") {
g_num_of_views_offset = g_custom_entries->size();
g_custom_entries->push_back(
@@ -183,26 +242,6 @@ google_breakpad::CustomClientInfo* GetCustomInfo(const std::wstring& dll_path,
} else {
g_custom_entries->push_back(
google_breakpad::CustomInfoEntry(L"num-views", L"N/A"));
-
- // Browser-specific g_custom_entries.
- google_breakpad::CustomInfoEntry switch1(L"switch-1", L"");
- google_breakpad::CustomInfoEntry switch2(L"switch-2", L"");
-
- // Get the first two command line switches if they exist. The CommandLine
- // class does not allow to enumerate the switches so we do it by hand.
- int num_args = 0;
- wchar_t** args = ::CommandLineToArgvW(::GetCommandLineW(), &num_args);
- if (args) {
- if (num_args > 1)
- switch1.set_value(TrimToBreakpadMax(args[1]).c_str());
- if (num_args > 2)
- switch2.set_value(TrimToBreakpadMax(args[2]).c_str());
- // The caller must free the memory allocated for |args|.
- ::LocalFree(args);
- }
-
- g_custom_entries->push_back(switch1);
- g_custom_entries->push_back(switch2);
}
static google_breakpad::CustomClientInfo custom_client_info;
@@ -337,15 +376,6 @@ extern "C" void __declspec(dllexport) __cdecl SetClientId(
client_id);
}
-static void SetIntegerValue(size_t offset, int value) {
- if (!g_custom_entries)
- return;
-
- wcscpy_s((*g_custom_entries)[offset].value,
- google_breakpad::CustomInfoEntry::kValueMaxLength,
- base::StringPrintf(L"%d", value).c_str());
-}
-
extern "C" void __declspec(dllexport) __cdecl SetNumberOfExtensions(
int number_of_extensions) {
SetIntegerValue(g_num_of_extensions_offset, number_of_extensions);