summaryrefslogtreecommitdiff
path: root/src/kernel
diff options
context:
space:
mode:
authordzwdz2021-09-12 17:53:48 +0200
committerdzwdz2021-09-12 17:53:48 +0200
commit3ea9657aaaf02709c8f216285b095af29e76491c (patch)
tree36dfe7038f2ff829791ccdaf18b0ee71744207a6 /src/kernel
parent5090829984cd4b3238d3f42eaad2431f1fd5282b (diff)
implement part of `fs_respond`
Diffstat (limited to 'src/kernel')
-rw-r--r--src/kernel/proc.c1
-rw-r--r--src/kernel/proc.h1
-rw-r--r--src/kernel/syscalls.c13
-rw-r--r--src/kernel/vfs/request.c28
4 files changed, 32 insertions, 11 deletions
diff --git a/src/kernel/proc.c b/src/kernel/proc.c
index 21a1d54..e82971c 100644
--- a/src/kernel/proc.c
+++ b/src/kernel/proc.c
@@ -19,6 +19,7 @@ struct process *process_seed(void) {
proc->parent = NULL;
proc->mount = vfs_mount_seed();
proc->id = next_pid++;
+ proc->handled_req = NULL;
process_first = proc;
diff --git a/src/kernel/proc.h b/src/kernel/proc.h
index 595d62c..73a6651 100644
--- a/src/kernel/proc.h
+++ b/src/kernel/proc.h
@@ -36,6 +36,7 @@ struct process {
int __user *len;
} awaited_req; // PS_WAITS4REQUEST
};
+ struct vfs_request *handled_req;
struct vfs_mount *mount;
diff --git a/src/kernel/syscalls.c b/src/kernel/syscalls.c
index 1180197..b247619 100644
--- a/src/kernel/syscalls.c
+++ b/src/kernel/syscalls.c
@@ -222,6 +222,17 @@ int _syscall_fs_wait(handle_t back, char __user *buf, int __user *len) {
}
}
+int _syscall_fs_respond(int ret, char __user *buf, int len) {
+ struct vfs_request *req = process_current->handled_req;
+ if (!req) return -1;
+
+ // TODO copy buffer
+
+ process_current->handled_req = NULL;
+ vfs_request_finish(req, ret);
+ return 0;
+}
+
int _syscall_memflag(void __user *addr, size_t len, int flags) {
userptr_t goal = addr + len;
struct pagedir *pages = process_current->pages;
@@ -263,6 +274,8 @@ int syscall_handler(int num, int a, int b, int c) {
return _syscall_fs_create((userptr_t)a);
case _SYSCALL_FS_WAIT:
return _syscall_fs_wait(a, (userptr_t)b, (userptr_t)c);
+ case _SYSCALL_FS_RESPOND:
+ return _syscall_fs_respond(a, (userptr_t)b, c);
case _SYSCALL_MEMFLAG:
return _syscall_memflag((userptr_t)a, b, c);
default:
diff --git a/src/kernel/vfs/request.c b/src/kernel/vfs/request.c
index 2ff5d72..a8f2fc0 100644
--- a/src/kernel/vfs/request.c
+++ b/src/kernel/vfs/request.c
@@ -6,23 +6,27 @@
#include <kernel/vfs/root.h>
// dispatches a VFS operation to the correct process
-_Noreturn void vfs_request_create(struct vfs_request req) {
+_Noreturn void vfs_request_create(struct vfs_request req_) {
+ struct vfs_request *req;
int ret;
- switch (req.backend->type) {
+ process_current->state = PS_WAITS4FS;
+
+ // the request is owned by the caller
+ process_current->pending_req = req_;
+ req = &process_current->pending_req;
+
+ switch (req->backend->type) {
case VFS_BACK_ROOT:
- ret = vfs_root_handler(&req);
- vfs_request_finish(&req, ret);
+ ret = vfs_root_handler(req);
+ vfs_request_finish(req, ret);
case VFS_BACK_USER:
- process_current->state = PS_WAITS4FS;
- if (req.backend->handler == NULL) {
+ if (req->backend->handler == NULL) {
// backend isn't ready yet, join the queue
- assert(req.backend->queue == NULL); // TODO implement a proper queue
-
- req.backend->queue = process_current;
- process_current->pending_req = req;
+ assert(req->backend->queue == NULL); // TODO implement a proper queue
+ req->backend->queue = process_current;
process_switch_any();
} else {
- vfs_request_pass2handler(&req);
+ vfs_request_pass2handler(req);
}
default:
panic();
@@ -34,7 +38,9 @@ _Noreturn void vfs_request_pass2handler(struct vfs_request *req) {
int len;
assert(handler);
assert(handler->state == PS_WAITS4REQUEST);
+ assert(!handler->handled_req);
handler->state = PS_RUNNING;
+ handler->handled_req = req;
if (!virt_cpy_from(handler->pages,
&len, handler->awaited_req.len, sizeof len))