forked from Bananymous/banan-os
133 lines
2.8 KiB
C++
133 lines
2.8 KiB
C++
#include <kernel/DeviceManager.h>
|
|
#include <kernel/LockGuard.h>
|
|
#include <kernel/PCI.h>
|
|
#include <kernel/Process.h>
|
|
#include <kernel/Storage/ATAController.h>
|
|
|
|
namespace Kernel
|
|
{
|
|
|
|
DeviceManager& DeviceManager::get()
|
|
{
|
|
static DeviceManager instance;
|
|
return instance;
|
|
}
|
|
|
|
void DeviceManager::initialize_pci_devices()
|
|
{
|
|
for (const auto& pci_device : PCI::get().devices())
|
|
{
|
|
switch (pci_device.class_code())
|
|
{
|
|
case 0x01:
|
|
{
|
|
StorageController* controller = nullptr;
|
|
switch (pci_device.subclass())
|
|
{
|
|
case 0x01:
|
|
if (auto res = ATAController::create(pci_device); res.is_error())
|
|
dprintln("{}", res.error());
|
|
else
|
|
controller = res.value();
|
|
break;
|
|
default:
|
|
dprintln("unsupported storage device (pci {2H}.{2H}.{2H})", pci_device.class_code(), pci_device.subclass(), pci_device.prog_if());
|
|
break;
|
|
}
|
|
|
|
if (controller)
|
|
{
|
|
add_device(controller);
|
|
for (auto* device : controller->devices())
|
|
{
|
|
add_device(device);
|
|
if (auto res = device->initialize_partitions(); res.is_error())
|
|
dprintln("{}", res.error());
|
|
else
|
|
{
|
|
for (auto* partition : device->partitions())
|
|
add_device(partition);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void DeviceManager::initialize_updater()
|
|
{
|
|
Process::create_kernel(
|
|
[](void*)
|
|
{
|
|
while (true)
|
|
{
|
|
DeviceManager::get().update();
|
|
PIT::sleep(1);
|
|
}
|
|
}, nullptr
|
|
);
|
|
}
|
|
|
|
ino_t DeviceManager::get_next_ino() const
|
|
{
|
|
static ino_t next_ino = 1;
|
|
return next_ino++;
|
|
}
|
|
|
|
dev_t DeviceManager::get_next_rdev() const
|
|
{
|
|
static dev_t next_dev = 1;
|
|
return next_dev++;
|
|
}
|
|
|
|
uint8_t DeviceManager::get_next_input_dev() const
|
|
{
|
|
static uint8_t next_dev = 0;
|
|
return next_dev++;
|
|
}
|
|
|
|
void DeviceManager::update()
|
|
{
|
|
LockGuard _(m_lock);
|
|
for (Device* device : m_devices)
|
|
device->update();
|
|
}
|
|
|
|
void DeviceManager::add_device(Device* device)
|
|
{
|
|
LockGuard _(m_lock);
|
|
MUST(m_devices.push_back(device));
|
|
}
|
|
|
|
BAN::ErrorOr<BAN::RefPtr<Inode>> DeviceManager::read_directory_inode(BAN::StringView name)
|
|
{
|
|
LockGuard _(m_lock);
|
|
if (name == "."sv || name == ".."sv)
|
|
return BAN::RefPtr<Inode>(this);
|
|
for (Device* device : m_devices)
|
|
if (device->name() == name)
|
|
return BAN::RefPtr<Inode>(device);
|
|
return BAN::Error::from_errno(ENOENT);
|
|
}
|
|
|
|
BAN::ErrorOr<BAN::Vector<BAN::String>> DeviceManager::read_directory_entries(size_t index)
|
|
{
|
|
BAN::Vector<BAN::String> result;
|
|
if (index > 0)
|
|
return result;
|
|
|
|
LockGuard _(m_lock);
|
|
TRY(result.reserve(m_devices.size() + 2));
|
|
MUST(result.emplace_back("."sv));
|
|
MUST(result.emplace_back(".."sv));
|
|
for (Device* device : m_devices)
|
|
MUST(result.emplace_back(device->name()));
|
|
return result;
|
|
}
|
|
|
|
|
|
} |