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