aboutsummaryrefslogtreecommitdiff
path: root/kernel/features/timer.c
blob: 674dd1a470583f6c515c7f7536adad31aba0aaec (plain) (blame)
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
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
// MIT License, Copyright (c) 2020 Marvin Borner

#include <assert.h>
#include <def.h>
#include <dev.h>
#include <drivers/cpu.h>
#include <drivers/int.h>
#include <drivers/pit.h>
#include <drivers/rtc.h>
#include <mem.h>
#include <proc.h>
#include <timer.h>

static u32 timer_ticks = 0;

u32 timer_get(void)
{
	return timer_ticks;
}

static void timer_handler(void)
{
	int_disable();
	if (timer_ticks >= U32_MAX)
		timer_ticks = 0;
	else
		timer_ticks++;
	proc_timer_check(timer_ticks);
	int_enable();
}

CLEAR void timer_install_handler(void)
{
	static u8 check = 0;
	assert(check++ == 0);
	int_event_handler_add(0, timer_handler);
}

// "Delay" function with CPU sleep
void timer_wait(u32 ticks)
{
	u32 eticks = timer_ticks + ticks;
	while (timer_ticks < eticks)
		__asm__ volatile("sti\nhlt\ncli");
}

static struct timer timer_struct(void)
{
	struct proc *proc = proc_current();
	struct timer timer = {
		.rtc = rtc_stamp(),
		.ticks.user = proc->ticks.user,
		.ticks.kernel = proc->ticks.kernel,
		.time = timer_get(),
	};
	return timer;
}

static res timer_read(void *buf, u32 offset, u32 count)
{
	UNUSED(offset);

	// TODO: Make sleeping more accurate
	struct proc *proc = proc_current();
	if (proc->timer.mode == TIMER_MODE_SLEEP) {
		dev_block(DEV_TIMER, proc);
		proc->timer.mode = TIMER_MODE_DEFAULT;
	}

	struct timer timer = timer_struct();
	memcpy_user(buf, &timer, MIN(count, sizeof(timer)));

	return MIN(count, sizeof(timer));
}

static res timer_control(u32 request, void *arg1, void *arg2, void *arg3)
{
	UNUSED(arg2);
	UNUSED(arg3);

	switch (request) {
	case DEVCTL_TIMER_SLEEP: {
		struct proc *proc = proc_current();
		proc->timer.mode = TIMER_MODE_SLEEP;
		proc->timer.data = timer_ticks + (u32)arg1;
		return EOK;
	}
	default:
		return -EINVAL;
	}
}

static res timer_ready(void)
{
	struct proc *proc = proc_current();
	if (proc->timer.mode == TIMER_MODE_SLEEP) {
		if (timer_ticks >= proc->timer.data)
			return EOK;
		return -EAGAIN;
	}
	return EOK;
}

// Install timer handler into IRQ0
CLEAR void timer_install(void)
{
	// hpet_install(10000); // TODO: Reimplement hpet
	pit_install();

	struct dev_dev *dev = zalloc(sizeof(*dev));
	dev->read = timer_read;
	dev->control = timer_control;
	dev->ready = timer_ready;
	dev_add(DEV_TIMER, dev);
}