blob: e0681b44838aa58d69977a5a9815e486a65763e2 [file] [log] [blame] [edit]
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/common/crash_keys.h"
#include <array>
#include <deque>
#include <string_view>
#include "base/base_switches.h"
#include "base/command_line.h"
#include "base/format_macros.h"
#include "base/no_destructor.h"
#include "base/strings/strcat.h"
#include "base/strings/string_split.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "build/build_config.h"
#include "chrome/common/chrome_switches.h"
#include "components/crash/core/common/crash_key.h"
#include "components/crash/core/common/crash_keys.h"
#include "components/webui/flags/flags_ui_switches.h"
#include "content/public/common/content_switches.h"
#if BUILDFLAG(IS_CHROMEOS)
#include "components/crash/core/app/crash_switches.h"
#include "gpu/command_buffer/service/gpu_switches.h"
#include "ui/gl/gl_switches.h"
#endif
namespace crash_keys {
namespace {
constexpr std::string_view kStringAnnotationsSwitch = "string-annotations";
// A convenient wrapper around a crash key and its name.
//
// The CrashKey contract requires that CrashKeyStrings are never
// moved, copied, or deleted (see
// third_party/crashpad/crashpad/client/annotation.h); since this class holds
// a CrashKeyString, it likewise cannot be moved, copied, or deleted.
class CrashKeyWithName {
public:
explicit CrashKeyWithName(std::string name)
: name_(std::move(name)), crash_key_(name_.c_str()) {}
CrashKeyWithName(const CrashKeyWithName&) = delete;
CrashKeyWithName& operator=(const CrashKeyWithName&) = delete;
CrashKeyWithName(CrashKeyWithName&&) = delete;
CrashKeyWithName& operator=(CrashKeyWithName&&) = delete;
~CrashKeyWithName() = delete;
std::string_view Name() const { return name_; }
std::string_view Value() const { return crash_key_.value(); }
void Clear() { crash_key_.Clear(); }
void Set(std::string_view value) { crash_key_.Set(value); }
private:
std::string name_;
crash_reporter::CrashKeyString<64> crash_key_;
};
void SplitAndPopulateCrashKeys(std::deque<CrashKeyWithName>& crash_keys,
std::string_view comma_separated_feature_list,
std::string crash_key_name_prefix) {
// Crash keys are indestructable so we can not simply empty the deque.
// Instead we must keep the previous crash keys alive and clear their values.
for (CrashKeyWithName& crash_key : crash_keys)
crash_key.Clear();
auto features =
base::SplitString(comma_separated_feature_list, ",",
base::TRIM_WHITESPACE, base::SPLIT_WANT_NONEMPTY);
for (size_t i = 0; i < features.size(); i++) {
if (crash_keys.size() <= i) {
crash_keys.emplace_back(base::StringPrintf(
"%s-%" PRIuS, crash_key_name_prefix.c_str(), i + 1));
}
CrashKeyWithName& crash_key = crash_keys[i];
crash_key.Set(features[i]);
}
}
// --enable-features and --disable-features often contain a long list not
// fitting into 64 bytes, hiding important information when analysing crashes.
// Therefore they are separated out in a list of CrashKeys, one for each enabled
// or disabled feature.
// They are also excluded from the default "switches".
void HandleEnableDisableFeatures(const base::CommandLine& command_line) {
static base::NoDestructor<std::deque<CrashKeyWithName>>
enabled_features_crash_keys;
static base::NoDestructor<std::deque<CrashKeyWithName>>
disabled_features_crash_keys;
SplitAndPopulateCrashKeys(
*enabled_features_crash_keys,
command_line.GetSwitchValueASCII(switches::kEnableFeatures),
"commandline-enabled-feature");
SplitAndPopulateCrashKeys(
*disabled_features_crash_keys,
command_line.GetSwitchValueASCII(switches::kDisableFeatures),
"commandline-disabled-feature");
}
// Return true if we DON'T want to upload this flag to the crash server.
bool IsBoringSwitch(const std::string& flag) {
static const auto kIgnoreSwitches = std::to_array<std::string_view>({
kStringAnnotationsSwitch,
switches::kEnableLogging,
switches::kFlagSwitchesBegin,
switches::kFlagSwitchesEnd,
switches::kLoggingLevel,
switches::kProcessType,
switches::kV,
switches::kVModule,
// This is a serialized buffer which won't fit in the default 64 bytes
// anyways. Should be switches::kGpuPreferences but we run into linking
// errors on Windows if we try to use that directly.
"gpu-preferences",
switches::kEnableFeatures,
switches::kDisableFeatures,
#if BUILDFLAG(IS_MAC)
switches::kMetricsClientID,
#elif BUILDFLAG(IS_CHROMEOS)
// --crash-loop-before is a "boring" switch because it is redundant;
// crash_reporter separately informs the crash server if it is doing
// crash-loop handling.
crash_reporter::switches::kCrashLoopBefore,
switches::kUseGL,
switches::kUserDataDir,
// Cros/CC flags are specified as raw strings to avoid dependency.
"child-wallpaper-large",
"child-wallpaper-small",
"default-wallpaper-large",
"default-wallpaper-small",
"guest-wallpaper-large",
"guest-wallpaper-small",
"enterprise-enable-forced-re-enrollment",
"enterprise-enable-forced-re-enrollment-on-flex",
"enterprise-enrollment-initial-modulus",
"enterprise-enrollment-modulus-limit",
"login-profile",
"login-user",
"use-cras",
#endif
});
#if BUILDFLAG(IS_WIN)
// Just about everything has this, don't bother.
if (base::StartsWith(flag, "/prefetch:", base::CompareCase::SENSITIVE))
return true;
#endif
if (!base::StartsWith(flag, "--", base::CompareCase::SENSITIVE))
return false;
size_t end = flag.find("=");
size_t len = (end == std::string::npos) ? flag.length() - 2 : end - 2;
for (size_t i = 0; i < std::size(kIgnoreSwitches); ++i) {
if (flag.compare(2, len, kIgnoreSwitches[i]) == 0)
return true;
}
return false;
}
std::deque<CrashKeyWithName>& GetCommandLineStringAnnotations() {
static base::NoDestructor<std::deque<CrashKeyWithName>>
command_line_string_annotations;
return *command_line_string_annotations;
}
void SetStringAnnotations(const base::CommandLine& command_line) {
// This is only meant to be used to pass annotations from the browser to
// children and not to be used on the browser command line.
if (!command_line.HasSwitch(switches::kProcessType)) {
return;
}
base::StringPairs annotations;
if (!base::SplitStringIntoKeyValuePairs(
command_line.GetSwitchValueASCII(kStringAnnotationsSwitch), '=', ',',
&annotations)) {
return;
}
for (const auto& [key, value] : annotations) {
GetCommandLineStringAnnotations().emplace_back(key).Set(value);
}
}
} // namespace
void AllocateCrashKeyInBrowserAndChildren(std::string_view key,
std::string_view value) {
GetCommandLineStringAnnotations().emplace_back(std::string(key)).Set(value);
}
void AppendStringAnnotationsCommandLineSwitch(base::CommandLine* command_line) {
std::string string_annotations;
for (const auto& crash_key : GetCommandLineStringAnnotations()) {
if (!string_annotations.empty()) {
string_annotations.push_back(',');
}
string_annotations = base::StrCat(
{string_annotations, crash_key.Name(), "=", crash_key.Value()});
}
if (string_annotations.empty()) {
return;
}
command_line->AppendSwitchASCII(kStringAnnotationsSwitch, string_annotations);
}
void SetCrashKeysFromCommandLine(const base::CommandLine& command_line) {
SetStringAnnotations(command_line);
HandleEnableDisableFeatures(command_line);
SetSwitchesFromCommandLine(command_line, &IsBoringSwitch);
}
} // namespace crash_keys