/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include namespace Kernel { KResultOr Process::sys$fstat(int fd, Userspace user_statbuf) { VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this) REQUIRE_PROMISE(stdio); auto description = fds().file_description(fd); if (!description) return EBADF; stat buffer = {}; int rc = description->stat(buffer); if (!copy_to_user(user_statbuf, &buffer)) return EFAULT; return rc; } KResultOr Process::sys$stat(Userspace user_params) { VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this) REQUIRE_PROMISE(rpath); Syscall::SC_stat_params params; if (!copy_from_user(¶ms, user_params)) return EFAULT; auto path = get_syscall_path_argument(params.path); if (path.is_error()) return path.error(); RefPtr base; if (params.dirfd == AT_FDCWD) { base = current_directory(); } else { auto base_description = fds().file_description(params.dirfd); if (!base_description) return EBADF; if (!base_description->is_directory()) return ENOTDIR; if (!base_description->custody()) return EINVAL; base = base_description->custody(); } auto metadata_or_error = VirtualFileSystem::the().lookup_metadata(path.value()->view(), *base, params.follow_symlinks ? 0 : O_NOFOLLOW_NOERROR); if (metadata_or_error.is_error()) return metadata_or_error.error(); stat statbuf; auto result = metadata_or_error.value().stat(statbuf); if (result.is_error()) return result; if (!copy_to_user(params.statbuf, &statbuf)) return EFAULT; return 0; } }