/* * Copyright (c) 2018-2022, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include #include namespace Kernel { class OpenFileDescriptionData { public: virtual ~OpenFileDescriptionData() = default; }; class OpenFileDescription final : public AtomicRefCounted { public: static ErrorOr> try_create(Custody&); static ErrorOr> try_create(File&); ~OpenFileDescription(); Thread::FileBlocker::BlockFlags should_unblock(Thread::FileBlocker::BlockFlags) const; bool is_readable() const; void set_readable(bool); bool is_writable() const; void set_writable(bool); void set_rw_mode(int options); ErrorOr close(); ErrorOr seek(off_t, int whence); ErrorOr read(UserOrKernelBuffer&, size_t); ErrorOr write(UserOrKernelBuffer const& data, size_t); ErrorOr stat(); // NOTE: These ignore the current offset of this file description. ErrorOr read(UserOrKernelBuffer&, u64 offset, size_t); ErrorOr write(u64 offset, UserOrKernelBuffer const&, size_t); ErrorOr chmod(Credentials const& credentials, mode_t); bool can_read() const; bool can_write() const; ErrorOr get_dir_entries(UserOrKernelBuffer& buffer, size_t); ErrorOr> original_absolute_path() const; ErrorOr> pseudo_path() const; bool is_direct() const; bool is_directory() const; File& file() { return *m_file; } File const& file() const { return *m_file; } bool is_device() const; Device const* device() const; Device* device(); bool is_tty() const; const TTY* tty() const; TTY* tty(); bool is_inode_watcher() const; InodeWatcher const* inode_watcher() const; InodeWatcher* inode_watcher(); bool is_mount_file() const; MountFile const* mount_file() const; MountFile* mount_file(); bool is_master_pty() const; MasterPTY const* master_pty() const; MasterPTY* master_pty(); InodeMetadata metadata() const; Inode* inode() { return m_inode.ptr(); } Inode const* inode() const { return m_inode.ptr(); } RefPtr custody(); RefPtr custody() const; ErrorOr> vmobject_for_mmap(Process&, Memory::VirtualRange const&, u64& offset, bool shared); bool is_blocking() const; void set_blocking(bool b); bool should_append() const; u32 file_flags() const; void set_file_flags(u32); bool is_socket() const; Socket* socket(); Socket const* socket() const; bool is_fifo() const; FIFO* fifo(); FIFO::Direction fifo_direction() const; void set_fifo_direction(Badge, FIFO::Direction direction); OwnPtr& data(); void set_original_inode(Badge, NonnullRefPtr inode) { m_inode = move(inode); } void set_original_custody(Badge, Custody& custody); ErrorOr truncate(u64); ErrorOr sync(); off_t offset() const; ErrorOr chown(Credentials const& credentials, UserID, GroupID); FileBlockerSet& blocker_set(); ErrorOr apply_flock(Process const&, Userspace, ShouldBlock); ErrorOr get_flock(Userspace) const; private: friend class VirtualFileSystem; explicit OpenFileDescription(File&); ErrorOr attach(); void evaluate_block_conditions() { blocker_set().unblock_all_blockers_whose_conditions_are_met(); } RefPtr m_inode; NonnullRefPtr const m_file; struct State { OwnPtr data; RefPtr custody; off_t current_offset { 0 }; u32 file_flags { 0 }; bool readable : 1 { false }; bool writable : 1 { false }; bool is_blocking : 1 { true }; bool is_directory : 1 { false }; bool should_append : 1 { false }; bool direct : 1 { false }; FIFO::Direction fifo_direction : 2 { FIFO::Direction::Neither }; }; SpinlockProtected m_state {}; }; }