/*
 * Copyright (c) Atmosphère-NX
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
#pragma once
#include 
#include 
namespace ams::os::impl {
    class TickManagerImpl {
        private:
            using StandardClock = typename std::conditional::type;
            using TimePoint     = std::chrono::time_point;
        private:
            TimePoint m_start_time;
        public:
            TickManagerImpl() : m_start_time(StandardClock::now()) { /* ... */ }
            ALWAYS_INLINE Tick GetTick() const {
                return Tick(static_cast((StandardClock::now() - m_start_time).count()));
            }
            ALWAYS_INLINE Tick GetSystemTickOrdered() const {
                PerformOrderingForGetSystemTickOrdered(true);
                ON_SCOPE_EXIT { PerformOrderingForGetSystemTickOrdered(false); };
                return Tick(static_cast((StandardClock::now() - m_start_time).count()));
            }
            static constexpr ALWAYS_INLINE s64 GetTickFrequency() {
                return static_cast(StandardClock::period::den) / static_cast(StandardClock::period::num);
            }
            static constexpr ALWAYS_INLINE s64 GetMaxTick() {
                static_assert(GetTickFrequency() <= TimeSpan::FromSeconds(1).GetNanoSeconds());
                return (std::numeric_limits::max() / TimeSpan::FromSeconds(1).GetNanoSeconds()) * GetTickFrequency();
            }
            static constexpr ALWAYS_INLINE s64 GetMaxTimeSpanNs() {
                static_assert(GetTickFrequency() <= TimeSpan::FromSeconds(1).GetNanoSeconds());
                return TimeSpan::FromNanoSeconds(std::numeric_limits::max()).GetNanoSeconds();
            }
        private:
            static ALWAYS_INLINE void PerformOrderingForGetSystemTickOrdered(bool before) {
                #if defined(ATMOSPHERE_ARCH_X64) || defined(ATMOSPHERE_ARCH_X86)
                int a = 0, b, c = 0, d;
                __asm__ __volatile__("cpuid" : "=a"(a), "=b"(b), "=c"(c), "=d"(d) : "0"(a), "2"(c) : "memory");
                AMS_UNUSED(before);
                #elif defined(ATMOSPHERE_ARCH_ARM64)
                if (before) {
                    __asm__ __volatile__("dsb ish" ::: "memory");
                }
                __asm__ __volatile__("isb" ::: "memory");
                #else
                    #error "Unknown architecture for std::chrono TickManager ordering."
                #endif
            }
    };
}