mirror of
https://github.com/RGBCube/serenity
synced 2025-05-31 07:08:10 +00:00
ptrace: Add PT_SETREGS
PT_SETTREGS sets the regsiters of the traced thread. It can only be used when the tracee is stopped. Also, refactor ptrace. The implementation was getting long and cluttered the alraedy large Process.cpp file. This commit moves the bulk of the implementation to Kernel/Ptrace.cpp, and factors out peek & poke to separate methods of the Process class.
This commit is contained in:
parent
0431712660
commit
9e51e295cf
11 changed files with 299 additions and 157 deletions
157
Kernel/Ptrace.cpp
Normal file
157
Kernel/Ptrace.cpp
Normal file
|
@ -0,0 +1,157 @@
|
|||
#include <Kernel/Process.h>
|
||||
#include <Kernel/Ptrace.h>
|
||||
#include <Kernel/Thread.h>
|
||||
#include <Kernel/ThreadTracer.h>
|
||||
#include <Kernel/VM/MemoryManager.h>
|
||||
#include <Kernel/VM/ProcessPagingScope.h>
|
||||
|
||||
namespace Ptrace {
|
||||
|
||||
KResultOr<u32> handle_syscall(const Kernel::Syscall::SC_ptrace_params& params, Process& caller)
|
||||
{
|
||||
if (params.request == PT_TRACE_ME) {
|
||||
if (Thread::current->tracer())
|
||||
return KResult(-EBUSY);
|
||||
|
||||
caller.set_wait_for_tracer_at_next_execve(true);
|
||||
return KSuccess;
|
||||
}
|
||||
|
||||
if (params.pid == caller.pid())
|
||||
return KResult(-EINVAL);
|
||||
|
||||
Thread* peer = nullptr;
|
||||
{
|
||||
InterruptDisabler disabler;
|
||||
peer = Thread::from_tid(params.pid);
|
||||
}
|
||||
if (!peer)
|
||||
return KResult(-ESRCH);
|
||||
|
||||
if ((peer->process().uid() != caller.euid())
|
||||
|| (peer->process().uid() != peer->process().euid())) // Disallow tracing setuid processes
|
||||
return KResult(-EACCES);
|
||||
|
||||
if (params.request == PT_ATTACH) {
|
||||
if (peer->tracer()) {
|
||||
return KResult(-EBUSY);
|
||||
}
|
||||
peer->start_tracing_from(caller.pid());
|
||||
if (peer->state() != Thread::State::Stopped && !(peer->has_blocker() && peer->blocker().is_reason_signal()))
|
||||
peer->send_signal(SIGSTOP, &caller);
|
||||
return KSuccess;
|
||||
}
|
||||
|
||||
auto* tracer = peer->tracer();
|
||||
|
||||
if (!tracer)
|
||||
return KResult(-EPERM);
|
||||
|
||||
if (tracer->tracer_pid() != caller.pid())
|
||||
return KResult(-EBUSY);
|
||||
|
||||
if (peer->state() == Thread::State::Running)
|
||||
return KResult(-EBUSY);
|
||||
|
||||
switch (params.request) {
|
||||
case PT_CONTINUE:
|
||||
peer->send_signal(SIGCONT, &caller);
|
||||
break;
|
||||
|
||||
case PT_DETACH:
|
||||
peer->stop_tracing();
|
||||
peer->send_signal(SIGCONT, &caller);
|
||||
break;
|
||||
|
||||
case PT_SYSCALL:
|
||||
tracer->set_trace_syscalls(true);
|
||||
peer->send_signal(SIGCONT, &caller);
|
||||
break;
|
||||
|
||||
case PT_GETREGS: {
|
||||
if (!tracer->has_regs())
|
||||
return KResult(-EINVAL);
|
||||
|
||||
PtraceRegisters* regs = reinterpret_cast<PtraceRegisters*>(params.addr);
|
||||
if (!caller.validate_write(regs, sizeof(PtraceRegisters)))
|
||||
return KResult(-EFAULT);
|
||||
|
||||
{
|
||||
SmapDisabler disabler;
|
||||
*regs = tracer->regs();
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case PT_SETREGS: {
|
||||
if (!tracer->has_regs())
|
||||
return KResult(-EINVAL);
|
||||
|
||||
if (!caller.validate_read(params.addr, sizeof(PtraceRegisters)))
|
||||
return KResult(-EFAULT);
|
||||
|
||||
auto& peer_saved_registers = peer->get_register_dump_from_stack();
|
||||
// Verify that the saved registers are in usermode context
|
||||
if ((peer_saved_registers.cs & 0x03) != 3)
|
||||
return -EFAULT;
|
||||
{
|
||||
SmapDisabler disabler;
|
||||
PtraceRegisters* regs = reinterpret_cast<PtraceRegisters*>(params.addr);
|
||||
tracer->set_regs(*regs);
|
||||
copy_ptrace_registers_into_kernel_registers(peer_saved_registers, *regs);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
case PT_PEEK: {
|
||||
u32* addr = reinterpret_cast<u32*>(params.addr);
|
||||
return peer->process().peek_user_data(addr);
|
||||
}
|
||||
|
||||
case PT_POKE: {
|
||||
u32* addr = reinterpret_cast<u32*>(params.addr);
|
||||
return peer->process().poke_user_data(addr, params.data);
|
||||
}
|
||||
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void copy_kernel_registers_into_ptrace_registers(PtraceRegisters& ptrace_regs, const RegisterState& kernel_regs)
|
||||
{
|
||||
ptrace_regs.eax = kernel_regs.eax,
|
||||
ptrace_regs.ecx = kernel_regs.ecx,
|
||||
ptrace_regs.edx = kernel_regs.edx,
|
||||
ptrace_regs.ebx = kernel_regs.ebx,
|
||||
ptrace_regs.esp = kernel_regs.userspace_esp,
|
||||
ptrace_regs.ebp = kernel_regs.ebp,
|
||||
ptrace_regs.esi = kernel_regs.esi,
|
||||
ptrace_regs.edi = kernel_regs.edi,
|
||||
ptrace_regs.eip = kernel_regs.eip,
|
||||
ptrace_regs.eflags = kernel_regs.eflags,
|
||||
ptrace_regs.cs = 0;
|
||||
ptrace_regs.ss = 0;
|
||||
ptrace_regs.ds = 0;
|
||||
ptrace_regs.es = 0;
|
||||
ptrace_regs.fs = 0;
|
||||
ptrace_regs.gs = 0;
|
||||
}
|
||||
|
||||
void copy_ptrace_registers_into_kernel_registers(RegisterState& kernel_regs, const PtraceRegisters& ptrace_regs)
|
||||
{
|
||||
kernel_regs.eax = ptrace_regs.eax;
|
||||
kernel_regs.ecx = ptrace_regs.ecx;
|
||||
kernel_regs.edx = ptrace_regs.edx;
|
||||
kernel_regs.ebx = ptrace_regs.ebx;
|
||||
kernel_regs.esp = ptrace_regs.esp;
|
||||
kernel_regs.ebp = ptrace_regs.ebp;
|
||||
kernel_regs.esi = ptrace_regs.esi;
|
||||
kernel_regs.edi = ptrace_regs.edi;
|
||||
kernel_regs.eip = ptrace_regs.eip;
|
||||
kernel_regs.eflags = ptrace_regs.eflags;
|
||||
}
|
||||
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue