summaryrefslogtreecommitdiff
path: root/src/kernel/handle.c
blob: f5255c08c454805dff135598e1e1b896bb763851 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
#include <kernel/handle.h>
#include <kernel/mem/alloc.h>
#include <kernel/panic.h>
#include <kernel/proc.h>
#include <kernel/vfs/request.h>

struct handle *handle_init(enum handle_type type) {
	struct handle *h = kmalloc(sizeof *h);
	h->type = type;
	h->refcount = 1;
	return h;
}

void handle_close(struct handle *h) {
	if (!h) return;
	assert(h->refcount > 0);
	if (--(h->refcount) > 0) return;

	switch (h->type) {
		case HANDLE_FILE:
			vfs_request_create((struct vfs_request) {
					.type = VFSOP_CLOSE,
					.id = h->file.id,
					.caller = NULL,
					.backend = h->file.backend,
				});
			vfs_backend_refdown(h->file.backend);
			break;
		case HANDLE_FS_FRONT:
			vfs_backend_refdown(h->fs.backend);
			break;
		case HANDLE_INVALID: panic_invalid_state();
	}

	// TODO sanity check to check if refcount is true. handle_sanity?

	// TODO tests which would catch premature frees
	// by that i mean duplicating a handle and killing the original process
	h->type = HANDLE_INVALID;
	kfree(h);
}