2023-03-30 14:22:15 +03:00
|
|
|
#include <BAN/ScopeGuard.h>
|
2023-02-22 01:23:11 +02:00
|
|
|
#include <BAN/StringView.h>
|
2023-03-30 22:39:45 +03:00
|
|
|
#include <kernel/DeviceManager.h>
|
2023-02-26 03:00:29 +02:00
|
|
|
#include <kernel/FS/Ext2.h>
|
2023-02-20 01:46:00 +02:00
|
|
|
#include <kernel/FS/VirtualFileSystem.h>
|
2023-03-30 22:02:16 +03:00
|
|
|
#include <kernel/LockGuard.h>
|
2023-02-20 01:46:00 +02:00
|
|
|
|
|
|
|
namespace Kernel
|
|
|
|
{
|
|
|
|
|
|
|
|
static VirtualFileSystem* s_instance = nullptr;
|
|
|
|
|
2023-03-30 16:37:53 +03:00
|
|
|
BAN::ErrorOr<void> VirtualFileSystem::initialize(BAN::StringView root)
|
2023-02-20 01:46:00 +02:00
|
|
|
{
|
|
|
|
ASSERT(s_instance == nullptr);
|
2023-02-26 03:00:29 +02:00
|
|
|
s_instance = new VirtualFileSystem();
|
|
|
|
if (s_instance == nullptr)
|
2023-03-02 21:10:44 +02:00
|
|
|
return BAN::Error::from_errno(ENOMEM);
|
2023-03-30 14:22:15 +03:00
|
|
|
BAN::ScopeGuard guard([] { delete s_instance; s_instance = nullptr; } );
|
2023-03-09 02:17:42 +02:00
|
|
|
|
2023-04-11 23:25:21 +03:00
|
|
|
ASSERT(root.size() >= 5 && root.substring(0, 5) == "/dev/"sv);;
|
2023-03-30 16:37:53 +03:00
|
|
|
root = root.substring(5);
|
|
|
|
|
|
|
|
auto partition_inode = TRY(DeviceManager::get().read_directory_inode(root));
|
2023-03-30 14:22:15 +03:00
|
|
|
s_instance->m_root_fs = TRY(Ext2FS::create(*(Partition*)partition_inode.ptr()));
|
2023-03-30 16:37:53 +03:00
|
|
|
|
2023-04-03 11:43:16 +03:00
|
|
|
DeviceManager::get().set_blksize(s_instance->m_root_fs->root_inode()->blksize());
|
2023-03-30 14:22:15 +03:00
|
|
|
TRY(s_instance->mount(&DeviceManager::get(), "/dev"));
|
|
|
|
|
|
|
|
guard.disable();
|
2023-03-09 02:17:42 +02:00
|
|
|
|
|
|
|
return {};
|
2023-02-20 01:46:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
VirtualFileSystem& VirtualFileSystem::get()
|
|
|
|
{
|
|
|
|
ASSERT(s_instance);
|
|
|
|
return *s_instance;
|
|
|
|
}
|
|
|
|
|
2023-03-30 15:06:41 +03:00
|
|
|
BAN::ErrorOr<void> VirtualFileSystem::mount(BAN::StringView partition, BAN::StringView target)
|
|
|
|
{
|
2023-06-01 00:24:45 +03:00
|
|
|
auto partition_file = TRY(file_from_absolute_path(partition, true));
|
2023-03-30 15:06:41 +03:00
|
|
|
if (partition_file.inode->inode_type() != Inode::InodeType::Device)
|
2023-04-11 23:25:21 +03:00
|
|
|
return BAN::Error::from_errno(ENOTBLK);
|
2023-03-30 22:02:16 +03:00
|
|
|
|
2023-03-30 15:06:41 +03:00
|
|
|
Device* device = (Device*)partition_file.inode.ptr();
|
2023-04-11 23:25:21 +03:00
|
|
|
if (device->device_type() != Device::DeviceType::BlockDevice)
|
|
|
|
return BAN::Error::from_errno(ENOTBLK);
|
2023-03-30 22:02:16 +03:00
|
|
|
|
2023-03-30 15:06:41 +03:00
|
|
|
auto* file_system = TRY(Ext2FS::create(*(Partition*)device));
|
|
|
|
return mount(file_system, target);
|
|
|
|
}
|
|
|
|
|
2023-03-29 11:50:46 +03:00
|
|
|
BAN::ErrorOr<void> VirtualFileSystem::mount(FileSystem* file_system, BAN::StringView path)
|
|
|
|
{
|
2023-06-01 00:24:45 +03:00
|
|
|
auto file = TRY(file_from_absolute_path(path, true));
|
2023-03-30 14:41:15 +03:00
|
|
|
if (!file.inode->mode().ifdir())
|
2023-03-29 11:50:46 +03:00
|
|
|
return BAN::Error::from_errno(ENOTDIR);
|
2023-03-30 22:02:16 +03:00
|
|
|
|
|
|
|
LockGuard _(m_lock);
|
2023-03-29 21:34:48 +03:00
|
|
|
TRY(m_mount_points.push_back({ file, file_system }));
|
2023-03-30 22:02:16 +03:00
|
|
|
|
2023-03-19 05:51:25 +02:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-04-01 01:54:35 +03:00
|
|
|
VirtualFileSystem::MountPoint* VirtualFileSystem::mount_from_host_inode(BAN::RefPtr<Inode> inode)
|
2023-03-29 21:34:48 +03:00
|
|
|
{
|
2023-03-30 22:02:16 +03:00
|
|
|
ASSERT(m_lock.is_locked());
|
2023-03-29 21:34:48 +03:00
|
|
|
for (MountPoint& mount : m_mount_points)
|
|
|
|
if (*mount.host.inode == *inode)
|
|
|
|
return &mount;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2023-04-01 01:54:35 +03:00
|
|
|
VirtualFileSystem::MountPoint* VirtualFileSystem::mount_from_root_inode(BAN::RefPtr<Inode> inode)
|
|
|
|
{
|
|
|
|
ASSERT(m_lock.is_locked());
|
|
|
|
for (MountPoint& mount : m_mount_points)
|
|
|
|
if (*mount.target->root_inode() == *inode)
|
|
|
|
return &mount;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2023-06-02 11:43:46 +03:00
|
|
|
BAN::ErrorOr<VirtualFileSystem::File> VirtualFileSystem::file_from_absolute_path(BAN::StringView path, bool follow_link)
|
2023-02-22 01:23:11 +02:00
|
|
|
{
|
2023-03-30 22:02:16 +03:00
|
|
|
LockGuard _(m_lock);
|
|
|
|
|
2023-03-16 15:31:33 +02:00
|
|
|
ASSERT(path.front() == '/');
|
2023-02-22 01:23:11 +02:00
|
|
|
|
|
|
|
auto inode = root_inode();
|
2023-04-11 23:25:21 +03:00
|
|
|
ASSERT(inode);
|
2023-03-09 02:31:24 +02:00
|
|
|
|
2023-03-29 21:34:48 +03:00
|
|
|
BAN::String canonical_path;
|
2023-03-16 15:31:33 +02:00
|
|
|
|
2023-06-02 11:43:46 +03:00
|
|
|
BAN::Vector<BAN::String> path_parts;
|
2023-03-29 21:34:48 +03:00
|
|
|
|
2023-03-17 21:16:22 +02:00
|
|
|
{
|
2023-06-02 11:43:46 +03:00
|
|
|
auto temp = TRY(path.split('/'));
|
|
|
|
for (size_t i = 0; i < temp.size(); i++)
|
2023-06-02 12:50:40 +03:00
|
|
|
TRY(path_parts.emplace_back(temp[temp.size() - i - 1]));
|
2023-06-02 11:43:46 +03:00
|
|
|
}
|
|
|
|
|
2023-06-02 12:50:40 +03:00
|
|
|
size_t link_depth = 0;
|
|
|
|
|
2023-06-02 11:43:46 +03:00
|
|
|
while (!path_parts.empty())
|
|
|
|
{
|
|
|
|
const auto& path_part = path_parts.back();
|
|
|
|
auto orig = inode;
|
|
|
|
|
2023-03-29 21:34:48 +03:00
|
|
|
if (path_part.empty() || path_part == "."sv)
|
2023-03-17 21:16:22 +02:00
|
|
|
{
|
2023-06-02 11:43:46 +03:00
|
|
|
|
2023-03-17 21:16:22 +02:00
|
|
|
}
|
2023-03-29 21:34:48 +03:00
|
|
|
else if (path_part == ".."sv)
|
2023-03-17 21:16:22 +02:00
|
|
|
{
|
2023-04-01 01:54:35 +03:00
|
|
|
if (auto* mount_point = mount_from_root_inode(inode))
|
2023-03-29 21:34:48 +03:00
|
|
|
inode = TRY(mount_point->host.inode->read_directory_inode(".."sv));
|
|
|
|
else
|
|
|
|
inode = TRY(inode->read_directory_inode(".."sv));
|
|
|
|
|
|
|
|
if (!canonical_path.empty())
|
2023-03-17 21:16:22 +02:00
|
|
|
{
|
2023-04-01 01:54:35 +03:00
|
|
|
ASSERT(canonical_path.front() == '/');
|
2023-03-29 21:34:48 +03:00
|
|
|
while (canonical_path.back() != '/')
|
|
|
|
canonical_path.pop_back();
|
|
|
|
canonical_path.pop_back();
|
2023-03-17 21:16:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-03-29 21:34:48 +03:00
|
|
|
inode = TRY(inode->read_directory_inode(path_part));
|
|
|
|
|
2023-04-01 01:54:35 +03:00
|
|
|
if (auto* mount_point = mount_from_host_inode(inode))
|
|
|
|
inode = mount_point->target->root_inode();
|
2023-06-01 00:24:45 +03:00
|
|
|
|
2023-06-02 11:43:46 +03:00
|
|
|
TRY(canonical_path.push_back('/'));
|
|
|
|
TRY(canonical_path.append(path_part));
|
|
|
|
}
|
|
|
|
|
|
|
|
path_parts.pop_back();
|
|
|
|
|
|
|
|
if (inode->mode().iflnk() && (follow_link || !path_parts.empty()))
|
|
|
|
{
|
|
|
|
auto target = TRY(inode->link_target());
|
|
|
|
if (target.empty())
|
|
|
|
return BAN::Error::from_errno(ENOENT);
|
|
|
|
|
|
|
|
if (target.front() == '/')
|
|
|
|
{
|
|
|
|
inode = root_inode();
|
|
|
|
canonical_path.clear();
|
|
|
|
|
|
|
|
auto temp = TRY(target.sv().split('/'));
|
|
|
|
for (size_t i = 0; i < temp.size(); i++)
|
|
|
|
TRY(path_parts.emplace_back(temp[temp.size() - i - 1]));
|
|
|
|
}
|
|
|
|
else
|
2023-06-01 00:24:45 +03:00
|
|
|
{
|
2023-06-02 11:43:46 +03:00
|
|
|
inode = orig;
|
|
|
|
|
|
|
|
while (canonical_path.back() != '/')
|
|
|
|
canonical_path.pop_back();
|
|
|
|
canonical_path.pop_back();
|
|
|
|
|
|
|
|
auto new_parts = TRY(target.sv().split('/'));
|
|
|
|
for (size_t i = 0; i < new_parts.size(); i++)
|
|
|
|
TRY(path_parts.emplace_back(new_parts[new_parts.size() - i - 1]));
|
2023-06-01 00:24:45 +03:00
|
|
|
}
|
2023-06-02 12:50:40 +03:00
|
|
|
|
|
|
|
link_depth++;
|
|
|
|
if (link_depth > 100)
|
|
|
|
return BAN::Error::from_errno(ELOOP);
|
2023-04-01 01:54:35 +03:00
|
|
|
}
|
2023-03-17 21:16:22 +02:00
|
|
|
}
|
|
|
|
|
2023-03-29 21:34:48 +03:00
|
|
|
if (canonical_path.empty())
|
|
|
|
TRY(canonical_path.push_back('/'));
|
|
|
|
|
2023-03-17 21:16:22 +02:00
|
|
|
File file;
|
|
|
|
file.inode = inode;
|
2023-03-29 21:34:48 +03:00
|
|
|
file.canonical_path = BAN::move(canonical_path);
|
2023-03-17 21:16:22 +02:00
|
|
|
|
|
|
|
if (file.canonical_path.empty())
|
|
|
|
TRY(file.canonical_path.push_back('/'));
|
|
|
|
|
|
|
|
return file;
|
2023-02-22 01:23:11 +02:00
|
|
|
}
|
|
|
|
|
2023-02-20 01:46:00 +02:00
|
|
|
}
|