/* * Copyright (c) 2020, Liav A. * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include namespace Kernel { class DevTmpFS final : public FileSystem { friend class DevTmpFSInode; friend class DevTmpFSRootDirectoryInode; public: virtual ~DevTmpFS() override; static KResultOr> try_create(); virtual KResult initialize() override; virtual StringView class_name() const override { return "DevTmpFS"sv; } virtual Inode& root_inode() override; private: DevTmpFS(); size_t allocate_inode_index(); RefPtr m_root_inode; InodeIndex m_next_inode_index { 0 }; }; class DevTmpFSInode : public Inode { friend class DevTmpFS; friend class DevTmpFSRootDirectoryInode; friend class DevTmpFSDirectoryInode; public: virtual StringView name() const = 0; DevTmpFS& fs() { return static_cast(Inode::fs()); } DevTmpFS const& fs() const { return static_cast(Inode::fs()); } protected: DevTmpFSInode(DevTmpFS&); virtual KResultOr read_bytes(off_t, size_t, UserOrKernelBuffer& buffer, OpenFileDescription*) const override; virtual KResult traverse_as_directory(Function) const override; virtual KResultOr> lookup(StringView name) override; virtual void flush_metadata() override; virtual KResultOr write_bytes(off_t, size_t, const UserOrKernelBuffer& buffer, OpenFileDescription*) override; virtual KResultOr> create_child(StringView name, mode_t, dev_t, UserID, GroupID) override; virtual KResult add_child(Inode&, const StringView& name, mode_t) override; virtual KResult remove_child(const StringView& name) override; virtual KResult chmod(mode_t) override; virtual KResult chown(UserID, GroupID) override; virtual KResult truncate(u64) override; private: IntrusiveListNode> m_list_node; }; class DevTmpFSDeviceInode : public DevTmpFSInode { friend class DevTmpFS; friend class DevTmpFSRootDirectoryInode; friend class DevTmpFSDirectoryInode; public: virtual StringView name() const override; virtual ~DevTmpFSDeviceInode() override; private: DevTmpFSDeviceInode(DevTmpFS&, unsigned, unsigned, bool, NonnullOwnPtr name); // ^Inode virtual KResultOr read_bytes(off_t, size_t, UserOrKernelBuffer& buffer, OpenFileDescription*) const override; virtual InodeMetadata metadata() const override; virtual KResultOr write_bytes(off_t, size_t, const UserOrKernelBuffer& buffer, OpenFileDescription*) override; virtual KResult chown(UserID, GroupID) override; virtual KResult chmod(mode_t) override; NonnullOwnPtr m_name; const unsigned m_major_number; const unsigned m_minor_number; const bool m_block_device; mode_t m_required_mode; UserID m_uid { 0 }; GroupID m_gid { 0 }; }; class DevTmpFSLinkInode : public DevTmpFSInode { friend class DevTmpFS; friend class DevTmpFSRootDirectoryInode; public: virtual StringView name() const override; virtual ~DevTmpFSLinkInode() override; protected: DevTmpFSLinkInode(DevTmpFS&, NonnullOwnPtr); // ^Inode virtual KResultOr read_bytes(off_t, size_t, UserOrKernelBuffer& buffer, OpenFileDescription*) const override; virtual InodeMetadata metadata() const override; virtual KResultOr write_bytes(off_t, size_t, const UserOrKernelBuffer& buffer, OpenFileDescription*) override; NonnullOwnPtr m_name; OwnPtr m_link; }; class DevTmpFSDirectoryInode : public DevTmpFSInode { friend class DevTmpFS; friend class DevTmpFSRootDirectoryInode; public: virtual ~DevTmpFSDirectoryInode() override; protected: DevTmpFSDirectoryInode(DevTmpFS&); // ^Inode virtual InodeMetadata metadata() const override; IntrusiveList, &DevTmpFSInode::m_list_node> m_nodes; }; class DevTmpFSPtsDirectoryInode final : public DevTmpFSDirectoryInode { friend class DevTmpFS; friend class DevTmpFSRootDirectoryInode; public: virtual ~DevTmpFSPtsDirectoryInode() override; virtual StringView name() const override { return "pts"; }; private: explicit DevTmpFSPtsDirectoryInode(DevTmpFS&); virtual KResult traverse_as_directory(Function) const override; virtual KResultOr> lookup(StringView name) override; virtual InodeMetadata metadata() const override; }; class DevTmpFSRootDirectoryInode final : public DevTmpFSDirectoryInode { friend class DevTmpFS; public: virtual ~DevTmpFSRootDirectoryInode() override; virtual StringView name() const override { return "."; } private: explicit DevTmpFSRootDirectoryInode(DevTmpFS&); virtual KResultOr> create_child(StringView name, mode_t, dev_t, UserID, GroupID) override; virtual KResult traverse_as_directory(Function) const override; virtual KResultOr> lookup(StringView name) override; virtual InodeMetadata metadata() const override; virtual KResult remove_child(const StringView& name) override; }; }