| /* |
| * Copyright (C) 2000 - 2007 Jeff Dike (jdike{addtoit,linux.intel}.com) |
| * Licensed under the GPL |
| */ |
| |
| #include <stddef.h> |
| #include <errno.h> |
| #include <signal.h> |
| #include <time.h> |
| #include <sys/time.h> |
| #include "kern_util.h" |
| #include "kern_constants.h" |
| #include "os.h" |
| #include "user.h" |
| |
| int set_interval(int is_virtual) |
| { |
| int usec = 1000000/hz(); |
| int timer_type = is_virtual ? ITIMER_VIRTUAL : ITIMER_REAL; |
| struct itimerval interval = ((struct itimerval) { { 0, usec }, |
| { 0, usec } }); |
| |
| if (setitimer(timer_type, &interval, NULL) == -1) |
| return -errno; |
| |
| return 0; |
| } |
| |
| void disable_timer(void) |
| { |
| struct itimerval disable = ((struct itimerval) { { 0, 0 }, { 0, 0 }}); |
| |
| if ((setitimer(ITIMER_VIRTUAL, &disable, NULL) < 0) || |
| (setitimer(ITIMER_REAL, &disable, NULL) < 0)) |
| printk(UM_KERN_ERR "disable_timer - setitimer failed, " |
| "errno = %d\n", errno); |
| |
| /* If there are signals already queued, after unblocking ignore them */ |
| signal(SIGALRM, SIG_IGN); |
| signal(SIGVTALRM, SIG_IGN); |
| } |
| |
| void switch_timers(int to_real) |
| { |
| struct itimerval disable = ((struct itimerval) { { 0, 0 }, { 0, 0 }}); |
| struct itimerval enable = ((struct itimerval) { { 0, 1000000/hz() }, |
| { 0, 1000000/hz() }}); |
| int old, new; |
| |
| if (to_real) { |
| old = ITIMER_VIRTUAL; |
| new = ITIMER_REAL; |
| } |
| else { |
| old = ITIMER_REAL; |
| new = ITIMER_VIRTUAL; |
| } |
| |
| if ((setitimer(old, &disable, NULL) < 0) || |
| (setitimer(new, &enable, NULL))) |
| printk(UM_KERN_ERR "switch_timers - setitimer failed, " |
| "errno = %d\n", errno); |
| } |
| |
| unsigned long long os_nsecs(void) |
| { |
| struct timeval tv; |
| |
| gettimeofday(&tv, NULL); |
| return (unsigned long long) tv.tv_sec * BILLION + tv.tv_usec * 1000; |
| } |
| |
| void idle_sleep(int secs) |
| { |
| struct timespec ts; |
| |
| ts.tv_sec = secs; |
| ts.tv_nsec = 0; |
| nanosleep(&ts, NULL); |
| } |