diff options
author | Anton Kling <anton@kling.gg> | 2024-11-23 15:00:16 +0100 |
---|---|---|
committer | Anton Kling <anton@kling.gg> | 2024-11-23 16:31:25 +0100 |
commit | 02194a2557b50bf850ff2a26d6e1eb6e7b67bb7c (patch) | |
tree | ae6836f20fc6b06c3f82429a8fc4683cd3767e75 /kernel/fs/vfs.c | |
parent | 934c30d35a3ca0e0bf6cd8709d779e060e27f6d2 (diff) |
vfs: Add O_APPEND support + refactoring
Diffstat (limited to 'kernel/fs/vfs.c')
-rw-r--r-- | kernel/fs/vfs.c | 44 |
1 files changed, 44 insertions, 0 deletions
diff --git a/kernel/fs/vfs.c b/kernel/fs/vfs.c index 1f28f13..25e2423 100644 --- a/kernel/fs/vfs.c +++ b/kernel/fs/vfs.c @@ -382,6 +382,50 @@ int vfs_pwrite(int fd, void *buf, u64 count, u64 offset) { return rc; } +// FIXME: These should be in a shared header file with libc +#define SEEK_SET 0 +#define SEEK_CUR 1 +#define SEEK_END 2 + +int vfs_lseek(int fd, int offset, int whence) { + vfs_fd_t *fd_ptr = get_vfs_fd(fd, NULL); + if (!fd_ptr) { + return -EBADF; + } + + off_t ret_offset = fd_ptr->offset; + switch (whence) { + case SEEK_SET: + ret_offset = offset; + break; + case SEEK_CUR: + ret_offset += offset; + break; + case SEEK_END: + assert(fd_ptr->inode); + ret_offset = fd_ptr->inode->file_size + offset; + break; + default: + return -EINVAL; + break; + } + fd_ptr->offset = ret_offset; + return ret_offset; +} + +int vfs_write(int fd, const char *buf, u64 count) { + vfs_fd_t *fd_ptr = get_vfs_fd(fd, NULL); + if (!fd_ptr) { + return -EBADF; + } + if (fd_ptr->mode & O_APPEND) { + vfs_lseek(fd, 0, SEEK_END); + } + int rc = vfs_pwrite(fd, (char *)buf, count, fd_ptr->offset); + fd_ptr->offset += rc; + return rc; +} + vfs_vm_object_t *vfs_get_vm_object(int fd, u64 length, u64 offset) { vfs_fd_t *vfs_fd = get_vfs_fd(fd, NULL); if (!vfs_fd) { |