Blame sysdeps/unix/clock_gettime.c

Packit 6c4009
/* clock_gettime -- Get the current time from a POSIX clockid_t.  Unix version.
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 Service 541783
#include <stdint.h>
Packit 6c4009
#include <time.h>
Packit 6c4009
#include <sys/time.h>
Packit Service 541783
#include <libc-internal.h>
Packit Service 541783
#include <ldsodefs.h>
Packit Service 541783
Packit Service 541783
Packit Service 541783
#if HP_TIMING_AVAIL
Packit Service 541783
/* Clock frequency of the processor.  We make it a 64-bit variable
Packit Service 541783
   because some jokers are already playing with processors with more
Packit Service 541783
   than 4GHz.  */
Packit Service 541783
static hp_timing_t freq;
Packit Service 541783
Packit Service 541783
Packit Service 541783
/* This function is defined in the thread library.  */
Packit Service 541783
extern int __pthread_clock_gettime (clockid_t clock_id, hp_timing_t freq,
Packit Service 541783
				    struct timespec *tp)
Packit Service 541783
     __attribute__ ((__weak__));
Packit Service 541783
Packit Service 541783
static int
Packit Service 541783
hp_timing_gettime (clockid_t clock_id, struct timespec *tp)
Packit Service 541783
{
Packit Service 541783
  hp_timing_t tsc;
Packit Service 541783
Packit Service 541783
  if (__glibc_unlikely (freq == 0))
Packit Service 541783
    {
Packit Service 541783
      /* This can only happen if we haven't initialized the `freq'
Packit Service 541783
	 variable yet.  Do this now. We don't have to protect this
Packit Service 541783
	 code against multiple execution since all of them should
Packit Service 541783
	 lead to the same result.  */
Packit Service 541783
      freq = __get_clockfreq ();
Packit Service 541783
      if (__glibc_unlikely (freq == 0))
Packit Service 541783
	/* Something went wrong.  */
Packit Service 541783
	return -1;
Packit Service 541783
    }
Packit Service 541783
Packit Service 541783
  if (clock_id != CLOCK_PROCESS_CPUTIME_ID
Packit Service 541783
      && __pthread_clock_gettime != NULL)
Packit Service 541783
    return __pthread_clock_gettime (clock_id, freq, tp);
Packit Service 541783
Packit Service 541783
  /* Get the current counter.  */
Packit Service 541783
  HP_TIMING_NOW (tsc);
Packit Service 541783
Packit Service 541783
  /* Compute the offset since the start time of the process.  */
Packit Service 541783
  tsc -= GL(dl_cpuclock_offset);
Packit Service 541783
Packit Service 541783
  /* Compute the seconds.  */
Packit Service 541783
  tp->tv_sec = tsc / freq;
Packit Service 541783
Packit Service 541783
  /* And the nanoseconds.  This computation should be stable until
Packit Service 541783
     we get machines with about 16GHz frequency.  */
Packit Service 541783
  tp->tv_nsec = ((tsc % freq) * UINT64_C (1000000000)) / freq;
Packit Service 541783
Packit Service 541783
  return 0;
Packit Service 541783
}
Packit Service 541783
#endif
Packit Service 541783
Packit Service 541783
Packit Service 541783
static inline int
Packit Service 541783
realtime_gettime (struct timespec *tp)
Packit Service 541783
{
Packit Service 541783
  struct timeval tv;
Packit Service 541783
  int retval = __gettimeofday (&tv, NULL);
Packit Service 541783
  if (retval == 0)
Packit Service 541783
    /* Convert into `timespec'.  */
Packit Service 541783
    TIMEVAL_TO_TIMESPEC (&tv, tp);
Packit Service 541783
  return retval;
Packit Service 541783
}
Packit Service 541783
Packit 6c4009
Packit 6c4009
/* Get current value of CLOCK and store it in TP.  */
Packit 6c4009
int
Packit 6c4009
__clock_gettime (clockid_t clock_id, struct timespec *tp)
Packit 6c4009
{
Packit 6c4009
  int retval = -1;
Packit 6c4009
Packit 6c4009
  switch (clock_id)
Packit 6c4009
    {
Packit 6c4009
    case CLOCK_REALTIME:
Packit 6c4009
      {
Packit 6c4009
	struct timeval tv;
Packit 6c4009
	retval = __gettimeofday (&tv, NULL);
Packit 6c4009
	if (retval == 0)
Packit 6c4009
	  TIMEVAL_TO_TIMESPEC (&tv, tp);
Packit 6c4009
      }
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    default:
Packit Service 541783
#if HP_TIMING_AVAIL
Packit Service 541783
      if ((clock_id & ((1 << CLOCK_IDFIELD_SIZE) - 1))
Packit Service 541783
	  == CLOCK_THREAD_CPUTIME_ID)
Packit Service 541783
	retval = hp_timing_gettime (clock_id, tp);
Packit Service 541783
      else
Packit Service 541783
#endif
Packit Service 541783
	__set_errno (EINVAL);
Packit Service 541783
      break;
Packit Service 541783
Packit Service 0b1af7
#if HP_TIMING_AVAIL
Packit Service 541783
    case CLOCK_PROCESS_CPUTIME_ID:
Packit Service 541783
      retval = hp_timing_gettime (clock_id, tp);
Packit 6c4009
      break;
Packit Service 541783
#endif
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return retval;
Packit 6c4009
}
Packit Service 541783
weak_alias (__clock_gettime, clock_gettime)
Packit 6c4009
libc_hidden_def (__clock_gettime)