2023-06-23 02:28:14 +02:00
|
|
|
#include "io-device.hpp"
|
2023-07-04 18:19:17 +02:00
|
|
|
#include "orbis/KernelAllocator.hpp"
|
2023-07-29 18:53:34 +02:00
|
|
|
#include "orbis/file.hpp"
|
|
|
|
|
#include "orbis/stat.hpp"
|
|
|
|
|
#include "orbis/uio.hpp"
|
2023-07-19 17:04:26 +02:00
|
|
|
#include "orbis/utils/Logs.hpp"
|
2023-07-29 21:46:28 +02:00
|
|
|
#include "vm.hpp"
|
2023-07-29 18:53:34 +02:00
|
|
|
#include <cerrno>
|
2023-06-23 02:28:14 +02:00
|
|
|
#include <fcntl.h>
|
2023-07-29 18:53:34 +02:00
|
|
|
#include <span>
|
2023-06-23 02:28:14 +02:00
|
|
|
#include <string>
|
2023-07-29 21:46:28 +02:00
|
|
|
#include <sys/mman.h>
|
2023-07-19 17:04:26 +02:00
|
|
|
#include <sys/socket.h>
|
2023-07-29 18:53:34 +02:00
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#include <sys/uio.h>
|
2023-06-23 02:28:14 +02:00
|
|
|
#include <unistd.h>
|
2023-07-29 18:53:34 +02:00
|
|
|
#include <vector>
|
2023-06-23 02:28:14 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
struct HostFile : orbis::File {
|
|
|
|
|
int hostFd = -1;
|
2023-06-23 02:28:14 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
~HostFile() {
|
|
|
|
|
if (hostFd > 0) {
|
|
|
|
|
::close(hostFd);
|
|
|
|
|
}
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
2023-07-29 18:53:34 +02:00
|
|
|
};
|
2023-06-23 02:28:14 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
struct SocketFile : orbis::File {
|
|
|
|
|
orbis::utils::kstring name;
|
|
|
|
|
int hostFd = -1;
|
2023-06-23 02:28:14 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
~SocketFile() {
|
|
|
|
|
if (hostFd > 0) {
|
|
|
|
|
::close(hostFd);
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-06-23 02:28:14 +02:00
|
|
|
};
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
struct HostFsDevice : IoDevice {
|
|
|
|
|
orbis::kstring hostPath;
|
2023-06-23 02:28:14 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
explicit HostFsDevice(orbis::kstring path) : hostPath(std::move(path)) {}
|
|
|
|
|
orbis::ErrorCode open(orbis::Ref<orbis::File> *file, const char *path,
|
2023-07-29 21:46:28 +02:00
|
|
|
std::uint32_t flags, std::uint32_t mode,
|
|
|
|
|
orbis::Thread *thread) override;
|
2023-07-19 17:04:26 +02:00
|
|
|
};
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
static orbis::ErrorCode convertErrno() {
|
|
|
|
|
switch (auto error = errno) {
|
|
|
|
|
case EPERM:
|
|
|
|
|
return orbis::ErrorCode::PERM;
|
|
|
|
|
case ENOENT:
|
|
|
|
|
return orbis::ErrorCode::NOENT;
|
|
|
|
|
case EBADF:
|
|
|
|
|
return orbis::ErrorCode::BADF;
|
|
|
|
|
case EIO:
|
|
|
|
|
return orbis::ErrorCode::IO;
|
|
|
|
|
case EACCES:
|
|
|
|
|
return orbis::ErrorCode::ACCES;
|
|
|
|
|
case EEXIST:
|
|
|
|
|
return orbis::ErrorCode::EXIST;
|
|
|
|
|
case EBUSY:
|
|
|
|
|
return orbis::ErrorCode::BUSY;
|
|
|
|
|
case ENOTDIR:
|
|
|
|
|
return orbis::ErrorCode::NOTDIR;
|
|
|
|
|
case EISDIR:
|
|
|
|
|
return orbis::ErrorCode::ISDIR;
|
|
|
|
|
case EFBIG:
|
|
|
|
|
return orbis::ErrorCode::FBIG;
|
|
|
|
|
case ENOSPC:
|
|
|
|
|
return orbis::ErrorCode::NOSPC;
|
|
|
|
|
case ESPIPE:
|
|
|
|
|
return orbis::ErrorCode::SPIPE;
|
|
|
|
|
case EPIPE:
|
|
|
|
|
return orbis::ErrorCode::PIPE;
|
2023-07-29 23:34:52 +02:00
|
|
|
case EINVAL:
|
|
|
|
|
return orbis::ErrorCode::INVAL;
|
2023-07-29 18:53:34 +02:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
ORBIS_LOG_ERROR("Unconverted errno", error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return orbis::ErrorCode::IO;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static orbis::ErrorCode host_read(orbis::File *file, orbis::Uio *uio,
|
|
|
|
|
orbis::Thread *) {
|
|
|
|
|
auto hostFile = static_cast<HostFile *>(file);
|
|
|
|
|
std::vector<iovec> vec;
|
|
|
|
|
for (auto entry : std::span(uio->iov, uio->iovcnt)) {
|
|
|
|
|
vec.push_back({.iov_base = entry.base, .iov_len = entry.len});
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-29 19:45:16 +02:00
|
|
|
ssize_t cnt = 0;
|
2023-07-29 18:53:34 +02:00
|
|
|
if (hostFile->hostFd == 0) {
|
2023-07-29 19:45:16 +02:00
|
|
|
for (auto io : vec) {
|
|
|
|
|
cnt += ::read(hostFile->hostFd, io.iov_base, io.iov_len);
|
|
|
|
|
|
|
|
|
|
if (cnt != io.iov_len) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-29 18:53:34 +02:00
|
|
|
} else {
|
|
|
|
|
cnt = ::preadv(hostFile->hostFd, vec.data(), vec.size(), uio->offset);
|
|
|
|
|
}
|
|
|
|
|
if (cnt < 0) {
|
|
|
|
|
return convertErrno();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uio->resid -= cnt;
|
|
|
|
|
uio->offset += cnt;
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static orbis::ErrorCode host_write(orbis::File *file, orbis::Uio *uio,
|
|
|
|
|
orbis::Thread *) {
|
|
|
|
|
auto hostFile = static_cast<HostFile *>(file);
|
|
|
|
|
std::vector<iovec> vec;
|
|
|
|
|
for (auto entry : std::span(uio->iov, uio->iovcnt)) {
|
|
|
|
|
vec.push_back({.iov_base = entry.base, .iov_len = entry.len});
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-29 19:45:16 +02:00
|
|
|
ssize_t cnt = 0;
|
2023-07-29 18:53:34 +02:00
|
|
|
if (hostFile->hostFd == 1 || hostFile->hostFd == 2) {
|
2023-07-29 19:45:16 +02:00
|
|
|
for (auto io : vec) {
|
|
|
|
|
cnt += ::write(hostFile->hostFd, io.iov_base, io.iov_len);
|
|
|
|
|
|
|
|
|
|
if (cnt != io.iov_len) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-29 18:53:34 +02:00
|
|
|
} else {
|
|
|
|
|
cnt = ::pwritev(hostFile->hostFd, vec.data(), vec.size(), uio->offset);
|
|
|
|
|
}
|
|
|
|
|
if (cnt < 0) {
|
|
|
|
|
return convertErrno();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uio->resid -= cnt;
|
|
|
|
|
uio->offset += cnt;
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-29 21:46:28 +02:00
|
|
|
static orbis::ErrorCode host_mmap(orbis::File *file, void **address,
|
|
|
|
|
std::uint64_t size, std::int32_t prot,
|
|
|
|
|
std::int32_t flags, std::int64_t offset,
|
|
|
|
|
orbis::Thread *thread) {
|
|
|
|
|
auto hostFile = static_cast<HostFile *>(file);
|
|
|
|
|
|
|
|
|
|
auto result =
|
|
|
|
|
rx::vm::map(*address, size, prot, flags, rx::vm::kMapInternalReserveOnly);
|
|
|
|
|
|
|
|
|
|
if (result == (void *)-1) {
|
|
|
|
|
return orbis::ErrorCode::NOMEM;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result = ::mmap(result, size, prot & rx::vm::kMapProtCpuAll,
|
|
|
|
|
MAP_SHARED | MAP_FIXED, hostFile->hostFd, offset);
|
|
|
|
|
if (result == (void *)-1) {
|
|
|
|
|
auto result = convertErrno();
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::printf("shm mapped at %p-%p\n", result, (char *)result + size);
|
|
|
|
|
|
|
|
|
|
*address = result;
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
static orbis::ErrorCode host_stat(orbis::File *file, orbis::Stat *sb,
|
|
|
|
|
orbis::Thread *thread) {
|
|
|
|
|
auto hostFile = static_cast<HostFile *>(file);
|
|
|
|
|
struct stat hostStat;
|
|
|
|
|
::fstat(hostFile->hostFd, &hostStat);
|
|
|
|
|
sb->dev = hostStat.st_dev; // TODO
|
|
|
|
|
sb->ino = hostStat.st_ino;
|
|
|
|
|
sb->mode = hostStat.st_mode;
|
|
|
|
|
sb->nlink = hostStat.st_nlink;
|
|
|
|
|
sb->uid = hostStat.st_uid;
|
|
|
|
|
sb->gid = hostStat.st_gid;
|
|
|
|
|
sb->rdev = hostStat.st_rdev;
|
|
|
|
|
sb->atim = {
|
|
|
|
|
.sec = static_cast<uint64_t>(hostStat.st_atim.tv_sec),
|
|
|
|
|
.nsec = static_cast<uint64_t>(hostStat.st_atim.tv_nsec),
|
|
|
|
|
};
|
|
|
|
|
sb->mtim = {
|
|
|
|
|
.sec = static_cast<uint64_t>(hostStat.st_mtim.tv_sec),
|
|
|
|
|
.nsec = static_cast<uint64_t>(hostStat.st_mtim.tv_nsec),
|
|
|
|
|
};
|
|
|
|
|
sb->ctim = {
|
|
|
|
|
.sec = static_cast<uint64_t>(hostStat.st_mtim.tv_sec),
|
|
|
|
|
.nsec = static_cast<uint64_t>(hostStat.st_mtim.tv_nsec),
|
|
|
|
|
};
|
|
|
|
|
sb->size = hostStat.st_size;
|
|
|
|
|
sb->blocks = hostStat.st_blocks;
|
|
|
|
|
sb->blksize = hostStat.st_blksize;
|
|
|
|
|
// TODO
|
|
|
|
|
sb->flags = 0;
|
|
|
|
|
sb->gen = 0;
|
|
|
|
|
sb->lspare = 0;
|
|
|
|
|
sb->birthtim = {
|
|
|
|
|
.sec = static_cast<uint64_t>(hostStat.st_mtim.tv_sec),
|
|
|
|
|
.nsec = static_cast<uint64_t>(hostStat.st_mtim.tv_nsec),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return {};
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
static orbis::ErrorCode host_truncate(orbis::File *file, std::uint64_t len,
|
|
|
|
|
orbis::Thread *thread) {
|
|
|
|
|
auto hostFile = static_cast<HostFile *>(file);
|
|
|
|
|
if (::ftruncate(hostFile->hostFd, len)) {
|
|
|
|
|
return convertErrno();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static orbis::ErrorCode socket_ioctl(orbis::File *file, std::uint64_t request,
|
|
|
|
|
void *argp, orbis::Thread *thread) {
|
|
|
|
|
auto soc = static_cast<SocketFile *>(file);
|
|
|
|
|
ORBIS_LOG_FATAL("Unhandled socket ioctl", request, soc->name);
|
|
|
|
|
return {};
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
static const orbis::FileOps hostOps = {
|
|
|
|
|
.read = host_read,
|
|
|
|
|
.write = host_write,
|
|
|
|
|
.truncate = host_truncate,
|
|
|
|
|
.stat = host_stat,
|
2023-07-29 21:46:28 +02:00
|
|
|
.mmap = host_mmap,
|
2023-07-29 18:53:34 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static const orbis::FileOps socketOps = {
|
|
|
|
|
.ioctl = socket_ioctl,
|
|
|
|
|
};
|
2023-06-23 02:28:14 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
IoDevice *createHostIoDevice(orbis::kstring hostPath) {
|
|
|
|
|
return orbis::knew<HostFsDevice>(std::move(hostPath));
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
orbis::ErrorCode createSocket(orbis::Ref<orbis::File> *file,
|
|
|
|
|
orbis::kstring name, int dom, int type,
|
|
|
|
|
int prot) {
|
|
|
|
|
auto fd = ::socket(dom, type, prot);
|
|
|
|
|
if (fd < 0) {
|
|
|
|
|
return convertErrno();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
auto s = orbis::knew<SocketFile>();
|
|
|
|
|
s->name = std::move(name);
|
|
|
|
|
s->hostFd = fd;
|
|
|
|
|
s->ops = &socketOps;
|
|
|
|
|
*file = s;
|
|
|
|
|
return {};
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
orbis::ErrorCode HostFsDevice::open(orbis::Ref<orbis::File> *file,
|
|
|
|
|
const char *path, std::uint32_t flags,
|
2023-07-29 21:46:28 +02:00
|
|
|
std::uint32_t mode, orbis::Thread *thread) {
|
2023-07-29 18:53:34 +02:00
|
|
|
auto realPath = hostPath + "/" + path;
|
2023-06-23 02:28:14 +02:00
|
|
|
|
|
|
|
|
int realFlags = flags & O_ACCMODE;
|
|
|
|
|
flags &= ~O_ACCMODE;
|
|
|
|
|
|
|
|
|
|
if ((flags & kOpenFlagAppend) != 0) {
|
|
|
|
|
realFlags |= O_APPEND;
|
|
|
|
|
flags &= ~kOpenFlagAppend;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & kOpenFlagNonBlock) != 0) {
|
|
|
|
|
realFlags |= O_NONBLOCK;
|
|
|
|
|
flags &= ~kOpenFlagNonBlock;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & kOpenFlagFsync) != 0) {
|
|
|
|
|
realFlags |= O_FSYNC;
|
|
|
|
|
flags &= ~kOpenFlagFsync;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & kOpenFlagAsync) != 0) {
|
|
|
|
|
realFlags |= O_ASYNC;
|
|
|
|
|
flags &= ~kOpenFlagAsync;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & kOpenFlagTrunc) != 0) {
|
|
|
|
|
realFlags |= O_TRUNC;
|
|
|
|
|
flags &= ~kOpenFlagTrunc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & kOpenFlagCreat) != 0) {
|
|
|
|
|
realFlags |= O_CREAT;
|
|
|
|
|
flags &= ~kOpenFlagCreat;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & kOpenFlagExcl) != 0) {
|
|
|
|
|
realFlags |= O_EXCL;
|
|
|
|
|
flags &= ~kOpenFlagExcl;
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
if ((flags & kOpenFlagDirectory) != 0) {
|
|
|
|
|
realFlags |= O_DIRECTORY;
|
|
|
|
|
flags &= ~kOpenFlagDirectory;
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-23 02:28:14 +02:00
|
|
|
if (flags != 0) {
|
2023-07-29 18:53:34 +02:00
|
|
|
ORBIS_LOG_ERROR("host_open: ***ERROR*** Unhandled open flags", flags);
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int hostFd = ::open(realPath.c_str(), realFlags, 0777);
|
|
|
|
|
|
|
|
|
|
if (hostFd < 0) {
|
2023-07-29 18:53:34 +02:00
|
|
|
auto error = convertErrno();
|
|
|
|
|
ORBIS_LOG_ERROR("host_open failed", realPath, error);
|
|
|
|
|
return error;
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
auto newFile = orbis::knew<HostFile>();
|
|
|
|
|
newFile->hostFd = hostFd;
|
|
|
|
|
newFile->ops = &hostOps;
|
|
|
|
|
newFile->device = this;
|
|
|
|
|
*file = newFile;
|
|
|
|
|
return {};
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
|
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
orbis::File *createHostFile(int hostFd, orbis::Ref<IoDevice> device) {
|
|
|
|
|
auto newFile = orbis::knew<HostFile>();
|
|
|
|
|
newFile->hostFd = hostFd;
|
|
|
|
|
newFile->ops = &hostOps;
|
|
|
|
|
newFile->device = device;
|
|
|
|
|
return newFile;
|
2023-06-23 02:28:14 +02:00
|
|
|
}
|
2023-07-19 17:04:26 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
struct FdWrapDevice : public IoDevice {
|
|
|
|
|
int fd;
|
2023-07-19 17:04:26 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
orbis::ErrorCode open(orbis::Ref<orbis::File> *file, const char *path,
|
2023-07-29 21:46:28 +02:00
|
|
|
std::uint32_t flags, std::uint32_t mode,
|
|
|
|
|
orbis::Thread *thread) override {
|
2023-07-29 18:53:34 +02:00
|
|
|
*file = createHostFile(fd, this);
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
};
|
2023-07-19 17:04:26 +02:00
|
|
|
|
2023-07-29 18:53:34 +02:00
|
|
|
IoDevice *createFdWrapDevice(int fd) {
|
|
|
|
|
auto result = orbis::knew<FdWrapDevice>();
|
|
|
|
|
result->fd = fd;
|
|
|
|
|
return result;
|
2023-07-19 17:04:26 +02:00
|
|
|
}
|