xref: /openbsd-src/gnu/llvm/compiler-rt/lib/xray/xray_tsc.h (revision 810390e339a5425391477d5d41c78d7cab2424ac)
13cab2bb3Spatrick //===-- xray_tsc.h ----------------------------------------------*- C++ -*-===//
23cab2bb3Spatrick //
33cab2bb3Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
43cab2bb3Spatrick // See https://llvm.org/LICENSE.txt for license information.
53cab2bb3Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
63cab2bb3Spatrick //
73cab2bb3Spatrick //===----------------------------------------------------------------------===//
83cab2bb3Spatrick //
93cab2bb3Spatrick // This file is a part of XRay, a dynamic runtime instrumentation system.
103cab2bb3Spatrick //
113cab2bb3Spatrick //===----------------------------------------------------------------------===//
123cab2bb3Spatrick #ifndef XRAY_EMULATE_TSC_H
133cab2bb3Spatrick #define XRAY_EMULATE_TSC_H
143cab2bb3Spatrick 
153cab2bb3Spatrick #include "sanitizer_common/sanitizer_common.h"
163cab2bb3Spatrick 
173cab2bb3Spatrick namespace __xray {
183cab2bb3Spatrick static constexpr uint64_t NanosecondsPerSecond = 1000ULL * 1000 * 1000;
193cab2bb3Spatrick }
203cab2bb3Spatrick 
213cab2bb3Spatrick #if SANITIZER_FUCHSIA
223cab2bb3Spatrick #include <zircon/syscalls.h>
233cab2bb3Spatrick 
243cab2bb3Spatrick namespace __xray {
253cab2bb3Spatrick 
probeRequiredCPUFeatures()263cab2bb3Spatrick inline bool probeRequiredCPUFeatures() XRAY_NEVER_INSTRUMENT { return true; }
273cab2bb3Spatrick 
readTSC(uint8_t & CPU)283cab2bb3Spatrick ALWAYS_INLINE uint64_t readTSC(uint8_t &CPU) XRAY_NEVER_INSTRUMENT {
293cab2bb3Spatrick   CPU = 0;
303cab2bb3Spatrick   return _zx_ticks_get();
313cab2bb3Spatrick }
323cab2bb3Spatrick 
getTSCFrequency()333cab2bb3Spatrick inline uint64_t getTSCFrequency() XRAY_NEVER_INSTRUMENT {
343cab2bb3Spatrick   return _zx_ticks_per_second();
353cab2bb3Spatrick }
363cab2bb3Spatrick 
373cab2bb3Spatrick } // namespace __xray
383cab2bb3Spatrick 
393cab2bb3Spatrick #else // SANITIZER_FUCHSIA
403cab2bb3Spatrick 
413cab2bb3Spatrick #if defined(__x86_64__)
423cab2bb3Spatrick #include "xray_x86_64.inc"
433cab2bb3Spatrick #elif defined(__powerpc64__)
443cab2bb3Spatrick #include "xray_powerpc64.inc"
45*810390e3Srobert #elif defined(__arm__) || defined(__aarch64__) || defined(__mips__) ||         \
46*810390e3Srobert     defined(__hexagon__)
473cab2bb3Spatrick // Emulated TSC.
483cab2bb3Spatrick // There is no instruction like RDTSCP in user mode on ARM. ARM's CP15 does
493cab2bb3Spatrick //   not have a constant frequency like TSC on x86(_64), it may go faster
503cab2bb3Spatrick //   or slower depending on CPU turbo or power saving mode. Furthermore,
513cab2bb3Spatrick //   to read from CP15 on ARM a kernel modification or a driver is needed.
523cab2bb3Spatrick //   We can not require this from users of compiler-rt.
533cab2bb3Spatrick // So on ARM we use clock_gettime() which gives the result in nanoseconds.
543cab2bb3Spatrick //   To get the measurements per second, we scale this by the number of
553cab2bb3Spatrick //   nanoseconds per second, pretending that the TSC frequency is 1GHz and
563cab2bb3Spatrick //   one TSC tick is 1 nanosecond.
573cab2bb3Spatrick #include "sanitizer_common/sanitizer_common.h"
583cab2bb3Spatrick #include "sanitizer_common/sanitizer_internal_defs.h"
593cab2bb3Spatrick #include "xray_defs.h"
603cab2bb3Spatrick #include <cerrno>
613cab2bb3Spatrick #include <cstdint>
623cab2bb3Spatrick #include <time.h>
633cab2bb3Spatrick 
643cab2bb3Spatrick namespace __xray {
653cab2bb3Spatrick 
probeRequiredCPUFeatures()663cab2bb3Spatrick inline bool probeRequiredCPUFeatures() XRAY_NEVER_INSTRUMENT { return true; }
673cab2bb3Spatrick 
readTSC(uint8_t & CPU)683cab2bb3Spatrick ALWAYS_INLINE uint64_t readTSC(uint8_t &CPU) XRAY_NEVER_INSTRUMENT {
693cab2bb3Spatrick   timespec TS;
703cab2bb3Spatrick   int result = clock_gettime(CLOCK_REALTIME, &TS);
713cab2bb3Spatrick   if (result != 0) {
723cab2bb3Spatrick     Report("clock_gettime(2) returned %d, errno=%d.", result, int(errno));
733cab2bb3Spatrick     TS.tv_sec = 0;
743cab2bb3Spatrick     TS.tv_nsec = 0;
753cab2bb3Spatrick   }
763cab2bb3Spatrick   CPU = 0;
773cab2bb3Spatrick   return TS.tv_sec * NanosecondsPerSecond + TS.tv_nsec;
783cab2bb3Spatrick }
793cab2bb3Spatrick 
getTSCFrequency()803cab2bb3Spatrick inline uint64_t getTSCFrequency() XRAY_NEVER_INSTRUMENT {
813cab2bb3Spatrick   return NanosecondsPerSecond;
823cab2bb3Spatrick }
833cab2bb3Spatrick 
843cab2bb3Spatrick } // namespace __xray
853cab2bb3Spatrick 
863cab2bb3Spatrick #else
873cab2bb3Spatrick #error Target architecture is not supported.
883cab2bb3Spatrick #endif // CPU architecture
893cab2bb3Spatrick #endif // SANITIZER_FUCHSIA
903cab2bb3Spatrick 
913cab2bb3Spatrick #endif // XRAY_EMULATE_TSC_H
92