aboutsummaryrefslogtreecommitdiffstats
path: root/arch/um/os-Linux/time.c
diff options
context:
space:
mode:
authorJeff Dike <jdike@addtoit.com>2007-10-16 04:27:22 -0400
committerLinus Torvalds <torvalds@woody.linux-foundation.org>2007-10-16 12:43:08 -0400
commit181bde801afae4304843ce95c868d4419ae87310 (patch)
treef33e6c88e36cda7699fc2dfe2b6ac9ccb3c6963f /arch/um/os-Linux/time.c
parent532d0fa4d104ca3e37dd38bc5073376fcc3c2712 (diff)
uml: fix timer switching
Fix up the switching between virtual and real timers. The idle loop sleeps, so the timer at that point must be real time. At all other times, the timer must be virtual. Even when userspace is running, and the kernel is asleep, the virtual timer is correct because the process timer will be running and the process timer will be firing. The timer switch used to be in the context switch and timer handler code. This is moved to the idle loop and the signal handler, making it much more clear why it is happening. switch_timers now returns the old timer type so that it may be restored. The signal handler uses this in order to restore the previous timer type when it returns. Signed-off-by: Jeff Dike <jdike@linux.intel.com> Cc: Thomas Gleixner <tglx@linutronix.de> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'arch/um/os-Linux/time.c')
-rw-r--r--arch/um/os-Linux/time.c27
1 files changed, 20 insertions, 7 deletions
diff --git a/arch/um/os-Linux/time.c b/arch/um/os-Linux/time.c
index 5939653f99ea..a16a0f484edc 100644
--- a/arch/um/os-Linux/time.c
+++ b/arch/um/os-Linux/time.c
@@ -12,6 +12,8 @@
12#include "os.h" 12#include "os.h"
13#include "user.h" 13#include "user.h"
14 14
15static int is_real_timer = 0;
16
15int set_interval(int is_virtual) 17int set_interval(int is_virtual)
16{ 18{
17 int usec = 1000000/UM_HZ; 19 int usec = 1000000/UM_HZ;
@@ -39,12 +41,14 @@ void disable_timer(void)
39 signal(SIGVTALRM, SIG_IGN); 41 signal(SIGVTALRM, SIG_IGN);
40} 42}
41 43
42void switch_timers(int to_real) 44int switch_timers(int to_real)
43{ 45{
44 struct itimerval disable = ((struct itimerval) { { 0, 0 }, { 0, 0 }}); 46 struct itimerval disable = ((struct itimerval) { { 0, 0 }, { 0, 0 }});
45 struct itimerval enable = ((struct itimerval) { { 0, 1000000/UM_HZ }, 47 struct itimerval enable;
46 { 0, 1000000/UM_HZ }}); 48 int old, new, old_type = is_real_timer;
47 int old, new; 49
50 if(to_real == old_type)
51 return to_real;
48 52
49 if (to_real) { 53 if (to_real) {
50 old = ITIMER_VIRTUAL; 54 old = ITIMER_VIRTUAL;
@@ -55,10 +59,19 @@ void switch_timers(int to_real)
55 new = ITIMER_VIRTUAL; 59 new = ITIMER_VIRTUAL;
56 } 60 }
57 61
58 if ((setitimer(old, &disable, NULL) < 0) || 62 if (setitimer(old, &disable, &enable) < 0)
59 (setitimer(new, &enable, NULL))) 63 printk(UM_KERN_ERR "switch_timers - setitimer disable failed, "
60 printk(UM_KERN_ERR "switch_timers - setitimer failed, " 64 "errno = %d\n", errno);
65
66 if((enable.it_value.tv_sec == 0) && (enable.it_value.tv_usec == 0))
67 enable.it_value = enable.it_interval;
68
69 if (setitimer(new, &enable, NULL))
70 printk(UM_KERN_ERR "switch_timers - setitimer enable failed, "
61 "errno = %d\n", errno); 71 "errno = %d\n", errno);
72
73 is_real_timer = to_real;
74 return old_type;
62} 75}
63 76
64unsigned long long os_nsecs(void) 77unsigned long long os_nsecs(void)