From a79ce29cd6953c54da583affd1049090af47b70d Mon Sep 17 00:00:00 2001
From: Gabe Black <gabe.black@gmail.com>
Date: Tue, 20 Oct 2020 17:36:50 -0700
Subject: [PATCH] x86: Move syscall handling for Linux into the EmuLinux
 workload.
MIME-Version: 1.0
Content-Type: text/plain; charset=utf8
Content-Transfer-Encoding: 8bit

Change-Id: I3fe1997e62491e9576b787660b7fae5ae99fb5c9
Reviewed-on: https://gem5-review.googlesource.com/c/public/gem5/+/33903
Reviewed-by: Gabe Black <gabe.black@gmail.com>
Reviewed-by: Alexandru Duțu <alexandru.dutu@amd.com>
Maintainer: Gabe Black <gabe.black@gmail.com>
Tested-by: kokoro <noreply+kokoro@google.com>
---
 src/arch/x86/linux/process.cc     | 863 +-----------------------------
 src/arch/x86/linux/process.hh     |  39 --
 src/arch/x86/linux/se_workload.cc | 844 ++++++++++++++++++++++++++++-
 src/arch/x86/linux/se_workload.hh |  37 ++
 src/arch/x86/process.hh           |   2 +
 5 files changed, 884 insertions(+), 901 deletions(-)

diff --git a/src/arch/x86/linux/process.cc b/src/arch/x86/linux/process.cc
index e95cf56df..7d9f174f7 100644
--- a/src/arch/x86/linux/process.cc
+++ b/src/arch/x86/linux/process.cc
@@ -37,21 +37,10 @@
 
 #include "arch/x86/linux/process.hh"
 
-#include <sys/syscall.h>
-
-#include "arch/x86/isa_traits.hh"
-#include "arch/x86/linux/linux.hh"
-#include "arch/x86/registers.hh"
 #include "base/loader/object_file.hh"
-#include "base/trace.hh"
-#include "cpu/thread_context.hh"
-#include "kern/linux/linux.hh"
 #include "sim/process.hh"
-#include "sim/syscall_desc.hh"
-#include "sim/syscall_emul.hh"
 
 using namespace std;
-using namespace X86ISA;
 
 namespace
 {
@@ -77,860 +66,12 @@ class X86LinuxObjectFileLoader : public Process::Loader
             return nullptr;
 
         if (arch == ::Loader::X86_64)
-            return new X86_64LinuxProcess(params, obj_file);
+            return new X86ISA::X86_64LinuxProcess(params, obj_file);
         else
-            return new I386LinuxProcess(params, obj_file);
+            return new X86ISA::I386LinuxProcess(params, obj_file);
     }
 };
 
 X86LinuxObjectFileLoader loader;
 
 } // anonymous namespace
