mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 21:22:35 +00:00 
			
		
		
		
	 c1af2f28e3
			
		
	
	
		c1af2f28e3
		
	
	
	
	
		
			
			Since we already have the directory open, let's have an API to fchown() the underlying file descriptor instead of forcing clients to do another path lookup.
		
			
				
	
	
		
			108 lines
		
	
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			108 lines
		
	
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2022, kleines Filmröllchen <filmroellchen@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include "Directory.h"
 | |
| #include "DirIterator.h"
 | |
| #include "System.h"
 | |
| #include <dirent.h>
 | |
| 
 | |
| namespace Core {
 | |
| 
 | |
| // We assume that the fd is a valid directory.
 | |
| Directory::Directory(int fd, Optional<LexicalPath> path)
 | |
|     : m_path(move(path))
 | |
|     , m_directory_fd(fd)
 | |
| {
 | |
| }
 | |
| 
 | |
| Directory::Directory(Directory&& other)
 | |
|     : m_path(move(other.m_path))
 | |
|     , m_directory_fd(other.m_directory_fd)
 | |
| {
 | |
|     other.m_directory_fd = -1;
 | |
| }
 | |
| 
 | |
| Directory::~Directory()
 | |
| {
 | |
|     if (m_directory_fd != -1)
 | |
|         MUST(System::close(m_directory_fd));
 | |
| }
 | |
| 
 | |
| ErrorOr<void> Directory::chown(uid_t uid, gid_t gid)
 | |
| {
 | |
|     if (m_directory_fd == -1)
 | |
|         return Error::from_syscall("fchown"sv, -EBADF);
 | |
|     TRY(Core::System::fchown(m_directory_fd, uid, gid));
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| ErrorOr<bool> Directory::is_valid_directory(int fd)
 | |
| {
 | |
|     auto stat = TRY(System::fstat(fd));
 | |
|     return stat.st_mode & S_IFDIR;
 | |
| }
 | |
| 
 | |
| ErrorOr<Directory> Directory::adopt_fd(int fd, Optional<LexicalPath> path)
 | |
| {
 | |
|     // This will also fail if the fd is invalid in the first place.
 | |
|     if (!TRY(Directory::is_valid_directory(fd)))
 | |
|         return Error::from_errno(ENOTDIR);
 | |
|     return Directory { fd, move(path) };
 | |
| }
 | |
| 
 | |
| ErrorOr<Directory> Directory::create(String path, CreateDirectories create_directories, mode_t creation_mode)
 | |
| {
 | |
|     return create(LexicalPath { move(path) }, create_directories, creation_mode);
 | |
| }
 | |
| 
 | |
| ErrorOr<Directory> Directory::create(LexicalPath path, CreateDirectories create_directories, mode_t creation_mode)
 | |
| {
 | |
|     if (create_directories == CreateDirectories::Yes)
 | |
|         TRY(ensure_directory(path, creation_mode));
 | |
|     // FIXME: doesn't work on Linux probably
 | |
|     auto fd = TRY(System::open(path.string(), O_CLOEXEC));
 | |
|     return adopt_fd(fd, move(path));
 | |
| }
 | |
| 
 | |
| ErrorOr<void> Directory::ensure_directory(LexicalPath const& path, mode_t creation_mode)
 | |
| {
 | |
|     if (path.basename() == "/" || path.basename() == ".")
 | |
|         return {};
 | |
| 
 | |
|     TRY(ensure_directory(path.parent(), creation_mode));
 | |
| 
 | |
|     auto return_value = System::mkdir(path.string(), creation_mode);
 | |
|     // We don't care if the directory already exists.
 | |
|     if (return_value.is_error() && return_value.error().code() != EEXIST)
 | |
|         return return_value;
 | |
| 
 | |
|     return {};
 | |
| }
 | |
| 
 | |
| ErrorOr<LexicalPath> Directory::path() const
 | |
| {
 | |
|     if (!m_path.has_value())
 | |
|         return Error::from_string_literal("Directory wasn't created with a path");
 | |
|     return m_path.value();
 | |
| }
 | |
| 
 | |
| ErrorOr<NonnullOwnPtr<Stream::File>> Directory::open(StringView filename, Stream::OpenMode mode) const
 | |
| {
 | |
|     auto fd = TRY(System::openat(m_directory_fd, filename, Stream::File::open_mode_to_options(mode)));
 | |
|     return Stream::File::adopt_fd(fd, mode);
 | |
| }
 | |
| 
 | |
| ErrorOr<struct stat> Directory::stat() const
 | |
| {
 | |
|     return System::fstat(m_directory_fd);
 | |
| }
 | |
| 
 | |
| ErrorOr<DirIterator> Directory::create_iterator() const
 | |
| {
 | |
|     return DirIterator { TRY(path()).string() };
 | |
| }
 | |
| 
 | |
| }
 |