#include "io-device.hpp" #include "orbis/KernelAllocator.hpp" #include "orbis/error/ErrorCode.hpp" #include "orbis/file.hpp" #include "orbis/thread/Thread.hpp" #include "orbis/uio.hpp" #include "orbis/utils/Logs.hpp" #include struct HddFile : orbis::File {}; struct HddDevice : IoDevice { std::uint64_t size; HddDevice(std::uint64_t size) : size(size) {} orbis::ErrorCode open(orbis::Ref *fs, const char *path, std::uint32_t flags, std::uint32_t mode, orbis::Thread *thread) override; }; static_assert(0x120 - 24 == 0x108); static orbis::ErrorCode hdd_ioctl(orbis::File *fs, std::uint64_t request, void *argp, orbis::Thread *thread) { auto device = fs->device.cast(); if (request == 0x40046480) { // DIOCGSECTORSIZE return orbis::uwrite(orbis::ptr(argp), 0x1000u); } if (request == 0x40086481) { // hdd size if (device->size == 0) { ORBIS_LOG_FATAL("Unknown hdd size request", request); thread->where(); } return orbis::uwrite(orbis::ptr(argp), device->size); } ORBIS_LOG_FATAL("Unhandled hdd ioctl", request); thread->where(); return {}; } static orbis::ErrorCode hdd_read(orbis::File *file, orbis::Uio *uio, orbis::Thread *thread) { auto dev = file->device.get(); ORBIS_LOG_ERROR(__FUNCTION__, uio->offset); for (auto vec : std::span(uio->iov, uio->iovcnt)) { std::memset(vec.base, 0, vec.len); // HACK: dummy UFS header if (uio->offset == 0x10000) { *(uint32_t *)((char *)vec.base + 0x55c) = 0x19540119; *(uint64_t *)((char *)vec.base + 0x3e8) = 0x1000; *(uint8_t *)((char *)vec.base + 0xd3) = 1; *(uint8_t *)((char *)vec.base + 0xd1) = 1; } uio->offset += vec.len; } return {}; } static orbis::ErrorCode hdd_stat(orbis::File *fs, orbis::Stat *sb, orbis::Thread *thread) { // TODO ORBIS_LOG_ERROR(__FUNCTION__); *sb = {}; sb->mode = 0x2000; return {}; } static const orbis::FileOps fsOps = { .ioctl = hdd_ioctl, .read = hdd_read, .stat = hdd_stat, }; orbis::ErrorCode HddDevice::open(orbis::Ref *fs, const char *path, std::uint32_t flags, std::uint32_t mode, orbis::Thread *thread) { auto newFile = orbis::knew(); newFile->ops = &fsOps; newFile->device = this; *fs = newFile; return {}; } IoDevice *createHddCharacterDevice(std::uint64_t size) { return orbis::knew(size); }