-
-/// Target uname() handler.
-static SyscallReturn
-unameFunc(SyscallDesc *desc, ThreadContext *tc, VPtr<Linux::utsname> name)
-{
-    auto process = tc->getProcessPtr();
-
-    strcpy(name->sysname, "Linux");
-    strcpy(name->nodename, "sim.gem5.org");
-    strcpy(name->release, process->release.c_str());
-    strcpy(name->version, "#1 Mon Aug 18 11:32:15 EDT 2003");
-    strcpy(name->machine, "x86_64");
-
-    return 0;
-}
-
-static SyscallReturn
-archPrctlFunc(SyscallDesc *desc, ThreadContext *tc, int code, uint64_t addr)
-{
-    enum ArchPrctlCodes
-    {
-        SetFS = 0x1002,
-        GetFS = 0x1003,
-        SetGS = 0x1001,
-        GetGS = 0x1004
-    };
-
-    uint64_t fsBase, gsBase;
-    PortProxy &p = tc->getVirtProxy();
-    switch(code)
-    {
-      // Each of these valid options should actually check addr.
-      case SetFS:
-        tc->setMiscRegNoEffect(MISCREG_FS_BASE, addr);
-        tc->setMiscRegNoEffect(MISCREG_FS_EFF_BASE, addr);
-        return 0;
-      case GetFS:
-        fsBase = tc->readMiscRegNoEffect(MISCREG_FS_BASE);
-        p.write(addr, fsBase);
-        return 0;
-      case SetGS:
-        tc->setMiscRegNoEffect(MISCREG_GS_BASE, addr);
-        tc->setMiscRegNoEffect(MISCREG_GS_EFF_BASE, addr);
-        return 0;
-      case GetGS:
-        gsBase = tc->readMiscRegNoEffect(MISCREG_GS_BASE);
-        p.write(addr, gsBase);
-        return 0;
-      default:
-        return -EINVAL;
-    }
-}
-
-BitUnion32(UserDescFlags)
-    Bitfield<0> seg_32bit;
-    Bitfield<2, 1> contents;
-    Bitfield<3> read_exec_only;
-    Bitfield<4> limit_in_pages;
-    Bitfield<5> seg_not_present;
-    Bitfield<6> useable;
-EndBitUnion(UserDescFlags)
-
-struct UserDesc32 {
-    uint32_t entry_number;
-    uint32_t base_addr;
-    uint32_t limit;
-    uint32_t flags;
-};
-
-struct UserDesc64 {
-    uint32_t entry_number;
-    uint32_t __padding1;
-    uint64_t base_addr;
-    uint32_t limit;
-    uint32_t flags;
-};
-
-static SyscallReturn
-setThreadArea32Func(SyscallDesc *desc, ThreadContext *tc,
-                    VPtr<UserDesc32> userDesc)
-{
-    const int minTLSEntry = 6;
-    const int numTLSEntries = 3;
-    const int maxTLSEntry = minTLSEntry + numTLSEntries - 1;
-
-    auto process = tc->getProcessPtr();
-
-    X86Process *x86p = dynamic_cast<X86Process *>(process);
-    assert(x86p);
-
-    assert((maxTLSEntry + 1) * sizeof(uint64_t) <= x86p->gdtSize());
-
-    TypedBufferArg<uint64_t>
-        gdt(x86p->gdtStart() + minTLSEntry * sizeof(uint64_t),
-            numTLSEntries * sizeof(uint64_t));
-
-    if (!gdt.copyIn(tc->getVirtProxy()))
-        panic("Failed to copy in GDT for %s.\n", desc->name());
-
-    if (userDesc->entry_number == (uint32_t)(-1)) {
-        // Find a free TLS entry.
-        for (int i = 0; i < numTLSEntries; i++) {
-            if (gdt[i] == 0) {
-                userDesc->entry_number = i + minTLSEntry;
-                break;
-            }
-        }
-        // We failed to find one.
-        if (userDesc->entry_number == (uint32_t)(-1))
-            return -ESRCH;
-    }
-
-    int index = userDesc->entry_number;
-
-    if (index < minTLSEntry || index > maxTLSEntry)
-        return -EINVAL;
-
-    index -= minTLSEntry;
-
-    // Build the entry we're going to add.
-    SegDescriptor segDesc = 0;
-    UserDescFlags flags = userDesc->flags;
-
-    segDesc.limitLow = bits(userDesc->limit, 15, 0);
-    segDesc.baseLow = bits(userDesc->base_addr, 23, 0);
-    segDesc.type.a = 1;
-    if (!flags.read_exec_only)
-        segDesc.type.w = 1;
-    if (bits((uint8_t)flags.contents, 0))
-        segDesc.type.e = 1;
-    if (bits((uint8_t)flags.contents, 1))
-        segDesc.type.codeOrData = 1;
-    segDesc.s = 1;
-    segDesc.dpl = 3;
-    if (!flags.seg_not_present)
-        segDesc.p = 1;
-    segDesc.limitHigh = bits(userDesc->limit, 19, 16);
-    if (flags.useable)
-        segDesc.avl = 1;
-    segDesc.l = 0;
-    if (flags.seg_32bit)
-        segDesc.d = 1;
-    if (flags.limit_in_pages)
-        segDesc.g = 1;
-    segDesc.baseHigh = bits(userDesc->base_addr, 31, 24);
-
-    gdt[index] = (uint64_t)segDesc;
-
-    if (!gdt.copyOut(tc->getVirtProxy()))
-        panic("Failed to copy out GDT for %s.\n", desc->name());
-
-    return 0;
-}
-
-const std::vector<IntRegIndex> X86_64LinuxProcess::SyscallABI::ArgumentRegs = {
-    INTREG_RDI, INTREG_RSI, INTREG_RDX, INTREG_R10W, INTREG_R8W, INTREG_R9W
-};
-
-static SyscallDescTable<X86_64LinuxProcess::SyscallABI> syscallDescs64 = {
-    {   0, "read", readFunc<X86Linux64> },
-    {   1, "write", writeFunc<X86Linux64> },
-    {   2, "open", openFunc<X86Linux64> },
-    {   3, "close", closeFunc },
-    {   4, "stat", stat64Func<X86Linux64> },
-    {   5, "fstat", fstat64Func<X86Linux64> },
-    {   6, "lstat", lstat64Func<X86Linux64> },
-    {   7, "poll", pollFunc<X86Linux64> },
-    {   8, "lseek", lseekFunc },
-    {   9, "mmap", mmapFunc<X86Linux64> },
-    {  10, "mprotect", ignoreFunc },
-    {  11, "munmap", munmapFunc },
-    {  12, "brk", brkFunc },
-    {  13, "rt_sigaction", ignoreWarnOnceFunc },
-    {  14, "rt_sigprocmask", ignoreWarnOnceFunc },
-    {  15, "rt_sigreturn" },
-    {  16, "ioctl", ioctlFunc<X86Linux64> },
-    {  17, "pread64", pread64Func<X86Linux64> },
-    {  18, "pwrite64", pwrite64Func<X86Linux64> },
-    {  19, "readv", readvFunc<X86Linux64> },
-    {  20, "writev", writevFunc<X86Linux64> },
-    {  21, "access", ignoreFunc },
-    {  22, "pipe", pipeFunc },
-    {  23, "select", selectFunc<X86Linux64> },
-    {  24, "sched_yield", ignoreFunc },
-    {  25, "mremap", mremapFunc<X86Linux64> },
-    {  26, "msync" },
-    {  27, "mincore" },
-    {  28, "madvise", ignoreFunc },
-    {  29, "shmget" },
-    {  30, "shmat" },
-    {  31, "shmctl" },
-    {  32, "dup", dupFunc },
-    {  33, "dup2", dup2Func },
-    {  34, "pause" },
-    {  35, "nanosleep", ignoreWarnOnceFunc },
-    {  36, "getitimer" },
-    {  37, "alarm" },
-    {  38, "setitimer" },
-    {  39, "getpid", getpidFunc },
-    {  40, "sendfile" },
-    {  41, "socket", socketFunc<X86Linux64> },
-    {  42, "connect", connectFunc },
-    {  43, "accept", acceptFunc<X86Linux64> },
-    {  44, "sendto", sendtoFunc },
-    {  45, "recvfrom", recvfromFunc },
-    {  46, "sendmsg", sendmsgFunc },
-    {  47, "recvmsg", recvmsgFunc },
-    {  48, "shutdown", shutdownFunc },
-    {  49, "bind", bindFunc },
-    {  50, "listen", listenFunc },
-    {  51, "getsockname", getsocknameFunc },
-    {  52, "getpeername", getpeernameFunc },
-    {  53, "socketpair", socketpairFunc<X86Linux64> },
-    {  54, "setsockopt", setsockoptFunc },
-    {  55, "getsockopt", getsockoptFunc },
-    {  56, "clone", cloneFunc<X86Linux64> },
-    {  57, "fork" },
-    {  58, "vfork" },
-    {  59, "execve", execveFunc<X86Linux64> },
-    {  60, "exit", exitFunc },
-    {  61, "wait4", wait4Func<X86Linux64> },
-    {  62, "kill" },
-    {  63, "uname", unameFunc },
-    {  64, "semget" },
-    {  65, "semop" },
-    {  66, "semctl" },
-    {  67, "shmdt" },
-    {  68, "msgget" },
-    {  69, "msgsnd" },
-    {  70, "msgrcv" },
-    {  71, "msgctl" },
-    {  72, "fcntl", fcntlFunc },
-    {  73, "flock" },
-    {  74, "fsync" },
-    {  75, "fdatasync" },
-    {  76, "truncate", truncateFunc },
-    {  77, "ftruncate", ftruncateFunc },
-#if defined(SYS_getdents)
-    {  78, "getdents", getdentsFunc },
-#else
-    {  78, "getdents" },
-#endif
-    {  79, "getcwd", getcwdFunc },
-    {  80, "chdir", chdirFunc },
-    {  81, "fchdir" },
-    {  82, "rename", renameFunc },
-    {  83, "mkdir", mkdirFunc },
-    {  84, "rmdir", rmdirFunc },
-    {  85, "creat" },
-    {  86, "link", linkFunc },
-    {  87, "unlink", unlinkFunc },
-    {  88, "symlink", symlinkFunc },
-    {  89, "readlink", readlinkFunc },
-    {  90, "chmod", ignoreFunc },
-    {  91, "fchmod" },
-    {  92, "chown" },
-    {  93, "fchown" },
-    {  94, "lchown" },
-    {  95, "umask", umaskFunc },
-    {  96, "gettimeofday", gettimeofdayFunc<X86Linux64> },
-    {  97, "getrlimit", getrlimitFunc<X86Linux64> },
-    {  98, "getrusage", getrusageFunc<X86Linux64> },
-    {  99, "sysinfo", sysinfoFunc<X86Linux64> },
-    { 100, "times", timesFunc<X86Linux64> },
-    { 101, "ptrace" },
-    { 102, "getuid", getuidFunc },
-    { 103, "syslog" },
-    { 104, "getgid", getgidFunc },
-    { 105, "setuid" },
-    { 106, "setgid" },
-    { 107, "geteuid", geteuidFunc },
-    { 108, "getegid", getegidFunc },
-    { 109, "setpgid", setpgidFunc },
-    { 110, "getppid", getppidFunc },
-    { 111, "getpgrp", getpgrpFunc },
-    { 112, "setsid" },
-    { 113, "setreuid" },
-    { 114, "setregid" },
-    { 115, "getgroups" },
-    { 116, "setgroups" },
-    { 117, "setresuid", ignoreFunc },
-    { 118, "getresuid" },
-    { 119, "setresgid" },
-    { 120, "getresgid" },
-    { 121, "getpgid" },
-    { 122, "setfsuid" },
-    { 123, "setfsgid" },
-    { 124, "getsid" },
-    { 125, "capget" },
-    { 126, "capset" },
-    { 127, "rt_sigpending" },
-    { 128, "rt_sigtimedwait" },
-    { 129, "rt_sigqueueinfo" },
-    { 130, "rt_sigsuspend" },
-    { 131, "sigaltstack" },
-    { 132, "utime" },
-    { 133, "mknod", mknodFunc },
-    { 134, "uselib" },
-    { 135, "personality" },
-    { 136, "ustat" },
-    { 137, "statfs", statfsFunc<X86Linux64> },
-    { 138, "fstatfs", fstatfsFunc<X86Linux64> },
-    { 139, "sysfs" },
-    { 140, "getpriority" },
-    { 141, "setpriority", ignoreFunc },
-    { 142, "sched_setparam" },
-    { 143, "sched_getparam" },
-    { 144, "sched_setscheduler" },
-    { 145, "sched_getscheduler" },
-    { 146, "sched_get_priority_max" },
-    { 147, "sched_get_priority_min" },
-    { 148, "sched_rr_get_interval" },
-    { 149, "mlock" },
-    { 150, "munlock" },
-    { 151, "mlockall" },
-    { 152, "munlockall" },
-    { 153, "vhangup" },
-    { 154, "modify_ldt" },
-    { 155, "pivot_root" },
-    { 156, "_sysctl" },
-    { 157, "prctl" },
-    { 158, "arch_prctl", archPrctlFunc },
-    { 159, "adjtimex" },
-    { 160, "setrlimit", ignoreFunc },
-    { 161, "chroot" },
-    { 162, "sync" },
-    { 163, "acct" },
-    { 164, "settimeofday" },
-    { 165, "mount" },
-    { 166, "umount2" },
-    { 167, "swapon" },
-    { 168, "swapoff" },
-    { 169, "reboot" },
-    { 170, "sethostname" },
-    { 171, "setdomainname" },
-    { 172, "iopl" },
-    { 173, "ioperm" },
-    { 174, "create_module" },
-    { 175, "init_module" },
-    { 176, "delete_module" },
-    { 177, "get_kernel_syms" },
-    { 178, "query_module" },
-    { 179, "quotactl" },
-    { 180, "nfsservctl" },
-    { 181, "getpmsg" },
-    { 182, "putpmsg" },
-    { 183, "afs_syscall" },
-    { 184, "tuxcall" },
-    { 185, "security" },
-    { 186, "gettid", gettidFunc },
-    { 187, "readahead" },
-    { 188, "setxattr" },
-    { 189, "lsetxattr" },
-    { 190, "fsetxattr" },
-    { 191, "getxattr" },
-    { 192, "lgetxattr" },
-    { 193, "fgetxattr" },
-    { 194, "listxattr" },
-    { 195, "llistxattr" },
-    { 196, "flistxattr" },
-    { 197, "removexattr" },
-    { 198, "lremovexattr" },
-    { 199, "fremovexattr" },
-    { 200, "tkill" },
-    { 201, "time", timeFunc<X86Linux64> },
-    { 202, "futex", futexFunc<X86Linux64> },
-    { 203, "sched_setaffinity", ignoreFunc },
-    { 204, "sched_getaffinity", ignoreFunc },
-    { 205, "set_thread_area" },
-    { 206, "io_setup" },
-    { 207, "io_destroy" },
-    { 208, "io_getevents" },
-    { 209, "io_submit" },
-    { 210, "io_cancel" },
-    { 211, "get_thread_area" },
-    { 212, "lookup_dcookie" },
-    { 213, "epoll_create" },
-    { 214, "epoll_ctl_old" },
-    { 215, "epoll_wait_old" },
-    { 216, "remap_file_pages" },
-    { 217, "getdents64" },
-    { 218, "set_tid_address", setTidAddressFunc },
-    { 219, "restart_syscall" },
-    { 220, "semtimedop" },
-    { 221, "fadvise64", ignoreFunc },
-    { 222, "timer_create" },
-    { 223, "timer_settime" },
-    { 224, "timer_gettime" },
-    { 225, "timer_getoverrun" },
-    { 226, "timer_delete" },
-    { 227, "clock_settime" },
-    { 228, "clock_gettime", clock_gettimeFunc<X86Linux64> },
-    { 229, "clock_getres", clock_getresFunc<X86Linux64> },
-    { 230, "clock_nanosleep" },
-    { 231, "exit_group", exitGroupFunc },
-    { 232, "epoll_wait" },
-    { 233, "epoll_ctl" },
-    { 234, "tgkill", tgkillFunc<X86Linux64> },
-    { 235, "utimes" },
-    { 236, "vserver" },
-    { 237, "mbind" },
-    { 238, "set_mempolicy" },
-    { 239, "get_mempolicy", ignoreFunc },
-    { 240, "mq_open" },
-    { 241, "mq_unlink" },
-    { 242, "mq_timedsend" },
-    { 243, "mq_timedreceive" },
-    { 244, "mq_notify" },
-    { 245, "mq_getsetattr" },
-    { 246, "kexec_load" },
-    { 247, "waitid" },
-    { 248, "add_key" },
-    { 249, "request_key" },
-    { 250, "keyctl" },
-    { 251, "ioprio_set" },
-    { 252, "ioprio_get" },
-    { 253, "inotify_init" },
-    { 254, "inotify_add_watch" },
-    { 255, "inotify_rm_watch" },
-    { 256, "migrate_pages" },
-    { 257, "openat", openatFunc<X86Linux64> },
-    { 258, "mkdirat" },
-    { 259, "mknodat" },
-    { 260, "fchownat" },
-    { 261, "futimesat" },
-    { 262, "newfstatat" },
-    { 263, "unlinkat" },
-    { 264, "renameat" },
-    { 265, "linkat" },
-    { 266, "symlinkat" },
-    { 267, "readlinkat", readlinkFunc },
-    { 268, "fchmodat" },
-    { 269, "faccessat" },
-    { 270, "pselect6" },
-    { 271, "ppoll" },
-    { 272, "unshare" },
-    { 273, "set_robust_list", ignoreFunc },
-    { 274, "get_robust_list" },
-    { 275, "splice" },
-    { 276, "tee" },
-    { 277, "sync_file_range" },
-    { 278, "vmsplice" },
-    { 279, "move_pages" },
-    { 280, "utimensat" },
-    { 281, "epoll_pwait" },
-    { 282, "signalfd" },
-    { 283, "timerfd_create" },
-    { 284, "eventfd", eventfdFunc<X86Linux64> },
-    { 285, "fallocate", fallocateFunc },
-    { 286, "timerfd_settime" },
-    { 287, "timerfd_gettime" },
-    { 288, "accept4" },
-    { 289, "signalfd4" },
-    { 290, "eventfd2", eventfdFunc<X86Linux64> },
-    { 291, "epoll_create1" },
-    { 292, "dup3" },
-    { 293, "pipe2", pipe2Func },
-    { 294, "inotify_init1" },
-    { 295, "preadv" },
-    { 296, "pwritev" },
-    { 297, "rt_tgsigqueueinfo" },
-    { 298, "perf_event_open" },
-    { 299, "recvmmsg" },
-    { 300, "fanotify_init" },
-    { 301, "fanotify_mark" },
-    { 302, "prlimit64", prlimitFunc<X86Linux64> },
-    { 303, "name_to_handle_at" },
-    { 304, "open_by_handle_at" },
-    { 305, "clock_adjtime" },
-    { 306, "syncfs" },
-    { 307, "sendmmsg" },
-    { 308, "setns" },
-    { 309, "getcpu", getcpuFunc },
-    { 310, "proess_vm_readv" },
-    { 311, "proess_vm_writev" },
-    { 312, "kcmp" },
-    { 313, "finit_module" },
-};
-
-void
-X86_64LinuxProcess::syscall(ThreadContext *tc)
-{
-    X86_64Process::syscall(tc);
-    syscallDescs64.get(tc->readIntReg(INTREG_RAX))->doSyscall(tc);
-}
-
-void
-X86_64LinuxProcess::clone(ThreadContext *old_tc, ThreadContext *new_tc,
-                          Process *process, RegVal flags)
-{
-    X86_64Process::clone(old_tc, new_tc, (X86_64Process*)process, flags);
-}
-
-const std::vector<IntRegIndex> I386LinuxProcess::SyscallABI::ArgumentRegs = {
-    INTREG_EBX, INTREG_ECX, INTREG_EDX, INTREG_ESI, INTREG_EDI, INTREG_EBP
-};
-
-static SyscallDescTable<I386LinuxProcess::SyscallABI> syscallDescs32 = {
-    {   0, "restart_syscall" },
-    {   1, "exit", exitFunc },
-    {   2, "fork" },
-    {   3, "read", readFunc<X86Linux32> },
-    {   4, "write", writeFunc<X86Linux32> },
-    {   5, "open", openFunc<X86Linux32> },
-    {   6, "close", closeFunc },
-    {   7, "waitpid" },
-    {   8, "creat" },
-    {   9, "link" },
-    {  10, "unlink" },
-    {  11, "execve", execveFunc<X86Linux32> },
-    {  12, "chdir", chdirFunc },
-    {  13, "time", timeFunc<X86Linux32> },
-    {  14, "mknod", mknodFunc },
-    {  15, "chmod" },
-    {  16, "lchown" },
-    {  17, "break" },
-    {  18, "oldstat" },
-    {  19, "lseek" },
-    {  20, "getpid", getpidFunc },
-    {  21, "mount" },
-    {  22, "umount" },
-    {  23, "setuid" },
-    {  24, "getuid", getuidFunc },
-    {  25, "stime" },
-    {  26, "ptrace" },
-    {  27, "alarm" },
-    {  28, "oldfstat" },
-    {  29, "pause" },
-    {  30, "utime" },
-    {  31, "stty" },
-    {  32, "gtty" },
-    {  33, "access", ignoreFunc },
-    {  34, "nice" },
-    {  35, "ftime" },
-    {  36, "sync" },
-    {  37, "kill" },
-    {  38, "rename" },
-    {  39, "mkdir", mkdirFunc },
-    {  40, "rmdir", mkdirFunc },
-    {  41, "dup", dupFunc },
-    {  42, "pipe", pipeFunc },
-    {  43, "times", timesFunc<X86Linux32> },
-    {  44, "prof" },
-    {  45, "brk", brkFunc },
-    {  46, "setgid" },
-    {  47, "getgid", getgidFunc },
-    {  48, "signal" },
-    {  49, "geteuid", geteuidFunc },
-    {  50, "getegid", getegidFunc },
-    {  51, "acct" },
-    {  52, "umount2" },
-    {  53, "lock" },
-    {  54, "ioctl", ioctlFunc<X86Linux32> },
-    {  55, "fcntl", fcntlFunc },
-    {  56, "mpx" },
-    {  57, "setpgid", setpgidFunc },
-    {  58, "ulimit" },
-    {  59, "oldolduname" },
-    {  60, "umask", umaskFunc },
-    {  61, "chroot" },
-    {  62, "ustat" },
-    {  63, "dup2", dup2Func },
-    {  64, "getppid" },
-    {  65, "getpgrp" },
-    {  66, "setsid" },
-    {  67, "sigaction" },
-    {  68, "sgetmask" },
-    {  69, "ssetmask" },
-    {  70, "setreuid" },
-    {  71, "setregid" },
-    {  72, "sigsuspend" },
-    {  73, "sigpending" },
-    {  74, "sethostname" },
-    {  75, "setrlimit", ignoreFunc },
-    {  76, "getrlimit", getrlimitFunc<X86Linux32> },
-    {  77, "getrusage", getrusageFunc<X86Linux32> },
-    {  78, "gettimeofday" },
-    {  79, "settimeofday" },
-    {  80, "getgroups" },
-    {  81, "setgroups" },
-    {  82, "select", selectFunc<X86Linux32> },
-    {  83, "symlink" },
-    {  84, "oldlstat" },
-    {  85, "readlink", readlinkFunc },
-    {  86, "uselib" },
-    {  87, "swapon" },
-    {  88, "reboot" },
-    {  89, "readdir" },
-    {  90, "mmap" },
-    {  91, "munmap", munmapFunc },
-    {  92, "truncate", truncateFunc },
-    {  93, "ftruncate", ftruncateFunc },
-    {  94, "fchmod" },
-    {  95, "fchown" },
-    {  96, "getpriority" },
-    {  97, "setpriority", ignoreFunc },
-    {  98, "profil" },
-    {  99, "statfs", ignoreFunc },
-    { 100, "fstatfs" },
-    { 101, "ioperm" },
-    { 102, "socketcall" },
-    { 103, "syslog" },
-    { 104, "setitimer" },
-    { 105, "getitimer" },
-    { 106, "stat" },
-    { 107, "lstat" },
-    { 108, "fstat" },
-    { 109, "olduname" },
-    { 110, "iopl" },
-    { 111, "vhangup" },
-    { 112, "idle" },
-    { 113, "vm86old" },
-    { 114, "wait4", wait4Func<X86Linux32> },
-    { 115, "swapoff" },
-    { 116, "sysinfo", sysinfoFunc<X86Linux32> },
-    { 117, "ipc" },
-    { 118, "fsync" },
-    { 119, "sigreturn" },
-    { 120, "clone", cloneFunc<X86Linux32> },
-    { 121, "setdomainname" },
-    { 122, "uname", unameFunc },
-    { 123, "modify_ldt" },
-    { 124, "adjtimex" },
-    { 125, "mprotect", ignoreFunc },
-    { 126, "sigprocmask" },
-    { 127, "create_module" },
-    { 128, "init_module" },
-    { 129, "delete_module" },
-    { 130, "get_kernel_syms" },
-    { 131, "quotactl" },
-    { 132, "getpgid" },
-    { 133, "fchdir" },
-    { 134, "bdflush" },
-    { 135, "sysfs" },
-    { 136, "personality" },
-    { 137, "afs_syscall" },
-    { 138, "setfsuid" },
-    { 139, "setfsgid" },
-    { 140, "_llseek", _llseekFunc },
-#if defined(SYS_getdents)
-    { 141, "getdents", getdentsFunc },
-#else
-    { 141, "getdents" },
-#endif
-    { 142, "_newselect" },
-    { 143, "flock" },
-    { 144, "msync" },
-    { 145, "readv", readvFunc<X86Linux32> },
-    { 146, "writev", writevFunc<X86Linux32> },
-    { 147, "getsid" },
-    { 148, "fdatasync" },
-    { 149, "_sysctl" },
-    { 150, "mlock" },
-    { 151, "munlock" },
-    { 152, "mlockall" },
-    { 153, "munlockall" },
-    { 154, "sched_setparam" },
-    { 155, "sched_getparam" },
-    { 156, "sched_setscheduler" },
-    { 157, "sched_getscheduler" },
-    { 158, "sched_yield", ignoreFunc },
-    { 159, "sched_get_priority_max" },
-    { 160, "sched_get_priority_min" },
-    { 161, "sched_rr_get_interval" },
-    { 162, "nanosleep", ignoreFunc },
-    { 163, "mremap" },
-    { 164, "setresuid", ignoreFunc },
-    { 165, "getresuid" },
-    { 166, "vm86" },
-    { 167, "query_module" },
-    { 168, "poll", pollFunc<X86Linux32> },
-    { 169, "nfsservctl" },
-    { 170, "setresgid" },
-    { 171, "getresgid" },
-    { 172, "prctl" },
-    { 173, "rt_sigreturn" },
-    { 174, "rt_sigaction", ignoreFunc },
-    { 175, "rt_sigprocmask", ignoreFunc },
-    { 176, "rt_sigpending" },
-    { 177, "rt_sigtimedwait" },
-    { 178, "rt_sigqueueinfo" },
-    { 179, "rt_sigsuspend" },
-    { 180, "pread64", pread64Func<X86Linux64> },
-    { 181, "pwrite64", pwrite64Func<X86Linux64> },
-    { 182, "chown" },
-    { 183, "getcwd", getcwdFunc },
-    { 184, "capget" },
-    { 185, "capset" },
-    { 186, "sigaltstack" },
-    { 187, "sendfile" },
-    { 188, "getpmsg" },
-    { 189, "putpmsg" },
-    { 190, "vfork" },
-    { 191, "ugetrlimit", ignoreFunc },
-    { 192, "mmap2", mmap2Func<X86Linux32> },
-    { 193, "truncate64", truncate64Func },
-    { 194, "ftruncate64", ftruncate64Func },
-    { 195, "stat64", stat64Func<X86Linux32> },
-    { 196, "lstat64" },
-    { 197, "fstat64", fstat64Func<X86Linux32> },
-    { 198, "lchown32" },
-    { 199, "getuid32", getuidFunc },
-    { 200, "getgid32", getgidFunc },
-    { 201, "geteuid32", geteuidFunc },
-    { 202, "getegid32", getegidFunc },
-    { 203, "setreuid32" },
-    { 204, "setregid32" },
-    { 205, "getgroups32" },
-    { 206, "setgroups32" },
-    { 207, "fchown32" },
-    { 208, "setresuid32" },
-    { 209, "getresuid32" },
-    { 210, "setresgid32" },
-    { 211, "getresgid32" },
-    { 212, "chown32" },
-    { 213, "setuid32" },
-    { 214, "setgid32" },
-    { 215, "setfsuid32" },
-    { 216, "setfsgid32" },
-    { 217, "pivot_root" },
-    { 218, "mincore" },
-    { 219, "madvise", ignoreFunc },
-    { 220, "madvise1" },
-    { 221, "getdents64" },
-    { 222, "fcntl64" },
-    { 223, "unused" },
-    { 224, "gettid", gettidFunc },
-    { 225, "readahead" },
-    { 226, "setxattr" },
-    { 227, "lsetxattr" },
-    { 228, "fsetxattr" },
-    { 229, "getxattr" },
-    { 230, "lgetxattr" },
-    { 231, "fgetxattr" },
-    { 232, "listxattr" },
-    { 233, "llistxattr" },
-    { 234, "flistxattr" },
-    { 235, "removexattr" },
-    { 236, "lremovexattr" },
-    { 237, "fremovexattr" },
-    { 238, "tkill" },
-    { 239, "sendfile64" },
-    { 240, "futex" },
-    { 241, "sched_setaffinity", ignoreFunc },
-    { 242, "sched_getaffinity", ignoreFunc },
-    { 243, "set_thread_area", setThreadArea32Func },
-    { 244, "get_thread_area" },
-    { 245, "io_setup" },
-    { 246, "io_destroy" },
-    { 247, "io_getevents" },
-    { 248, "io_submit" },
-    { 249, "io_cancel" },
-    { 250, "fadvise64" },
-    { 251, "unused" },
-    { 252, "exit_group", exitFunc },
-    { 253, "lookup_dcookie" },
-    { 254, "epoll_create" },
-    { 255, "epoll_ctl" },
-    { 256, "epoll_wait" },
-    { 257, "remap_file_pages" },
-    { 258, "set_tid_address", setTidAddressFunc },
-    { 259, "timer_create" },
-    { 260, "timer_settime" },
-    { 261, "timer_gettime" },
-    { 262, "timer_getoverrun" },
-    { 263, "timer_delete" },
-    { 264, "clock_settime" },
-    { 265, "clock_gettime", clock_gettimeFunc<X86Linux32> },
-    { 266, "clock_getres" },
-    { 267, "clock_nanosleep" },
-    { 268, "statfs64" },
-    { 269, "fstatfs64" },
-    { 270, "tgkill", tgkillFunc<X86Linux32> },
-    { 271, "utimes" },
-    { 272, "fadvise64_64" },
-    { 273, "vserver" },
-    { 274, "mbind" },
-    { 275, "get_mempolicy", ignoreFunc },
-    { 276, "set_mempolicy" },
-    { 277, "mq_open" },
-    { 278, "mq_unlink" },
-    { 279, "mq_timedsend" },
-    { 280, "mq_timedreceive" },
-    { 281, "mq_notify" },
-    { 282, "mq_getsetattr" },
-    { 283, "kexec_load" },
-    { 284, "waitid" },
-    { 285, "sys_setaltroot" },
-    { 286, "add_key" },
-    { 287, "request_key" },
-    { 288, "keyctl" },
-    { 289, "ioprio_set" },
-    { 290, "ioprio_get" },
-    { 291, "inotify_init" },
-    { 292, "inotify_add_watch" },
-    { 293, "inotify_rm_watch" },
-    { 294, "migrate_pages" },
-    { 295, "openat", openatFunc<X86Linux32> },
-    { 296, "mkdirat" },
-    { 297, "mknodat" },
-    { 298, "fchownat" },
-    { 299, "futimesat" },
-    { 300, "fstatat64" },
-    { 301, "unlinkat" },
-    { 302, "renameat" },
-    { 303, "linkat" },
-    { 304, "symlinkat" },
-    { 305, "readlinkat", readlinkFunc },
-    { 306, "fchmodat" },
-    { 307, "faccessat" },
-    { 308, "pselect6" },
-    { 309, "ppoll" },
-    { 310, "unshare" },
-    { 311, "set_robust_list", ignoreFunc },
-    { 312, "get_robust_list", ignoreFunc },
-    { 313, "splice" },
-    { 314, "sync_file_range" },
-    { 315, "tee" },
-    { 316, "vmsplice" },
-    { 317, "move_pages" },
-    { 318, "getcpu", getcpuFunc },
-    { 319, "epoll_pwait" },
-    { 320, "utimensat" },
-    { 321, "signalfd" },
-    { 322, "timerfd" },
-    { 323, "eventfd", eventfdFunc<X86Linux32> }
-};
-
-void
-I386LinuxProcess::syscall(ThreadContext *tc)
-{
-    I386Process::syscall(tc);
-    PCState pc = tc->pcState();
-    Addr eip = pc.pc();
-    if (eip >= vsyscallPage.base &&
-            eip < vsyscallPage.base + vsyscallPage.size) {
-        pc.npc(vsyscallPage.base + vsyscallPage.vsysexitOffset);
-        tc->pcState(pc);
-    }
-    syscallDescs32.get(tc->readIntReg(INTREG_RAX))->doSyscall(tc);
-}
-
-void
-I386LinuxProcess::clone(ThreadContext *old_tc, ThreadContext *new_tc,
-                        Process *process, RegVal flags)
-{
-    I386Process::clone(old_tc, new_tc, (I386Process*)process, flags);
-}
diff --git a/src/arch/x86/linux/process.hh b/src/arch/x86/linux/process.hh
index 06b669235..4aa6e11d6 100644
--- a/src/arch/x86/linux/process.hh
+++ b/src/arch/x86/linux/process.hh
@@ -53,53 +53,14 @@ class X86_64LinuxProcess : public X86_64Process
 {
   public:
     using X86_64Process::X86_64Process;
-    void syscall(ThreadContext *tc) override;
-    void clone(ThreadContext *old_tc, ThreadContext *new_tc, Process *process,
-               RegVal flags) override;
-
-    struct SyscallABI : public GenericSyscallABI64, public X86Linux::SyscallABI
-    {
-        static const std::vector<IntRegIndex> ArgumentRegs;
-    };
 };
 
 class I386LinuxProcess : public I386Process
 {
   public:
     using I386Process::I386Process;
-    void syscall(ThreadContext *tc) override;
-    void clone(ThreadContext *old_tc, ThreadContext *new_tc, Process *process,
-               RegVal flags) override;
-
-    struct SyscallABI : public GenericSyscallABI32, public X86Linux::SyscallABI
-    {
-        static const std::vector<IntRegIndex> ArgumentRegs;
-    };
 };
 
 } // namespace X86ISA
 
