2023-11-17 21:44:53 +00:00
|
|
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
|
|
|
#include <cinttypes>
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
#include "common/signal_chain.h"
|
|
|
|
#include "core/arm/nce/arm_nce.h"
|
2023-11-30 19:05:02 +00:00
|
|
|
#include "core/arm/nce/interpreter_visitor.h"
|
2023-11-29 21:49:16 +00:00
|
|
|
#include "core/arm/nce/patcher.h"
|
2023-11-17 21:44:53 +00:00
|
|
|
#include "core/core.h"
|
|
|
|
#include "core/memory.h"
|
|
|
|
|
|
|
|
#include "core/hle/kernel/k_process.h"
|
|
|
|
|
|
|
|
#include <signal.h>
|
|
|
|
#include <sys/syscall.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
namespace Core {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2023-11-30 19:05:02 +00:00
|
|
|
struct sigaction g_orig_bus_action;
|
|
|
|
struct sigaction g_orig_segv_action;
|
2023-11-17 21:44:53 +00:00
|
|
|
|
|
|
|
// Verify assembly offsets.
|
|
|
|
using NativeExecutionParameters = Kernel::KThread::NativeExecutionParameters;
|
|
|
|
static_assert(offsetof(NativeExecutionParameters, native_context) == TpidrEl0NativeContext);
|
|
|
|
static_assert(offsetof(NativeExecutionParameters, lock) == TpidrEl0Lock);
|
|
|
|
static_assert(offsetof(NativeExecutionParameters, magic) == TpidrEl0TlsMagic);
|
|
|
|
|
|
|
|
fpsimd_context* GetFloatingPointState(mcontext_t& host_ctx) {
|
|
|
|
_aarch64_ctx* header = reinterpret_cast<_aarch64_ctx*>(&host_ctx.__reserved);
|
|
|
|
while (header->magic != FPSIMD_MAGIC) {
|
2023-11-19 09:21:53 +00:00
|
|
|
header = reinterpret_cast<_aarch64_ctx*>(reinterpret_cast<char*>(header) + header->size);
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
return reinterpret_cast<fpsimd_context*>(header);
|
|
|
|
}
|
|
|
|
|
2023-11-30 19:05:02 +00:00
|
|
|
using namespace Common::Literals;
|
|
|
|
constexpr u32 StackSize = 32_KiB;
|
|
|
|
|
2023-11-17 21:44:53 +00:00
|
|
|
} // namespace
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void* ArmNce::RestoreGuestContext(void* raw_context) {
|
2023-11-17 21:44:53 +00:00
|
|
|
// Retrieve the host context.
|
|
|
|
auto& host_ctx = static_cast<ucontext_t*>(raw_context)->uc_mcontext;
|
|
|
|
|
|
|
|
// Thread-local parameters will be located in x9.
|
|
|
|
auto* tpidr = reinterpret_cast<NativeExecutionParameters*>(host_ctx.regs[9]);
|
|
|
|
auto* guest_ctx = static_cast<GuestContext*>(tpidr->native_context);
|
|
|
|
|
|
|
|
// Retrieve the host floating point state.
|
|
|
|
auto* fpctx = GetFloatingPointState(host_ctx);
|
|
|
|
|
|
|
|
// Save host callee-saved registers.
|
|
|
|
std::memcpy(guest_ctx->host_ctx.host_saved_vregs.data(), &fpctx->vregs[8],
|
|
|
|
sizeof(guest_ctx->host_ctx.host_saved_vregs));
|
|
|
|
std::memcpy(guest_ctx->host_ctx.host_saved_regs.data(), &host_ctx.regs[19],
|
|
|
|
sizeof(guest_ctx->host_ctx.host_saved_regs));
|
|
|
|
|
|
|
|
// Save stack pointer.
|
|
|
|
guest_ctx->host_ctx.host_sp = host_ctx.sp;
|
|
|
|
|
|
|
|
// Restore all guest state except tpidr_el0.
|
|
|
|
host_ctx.sp = guest_ctx->sp;
|
|
|
|
host_ctx.pc = guest_ctx->pc;
|
|
|
|
host_ctx.pstate = guest_ctx->pstate;
|
|
|
|
fpctx->fpcr = guest_ctx->fpcr;
|
|
|
|
fpctx->fpsr = guest_ctx->fpsr;
|
|
|
|
std::memcpy(host_ctx.regs, guest_ctx->cpu_registers.data(), sizeof(host_ctx.regs));
|
|
|
|
std::memcpy(fpctx->vregs, guest_ctx->vector_registers.data(), sizeof(fpctx->vregs));
|
|
|
|
|
|
|
|
// Return the new thread-local storage pointer.
|
|
|
|
return tpidr;
|
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::SaveGuestContext(GuestContext* guest_ctx, void* raw_context) {
|
2023-11-17 21:44:53 +00:00
|
|
|
// Retrieve the host context.
|
|
|
|
auto& host_ctx = static_cast<ucontext_t*>(raw_context)->uc_mcontext;
|
|
|
|
|
|
|
|
// Retrieve the host floating point state.
|
|
|
|
auto* fpctx = GetFloatingPointState(host_ctx);
|
|
|
|
|
|
|
|
// Save all guest registers except tpidr_el0.
|
|
|
|
std::memcpy(guest_ctx->cpu_registers.data(), host_ctx.regs, sizeof(host_ctx.regs));
|
|
|
|
std::memcpy(guest_ctx->vector_registers.data(), fpctx->vregs, sizeof(fpctx->vregs));
|
|
|
|
guest_ctx->fpsr = fpctx->fpsr;
|
|
|
|
guest_ctx->fpcr = fpctx->fpcr;
|
|
|
|
guest_ctx->pstate = static_cast<u32>(host_ctx.pstate);
|
|
|
|
guest_ctx->pc = host_ctx.pc;
|
|
|
|
guest_ctx->sp = host_ctx.sp;
|
|
|
|
|
|
|
|
// Restore stack pointer.
|
|
|
|
host_ctx.sp = guest_ctx->host_ctx.host_sp;
|
|
|
|
|
|
|
|
// Restore host callee-saved registers.
|
|
|
|
std::memcpy(&host_ctx.regs[19], guest_ctx->host_ctx.host_saved_regs.data(),
|
|
|
|
sizeof(guest_ctx->host_ctx.host_saved_regs));
|
|
|
|
std::memcpy(&fpctx->vregs[8], guest_ctx->host_ctx.host_saved_vregs.data(),
|
|
|
|
sizeof(guest_ctx->host_ctx.host_saved_vregs));
|
|
|
|
|
|
|
|
// Return from the call on exit by setting pc to x30.
|
|
|
|
host_ctx.pc = guest_ctx->host_ctx.host_saved_regs[11];
|
|
|
|
|
|
|
|
// Clear esr_el1 and return it.
|
|
|
|
host_ctx.regs[0] = guest_ctx->esr_el1.exchange(0);
|
|
|
|
}
|
|
|
|
|
2023-11-30 19:05:02 +00:00
|
|
|
bool ArmNce::HandleFailedGuestFault(GuestContext* guest_ctx, void* raw_info, void* raw_context) {
|
2023-11-17 21:44:53 +00:00
|
|
|
auto& host_ctx = static_cast<ucontext_t*>(raw_context)->uc_mcontext;
|
|
|
|
auto* info = static_cast<siginfo_t*>(raw_info);
|
|
|
|
|
2023-11-22 04:23:48 +00:00
|
|
|
// We can't handle the access, so determine why we crashed.
|
2023-11-17 21:44:53 +00:00
|
|
|
const bool is_prefetch_abort = host_ctx.pc == reinterpret_cast<u64>(info->si_addr);
|
2023-11-22 04:23:48 +00:00
|
|
|
|
|
|
|
// For data aborts, skip the instruction and return to guest code.
|
|
|
|
// This will allow games to continue in many scenarios where they would otherwise crash.
|
|
|
|
if (!is_prefetch_abort) {
|
|
|
|
host_ctx.pc += 4;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is a prefetch abort.
|
|
|
|
guest_ctx->esr_el1.fetch_or(static_cast<u64>(HaltReason::PrefetchAbort));
|
2023-11-17 21:44:53 +00:00
|
|
|
|
|
|
|
// Forcibly mark the context as locked. We are still running.
|
|
|
|
// We may race with SignalInterrupt here:
|
2023-11-19 09:21:53 +00:00
|
|
|
// - If we lose the race, then SignalInterrupt will send us a signal we are masking,
|
2023-11-17 21:44:53 +00:00
|
|
|
// and it will do nothing when it is unmasked, as we have already left guest code.
|
|
|
|
// - If we win the race, then SignalInterrupt will wait for us to unlock first.
|
2023-11-28 19:30:39 +00:00
|
|
|
auto& thread_params = guest_ctx->parent->m_running_thread->GetNativeExecutionParameters();
|
2023-11-17 21:44:53 +00:00
|
|
|
thread_params.lock.store(SpinLockLocked);
|
|
|
|
|
|
|
|
// Return to host.
|
|
|
|
SaveGuestContext(guest_ctx, raw_context);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-11-30 19:05:02 +00:00
|
|
|
bool ArmNce::HandleGuestAlignmentFault(GuestContext* guest_ctx, void* raw_info, void* raw_context) {
|
|
|
|
auto& host_ctx = static_cast<ucontext_t*>(raw_context)->uc_mcontext;
|
|
|
|
auto* fpctx = GetFloatingPointState(host_ctx);
|
|
|
|
auto& memory = guest_ctx->system->ApplicationMemory();
|
|
|
|
|
|
|
|
// Match and execute an instruction.
|
|
|
|
auto next_pc = MatchAndExecuteOneInstruction(memory, &host_ctx, fpctx);
|
|
|
|
if (next_pc) {
|
|
|
|
host_ctx.pc = *next_pc;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We couldn't handle the access.
|
|
|
|
return HandleFailedGuestFault(guest_ctx, raw_info, raw_context);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ArmNce::HandleGuestAccessFault(GuestContext* guest_ctx, void* raw_info, void* raw_context) {
|
|
|
|
auto* info = static_cast<siginfo_t*>(raw_info);
|
|
|
|
|
|
|
|
// Try to handle an invalid access.
|
|
|
|
// TODO: handle accesses which split a page?
|
|
|
|
const Common::ProcessAddress addr =
|
|
|
|
(reinterpret_cast<u64>(info->si_addr) & ~Memory::YUZU_PAGEMASK);
|
|
|
|
if (guest_ctx->system->ApplicationMemory().InvalidateNCE(addr, Memory::YUZU_PAGESIZE)) {
|
|
|
|
// We handled the access successfully and are returning to guest code.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We couldn't handle the access.
|
|
|
|
return HandleFailedGuestFault(guest_ctx, raw_info, raw_context);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArmNce::HandleHostAlignmentFault(int sig, void* raw_info, void* raw_context) {
|
|
|
|
return g_orig_bus_action.sa_sigaction(sig, static_cast<siginfo_t*>(raw_info), raw_context);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArmNce::HandleHostAccessFault(int sig, void* raw_info, void* raw_context) {
|
|
|
|
return g_orig_segv_action.sa_sigaction(sig, static_cast<siginfo_t*>(raw_info), raw_context);
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::LockThread(Kernel::KThread* thread) {
|
2023-11-17 21:44:53 +00:00
|
|
|
auto* thread_params = &thread->GetNativeExecutionParameters();
|
2023-11-28 19:30:39 +00:00
|
|
|
LockThreadParameters(thread_params);
|
|
|
|
}
|
2023-11-17 21:44:53 +00:00
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::UnlockThread(Kernel::KThread* thread) {
|
|
|
|
auto* thread_params = &thread->GetNativeExecutionParameters();
|
|
|
|
UnlockThreadParameters(thread_params);
|
|
|
|
}
|
2023-11-17 21:44:53 +00:00
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
HaltReason ArmNce::RunThread(Kernel::KThread* thread) {
|
|
|
|
// Check if we're already interrupted.
|
|
|
|
// If we are, we can just return immediately.
|
|
|
|
HaltReason hr = static_cast<HaltReason>(m_guest_ctx.esr_el1.exchange(0));
|
|
|
|
if (True(hr)) {
|
|
|
|
return hr;
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
// Get the thread context.
|
|
|
|
auto* thread_params = &thread->GetNativeExecutionParameters();
|
|
|
|
auto* process = thread->GetOwnerProcess();
|
|
|
|
|
2023-11-17 21:44:53 +00:00
|
|
|
// Assign current members.
|
2023-11-28 19:30:39 +00:00
|
|
|
m_running_thread = thread;
|
|
|
|
m_guest_ctx.parent = this;
|
|
|
|
thread_params->native_context = &m_guest_ctx;
|
|
|
|
thread_params->tpidr_el0 = m_guest_ctx.tpidr_el0;
|
|
|
|
thread_params->tpidrro_el0 = m_guest_ctx.tpidrro_el0;
|
2023-11-17 21:44:53 +00:00
|
|
|
thread_params->is_running = true;
|
|
|
|
|
|
|
|
// TODO: finding and creating the post handler needs to be locked
|
|
|
|
// to deal with dynamic loading of NROs.
|
2023-11-28 19:30:39 +00:00
|
|
|
const auto& post_handlers = process->GetPostHandlers();
|
|
|
|
if (auto it = post_handlers.find(m_guest_ctx.pc); it != post_handlers.end()) {
|
|
|
|
hr = ReturnToRunCodeByTrampoline(thread_params, &m_guest_ctx, it->second);
|
2023-11-17 21:44:53 +00:00
|
|
|
} else {
|
2023-11-28 19:30:39 +00:00
|
|
|
hr = ReturnToRunCodeByExceptionLevelChange(m_thread_id, thread_params);
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Unload members.
|
|
|
|
// The thread does not change, so we can persist the old reference.
|
2023-11-28 19:30:39 +00:00
|
|
|
m_running_thread = nullptr;
|
|
|
|
m_guest_ctx.tpidr_el0 = thread_params->tpidr_el0;
|
2023-11-17 21:44:53 +00:00
|
|
|
thread_params->native_context = nullptr;
|
|
|
|
thread_params->is_running = false;
|
|
|
|
|
|
|
|
// Return the halt reason.
|
2023-11-28 19:30:39 +00:00
|
|
|
return hr;
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
HaltReason ArmNce::StepThread(Kernel::KThread* thread) {
|
2023-11-17 21:44:53 +00:00
|
|
|
return HaltReason::StepThread;
|
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
u32 ArmNce::GetSvcNumber() const {
|
|
|
|
return m_guest_ctx.svc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArmNce::GetSvcArguments(std::span<uint64_t, 8> args) const {
|
|
|
|
for (size_t i = 0; i < 8; i++) {
|
|
|
|
args[i] = m_guest_ctx.cpu_registers[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArmNce::SetSvcArguments(std::span<const uint64_t, 8> args) {
|
|
|
|
for (size_t i = 0; i < 8; i++) {
|
|
|
|
m_guest_ctx.cpu_registers[i] = args[i];
|
|
|
|
}
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
ArmNce::ArmNce(System& system, bool uses_wall_clock, std::size_t core_index)
|
|
|
|
: ArmInterface{uses_wall_clock}, m_system{system}, m_core_index{core_index} {
|
|
|
|
m_guest_ctx.system = &m_system;
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
ArmNce::~ArmNce() = default;
|
2023-11-17 21:44:53 +00:00
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::Initialize() {
|
2023-12-12 04:02:01 +00:00
|
|
|
if (m_thread_id == -1) {
|
|
|
|
m_thread_id = gettid();
|
|
|
|
}
|
2023-11-17 21:44:53 +00:00
|
|
|
|
2023-11-30 19:05:02 +00:00
|
|
|
// Configure signal stack.
|
2023-12-12 04:02:01 +00:00
|
|
|
if (!m_stack) {
|
|
|
|
m_stack = std::make_unique<u8[]>(StackSize);
|
|
|
|
|
|
|
|
stack_t ss{};
|
|
|
|
ss.ss_sp = m_stack.get();
|
|
|
|
ss.ss_size = StackSize;
|
|
|
|
sigaltstack(&ss, nullptr);
|
|
|
|
}
|
2023-11-30 19:05:02 +00:00
|
|
|
|
|
|
|
// Set up signals.
|
|
|
|
static std::once_flag flag;
|
|
|
|
std::call_once(flag, [] {
|
2023-11-17 21:44:53 +00:00
|
|
|
using HandlerType = decltype(sigaction::sa_sigaction);
|
|
|
|
|
|
|
|
sigset_t signal_mask;
|
|
|
|
sigemptyset(&signal_mask);
|
|
|
|
sigaddset(&signal_mask, ReturnToRunCodeByExceptionLevelChangeSignal);
|
|
|
|
sigaddset(&signal_mask, BreakFromRunCodeSignal);
|
2023-11-30 19:05:02 +00:00
|
|
|
sigaddset(&signal_mask, GuestAlignmentFaultSignal);
|
|
|
|
sigaddset(&signal_mask, GuestAccessFaultSignal);
|
2023-11-17 21:44:53 +00:00
|
|
|
|
|
|
|
struct sigaction return_to_run_code_action {};
|
|
|
|
return_to_run_code_action.sa_flags = SA_SIGINFO | SA_ONSTACK;
|
|
|
|
return_to_run_code_action.sa_sigaction = reinterpret_cast<HandlerType>(
|
2023-11-28 19:30:39 +00:00
|
|
|
&ArmNce::ReturnToRunCodeByExceptionLevelChangeSignalHandler);
|
2023-11-17 21:44:53 +00:00
|
|
|
return_to_run_code_action.sa_mask = signal_mask;
|
|
|
|
Common::SigAction(ReturnToRunCodeByExceptionLevelChangeSignal, &return_to_run_code_action,
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
struct sigaction break_from_run_code_action {};
|
|
|
|
break_from_run_code_action.sa_flags = SA_SIGINFO | SA_ONSTACK;
|
|
|
|
break_from_run_code_action.sa_sigaction =
|
2023-11-28 19:30:39 +00:00
|
|
|
reinterpret_cast<HandlerType>(&ArmNce::BreakFromRunCodeSignalHandler);
|
2023-11-17 21:44:53 +00:00
|
|
|
break_from_run_code_action.sa_mask = signal_mask;
|
|
|
|
Common::SigAction(BreakFromRunCodeSignal, &break_from_run_code_action, nullptr);
|
|
|
|
|
2023-11-30 19:05:02 +00:00
|
|
|
struct sigaction alignment_fault_action {};
|
|
|
|
alignment_fault_action.sa_flags = SA_SIGINFO | SA_ONSTACK;
|
|
|
|
alignment_fault_action.sa_sigaction =
|
|
|
|
reinterpret_cast<HandlerType>(&ArmNce::GuestAlignmentFaultSignalHandler);
|
|
|
|
alignment_fault_action.sa_mask = signal_mask;
|
|
|
|
Common::SigAction(GuestAlignmentFaultSignal, &alignment_fault_action, nullptr);
|
|
|
|
|
|
|
|
struct sigaction access_fault_action {};
|
|
|
|
access_fault_action.sa_flags = SA_SIGINFO | SA_ONSTACK | SA_RESTART;
|
|
|
|
access_fault_action.sa_sigaction =
|
|
|
|
reinterpret_cast<HandlerType>(&ArmNce::GuestAccessFaultSignalHandler);
|
|
|
|
access_fault_action.sa_mask = signal_mask;
|
|
|
|
Common::SigAction(GuestAccessFaultSignal, &access_fault_action, &g_orig_segv_action);
|
2023-11-17 21:44:53 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::SetTpidrroEl0(u64 value) {
|
|
|
|
m_guest_ctx.tpidrro_el0 = value;
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::GetContext(Kernel::Svc::ThreadContext& ctx) const {
|
|
|
|
for (size_t i = 0; i < 29; i++) {
|
|
|
|
ctx.r[i] = m_guest_ctx.cpu_registers[i];
|
|
|
|
}
|
|
|
|
ctx.fp = m_guest_ctx.cpu_registers[29];
|
|
|
|
ctx.lr = m_guest_ctx.cpu_registers[30];
|
|
|
|
ctx.sp = m_guest_ctx.sp;
|
|
|
|
ctx.pc = m_guest_ctx.pc;
|
|
|
|
ctx.pstate = m_guest_ctx.pstate;
|
|
|
|
ctx.v = m_guest_ctx.vector_registers;
|
|
|
|
ctx.fpcr = m_guest_ctx.fpcr;
|
|
|
|
ctx.fpsr = m_guest_ctx.fpsr;
|
|
|
|
ctx.tpidr = m_guest_ctx.tpidr_el0;
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::SetContext(const Kernel::Svc::ThreadContext& ctx) {
|
|
|
|
for (size_t i = 0; i < 29; i++) {
|
|
|
|
m_guest_ctx.cpu_registers[i] = ctx.r[i];
|
|
|
|
}
|
|
|
|
m_guest_ctx.cpu_registers[29] = ctx.fp;
|
|
|
|
m_guest_ctx.cpu_registers[30] = ctx.lr;
|
|
|
|
m_guest_ctx.sp = ctx.sp;
|
|
|
|
m_guest_ctx.pc = ctx.pc;
|
|
|
|
m_guest_ctx.pstate = ctx.pstate;
|
|
|
|
m_guest_ctx.vector_registers = ctx.v;
|
|
|
|
m_guest_ctx.fpcr = ctx.fpcr;
|
|
|
|
m_guest_ctx.fpsr = ctx.fpsr;
|
|
|
|
m_guest_ctx.tpidr_el0 = ctx.tpidr;
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::SignalInterrupt(Kernel::KThread* thread) {
|
2023-11-17 21:44:53 +00:00
|
|
|
// Add break loop condition.
|
2023-11-28 19:30:39 +00:00
|
|
|
m_guest_ctx.esr_el1.fetch_or(static_cast<u64>(HaltReason::BreakLoop));
|
2023-11-17 21:44:53 +00:00
|
|
|
|
|
|
|
// Lock the thread context.
|
2023-11-28 19:30:39 +00:00
|
|
|
auto* params = &thread->GetNativeExecutionParameters();
|
2023-11-17 21:44:53 +00:00
|
|
|
LockThreadParameters(params);
|
|
|
|
|
|
|
|
if (params->is_running) {
|
|
|
|
// We should signal to the running thread.
|
|
|
|
// The running thread will unlock the thread context.
|
2023-11-28 19:30:39 +00:00
|
|
|
syscall(SYS_tkill, m_thread_id, BreakFromRunCodeSignal);
|
2023-11-17 21:44:53 +00:00
|
|
|
} else {
|
|
|
|
// If the thread is no longer running, we have nothing to do.
|
|
|
|
UnlockThreadParameters(params);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::ClearInstructionCache() {
|
2023-11-17 21:44:53 +00:00
|
|
|
// TODO: This is not possible to implement correctly on Linux because
|
|
|
|
// we do not have any access to ic iallu.
|
|
|
|
|
|
|
|
// Require accesses to complete.
|
|
|
|
std::atomic_thread_fence(std::memory_order_seq_cst);
|
|
|
|
}
|
|
|
|
|
2023-11-28 19:30:39 +00:00
|
|
|
void ArmNce::InvalidateCacheRange(u64 addr, std::size_t size) {
|
2023-11-19 18:57:23 +00:00
|
|
|
this->ClearInstructionCache();
|
2023-11-17 21:44:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Core
|