2006-01-19 04:42:42 +03:00
/*
* Copyright ( C ) 2000 , 2001 , 2002 Jeff Dike ( jdike @ karaya . com )
* Licensed under the GPL
*/
# include <stdio.h>
2005-04-17 02:20:36 +04:00
# include <stdlib.h>
2006-01-19 04:42:42 +03:00
# include <unistd.h>
# include <time.h>
2005-04-17 02:20:36 +04:00
# include <sys/time.h>
2006-01-19 04:42:42 +03:00
# include <signal.h>
# include <errno.h>
# include "user_util.h"
# include "kern_util.h"
# include "user.h"
# include "process.h"
# include "kern_constants.h"
# include "os.h"
static void set_interval ( int timer_type )
{
int usec = 1000000 / hz ( ) ;
struct itimerval interval = ( ( struct itimerval ) { { 0 , usec } ,
{ 0 , usec } } ) ;
if ( setitimer ( timer_type , & interval , NULL ) = = - 1 )
panic ( " setitimer failed - errno = %d \n " , errno ) ;
}
void enable_timer ( void )
{
set_interval ( ITIMER_VIRTUAL ) ;
}
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 ( " disnable_timer - setitimer failed, errno = %d \n " ,
errno ) ;
/* If there are signals already queued, after unblocking ignore them */
set_handler ( SIGALRM , SIG_IGN , 0 , - 1 ) ;
set_handler ( SIGVTALRM , SIG_IGN , 0 , - 1 ) ;
}
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 ( " switch_timers - setitimer failed, errno = %d \n " ,
errno ) ;
}
2005-04-17 02:20:36 +04:00
2006-07-10 15:45:09 +04:00
# ifdef UML_CONFIG_MODE_TT
2006-01-19 04:42:42 +03:00
void uml_idle_timer ( void )
{
if ( signal ( SIGVTALRM , SIG_IGN ) = = SIG_ERR )
panic ( " Couldn't unset SIGVTALRM handler " ) ;
set_handler ( SIGALRM , ( __sighandler_t ) alarm_handler ,
SA_RESTART , SIGUSR1 , SIGIO , SIGWINCH , SIGVTALRM , - 1 ) ;
set_interval ( ITIMER_REAL ) ;
}
2006-07-10 15:45:09 +04:00
# endif
2006-01-19 04:42:42 +03:00
unsigned long long os_nsecs ( void )
2005-04-17 02:20:36 +04:00
{
struct timeval tv ;
gettimeofday ( & tv , NULL ) ;
2006-01-19 04:42:42 +03:00
return ( ( unsigned long long ) tv . tv_sec * BILLION + tv . tv_usec * 1000 ) ;
2005-04-17 02:20:36 +04:00
}
2006-01-19 04:42:42 +03:00
void idle_sleep ( int secs )
{
struct timespec ts ;
ts . tv_sec = secs ;
ts . tv_nsec = 0 ;
nanosleep ( & ts , NULL ) ;
}
void user_time_init ( void )
{
set_interval ( ITIMER_VIRTUAL ) ;
}