-namespace GuestABI
-{
-
-template <typename Arg>
-struct Argument<X86ISA::I386LinuxProcess::SyscallABI, Arg,
-    typename std::enable_if<
-        X86ISA::I386LinuxProcess::SyscallABI::IsWide<Arg>::value>::type>
-{
-    using ABI = X86ISA::I386LinuxProcess::SyscallABI;
-
-    static Arg
-    get(ThreadContext *tc, typename ABI::State &state)
-    {
-        panic_if(state + 1 >= ABI::ArgumentRegs.size(),
-                "Ran out of syscall argument registers.");
-        auto low = ABI::ArgumentRegs[state++];
-        auto high = ABI::ArgumentRegs[state++];
-        return (Arg)ABI::mergeRegs(tc, low, high);
-    }
-};
-
-} // namespace GuestABI
-
 #endif // __X86_LINUX_PROCESS_HH__
diff --git a/src/arch/x86/linux/se_workload.cc b/src/arch/x86/linux/se_workload.cc
index 2f6dd6397..6f4384a98 100644
--- a/src/arch/x86/linux/se_workload.cc
+++ b/src/arch/x86/linux/se_workload.cc
@@ -27,8 +27,18 @@
 
 #include "arch/x86/linux/se_workload.hh"
 
+#include <sys/syscall.h>
+
+#include "arch/x86/isa_traits.hh"
+#include "arch/x86/linux/linux.hh"
+#include "arch/x86/linux/process.hh"
+#include "arch/x86/registers.hh"
+#include "base/trace.hh"
 #include "cpu/thread_context.hh"
