aboutsummaryrefslogtreecommitdiff
path: root/kernel/src/filesystem/ext2/filesystem.cpp
blob: a3425b5835f3f78a6a691e617c7f34aa6719de06 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
#include "kernel/filesystem/ext2/filesystem.hpp"

#include "kapi/devices/device.hpp"

#include "kernel/devices/block_device_utils.hpp"
#include "kernel/filesystem/ext2/inode.hpp"
#include "kernel/filesystem/ext2/superblock.hpp"
#include "kernel/filesystem/inode.hpp"

#include <kstd/memory>

#include <cstddef>
#include <cstdint>
#include <string_view>

namespace kernel::filesystem::ext2
{
  namespace
  {
    constexpr size_t SUPERBLOCK_OFFSET = 1024;
    constexpr uint16_t EXT2_MAGIC = 0xEF53;

    // 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<kapi::devices::device> const & device) -> int
  {
    m_device = device;

    kernel::devices::block_device_utils::read(m_device, &m_superblock, SUPERBLOCK_OFFSET, sizeof(m_superblock));

    if (m_superblock.magic != EXT2_MAGIC)
    {
      return -1;
    }

    // TODO BA-FS26 load proper root inode from ext2 metadata
    // m_root_inode = inode{inode_kind::directory};

    // TODO BA-FS26 implement
    m_root_inode = kstd::make_shared<inode>();
    // devices::block_device_utils::read(device, nullptr, 0, 0);  // TODO BA-FS26 just for testing
    return 0;
  }

  auto filesystem::lookup(kstd::shared_ptr<kernel::filesystem::inode> const & /*parent*/, std::string_view name)
      -> kstd::shared_ptr<kernel::filesystem::inode>
  {
    // 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<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