Luna/kernel/src/sys/file.cpp

261 lines
6.6 KiB
C++
Raw Normal View History

2023-03-11 17:02:50 +00:00
#include "Log.h"
#include "fs/Pipe.h"
2023-03-11 17:02:50 +00:00
#include "fs/VFS.h"
#include "memory/MemoryManager.h"
#include "sys/Syscall.h"
#include "thread/Scheduler.h"
#include <bits/fcntl.h>
#include <bits/open-flags.h>
#include <bits/seek.h>
#include <luna/SafeArithmetic.h>
#include <sys/types.h>
2023-03-11 17:02:50 +00:00
Result<u64> sys_read(Registers* regs, SyscallArgs args)
2023-03-11 17:02:50 +00:00
{
int fd = (int)args[0];
u8* buf = (u8*)args[1];
usize size = (usize)args[2];
if (!size) return 0;
2023-03-11 17:02:50 +00:00
if (!MemoryManager::validate_user_write(buf, size)) return err(EFAULT);
Thread* current = Scheduler::current();
auto& descriptor = *TRY(current->resolve_fd(fd));
2023-03-11 17:02:50 +00:00
if (!descriptor.is_readable()) return err(EBADF);
while (descriptor.inode()->will_block_if_read())
2023-03-19 10:25:14 +00:00
{
if (descriptor.should_block()) kernel_sleep(10);
else
return err(EAGAIN);
if (current->interrupted)
{
kdbgln("signal: read interrupted by signal");
if (current->will_invoke_signal_handler()) return err(EINTR);
current->process_pending_signals(regs);
}
2023-03-19 10:25:14 +00:00
}
2023-03-19 10:21:50 +00:00
usize nread = TRY(descriptor.inode()->read(buf, descriptor.offset, size));
2023-03-11 17:02:50 +00:00
if (VFS::is_seekable(descriptor.inode())) descriptor.offset += nread;
2023-03-11 17:02:50 +00:00
return nread;
}
Result<u64> sys_write(Registers*, SyscallArgs args)
{
int fd = (int)args[0];
const u8* buf = (const u8*)args[1];
usize size = (usize)args[2];
if (!size) return 0;
if (!MemoryManager::validate_user_read(buf, size)) return err(EFAULT);
Thread* current = Scheduler::current();
auto& descriptor = *TRY(current->resolve_fd(fd));
if (!descriptor.is_writable()) return err(EBADF);
if (descriptor.should_append() && VFS::is_seekable(descriptor.inode()))
descriptor.offset = descriptor.inode()->metadata().size;
usize nwritten = TRY(descriptor.inode()->write(buf, descriptor.offset, size));
if (VFS::is_seekable(descriptor.inode())) descriptor.offset += nwritten;
return nwritten;
}
Result<u64> sys_lseek(Registers*, SyscallArgs args)
{
int fd = (int)args[0];
off_t offset = (long)args[1];
int whence = (int)args[2];
Thread* current = Scheduler::current();
auto& descriptor = *TRY(current->resolve_fd(fd));
if (descriptor.inode()->type() == VFS::InodeType::FIFO) return err(ESPIPE);
if (!VFS::is_seekable(descriptor.inode())) return descriptor.offset;
off_t new_offset;
switch (whence)
{
case SEEK_SET: new_offset = offset; break;
case SEEK_CUR: new_offset = TRY(safe_add((long)descriptor.offset, offset)); break;
case SEEK_END: new_offset = TRY(safe_add((long)descriptor.inode()->metadata().size, offset)); break;
default: return err(EINVAL);
}
if (new_offset < 0) return err(EINVAL);
descriptor.offset = (usize)new_offset;
return (u64)new_offset;
}
Result<u64> sys_fcntl(Registers*, SyscallArgs args)
{
int fd = (int)args[0];
int cmd = (int)args[1];
Thread* current = Scheduler::current();
auto& descriptor = *TRY(current->resolve_fd(fd));
bool is_cloexec = true;
switch (cmd)
{
case F_DUPFD: is_cloexec = false; [[fallthrough]];
case F_DUPFD_CLOEXEC: {
int arg = (int)args[2];
int new_fd = TRY(current->allocate_fd(arg));
current->fd_table[new_fd] = descriptor;
if (is_cloexec) current->fd_table[new_fd]->flags() |= O_CLOEXEC;
else
current->fd_table[new_fd]->flags() &= ~O_CLOEXEC;
return (u64)new_fd;
}
case F_GETFD: return (u64) !!(descriptor.flags() & O_CLOEXEC);
case F_SETFD: {
int arg = (int)args[2];
if (arg == FD_CLOEXEC) descriptor.flags() |= O_CLOEXEC;
else
descriptor.flags() &= ~O_CLOEXEC;
return 0;
}
case F_GETFL: return (u64)(descriptor.flags() & ~O_CLOEXEC);
case F_SETFL: {
int arg = (int)args[2];
descriptor.flags() &= ~(O_APPEND | O_NONBLOCK);
arg &= (O_APPEND | O_NONBLOCK);
descriptor.flags() |= arg;
return 0;
}
default: return err(EINVAL);
}
}
2023-04-09 09:22:57 +00:00
Result<u64> sys_ioctl(Registers*, SyscallArgs args)
{
int fd = (int)args[0];
int request = (int)args[1];
void* arg = (void*)args[2];
Thread* current = Scheduler::current();
auto& descriptor = *TRY(current->resolve_fd(fd));
return descriptor.inode()->ioctl(request, arg);
2023-04-09 09:22:57 +00:00
}
2023-04-25 18:37:30 +00:00
2023-07-11 10:05:09 +00:00
Result<u64> sys_isatty(Registers*, SyscallArgs args)
{
int fd = (int)args[0];
Thread* current = Scheduler::current();
auto& descriptor = *TRY(current->resolve_fd(fd));
return descriptor.inode()->isatty();
2023-07-11 10:05:09 +00:00
}
2023-04-25 18:37:30 +00:00
Result<u64> sys_dup2(Registers*, SyscallArgs args)
{
int oldfd = (int)args[0];
int newfd = (int)args[1];
Thread* current = Scheduler::current();
if (newfd < 0 || newfd >= FD_MAX) return err(EBADF);
auto descriptor = *TRY(current->resolve_fd(oldfd));
if (newfd == oldfd) return (u64)newfd;
current->fd_table[newfd] = descriptor;
current->fd_table[newfd]->flags() &= ~O_CLOEXEC;
2023-04-25 18:37:30 +00:00
return (u64)newfd;
}
Result<u64> sys_pipe(Registers*, SyscallArgs args)
{
int* pfds = (int*)args[0];
Thread* current = Scheduler::current();
int rfd = TRY(current->allocate_fd(0));
int wfd = TRY(current->allocate_fd(rfd + 1));
if (!MemoryManager::copy_to_user_typed(pfds, &rfd)) return err(EFAULT);
if (!MemoryManager::copy_to_user_typed(pfds + 1, &wfd)) return err(EFAULT);
SharedPtr<VFS::Inode> rpipe;
SharedPtr<VFS::Inode> wpipe;
TRY(Pipe::create(rpipe, wpipe));
current->fd_table[rfd] = FileDescriptor { TRY(make_shared<OpenFileDescription>(rpipe, O_RDONLY)), 0 };
current->fd_table[wfd] = FileDescriptor { TRY(make_shared<OpenFileDescription>(wpipe, O_WRONLY)), 0 };
return 0;
}
2023-05-26 20:27:49 +00:00
Result<u64> sys_umask(Registers*, SyscallArgs args)
{
mode_t new_umask = (mode_t)args[0];
auto* current = Scheduler::current();
mode_t old_umask = current->umask;
current->umask = new_umask & 0777;
return old_umask;
}
Result<u64> sys_truncate(Registers*, SyscallArgs args)
{
auto path = TRY(MemoryManager::strdup_from_user(args[0]));
size_t length = (size_t)args[1];
auto* current = Scheduler::current();
auto inode = TRY(VFS::resolve_path(path.chars(), current->auth, current->current_directory));
if (!VFS::can_write(inode, current->auth)) return err(EACCES);
TRY(inode->truncate(length));
return 0;
}
Result<u64> sys_ftruncate(Registers*, SyscallArgs args)
{
int fd = (int)args[0];
size_t length = (size_t)args[1];
auto* current = Scheduler::current();
auto description = TRY(current->resolve_fd(fd))->description;
if (!(description->flags & O_WRONLY)) return err(EBADF);
TRY(description->inode->truncate(length));
return 0;
}