// Copyright 2019 The Crashpad Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "handler/linux/cros_crash_report_exception_handler.h" #include #include "base/logging.h" #include "base/strings/stringprintf.h" #if defined(CRASHPAD_IS_IN_CHROMIUM) #include "base/system/sys_info.h" #endif // CRASHPAD_IS_IN_CHROMIUM #include "client/settings.h" #include "handler/linux/capture_snapshot.h" #include "handler/minidump_to_upload_parameters.h" #include "minidump/minidump_file_writer.h" #include "snapshot/linux/process_snapshot_linux.h" #include "snapshot/minidump/process_snapshot_minidump.h" #include "snapshot/sanitized/process_snapshot_sanitized.h" #include "util/file/file_writer.h" #include "util/linux/direct_ptrace_connection.h" #include "util/linux/ptrace_client.h" #include "util/misc/metrics.h" #include "util/misc/uuid.h" #include "util/posix/spawn_subprocess.h" namespace crashpad { namespace { // Returns the process name for a pid. const std::string GetProcessNameFromPid(pid_t pid) { // Symlink to process binary is at /proc/###/exe. std::string link_path = "/proc/" + std::to_string(pid) + "/exe"; std::string result(4096, '\0'); ssize_t size = readlink(link_path.c_str(), result.data(), result.size()); if (size < 0) { PLOG(ERROR) << "Failed to readlink " << link_path; result.clear(); } else { result.resize(size); size_t last_slash_pos = result.rfind('/'); if (last_slash_pos != std::string::npos) { result = result.substr(last_slash_pos + 1); } } return result; } bool WriteAnnotationsAndMinidump( const std::map& parameters, MinidumpFileWriter& minidump, FileWriter& file_writer) { for (const auto& kv : parameters) { if (kv.first.find(':') != std::string::npos) { LOG(ERROR) << "Annotation key cannot have ':' in it " << kv.first; return false; } if (!file_writer.Write(kv.first.c_str(), strlen(kv.first.c_str()))) { return false; } if (!file_writer.Write(":", 1)) { return false; } size_t value_size = strlen(kv.second.c_str()); std::string value_size_str = std::to_string(value_size); if (!file_writer.Write(value_size_str.c_str(), value_size_str.size())) { return false; } if (!file_writer.Write(":", 1)) { return false; } if (!file_writer.Write(kv.second.c_str(), strlen(kv.second.c_str()))) { return false; } } static constexpr char kMinidumpName[] = "upload_file_minidump\"; filename=\"dump\":"; if (!file_writer.Write(kMinidumpName, sizeof(kMinidumpName) - 1)) { return false; } crashpad::FileOffset dump_size_start_offset = file_writer.Seek(0, SEEK_CUR); if (dump_size_start_offset < 0) { LOG(ERROR) << "Failed to get minidump size start offset"; return false; } static constexpr char kMinidumpLengthFilling[] = "00000000000000000000:"; if (!file_writer.Write(kMinidumpLengthFilling, sizeof(kMinidumpLengthFilling) - 1)) { return false; } crashpad::FileOffset dump_start_offset = file_writer.Seek(0, SEEK_CUR); if (dump_start_offset < 0) { LOG(ERROR) << "Failed to get minidump start offset"; return false; } if (!minidump.WriteEverything(&file_writer)) { return false; } crashpad::FileOffset dump_end_offset = file_writer.Seek(0, SEEK_CUR); if (dump_end_offset < 0) { LOG(ERROR) << "Failed to get minidump end offset"; return false; } size_t dump_data_size = dump_end_offset - dump_start_offset; std::string dump_data_size_str = std::to_string(dump_data_size); file_writer.Seek(dump_size_start_offset + strlen(kMinidumpLengthFilling) - 1 - dump_data_size_str.size(), SEEK_SET); if (!file_writer.Write(dump_data_size_str.c_str(), dump_data_size_str.size())) { return false; } return true; } } // namespace CrosCrashReportExceptionHandler::CrosCrashReportExceptionHandler( CrashReportDatabase* database, const std::map* process_annotations, const UserStreamDataSources* user_stream_data_sources) : database_(database), process_annotations_(process_annotations), user_stream_data_sources_(user_stream_data_sources), always_allow_feedback_(false) {} CrosCrashReportExceptionHandler::~CrosCrashReportExceptionHandler() = default; bool CrosCrashReportExceptionHandler::HandleException( pid_t client_process_id, uid_t client_uid, const ExceptionHandlerProtocol::ClientInformation& info, VMAddress requesting_thread_stack_address, pid_t* requesting_thread_id, UUID* local_report_id) { Metrics::ExceptionEncountered(); DirectPtraceConnection connection; if (!connection.Initialize(client_process_id)) { Metrics::ExceptionCaptureResult( Metrics::CaptureResult::kDirectPtraceFailed); return false; } return HandleExceptionWithConnection(&connection, info, client_uid, requesting_thread_stack_address, requesting_thread_id, local_report_id); } bool CrosCrashReportExceptionHandler::HandleExceptionWithBroker( pid_t client_process_id, uid_t client_uid, const ExceptionHandlerProtocol::ClientInformation& info, int broker_sock, UUID* local_report_id) { Metrics::ExceptionEncountered(); PtraceClient client; if (!client.Initialize(broker_sock, client_process_id)) { Metrics::ExceptionCaptureResult( Metrics::CaptureResult::kBrokeredPtraceFailed); return false; } return HandleExceptionWithConnection( &client, info, client_uid, 0, nullptr, local_report_id); } bool CrosCrashReportExceptionHandler::HandleExceptionWithConnection( PtraceConnection* connection, const ExceptionHandlerProtocol::ClientInformation& info, uid_t client_uid, VMAddress requesting_thread_stack_address, pid_t* requesting_thread_id, UUID* local_report_id) { std::unique_ptr process_snapshot; std::unique_ptr sanitized_snapshot; if (!CaptureSnapshot(connection, info, *process_annotations_, client_uid, requesting_thread_stack_address, requesting_thread_id, &process_snapshot, &sanitized_snapshot)) { return false; } UUID client_id; Settings* const settings = database_->GetSettings(); if (settings) { // If GetSettings() or GetClientID() fails, something else will log a // message and client_id will be left at its default value, all zeroes, // which is appropriate. settings->GetClientID(&client_id); } process_snapshot->SetClientID(client_id); UUID uuid; uuid.InitializeWithNew(); process_snapshot->SetReportID(uuid); ProcessSnapshot* snapshot = sanitized_snapshot ? implicit_cast(sanitized_snapshot.get()) : implicit_cast(process_snapshot.get()); MinidumpFileWriter minidump; minidump.InitializeFromSnapshot(snapshot); AddUserExtensionStreams(user_stream_data_sources_, snapshot, &minidump); FileWriter file_writer; if (!file_writer.OpenMemfd(base::FilePath("minidump"))) { Metrics::ExceptionCaptureResult(Metrics::CaptureResult::kOpenMemfdFailed); return false; } std::map parameters = BreakpadHTTPFormParametersFromMinidump(snapshot); // Used to differentiate between breakpad and crashpad while the switch is // ramping up. parameters.emplace("crash_library", "crashpad"); if (!WriteAnnotationsAndMinidump(parameters, minidump, file_writer)) { Metrics::ExceptionCaptureResult( Metrics::CaptureResult::kMinidumpWriteFailed); return false; } // CrOS uses crash_reporter instead of Crashpad to report crashes. // crash_reporter needs to know the pid and uid of the crashing process. std::vector argv({"/sbin/crash_reporter"}); #if defined(CRASHPAD_IS_IN_CHROMIUM) int32_t major_version = 0, minor_version = 0, bugfix_version = 0; base::SysInfo::OperatingSystemVersionNumbers( &major_version, &minor_version, &bugfix_version); // The version on which https://crrev.com/c/4265753 landed. constexpr int32_t kFixedVersion = 15363; // TODO(https://crbug.com/1420445): Remove this check (and the // CRASHPAD_IS_IN_CHROMIUM defines) when M115 branches. // (Lacros is guaranteed not to be more than 2 milestones ahead of ash, and // M113 on ash has the relevant crash_reporter change.) if (major_version >= kFixedVersion) { // Used to distinguish between non-fatal and fatal crashes. const ExceptionSnapshot* const exception_snapshot = snapshot->Exception(); if (exception_snapshot) { // convert to int32, since crashpad uses -1 as a signal for non-fatal // crashes. argv.push_back(base::StringPrintf( "--chrome_signal=%d", static_cast(exception_snapshot->Exception()))); } } #endif // CRASHPAD_IS_IN_CHROMIUM argv.push_back("--chrome_memfd=" + std::to_string(file_writer.fd())); const pid_t pid = process_snapshot->ProcessID(); argv.push_back("--pid=" + std::to_string(pid)); argv.push_back("--uid=" + std::to_string(client_uid)); std::string process_name = GetProcessNameFromPid(pid); argv.push_back("--exe=" + (process_name.empty() ? "chrome" : process_name)); if (info.crash_loop_before_time != 0) { argv.push_back("--crash_loop_before=" + std::to_string(info.crash_loop_before_time)); } if (!dump_dir_.empty()) { argv.push_back("--chrome_dump_dir=" + dump_dir_.value()); } if (always_allow_feedback_) { argv.push_back("--always_allow_feedback"); } if (!SpawnSubprocess(argv, nullptr /* envp */, file_writer.fd() /* preserve_fd */, false /* use_path */, nullptr /* child_function */)) { Metrics::ExceptionCaptureResult( Metrics::CaptureResult::kFinishedWritingCrashReportFailed); return false; } if (local_report_id != nullptr) { *local_report_id = uuid; } LOG(INFO) << "Successfully wrote report " << uuid.ToString(); Metrics::ExceptionCaptureResult(Metrics::CaptureResult::kSuccess); return true; } } // namespace crashpad