diff options
Diffstat (limited to 'src/kernel/arch/amd64/driver')
-rw-r--r-- | src/kernel/arch/amd64/driver/driver.h | 1 | ||||
-rw-r--r-- | src/kernel/arch/amd64/driver/time.c | 80 |
2 files changed, 81 insertions, 0 deletions
diff --git a/src/kernel/arch/amd64/driver/driver.h b/src/kernel/arch/amd64/driver/driver.h index 0c4ecd5..6f862cf 100644 --- a/src/kernel/arch/amd64/driver/driver.h +++ b/src/kernel/arch/amd64/driver/driver.h @@ -14,5 +14,6 @@ struct GfxInfo { void pata_init(void); void ps2_init(void); void rtl8139_init(uint32_t bdf); +void time_init(void); void vfs_root_init(void); void video_init(GfxInfo); 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); +} |