diff options
author | dzwdz | 2022-08-03 12:42:52 +0200 |
---|---|---|
committer | dzwdz | 2022-08-03 12:42:52 +0200 |
commit | d9da70c7c6230b9698dc4a1dbc4d7f05794d2740 (patch) | |
tree | afa28cfc7069f7649632f7b33daac47ee56d3819 /src/kernel/vfs | |
parent | 092f732b893c3ecc0f8cee6699ab21ea42ad10ff (diff) |
kernel: reuse a single allocation for all vfs_requests of a process
$ iostress 32 512 0 > /vtty # before
512 calls, 0 bytes. avg 121133
$ iostress 32 512 0 > /vtty # after
512 calls, 0 bytes. avg 103540
103540/121133 = ~85%
I think the tiny bit of added complexity is worth it here.
Diffstat (limited to 'src/kernel/vfs')
-rw-r--r-- | src/kernel/vfs/request.c | 44 | ||||
-rw-r--r-- | src/kernel/vfs/request.h | 2 |
2 files changed, 24 insertions, 22 deletions
diff --git a/src/kernel/vfs/request.c b/src/kernel/vfs/request.c index 0164dfd..d942fba 100644 --- a/src/kernel/vfs/request.c +++ b/src/kernel/vfs/request.c @@ -8,29 +8,30 @@ #include <shared/mem.h> void vfsreq_create(struct vfs_request req_) { - struct vfs_request *req = kmalloc(sizeof *req); // freed in vfsreq_finish + struct vfs_request *req; + if (req_.caller) { + process_transition(req_.caller, PS_WAITS4FS); + if (!req_.caller->reqslot) + req_.caller->reqslot = kmalloc(sizeof *req); + req = req_.caller->reqslot; + /* (re)using a single allocation for all request a process makes */ + } else { + req = kmalloc(sizeof *req); + } memcpy(req, &req_, sizeof *req); if (req->backend) req->backend->refcount++; - if (req->caller) { - // TODO after running `run_tests` from vga, this is entered with state == PS_WAITS4FS - process_transition(req->caller, PS_WAITS4FS); - req->caller->waits4fs.req = req; - } - - if (!req->backend || !req->backend->potential_handlers) { + if (req->backend && req->backend->potential_handlers) { + struct vfs_request **iter = &req->backend->queue; + while (*iter != NULL) // find free spot in queue + iter = &(*iter)->queue_next; + *iter = req; + vfs_backend_tryaccept(req->backend); + } else { vfsreq_finish_short(req, -1); - return; } - - struct vfs_request **iter = &req->backend->queue; - while (*iter != NULL) // find free spot in queue - iter = &(*iter)->queue_next; - *iter = req; - - vfs_backend_tryaccept(req->backend); } void vfsreq_finish(struct vfs_request *req, char __user *stored, long ret, @@ -70,17 +71,16 @@ void vfsreq_finish(struct vfs_request *req, char __user *stored, long ret, if (req->input.kern) kfree(req->input.buf_kern); + if (req->backend) + vfs_backend_refdown(req->backend); + if (req->caller) { assert(req->caller->state == PS_WAITS4FS); regs_savereturn(&req->caller->regs, ret); process_transition(req->caller, PS_RUNNING); + } else { + kfree(req); } - - if (req->backend) - vfs_backend_refdown(req->backend); - - kfree(req); - return; } void vfs_backend_tryaccept(struct vfs_backend *backend) { diff --git a/src/kernel/vfs/request.h b/src/kernel/vfs/request.h index cb560c9..ee33623 100644 --- a/src/kernel/vfs/request.h +++ b/src/kernel/vfs/request.h @@ -51,6 +51,8 @@ struct vfs_request { long offset; int flags; + /* if caller != NULL, owned by it - don't free, the allocation will be reused + * if caller == NULL, free on finish */ struct process *caller; struct vfs_backend *backend; |