/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include namespace Kernel { class Ext2FSInode final : public Inode { friend class Ext2FS; friend class Ext2FSBlockView; public: virtual ~Ext2FSInode() override; u64 size() const; bool is_symlink() const { return Kernel::is_symlink(m_raw_inode.i_mode); } bool is_directory() const { return Kernel::is_directory(m_raw_inode.i_mode); } private: // ^Inode virtual ErrorOr read_bytes_locked(off_t, size_t, UserOrKernelBuffer& buffer, OpenFileDescription*) const override; virtual InodeMetadata metadata() const override; virtual ErrorOr traverse_as_directory(Function(FileSystem::DirectoryEntryView const&)>) const override; virtual ErrorOr> lookup(StringView name) override; virtual ErrorOr flush_metadata() override; virtual ErrorOr write_bytes_locked(off_t, size_t, UserOrKernelBuffer const& data, OpenFileDescription*) override; virtual ErrorOr> create_child(StringView name, mode_t, dev_t, UserID, GroupID) override; virtual ErrorOr add_child(Inode& child, StringView name, mode_t) override; virtual ErrorOr remove_child(StringView name) override; virtual ErrorOr replace_child(StringView name, Inode& child) override; virtual ErrorOr update_timestamps(Optional atime, Optional ctime, Optional mtime) override; virtual ErrorOr increment_link_count() override; virtual ErrorOr decrement_link_count() override; virtual ErrorOr chmod(mode_t) override; virtual ErrorOr chown(UserID, GroupID) override; virtual ErrorOr truncate_locked(u64) override; virtual ErrorOr get_block_address(int) override; ErrorOr allocate_block(BlockBasedFileSystem::BlockIndex, bool zero_newly_allocated_block, bool allow_cache); ErrorOr allocate_and_zero_block(); ErrorOr write_directory(Vector&); ErrorOr populate_lookup_cache(); ErrorOr resize(u64); ErrorOr write_singly_indirect_block_pointer(BlockBasedFileSystem::BlockIndex logical_block_index, BlockBasedFileSystem::BlockIndex on_disk_index); ErrorOr write_doubly_indirect_block_pointer(BlockBasedFileSystem::BlockIndex logical_block_index, BlockBasedFileSystem::BlockIndex on_disk_index); ErrorOr write_triply_indirect_block_pointer(BlockBasedFileSystem::BlockIndex logical_block_index, BlockBasedFileSystem::BlockIndex on_disk_index); ErrorOr write_block_pointer(BlockBasedFileSystem::BlockIndex logical_block_index, BlockBasedFileSystem::BlockIndex on_disk_index); ErrorOr compute_block_list(BlockBasedFileSystem::BlockIndex, BlockBasedFileSystem::BlockIndex) const; ErrorOr free_all_blocks(); u64 singly_indirect_block_capacity() const { auto const entries_per_block = EXT2_ADDR_PER_BLOCK(&fs().super_block()); return EXT2_NDIR_BLOCKS + entries_per_block; } u64 doubly_indirect_block_capacity() const { auto const entries_per_block = EXT2_ADDR_PER_BLOCK(&fs().super_block()); return singly_indirect_block_capacity() + entries_per_block * entries_per_block; } u64 triply_indirect_block_capacity() const { auto const entries_per_block = EXT2_ADDR_PER_BLOCK(&fs().super_block()); return doubly_indirect_block_capacity() + entries_per_block * entries_per_block * entries_per_block; } Ext2FS& fs(); Ext2FS const& fs() const; Ext2FSInode(Ext2FS&, InodeIndex); mutable Ext2FSBlockView m_block_view; HashMap, InodeIndex> m_lookup_cache; ext2_inode m_raw_inode {}; }; inline Ext2FS& Ext2FSInode::fs() { return static_cast(Inode::fs()); } inline Ext2FS const& Ext2FSInode::fs() const { return static_cast(Inode::fs()); } }