osutils.cpp 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. // [AsmJit]
  2. // Machine Code Generation for C++.
  3. //
  4. // [License]
  5. // Zlib - See LICENSE.md file in the package.
  6. #define ASMJIT_EXPORTS
  7. #include "../core/osutils.h"
  8. #include "../core/support.h"
  9. #if defined(_WIN32)
  10. #include <atomic>
  11. #elif defined(__APPLE__)
  12. #include <mach/mach_time.h>
  13. #else
  14. #include <time.h>
  15. #include <unistd.h>
  16. #endif
  17. ASMJIT_BEGIN_NAMESPACE
  18. // ============================================================================
  19. // [asmjit::OSUtils - GetTickCount]
  20. // ============================================================================
  21. uint32_t OSUtils::getTickCount() noexcept {
  22. #if defined(_WIN32)
  23. enum HiResStatus : uint32_t {
  24. kHiResUnknown = 0,
  25. kHiResAvailable = 1,
  26. kHiResNotAvailable = 2
  27. };
  28. static std::atomic<uint32_t> _hiResStatus(kHiResUnknown);
  29. static volatile double _hiResFreq(0);
  30. uint32_t status = _hiResStatus.load();
  31. LARGE_INTEGER now, qpf;
  32. if (status != kHiResNotAvailable && ::QueryPerformanceCounter(&now)) {
  33. double freq = _hiResFreq;
  34. if (status == kHiResUnknown) {
  35. // Detects the availability of high resolution counter.
  36. if (::QueryPerformanceFrequency(&qpf)) {
  37. freq = double(qpf.QuadPart) / 1000.0;
  38. _hiResFreq = freq;
  39. _hiResStatus.compare_exchange_strong(status, kHiResAvailable);
  40. status = kHiResAvailable;
  41. }
  42. else {
  43. // High resolution not available.
  44. _hiResStatus.compare_exchange_strong(status, kHiResNotAvailable);
  45. }
  46. }
  47. if (status == kHiResAvailable)
  48. return uint32_t(uint64_t(int64_t(double(now.QuadPart) / freq)) & 0xFFFFFFFFu);
  49. }
  50. // Bail to `GetTickCount()` if we cannot use high resolution.
  51. return ::GetTickCount();
  52. #elif defined(__APPLE__)
  53. // See Apple's QA1398.
  54. static mach_timebase_info_data_t _machTime;
  55. uint32_t denom = _machTime.denom;
  56. if (ASMJIT_UNLIKELY(!denom)) {
  57. if (mach_timebase_info(&_machTime) != KERN_SUCCESS || !(denom = _machTime.denom))
  58. return 0;
  59. }
  60. // `mach_absolute_time()` returns nanoseconds, we want milliseconds.
  61. uint64_t t = mach_absolute_time() / 1000000u;
  62. t = (t * _machTime.numer) / _machTime.denom;
  63. return uint32_t(t & 0xFFFFFFFFu);
  64. #elif defined(_POSIX_MONOTONIC_CLOCK) && _POSIX_MONOTONIC_CLOCK >= 0
  65. struct timespec ts;
  66. if (ASMJIT_UNLIKELY(clock_gettime(CLOCK_MONOTONIC, &ts) != 0))
  67. return 0;
  68. uint64_t t = (uint64_t(ts.tv_sec ) * 1000u) + (uint64_t(ts.tv_nsec) / 1000000u);
  69. return uint32_t(t & 0xFFFFFFFFu);
  70. #else
  71. #pragma message("asmjit::OSUtils::getTickCount() doesn't have implementation for the target OS.")
  72. return 0;
  73. #endif
  74. }
  75. ASMJIT_END_NAMESPACE