+#include "kern/linux/linux.hh"
 #include "sim/process.hh"
+#include "sim/syscall_desc.hh"
+#include "sim/syscall_emul.hh"
 
 namespace X86ISA
 {
@@ -36,10 +46,842 @@ namespace X86ISA
 EmuLinux::EmuLinux(const Params &p) : SEWorkload(p), _params(p)
 {}
 
+const std::vector<IntRegIndex> EmuLinux::SyscallABI64::ArgumentRegs = {
+    INTREG_RDI, INTREG_RSI, INTREG_RDX, INTREG_R10W, INTREG_R8W, INTREG_R9W
+};
+
+const std::vector<IntRegIndex> EmuLinux::SyscallABI32::ArgumentRegs = {
+    INTREG_EBX, INTREG_ECX, INTREG_EDX, INTREG_ESI, INTREG_EDI, INTREG_EBP
+};
+
+/// Target uname() handler.
+static SyscallReturn
+unameFunc(SyscallDesc *desc, ThreadContext *tc, VPtr<Linux::utsname> name)
+{
+    auto process = tc->getProcessPtr();
+
+    strcpy(name->sysname, "Linux");
+    strcpy(name->nodename, "sim.gem5.org");
+    strcpy(name->release, process->release.c_str());
+    strcpy(name->version, "#1 Mon Aug 18 11:32:15 EDT 2003");
+    strcpy(name->machine, "x86_64");
+
+    return 0;
+}
+
+static SyscallReturn
+archPrctlFunc(SyscallDesc *desc, ThreadContext *tc, int code, uint64_t addr)
+{
+    enum ArchPrctlCodes
+    {
+        SetFS = 0x1002,
+        GetFS = 0x1003,
+        SetGS = 0x1001,
+        GetGS = 0x1004
+    };
+
+    uint64_t fsBase, gsBase;
+    PortProxy &p = tc->getVirtProxy();
+    switch(code)
+    {
+      // Each of these valid options should actually check addr.
+      case SetFS:
+        tc->setMiscRegNoEffect(MISCREG_FS_BASE, addr);
+        tc->setMiscRegNoEffect(MISCREG_FS_EFF_BASE, addr);
+        return 0;
+      case GetFS:
+        fsBase = tc->readMiscRegNoEffect(MISCREG_FS_BASE);
+        p.write(addr, fsBase);
+        return 0;
+      case SetGS:
+        tc->setMiscRegNoEffect(MISCREG_GS_BASE, addr);
+        tc->setMiscRegNoEffect(MISCREG_GS_EFF_BASE, addr);
+        return 0;
+      case GetGS:
+        gsBase = tc->readMiscRegNoEffect(MISCREG_GS_BASE);
+        p.write(addr, gsBase);
+        return 0;
+      default:
+        return -EINVAL;
+    }
+}
+
+BitUnion32(UserDescFlags)
+    Bitfield<0> seg_32bit;
+    Bitfield<2, 1> contents;
+    Bitfield<3> read_exec_only;
+    Bitfield<4> limit_in_pages;
+    Bitfield<5> seg_not_present;
+    Bitfield<6> useable;
+EndBitUnion(UserDescFlags)
+
+struct UserDesc32 {
+    uint32_t entry_number;
+    uint32_t base_addr;
+    uint32_t limit;
+    uint32_t flags;
+};
+
+struct UserDesc64 {
+    uint32_t entry_number;
+    uint32_t __padding1;
+    uint64_t base_addr;
+    uint32_t limit;
+    uint32_t flags;
+};
+
+static SyscallReturn
+setThreadArea32Func(SyscallDesc *desc, ThreadContext *tc,
+                    VPtr<UserDesc32> userDesc)
+{
+    const int minTLSEntry = 6;
+    const int numTLSEntries = 3;
+    const int maxTLSEntry = minTLSEntry + numTLSEntries - 1;
+
+    auto process = tc->getProcessPtr();
+
+    X86Process *x86p = dynamic_cast<X86Process *>(process);
+    assert(x86p);
+
+    assert((maxTLSEntry + 1) * sizeof(uint64_t) <= x86p->gdtSize());
+
+    TypedBufferArg<uint64_t>
+        gdt(x86p->gdtStart() + minTLSEntry * sizeof(uint64_t),
+            numTLSEntries * sizeof(uint64_t));
+
+    if (!gdt.copyIn(tc->getVirtProxy()))
+        panic("Failed to copy in GDT for %s.\n", desc->name());
+
+    if (userDesc->entry_number == (uint32_t)(-1)) {
+        // Find a free TLS entry.
+        for (int i = 0; i < numTLSEntries; i++) {
+            if (gdt[i] == 0) {
+                userDesc->entry_number = i + minTLSEntry;
+                break;
+            }
+        }
+        // We failed to find one.
+        if (userDesc->entry_number == (uint32_t)(-1))
+            return -ESRCH;
+    }
+
+    int index = userDesc->entry_number;
+
+    if (index < minTLSEntry || index > maxTLSEntry)
+        return -EINVAL;
+
+    index -= minTLSEntry;
+
+    // Build the entry we're going to add.
+    SegDescriptor segDesc = 0;
+    UserDescFlags flags = userDesc->flags;
+
+    segDesc.limitLow = bits(userDesc->limit, 15, 0);
+    segDesc.baseLow = bits(userDesc->base_addr, 23, 0);
+    segDesc.type.a = 1;
+    if (!flags.read_exec_only)
+        segDesc.type.w = 1;
+    if (bits((uint8_t)flags.contents, 0))
+        segDesc.type.e = 1;
+    if (bits((uint8_t)flags.contents, 1))
+        segDesc.type.codeOrData = 1;
+    segDesc.s = 1;
+    segDesc.dpl = 3;
+    if (!flags.seg_not_present)
+        segDesc.p = 1;
+    segDesc.limitHigh = bits(userDesc->limit, 19, 16);
+    if (flags.useable)
+        segDesc.avl = 1;
+    segDesc.l = 0;
+    if (flags.seg_32bit)
+        segDesc.d = 1;
+    if (flags.limit_in_pages)
+        segDesc.g = 1;
+    segDesc.baseHigh = bits(userDesc->base_addr, 31, 24);
+
+    gdt[index] = (uint64_t)segDesc;
+
+    if (!gdt.copyOut(tc->getVirtProxy()))
+        panic("Failed to copy out GDT for %s.\n", desc->name());
+
+    return 0;
+}
+
+static SyscallDescTable<EmuLinux::SyscallABI64> syscallDescs64 = {
+    {   0, "read", readFunc<X86Linux64> },
+    {   1, "write", writeFunc<X86Linux64> },
+    {   2, "open", openFunc<X86Linux64> },
+    {   3, "close", closeFunc },
+    {   4, "stat", stat64Func<X86Linux64> },
+    {   5, "fstat", fstat64Func<X86Linux64> },
+    {   6, "lstat", lstat64Func<X86Linux64> },
+    {   7, "poll", pollFunc<X86Linux64> },
+    {   8, "lseek", lseekFunc },
+    {   9, "mmap", mmapFunc<X86Linux64> },
+    {  10, "mprotect", ignoreFunc },
+    {  11, "munmap", munmapFunc },
+    {  12, "brk", brkFunc },
+    {  13, "rt_sigaction", ignoreWarnOnceFunc },
+    {  14, "rt_sigprocmask", ignoreWarnOnceFunc },
+    {  15, "rt_sigreturn" },
+    {  16, "ioctl", ioctlFunc<X86Linux64> },
+    {  17, "pread64", pread64Func<X86Linux64> },
+    {  18, "pwrite64", pwrite64Func<X86Linux64> },
+    {  19, "readv", readvFunc<X86Linux64> },
+    {  20, "writev", writevFunc<X86Linux64> },
+    {  21, "access", ignoreFunc },
+    {  22, "pipe", pipeFunc },
+    {  23, "select", selectFunc<X86Linux64> },
+    {  24, "sched_yield", ignoreFunc },
+    {  25, "mremap", mremapFunc<X86Linux64> },
+    {  26, "msync" },
+    {  27, "mincore" },
+    {  28, "madvise", ignoreFunc },
+    {  29, "shmget" },
+    {  30, "shmat" },
+    {  31, "shmctl" },
+    {  32, "dup", dupFunc },
+    {  33, "dup2", dup2Func },
+    {  34, "pause" },
+    {  35, "nanosleep", ignoreWarnOnceFunc },
+    {  36, "getitimer" },
+    {  37, "alarm" },
+    {  38, "setitimer" },
+    {  39, "getpid", getpidFunc },
+    {  40, "sendfile" },
+    {  41, "socket", socketFunc<X86Linux64> },
+    {  42, "connect", connectFunc },
+    {  43, "accept", acceptFunc<X86Linux64> },
+    {  44, "sendto", sendtoFunc },
+    {  45, "recvfrom", recvfromFunc },
+    {  46, "sendmsg", sendmsgFunc },
+    {  47, "recvmsg", recvmsgFunc },
+    {  48, "shutdown", shutdownFunc },
+    {  49, "bind", bindFunc },
+    {  50, "listen", listenFunc },
+    {  51, "getsockname", getsocknameFunc },
+    {  52, "getpeername", getpeernameFunc },
+    {  53, "socketpair", socketpairFunc<X86Linux64> },
+    {  54, "setsockopt", setsockoptFunc },
+    {  55, "getsockopt", getsockoptFunc },
+    {  56, "clone", cloneFunc<X86Linux64> },
+    {  57, "fork" },
+    {  58, "vfork" },
+    {  59, "execve", execveFunc<X86Linux64> },
+    {  60, "exit", exitFunc },
+    {  61, "wait4", wait4Func<X86Linux64> },
+    {  62, "kill" },
+    {  63, "uname", unameFunc },
+    {  64, "semget" },
+    {  65, "semop" },
+    {  66, "semctl" },
+    {  67, "shmdt" },
+    {  68, "msgget" },
+    {  69, "msgsnd" },
+    {  70, "msgrcv" },
+    {  71, "msgctl" },
+    {  72, "fcntl", fcntlFunc },
+    {  73, "flock" },
+    {  74, "fsync" },
+    {  75, "fdatasync" },
+    {  76, "truncate", truncateFunc },
+    {  77, "ftruncate", ftruncateFunc },
+#if defined(SYS_getdents)
+    {  78, "getdents", getdentsFunc },
+#else
+    {  78, "getdents" },
+#endif
+    {  79, "getcwd", getcwdFunc },
+    {  80, "chdir", chdirFunc },
+    {  81, "fchdir" },
+    {  82, "rename", renameFunc },
+    {  83, "mkdir", mkdirFunc },
+    {  84, "rmdir", rmdirFunc },
+    {  85, "creat" },
+    {  86, "link", linkFunc },
+    {  87, "unlink", unlinkFunc },
+    {  88, "symlink", symlinkFunc },
+    {  89, "readlink", readlinkFunc },
+    {  90, "chmod", ignoreFunc },
+    {  91, "fchmod" },
+    {  92, "chown" },
+    {  93, "fchown" },
+    {  94, "lchown" },
+    {  95, "umask", umaskFunc },
+    {  96, "gettimeofday", gettimeofdayFunc<X86Linux64> },
+    {  97, "getrlimit", getrlimitFunc<X86Linux64> },
+    {  98, "getrusage", getrusageFunc<X86Linux64> },
+    {  99, "sysinfo", sysinfoFunc<X86Linux64> },
+    { 100, "times", timesFunc<X86Linux64> },
+    { 101, "ptrace" },
+    { 102, "getuid", getuidFunc },
+    { 103, "syslog" },
+    { 104, "getgid", getgidFunc },
+    { 105, "setuid" },
+    { 106, "setgid" },
+    { 107, "geteuid", geteuidFunc },
+    { 108, "getegid", getegidFunc },
+    { 109, "setpgid", setpgidFunc },
+    { 110, "getppid", getppidFunc },
+    { 111, "getpgrp", getpgrpFunc },
+    { 112, "setsid" },
+    { 113, "setreuid" },
+    { 114, "setregid" },
+    { 115, "getgroups" },
+    { 116, "setgroups" },
+    { 117, "setresuid", ignoreFunc },
+    { 118, "getresuid" },
+    { 119, "setresgid" },
+    { 120, "getresgid" },
+    { 121, "getpgid" },
+    { 122, "setfsuid" },
+    { 123, "setfsgid" },
+    { 124, "getsid" },
+    { 125, "capget" },
+    { 126, "capset" },
+    { 127, "rt_sigpending" },
+    { 128, "rt_sigtimedwait" },
+    { 129, "rt_sigqueueinfo" },
+    { 130, "rt_sigsuspend" },
+    { 131, "sigaltstack" },
+    { 132, "utime" },
+    { 133, "mknod", mknodFunc },
+    { 134, "uselib" },
+    { 135, "personality" },
+    { 136, "ustat" },
+    { 137, "statfs", statfsFunc<X86Linux64> },
+    { 138, "fstatfs", fstatfsFunc<X86Linux64> },
+    { 139, "sysfs" },
+    { 140, "getpriority" },
+    { 141, "setpriority", ignoreFunc },
+    { 142, "sched_setparam" },
+    { 143, "sched_getparam" },
+    { 144, "sched_setscheduler" },
+    { 145, "sched_getscheduler" },
+    { 146, "sched_get_priority_max" },
+    { 147, "sched_get_priority_min" },
+    { 148, "sched_rr_get_interval" },
+    { 149, "mlock" },
+    { 150, "munlock" },
+    { 151, "mlockall" },
+    { 152, "munlockall" },
+    { 153, "vhangup" },
+    { 154, "modify_ldt" },
+    { 155, "pivot_root" },
+    { 156, "_sysctl" },
+    { 157, "prctl" },
+    { 158, "arch_prctl", archPrctlFunc },
+    { 159, "adjtimex" },
+    { 160, "setrlimit", ignoreFunc },
+    { 161, "chroot" },
+    { 162, "sync" },
+    { 163, "acct" },
+    { 164, "settimeofday" },
+    { 165, "mount" },
+    { 166, "umount2" },
+    { 167, "swapon" },
+    { 168, "swapoff" },
+    { 169, "reboot" },
+    { 170, "sethostname" },
+    { 171, "setdomainname" },
+    { 172, "iopl" },
+    { 173, "ioperm" },
+    { 174, "create_module" },
+    { 175, "init_module" },
+    { 176, "delete_module" },
+    { 177, "get_kernel_syms" },
+    { 178, "query_module" },
+    { 179, "quotactl" },
+    { 180, "nfsservctl" },
+    { 181, "getpmsg" },
+    { 182, "putpmsg" },
+    { 183, "afs_syscall" },
+    { 184, "tuxcall" },
+    { 185, "security" },
+    { 186, "gettid", gettidFunc },
+    { 187, "readahead" },
+    { 188, "setxattr" },
+    { 189, "lsetxattr" },
+    { 190, "fsetxattr" },
+    { 191, "getxattr" },
+    { 192, "lgetxattr" },
+    { 193, "fgetxattr" },
+    { 194, "listxattr" },
+    { 195, "llistxattr" },
+    { 196, "flistxattr" },
+    { 197, "removexattr" },
+    { 198, "lremovexattr" },
+    { 199, "fremovexattr" },
+    { 200, "tkill" },
+    { 201, "time", timeFunc<X86Linux64> },
+    { 202, "futex", futexFunc<X86Linux64> },
+    { 203, "sched_setaffinity", ignoreFunc },
+    { 204, "sched_getaffinity", ignoreFunc },
+    { 205, "set_thread_area" },
+    { 206, "io_setup" },
+    { 207, "io_destroy" },
+    { 208, "io_getevents" },
+    { 209, "io_submit" },
+    { 210, "io_cancel" },
+    { 211, "get_thread_area" },
+    { 212, "lookup_dcookie" },
+    { 213, "epoll_create" },
+    { 214, "epoll_ctl_old" },
+    { 215, "epoll_wait_old" },
+    { 216, "remap_file_pages" },
+    { 217, "getdents64" },
+    { 218, "set_tid_address", setTidAddressFunc },
+    { 219, "restart_syscall" },
+    { 220, "semtimedop" },
+    { 221, "fadvise64", ignoreFunc },
+    { 222, "timer_create" },
+    { 223, "timer_settime" },
+    { 224, "timer_gettime" },
+    { 225, "timer_getoverrun" },
+    { 226, "timer_delete" },
+    { 227, "clock_settime" },
+    { 228, "clock_gettime", clock_gettimeFunc<X86Linux64> },
+    { 229, "clock_getres", clock_getresFunc<X86Linux64> },
+    { 230, "clock_nanosleep" },
+    { 231, "exit_group", exitGroupFunc },
+    { 232, "epoll_wait" },
+    { 233, "epoll_ctl" },
+    { 234, "tgkill", tgkillFunc<X86Linux64> },
+    { 235, "utimes" },
+    { 236, "vserver" },
+    { 237, "mbind" },
+    { 238, "set_mempolicy" },
+    { 239, "get_mempolicy", ignoreFunc },
+    { 240, "mq_open" },
+    { 241, "mq_unlink" },
+    { 242, "mq_timedsend" },
+    { 243, "mq_timedreceive" },
+    { 244, "mq_notify" },
+    { 245, "mq_getsetattr" },
+    { 246, "kexec_load" },
+    { 247, "waitid" },
+    { 248, "add_key" },
+    { 249, "request_key" },
+    { 250, "keyctl" },
+    { 251, "ioprio_set" },
+    { 252, "ioprio_get" },
+    { 253, "inotify_init" },
+    { 254, "inotify_add_watch" },
+    { 255, "inotify_rm_watch" },
+    { 256, "migrate_pages" },
+    { 257, "openat", openatFunc<X86Linux64> },
+    { 258, "mkdirat" },
+    { 259, "mknodat" },
+    { 260, "fchownat" },
+    { 261, "futimesat" },
+    { 262, "newfstatat" },
+    { 263, "unlinkat" },
+    { 264, "renameat" },
+    { 265, "linkat" },
+    { 266, "symlinkat" },
+    { 267, "readlinkat", readlinkFunc },
+    { 268, "fchmodat" },
+    { 269, "faccessat" },
+    { 270, "pselect6" },
+    { 271, "ppoll" },
+    { 272, "unshare" },
+    { 273, "set_robust_list", ignoreFunc },
+    { 274, "get_robust_list" },
+    { 275, "splice" },
+    { 276, "tee" },
+    { 277, "sync_file_range" },
+    { 278, "vmsplice" },
+    { 279, "move_pages" },
+    { 280, "utimensat" },
+    { 281, "epoll_pwait" },
+    { 282, "signalfd" },
+    { 283, "timerfd_create" },
+    { 284, "eventfd", eventfdFunc<X86Linux64> },
+    { 285, "fallocate", fallocateFunc },
+    { 286, "timerfd_settime" },
+    { 287, "timerfd_gettime" },
+    { 288, "accept4" },
+    { 289, "signalfd4" },
+    { 290, "eventfd2", eventfdFunc<X86Linux64> },
+    { 291, "epoll_create1" },
+    { 292, "dup3" },
+    { 293, "pipe2", pipe2Func },
+    { 294, "inotify_init1" },
+    { 295, "preadv" },
+    { 296, "pwritev" },
+    { 297, "rt_tgsigqueueinfo" },
+    { 298, "perf_event_open" },
+    { 299, "recvmmsg" },
+    { 300, "fanotify_init" },
+    { 301, "fanotify_mark" },
+    { 302, "prlimit64", prlimitFunc<X86Linux64> },
+    { 303, "name_to_handle_at" },
+    { 304, "open_by_handle_at" },
+    { 305, "clock_adjtime" },
+    { 306, "syncfs" },
+    { 307, "sendmmsg" },
+    { 308, "setns" },
+    { 309, "getcpu", getcpuFunc },
+    { 310, "proess_vm_readv" },
+    { 311, "proess_vm_writev" },
+    { 312, "kcmp" },
+    { 313, "finit_module" },
+};
+
+static SyscallDescTable<EmuLinux::SyscallABI32> syscallDescs32 = {
+    {   0, "restart_syscall" },
+    {   1, "exit", exitFunc },
+    {   2, "fork" },
+    {   3, "read", readFunc<X86Linux32> },
+    {   4, "write", writeFunc<X86Linux32> },
+    {   5, "open", openFunc<X86Linux32> },
+    {   6, "close", closeFunc },
+    {   7, "waitpid" },
+    {   8, "creat" },
+    {   9, "link" },
+    {  10, "unlink" },
+    {  11, "execve", execveFunc<X86Linux32> },
+    {  12, "chdir", chdirFunc },
+    {  13, "time", timeFunc<X86Linux32> },
+    {  14, "mknod", mknodFunc },
+    {  15, "chmod" },
+    {  16, "lchown" },
+    {  17, "break" },
+    {  18, "oldstat" },
+    {  19, "lseek" },
+    {  20, "getpid", getpidFunc },
+    {  21, "mount" },
+    {  22, "umount" },
+    {  23, "setuid" },
+    {  24, "getuid", getuidFunc },
+    {  25, "stime" },
+    {  26, "ptrace" },
+    {  27, "alarm" },
+    {  28, "oldfstat" },
+    {  29, "pause" },
+    {  30, "utime" },
+    {  31, "stty" },
+    {  32, "gtty" },
+    {  33, "access", ignoreFunc },
+    {  34, "nice" },
+    {  35, "ftime" },
+    {  36, "sync" },
+    {  37, "kill" },
+    {  38, "rename" },
+    {  39, "mkdir", mkdirFunc },
+    {  40, "rmdir", mkdirFunc },
+    {  41, "dup", dupFunc },
+    {  42, "pipe", pipeFunc },
+    {  43, "times", timesFunc<X86Linux32> },
+    {  44, "prof" },
+    {  45, "brk", brkFunc },
+    {  46, "setgid" },
+    {  47, "getgid", getgidFunc },
+    {  48, "signal" },
+    {  49, "geteuid", geteuidFunc },
+    {  50, "getegid", getegidFunc },
+    {  51, "acct" },
+    {  52, "umount2" },
+    {  53, "lock" },
+    {  54, "ioctl", ioctlFunc<X86Linux32> },
+    {  55, "fcntl", fcntlFunc },
+    {  56, "mpx" },
+    {  57, "setpgid", setpgidFunc },
+    {  58, "ulimit" },
+    {  59, "oldolduname" },
+    {  60, "umask", umaskFunc },
+    {  61, "chroot" },
+    {  62, "ustat" },
+    {  63, "dup2", dup2Func },
+    {  64, "getppid" },
+    {  65, "getpgrp" },
+    {  66, "setsid" },
+    {  67, "sigaction" },
+    {  68, "sgetmask" },
+    {  69, "ssetmask" },
+    {  70, "setreuid" },
+    {  71, "setregid" },
+    {  72, "sigsuspend" },
+    {  73, "sigpending" },
+    {  74, "sethostname" },
+    {  75, "setrlimit", ignoreFunc },
+    {  76, "getrlimit", getrlimitFunc<X86Linux32> },
+    {  77, "getrusage", getrusageFunc<X86Linux32> },
+    {  78, "gettimeofday" },
+    {  79, "settimeofday" },
+    {  80, "getgroups" },
+    {  81, "setgroups" },
+    {  82, "select", selectFunc<X86Linux32> },
+    {  83, "symlink" },
+    {  84, "oldlstat" },
+    {  85, "readlink", readlinkFunc },
+    {  86, "uselib" },
+    {  87, "swapon" },
+    {  88, "reboot" },
+    {  89, "readdir" },
+    {  90, "mmap" },
+    {  91, "munmap", munmapFunc },
+    {  92, "truncate", truncateFunc },
+    {  93, "ftruncate", ftruncateFunc },
+    {  94, "fchmod" },
+    {  95, "fchown" },
+    {  96, "getpriority" },
+    {  97, "setpriority", ignoreFunc },
+    {  98, "profil" },
+    {  99, "statfs", ignoreFunc },
+    { 100, "fstatfs" },
+    { 101, "ioperm" },
+    { 102, "socketcall" },
+    { 103, "syslog" },
+    { 104, "setitimer" },
+    { 105, "getitimer" },
+    { 106, "stat" },
+    { 107, "lstat" },
+    { 108, "fstat" },
+    { 109, "olduname" },
+    { 110, "iopl" },
+    { 111, "vhangup" },
+    { 112, "idle" },
+    { 113, "vm86old" },
+    { 114, "wait4", wait4Func<X86Linux32> },
+    { 115, "swapoff" },
+    { 116, "sysinfo", sysinfoFunc<X86Linux32> },
+    { 117, "ipc" },
+    { 118, "fsync" },
+    { 119, "sigreturn" },
+    { 120, "clone", cloneFunc<X86Linux32> },
+    { 121, "setdomainname" },
+    { 122, "uname", unameFunc },
+    { 123, "modify_ldt" },
+    { 124, "adjtimex" },
+    { 125, "mprotect", ignoreFunc },
+    { 126, "sigprocmask" },
+    { 127, "create_module" },
+    { 128, "init_module" },
+    { 129, "delete_module" },
+    { 130, "get_kernel_syms" },
+    { 131, "quotactl" },
+    { 132, "getpgid" },
+    { 133, "fchdir" },
+    { 134, "bdflush" },
+    { 135, "sysfs" },
+    { 136, "personality" },
+    { 137, "afs_syscall" },
+    { 138, "setfsuid" },
+    { 139, "setfsgid" },
+    { 140, "_llseek", _llseekFunc },
+#if defined(SYS_getdents)
+    { 141, "getdents", getdentsFunc },
+#else
+    { 141, "getdents" },
+#endif
+    { 142, "_newselect" },
+    { 143, "flock" },
+    { 144, "msync" },
+    { 145, "readv", readvFunc<X86Linux32> },
+    { 146, "writev", writevFunc<X86Linux32> },
+    { 147, "getsid" },
+    { 148, "fdatasync" },
+    { 149, "_sysctl" },
+    { 150, "mlock" },
+    { 151, "munlock" },
+    { 152, "mlockall" },
+    { 153, "munlockall" },
+    { 154, "sched_setparam" },
+    { 155, "sched_getparam" },
+    { 156, "sched_setscheduler" },
+    { 157, "sched_getscheduler" },
+    { 158, "sched_yield", ignoreFunc },
+    { 159, "sched_get_priority_max" },
+    { 160, "sched_get_priority_min" },
+    { 161, "sched_rr_get_interval" },
+    { 162, "nanosleep", ignoreFunc },
+    { 163, "mremap" },
+    { 164, "setresuid", ignoreFunc },
+    { 165, "getresuid" },
+    { 166, "vm86" },
+    { 167, "query_module" },
+    { 168, "poll", pollFunc<X86Linux32> },
+    { 169, "nfsservctl" },
+    { 170, "setresgid" },
+    { 171, "getresgid" },
+    { 172, "prctl" },
+    { 173, "rt_sigreturn" },
+    { 174, "rt_sigaction", ignoreFunc },
+    { 175, "rt_sigprocmask", ignoreFunc },
+    { 176, "rt_sigpending" },
+    { 177, "rt_sigtimedwait" },
+    { 178, "rt_sigqueueinfo" },
+    { 179, "rt_sigsuspend" },
+    { 180, "pread64", pread64Func<X86Linux64> },
+    { 181, "pwrite64", pwrite64Func<X86Linux64> },
+    { 182, "chown" },
+    { 183, "getcwd", getcwdFunc },
+    { 184, "capget" },
+    { 185, "capset" },
+    { 186, "sigaltstack" },
+    { 187, "sendfile" },
+    { 188, "getpmsg" },
+    { 189, "putpmsg" },
+    { 190, "vfork" },
+    { 191, "ugetrlimit", ignoreFunc },
+    { 192, "mmap2", mmap2Func<X86Linux32> },
+    { 193, "truncate64", truncate64Func },
+    { 194, "ftruncate64", ftruncate64Func },
+    { 195, "stat64", stat64Func<X86Linux32> },
+    { 196, "lstat64" },
+    { 197, "fstat64", fstat64Func<X86Linux32> },
+    { 198, "lchown32" },
+    { 199, "getuid32", getuidFunc },
+    { 200, "getgid32", getgidFunc },
+    { 201, "geteuid32", geteuidFunc },
+    { 202, "getegid32", getegidFunc },
+    { 203, "setreuid32" },
+    { 204, "setregid32" },
+    { 205, "getgroups32" },
+    { 206, "setgroups32" },
+    { 207, "fchown32" },
+    { 208, "setresuid32" },
+    { 209, "getresuid32" },
+    { 210, "setresgid32" },
+    { 211, "getresgid32" },
+    { 212, "chown32" },
+    { 213, "setuid32" },
+    { 214, "setgid32" },
+    { 215, "setfsuid32" },
+    { 216, "setfsgid32" },
+    { 217, "pivot_root" },
+    { 218, "mincore" },
+    { 219, "madvise", ignoreFunc },
+    { 220, "madvise1" },
+    { 221, "getdents64" },
+    { 222, "fcntl64" },
+    { 223, "unused" },
+    { 224, "gettid", gettidFunc },
+    { 225, "readahead" },
+    { 226, "setxattr" },
+    { 227, "lsetxattr" },
+    { 228, "fsetxattr" },
+    { 229, "getxattr" },
+    { 230, "lgetxattr" },
+    { 231, "fgetxattr" },
+    { 232, "listxattr" },
+    { 233, "llistxattr" },
+    { 234, "flistxattr" },
+    { 235, "removexattr" },
+    { 236, "lremovexattr" },
+    { 237, "fremovexattr" },
+    { 238, "tkill" },
+    { 239, "sendfile64" },
+    { 240, "futex" },
+    { 241, "sched_setaffinity", ignoreFunc },
+    { 242, "sched_getaffinity", ignoreFunc },
+    { 243, "set_thread_area", setThreadArea32Func },
+    { 244, "get_thread_area" },
+    { 245, "io_setup" },
+    { 246, "io_destroy" },
+    { 247, "io_getevents" },
+    { 248, "io_submit" },
+    { 249, "io_cancel" },
+    { 250, "fadvise64" },
+    { 251, "unused" },
+    { 252, "exit_group", exitFunc },
+    { 253, "lookup_dcookie" },
+    { 254, "epoll_create" },
+    { 255, "epoll_ctl" },
+    { 256, "epoll_wait" },
+    { 257, "remap_file_pages" },
+    { 258, "set_tid_address", setTidAddressFunc },
+    { 259, "timer_create" },
+    { 260, "timer_settime" },
+    { 261, "timer_gettime" },
+    { 262, "timer_getoverrun" },
+    { 263, "timer_delete" },
+    { 264, "clock_settime" },
+    { 265, "clock_gettime", clock_gettimeFunc<X86Linux32> },
+    { 266, "clock_getres" },
+    { 267, "clock_nanosleep" },
+    { 268, "statfs64" },
+    { 269, "fstatfs64" },
+    { 270, "tgkill", tgkillFunc<X86Linux32> },
+    { 271, "utimes" },
+    { 272, "fadvise64_64" },
+    { 273, "vserver" },
+    { 274, "mbind" },
+    { 275, "get_mempolicy", ignoreFunc },
+    { 276, "set_mempolicy" },
+    { 277, "mq_open" },
+    { 278, "mq_unlink" },
+    { 279, "mq_timedsend" },
+    { 280, "mq_timedreceive" },
+    { 281, "mq_notify" },
+    { 282, "mq_getsetattr" },
+    { 283, "kexec_load" },
+    { 284, "waitid" },
+    { 285, "sys_setaltroot" },
+    { 286, "add_key" },
+    { 287, "request_key" },
+    { 288, "keyctl" },
+    { 289, "ioprio_set" },
+    { 290, "ioprio_get" },
+    { 291, "inotify_init" },
+    { 292, "inotify_add_watch" },
+    { 293, "inotify_rm_watch" },
+    { 294, "migrate_pages" },
+    { 295, "openat", openatFunc<X86Linux32> },
+    { 296, "mkdirat" },
+    { 297, "mknodat" },
+    { 298, "fchownat" },
+    { 299, "futimesat" },
+    { 300, "fstatat64" },
+    { 301, "unlinkat" },
+    { 302, "renameat" },
+    { 303, "linkat" },
+    { 304, "symlinkat" },
+    { 305, "readlinkat", readlinkFunc },
+    { 306, "fchmodat" },
+    { 307, "faccessat" },
+    { 308, "pselect6" },
+    { 309, "ppoll" },
+    { 310, "unshare" },
+    { 311, "set_robust_list", ignoreFunc },
+    { 312, "get_robust_list", ignoreFunc },
+    { 313, "splice" },
+    { 314, "sync_file_range" },
+    { 315, "tee" },
+    { 316, "vmsplice" },
+    { 317, "move_pages" },
+    { 318, "getcpu", getcpuFunc },
+    { 319, "epoll_pwait" },
+    { 320, "utimensat" },
+    { 321, "signalfd" },
+    { 322, "timerfd" },
+    { 323, "eventfd", eventfdFunc<X86Linux32> }
+};
+
 void
 EmuLinux::syscall(ThreadContext *tc)
 {
-    tc->getProcessPtr()->syscall(tc);
+    Process *process = tc->getProcessPtr();
+    // Call the syscall function in the base Process class to update stats.
+    // This will move into the base SEWorkload function at some point.
+    process->Process::syscall(tc);
+
+    RegVal rax = tc->readIntReg(INTREG_RAX);
+    if (dynamic_cast<X86_64Process *>(process)) {
+        syscallDescs64.get(rax)->doSyscall(tc);
+    } else if (auto *proc32 = dynamic_cast<I386Process *>(process)) {
+        PCState pc = tc->pcState();
+        Addr eip = pc.pc();
+        const auto &vsyscall = proc32->getVSyscallPage();
+        if (eip >= vsyscall.base && eip < vsyscall.base + vsyscall.size) {
+            pc.npc(vsyscall.base + vsyscall.vsysexitOffset);
+            tc->pcState(pc);
+        }
+        syscallDescs32.get(rax)->doSyscall(tc);
+    } else {
+        panic("Unrecognized process type.");
+    }
 }
 
 } // namespace X86ISA
