2005-05-31 12:57:19 +02:00
|
|
|
/* This file takes care of those system calls that deal with time.
|
|
|
|
*
|
|
|
|
* The entry points into this file are
|
2013-03-29 22:48:22 +01:00
|
|
|
* do_getres: perform the CLOCK_GETRES system call
|
|
|
|
* do_gettime: perform the CLOCK_GETTIME system call
|
2013-03-30 17:59:21 +01:00
|
|
|
* do_settime: perform the CLOCK_SETTIME system call
|
2013-11-04 22:48:08 +01:00
|
|
|
* do_time: perform the GETTIMEOFDAY system call
|
2005-05-31 12:57:19 +02:00
|
|
|
* do_stime: perform the STIME system call
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "pm.h"
|
|
|
|
#include <minix/callnr.h>
|
|
|
|
#include <minix/com.h>
|
|
|
|
#include <signal.h>
|
2013-03-29 22:48:22 +01:00
|
|
|
#include <sys/time.h>
|
2005-05-31 12:57:19 +02:00
|
|
|
#include "mproc.h"
|
|
|
|
|
2013-03-29 22:48:22 +01:00
|
|
|
/*===========================================================================*
|
|
|
|
* do_gettime *
|
|
|
|
*===========================================================================*/
|
|
|
|
int do_gettime()
|
|
|
|
{
|
|
|
|
clock_t ticks, realtime, clock;
|
|
|
|
time_t boottime;
|
|
|
|
int s;
|
|
|
|
|
|
|
|
if ( (s=getuptime(&ticks, &realtime, &boottime)) != OK)
|
|
|
|
panic("do_time couldn't get uptime: %d", s);
|
|
|
|
|
2014-05-12 23:40:11 +02:00
|
|
|
switch (m_in.m_lc_pm_time.clk_id) {
|
2013-03-29 22:48:22 +01:00
|
|
|
case CLOCK_REALTIME:
|
|
|
|
clock = realtime;
|
|
|
|
break;
|
|
|
|
case CLOCK_MONOTONIC:
|
|
|
|
clock = ticks;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return EINVAL; /* invalid/unsupported clock_id */
|
|
|
|
}
|
|
|
|
|
2014-05-12 23:40:11 +02:00
|
|
|
mp->mp_reply.m_pm_lc_time.sec = boottime + (clock / system_hz);
|
|
|
|
mp->mp_reply.m_pm_lc_time.nsec =
|
2013-09-20 19:47:18 +02:00
|
|
|
(uint32_t) ((clock % system_hz) * 1000000000ULL / system_hz);
|
2013-03-29 22:48:22 +01:00
|
|
|
|
|
|
|
return(OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* do_getres *
|
|
|
|
*===========================================================================*/
|
|
|
|
int do_getres()
|
|
|
|
{
|
2014-05-12 23:40:11 +02:00
|
|
|
switch (m_in.m_lc_pm_time.clk_id) {
|
2013-03-29 22:48:22 +01:00
|
|
|
case CLOCK_REALTIME:
|
|
|
|
case CLOCK_MONOTONIC:
|
|
|
|
/* tv_sec is always 0 since system_hz is an int */
|
2014-05-12 23:40:11 +02:00
|
|
|
mp->mp_reply.m_pm_lc_time.sec = 0;
|
|
|
|
mp->mp_reply.m_pm_lc_time.nsec = 1000000000 / system_hz;
|
2013-03-29 22:48:22 +01:00
|
|
|
return(OK);
|
|
|
|
default:
|
|
|
|
return EINVAL; /* invalid/unsupported clock_id */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-30 17:59:21 +01:00
|
|
|
/*===========================================================================*
|
|
|
|
* do_settime *
|
|
|
|
*===========================================================================*/
|
|
|
|
int do_settime()
|
|
|
|
{
|
|
|
|
int s;
|
|
|
|
|
|
|
|
if (mp->mp_effuid != SUPER_USER) {
|
|
|
|
return(EPERM);
|
|
|
|
}
|
|
|
|
|
2014-05-12 23:40:11 +02:00
|
|
|
switch (m_in.m_lc_pm_time.clk_id) {
|
2013-03-30 17:59:21 +01:00
|
|
|
case CLOCK_REALTIME:
|
2014-05-12 23:40:11 +02:00
|
|
|
s = sys_settime(m_in.m_lc_pm_time.now, m_in.m_lc_pm_time.clk_id,
|
|
|
|
m_in.m_lc_pm_time.sec, m_in.m_lc_pm_time.nsec);
|
2013-03-30 17:59:21 +01:00
|
|
|
return(s);
|
|
|
|
case CLOCK_MONOTONIC: /* monotonic cannot be changed */
|
|
|
|
default:
|
|
|
|
return EINVAL; /* invalid/unsupported clock_id */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-05-31 12:57:19 +02:00
|
|
|
/*===========================================================================*
|
|
|
|
* do_time *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int do_time()
|
2005-05-31 12:57:19 +02:00
|
|
|
{
|
|
|
|
/* Perform the time(tp) system call. This returns the time in seconds since
|
|
|
|
* 1.1.1970. MINIX is an astrophysically naive system that assumes the earth
|
|
|
|
* rotates at a constant rate and that such things as leap seconds do not
|
|
|
|
* exist.
|
|
|
|
*/
|
2013-03-29 22:48:22 +01:00
|
|
|
clock_t ticks, realtime;
|
|
|
|
time_t boottime;
|
2005-05-31 12:57:19 +02:00
|
|
|
int s;
|
|
|
|
|
2013-03-29 22:48:22 +01:00
|
|
|
if ( (s=getuptime(&ticks, &realtime, &boottime)) != OK)
|
2010-03-05 16:05:11 +01:00
|
|
|
panic("do_time couldn't get uptime: %d", s);
|
2005-05-31 12:57:19 +02:00
|
|
|
|
2014-05-12 23:40:11 +02:00
|
|
|
mp->mp_reply.m_pm_lc_time.sec = boottime + (realtime / system_hz);
|
|
|
|
mp->mp_reply.m_pm_lc_time.nsec =
|
|
|
|
(uint32_t) ((realtime % system_hz) * 1000000000ULL / system_hz);
|
2005-05-31 12:57:19 +02:00
|
|
|
return(OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*===========================================================================*
|
|
|
|
* do_stime *
|
|
|
|
*===========================================================================*/
|
2012-03-25 20:25:53 +02:00
|
|
|
int do_stime()
|
2005-05-31 12:57:19 +02:00
|
|
|
{
|
|
|
|
/* Perform the stime(tp) system call. Retrieve the system's uptime (ticks
|
2009-11-28 14:22:01 +01:00
|
|
|
* since boot) and pass the new time in seconds at system boot to the kernel.
|
2005-05-31 12:57:19 +02:00
|
|
|
*/
|
2013-03-29 22:48:22 +01:00
|
|
|
clock_t uptime, realtime;
|
|
|
|
time_t boottime;
|
2005-05-31 12:57:19 +02:00
|
|
|
int s;
|
|
|
|
|
|
|
|
if (mp->mp_effuid != SUPER_USER) {
|
|
|
|
return(EPERM);
|
|
|
|
}
|
2013-03-29 22:48:22 +01:00
|
|
|
if ( (s=getuptime(&uptime, &realtime, &boottime)) != OK)
|
2010-03-05 16:05:11 +01:00
|
|
|
panic("do_stime couldn't get uptime: %d", s);
|
2014-05-12 23:40:11 +02:00
|
|
|
boottime = m_in.m_lc_pm_time.sec - (realtime/system_hz);
|
2005-05-31 12:57:19 +02:00
|
|
|
|
2007-08-07 14:28:42 +02:00
|
|
|
s= sys_stime(boottime); /* Tell kernel about boottime */
|
|
|
|
if (s != OK)
|
2010-03-05 16:05:11 +01:00
|
|
|
panic("pm: sys_stime failed: %d", s);
|
2005-05-31 12:57:19 +02:00
|
|
|
|
2007-08-07 14:28:42 +02:00
|
|
|
return(OK);
|
2005-05-31 12:57:19 +02:00
|
|
|
}
|