| // Copyright 2019 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef QUICHE_COMMON_QUICHE_ENDIAN_H_ |
| #define QUICHE_COMMON_QUICHE_ENDIAN_H_ |
| |
| #include <algorithm> |
| #include <cstdint> |
| #include <type_traits> |
| |
| #include "common/platform/api/quiche_export.h" |
| |
| namespace quiche { |
| |
| enum Endianness { |
| NETWORK_BYTE_ORDER, // big endian |
| HOST_BYTE_ORDER // little endian |
| }; |
| |
| // Provide utility functions that convert from/to network order (big endian) |
| // to/from host order (can be either little or big endian depending on the |
| // platform). |
| class QUICHE_EXPORT_PRIVATE QuicheEndian { |
| public: |
| // Convert |x| from host order (little endian) to network order (big endian). |
| #if defined(__clang__) || \ |
| (defined(__GNUC__) && \ |
| ((__GNUC__ == 4 && __GNUC_MINOR__ >= 8) || __GNUC__ >= 5)) |
| static uint16_t HostToNet16(uint16_t x) { return __builtin_bswap16(x); } |
| static uint32_t HostToNet32(uint32_t x) { return __builtin_bswap32(x); } |
| static uint64_t HostToNet64(uint64_t x) { return __builtin_bswap64(x); } |
| #else |
| static uint16_t HostToNet16(uint16_t x) { return PortableByteSwap(x); } |
| static uint32_t HostToNet32(uint32_t x) { return PortableByteSwap(x); } |
| static uint64_t HostToNet64(uint64_t x) { return PortableByteSwap(x); } |
| #endif |
| |
| // Convert |x| from network order (big endian) to host order (little endian). |
| static uint16_t NetToHost16(uint16_t x) { return HostToNet16(x); } |
| static uint32_t NetToHost32(uint32_t x) { return HostToNet32(x); } |
| static uint64_t NetToHost64(uint64_t x) { return HostToNet64(x); } |
| |
| // Returns true if current host order is little endian. |
| static bool HostIsLittleEndian() { return true; } |
| |
| // Left public for tests. |
| template <typename T> |
| static T PortableByteSwap(T input) { |
| static_assert(std::is_unsigned<T>::value, "T has to be uintNN_t"); |
| union { |
| T number; |
| char bytes[sizeof(T)]; |
| } value; |
| value.number = input; |
| std::reverse(std::begin(value.bytes), std::end(value.bytes)); |
| return value.number; |
| } |
| }; |
| |
| } // namespace quiche |
| |
| #endif // QUICHE_COMMON_QUICHE_ENDIAN_H_ |