diff --git a/src/arch/x86/linux/se_workload.hh b/src/arch/x86/linux/se_workload.hh
index d1dca672e..efe84f3e9 100644
--- a/src/arch/x86/linux/se_workload.hh
+++ b/src/arch/x86/linux/se_workload.hh
@@ -28,8 +28,10 @@
 #ifndef __ARCH_X86_LINUX_SE_WORKLOAD_HH__
 #define __ARCH_X86_LINUX_SE_WORKLOAD_HH__
 
+#include "arch/x86/linux/linux.hh"
 #include "params/X86EmuLinux.hh"
 #include "sim/se_workload.hh"
+#include "sim/syscall_abi.hh"
 
 namespace X86ISA
 {
@@ -50,8 +52,43 @@ class EmuLinux : public SEWorkload
     Loader::Arch getArch() const override { return Loader::X86_64; }
 
     void syscall(ThreadContext *tc) override;
+
+    struct SyscallABI64 :
+        public GenericSyscallABI64, public X86Linux::SyscallABI
+    {
+        static const std::vector<IntRegIndex> ArgumentRegs;
+    };
+
+    struct SyscallABI32 :
+        public GenericSyscallABI32, public X86Linux::SyscallABI
+    {
+        static const std::vector<IntRegIndex> ArgumentRegs;
+    };
 };
 
 } // namespace X86ISA
 
