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