summaryrefslogtreecommitdiff
path: root/Kernel/Devices/DeviceManagement.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Kernel/Devices/DeviceManagement.cpp')
-rw-r--r--Kernel/Devices/DeviceManagement.cpp88
1 files changed, 88 insertions, 0 deletions
diff --git a/Kernel/Devices/DeviceManagement.cpp b/Kernel/Devices/DeviceManagement.cpp
new file mode 100644
index 0000000000..b02fc06cbf
--- /dev/null
+++ b/Kernel/Devices/DeviceManagement.cpp
@@ -0,0 +1,88 @@
+/*
+ * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il>
+ *
+ * SPDX-License-Identifier: BSD-2-Clause
+ */
+
+#include <AK/HashTable.h>
+#include <AK/Singleton.h>
+#include <Kernel/Devices/DeviceManagement.h>
+#include <Kernel/FileSystem/InodeMetadata.h>
+#include <Kernel/Sections.h>
+
+namespace Kernel {
+
+static Singleton<DeviceManagement> s_the;
+
+UNMAP_AFTER_INIT DeviceManagement::DeviceManagement()
+{
+}
+UNMAP_AFTER_INIT void DeviceManagement::initialize()
+{
+ s_the.ensure_instance();
+}
+
+UNMAP_AFTER_INIT void DeviceManagement::attach_null_device(NullDevice const& device)
+{
+ m_null_device = device;
+}
+
+DeviceManagement& DeviceManagement::the()
+{
+ return *s_the;
+}
+
+Device* DeviceManagement::get_device(unsigned major, unsigned minor)
+{
+ return m_devices.with_exclusive([&](auto& map) -> Device* {
+ auto it = map.find(encoded_device(major, minor));
+ if (it == map.end())
+ return nullptr;
+ return it->value;
+ });
+}
+
+void DeviceManagement::before_device_removal(Badge<Device>, Device& device)
+{
+ u32 device_id = encoded_device(device.major(), device.minor());
+ m_devices.with_exclusive([&](auto& map) -> void {
+ VERIFY(map.contains(device_id));
+ map.remove(encoded_device(device.major(), device.minor()));
+ });
+}
+
+void DeviceManagement::after_inserting_device(Badge<Device>, Device& device)
+{
+ u32 device_id = encoded_device(device.major(), device.minor());
+ m_devices.with_exclusive([&](auto& map) -> void {
+ if (map.contains(device_id)) {
+ dbgln("Already registered {},{}: {}", device.major(), device.minor(), device.class_name());
+ VERIFY_NOT_REACHED();
+ }
+ auto result = map.set(device_id, &device);
+ if (result != AK::HashSetResult::InsertedNewEntry) {
+ dbgln("Failed to register {},{}: {}", device.major(), device.minor(), device.class_name());
+ VERIFY_NOT_REACHED();
+ }
+ });
+}
+
+void DeviceManagement::for_each(Function<void(Device&)> callback)
+{
+ m_devices.with_exclusive([&](auto& map) -> void {
+ for (auto& entry : map)
+ callback(*entry.value);
+ });
+}
+
+NullDevice& DeviceManagement::null_device()
+{
+ return *m_null_device;
+}
+
+NullDevice const& DeviceManagement::null_device() const
+{
+ return *m_null_device;
+}
+
+}