mirror of
https://github.com/chromium/crashpad.git
synced 2025-03-10 06:36:02 +00:00
Gather most of the necessary information for the thread snapshot. Note that: - The 'capture' portion of this CL will be moved out of the snapshot interface and into a separate in-process dump to disk location. - All of the pointer dereferences need to be wrapped in vm_read. - The read-fast-and-dump logic in thread_snapshot may end up in a different file completely, but until we pick a serialization/deserialization method, keep it as-is. Change-Id: I80ba323cb6a59ac0dd1bba9150d047ba83cc4dad Reviewed-on: https://chromium-review.googlesource.com/c/crashpad/crashpad/+/2085572 Reviewed-by: Mark Mentovai <mark@chromium.org> Reviewed-by: Justin Cohen <justincohen@chromium.org>
188 lines
5.7 KiB
C++
188 lines
5.7 KiB
C++
// Copyright 2020 The Crashpad Authors. All rights reserved.
|
|
//
|
|
// 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 "snapshot/ios/process_snapshot_ios.h"
|
|
|
|
#include <mach-o/loader.h>
|
|
#include <mach/mach.h>
|
|
|
|
#include <utility>
|
|
|
|
#include "base/logging.h"
|
|
#include "base/mac/mach_logging.h"
|
|
|
|
namespace crashpad {
|
|
|
|
ProcessSnapshotIOS::ProcessSnapshotIOS()
|
|
: ProcessSnapshot(),
|
|
threads_(),
|
|
modules_(),
|
|
report_id_(),
|
|
client_id_(),
|
|
annotations_simple_map_(),
|
|
snapshot_time_(),
|
|
initialized_() {}
|
|
|
|
ProcessSnapshotIOS::~ProcessSnapshotIOS() {}
|
|
|
|
bool ProcessSnapshotIOS::Initialize() {
|
|
INITIALIZATION_STATE_SET_INITIALIZING(initialized_);
|
|
|
|
if (gettimeofday(&snapshot_time_, nullptr) != 0) {
|
|
PLOG(ERROR) << "gettimeofday";
|
|
return false;
|
|
}
|
|
|
|
InitializeThreads();
|
|
InitializeModules();
|
|
|
|
INITIALIZATION_STATE_SET_VALID(initialized_);
|
|
return true;
|
|
}
|
|
|
|
pid_t ProcessSnapshotIOS::ProcessID() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return getpid();
|
|
}
|
|
|
|
pid_t ProcessSnapshotIOS::ParentProcessID() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return 0;
|
|
}
|
|
|
|
void ProcessSnapshotIOS::SnapshotTime(timeval* snapshot_time) const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
*snapshot_time = snapshot_time_;
|
|
}
|
|
|
|
void ProcessSnapshotIOS::ProcessStartTime(timeval* start_time) const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
}
|
|
|
|
void ProcessSnapshotIOS::ProcessCPUTimes(timeval* user_time,
|
|
timeval* system_time) const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
}
|
|
|
|
void ProcessSnapshotIOS::ReportID(UUID* report_id) const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
*report_id = report_id_;
|
|
}
|
|
|
|
void ProcessSnapshotIOS::ClientID(UUID* client_id) const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
*client_id = client_id_;
|
|
}
|
|
|
|
const std::map<std::string, std::string>&
|
|
ProcessSnapshotIOS::AnnotationsSimpleMap() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return annotations_simple_map_;
|
|
}
|
|
|
|
const SystemSnapshot* ProcessSnapshotIOS::System() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return nullptr;
|
|
}
|
|
|
|
std::vector<const ThreadSnapshot*> ProcessSnapshotIOS::Threads() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
std::vector<const ThreadSnapshot*> threads;
|
|
for (const auto& thread : threads_) {
|
|
threads.push_back(thread.get());
|
|
}
|
|
return threads;
|
|
}
|
|
|
|
std::vector<const ModuleSnapshot*> ProcessSnapshotIOS::Modules() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
std::vector<const ModuleSnapshot*> modules;
|
|
for (const auto& module : modules_) {
|
|
modules.push_back(module.get());
|
|
}
|
|
return modules;
|
|
}
|
|
|
|
std::vector<UnloadedModuleSnapshot> ProcessSnapshotIOS::UnloadedModules()
|
|
const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return std::vector<UnloadedModuleSnapshot>();
|
|
}
|
|
|
|
const ExceptionSnapshot* ProcessSnapshotIOS::Exception() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return nullptr;
|
|
}
|
|
|
|
std::vector<const MemoryMapRegionSnapshot*> ProcessSnapshotIOS::MemoryMap()
|
|
const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return std::vector<const MemoryMapRegionSnapshot*>();
|
|
}
|
|
|
|
std::vector<HandleSnapshot> ProcessSnapshotIOS::Handles() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return std::vector<HandleSnapshot>();
|
|
}
|
|
|
|
std::vector<const MemorySnapshot*> ProcessSnapshotIOS::ExtraMemory() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return std::vector<const MemorySnapshot*>();
|
|
}
|
|
|
|
const ProcessMemory* ProcessSnapshotIOS::Memory() const {
|
|
INITIALIZATION_STATE_DCHECK_VALID(initialized_);
|
|
return nullptr;
|
|
}
|
|
|
|
void ProcessSnapshotIOS::InitializeThreads() {
|
|
mach_msg_type_number_t thread_count = 0;
|
|
const thread_act_array_t threads =
|
|
internal::ThreadSnapshotIOS::GetThreads(&thread_count);
|
|
for (uint32_t thread_index = 0; thread_index < thread_count; ++thread_index) {
|
|
thread_t thread = threads[thread_index];
|
|
auto thread_snapshot = std::make_unique<internal::ThreadSnapshotIOS>();
|
|
if (thread_snapshot->Initialize(thread)) {
|
|
threads_.push_back(std::move(thread_snapshot));
|
|
}
|
|
mach_port_deallocate(mach_task_self(), thread);
|
|
}
|
|
// TODO(justincohen): This dealloc above and below needs to move with the
|
|
// call to task_threads inside internal::ThreadSnapshotIOS::GetThreads.
|
|
vm_deallocate(mach_task_self(),
|
|
reinterpret_cast<vm_address_t>(threads),
|
|
sizeof(thread_t) * thread_count);
|
|
}
|
|
|
|
void ProcessSnapshotIOS::InitializeModules() {
|
|
const dyld_all_image_infos* image_infos =
|
|
internal::ModuleSnapshotIOS::DyldAllImageInfo();
|
|
|
|
uint32_t image_count = image_infos->infoArrayCount;
|
|
const dyld_image_info* image_array = image_infos->infoArray;
|
|
for (uint32_t image_index = 0; image_index < image_count; ++image_index) {
|
|
const dyld_image_info* image = &image_array[image_index];
|
|
auto module = std::make_unique<internal::ModuleSnapshotIOS>();
|
|
if (module->Initialize(image)) {
|
|
modules_.push_back(std::move(module));
|
|
}
|
|
}
|
|
auto module = std::make_unique<internal::ModuleSnapshotIOS>();
|
|
if (module->InitializeDyld(image_infos)) {
|
|
modules_.push_back(std::move(module));
|
|
}
|
|
}
|
|
|
|
} // namespace crashpad
|