//===--- clock_gettime linux implementation ---------------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

#ifndef LLVM_LIBC_SRC___SUPPORT_TIME_LINUX_CLOCK_GETTIME_H
#define LLVM_LIBC_SRC___SUPPORT_TIME_LINUX_CLOCK_GETTIME_H

#include "hdr/types/clockid_t.h"
#include "hdr/types/struct_timespec.h"
#include "src/__support/OSUtil/linux/vdso.h"
#include "src/__support/OSUtil/syscall.h"
#include "src/__support/common.h"
#include "src/__support/error_or.h"
#include "src/__support/macros/config.h"
#include <sys/syscall.h>

#if defined(SYS_clock_gettime64)
#include <linux/time_types.h>
#endif

namespace LIBC_NAMESPACE_DECL {
namespace internal {
LIBC_INLINE ErrorOr<int> clock_gettime(clockid_t clockid, timespec *ts) {
  using namespace vdso;
  int ret;
#if defined(SYS_clock_gettime)
  TypedSymbol<VDSOSym::ClockGetTime> clock_gettime;
  if (LIBC_LIKELY(clock_gettime != nullptr))
    ret = clock_gettime(clockid, ts);
  else
    ret = LIBC_NAMESPACE::syscall_impl<int>(SYS_clock_gettime,
                                            static_cast<long>(clockid),
                                            reinterpret_cast<long>(ts));
#elif defined(SYS_clock_gettime64)
  static_assert(
      sizeof(time_t) == sizeof(int64_t),
      "SYS_clock_gettime64 requires struct timespec with 64-bit members.");

  TypedSymbol<VDSOSym::ClockGetTime64> clock_gettime64;
  __kernel_timespec ts64{};
  if (LIBC_LIKELY(clock_gettime64 != nullptr))
    ret = clock_gettime64(clockid, &ts64);
  else
    ret = LIBC_NAMESPACE::syscall_impl<int>(SYS_clock_gettime64,
                                            static_cast<long>(clockid),
                                            reinterpret_cast<long>(&ts64));
  if (ret == 0) {
    ts->tv_sec = static_cast<decltype(ts->tv_sec)>(ts64.tv_sec);
    ts->tv_nsec = static_cast<decltype(ts->tv_nsec)>(ts64.tv_nsec);
  }
#else
#error "SYS_clock_gettime and SYS_clock_gettime64 syscalls not available."
#endif
  if (ret < 0)
    return Error(-ret);
  return ret;
}

} // namespace internal
} // namespace LIBC_NAMESPACE_DECL

#endif // LLVM_LIBC_SRC___SUPPORT_TIME_LINUX_CLOCK_GETTIME_H
