From c5d617d8a25e2397e3e66a40038d270260e0bfc4 Mon Sep 17 00:00:00 2001 From: Asahi Lina Date: Fri, 20 Dec 2024 01:55:17 +0900 Subject: [PATCH] FileManagement: Use openat2() with RESOLVE_IN_ROOT for RootFS open ops This avoids having to do the symlink chasing in GetEmulatedFDPath, since the kernel does it for us. On top of that, with a merged RootFS setup, this will correctly handle symlinks from user directories into the RootFS, fixing wine on Fedora. --- .../LinuxSyscalls/FileManagement.cpp | 53 ++++++++++++++++--- 1 file changed, 45 insertions(+), 8 deletions(-) diff --git a/Source/Tools/LinuxEmulation/LinuxSyscalls/FileManagement.cpp b/Source/Tools/LinuxEmulation/LinuxSyscalls/FileManagement.cpp index ff1d312128..14e2b2c2be 100644 --- a/Source/Tools/LinuxEmulation/LinuxSyscalls/FileManagement.cpp +++ b/Source/Tools/LinuxEmulation/LinuxSyscalls/FileManagement.cpp @@ -29,6 +29,7 @@ desc: Rootfs overlay logic #include #include #include +#include #include #include #include @@ -645,10 +646,23 @@ uint64_t FileManager::Open(const char* pathname, int flags, uint32_t mode) { if (!ShouldSkipOpenInEmu(flags)) { FDPathTmpData TmpFilename; - auto Path = GetEmulatedFDPath(AT_FDCWD, SelfPath, true, TmpFilename); + auto Path = GetEmulatedFDPath(AT_FDCWD, SelfPath, false, TmpFilename); if (Path.first != -1) { - fd = ::openat(Path.first, Path.second, flags, mode); - } else { + FEX::HLE::open_how how = { + .flags = (uint64_t)flags, + .mode = (flags & (O_CREAT | O_TMPFILE)) ? mode & 07777 : 0, // openat2() is stricter about this + .resolve = RESOLVE_IN_ROOT, + }; + fd = ::syscall(SYSCALL_DEF(openat2), Path.first, Path.second, &how, sizeof(how)); + if (fd < 0 && errno == EXDEV) { + // This means a magic symlink (/proc/foo) was involved. In this case we + // just punt and do the access without RESOLVE_IN_ROOT. + fd = ::syscall(SYSCALL_DEF(openat), Path.first, Path.second, flags, mode); + } + } + + // Open through RootFS failed (probably nonexistent), so open directly. + if (fd == -1) { fd = ::open(SelfPath, flags, mode); } @@ -885,10 +899,23 @@ uint64_t FileManager::Openat([[maybe_unused]] int dirfs, const char* pathname, i if (!ShouldSkipOpenInEmu(flags)) { FDPathTmpData TmpFilename; - auto Path = GetEmulatedFDPath(dirfs, SelfPath, true, TmpFilename); + auto Path = GetEmulatedFDPath(dirfs, SelfPath, false, TmpFilename); if (Path.first != -1) { - fd = ::syscall(SYSCALL_DEF(openat), Path.first, Path.second, flags, mode); - } else { + FEX::HLE::open_how how = { + .flags = (uint64_t)flags, + .mode = (flags & (O_CREAT | O_TMPFILE)) ? mode & 07777 : 0, // openat2() is stricter about this, + .resolve = RESOLVE_IN_ROOT, + }; + fd = ::syscall(SYSCALL_DEF(openat2), Path.first, Path.second, &how, sizeof(how)); + if (fd < 0 && errno == EXDEV) { + // This means a magic symlink (/proc/foo) was involved. In this case we + // just punt and do the access without RESOLVE_IN_ROOT. + fd = ::syscall(SYSCALL_DEF(openat), Path.first, Path.second, flags, mode); + } + } + + // Open through RootFS failed (probably nonexistent), so open directly. + if (fd == -1) { fd = ::syscall(SYSCALL_DEF(openat), dirfs, SelfPath, flags, mode); } @@ -908,10 +935,20 @@ uint64_t FileManager::Openat2(int dirfs, const char* pathname, FEX::HLE::open_ho if (!ShouldSkipOpenInEmu(how->flags)) { FDPathTmpData TmpFilename; - auto Path = GetEmulatedFDPath(dirfs, SelfPath, true, TmpFilename); + auto Path = GetEmulatedFDPath(dirfs, SelfPath, false, TmpFilename); if (Path.first != -1) { + how->resolve |= RESOLVE_IN_ROOT; fd = ::syscall(SYSCALL_DEF(openat2), Path.first, Path.second, how, usize); - } else { + if (fd < 0 && errno == EXDEV) { + // This means a magic symlink (/proc/foo) was involved. In this case we + // just punt and do the access without RESOLVE_IN_ROOT. + how->resolve &= RESOLVE_IN_ROOT; + fd = ::syscall(SYSCALL_DEF(openat2), Path.first, Path.second, how, usize); + } + } + + // Open through RootFS failed (probably nonexistent), so open directly. + if (fd == -1) { fd = ::syscall(SYSCALL_DEF(openat2), dirfs, SelfPath, how, usize); }