uml: tickless support
[linux-2.6.git] / arch / um / os-Linux / time.c
1 /*
2  * Copyright (C) 2000 - 2007 Jeff Dike (jdike{addtoit,linux.intel}.com)
3  * Licensed under the GPL
4  */
5
6 #include <stddef.h>
7 #include <errno.h>
8 #include <signal.h>
9 #include <time.h>
10 #include <sys/time.h>
11 #include "kern_constants.h"
12 #include "os.h"
13 #include "user.h"
14
15 static int is_real_timer = 0;
16
17 int set_interval(void)
18 {
19         int usec = 1000000/UM_HZ;
20         struct itimerval interval = ((struct itimerval) { { 0, usec },
21                                                           { 0, usec } });
22
23         if (setitimer(ITIMER_VIRTUAL, &interval, NULL) == -1)
24                 return -errno;
25
26         return 0;
27 }
28
29 int timer_one_shot(int ticks)
30 {
31         unsigned long usec = ticks * 1000000 / UM_HZ;
32         unsigned long sec = usec / 1000000;
33         struct itimerval interval;
34
35         usec %= 1000000;
36         interval = ((struct itimerval) { { 0, 0 }, { sec, usec } });
37
38         if (setitimer(ITIMER_VIRTUAL, &interval, NULL) == -1)
39                 return -errno;
40
41         return 0;
42 }
43
44 void disable_timer(void)
45 {
46         struct itimerval disable = ((struct itimerval) { { 0, 0 }, { 0, 0 }});
47
48         if ((setitimer(ITIMER_VIRTUAL, &disable, NULL) < 0) ||
49             (setitimer(ITIMER_REAL, &disable, NULL) < 0))
50                 printk(UM_KERN_ERR "disable_timer - setitimer failed, "
51                        "errno = %d\n", errno);
52 }
53
54 int switch_timers(int to_real)
55 {
56         struct itimerval disable = ((struct itimerval) { { 0, 0 }, { 0, 0 }});
57         struct itimerval enable;
58         int old, new, old_type = is_real_timer;
59
60         if(to_real == old_type)
61                 return to_real;
62
63         if (to_real) {
64                 old = ITIMER_VIRTUAL;
65                 new = ITIMER_REAL;
66         }
67         else {
68                 old = ITIMER_REAL;
69                 new = ITIMER_VIRTUAL;
70         }
71
72         if (setitimer(old, &disable, &enable) < 0)
73                 printk(UM_KERN_ERR "switch_timers - setitimer disable failed, "
74                        "errno = %d\n", errno);
75
76         if((enable.it_value.tv_sec == 0) && (enable.it_value.tv_usec == 0))
77                 enable.it_value = enable.it_interval;
78
79         if (setitimer(new, &enable, NULL))
80                 printk(UM_KERN_ERR "switch_timers - setitimer enable failed, "
81                        "errno = %d\n", errno);
82
83         is_real_timer = to_real;
84         return old_type;
85 }
86
87 unsigned long long os_nsecs(void)
88 {
89         struct timeval tv;
90
91         gettimeofday(&tv, NULL);
92         return timeval_to_ns(&tv);
93 }
94
95 void idle_sleep(int secs)
96 {
97         struct timespec ts;
98
99         ts.tv_sec = secs;
100         ts.tv_nsec = 0;
101         nanosleep(&ts, NULL);
102 }