#pragma once
#include <stdint.h>

#ifdef _MSC_VER
#  include <stdlib.h>
#endif

namespace panda {

namespace detail {
    union check_endianess { unsigned x; unsigned char c; };
    static const bool am_i_little = (check_endianess{1}).c;

#ifdef _MSC_VER

    inline uint16_t swap_bytes16 (uint16_t x) { return _byteswap_ushort(x); }
    inline uint32_t swap_bytes32 (uint32_t x) { return _byteswap_ulong(x); }
    inline uint64_t swap_bytes64 (uint64_t x) { return _byteswap_uint64(x); }

#elif defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)) && 0

    inline uint16_t swap_bytes16 (uint16_t x) { return __builtin_swap_bytes16(x); }
    inline uint32_t swap_bytes32 (uint32_t x) { return __builtin_swap_bytes32(x); }
    inline uint64_t swap_bytes64 (uint64_t x) { return __builtin_swap_bytes64(x); }

#else

    inline uint16_t swap_bytes16 (uint16_t x) {
        return ((x >> 8) & 0xff) | (x << 8);
    }

    inline uint32_t swap_bytes32 (uint32_t x) {
        return ((x & 0xff000000) >> 24) | ((x & 0x00ff0000) >> 8) | ((x & 0x0000ff00) << 8) | (x << 24);
    }

    inline uint64_t swap_bytes64 (uint64_t x) {
        union { uint64_t u64; uint32_t u32[2]; } v1, v2;
        v1.u64 = x;
        v2.u32[0] = swap_bytes32(v1.u32[1]);
        v2.u32[1] = swap_bytes32(v1.u32[0]);
        return v2.u64;
    }

#endif

}

inline uint16_t h2be16 (uint16_t x) { return detail::am_i_little ? detail::swap_bytes16(x) : x; }
inline uint16_t h2le16 (uint16_t x) { return detail::am_i_little ? x : detail::swap_bytes16(x); }
inline uint16_t be2h16 (uint16_t x) { return detail::am_i_little ? detail::swap_bytes16(x) : x; }
inline uint16_t le2h16 (uint16_t x) { return detail::am_i_little ? x : detail::swap_bytes16(x); }

inline uint32_t h2be32 (uint32_t x) { return detail::am_i_little ? detail::swap_bytes32(x) : x; }
inline uint32_t h2le32 (uint32_t x) { return detail::am_i_little ? x : detail::swap_bytes32(x); }
inline uint32_t be2h32 (uint32_t x) { return detail::am_i_little ? detail::swap_bytes32(x) : x; }
inline uint32_t le2h32 (uint32_t x) { return detail::am_i_little ? x : detail::swap_bytes32(x); }

inline uint64_t h2be64 (uint64_t x) { return detail::am_i_little ? detail::swap_bytes64(x) : x; }
inline uint64_t h2le64 (uint64_t x) { return detail::am_i_little ? x : detail::swap_bytes64(x); }
inline uint64_t be2h64 (uint64_t x) { return detail::am_i_little ? detail::swap_bytes64(x) : x; }
inline uint64_t le2h64 (uint64_t x) { return detail::am_i_little ? x : detail::swap_bytes64(x); }

// just to make templates simpler
inline uint8_t h2be (uint8_t x) { return x; }
inline uint8_t h2le (uint8_t x) { return x; }
inline uint8_t be2h (uint8_t x) { return x; }
inline uint8_t le2h (uint8_t x) { return x; }

inline uint16_t h2be (uint16_t x) { return h2be16(x); }
inline uint16_t h2le (uint16_t x) { return h2le16(x); }
inline uint16_t be2h (uint16_t x) { return be2h16(x); }
inline uint16_t le2h (uint16_t x) { return le2h16(x); }

inline uint32_t h2be (uint32_t x) { return h2be32(x); }
inline uint32_t h2le (uint32_t x) { return h2le32(x); }
inline uint32_t be2h (uint32_t x) { return be2h32(x); }
inline uint32_t le2h (uint32_t x) { return le2h32(x); }

inline uint64_t h2be (uint64_t x) { return h2be64(x); }
inline uint64_t h2le (uint64_t x) { return h2le64(x); }
inline uint64_t be2h (uint64_t x) { return be2h64(x); }
inline uint64_t le2h (uint64_t x) { return le2h64(x); }

}