Blame sysdeps/unix/clock_settime.c

Packit 6c4009
/* Copyright (C) 1999-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <time.h>
Packit 6c4009
#include <sys/time.h>
Packit Service 8bbac7
#include <ldsodefs.h>
Packit Service 8bbac7
Packit Service 8bbac7
Packit Service 8bbac7
#if HP_TIMING_AVAIL && !defined HANDLED_CPUTIME
Packit Service 8bbac7
/* Clock frequency of the processor.  We make it a 64-bit variable
Packit Service 8bbac7
   because some jokers are already playing with processors with more
Packit Service 8bbac7
   than 4GHz.  */
Packit Service 8bbac7
static hp_timing_t freq;
Packit Service 8bbac7
Packit Service 8bbac7
Packit Service 8bbac7
/* This function is defined in the thread library.  */
Packit Service 8bbac7
extern void __pthread_clock_settime (clockid_t clock_id, hp_timing_t offset)
Packit Service 8bbac7
     __attribute__ ((__weak__));
Packit Service 8bbac7
Packit Service 8bbac7
Packit Service 8bbac7
static int
Packit Service 8bbac7
hp_timing_settime (clockid_t clock_id, const struct timespec *tp)
Packit Service 8bbac7
{
Packit Service 8bbac7
  hp_timing_t tsc;
Packit Service 8bbac7
  hp_timing_t usertime;
Packit Service 8bbac7
Packit Service 8bbac7
  /* First thing is to get the current time.  */
Packit Service 8bbac7
  HP_TIMING_NOW (tsc);
Packit Service 8bbac7
Packit Service 8bbac7
  if (__glibc_unlikely (freq == 0))
Packit Service 8bbac7
    {
Packit Service 8bbac7
      /* This can only happen if we haven't initialized the `freq'
Packit Service 8bbac7
	 variable yet.  Do this now. We don't have to protect this
Packit Service 8bbac7
	 code against multiple execution since all of them should lead
Packit Service 8bbac7
	 to the same result.  */
Packit Service 8bbac7
      freq = __get_clockfreq ();
Packit Service 8bbac7
      if (__glibc_unlikely (freq == 0))
Packit Service 8bbac7
	/* Something went wrong.  */
Packit Service 8bbac7
	return -1;
Packit Service 8bbac7
    }
Packit Service 8bbac7
Packit Service 8bbac7
  /* Convert the user-provided time into CPU ticks.  */
Packit Service 8bbac7
  usertime = tp->tv_sec * freq + (tp->tv_nsec * freq) / 1000000000ull;
Packit Service 8bbac7
Packit Service 8bbac7
  /* Determine the offset and use it as the new base value.  */
Packit Service 8bbac7
  if (clock_id == CLOCK_PROCESS_CPUTIME_ID
Packit Service 8bbac7
      || __pthread_clock_settime == NULL)
Packit Service 8bbac7
    GL(dl_cpuclock_offset) = tsc - usertime;
Packit Service 8bbac7
  else
Packit Service 8bbac7
    __pthread_clock_settime (clock_id, tsc - usertime);
Packit Service 8bbac7
Packit Service 8bbac7
  return 0;
Packit Service 8bbac7
}
Packit Service 8bbac7
#endif
Packit Service 8bbac7
Packit Service 6de65a
Packit 6c4009
/* Set CLOCK to value TP.  */
Packit 6c4009
int
Packit 6c4009
__clock_settime (clockid_t clock_id, const struct timespec *tp)
Packit 6c4009
{
Packit Service 8bbac7
  int retval;
Packit 6c4009
Packit 6c4009
  /* Make sure the time cvalue is OK.  */
Packit 6c4009
  if (tp->tv_nsec < 0 || tp->tv_nsec >= 1000000000)
Packit 6c4009
    {
Packit 6c4009
      __set_errno (EINVAL);
Packit 6c4009
      return -1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  switch (clock_id)
Packit 6c4009
    {
Packit Service 8bbac7
#define HANDLE_REALTIME \
Packit Service 8bbac7
      do {								      \
Packit Service 8bbac7
	struct timeval tv;						      \
Packit Service 8bbac7
	TIMESPEC_TO_TIMEVAL (&tv, tp);					      \
Packit Service 8bbac7
									      \
Packit Service 8bbac7
	retval = __settimeofday (&tv, NULL);				      \
Packit Service 8bbac7
      } while (0)
Packit Service 8bbac7
Packit Service 8bbac7
#ifdef SYSDEP_SETTIME
Packit Service 8bbac7
      SYSDEP_SETTIME;
Packit Service 8bbac7
#endif
Packit Service 8bbac7
Packit Service 8bbac7
#ifndef HANDLED_REALTIME
Packit 6c4009
    case CLOCK_REALTIME:
Packit Service 8bbac7
      HANDLE_REALTIME;
Packit 6c4009
      break;
Packit Service 8bbac7
#endif
Packit 6c4009
Packit 6c4009
    default:
Packit Service 8bbac7
#ifdef SYSDEP_SETTIME_CPU
Packit Service 8bbac7
      SYSDEP_SETTIME_CPU;
Packit Service 8bbac7
#endif
Packit Service 8bbac7
#ifndef HANDLED_CPUTIME
Packit Service 8bbac7
# if HP_TIMING_AVAIL
Packit Service 8bbac7
      if (CPUCLOCK_WHICH (clock_id) == CLOCK_PROCESS_CPUTIME_ID
Packit Service 8bbac7
	  || CPUCLOCK_WHICH (clock_id) == CLOCK_THREAD_CPUTIME_ID)
Packit Service 8bbac7
	retval = hp_timing_settime (clock_id, tp);
Packit Service 8bbac7
      else
Packit Service 8bbac7
# endif
Packit Service 8bbac7
	{
Packit Service 8bbac7
	  __set_errno (EINVAL);
Packit Service 8bbac7
	  retval = -1;
Packit Service 8bbac7
	}
Packit Service 8bbac7
#endif
Packit 6c4009
      break;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return retval;
Packit 6c4009
}
Packit Service 8bbac7
weak_alias (__clock_settime, clock_settime)