// Copyright 2018 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/fuchsia/thread_snapshot_fuchsia.h" #include "base/logging.h" #include "snapshot/fuchsia/cpu_context_fuchsia.h" namespace crashpad { namespace internal { ThreadSnapshotFuchsia::ThreadSnapshotFuchsia() : ThreadSnapshot(), context_arch_(), context_(), stack_(), thread_id_(ZX_KOID_INVALID), thread_specific_data_address_(0), initialized_() {} ThreadSnapshotFuchsia::~ThreadSnapshotFuchsia() {} bool ThreadSnapshotFuchsia::Initialize( ProcessReaderFuchsia* process_reader, const ProcessReaderFuchsia::Thread& thread) { INITIALIZATION_STATE_SET_INITIALIZING(initialized_); #if defined(ARCH_CPU_X86_64) context_.architecture = kCPUArchitectureX86_64; context_.x86_64 = &context_arch_; // TODO(scottmg): Float context, once Fuchsia has a debug API to capture // floating point registers. ZX-1750 upstream. InitializeCPUContextX86_64(thread.general_registers, context_.x86_64); #elif defined(ARCH_CPU_ARM64) context_.architecture = kCPUArchitectureARM64; context_.arm64 = &context_arch_; // TODO(scottmg): Implement context capture for arm64. #else #error Port. #endif if (thread.stack_regions.empty()) { stack_.Initialize(process_reader, 0, 0); } else { stack_.Initialize(process_reader, thread.stack_regions[0].base(), thread.stack_regions[0].size()); // TODO(scottmg): Handle split stack by adding other parts to ExtraMemory(). } thread_id_ = thread.id; INITIALIZATION_STATE_SET_VALID(initialized_); return true; } const CPUContext* ThreadSnapshotFuchsia::Context() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return &context_; } const MemorySnapshot* ThreadSnapshotFuchsia::Stack() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return &stack_; } uint64_t ThreadSnapshotFuchsia::ThreadID() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return thread_id_; } int ThreadSnapshotFuchsia::SuspendCount() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); // There is not (currently) a suspend count for threads on Fuchsia. return 0; } int ThreadSnapshotFuchsia::Priority() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); // There is not (currently) thread priorities on Fuchsia. return 0; } uint64_t ThreadSnapshotFuchsia::ThreadSpecificDataAddress() const { INITIALIZATION_STATE_DCHECK_VALID(initialized_); return thread_specific_data_address_; } std::vector ThreadSnapshotFuchsia::ExtraMemory() const { return std::vector(); } } // namespace internal } // namespace crashpad