/*
    This file is part of TON Blockchain Library.
    TON Blockchain Library is free software: you can redistribute it and/or modify
    it under the terms of the GNU Lesser General Public License as published by
    the Free Software Foundation, either version 2 of the License, or
    (at your option) any later version.
    TON Blockchain Library is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU Lesser General Public License for more details.
    You should have received a copy of the GNU Lesser General Public License
    along with TON Blockchain Library.  If not, see .
    Copyright 2017-2020 Telegram Systems LLP
*/
#pragma once
#include "td/utils/int_types.h"
namespace td {
struct Clocks {
  static int64 monotonic_nano();
  static double monotonic();
  static double system();
  static int tz_offset();
#if defined(__i386__) or defined(__M_IX86)
  static uint64 rdtsc() {
    unsigned long long int x;
    __asm__ volatile("rdtsc" : "=A"(x));
    return x;
  }
  static constexpr uint64 rdtsc_frequency() {
    return 2000'000'000;
  }
  static constexpr double ticks_per_second() {
    return 2e9;
  }
  static constexpr double inv_ticks_per_second() {
    return 0.5e-9;
  }
#elif defined(__x86_64__) or defined(__M_X64)
  static uint64 rdtsc() {
    unsigned hi, lo;
    __asm__ __volatile__("rdtsc" : "=a"(lo), "=d"(hi));
    return ((unsigned long long)lo) | (((unsigned long long)hi) << 32);
  }
  static constexpr uint64 rdtsc_frequency() {
    return 2000'000'000;
  }
  static constexpr double ticks_per_second() {
    return 2e9;
  }
  static constexpr double inv_ticks_per_second() {
    return 0.5e-9;
  }
#elif defined(__aarch64__) or defined(_M_ARM64)
  static uint64 rdtsc() {
    unsigned long long val;
    asm volatile("mrs %0, cntvct_el0" : "=r"(val));
    return val;
  }
  static uint64 rdtsc_frequency() {
    unsigned long long val;
    asm volatile("mrs %0, cntfrq_el0" : "=r"(val));
    return val;
  }
  static double ticks_per_second() {
    return static_cast(rdtsc_frequency());
  }
  static double inv_ticks_per_second() {
    return 1.0 / static_cast(rdtsc_frequency());
  }
#else
  static uint64 rdtsc() {
    return monotonic_nano();
  }
  static uint64 rdtsc_frequency() {
    return 1000'000'000;
  }
  static double ticks_per_second() {
    return static_cast(rdtsc_frequency());
  }
  static double inv_ticks_per_second() {
    return 1.0 / static_cast(rdtsc_frequency());
  }
#endif
};
}  // namespace td