summaryrefslogtreecommitdiff
path: root/src/kernel/arch/amd64/driver/time.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/kernel/arch/amd64/driver/time.c')
-rw-r--r--src/kernel/arch/amd64/driver/time.c80
1 files changed, 80 insertions, 0 deletions
diff --git a/src/kernel/arch/amd64/driver/time.c b/src/kernel/arch/amd64/driver/time.c
new file mode 100644
index 0000000..8ae6fb2
--- /dev/null
+++ b/src/kernel/arch/amd64/driver/time.c
@@ -0,0 +1,80 @@
+/* This is already sort of deprecated as I introduce it --
+ * it doesn't seem to provide much benefit over _sys_time, which I think I
+ * have to keep anyways, so processes in an empty namespace can still keep
+ * time. */
+#include <camellia/errno.h>
+#include <camellia/fsutil.h>
+#include <kernel/arch/amd64/driver/driver.h>
+#include <kernel/arch/amd64/driver/util.h>
+#include <kernel/malloc.h>
+#include <kernel/panic.h>
+#include <kernel/proc.h>
+#include <kernel/util.h>
+#include <kernel/vfs/mount.h>
+#include <kernel/vfs/request.h>
+#include <shared/mem.h>
+
+typedef struct {
+ uint64_t base;
+} TimeObj;
+
+static long
+handle(VfsReq *req)
+{
+ TimeObj *h;
+ if (req->type == VFSOP_OPEN) {
+ if (reqpathcmp(req, "")) {
+ h = kmalloc(sizeof *h, "dev/time");
+ h->base = uptime_ns();
+ return (uintptr_t)h;
+ } else {
+ return -ENOENT;
+ }
+ }
+ h = (__force void*)req->id;
+
+ if (req->type == VFSOP_CLOSE) {
+ assert(h);
+ kfree(h);
+ return 0;
+ }
+
+ uint64_t now = uptime_ns();
+
+ union {
+ char buf[8];
+ uint64_t t;
+ } u;
+ switch (req->type) {
+ case VFSOP_READ:
+ u.t = now - h->base;
+ return req_readcopy(req, u.buf, sizeof u.buf);
+ case VFSOP_GETSIZE:
+ return 8;
+ case VFSOP_WRITE:
+ if (req->input.len == 8) {
+ assert(!req->input.kern);
+ if (pcpy_from(req->caller, u.buf, req->input.buf, sizeof u.buf) != sizeof(u.buf)) {
+ return -EGENERIC;
+ }
+ h->base = now - u.t;
+ assert(u.t == now - h->base);
+ return 8;
+ }
+ return -EGENERIC;
+ default:
+ return -ENOSYS;
+ }
+}
+
+static void
+accept(VfsReq *req)
+{
+ vfsreq_finish_short(req, handle(req));
+}
+
+void
+time_init(void)
+{
+ vfs_root_register("/dev/bintime", accept);
+}