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