rpcsx/rpcsx-os/iodev/hdd.cpp
DH 65e653f5ef [rpcsx-os/orbis-kernel] random bugfixes
ipmi: fixed respond sync, get message, try get message, try send message
event: detach event emitter from file
signals: basic implementation
linker: fixed zero symbol relocation, fixed exec relocation
shared_cv/mutex: implement eintr response support
shared_cv: fixed possible loop instead of wait
ipmi: implement invoke async, respond async, get result, get client app id, client get name
rpcsx-os: add safemode flag
2024-01-13 20:57:02 +03:00

92 lines
2.6 KiB
C++

#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 <span>
struct HddFile : orbis::File {};
struct HddDevice : IoDevice {
std::uint64_t size;
HddDevice(std::uint64_t size) : size(size) {}
orbis::ErrorCode open(orbis::Ref<orbis::File> *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<HddDevice>();
if (request == 0x40046480) { // DIOCGSECTORSIZE
return orbis::uwrite(orbis::ptr<orbis::uint>(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<orbis::ulong>(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<orbis::File> *fs, const char *path,
std::uint32_t flags, std::uint32_t mode,
orbis::Thread *thread) {
auto newFile = orbis::knew<HddFile>();
newFile->ops = &fsOps;
newFile->device = this;
*fs = newFile;
return {};
}
IoDevice *createHddCharacterDevice(std::uint64_t size) {
return orbis::knew<HddDevice>(size);
}