blob: 226bfc6728e74f34587890fe5ae82984c99021c4 [file] [log] [blame]
// 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/cpu_context_fuchsia.h"
#include <string.h>
namespace crashpad {
namespace internal {
#if defined(ARCH_CPU_X86_64)
void InitializeCPUContextX86_64_NoFloatingPoint(
const zx_thread_state_general_regs_t& thread_context,
CPUContextX86_64* context) {
memset(context, 0, sizeof(*context));
context->rax = thread_context.rax;
context->rbx = thread_context.rbx;
context->rcx = thread_context.rcx;
context->rdx = thread_context.rdx;
context->rdi = thread_context.rdi;
context->rsi = thread_context.rsi;
context->rbp = thread_context.rbp;
context->rsp = thread_context.rsp;
context->r8 = thread_context.r8;
context->r9 = thread_context.r9;
context->r10 = thread_context.r10;
context->r11 = thread_context.r11;
context->r12 = thread_context.r12;
context->r13 = thread_context.r13;
context->r14 = thread_context.r14;
context->r15 = thread_context.r15;
context->rip = thread_context.rip;
context->rflags = thread_context.rflags;
}
#elif defined(ARCH_CPU_ARM64)
void InitializeCPUContextARM64(
const zx_thread_state_general_regs_t& thread_context,
const zx_thread_state_vector_regs_t& vector_context,
CPUContextARM64* context) {
memset(context, 0, sizeof(*context));
// Fuchsia stores the link register (x30) on its own while Crashpad stores it
// with the other general purpose x0-x28 and x29 frame pointer registers. So
// we expect the size and number of elements to be off by one unit.
static_assert(sizeof(context->regs) - sizeof(context->regs[30]) ==
sizeof(thread_context.r),
"registers size mismatch");
static_assert((sizeof(context->regs) - sizeof(context->regs[30])) /
sizeof(context->regs[0]) ==
sizeof(thread_context.r) / sizeof(thread_context.r[0]),
"registers number of elements mismatch");
memcpy(&context->regs, &thread_context.r, sizeof(thread_context.r));
context->regs[30] = thread_context.lr;
context->sp = thread_context.sp;
context->pc = thread_context.pc;
// Only the NZCV flags (bits 31 to 28 respectively) of the cpsr register are
// readable and writable by userland on ARM64.
constexpr uint32_t kNZCV = 0xf0000000;
// Fuchsia uses the old "cspr" terminology from armv7 while Crashpad uses the
// new "spsr" terminology for armv8.
context->spsr = thread_context.cpsr & kNZCV;
if (thread_context.cpsr >
std::numeric_limits<decltype(context->spsr)>::max()) {
LOG(WARNING) << "cpsr truncation: we only expect the first 32 bits to be "
"set in the cpsr";
}
context->spsr =
static_cast<decltype(context->spsr)>(thread_context.cpsr) & kNZCV;
context->fpcr = vector_context.fpcr;
context->fpsr = vector_context.fpsr;
static_assert(sizeof(context->fpsimd) == sizeof(vector_context.v),
"registers size mismatch");
memcpy(&context->fpsimd, &vector_context.v, sizeof(vector_context.v));
}
#endif // ARCH_CPU_X86_64
} // namespace internal
} // namespace crashpad