Commit fe2cc53e authored by Jeff Dike's avatar Jeff Dike Committed by Linus Torvalds

uml: track and make up lost ticks

Alarm delivery could be noticably late in the !CONFIG_NOHZ case because lost
ticks weren't being taken into account.  This is now treated more carefully,
with the time between ticks being calculated and the appropriate number of
ticks delivered to the timekeeping system.

Cc: Nix <nix@esperi.org.uk>
Signed-off-by: default avatarJeff Dike <jdike@linux.intel.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 3d88958e
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
/* Copied from linux/compiler-gcc.h since we can't include it directly */ /* Copied from linux/compiler-gcc.h since we can't include it directly */
#define barrier() __asm__ __volatile__("": : :"memory") #define barrier() __asm__ __volatile__("": : :"memory")
extern void sig_handler(int sig, struct sigcontext sc); extern void sig_handler(int sig, struct sigcontext *sc);
extern void alarm_handler(int sig, struct sigcontext sc); extern void alarm_handler(int sig, struct sigcontext *sc);
#endif #endif
...@@ -12,6 +12,7 @@ ...@@ -12,6 +12,7 @@
#include "as-layout.h" #include "as-layout.h"
#include "kern_util.h" #include "kern_util.h"
#include "os.h" #include "os.h"
#include "process.h"
#include "sysdep/barrier.h" #include "sysdep/barrier.h"
#include "sysdep/sigcontext.h" #include "sysdep/sigcontext.h"
#include "user.h" #include "user.h"
......
...@@ -9,7 +9,9 @@ ...@@ -9,7 +9,9 @@
#include <time.h> #include <time.h>
#include <sys/time.h> #include <sys/time.h>
#include "kern_constants.h" #include "kern_constants.h"
#include "kern_util.h"
#include "os.h" #include "os.h"
#include "process.h"
#include "user.h" #include "user.h"
int set_interval(void) int set_interval(void)
...@@ -58,12 +60,17 @@ static inline long long timeval_to_ns(const struct timeval *tv) ...@@ -58,12 +60,17 @@ static inline long long timeval_to_ns(const struct timeval *tv)
long long disable_timer(void) long long disable_timer(void)
{ {
struct itimerval time = ((struct itimerval) { { 0, 0 }, { 0, 0 } }); struct itimerval time = ((struct itimerval) { { 0, 0 }, { 0, 0 } });
int remain, max = UM_NSEC_PER_SEC / UM_HZ;
if (setitimer(ITIMER_VIRTUAL, &time, &time) < 0) if (setitimer(ITIMER_VIRTUAL, &time, &time) < 0)
printk(UM_KERN_ERR "disable_timer - setitimer failed, " printk(UM_KERN_ERR "disable_timer - setitimer failed, "
"errno = %d\n", errno); "errno = %d\n", errno);
return timeval_to_ns(&time.it_value); remain = timeval_to_ns(&time.it_value);
if (remain > max)
remain = max;
return remain;
} }
long long os_nsecs(void) long long os_nsecs(void)
...@@ -79,7 +86,44 @@ static int after_sleep_interval(struct timespec *ts) ...@@ -79,7 +86,44 @@ static int after_sleep_interval(struct timespec *ts)
{ {
return 0; return 0;
} }
static void deliver_alarm(void)
{
alarm_handler(SIGVTALRM, NULL);
}
static unsigned long long sleep_time(unsigned long long nsecs)
{
return nsecs;
}
#else #else
unsigned long long last_tick;
unsigned long long skew;
static void deliver_alarm(void)
{
unsigned long long this_tick = os_nsecs();
int one_tick = UM_NSEC_PER_SEC / UM_HZ;
if (last_tick == 0)
last_tick = this_tick - one_tick;
skew += this_tick - last_tick;
while (skew >= one_tick) {
alarm_handler(SIGVTALRM, NULL);
skew -= one_tick;
}
last_tick = this_tick;
}
static unsigned long long sleep_time(unsigned long long nsecs)
{
return nsecs > skew ? nsecs - skew : 0;
}
static inline long long timespec_to_us(const struct timespec *ts) static inline long long timespec_to_us(const struct timespec *ts)
{ {
return ((long long) ts->tv_sec * UM_USEC_PER_SEC) + return ((long long) ts->tv_sec * UM_USEC_PER_SEC) +
...@@ -102,6 +146,8 @@ static int after_sleep_interval(struct timespec *ts) ...@@ -102,6 +146,8 @@ static int after_sleep_interval(struct timespec *ts)
*/ */
if (start_usecs > usec) if (start_usecs > usec)
start_usecs = usec; start_usecs = usec;
start_usecs -= skew / UM_NSEC_PER_USEC;
tv = ((struct timeval) { .tv_sec = start_usecs / UM_USEC_PER_SEC, tv = ((struct timeval) { .tv_sec = start_usecs / UM_USEC_PER_SEC,
.tv_usec = start_usecs % UM_USEC_PER_SEC }); .tv_usec = start_usecs % UM_USEC_PER_SEC });
interval = ((struct itimerval) { { 0, usec }, tv }); interval = ((struct itimerval) { { 0, usec }, tv });
...@@ -113,8 +159,6 @@ static int after_sleep_interval(struct timespec *ts) ...@@ -113,8 +159,6 @@ static int after_sleep_interval(struct timespec *ts)
} }
#endif #endif
extern void alarm_handler(int sig, struct sigcontext *sc);
void idle_sleep(unsigned long long nsecs) void idle_sleep(unsigned long long nsecs)
{ {
struct timespec ts; struct timespec ts;
...@@ -126,10 +170,12 @@ void idle_sleep(unsigned long long nsecs) ...@@ -126,10 +170,12 @@ void idle_sleep(unsigned long long nsecs)
*/ */
if (nsecs == 0) if (nsecs == 0)
nsecs = UM_NSEC_PER_SEC / UM_HZ; nsecs = UM_NSEC_PER_SEC / UM_HZ;
nsecs = sleep_time(nsecs);
ts = ((struct timespec) { .tv_sec = nsecs / UM_NSEC_PER_SEC, ts = ((struct timespec) { .tv_sec = nsecs / UM_NSEC_PER_SEC,
.tv_nsec = nsecs % UM_NSEC_PER_SEC }); .tv_nsec = nsecs % UM_NSEC_PER_SEC });
if (nanosleep(&ts, &ts) == 0) if (nanosleep(&ts, &ts) == 0)
alarm_handler(SIGVTALRM, NULL); deliver_alarm();
after_sleep_interval(&ts); after_sleep_interval(&ts);
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment