1 //===-- xray_tsc.h ----------------------------------------------*- C++ -*-===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // This file is a part of XRay, a dynamic runtime instrumentation system.
11 //===----------------------------------------------------------------------===//
12 #ifndef XRAY_EMULATE_TSC_H
13 #define XRAY_EMULATE_TSC_H
15 #include "sanitizer_common/sanitizer_common.h"
18 static constexpr uint64_t NanosecondsPerSecond
= 1000ULL * 1000 * 1000;
22 #include <zircon/syscalls.h>
26 inline bool probeRequiredCPUFeatures() XRAY_NEVER_INSTRUMENT
{ return true; }
28 ALWAYS_INLINE
uint64_t readTSC(uint8_t &CPU
) XRAY_NEVER_INSTRUMENT
{
30 return _zx_ticks_get();
33 inline uint64_t getTSCFrequency() XRAY_NEVER_INSTRUMENT
{
34 return _zx_ticks_per_second();
39 #else // SANITIZER_FUCHSIA
41 #if defined(__x86_64__)
42 #include "xray_x86_64.inc"
43 #elif defined(__powerpc64__)
44 #include "xray_powerpc64.inc"
45 #elif defined(__arm__) || defined(__aarch64__) || defined(__mips__) || \
48 // There is no instruction like RDTSCP in user mode on ARM. ARM's CP15 does
49 // not have a constant frequency like TSC on x86(_64), it may go faster
50 // or slower depending on CPU turbo or power saving mode. Furthermore,
51 // to read from CP15 on ARM a kernel modification or a driver is needed.
52 // We can not require this from users of compiler-rt.
53 // So on ARM we use clock_gettime() which gives the result in nanoseconds.
54 // To get the measurements per second, we scale this by the number of
55 // nanoseconds per second, pretending that the TSC frequency is 1GHz and
56 // one TSC tick is 1 nanosecond.
57 #include "sanitizer_common/sanitizer_common.h"
58 #include "sanitizer_common/sanitizer_internal_defs.h"
59 #include "xray_defs.h"
66 inline bool probeRequiredCPUFeatures() XRAY_NEVER_INSTRUMENT
{ return true; }
68 ALWAYS_INLINE
uint64_t readTSC(uint8_t &CPU
) XRAY_NEVER_INSTRUMENT
{
70 int result
= clock_gettime(CLOCK_REALTIME
, &TS
);
72 Report("clock_gettime(2) returned %d, errno=%d.", result
, int(errno
));
77 return TS
.tv_sec
* NanosecondsPerSecond
+ TS
.tv_nsec
;
80 inline uint64_t getTSCFrequency() XRAY_NEVER_INSTRUMENT
{
81 return NanosecondsPerSecond
;
87 #error Target architecture is not supported.
88 #endif // CPU architecture
89 #endif // SANITIZER_FUCHSIA
91 #endif // XRAY_EMULATE_TSC_H