diff options
Diffstat (limited to 'src/kernel/vfs/request.c')
-rw-r--r-- | src/kernel/vfs/request.c | 62 |
1 files changed, 41 insertions, 21 deletions
diff --git a/src/kernel/vfs/request.c b/src/kernel/vfs/request.c index e3f0316..8fe0114 100644 --- a/src/kernel/vfs/request.c +++ b/src/kernel/vfs/request.c @@ -80,11 +80,13 @@ void vfsreq_finish(VfsReq *req, char __user *stored, long ret, assert((size_t)ret <= req->output.len); } - if (req->input.kern) + if (req->input.kern) { kfree(req->input.buf_kern); + } - if (req->backend) - vfs_backend_refdown(req->backend, true); + if (req->backend) { + vfsback_userdown(req->backend); + } if (req->caller) { assert(req->caller->state == PS_WAITS4FS); @@ -159,26 +161,28 @@ static void vfs_backend_user_accept(VfsReq *req) { return; } -void vfs_backend_refdown(VfsBackend *b, bool use) { - size_t *field = use ? &b->usehcnt : &b->provhcnt; - assert(b); - assert(0 < *field); - *field -= 1; - - if (b->provhcnt == 0 && use == false) { - VfsReq *q = b->queue; - while (q) { - VfsReq *q2 = q->queue_next; - vfsreq_finish_short(q, -1); - q = q2; - } - b->queue = NULL; +static void +vfsback_checkfree(VfsBackend *b) +{ + if (b->usehcnt == 0 && b->provhcnt == 0) { + assert(!b->queue); + kfree(b); } - if (b->usehcnt == 0 && use == true) { +} + +void +vfsback_userdown(VfsBackend *b) +{ + assert(b); + assert(0 < b->usehcnt); + b->usehcnt--; + if (b->usehcnt == 0) { if (!b->is_user && b->kern.cleanup) { b->kern.cleanup(b); } if (b->is_user && b->user.handler) { + /* tell the process that the filesystem won't receive any more + * requests */ Proc *p = b->user.handler; b->user.handler = NULL; assert(p->state == PS_WAITS4REQUEST); @@ -186,8 +190,24 @@ void vfs_backend_refdown(VfsBackend *b, bool use) { proc_setstate(p, PS_RUNNING); } } - if (b->usehcnt == 0 && b->provhcnt == 0) { - assert(!b->queue); - kfree(b); + vfsback_checkfree(b); +} + +void +vfsback_provdown(VfsBackend *b) +{ + assert(b); + assert(0 < b->provhcnt); + b->provhcnt--; + if (b->provhcnt == 0) { + /* discard everything in the queue */ + VfsReq *q = b->queue; + while (q) { + VfsReq *q2 = q->queue_next; + vfsreq_finish_short(q, -1); + q = q2; + } + b->queue = NULL; } + vfsback_checkfree(b); } |