+namespace GuestABI
+{
+
+template <typename Arg>
+struct Argument<X86ISA::EmuLinux::SyscallABI32, Arg,
+    typename std::enable_if<
+        X86ISA::EmuLinux::SyscallABI32::IsWide<Arg>::value>::type>
+{
+    using ABI = X86ISA::EmuLinux::SyscallABI32;
+
+    static Arg
+    get(ThreadContext *tc, typename ABI::State &state)
+    {
+        panic_if(state + 1 >= ABI::ArgumentRegs.size(),
+                "Ran out of syscall argument registers.");
+        auto low = ABI::ArgumentRegs[state++];
+        auto high = ABI::ArgumentRegs[state++];
+        return (Arg)ABI::mergeRegs(tc, low, high);
+    }
+};
+
+} // namespace GuestABI
+
 #endif // __ARCH_X86_LINUX_SE_WORKLOAD_HH__
diff --git a/src/arch/x86/process.hh b/src/arch/x86/process.hh
index a55212f40..7288aae05 100644
--- a/src/arch/x86/process.hh
+++ b/src/arch/x86/process.hh
@@ -159,6 +159,8 @@ namespace X86ISA
         I386Process(const ProcessParams &params,
                     ::Loader::ObjectFile *objFile);
 
+        const VSyscallPage &getVSyscallPage() { return vsyscallPage; }
+
         void argsInit(int pageSize);
         void initState() override;
 
-- 
2.30.2