mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-25 18:52:22 -05:00
d16d805d96
It makes much more sense to have these actions being performed via the prctl syscall, as they both require 2 plain arguments to be passed to the syscall layer, and in contrast to most syscalls, we don't get in these removed syscalls an automatic representation of Userspace<T>, but two FlatPtr(s) to perform casting on them in the prctl syscall which is suited to what has been done in the removed syscalls. Also, it makes sense to have these actions in the prctl syscall, because they are strongly related to the process control concept of the prctl syscall.
1638 lines
50 KiB
C++
1638 lines
50 KiB
C++
/*
|
|
* Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
|
|
* Copyright (c) 2022, Rummskartoffel <Rummskartoffel@protonmail.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include "Emulator.h"
|
|
#include "MmapRegion.h"
|
|
#include "SimpleRegion.h"
|
|
#include "SoftCPU.h"
|
|
#include <AK/Debug.h>
|
|
#include <AK/Format.h>
|
|
#include <Kernel/API/SyscallString.h>
|
|
#include <alloca.h>
|
|
#include <fcntl.h>
|
|
#include <sched.h>
|
|
#include <serenity.h>
|
|
#include <strings.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/poll.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/time.h>
|
|
#include <sys/uio.h>
|
|
#include <sys/utsname.h>
|
|
#include <syscall.h>
|
|
#include <termios.h>
|
|
|
|
#if defined(AK_COMPILER_GCC)
|
|
# pragma GCC optimize("O3")
|
|
#endif
|
|
|
|
namespace UserspaceEmulator {
|
|
|
|
u32 Emulator::virt_syscall(u32 function, u32 arg1, u32 arg2, u32 arg3)
|
|
{
|
|
if constexpr (SPAM_DEBUG)
|
|
reportln("Syscall: {} ({:x})"sv, Syscall::to_string((Syscall::Function)function), function);
|
|
switch (function) {
|
|
case SC_accept4:
|
|
return virt$accept4(arg1);
|
|
case SC_allocate_tls:
|
|
return virt$allocate_tls(arg1, arg2);
|
|
case SC_anon_create:
|
|
return virt$anon_create(arg1, arg2);
|
|
case SC_annotate_mapping:
|
|
return virt$annotate_mapping(arg1);
|
|
case SC_beep:
|
|
return virt$beep();
|
|
case SC_bind:
|
|
return virt$bind(arg1, arg2, arg3);
|
|
case SC_chdir:
|
|
return virt$chdir(arg1, arg2);
|
|
case SC_chmod:
|
|
return virt$chmod(arg1);
|
|
case SC_chown:
|
|
return virt$chown(arg1);
|
|
case SC_clock_gettime:
|
|
return virt$clock_gettime(arg1, arg2);
|
|
case SC_clock_nanosleep:
|
|
return virt$clock_nanosleep(arg1);
|
|
case SC_clock_settime:
|
|
return virt$clock_settime(arg1, arg2);
|
|
case SC_close:
|
|
return virt$close(arg1);
|
|
case SC_connect:
|
|
return virt$connect(arg1, arg2, arg3);
|
|
case SC_create_inode_watcher:
|
|
return virt$create_inode_watcher(arg1);
|
|
case SC_dbgputstr:
|
|
return virt$dbgputstr(arg1, arg2);
|
|
case SC_disown:
|
|
return virt$disown(arg1);
|
|
case SC_dup2:
|
|
return virt$dup2(arg1, arg2);
|
|
case SC_emuctl:
|
|
return virt$emuctl(arg1, arg2, arg3);
|
|
case SC_execve:
|
|
return virt$execve(arg1);
|
|
case SC_exit:
|
|
virt$exit((int)arg1);
|
|
return 0;
|
|
case SC_faccessat:
|
|
return virt$faccessat(arg1);
|
|
case SC_fchmod:
|
|
return virt$fchmod(arg1, arg2);
|
|
case SC_fchown:
|
|
return virt$fchown(arg1, arg2, arg3);
|
|
case SC_fcntl:
|
|
return virt$fcntl(arg1, arg2, arg3);
|
|
case SC_fork:
|
|
return virt$fork();
|
|
case SC_fstat:
|
|
return virt$fstat(arg1, arg2);
|
|
case SC_ftruncate:
|
|
return virt$ftruncate(arg1, arg2);
|
|
case SC_futex:
|
|
return virt$futex(arg1);
|
|
case SC_get_dir_entries:
|
|
return virt$get_dir_entries(arg1, arg2, arg3);
|
|
case SC_get_stack_bounds:
|
|
return virt$get_stack_bounds(arg1, arg2);
|
|
case SC_getcwd:
|
|
return virt$getcwd(arg1, arg2);
|
|
case SC_getegid:
|
|
return virt$getegid();
|
|
case SC_geteuid:
|
|
return virt$geteuid();
|
|
case SC_getgid:
|
|
return virt$getgid();
|
|
case SC_getgroups:
|
|
return virt$getgroups(arg1, arg2);
|
|
case SC_gethostname:
|
|
return virt$gethostname(arg1, arg2);
|
|
case SC_getpeername:
|
|
return virt$getpeername(arg1);
|
|
case SC_getpgid:
|
|
return virt$getpgid(arg1);
|
|
case SC_getpgrp:
|
|
return virt$getpgrp();
|
|
case SC_getpid:
|
|
return virt$getpid();
|
|
case SC_getppid:
|
|
return virt$getppid();
|
|
case SC_getrandom:
|
|
return virt$getrandom(arg1, arg2, arg3);
|
|
case SC_getsid:
|
|
return virt$getsid(arg1);
|
|
case SC_getsockname:
|
|
return virt$getsockname(arg1);
|
|
case SC_getsockopt:
|
|
return virt$getsockopt(arg1);
|
|
case SC_gettid:
|
|
return virt$gettid();
|
|
case SC_getuid:
|
|
return virt$getuid();
|
|
case SC_inode_watcher_add_watch:
|
|
return virt$inode_watcher_add_watch(arg1);
|
|
case SC_inode_watcher_remove_watch:
|
|
return virt$inode_watcher_remove_watch(arg1, arg2);
|
|
case SC_ioctl:
|
|
return virt$ioctl(arg1, arg2, arg3);
|
|
case SC_kill:
|
|
return virt$kill(arg1, arg2);
|
|
case SC_killpg:
|
|
return virt$killpg(arg1, arg2);
|
|
case SC_listen:
|
|
return virt$listen(arg1, arg2);
|
|
case SC_lseek:
|
|
return virt$lseek(arg1, arg2, arg3);
|
|
case SC_madvise:
|
|
return virt$madvise(arg1, arg2, arg3);
|
|
case SC_map_time_page:
|
|
return -ENOSYS;
|
|
case SC_mkdir:
|
|
return virt$mkdir(arg1, arg2, arg3);
|
|
case SC_mmap:
|
|
return virt$mmap(arg1);
|
|
case SC_mount:
|
|
return virt$mount(arg1);
|
|
case SC_mprotect:
|
|
return virt$mprotect(arg1, arg2, arg3);
|
|
case SC_mremap:
|
|
return virt$mremap(arg1);
|
|
case SC_munmap:
|
|
return virt$munmap(arg1, arg2);
|
|
case SC_open:
|
|
return virt$open(arg1);
|
|
case SC_perf_event:
|
|
return virt$perf_event((int)arg1, arg2, arg3);
|
|
case SC_perf_register_string:
|
|
return virt$perf_register_string(arg1, arg2);
|
|
case SC_pipe:
|
|
return virt$pipe(arg1, arg2);
|
|
case SC_pledge:
|
|
return virt$pledge(arg1);
|
|
case SC_poll:
|
|
return virt$poll(arg1);
|
|
case SC_profiling_disable:
|
|
return virt$profiling_disable(arg1);
|
|
case SC_profiling_enable:
|
|
return virt$profiling_enable(arg1);
|
|
case SC_purge:
|
|
return virt$purge(arg1);
|
|
case SC_read:
|
|
return virt$read(arg1, arg2, arg3);
|
|
case SC_readlink:
|
|
return virt$readlink(arg1);
|
|
case SC_realpath:
|
|
return virt$realpath(arg1);
|
|
case SC_recvfd:
|
|
return virt$recvfd(arg1, arg2);
|
|
case SC_recvmsg:
|
|
return virt$recvmsg(arg1, arg2, arg3);
|
|
case SC_rename:
|
|
return virt$rename(arg1);
|
|
case SC_rmdir:
|
|
return virt$rmdir(arg1, arg2);
|
|
case SC_scheduler_get_parameters:
|
|
return virt$scheduler_get_parameters(arg1);
|
|
case SC_scheduler_set_parameters:
|
|
return virt$scheduler_set_parameters(arg1);
|
|
case SC_sendfd:
|
|
return virt$sendfd(arg1, arg2);
|
|
case SC_sendmsg:
|
|
return virt$sendmsg(arg1, arg2, arg3);
|
|
case SC_set_mmap_name:
|
|
return virt$set_mmap_name(arg1);
|
|
case SC_set_thread_name:
|
|
return virt$set_thread_name(arg1, arg2, arg3);
|
|
case SC_setgid:
|
|
return virt$setgid(arg2);
|
|
case SC_setgroups:
|
|
return virt$setgroups(arg1, arg2);
|
|
case SC_setpgid:
|
|
return virt$setpgid(arg1, arg2);
|
|
case SC_setsid:
|
|
return virt$setsid();
|
|
case SC_setsockopt:
|
|
return virt$setsockopt(arg1);
|
|
case SC_setuid:
|
|
return virt$setuid(arg1);
|
|
case SC_shutdown:
|
|
return virt$shutdown(arg1, arg2);
|
|
case SC_sigaction:
|
|
return virt$sigaction(arg1, arg2, arg3);
|
|
case SC_sigprocmask:
|
|
return virt$sigprocmask(arg1, arg2, arg3);
|
|
case SC_sigreturn:
|
|
return virt$sigreturn();
|
|
case SC_socket:
|
|
return virt$socket(arg1, arg2, arg3);
|
|
case SC_stat:
|
|
return virt$stat(arg1);
|
|
case SC_symlink:
|
|
return virt$symlink(arg1);
|
|
case SC_sync:
|
|
virt$sync();
|
|
return 0;
|
|
case SC_sysconf:
|
|
return virt$sysconf(arg1);
|
|
case SC_umask:
|
|
return virt$umask(arg1);
|
|
case SC_uname:
|
|
return virt$uname(arg1);
|
|
case SC_unlink:
|
|
return virt$unlink(arg1, arg2);
|
|
case SC_unveil:
|
|
return virt$unveil(arg1);
|
|
case SC_waitid:
|
|
return virt$waitid(arg1);
|
|
case SC_write:
|
|
return virt$write(arg1, arg2, arg3);
|
|
default:
|
|
reportln("\n=={}== \033[31;1mUnimplemented syscall: {}\033[0m, {:p}"sv, getpid(), Syscall::to_string((Syscall::Function)function), function);
|
|
dump_backtrace();
|
|
TODO();
|
|
}
|
|
}
|
|
|
|
int Emulator::virt$anon_create(size_t size, int options)
|
|
{
|
|
return syscall(SC_anon_create, size, options);
|
|
}
|
|
|
|
int Emulator::virt$sendfd(int socket, int fd)
|
|
{
|
|
return syscall(SC_sendfd, socket, fd);
|
|
}
|
|
|
|
int Emulator::virt$recvfd(int socket, int options)
|
|
{
|
|
return syscall(SC_recvfd, socket, options);
|
|
}
|
|
|
|
int Emulator::virt$profiling_enable(pid_t pid)
|
|
{
|
|
return syscall(SC_profiling_enable, pid);
|
|
}
|
|
|
|
int Emulator::virt$profiling_disable(pid_t pid)
|
|
{
|
|
return syscall(SC_profiling_disable, pid);
|
|
}
|
|
|
|
FlatPtr Emulator::virt$perf_event(int event, FlatPtr arg1, FlatPtr arg2)
|
|
{
|
|
if (event == PERF_EVENT_SIGNPOST) {
|
|
if (is_profiling()) {
|
|
if (profiler_string_id_map().size() > arg1)
|
|
emit_profile_event(profile_stream(), "signpost"sv, DeprecatedString::formatted("\"arg1\": {}, \"arg2\": {}", arg1, arg2));
|
|
syscall(SC_perf_event, PERF_EVENT_SIGNPOST, profiler_string_id_map().at(arg1), arg2);
|
|
} else {
|
|
syscall(SC_perf_event, PERF_EVENT_SIGNPOST, arg1, arg2);
|
|
}
|
|
return 0;
|
|
}
|
|
return -ENOSYS;
|
|
}
|
|
|
|
FlatPtr Emulator::virt$perf_register_string(FlatPtr string, size_t size)
|
|
{
|
|
char* buffer = (char*)alloca(size + 4);
|
|
// FIXME: not nice, but works
|
|
__builtin_memcpy(buffer, "UE: ", 4);
|
|
mmu().copy_from_vm((buffer + 4), string, size);
|
|
auto ret = (int)syscall(SC_perf_register_string, buffer, size + 4);
|
|
|
|
if (ret >= 0 && is_profiling()) {
|
|
profiler_strings().append(make<DeprecatedString>(StringView { buffer + 4, size }));
|
|
profiler_string_id_map().append(ret);
|
|
ret = profiler_string_id_map().size() - 1;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int Emulator::virt$disown(pid_t pid)
|
|
{
|
|
return syscall(SC_disown, pid);
|
|
}
|
|
|
|
int Emulator::virt$purge(int mode)
|
|
{
|
|
return syscall(SC_purge, mode);
|
|
}
|
|
|
|
int Emulator::virt$fstat(int fd, FlatPtr statbuf)
|
|
{
|
|
struct stat local_statbuf;
|
|
int rc = syscall(SC_fstat, fd, &local_statbuf);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(statbuf, &local_statbuf, sizeof(local_statbuf));
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$close(int fd)
|
|
{
|
|
return syscall(SC_close, fd);
|
|
}
|
|
|
|
int Emulator::virt$mkdir(FlatPtr path, size_t path_length, mode_t mode)
|
|
{
|
|
auto buffer = mmu().copy_buffer_from_vm(path, path_length);
|
|
return syscall(SC_mkdir, buffer.data(), buffer.size(), mode);
|
|
}
|
|
|
|
int Emulator::virt$rmdir(FlatPtr path, size_t path_length)
|
|
{
|
|
auto buffer = mmu().copy_buffer_from_vm(path, path_length);
|
|
return syscall(SC_rmdir, buffer.data(), buffer.size());
|
|
}
|
|
|
|
int Emulator::virt$unlink(FlatPtr path, size_t path_length)
|
|
{
|
|
auto buffer = mmu().copy_buffer_from_vm(path, path_length);
|
|
return syscall(SC_unlink, AT_FDCWD, buffer.data(), buffer.size(), 0);
|
|
}
|
|
|
|
int Emulator::virt$symlink(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_symlink_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto target = mmu().copy_buffer_from_vm((FlatPtr)params.target.characters, params.target.length);
|
|
params.target.characters = (char const*)target.data();
|
|
params.target.length = target.size();
|
|
|
|
auto link = mmu().copy_buffer_from_vm((FlatPtr)params.linkpath.characters, params.linkpath.length);
|
|
params.linkpath.characters = (char const*)link.data();
|
|
params.linkpath.length = link.size();
|
|
|
|
return syscall(SC_symlink, ¶ms);
|
|
}
|
|
|
|
int Emulator::virt$rename(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_rename_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto new_path = mmu().copy_buffer_from_vm((FlatPtr)params.new_path.characters, params.new_path.length);
|
|
params.new_path.characters = (char const*)new_path.data();
|
|
params.new_path.length = new_path.size();
|
|
|
|
auto old_path = mmu().copy_buffer_from_vm((FlatPtr)params.old_path.characters, params.old_path.length);
|
|
params.old_path.characters = (char const*)old_path.data();
|
|
params.old_path.length = old_path.size();
|
|
|
|
return syscall(SC_rename, ¶ms);
|
|
}
|
|
|
|
int Emulator::virt$dbgputstr(FlatPtr characters, int length)
|
|
{
|
|
auto buffer = mmu().copy_buffer_from_vm(characters, length);
|
|
dbgputstr((char const*)buffer.data(), buffer.size());
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$chmod(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_chmod_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto path = mmu().copy_buffer_from_vm((FlatPtr)params.path.characters, params.path.length);
|
|
params.path.characters = (char const*)path.data();
|
|
params.path.length = path.size();
|
|
return syscall(SC_chmod, ¶ms);
|
|
}
|
|
|
|
int Emulator::virt$chown(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_chown_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto path = mmu().copy_buffer_from_vm((FlatPtr)params.path.characters, params.path.length);
|
|
params.path.characters = (char const*)path.data();
|
|
params.path.length = path.size();
|
|
|
|
return syscall(SC_chown, ¶ms);
|
|
}
|
|
|
|
int Emulator::virt$fchmod(int fd, mode_t mode)
|
|
{
|
|
return syscall(SC_fchmod, fd, mode);
|
|
}
|
|
|
|
int Emulator::virt$fchown(int fd, uid_t uid, gid_t gid)
|
|
{
|
|
return syscall(SC_fchown, fd, uid, gid);
|
|
}
|
|
|
|
int Emulator::virt$setsockopt(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_setsockopt_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
if (params.option == SO_RCVTIMEO || params.option == SO_TIMESTAMP) {
|
|
auto host_value_buffer_result = ByteBuffer::create_zeroed(params.value_size);
|
|
if (host_value_buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& host_value_buffer = host_value_buffer_result.value();
|
|
mmu().copy_from_vm(host_value_buffer.data(), (FlatPtr)params.value, params.value_size);
|
|
int rc = setsockopt(params.sockfd, params.level, params.option, host_value_buffer.data(), host_value_buffer.size());
|
|
if (rc < 0)
|
|
return -errno;
|
|
return rc;
|
|
}
|
|
|
|
if (params.option == SO_BINDTODEVICE) {
|
|
auto ifname = mmu().copy_buffer_from_vm((FlatPtr)params.value, params.value_size);
|
|
params.value = ifname.data();
|
|
params.value_size = ifname.size();
|
|
return syscall(SC_setsockopt, ¶ms);
|
|
}
|
|
|
|
TODO();
|
|
}
|
|
|
|
int Emulator::virt$get_stack_bounds(FlatPtr base, FlatPtr size)
|
|
{
|
|
auto* region = mmu().find_region({ m_cpu->ss(), m_cpu->esp().value() });
|
|
FlatPtr b = region->base();
|
|
size_t s = region->size();
|
|
mmu().copy_to_vm(base, &b, sizeof(b));
|
|
mmu().copy_to_vm(size, &s, sizeof(s));
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$ftruncate(int fd, FlatPtr length_addr)
|
|
{
|
|
off_t length;
|
|
mmu().copy_from_vm(&length, length_addr, sizeof(off_t));
|
|
return syscall(SC_ftruncate, fd, &length);
|
|
}
|
|
|
|
int Emulator::virt$uname(FlatPtr params_addr)
|
|
{
|
|
struct utsname local_uname;
|
|
auto rc = syscall(SC_uname, &local_uname);
|
|
mmu().copy_to_vm(params_addr, &local_uname, sizeof(local_uname));
|
|
return rc;
|
|
}
|
|
|
|
mode_t Emulator::virt$umask(mode_t mask)
|
|
{
|
|
return syscall(SC_umask, mask);
|
|
}
|
|
|
|
int Emulator::virt$accept4(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_accept4_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
sockaddr_storage addr = {};
|
|
socklen_t addrlen;
|
|
mmu().copy_from_vm(&addrlen, (FlatPtr)params.addrlen, sizeof(socklen_t));
|
|
VERIFY(addrlen <= sizeof(addr));
|
|
int rc = accept4(params.sockfd, (sockaddr*)&addr, &addrlen, params.flags);
|
|
if (rc == 0) {
|
|
mmu().copy_to_vm((FlatPtr)params.addr, &addr, addrlen);
|
|
mmu().copy_to_vm((FlatPtr)params.addrlen, &addrlen, sizeof(socklen_t));
|
|
}
|
|
return rc < 0 ? -errno : rc;
|
|
}
|
|
|
|
int Emulator::virt$bind(int sockfd, FlatPtr address, socklen_t address_length)
|
|
{
|
|
auto buffer = mmu().copy_buffer_from_vm(address, address_length);
|
|
return syscall(SC_bind, sockfd, buffer.data(), buffer.size());
|
|
}
|
|
|
|
int Emulator::virt$connect(int sockfd, FlatPtr address, socklen_t address_size)
|
|
{
|
|
auto buffer = mmu().copy_buffer_from_vm(address, address_size);
|
|
return syscall(SC_connect, sockfd, buffer.data(), buffer.size());
|
|
}
|
|
|
|
int Emulator::virt$shutdown(int sockfd, int how)
|
|
{
|
|
return syscall(SC_shutdown, sockfd, how);
|
|
}
|
|
|
|
int Emulator::virt$listen(int fd, int backlog)
|
|
{
|
|
return syscall(SC_listen, fd, backlog);
|
|
}
|
|
|
|
int Emulator::virt$kill(pid_t pid, int signal)
|
|
{
|
|
return syscall(SC_kill, pid, signal);
|
|
}
|
|
|
|
int Emulator::virt$killpg(int pgrp, int sig)
|
|
{
|
|
return syscall(SC_killpg, pgrp, sig);
|
|
}
|
|
|
|
int Emulator::virt$clock_gettime(int clockid, FlatPtr timespec)
|
|
{
|
|
struct timespec host_timespec;
|
|
int rc = syscall(SC_clock_gettime, clockid, &host_timespec);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(timespec, &host_timespec, sizeof(host_timespec));
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$clock_settime(uint32_t clock_id, FlatPtr user_ts)
|
|
{
|
|
struct timespec user_timespec;
|
|
mmu().copy_from_vm(&user_timespec, user_ts, sizeof(user_timespec));
|
|
int rc = syscall(SC_clock_settime, clock_id, &user_timespec);
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$set_mmap_name(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_set_mmap_name_params params {};
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
auto name = mmu().copy_buffer_from_vm((FlatPtr)params.name.characters, params.name.length);
|
|
|
|
auto* region = mmu().find_region({ 0x23, (FlatPtr)params.addr });
|
|
if (!region || !is<MmapRegion>(*region))
|
|
return -EINVAL;
|
|
static_cast<MmapRegion&>(*region).set_name(DeprecatedString::copy(name));
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$lseek(int fd, FlatPtr offset_addr, int whence)
|
|
{
|
|
off_t offset;
|
|
mmu().copy_from_vm(&offset, offset_addr, sizeof(off_t));
|
|
auto rc = syscall(SC_lseek, fd, &offset, whence);
|
|
mmu().copy_to_vm(offset_addr, &offset, sizeof(off_t));
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$socket(int domain, int type, int protocol)
|
|
{
|
|
return syscall(SC_socket, domain, type, protocol);
|
|
}
|
|
|
|
int Emulator::virt$recvmsg(int sockfd, FlatPtr msg_addr, int flags)
|
|
{
|
|
msghdr mmu_msg;
|
|
mmu().copy_from_vm(&mmu_msg, msg_addr, sizeof(mmu_msg));
|
|
|
|
Vector<iovec, 1> mmu_iovs;
|
|
mmu_iovs.resize(mmu_msg.msg_iovlen);
|
|
mmu().copy_from_vm(mmu_iovs.data(), (FlatPtr)mmu_msg.msg_iov, mmu_msg.msg_iovlen * sizeof(iovec));
|
|
Vector<ByteBuffer, 1> buffers;
|
|
Vector<iovec, 1> iovs;
|
|
for (auto const& iov : mmu_iovs) {
|
|
auto buffer_result = ByteBuffer::create_uninitialized(iov.iov_len);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
buffers.append(buffer_result.release_value());
|
|
iovs.append({ buffers.last().data(), buffers.last().size() });
|
|
}
|
|
|
|
ByteBuffer control_buffer;
|
|
if (mmu_msg.msg_control) {
|
|
auto buffer_result = ByteBuffer::create_uninitialized(mmu_msg.msg_controllen);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
control_buffer = buffer_result.release_value();
|
|
}
|
|
|
|
sockaddr_storage addr;
|
|
msghdr msg = { &addr, sizeof(addr), iovs.data(), (int)iovs.size(), mmu_msg.msg_control ? control_buffer.data() : nullptr, mmu_msg.msg_controllen, mmu_msg.msg_flags };
|
|
int rc = recvmsg(sockfd, &msg, flags);
|
|
if (rc < 0)
|
|
return -errno;
|
|
|
|
for (size_t i = 0; i < buffers.size(); ++i)
|
|
mmu().copy_to_vm((FlatPtr)mmu_iovs[i].iov_base, buffers[i].data(), mmu_iovs[i].iov_len);
|
|
|
|
if (mmu_msg.msg_name)
|
|
mmu().copy_to_vm((FlatPtr)mmu_msg.msg_name, &addr, min(sizeof(addr), (size_t)mmu_msg.msg_namelen));
|
|
if (mmu_msg.msg_control)
|
|
mmu().copy_to_vm((FlatPtr)mmu_msg.msg_control, control_buffer.data(), min(mmu_msg.msg_controllen, msg.msg_controllen));
|
|
mmu_msg.msg_namelen = msg.msg_namelen;
|
|
mmu_msg.msg_controllen = msg.msg_controllen;
|
|
mmu_msg.msg_flags = msg.msg_flags;
|
|
mmu().copy_to_vm(msg_addr, &mmu_msg, sizeof(mmu_msg));
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$sendmsg(int sockfd, FlatPtr msg_addr, int flags)
|
|
{
|
|
msghdr mmu_msg;
|
|
mmu().copy_from_vm(&mmu_msg, msg_addr, sizeof(mmu_msg));
|
|
|
|
Vector<iovec, 1> iovs;
|
|
iovs.resize(mmu_msg.msg_iovlen);
|
|
mmu().copy_from_vm(iovs.data(), (FlatPtr)mmu_msg.msg_iov, mmu_msg.msg_iovlen * sizeof(iovec));
|
|
Vector<ByteBuffer, 1> buffers;
|
|
for (auto& iov : iovs) {
|
|
buffers.append(mmu().copy_buffer_from_vm((FlatPtr)iov.iov_base, iov.iov_len));
|
|
iov = { buffers.last().data(), buffers.last().size() };
|
|
}
|
|
|
|
ByteBuffer control_buffer;
|
|
if (mmu_msg.msg_control) {
|
|
auto buffer_result = ByteBuffer::create_uninitialized(mmu_msg.msg_controllen);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
control_buffer = buffer_result.release_value();
|
|
}
|
|
|
|
sockaddr_storage address;
|
|
socklen_t address_length = 0;
|
|
if (mmu_msg.msg_name) {
|
|
address_length = min(sizeof(address), (size_t)mmu_msg.msg_namelen);
|
|
mmu().copy_from_vm(&address, (FlatPtr)mmu_msg.msg_name, address_length);
|
|
}
|
|
|
|
msghdr msg = { mmu_msg.msg_name ? &address : nullptr, address_length, iovs.data(), (int)iovs.size(), mmu_msg.msg_control ? control_buffer.data() : nullptr, mmu_msg.msg_controllen, mmu_msg.msg_flags };
|
|
return sendmsg(sockfd, &msg, flags);
|
|
}
|
|
|
|
int Emulator::virt$getsockopt(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_getsockopt_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
if (params.option == SO_PEERCRED) {
|
|
struct ucred creds = {};
|
|
socklen_t creds_size = sizeof(creds);
|
|
int rc = getsockopt(params.sockfd, params.level, SO_PEERCRED, &creds, &creds_size);
|
|
if (rc < 0)
|
|
return -errno;
|
|
// FIXME: Check params.value_size
|
|
mmu().copy_to_vm((FlatPtr)params.value, &creds, sizeof(creds));
|
|
return rc;
|
|
}
|
|
if (params.option == SO_ERROR) {
|
|
int so_error;
|
|
socklen_t so_error_len = sizeof(so_error);
|
|
int rc = getsockopt(params.sockfd, params.level, SO_ERROR, &so_error, &so_error_len);
|
|
if (rc < 0)
|
|
return -errno;
|
|
// FIXME: Check params.value_size
|
|
mmu().copy_to_vm((FlatPtr)params.value, &so_error, sizeof(so_error));
|
|
return rc;
|
|
}
|
|
|
|
dbgln("Not implemented socket param: {}", params.option);
|
|
TODO();
|
|
}
|
|
|
|
int Emulator::virt$getsockname(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_getsockname_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
sockaddr_storage addr = {};
|
|
socklen_t addrlen;
|
|
mmu().copy_from_vm(&addrlen, (FlatPtr)params.addrlen, sizeof(socklen_t));
|
|
VERIFY(addrlen <= sizeof(addr));
|
|
auto rc = getsockname(params.sockfd, (sockaddr*)&addr, &addrlen);
|
|
if (rc == 0) {
|
|
mmu().copy_to_vm((FlatPtr)params.addr, &addr, sizeof(addr));
|
|
mmu().copy_to_vm((FlatPtr)params.addrlen, &addrlen, sizeof(addrlen));
|
|
}
|
|
return rc < 0 ? -errno : rc;
|
|
}
|
|
|
|
int Emulator::virt$getpeername(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_getpeername_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
sockaddr_storage addr = {};
|
|
socklen_t addrlen;
|
|
mmu().copy_from_vm(&addrlen, (FlatPtr)params.addrlen, sizeof(socklen_t));
|
|
VERIFY(addrlen <= sizeof(addr));
|
|
auto rc = getpeername(params.sockfd, (sockaddr*)&addr, &addrlen);
|
|
if (rc == 0) {
|
|
mmu().copy_to_vm((FlatPtr)params.addr, &addr, sizeof(addr));
|
|
mmu().copy_to_vm((FlatPtr)params.addrlen, &addrlen, sizeof(addrlen));
|
|
}
|
|
return rc < 0 ? -errno : rc;
|
|
}
|
|
|
|
int Emulator::virt$getgroups(ssize_t count, FlatPtr groups)
|
|
{
|
|
if (!count)
|
|
return syscall(SC_getgroups, 0, nullptr);
|
|
|
|
auto buffer_result = ByteBuffer::create_uninitialized(count * sizeof(gid_t));
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& buffer = buffer_result.value();
|
|
int rc = syscall(SC_getgroups, count, buffer.data());
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(groups, buffer.data(), buffer.size());
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$setgroups(ssize_t count, FlatPtr groups)
|
|
{
|
|
if (!count)
|
|
return syscall(SC_setgroups, 0, nullptr);
|
|
|
|
auto buffer = mmu().copy_buffer_from_vm(groups, count * sizeof(gid_t));
|
|
return syscall(SC_setgroups, count, buffer.data());
|
|
}
|
|
|
|
u32 Emulator::virt$fcntl(int fd, int cmd, u32 arg)
|
|
{
|
|
switch (cmd) {
|
|
case F_DUPFD:
|
|
case F_GETFD:
|
|
case F_SETFD:
|
|
case F_GETFL:
|
|
case F_SETFL:
|
|
case F_ISTTY:
|
|
break;
|
|
default:
|
|
dbgln("Invalid fcntl cmd: {}", cmd);
|
|
}
|
|
|
|
return syscall(SC_fcntl, fd, cmd, arg);
|
|
}
|
|
|
|
u32 Emulator::virt$open(u32 params_addr)
|
|
{
|
|
Syscall::SC_open_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto path = mmu().copy_buffer_from_vm((FlatPtr)params.path.characters, params.path.length);
|
|
|
|
Syscall::SC_open_params host_params {};
|
|
host_params.dirfd = params.dirfd;
|
|
host_params.mode = params.mode;
|
|
host_params.options = params.options;
|
|
host_params.path.characters = (char const*)path.data();
|
|
host_params.path.length = path.size();
|
|
|
|
return syscall(SC_open, &host_params);
|
|
}
|
|
|
|
int Emulator::virt$pipe(FlatPtr vm_pipefd, int flags)
|
|
{
|
|
int pipefd[2];
|
|
int rc = syscall(SC_pipe, pipefd, flags);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(vm_pipefd, pipefd, sizeof(pipefd));
|
|
return rc;
|
|
}
|
|
|
|
static void round_to_page_size(FlatPtr& address, size_t& size)
|
|
{
|
|
auto new_end = round_up_to_power_of_two(address + size, PAGE_SIZE);
|
|
address &= ~(PAGE_SIZE - 1);
|
|
size = new_end - address;
|
|
}
|
|
|
|
u32 Emulator::virt$munmap(FlatPtr address, size_t size)
|
|
{
|
|
if (is_profiling())
|
|
emit_profile_event(profile_stream(), "munmap"sv, DeprecatedString::formatted("\"ptr\": {}, \"size\": {}", address, size));
|
|
round_to_page_size(address, size);
|
|
Vector<Region*, 4> marked_for_deletion;
|
|
bool has_non_mmap_region = false;
|
|
mmu().for_regions_in({ 0x23, address }, size, [&](Region* region) {
|
|
if (region) {
|
|
if (!is<MmapRegion>(*region)) {
|
|
has_non_mmap_region = true;
|
|
return IterationDecision::Break;
|
|
}
|
|
marked_for_deletion.append(region);
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
if (has_non_mmap_region)
|
|
return -EINVAL;
|
|
|
|
for (Region* region : marked_for_deletion) {
|
|
m_range_allocator.deallocate(region->range());
|
|
mmu().remove_region(*region);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
u32 Emulator::virt$mmap(u32 params_addr)
|
|
{
|
|
Syscall::SC_mmap_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
params.alignment = params.alignment ? params.alignment : PAGE_SIZE;
|
|
|
|
if (params.size == 0)
|
|
return -EINVAL;
|
|
|
|
u32 requested_size = round_up_to_power_of_two(params.size, PAGE_SIZE);
|
|
FlatPtr final_address;
|
|
|
|
Optional<Range> result;
|
|
if (params.flags & MAP_RANDOMIZED) {
|
|
result = m_range_allocator.allocate_randomized(requested_size, params.alignment);
|
|
} else if (params.flags & MAP_FIXED || params.flags & MAP_FIXED_NOREPLACE) {
|
|
if (params.addr) {
|
|
// If MAP_FIXED is specified, existing mappings that intersect the requested range are removed.
|
|
if (params.flags & MAP_FIXED)
|
|
virt$munmap((FlatPtr)params.addr, requested_size);
|
|
result = m_range_allocator.allocate_specific(VirtualAddress { params.addr }, requested_size);
|
|
} else {
|
|
// mmap(nullptr, …, MAP_FIXED) is technically okay, but tends to be a bug.
|
|
// Therefore, refuse to be helpful.
|
|
reportln("\n=={}== \033[31;1mTried to mmap at nullptr with MAP_FIXED.\033[0m, {:#x} bytes."sv, getpid(), params.size);
|
|
dump_backtrace();
|
|
}
|
|
} else {
|
|
result = m_range_allocator.allocate_anywhere(requested_size, params.alignment);
|
|
}
|
|
if (!result.has_value())
|
|
return -ENOMEM;
|
|
final_address = result.value().base().get();
|
|
auto final_size = result.value().size();
|
|
|
|
DeprecatedString name_str;
|
|
if (params.name.characters) {
|
|
auto buffer_result = ByteBuffer::create_uninitialized(params.name.length);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& name = buffer_result.value();
|
|
mmu().copy_from_vm(name.data(), (FlatPtr)params.name.characters, params.name.length);
|
|
name_str = { name.data(), name.size() };
|
|
}
|
|
|
|
if (is_profiling())
|
|
emit_profile_event(profile_stream(), "mmap"sv, DeprecatedString::formatted(R"("ptr": {}, "size": {}, "name": "{}")", final_address, final_size, name_str));
|
|
|
|
if (params.flags & MAP_ANONYMOUS) {
|
|
mmu().add_region(MmapRegion::create_anonymous(final_address, final_size, params.prot, move(name_str)));
|
|
} else {
|
|
auto region = MmapRegion::create_file_backed(final_address, final_size, params.prot, params.flags, params.fd, params.offset, move(name_str));
|
|
if (region->name() == "libsystem.so: .text" && !m_libsystem_start) {
|
|
m_libsystem_start = final_address;
|
|
m_libsystem_end = final_address + final_size;
|
|
}
|
|
mmu().add_region(move(region));
|
|
}
|
|
|
|
return final_address;
|
|
}
|
|
|
|
FlatPtr Emulator::virt$mremap(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_mremap_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
// FIXME: Support regions that have been split in the past (e.g. due to mprotect or munmap).
|
|
if (auto* region = mmu().find_region({ m_cpu->ds(), (FlatPtr)params.old_address })) {
|
|
if (!is<MmapRegion>(*region))
|
|
return -EINVAL;
|
|
VERIFY(region->size() == params.old_size);
|
|
auto& mmap_region = *(MmapRegion*)region;
|
|
auto* ptr = mremap(mmap_region.data(), mmap_region.size(), mmap_region.size(), params.flags);
|
|
if (ptr == MAP_FAILED)
|
|
return -errno;
|
|
return (FlatPtr)ptr;
|
|
}
|
|
return -EINVAL;
|
|
}
|
|
|
|
u32 Emulator::virt$mount(u32 params_addr)
|
|
{
|
|
Syscall::SC_mount_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
auto target = mmu().copy_buffer_from_vm((FlatPtr)params.target.characters, params.target.length);
|
|
auto fs_path = mmu().copy_buffer_from_vm((FlatPtr)params.fs_type.characters, params.fs_type.length);
|
|
params.fs_type.characters = (char*)fs_path.data();
|
|
params.fs_type.length = fs_path.size();
|
|
params.target.characters = (char*)target.data();
|
|
params.target.length = target.size();
|
|
|
|
return syscall(SC_mount, ¶ms);
|
|
}
|
|
|
|
u32 Emulator::virt$gettid()
|
|
{
|
|
return gettid();
|
|
}
|
|
|
|
u32 Emulator::virt$getpid()
|
|
{
|
|
return getpid();
|
|
}
|
|
|
|
pid_t Emulator::virt$getppid()
|
|
{
|
|
return getppid();
|
|
}
|
|
|
|
u32 Emulator::virt$pledge(u32)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
u32 Emulator::virt$unveil(u32)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
u32 Emulator::virt$mprotect(FlatPtr base, size_t size, int prot)
|
|
{
|
|
round_to_page_size(base, size);
|
|
bool has_non_mmapped_region = false;
|
|
|
|
mmu().for_regions_in({ 0x23, base }, size, [&](Region* region) {
|
|
if (region) {
|
|
if (!is<MmapRegion>(*region)) {
|
|
has_non_mmapped_region = true;
|
|
return IterationDecision::Break;
|
|
}
|
|
auto& mmap_region = *(MmapRegion*)region;
|
|
mmap_region.set_prot(prot);
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
if (has_non_mmapped_region)
|
|
return -EINVAL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
u32 Emulator::virt$madvise(FlatPtr, size_t, int)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
uid_t Emulator::virt$getuid()
|
|
{
|
|
return getuid();
|
|
}
|
|
|
|
uid_t Emulator::virt$geteuid()
|
|
{
|
|
return geteuid();
|
|
}
|
|
|
|
gid_t Emulator::virt$getgid()
|
|
{
|
|
return getgid();
|
|
}
|
|
|
|
gid_t Emulator::virt$getegid()
|
|
{
|
|
return getegid();
|
|
}
|
|
|
|
int Emulator::virt$setuid(uid_t uid)
|
|
{
|
|
return syscall(SC_setuid, uid);
|
|
}
|
|
|
|
int Emulator::virt$setgid(gid_t gid)
|
|
{
|
|
return syscall(SC_setgid, gid);
|
|
}
|
|
|
|
u32 Emulator::virt$write(int fd, FlatPtr data, ssize_t size)
|
|
{
|
|
if (size < 0)
|
|
return -EINVAL;
|
|
auto buffer = mmu().copy_buffer_from_vm(data, size);
|
|
return syscall(SC_write, fd, buffer.data(), buffer.size());
|
|
}
|
|
|
|
u32 Emulator::virt$read(int fd, FlatPtr buffer, ssize_t size)
|
|
{
|
|
if (size < 0)
|
|
return -EINVAL;
|
|
auto buffer_result = ByteBuffer::create_uninitialized(size);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& local_buffer = buffer_result.value();
|
|
int nread = syscall(SC_read, fd, local_buffer.data(), local_buffer.size());
|
|
if (nread < 0) {
|
|
if (nread == -EPERM) {
|
|
dump_backtrace();
|
|
TODO();
|
|
}
|
|
return nread;
|
|
}
|
|
mmu().copy_to_vm(buffer, local_buffer.data(), local_buffer.size());
|
|
return nread;
|
|
}
|
|
|
|
void Emulator::virt$sync()
|
|
{
|
|
syscall(SC_sync);
|
|
}
|
|
|
|
void Emulator::virt$exit(int status)
|
|
{
|
|
reportln("\n=={}== \033[33;1mSyscall: exit({})\033[0m, shutting down!"sv, getpid(), status);
|
|
m_exit_status = status;
|
|
m_shutdown = true;
|
|
}
|
|
|
|
ssize_t Emulator::virt$getrandom(FlatPtr buffer, size_t buffer_size, unsigned int flags)
|
|
{
|
|
auto buffer_result = ByteBuffer::create_uninitialized(buffer_size);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& host_buffer = buffer_result.value();
|
|
int rc = syscall(SC_getrandom, host_buffer.data(), host_buffer.size(), flags);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(buffer, host_buffer.data(), host_buffer.size());
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$get_dir_entries(int fd, FlatPtr buffer, ssize_t size)
|
|
{
|
|
auto buffer_result = ByteBuffer::create_uninitialized(size);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& host_buffer = buffer_result.value();
|
|
int rc = syscall(SC_get_dir_entries, fd, host_buffer.data(), host_buffer.size());
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(buffer, host_buffer.data(), host_buffer.size());
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$ioctl([[maybe_unused]] int fd, unsigned request, [[maybe_unused]] FlatPtr arg)
|
|
{
|
|
switch (request) {
|
|
case TIOCGWINSZ: {
|
|
struct winsize ws;
|
|
int rc = syscall(SC_ioctl, fd, TIOCGWINSZ, &ws);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(arg, &ws, sizeof(winsize));
|
|
return 0;
|
|
}
|
|
case TIOCSWINSZ: {
|
|
struct winsize ws;
|
|
mmu().copy_from_vm(&ws, arg, sizeof(winsize));
|
|
return syscall(SC_ioctl, fd, request, &ws);
|
|
}
|
|
case TIOCGPGRP: {
|
|
pid_t pgid;
|
|
auto rc = syscall(SC_ioctl, fd, request, &pgid);
|
|
mmu().copy_to_vm(arg, &pgid, sizeof(pgid));
|
|
return rc;
|
|
}
|
|
case TIOCSPGRP:
|
|
return syscall(SC_ioctl, fd, request, arg);
|
|
case TCGETS: {
|
|
struct termios termios;
|
|
int rc = syscall(SC_ioctl, fd, request, &termios);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(arg, &termios, sizeof(termios));
|
|
return rc;
|
|
}
|
|
case TCSETS:
|
|
case TCSETSF:
|
|
case TCSETSW: {
|
|
struct termios termios;
|
|
mmu().copy_from_vm(&termios, arg, sizeof(termios));
|
|
return syscall(SC_ioctl, fd, request, &termios);
|
|
}
|
|
case TCFLSH:
|
|
return syscall(SC_ioctl, fd, request, arg);
|
|
case TIOCNOTTY:
|
|
case TIOCSCTTY:
|
|
return syscall(SC_ioctl, fd, request, 0);
|
|
case TIOCSTI:
|
|
return -EIO;
|
|
case GRAPHICS_IOCTL_GET_PROPERTIES: {
|
|
size_t size = 0;
|
|
auto rc = syscall(SC_ioctl, fd, request, &size);
|
|
mmu().copy_to_vm(arg, &size, sizeof(size));
|
|
return rc;
|
|
}
|
|
case GRAPHICS_IOCTL_SET_HEAD_VERTICAL_OFFSET_BUFFER:
|
|
return syscall(SC_ioctl, fd, request, arg);
|
|
case FIONBIO: {
|
|
int enabled;
|
|
mmu().copy_from_vm(&enabled, arg, sizeof(int));
|
|
return syscall(SC_ioctl, fd, request, &enabled);
|
|
}
|
|
default:
|
|
reportln("Unsupported ioctl: {}"sv, request);
|
|
dump_backtrace();
|
|
TODO();
|
|
}
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
|
|
int Emulator::virt$emuctl(FlatPtr arg1, FlatPtr arg2, FlatPtr arg3)
|
|
{
|
|
auto* tracer = malloc_tracer();
|
|
if (arg1 <= 4 && !tracer)
|
|
return 0;
|
|
switch (arg1) {
|
|
case 1:
|
|
tracer->target_did_malloc({}, arg3, arg2);
|
|
return 0;
|
|
case 2:
|
|
tracer->target_did_free({}, arg2);
|
|
return 0;
|
|
case 3:
|
|
tracer->target_did_realloc({}, arg3, arg2);
|
|
return 0;
|
|
case 4:
|
|
tracer->target_did_change_chunk_size({}, arg3, arg2);
|
|
return 0;
|
|
case 5: // mark ROI start
|
|
if (is_in_region_of_interest())
|
|
return -EINVAL;
|
|
m_is_in_region_of_interest = true;
|
|
return 0;
|
|
case 6: // mark ROI end
|
|
m_is_in_region_of_interest = false;
|
|
return 0;
|
|
case 7:
|
|
m_is_memory_auditing_suppressed = true;
|
|
return 0;
|
|
case 8:
|
|
m_is_memory_auditing_suppressed = false;
|
|
return 0;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
int Emulator::virt$fork()
|
|
{
|
|
int rc = fork();
|
|
if (rc < 0)
|
|
return -errno;
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$execve(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_execve_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto path = DeprecatedString::copy(mmu().copy_buffer_from_vm((FlatPtr)params.path.characters, params.path.length));
|
|
Vector<DeprecatedString> arguments;
|
|
Vector<DeprecatedString> environment;
|
|
|
|
auto copy_string_list = [this](auto& output_vector, auto& string_list) {
|
|
for (size_t i = 0; i < string_list.length; ++i) {
|
|
Syscall::StringArgument string;
|
|
mmu().copy_from_vm(&string, (FlatPtr)&string_list.strings[i], sizeof(string));
|
|
output_vector.append(DeprecatedString::copy(mmu().copy_buffer_from_vm((FlatPtr)string.characters, string.length)));
|
|
}
|
|
};
|
|
|
|
copy_string_list(arguments, params.arguments);
|
|
copy_string_list(environment, params.environment);
|
|
|
|
reportln("\n=={}== \033[33;1mSyscall:\033[0m execve"sv, getpid());
|
|
reportln("=={}== @ {}"sv, getpid(), path);
|
|
for (auto& argument : arguments)
|
|
reportln("=={}== - {}"sv, getpid(), argument);
|
|
|
|
if (access(path.characters(), X_OK) < 0) {
|
|
if (errno == ENOENT || errno == EACCES)
|
|
return -errno;
|
|
}
|
|
|
|
Vector<char*> argv;
|
|
Vector<char*> envp;
|
|
|
|
argv.append(const_cast<char*>("/bin/UserspaceEmulator"));
|
|
if (g_report_to_debug)
|
|
argv.append(const_cast<char*>("--report-to-debug"));
|
|
argv.append(const_cast<char*>("--"));
|
|
argv.append(const_cast<char*>(path.characters()));
|
|
|
|
auto create_string_vector = [](auto& output_vector, auto& input_vector) {
|
|
for (auto& string : input_vector)
|
|
output_vector.append(const_cast<char*>(string.characters()));
|
|
output_vector.append(nullptr);
|
|
};
|
|
|
|
create_string_vector(argv, arguments);
|
|
create_string_vector(envp, environment);
|
|
|
|
// Yoink duplicated program name.
|
|
argv.remove(3 + (g_report_to_debug ? 1 : 0));
|
|
|
|
return execve(argv[0], (char* const*)argv.data(), (char* const*)envp.data());
|
|
}
|
|
|
|
int Emulator::virt$stat(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_stat_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto path = DeprecatedString::copy(mmu().copy_buffer_from_vm((FlatPtr)params.path.characters, params.path.length));
|
|
struct stat host_statbuf;
|
|
int rc;
|
|
if (params.follow_symlinks)
|
|
rc = stat(path.characters(), &host_statbuf);
|
|
else
|
|
rc = lstat(path.characters(), &host_statbuf);
|
|
if (rc < 0)
|
|
return -errno;
|
|
mmu().copy_to_vm((FlatPtr)params.statbuf, &host_statbuf, sizeof(host_statbuf));
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$realpath(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_realpath_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto path = mmu().copy_buffer_from_vm((FlatPtr)params.path.characters, params.path.length);
|
|
auto buffer_result = ByteBuffer::create_zeroed(params.buffer.size);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& host_buffer = buffer_result.value();
|
|
|
|
Syscall::SC_realpath_params host_params;
|
|
host_params.path = { (char const*)path.data(), path.size() };
|
|
host_params.buffer = { (char*)host_buffer.data(), host_buffer.size() };
|
|
int rc = syscall(SC_realpath, &host_params);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm((FlatPtr)params.buffer.data, host_buffer.data(), host_buffer.size());
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$gethostname(FlatPtr buffer, ssize_t buffer_size)
|
|
{
|
|
if (buffer_size < 0)
|
|
return -EINVAL;
|
|
auto buffer_result = ByteBuffer::create_zeroed(buffer_size);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& host_buffer = buffer_result.value();
|
|
int rc = syscall(SC_gethostname, host_buffer.data(), host_buffer.size());
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(buffer, host_buffer.data(), host_buffer.size());
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$sigaction(int signum, FlatPtr act, FlatPtr oldact)
|
|
{
|
|
if (signum == SIGKILL) {
|
|
reportln("Attempted to sigaction() with SIGKILL"sv);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (signum <= 0 || signum >= NSIG)
|
|
return -EINVAL;
|
|
|
|
struct sigaction host_act;
|
|
mmu().copy_from_vm(&host_act, act, sizeof(host_act));
|
|
|
|
auto& handler = m_signal_handler[signum];
|
|
handler.handler = (FlatPtr)host_act.sa_handler;
|
|
handler.mask = host_act.sa_mask;
|
|
handler.flags = host_act.sa_flags;
|
|
|
|
if (oldact) {
|
|
struct sigaction host_oldact;
|
|
auto& old_handler = m_signal_handler[signum];
|
|
host_oldact.sa_handler = (void (*)(int))(old_handler.handler);
|
|
host_oldact.sa_mask = old_handler.mask;
|
|
host_oldact.sa_flags = old_handler.flags;
|
|
mmu().copy_to_vm(oldact, &host_oldact, sizeof(host_oldact));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$sigprocmask(int how, FlatPtr set, FlatPtr old_set)
|
|
{
|
|
if (old_set) {
|
|
mmu().copy_to_vm(old_set, &m_signal_mask, sizeof(sigset_t));
|
|
}
|
|
if (set) {
|
|
sigset_t set_value;
|
|
mmu().copy_from_vm(&set_value, set, sizeof(sigset_t));
|
|
switch (how) {
|
|
case SIG_BLOCK:
|
|
m_signal_mask |= set_value;
|
|
break;
|
|
case SIG_SETMASK:
|
|
m_signal_mask = set_value;
|
|
break;
|
|
case SIG_UNBLOCK:
|
|
m_signal_mask &= ~set_value;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$sigreturn()
|
|
{
|
|
u32 stack_ptr = m_cpu->esp().value();
|
|
auto local_pop = [&]<typename T>() {
|
|
auto value = m_cpu->read_memory<T>({ m_cpu->ss(), stack_ptr });
|
|
stack_ptr += sizeof(T);
|
|
return value;
|
|
};
|
|
|
|
// State from signal trampoline (note that we're assuming i386 here):
|
|
// saved_ax, ucontext, signal_info, fpu_state.
|
|
|
|
// Drop the FPU state
|
|
// FIXME: Read and restore from this.
|
|
stack_ptr += 512;
|
|
|
|
// Drop the signal info
|
|
stack_ptr += sizeof(siginfo_t);
|
|
|
|
auto ucontext = local_pop.operator()<ucontext_t>();
|
|
|
|
auto eax = local_pop.operator()<u32>();
|
|
|
|
m_signal_mask = ucontext.value().uc_sigmask;
|
|
|
|
auto mcontext_slice = ucontext.slice<&ucontext_t::uc_mcontext>();
|
|
|
|
m_cpu->set_edi(mcontext_slice.slice<&__mcontext::edi>());
|
|
m_cpu->set_esi(mcontext_slice.slice<&__mcontext::esi>());
|
|
m_cpu->set_ebp(mcontext_slice.slice<&__mcontext::ebp>());
|
|
m_cpu->set_esp(mcontext_slice.slice<&__mcontext::esp>());
|
|
m_cpu->set_ebx(mcontext_slice.slice<&__mcontext::ebx>());
|
|
m_cpu->set_edx(mcontext_slice.slice<&__mcontext::edx>());
|
|
m_cpu->set_ecx(mcontext_slice.slice<&__mcontext::ecx>());
|
|
m_cpu->set_eax(mcontext_slice.slice<&__mcontext::eax>());
|
|
m_cpu->set_eip(mcontext_slice.value().eip);
|
|
m_cpu->set_eflags(mcontext_slice.slice<&__mcontext::eflags>());
|
|
|
|
// FIXME: We're dropping the shadow bits here.
|
|
return eax.value();
|
|
}
|
|
|
|
int Emulator::virt$getpgrp()
|
|
{
|
|
return syscall(SC_getpgrp);
|
|
}
|
|
|
|
int Emulator::virt$getpgid(pid_t pid)
|
|
{
|
|
return syscall(SC_getpgid, pid);
|
|
}
|
|
|
|
int Emulator::virt$setpgid(pid_t pid, pid_t pgid)
|
|
{
|
|
return syscall(SC_setpgid, pid, pgid);
|
|
}
|
|
|
|
int Emulator::virt$getcwd(FlatPtr buffer, size_t buffer_size)
|
|
{
|
|
auto buffer_result = ByteBuffer::create_zeroed(buffer_size);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& host_buffer = buffer_result.value();
|
|
int rc = syscall(SC_getcwd, host_buffer.data(), host_buffer.size());
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm(buffer, host_buffer.data(), host_buffer.size());
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$getsid(pid_t pid)
|
|
{
|
|
return syscall(SC_getsid, pid);
|
|
}
|
|
|
|
int Emulator::virt$faccessat(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_faccessat_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto host_path = mmu().copy_buffer_from_vm(reinterpret_cast<FlatPtr>(params.pathname.characters), params.pathname.length);
|
|
Syscall::SC_faccessat_params host_params = params;
|
|
host_params.pathname = { reinterpret_cast<char const*>(host_path.data()), host_path.size() };
|
|
|
|
return syscall(SC_faccessat, &host_params);
|
|
}
|
|
|
|
int Emulator::virt$waitid(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_waitid_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
Syscall::SC_waitid_params host_params = params;
|
|
siginfo info {};
|
|
host_params.infop = &info;
|
|
|
|
int rc = syscall(SC_waitid, &host_params);
|
|
if (rc < 0)
|
|
return rc;
|
|
|
|
if (info.si_addr) {
|
|
// FIXME: Translate this somehow once we actually start setting it in the kernel.
|
|
dbgln("si_addr is set to {:p}, I did not expect this!", info.si_addr);
|
|
TODO();
|
|
}
|
|
|
|
if (params.infop)
|
|
mmu().copy_to_vm((FlatPtr)params.infop, &info, sizeof(info));
|
|
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$chdir(FlatPtr path, size_t path_length)
|
|
{
|
|
auto host_path = mmu().copy_buffer_from_vm(path, path_length);
|
|
return syscall(SC_chdir, host_path.data(), host_path.size());
|
|
}
|
|
|
|
int Emulator::virt$dup2(int old_fd, int new_fd)
|
|
{
|
|
return syscall(SC_dup2, old_fd, new_fd);
|
|
}
|
|
|
|
int Emulator::virt$scheduler_get_parameters(FlatPtr user_addr)
|
|
{
|
|
Syscall::SC_scheduler_parameters_params user_param;
|
|
mmu().copy_from_vm(&user_param, user_addr, sizeof(user_param));
|
|
auto rc = syscall(SC_scheduler_get_parameters, &user_param);
|
|
mmu().copy_to_vm(user_addr, &user_param, sizeof(user_param));
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$scheduler_set_parameters(FlatPtr user_addr)
|
|
{
|
|
Syscall::SC_scheduler_parameters_params user_param;
|
|
mmu().copy_from_vm(&user_param, user_addr, sizeof(user_param));
|
|
return syscall(SC_scheduler_set_parameters, &user_param);
|
|
}
|
|
|
|
int Emulator::virt$set_thread_name(pid_t pid, FlatPtr name_addr, size_t name_length)
|
|
{
|
|
auto user_name = mmu().copy_buffer_from_vm(name_addr, name_length);
|
|
auto name = DeprecatedString::formatted("(UE) {}", StringView { user_name.data(), user_name.size() });
|
|
return syscall(SC_set_thread_name, pid, name.characters(), name.length());
|
|
}
|
|
|
|
pid_t Emulator::virt$setsid()
|
|
{
|
|
return syscall(SC_setsid);
|
|
}
|
|
|
|
int Emulator::virt$create_inode_watcher(unsigned flags)
|
|
{
|
|
return syscall(SC_create_inode_watcher, flags);
|
|
}
|
|
|
|
int Emulator::virt$inode_watcher_add_watch(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_inode_watcher_add_watch_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
return syscall(SC_inode_watcher_add_watch, ¶ms);
|
|
}
|
|
|
|
int Emulator::virt$inode_watcher_remove_watch(int fd, int wd)
|
|
{
|
|
return syscall(SC_inode_watcher_add_watch, fd, wd);
|
|
}
|
|
|
|
int Emulator::virt$clock_nanosleep(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_clock_nanosleep_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
timespec requested_sleep;
|
|
mmu().copy_from_vm(&requested_sleep, (FlatPtr)params.requested_sleep, sizeof(timespec));
|
|
params.requested_sleep = &requested_sleep;
|
|
|
|
auto* remaining_vm_addr = params.remaining_sleep;
|
|
timespec remaining { 0, 0 };
|
|
params.remaining_sleep = &remaining;
|
|
|
|
int rc = syscall(SC_clock_nanosleep, ¶ms);
|
|
if (remaining_vm_addr)
|
|
mmu().copy_to_vm((FlatPtr)remaining_vm_addr, &remaining, sizeof(timespec));
|
|
|
|
return rc;
|
|
}
|
|
|
|
int Emulator::virt$readlink(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_readlink_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
auto path = mmu().copy_buffer_from_vm((FlatPtr)params.path.characters, params.path.length);
|
|
auto buffer_result = ByteBuffer::create_zeroed(params.buffer.size);
|
|
if (buffer_result.is_error())
|
|
return -ENOMEM;
|
|
auto& host_buffer = buffer_result.value();
|
|
|
|
Syscall::SC_readlink_params host_params;
|
|
host_params.path = { (char const*)path.data(), path.size() };
|
|
host_params.buffer = { (char*)host_buffer.data(), host_buffer.size() };
|
|
int rc = syscall(SC_readlink, &host_params);
|
|
if (rc < 0)
|
|
return rc;
|
|
mmu().copy_to_vm((FlatPtr)params.buffer.data, host_buffer.data(), host_buffer.size());
|
|
return rc;
|
|
}
|
|
|
|
u32 Emulator::virt$allocate_tls(FlatPtr initial_data, size_t size)
|
|
{
|
|
// TODO: This matches what Thread::make_thread_specific_region does. The kernel
|
|
// ends up allocating one more page. Figure out if this is intentional.
|
|
auto region_size = align_up_to(size, PAGE_SIZE) + PAGE_SIZE;
|
|
constexpr auto tls_location = VirtualAddress(0x20000000);
|
|
m_range_allocator.reserve_user_range(tls_location, region_size);
|
|
auto tcb_region = make<SimpleRegion>(tls_location.get(), region_size);
|
|
|
|
size_t offset = 0;
|
|
while (size - offset > 0) {
|
|
u8 buffer[512];
|
|
size_t read_bytes = min(sizeof(buffer), size - offset);
|
|
mmu().copy_from_vm(buffer, initial_data + offset, read_bytes);
|
|
memcpy(tcb_region->data() + offset, buffer, read_bytes);
|
|
offset += read_bytes;
|
|
}
|
|
memset(tcb_region->shadow_data(), 0x01, size);
|
|
|
|
auto tls_region = make<SimpleRegion>(0, 4);
|
|
tls_region->write32(0, shadow_wrap_as_initialized(tcb_region->base() + (u32)size));
|
|
memset(tls_region->shadow_data(), 0x01, 4);
|
|
|
|
u32 tls_base = tcb_region->base();
|
|
mmu().add_region(move(tcb_region));
|
|
mmu().set_tls_region(move(tls_region));
|
|
return tls_base;
|
|
}
|
|
|
|
int Emulator::virt$beep()
|
|
{
|
|
return syscall(SC_beep);
|
|
}
|
|
|
|
u32 Emulator::virt$sysconf(u32 name)
|
|
{
|
|
return syscall(SC_sysconf, name);
|
|
}
|
|
|
|
int Emulator::virt$annotate_mapping(FlatPtr)
|
|
{
|
|
// FIXME: Implement this.
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$futex(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_futex_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
// FIXME: Implement this.
|
|
return 0;
|
|
}
|
|
|
|
int Emulator::virt$poll(FlatPtr params_addr)
|
|
{
|
|
Syscall::SC_poll_params params;
|
|
mmu().copy_from_vm(¶ms, params_addr, sizeof(params));
|
|
|
|
if (params.nfds >= FD_SETSIZE)
|
|
return EINVAL;
|
|
|
|
Vector<pollfd, FD_SETSIZE> fds;
|
|
struct timespec timeout;
|
|
u32 sigmask;
|
|
|
|
if (params.fds)
|
|
mmu().copy_from_vm(fds.data(), (FlatPtr)params.fds, sizeof(pollfd) * params.nfds);
|
|
if (params.timeout)
|
|
mmu().copy_from_vm(&timeout, (FlatPtr)params.timeout, sizeof(timeout));
|
|
if (params.sigmask)
|
|
mmu().copy_from_vm(&sigmask, (FlatPtr)params.sigmask, sizeof(sigmask));
|
|
|
|
int rc = ppoll(params.fds ? fds.data() : nullptr, params.nfds, params.timeout ? &timeout : nullptr, params.sigmask ? &sigmask : nullptr);
|
|
if (rc < 0)
|
|
return -errno;
|
|
|
|
if (params.fds)
|
|
mmu().copy_to_vm((FlatPtr)params.fds, fds.data(), sizeof(pollfd) * params.nfds);
|
|
if (params.timeout)
|
|
mmu().copy_to_vm((FlatPtr)params.timeout, &timeout, sizeof(timeout));
|
|
|
|
return rc;
|
|
}
|
|
}
|