mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 11:42:33 +00:00

Like with the ProcFS, description data can change at anytime, so it's wise to ensure that when the userland reads from an Inode, data is consistent unless the userland indicated it wants to refresh the data (by seeking to offset 0, or re-attaching the Inode). Otherwise, if the data changes in the middle of the reading, it can cause silent corruption in output which can lead to random crashes.
245 lines
6.3 KiB
C++
245 lines
6.3 KiB
C++
/*
|
|
* Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/Singleton.h>
|
|
#include <AK/StringView.h>
|
|
#include <Kernel/FileSystem/SysFS.h>
|
|
#include <Kernel/Sections.h>
|
|
|
|
namespace Kernel {
|
|
|
|
static Singleton<SysFSComponentRegistry> s_the;
|
|
|
|
SysFSComponentRegistry& SysFSComponentRegistry::the()
|
|
{
|
|
return *s_the;
|
|
}
|
|
|
|
UNMAP_AFTER_INIT void SysFSComponentRegistry::initialize()
|
|
{
|
|
VERIFY(!s_the.is_initialized());
|
|
s_the.ensure_instance();
|
|
}
|
|
|
|
UNMAP_AFTER_INIT SysFSComponentRegistry::SysFSComponentRegistry()
|
|
: m_root_directory(SysFSRootDirectory::create())
|
|
{
|
|
}
|
|
|
|
UNMAP_AFTER_INIT void SysFSComponentRegistry::register_new_component(SysFSComponent& component)
|
|
{
|
|
MutexLocker locker(m_lock);
|
|
m_root_directory->m_components.append(component);
|
|
}
|
|
|
|
NonnullRefPtr<SysFSRootDirectory> SysFSRootDirectory::create()
|
|
{
|
|
return adopt_ref(*new (nothrow) SysFSRootDirectory);
|
|
}
|
|
|
|
KResult SysFSRootDirectory::traverse_as_directory(unsigned fsid, Function<bool(FileSystem::DirectoryEntryView const&)> callback) const
|
|
{
|
|
MutexLocker locker(SysFSComponentRegistry::the().get_lock());
|
|
callback({ ".", { fsid, component_index() }, 0 });
|
|
callback({ "..", { fsid, 0 }, 0 });
|
|
|
|
for (auto& component : m_components) {
|
|
InodeIdentifier identifier = { fsid, component.component_index() };
|
|
callback({ component.name(), identifier, 0 });
|
|
}
|
|
return KSuccess;
|
|
}
|
|
|
|
SysFSRootDirectory::SysFSRootDirectory()
|
|
: SysFSDirectory(".")
|
|
{
|
|
auto buses_directory = SysFSBusDirectory::must_create(*this);
|
|
m_components.append(buses_directory);
|
|
m_buses_directory = buses_directory;
|
|
}
|
|
|
|
NonnullRefPtr<SysFS> SysFS::create()
|
|
{
|
|
return adopt_ref(*new (nothrow) SysFS);
|
|
}
|
|
|
|
SysFS::SysFS()
|
|
: m_root_inode(SysFSComponentRegistry::the().root_directory().to_inode(*this))
|
|
{
|
|
}
|
|
|
|
SysFS::~SysFS()
|
|
{
|
|
}
|
|
|
|
KResult SysFS::initialize()
|
|
{
|
|
return KSuccess;
|
|
}
|
|
|
|
Inode& SysFS::root_inode()
|
|
{
|
|
return *m_root_inode;
|
|
}
|
|
|
|
NonnullRefPtr<SysFSInode> SysFSInode::create(SysFS const& fs, SysFSComponent const& component)
|
|
{
|
|
return adopt_ref(*new (nothrow) SysFSInode(fs, component));
|
|
}
|
|
|
|
SysFSInode::SysFSInode(SysFS const& fs, SysFSComponent const& component)
|
|
: Inode(const_cast<SysFS&>(fs), component.component_index())
|
|
, m_associated_component(component)
|
|
{
|
|
}
|
|
|
|
void SysFSInode::did_seek(FileDescription& description, off_t new_offset)
|
|
{
|
|
if (new_offset != 0)
|
|
return;
|
|
auto result = m_associated_component->refresh_data(description);
|
|
if (result.is_error()) {
|
|
// Subsequent calls to read will return EIO!
|
|
dbgln("SysFS: Could not refresh contents: {}", result.error());
|
|
}
|
|
}
|
|
|
|
KResult SysFSInode::attach(FileDescription& description)
|
|
{
|
|
return m_associated_component->refresh_data(description);
|
|
}
|
|
|
|
KResultOr<size_t> SysFSInode::read_bytes(off_t offset, size_t count, UserOrKernelBuffer& buffer, FileDescription* fd) const
|
|
{
|
|
return m_associated_component->read_bytes(offset, count, buffer, fd);
|
|
}
|
|
|
|
KResult SysFSInode::traverse_as_directory(Function<bool(FileSystem::DirectoryEntryView const&)>) const
|
|
{
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
|
|
KResultOr<NonnullRefPtr<Inode>> SysFSInode::lookup(StringView)
|
|
{
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
|
|
InodeMetadata SysFSInode::metadata() const
|
|
{
|
|
// NOTE: No locking required as m_associated_component or its component index will never change during our lifetime.
|
|
InodeMetadata metadata;
|
|
metadata.inode = { fsid(), m_associated_component->component_index() };
|
|
metadata.mode = S_IFREG | S_IRUSR | S_IRGRP | S_IROTH;
|
|
metadata.uid = 0;
|
|
metadata.gid = 0;
|
|
metadata.size = 0;
|
|
metadata.mtime = mepoch;
|
|
return metadata;
|
|
}
|
|
|
|
void SysFSInode::flush_metadata()
|
|
{
|
|
}
|
|
|
|
KResultOr<size_t> SysFSInode::write_bytes(off_t offset, size_t count, UserOrKernelBuffer const& buffer, FileDescription* fd)
|
|
{
|
|
return m_associated_component->write_bytes(offset, count, buffer, fd);
|
|
}
|
|
|
|
KResultOr<NonnullRefPtr<Inode>> SysFSInode::create_child(StringView, mode_t, dev_t, UserID, GroupID)
|
|
{
|
|
return EROFS;
|
|
}
|
|
|
|
KResult SysFSInode::add_child(Inode&, StringView const&, mode_t)
|
|
{
|
|
return EROFS;
|
|
}
|
|
|
|
KResult SysFSInode::remove_child(StringView const&)
|
|
{
|
|
return EROFS;
|
|
}
|
|
|
|
KResult SysFSInode::chmod(mode_t)
|
|
{
|
|
return EPERM;
|
|
}
|
|
|
|
KResult SysFSInode::chown(UserID, GroupID)
|
|
{
|
|
return EPERM;
|
|
}
|
|
|
|
KResult SysFSInode::truncate(u64)
|
|
{
|
|
return EPERM;
|
|
}
|
|
|
|
NonnullRefPtr<SysFSDirectoryInode> SysFSDirectoryInode::create(SysFS const& sysfs, SysFSComponent const& component)
|
|
{
|
|
return adopt_ref(*new (nothrow) SysFSDirectoryInode(sysfs, component));
|
|
}
|
|
|
|
SysFSDirectoryInode::SysFSDirectoryInode(SysFS const& fs, SysFSComponent const& component)
|
|
: SysFSInode(fs, component)
|
|
{
|
|
}
|
|
|
|
SysFSDirectoryInode::~SysFSDirectoryInode()
|
|
{
|
|
}
|
|
|
|
InodeMetadata SysFSDirectoryInode::metadata() const
|
|
{
|
|
// NOTE: No locking required as m_associated_component or its component index will never change during our lifetime.
|
|
InodeMetadata metadata;
|
|
metadata.inode = { fsid(), m_associated_component->component_index() };
|
|
metadata.mode = S_IFDIR | S_IRUSR | S_IRGRP | S_IROTH | S_IXOTH;
|
|
metadata.uid = 0;
|
|
metadata.gid = 0;
|
|
metadata.size = 0;
|
|
metadata.mtime = mepoch;
|
|
return metadata;
|
|
}
|
|
KResult SysFSDirectoryInode::traverse_as_directory(Function<bool(FileSystem::DirectoryEntryView const&)> callback) const
|
|
{
|
|
MutexLocker locker(fs().m_lock);
|
|
return m_associated_component->traverse_as_directory(fs().fsid(), move(callback));
|
|
}
|
|
|
|
KResultOr<NonnullRefPtr<Inode>> SysFSDirectoryInode::lookup(StringView name)
|
|
{
|
|
MutexLocker locker(fs().m_lock);
|
|
auto component = m_associated_component->lookup(name);
|
|
if (!component)
|
|
return ENOENT;
|
|
return component->to_inode(fs());
|
|
}
|
|
|
|
SysFSBusDirectory& SysFSComponentRegistry::buses_directory()
|
|
{
|
|
return *m_root_directory->m_buses_directory;
|
|
}
|
|
|
|
void SysFSComponentRegistry::register_new_bus_directory(SysFSDirectory& new_bus_directory)
|
|
{
|
|
VERIFY(!m_root_directory->m_buses_directory.is_null());
|
|
m_root_directory->m_buses_directory->m_components.append(new_bus_directory);
|
|
}
|
|
|
|
UNMAP_AFTER_INIT NonnullRefPtr<SysFSBusDirectory> SysFSBusDirectory::must_create(SysFSRootDirectory const& parent_directory)
|
|
{
|
|
auto directory = adopt_ref(*new (nothrow) SysFSBusDirectory(parent_directory));
|
|
return directory;
|
|
}
|
|
|
|
UNMAP_AFTER_INIT SysFSBusDirectory::SysFSBusDirectory(SysFSRootDirectory const& parent_directory)
|
|
: SysFSDirectory("bus"sv, parent_directory)
|
|
{
|
|
}
|
|
|
|
}
|