| // 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 "quiche/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 (little endian). | 
 | class QUICHE_EXPORT 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); } | 
 |  | 
 |   // 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(&value.bytes[0], &value.bytes[sizeof(T)]); | 
 |     return value.number; | 
 |   } | 
 | }; | 
 |  | 
 | enum QuicheVariableLengthIntegerLength : uint8_t { | 
 |   // Length zero means the variable length integer is not present. | 
 |   VARIABLE_LENGTH_INTEGER_LENGTH_0 = 0, | 
 |   VARIABLE_LENGTH_INTEGER_LENGTH_1 = 1, | 
 |   VARIABLE_LENGTH_INTEGER_LENGTH_2 = 2, | 
 |   VARIABLE_LENGTH_INTEGER_LENGTH_4 = 4, | 
 |   VARIABLE_LENGTH_INTEGER_LENGTH_8 = 8, | 
 |  | 
 |   // By default we write the IETF long header length using the 2-byte encoding | 
 |   // of variable length integers, even when the length is below 64, which allows | 
 |   // us to fill in the length before knowing what the length actually is. | 
 |   kQuicheDefaultLongHeaderLengthLength = VARIABLE_LENGTH_INTEGER_LENGTH_2, | 
 | }; | 
 |  | 
 | }  // namespace quiche | 
 |  | 
 | #endif  // QUICHE_COMMON_QUICHE_ENDIAN_H_ |