/*
 * 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::util {
    consteval bool IsLittleEndian() {
        return std::endian::native == std::endian::little;
    }
    consteval bool IsBigEndian() {
        return std::endian::native == std::endian::big;
    }
    static_assert(IsLittleEndian() ^ IsBigEndian());
    template
    constexpr ALWAYS_INLINE U SwapEndian(const U u) {
        static_assert(BITSIZEOF(u8) == 8);
        if constexpr (sizeof(U) * BITSIZEOF(u8) == 64) {
            static_assert(__builtin_bswap64(UINT64_C(0x0123456789ABCDEF)) == UINT64_C(0xEFCDAB8967452301));
            return __builtin_bswap64(u);
        } else if constexpr (sizeof(U) * BITSIZEOF(u8) == 32) {
            static_assert(__builtin_bswap32(0x01234567u) == 0x67452301u);
            return __builtin_bswap32(u);
        } else if constexpr (sizeof(U) * BITSIZEOF(u8) == 16) {
            static_assert(__builtin_bswap16(0x0123u) == 0x2301u);
            return __builtin_bswap16(u);
        } else if constexpr (sizeof(U) * BITSIZEOF(u8) == 8) {
            return u;
        } else {
            static_assert(!std::is_same::value);
        }
    }
    constexpr ALWAYS_INLINE u64 SwapEndian48(const u64 u) {
        using U = u64;
        static_assert(BITSIZEOF(u8) == 8);
        constexpr U ByteMask = 0xFFu;
        AMS_ASSERT((u & UINT64_C(0xFFFF000000000000)) == 0);
        return  ((u & (ByteMask << 40)) >> 40) |
                ((u & (ByteMask << 32)) >> 24) |
                ((u & (ByteMask << 24)) >>  8) |
                ((u & (ByteMask << 16)) <<  8) |
                ((u & (ByteMask <<  8)) << 24) |
                ((u & (ByteMask <<  0)) << 40);
    }
    template
    constexpr ALWAYS_INLINE void SwapEndian(T *ptr) {
        using U = typename std::make_unsigned::type;
        *ptr = static_cast(SwapEndian(static_cast(*ptr)));
    }
    template
    constexpr ALWAYS_INLINE T ConvertToBigEndian(const T val) {
        using U = typename std::make_unsigned::type;
        if constexpr (IsBigEndian()) {
            return static_cast(static_cast(val));
        } else {
            static_assert(IsLittleEndian());
            return static_cast(SwapEndian(static_cast(val)));
        }
    }
    template
    constexpr ALWAYS_INLINE T ConvertToLittleEndian(const T val) {
        using U = typename std::make_unsigned::type;
        if constexpr (IsBigEndian()) {
            return static_cast(SwapEndian(static_cast(val)));
        } else {
            static_assert(IsLittleEndian());
            return static_cast(static_cast(val));
        }
    }
    template
    constexpr ALWAYS_INLINE T ConvertFromBigEndian(const T val) {
        return ConvertToBigEndian(val);
    }
    template
    constexpr ALWAYS_INLINE T ConvertFromLittleEndian(const T val) {
        return ConvertToLittleEndian(val);
    }
    template
    constexpr ALWAYS_INLINE T ConvertToBigEndian48(const T val) {
        using U = typename std::make_unsigned::type;
        static_assert(sizeof(T) == sizeof(u64));
        if constexpr (IsBigEndian()) {
            AMS_ASSERT((static_cast(val) & UINT64_C(0xFFFF000000000000)) == 0);
            return static_cast(static_cast(val));
        } else {
            static_assert(IsLittleEndian());
            return static_cast(SwapEndian48(static_cast(val)));
        }
    }
    template
    constexpr ALWAYS_INLINE T ConvertToLittleEndian48(const T val) {
        using U = typename std::make_unsigned::type;
        static_assert(sizeof(T) == sizeof(u64));
        if constexpr (IsBigEndian()) {
            return static_cast(SwapEndian48(static_cast(val)));
        } else {
            static_assert(IsLittleEndian());
            AMS_ASSERT((static_cast(val) & UINT64_C(0xFFFF000000000000)) == 0);
            return static_cast(static_cast(val));
        }
    }
    template
    constexpr ALWAYS_INLINE T LoadBigEndian(const T *ptr) {
        return ConvertToBigEndian(*ptr);
    }
    template
    constexpr ALWAYS_INLINE T LoadLittleEndian(const T *ptr) {
        return ConvertToLittleEndian(*ptr);
    }
    template
    constexpr ALWAYS_INLINE void StoreBigEndian(T *ptr, T val) {
        *static_cast(ptr) = ConvertToBigEndian(val);
    }
    template
    constexpr ALWAYS_INLINE void StoreLittleEndian(T *ptr, T val) {
        *static_cast(ptr) = ConvertToLittleEndian(val);
    }
}