2023-09-03 21:22:20 +02:00
|
|
|
#include "io-device.hpp"
|
|
|
|
|
#include "orbis/KernelAllocator.hpp"
|
2023-11-02 14:26:58 +01:00
|
|
|
#include "orbis/error/ErrorCode.hpp"
|
2023-09-03 21:22:20 +02:00
|
|
|
#include "orbis/file.hpp"
|
2024-01-13 18:57:02 +01:00
|
|
|
#include "orbis/thread/Thread.hpp"
|
2023-11-02 14:26:58 +01:00
|
|
|
#include "orbis/uio.hpp"
|
2023-09-03 21:22:20 +02:00
|
|
|
#include "orbis/utils/Logs.hpp"
|
2023-11-02 14:26:58 +01:00
|
|
|
#include <span>
|
|
|
|
|
#include <unistd.h>
|
2023-09-03 21:22:20 +02:00
|
|
|
|
|
|
|
|
struct ConsoleFile : orbis::File {};
|
2023-11-02 14:26:58 +01:00
|
|
|
struct ConsoleDevice : IoDevice {
|
|
|
|
|
int inputFd;
|
|
|
|
|
int outputFd;
|
|
|
|
|
|
|
|
|
|
ConsoleDevice(int inputFd, int outputFd)
|
|
|
|
|
: inputFd(inputFd), outputFd(outputFd) {}
|
|
|
|
|
|
|
|
|
|
orbis::ErrorCode open(orbis::Ref<orbis::File> *file, const char *path,
|
|
|
|
|
std::uint32_t flags, std::uint32_t mode,
|
|
|
|
|
orbis::Thread *thread) override;
|
|
|
|
|
};
|
2023-09-03 21:22:20 +02:00
|
|
|
|
|
|
|
|
static orbis::ErrorCode console_ioctl(orbis::File *file, std::uint64_t request,
|
2023-11-02 14:26:58 +01:00
|
|
|
void *argp, orbis::Thread *thread) {
|
2023-09-03 21:22:20 +02:00
|
|
|
|
|
|
|
|
ORBIS_LOG_FATAL("Unhandled console ioctl", request);
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
2023-11-02 14:26:58 +01:00
|
|
|
static orbis::ErrorCode console_read(orbis::File *file, orbis::Uio *uio,
|
|
|
|
|
orbis::Thread *thread) {
|
|
|
|
|
auto dev = dynamic_cast<ConsoleDevice *>(file->device.get());
|
|
|
|
|
|
|
|
|
|
for (auto vec : std::span(uio->iov, uio->iovcnt)) {
|
|
|
|
|
auto result = ::read(dev->inputFd, vec.base, vec.len);
|
|
|
|
|
|
|
|
|
|
if (result < 0) {
|
|
|
|
|
return orbis::ErrorCode::IO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uio->offset += result;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static orbis::ErrorCode console_write(orbis::File *file, orbis::Uio *uio,
|
|
|
|
|
orbis::Thread *thread) {
|
|
|
|
|
auto dev = dynamic_cast<ConsoleDevice *>(file->device.get());
|
|
|
|
|
|
|
|
|
|
for (auto vec : std::span(uio->iov, uio->iovcnt)) {
|
2023-12-31 12:30:49 +01:00
|
|
|
uio->offset += vec.len;
|
2024-01-13 18:57:02 +01:00
|
|
|
|
2023-11-02 14:26:58 +01:00
|
|
|
::write(dev->outputFd, vec.base, vec.len);
|
2023-12-31 12:30:49 +01:00
|
|
|
::write(2, vec.base, vec.len);
|
2023-11-02 14:26:58 +01:00
|
|
|
}
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-03 21:22:20 +02:00
|
|
|
static const orbis::FileOps fileOps = {
|
|
|
|
|
.ioctl = console_ioctl,
|
2023-11-02 14:26:58 +01:00
|
|
|
.read = console_read,
|
|
|
|
|
.write = console_write,
|
2023-09-03 21:22:20 +02:00
|
|
|
};
|
|
|
|
|
|
2023-11-02 14:26:58 +01:00
|
|
|
orbis::ErrorCode ConsoleDevice::open(orbis::Ref<orbis::File> *file,
|
|
|
|
|
const char *path, std::uint32_t flags,
|
|
|
|
|
std::uint32_t mode,
|
|
|
|
|
orbis::Thread *thread) {
|
|
|
|
|
auto newFile = orbis::knew<ConsoleFile>();
|
|
|
|
|
newFile->ops = &fileOps;
|
|
|
|
|
newFile->device = this;
|
2023-09-03 21:22:20 +02:00
|
|
|
|
2023-11-02 14:26:58 +01:00
|
|
|
*file = newFile;
|
|
|
|
|
return {};
|
|
|
|
|
}
|
2023-09-03 21:22:20 +02:00
|
|
|
|
2023-11-02 14:26:58 +01:00
|
|
|
IoDevice *createConsoleCharacterDevice(int inputFd, int outputFd) {
|
|
|
|
|
return orbis::knew<ConsoleDevice>(inputFd, outputFd);
|
|
|
|
|
}
|