123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168 |
- /*
- * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <AK/Singleton.h>
- #include <Kernel/Devices/Device.h>
- #include <Kernel/Devices/DeviceManagement.h>
- #include <Kernel/FileSystem/InodeMetadata.h>
- #include <Kernel/FileSystem/SysFS.h>
- #include <Kernel/Sections.h>
- namespace Kernel {
- NonnullRefPtr<SysFSDeviceComponent> SysFSDeviceComponent::must_create(Device const& device)
- {
- // FIXME: Handle allocation failure gracefully
- auto device_name = MUST(KString::try_create(String::formatted("{}:{}", device.major(), device.minor())));
- return adopt_ref_if_nonnull(new SysFSDeviceComponent(move(device_name), device)).release_nonnull();
- }
- SysFSDeviceComponent::SysFSDeviceComponent(NonnullOwnPtr<KString> major_minor_formatted_device_name, Device const& device)
- : SysFSComponent()
- , m_block_device(device.is_block_device())
- , m_major_minor_formatted_device_name(move(major_minor_formatted_device_name))
- {
- VERIFY(device.is_block_device() || device.is_character_device());
- }
- UNMAP_AFTER_INIT NonnullRefPtr<SysFSDevicesDirectory> SysFSDevicesDirectory::must_create(SysFSRootDirectory const& root_directory)
- {
- auto devices_directory = adopt_ref_if_nonnull(new SysFSDevicesDirectory(root_directory)).release_nonnull();
- devices_directory->m_components.append(SysFSBlockDevicesDirectory::must_create(*devices_directory));
- devices_directory->m_components.append(SysFSCharacterDevicesDirectory::must_create(*devices_directory));
- return devices_directory;
- }
- SysFSDevicesDirectory::SysFSDevicesDirectory(SysFSRootDirectory const& root_directory)
- : SysFSDirectory(root_directory)
- {
- }
- NonnullRefPtr<SysFSBlockDevicesDirectory> SysFSBlockDevicesDirectory::must_create(SysFSDevicesDirectory const& devices_directory)
- {
- return adopt_ref_if_nonnull(new SysFSBlockDevicesDirectory(devices_directory)).release_nonnull();
- }
- SysFSBlockDevicesDirectory::SysFSBlockDevicesDirectory(SysFSDevicesDirectory const& devices_directory)
- : SysFSDirectory(devices_directory)
- {
- }
- ErrorOr<void> SysFSBlockDevicesDirectory::traverse_as_directory(FileSystemID fsid, Function<ErrorOr<void>(FileSystem::DirectoryEntryView const&)> callback) const
- {
- VERIFY(m_parent_directory);
- TRY(callback({ ".", { fsid, component_index() }, 0 }));
- TRY(callback({ "..", { fsid, m_parent_directory->component_index() }, 0 }));
- return SysFSComponentRegistry::the().devices_list().with_exclusive([&](auto& list) -> ErrorOr<void> {
- for (auto& exposed_device : list) {
- if (!exposed_device.is_block_device())
- continue;
- TRY(callback({ exposed_device.name(), { fsid, exposed_device.component_index() }, 0 }));
- }
- return {};
- });
- }
- RefPtr<SysFSComponent> SysFSBlockDevicesDirectory::lookup(StringView name)
- {
- return SysFSComponentRegistry::the().devices_list().with_exclusive([&](auto& list) -> RefPtr<SysFSComponent> {
- for (auto& exposed_device : list) {
- if (!exposed_device.is_block_device())
- continue;
- if (exposed_device.name() == name)
- return exposed_device;
- }
- return nullptr;
- });
- }
- NonnullRefPtr<SysFSCharacterDevicesDirectory> SysFSCharacterDevicesDirectory::must_create(SysFSDevicesDirectory const& devices_directory)
- {
- return adopt_ref_if_nonnull(new SysFSCharacterDevicesDirectory(devices_directory)).release_nonnull();
- }
- SysFSCharacterDevicesDirectory::SysFSCharacterDevicesDirectory(SysFSDevicesDirectory const& devices_directory)
- : SysFSDirectory(devices_directory)
- {
- }
- ErrorOr<void> SysFSCharacterDevicesDirectory::traverse_as_directory(FileSystemID fsid, Function<ErrorOr<void>(FileSystem::DirectoryEntryView const&)> callback) const
- {
- VERIFY(m_parent_directory);
- TRY(callback({ ".", { fsid, component_index() }, 0 }));
- TRY(callback({ "..", { fsid, m_parent_directory->component_index() }, 0 }));
- return SysFSComponentRegistry::the().devices_list().with_exclusive([&](auto& list) -> ErrorOr<void> {
- for (auto& exposed_device : list) {
- if (exposed_device.is_block_device())
- continue;
- TRY(callback({ exposed_device.name(), { fsid, exposed_device.component_index() }, 0 }));
- }
- return {};
- });
- }
- RefPtr<SysFSComponent> SysFSCharacterDevicesDirectory::lookup(StringView name)
- {
- return SysFSComponentRegistry::the().devices_list().with_exclusive([&](auto& list) -> RefPtr<SysFSComponent> {
- for (auto& exposed_device : list) {
- if (exposed_device.is_block_device())
- continue;
- if (exposed_device.name() == name)
- return exposed_device;
- }
- return nullptr;
- });
- }
- Device::Device(MajorNumber major, MinorNumber minor)
- : m_major(major)
- , m_minor(minor)
- {
- }
- void Device::after_inserting()
- {
- DeviceManagement::the().after_inserting_device({}, *this);
- VERIFY(!m_sysfs_component);
- auto sys_fs_component = SysFSDeviceComponent::must_create(*this);
- m_sysfs_component = sys_fs_component;
- SysFSComponentRegistry::the().devices_list().with_exclusive([&](auto& list) -> void {
- list.append(sys_fs_component);
- });
- }
- void Device::before_removing()
- {
- VERIFY(m_sysfs_component);
- SysFSComponentRegistry::the().devices_list().with_exclusive([&](auto& list) -> void {
- list.remove(*m_sysfs_component);
- });
- DeviceManagement::the().before_device_removal({}, *this);
- m_state = State::BeingRemoved;
- }
- Device::~Device()
- {
- VERIFY(m_state == State::BeingRemoved);
- }
- ErrorOr<NonnullOwnPtr<KString>> Device::pseudo_path(const OpenFileDescription&) const
- {
- return KString::formatted("device:{},{}", major(), minor());
- }
- void Device::process_next_queued_request(Badge<AsyncDeviceRequest>, const AsyncDeviceRequest& completed_request)
- {
- SpinlockLocker lock(m_requests_lock);
- VERIFY(!m_requests.is_empty());
- VERIFY(m_requests.first().ptr() == &completed_request);
- m_requests.remove(m_requests.begin());
- if (!m_requests.is_empty()) {
- auto* next_request = m_requests.first().ptr();
- next_request->do_start(move(lock));
- }
- evaluate_block_conditions();
- }
- }
|