#include <security/mac_framework.h>
#endif
#include <IOKit/IOBSD.h>
+#include <os/log.h>
typedef int64_t l_fp;
-#define L_ADD(v, u) ((v) += (u))
-#define L_SUB(v, u) ((v) -= (u))
-#define L_ADDHI(v, a) ((v) += (int64_t)(a) << 32)
-#define L_NEG(v) ((v) = -(v))
+#define L_ADD(v, u) ((v) += (u))
+#define L_SUB(v, u) ((v) -= (u))
+#define L_ADDHI(v, a) ((v) += (int64_t)(a) << 32)
+#define L_NEG(v) ((v) = -(v))
#define L_RSHIFT(v, n) \
do { \
- if ((v) < 0) \
- (v) = -(-(v) >> (n)); \
- else \
- (v) = (v) >> (n); \
+ if ((v) < 0) \
+ (v) = -(-(v) >> (n)); \
+ else \
+ (v) = (v) >> (n); \
} while (0)
-#define L_MPY(v, a) ((v) *= (a))
-#define L_CLR(v) ((v) = 0)
-#define L_ISNEG(v) ((v) < 0)
+#define L_MPY(v, a) ((v) *= (a))
+#define L_CLR(v) ((v) = 0)
+#define L_ISNEG(v) ((v) < 0)
#define L_LINT(v, a) \
do { \
- if ((a) > 0) \
- ((v) = (int64_t)(a) << 32); \
- else \
- ((v) = -((int64_t)(-(a)) << 32)); \
+ if ((a) > 0) \
+ ((v) = (int64_t)(a) << 32); \
+ else \
+ ((v) = -((int64_t)(-(a)) << 32)); \
} while (0)
-#define L_GINT(v) ((v) < 0 ? -(-(v) >> 32) : (v) >> 32)
+#define L_GINT(v) ((v) < 0 ? -(-(v) >> 32) : (v) >> 32)
/*
* Generic NTP kernel interface
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
*/
-#define SHIFT_PLL 4
-#define SHIFT_FLL 2
+#define SHIFT_PLL 4
+#define SHIFT_FLL 2
static int time_state = TIME_OK;
int time_status = STA_UNSYNC;
static lck_spin_t * ntp_lock;
static lck_grp_t * ntp_lock_grp;
static lck_attr_t * ntp_lock_attr;
-static lck_grp_attr_t *ntp_lock_grp_attr;
+static lck_grp_attr_t *ntp_lock_grp_attr;
-#define NTP_LOCK(enable) \
- enable = ml_set_interrupts_enabled(FALSE); \
- lck_spin_lock(ntp_lock);
+#define NTP_LOCK(enable) \
+ enable = ml_set_interrupts_enabled(FALSE); \
+ lck_spin_lock(ntp_lock);
-#define NTP_UNLOCK(enable) \
- lck_spin_unlock(ntp_lock);\
- ml_set_interrupts_enabled(enable);
+#define NTP_UNLOCK(enable) \
+ lck_spin_unlock(ntp_lock);\
+ ml_set_interrupts_enabled(enable);
-#define NTP_ASSERT_LOCKED() LCK_SPIN_ASSERT(ntp_lock, LCK_ASSERT_OWNED)
+#define NTP_ASSERT_LOCKED() LCK_SPIN_ASSERT(ntp_lock, LCK_ASSERT_OWNED)
static timer_call_data_t ntp_loop_update;
static uint64_t ntp_loop_deadline;
static void refresh_ntp_loop(void);
static void start_ntp_loop(void);
+#if DEVELOPMENT || DEBUG
+uint32_t g_should_log_clock_adjustments = 0;
+SYSCTL_INT(_kern, OID_AUTO, log_clock_adjustments, CTLFLAG_RW | CTLFLAG_LOCKED, &g_should_log_clock_adjustments, 0, "enable kernel clock adjustment logging");
+#endif
+
static bool
ntp_is_time_error(int tsl)
{
+ if (tsl & (STA_UNSYNC | STA_CLOCKERR)) {
+ return true;
+ }
- if (tsl & (STA_UNSYNC | STA_CLOCKERR))
- return (true);
-
- return (false);
+ return false;
}
static void
ntvp->time.tv_sec = atv.tv_sec;
ntvp->time.tv_nsec = atv.tv_nsec;
if ((unsigned long)atv.tv_sec > last_time_maxerror_update) {
- time_maxerror += (MAXFREQ / 1000)*(atv.tv_sec-last_time_maxerror_update);
+ time_maxerror += (MAXFREQ / 1000) * (atv.tv_sec - last_time_maxerror_update);
last_time_maxerror_update = atv.tv_sec;
}
ntvp->maxerror = time_maxerror;
ntvp->tai = time_tai;
ntvp->time_state = time_state;
- if (ntp_is_time_error(time_status))
+ if (ntp_is_time_error(time_status)) {
ntvp->time_state = TIME_ERROR;
+ }
}
int
NTP_UNLOCK(enable);
if (IS_64BIT_PROCESS(p)) {
- struct user64_ntptimeval user_ntv;
+ struct user64_ntptimeval user_ntv = {};
user_ntv.time.tv_sec = ntv.time.tv_sec;
user_ntv.time.tv_nsec = ntv.time.tv_nsec;
user_ntv.maxerror = ntv.maxerror;
user_ntv.time_state = ntv.time_state;
error = copyout(&user_ntv, uap->ntvp, sizeof(user_ntv));
} else {
- struct user32_ntptimeval user_ntv;
+ struct user32_ntptimeval user_ntv = {};
user_ntv.time.tv_sec = ntv.time.tv_sec;
user_ntv.time.tv_nsec = ntv.time.tv_nsec;
user_ntv.maxerror = ntv.maxerror;
error = copyout(&user_ntv, uap->ntvp, sizeof(user_ntv));
}
- if (error)
+ if (error) {
return error;
+ }
return ntv.time_state;
}
int
-ntp_adjtime(struct proc *p, struct ntp_adjtime_args *uap, __unused int32_t *retval)
+ntp_adjtime(struct proc *p, struct ntp_adjtime_args *uap, int32_t *retval)
{
- struct timex ntv;
+ struct timex ntv = {};
long freq;
- int modes;
+ unsigned int modes;
int error, ret = 0;
clock_sec_t sec;
clock_usec_t microsecs;
ntv.constant = user_ntv.constant;
ntv.precision = user_ntv.precision;
ntv.tolerance = user_ntv.tolerance;
-
} else {
struct user32_timex user_ntv;
error = copyin(uap->tp, &user_ntv, sizeof(user_ntv));
ntv.precision = user_ntv.precision;
ntv.tolerance = user_ntv.tolerance;
}
- if (error)
- return (error);
+ if (error) {
+ return error;
+ }
+#if DEVELOPMENT || DEBUG
+ if (g_should_log_clock_adjustments) {
+ os_log(OS_LOG_DEFAULT, "%s: BEFORE modes %u offset %ld freq %ld status %d constant %ld time_adjtime %lld\n",
+ __func__, ntv.modes, ntv.offset, ntv.freq, ntv.status, ntv.constant, time_adjtime);
+ }
+#endif
/*
* Update selected clock variables - only the superuser can
* change anything. Note that there is no error checking here on
if (!IOTaskHasEntitlement(current_task(), SETTIME_ENTITLEMENT)) {
#if CONFIG_MACF
error = mac_system_check_settime(kauth_cred_get());
- if (error)
- return (error);
+ if (error) {
+ return error;
+ }
#endif
- if ((error = priv_check_cred(kauth_cred_get(), PRIV_ADJTIME, 0)))
- return (error);
-
+ if ((error = priv_check_cred(kauth_cred_get(), PRIV_ADJTIME, 0))) {
+ return error;
+ }
}
}
time_maxerror = ntv.maxerror;
last_time_maxerror_update = sec;
}
- if (modes & MOD_ESTERROR)
+ if (modes & MOD_ESTERROR) {
time_esterror = ntv.esterror;
+ }
if (modes & MOD_STATUS) {
if (time_status & STA_PLL && !(ntv.status & STA_PLL)) {
time_state = TIME_OK;
time_status &= STA_SUPPORTED;
}
if (modes & MOD_TIMECONST) {
- if (ntv.constant < 0)
+ if (ntv.constant < 0) {
time_constant = 0;
- else if (ntv.constant > MAXTC)
+ } else if (ntv.constant > MAXTC) {
time_constant = MAXTC;
- else
+ } else {
time_constant = ntv.constant;
+ }
}
if (modes & MOD_TAI) {
- if (ntv.constant > 0)
+ if (ntv.constant > 0) {
time_tai = ntv.constant;
+ }
}
- if (modes & MOD_NANO)
+ if (modes & MOD_NANO) {
time_status |= STA_NANO;
- if (modes & MOD_MICRO)
+ }
+ if (modes & MOD_MICRO) {
time_status &= ~STA_NANO;
- if (modes & MOD_CLKB)
+ }
+ if (modes & MOD_CLKB) {
time_status |= STA_CLK;
- if (modes & MOD_CLKA)
+ }
+ if (modes & MOD_CLKA) {
time_status &= ~STA_CLK;
+ }
if (modes & MOD_FREQUENCY) {
freq = (ntv.freq * 1000LL) >> 16;
- if (freq > MAXFREQ)
+ if (freq > MAXFREQ) {
L_LINT(time_freq, MAXFREQ);
- else if (freq < -MAXFREQ)
+ } else if (freq < -MAXFREQ) {
L_LINT(time_freq, -MAXFREQ);
- else {
+ } else {
/*
* ntv.freq is [PPM * 2^16] = [us/s * 2^16]
* time_freq is [ns/s * 2^32]
}
}
if (modes & MOD_OFFSET) {
- if (time_status & STA_NANO)
+ if (time_status & STA_NANO) {
hardupdate(ntv.offset);
- else
+ } else {
hardupdate(ntv.offset * 1000);
+ }
}
ret = ntp_is_time_error(time_status) ? TIME_ERROR : time_state;
+#if DEVELOPMENT || DEBUG
+ if (g_should_log_clock_adjustments) {
+ os_log(OS_LOG_DEFAULT, "%s: AFTER modes %u offset %lld freq %lld status %d constant %ld time_adjtime %lld\n",
+ __func__, modes, time_offset, time_freq, time_status, time_constant, time_adjtime);
+ }
+#endif
+
/*
* Retrieve all clock variables. Note that the TAI offset is
* returned only by ntp_gettime();
*/
if (IS_64BIT_PROCESS(p)) {
- struct user64_timex user_ntv;
+ struct user64_timex user_ntv = {};
- if (time_status & STA_NANO)
+ user_ntv.modes = modes;
+ if (time_status & STA_NANO) {
user_ntv.offset = L_GINT(time_offset);
- else
+ } else {
user_ntv.offset = L_GINT(time_offset) / 1000;
+ }
user_ntv.freq = L_GINT((time_freq / 1000LL) << 16);
user_ntv.maxerror = time_maxerror;
user_ntv.esterror = time_esterror;
user_ntv.status = time_status;
user_ntv.constant = time_constant;
- if (time_status & STA_NANO)
+ if (time_status & STA_NANO) {
user_ntv.precision = time_precision;
- else
+ } else {
user_ntv.precision = time_precision / 1000;
+ }
user_ntv.tolerance = MAXFREQ * SCALE_PPM;
/* unlock before copyout */
NTP_UNLOCK(enable);
error = copyout(&user_ntv, uap->tp, sizeof(user_ntv));
+ } else {
+ struct user32_timex user_ntv = {};
- }
- else{
- struct user32_timex user_ntv;
-
- if (time_status & STA_NANO)
+ user_ntv.modes = modes;
+ if (time_status & STA_NANO) {
user_ntv.offset = L_GINT(time_offset);
- else
+ } else {
user_ntv.offset = L_GINT(time_offset) / 1000;
+ }
user_ntv.freq = L_GINT((time_freq / 1000LL) << 16);
user_ntv.maxerror = time_maxerror;
user_ntv.esterror = time_esterror;
user_ntv.status = time_status;
user_ntv.constant = time_constant;
- if (time_status & STA_NANO)
+ if (time_status & STA_NANO) {
user_ntv.precision = time_precision;
- else
+ } else {
user_ntv.precision = time_precision / 1000;
+ }
user_ntv.tolerance = MAXFREQ * SCALE_PPM;
/* unlock before copyout */
error = copyout(&user_ntv, uap->tp, sizeof(user_ntv));
}
- if (modes)
+ if (modes) {
start_ntp_loop();
+ }
- if (error == 0)
+ if (error == 0) {
*retval = ret;
+ }
- return (error);
+ return error;
}
int64_t
-ntp_get_freq(void){
+ntp_get_freq(void)
+{
return time_freq;
}
NTP_ASSERT_LOCKED();
if (secs > last_time_maxerror_update) {
- time_maxerror += (MAXFREQ / 1000)*(secs-last_time_maxerror_update);
+ time_maxerror += (MAXFREQ / 1000) * (secs - last_time_maxerror_update);
last_time_maxerror_update = secs;
}
* until the last second is slewed the final < 500 usecs.
*/
if (time_adjtime != 0) {
- if (time_adjtime > 1000000)
+ if (time_adjtime > 1000000) {
tickrate = 5000;
- else if (time_adjtime < -1000000)
+ } else if (time_adjtime < -1000000) {
tickrate = -5000;
- else if (time_adjtime > 500)
+ } else if (time_adjtime > 500) {
tickrate = 500;
- else if (time_adjtime < -500)
+ } else if (time_adjtime < -500) {
tickrate = -500;
- else
+ } else {
tickrate = time_adjtime;
+ }
time_adjtime -= tickrate;
L_LINT(ftemp, tickrate * 1000);
L_ADD(time_adj, ftemp);
if (old_time_adjtime || ((time_offset || old_offset) && (time_offset != old_offset))) {
updated = 1;
- }
- else{
+ } else {
updated = 0;
}
+#if DEVELOPMENT || DEBUG
+ if (g_should_log_clock_adjustments) {
+ int64_t nano = (time_adj > 0)? time_adj >> 32 : -((-time_adj) >> 32);
+ int64_t frac = (time_adj > 0)? ((uint32_t) time_adj) : -((uint32_t) (-time_adj));
+
+ os_log(OS_LOG_DEFAULT, "%s:AFTER offset %lld (%lld) freq %lld status %d "
+ "constant %ld time_adjtime %lld nano %lld frac %lld adj %lld\n",
+ __func__, time_offset, (time_offset > 0)? time_offset >> 32 : -((-time_offset) >> 32),
+ time_freq, time_status, time_constant, time_adjtime, nano, frac, time_adj);
+ }
+#endif
+
*adjustment = time_adj;
}
*/
static void
hardupdate(offset)
- long offset;
+long offset;
{
long mtemp = 0;
long time_monitor;
NTP_ASSERT_LOCKED();
- if (!(time_status & STA_PLL))
+ if (!(time_status & STA_PLL)) {
return;
+ }
- if (offset > MAXPHASE)
+ if (offset > MAXPHASE) {
time_monitor = MAXPHASE;
- else if (offset < -MAXPHASE)
+ } else if (offset < -MAXPHASE) {
time_monitor = -MAXPHASE;
- else
+ } else {
time_monitor = offset;
+ }
L_LINT(time_offset, time_monitor);
clock_get_calendar_uptime(&time_uptime);
}
time_reftime = time_uptime;
- if (L_GINT(time_freq) > MAXFREQ)
+ if (L_GINT(time_freq) > MAXFREQ) {
L_LINT(time_freq, MAXFREQ);
- else if (L_GINT(time_freq) < -MAXFREQ)
+ } else if (L_GINT(time_freq) < -MAXFREQ) {
L_LINT(time_freq, -MAXFREQ);
+ }
}
int64_t ltr, ltw;
boolean_t enable;
- if (delta == NULL)
- return (EINVAL);
+ if (delta == NULL) {
+ return EINVAL;
+ }
ltw = (int64_t)delta->tv_sec * (int64_t)USEC_PER_SEC + delta->tv_usec;
NTP_LOCK(enable);
ltr = time_adjtime;
time_adjtime = ltw;
+#if DEVELOPMENT || DEBUG
+ if (g_should_log_clock_adjustments) {
+ os_log(OS_LOG_DEFAULT, "%s:AFTER offset %lld freq %lld status %d constant %ld time_adjtime %lld\n",
+ __func__, time_offset, time_freq, time_status, time_constant, time_adjtime);
+ }
+#endif
NTP_UNLOCK(enable);
atv.tv_sec = ltr / (int64_t)USEC_PER_SEC;
start_ntp_loop();
- return (0);
+ return 0;
}
int
adjtime(struct proc *p, struct adjtime_args *uap, __unused int32_t *retval)
{
-
struct timeval atv;
int error;
/* Check that this task is entitled to set the time or it is root */
if (!IOTaskHasEntitlement(current_task(), SETTIME_ENTITLEMENT)) {
-
#if CONFIG_MACF
error = mac_system_check_settime(kauth_cred_get());
- if (error)
- return (error);
+ if (error) {
+ return error;
+ }
#endif
- if ((error = priv_check_cred(kauth_cred_get(), PRIV_ADJTIME, 0)))
- return (error);
+ if ((error = priv_check_cred(kauth_cred_get(), PRIV_ADJTIME, 0))) {
+ return error;
+ }
}
if (IS_64BIT_PROCESS(p)) {
atv.tv_sec = user_atv.tv_sec;
atv.tv_usec = user_atv.tv_usec;
}
- if (error)
- return (error);
+ if (error) {
+ return error;
+ }
kern_adjtime(&atv);
if (uap->olddelta) {
if (IS_64BIT_PROCESS(p)) {
- struct user64_timeval user_atv;
+ struct user64_timeval user_atv = {};
user_atv.tv_sec = atv.tv_sec;
user_atv.tv_usec = atv.tv_usec;
error = copyout(&user_atv, uap->olddelta, sizeof(user_atv));
} else {
- struct user32_timeval user_atv;
+ struct user32_timeval user_atv = {};
user_atv.tv_sec = atv.tv_sec;
user_atv.tv_usec = atv.tv_usec;
error = copyout(&user_atv, uap->olddelta, sizeof(user_atv));
}
}
- return (error);
-
+ return error;
}
static void
static void
refresh_ntp_loop(void)
{
-
NTP_ASSERT_LOCKED();
if (--ntp_loop_active == 0) {
/*
if (updated) {
clock_deadline_for_periodic_event(ntp_loop_period, mach_absolute_time(), &ntp_loop_deadline);
- if (!timer_call_enter(&ntp_loop_update, ntp_loop_deadline, TIMER_CALL_SYS_CRITICAL))
- ntp_loop_active++;
+ if (!timer_call_enter(&ntp_loop_update, ntp_loop_deadline, TIMER_CALL_SYS_CRITICAL)) {
+ ntp_loop_active++;
+ }
}
}
-
}
/*
ntp_loop_deadline = mach_absolute_time() + ntp_loop_period;
if (!timer_call_enter(&ntp_loop_update, ntp_loop_deadline, TIMER_CALL_SYS_CRITICAL)) {
- ntp_loop_active++;
+ ntp_loop_active++;
}
NTP_UNLOCK(enable);
static void
init_ntp_loop(void)
{
- uint64_t abstime;
+ uint64_t abstime;
ntp_loop_active = 0;
nanoseconds_to_absolutetime(NTP_LOOP_PERIOD_INTERVAL, &abstime);
void
ntp_init(void)
{
-
L_CLR(time_offset);
L_CLR(time_freq);