#include "kernel/filesystem/ext2/filesystem.hpp" #include "kapi/devices/device.hpp" #include "kernel/devices/block_device_utils.hpp" #include "kernel/filesystem/ext2/block_group_descriptor.hpp" #include "kernel/filesystem/ext2/inode.hpp" #include "kernel/filesystem/ext2/superblock.hpp" #include "kernel/filesystem/filesystem.hpp" #include "kernel/filesystem/inode.hpp" #include #include #include #include #include namespace kernel::filesystem::ext2 { namespace { constexpr size_t SUPERBLOCK_OFFSET = 1024; constexpr uint16_t MAGIC_NUMBER = 0xEF53; constexpr uint32_t ROOT_INODE_NUMBER = 2; // Mode bits constexpr uint16_t S_IFMT = 0xF000; constexpr uint16_t S_IFREG = 0x8000; constexpr uint16_t S_IFDIR = 0x4000; auto S_ISREG(uint16_t mode) -> bool { return (mode & S_IFMT) == S_IFREG; } auto S_ISDIR(uint16_t mode) -> bool { return (mode & S_IFMT) == S_IFDIR; } } // namespace auto filesystem::mount(kstd::shared_ptr const & device) -> int { kernel::filesystem::filesystem::mount(device); kernel::devices::block_device_utils::read(m_device, &m_superblock, SUPERBLOCK_OFFSET, sizeof(m_superblock)); if (m_superblock.magic != MAGIC_NUMBER) { return -1; } auto const block_size = get_block_size(); auto const blocks_per_group = m_superblock.blocks_per_group; auto const num_block_groups = (m_superblock.blocks_count + blocks_per_group - 1) / blocks_per_group; m_block_group_descriptors = kstd::vector(num_block_groups); auto const block_group_descriptor_table_offset = block_size == 1024 ? 2 * block_size : block_size; kernel::devices::block_device_utils::read(m_device, m_block_group_descriptors.data(), block_group_descriptor_table_offset, num_block_groups * sizeof(block_group_descriptor)); m_root_inode = read_inode(ROOT_INODE_NUMBER); // TODO BA-FS26 check if root inode is valid and is a directory ?? return 0; } auto filesystem::lookup(kstd::shared_ptr const & /*parent*/, std::string_view name) -> kstd::shared_ptr { // TODO BA-FS26 implement ext2 directory traversal and inode loading if (name == "dev") { // TODO BA-FS26 just for testing return nullptr; } return kstd::make_shared(); } auto filesystem::read_inode(uint32_t inode_number) -> kstd::shared_ptr { auto const block_size = get_block_size(); auto const inodes_per_group = m_superblock.inodes_per_group; auto const block_group_index = (inode_number - 1) / inodes_per_group; auto const inode_index_within_group = (inode_number - 1) % inodes_per_group; if (block_group_index >= m_block_group_descriptors.size()) { return nullptr; } auto const & block_group_descriptor = m_block_group_descriptors.at(block_group_index); auto const inode_table_start_block = block_group_descriptor.inode_table; auto const inode_table_offset = static_cast(inode_table_start_block) * block_size; auto const inode_offset = inode_table_offset + inode_index_within_group * get_inode_size(); auto new_inode = kstd::make_shared(); kernel::devices::block_device_utils::read(m_device, &new_inode->m_data, inode_offset, sizeof(inode_data)); // TODO BA-FS26 improve inode_kind really needed? or just map it to the mode bits? if (S_ISREG(new_inode->m_data.mode)) { new_inode->m_kind = inode::inode_kind::regular; } else if (S_ISDIR(new_inode->m_data.mode)) { new_inode->m_kind = inode::inode_kind::directory; } else { // TODO BA-FS26 really correct?? return nullptr; } return new_inode; } auto filesystem::get_block_size() -> size_t { return 1024U << m_superblock.log_block_size; } auto filesystem::get_inode_size() -> size_t { return m_superblock.rev_level == 0 ? 128 : m_superblock.inode_size; } } // namespace kernel::filesystem::ext2