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 ¶ms, ::Loader::ObjectFile *objFile); + const VSyscallPage &getVSyscallPage() { return vsyscallPage; } + void argsInit(int pageSize); void initState() override; -- 2.30.2