C++程序  |  617行  |  17.9 KB

#ifndef ANDROID_PDX_RPC_ENCODING_H_
#define ANDROID_PDX_RPC_ENCODING_H_

#include <array>
#include <cstdint>
#include <cstring>
#include <map>
#include <numeric>
#include <string>
#include <tuple>
#include <unordered_map>
#include <vector>

#include <pdx/channel_handle.h>
#include <pdx/file_handle.h>

#include "array_wrapper.h"
#include "buffer_wrapper.h"
#include "string_wrapper.h"
#include "variant.h"

namespace android {
namespace pdx {
namespace rpc {

// This library uses a subset, or profile, of MessagePack (http://msgpack.org)
// to encode supported data types during serialization and to verify the
// expected data types during deserialization. One notable deviation from the
// MessagePack specification is that little-endian byte order is used for
// multi-byte numeric types to avoid unnecessary conversion on nearly all
// relevant architectures.
//
// Some data types, integers for example, support multiple encoding strategies.
// This library attempts to optimize for space based on the value of such types.
// However, during decode all valid encodings for a given type are accepted.

// Prefix byte for type encodings. This is the complete list of prefix bytes
// from the MessagePack specification, even though not all types are used by
// this library.
enum EncodingPrefix {
  ENCODING_TYPE_POSITIVE_FIXINT = 0x00,
  ENCODING_TYPE_POSITIVE_FIXINT_MIN = 0x00,
  ENCODING_TYPE_POSITIVE_FIXINT_MAX = 0x7f,
  ENCODING_TYPE_POSITIVE_FIXINT_MASK = 0x7f,
  ENCODING_TYPE_FIXMAP = 0x80,
  ENCODING_TYPE_FIXMAP_MIN = 0x80,
  ENCODING_TYPE_FIXMAP_MAX = 0x8f,
  ENCODING_TYPE_FIXMAP_MASK = 0x0f,
  ENCODING_TYPE_FIXARRAY = 0x90,
  ENCODING_TYPE_FIXARRAY_MIN = 0x90,
  ENCODING_TYPE_FIXARRAY_MAX = 0x9f,
  ENCODING_TYPE_FIXARRAY_MASK = 0x0f,
  ENCODING_TYPE_FIXSTR = 0xa0,
  ENCODING_TYPE_FIXSTR_MIN = 0xa0,
  ENCODING_TYPE_FIXSTR_MAX = 0xbf,
  ENCODING_TYPE_FIXSTR_MASK = 0x1f,
  ENCODING_TYPE_NIL = 0xc0,
  ENCODING_TYPE_RESERVED = 0xc1,
  ENCODING_TYPE_FALSE = 0xc2,
  ENCODING_TYPE_TRUE = 0xc3,
  ENCODING_TYPE_BIN8 = 0xc4,
  ENCODING_TYPE_BIN16 = 0xc5,
  ENCODING_TYPE_BIN32 = 0xc6,
  ENCODING_TYPE_EXT8 = 0xc7,
  ENCODING_TYPE_EXT16 = 0xc8,
  ENCODING_TYPE_EXT32 = 0xc9,
  ENCODING_TYPE_FLOAT32 = 0xca,
  ENCODING_TYPE_FLOAT64 = 0xcb,
  ENCODING_TYPE_UINT8 = 0xcc,
  ENCODING_TYPE_UINT16 = 0xcd,
  ENCODING_TYPE_UINT32 = 0xce,
  ENCODING_TYPE_UINT64 = 0xcf,
  ENCODING_TYPE_INT8 = 0xd0,
  ENCODING_TYPE_INT16 = 0xd1,
  ENCODING_TYPE_INT32 = 0xd2,
  ENCODING_TYPE_INT64 = 0xd3,
  ENCODING_TYPE_FIXEXT1 = 0xd4,
  ENCODING_TYPE_FIXEXT2 = 0xd5,
  ENCODING_TYPE_FIXEXT4 = 0xd6,
  ENCODING_TYPE_FIXEXT8 = 0xd7,
  ENCODING_TYPE_FIXEXT16 = 0xd8,
  ENCODING_TYPE_STR8 = 0xd9,
  ENCODING_TYPE_STR16 = 0xda,
  ENCODING_TYPE_STR32 = 0xdb,
  ENCODING_TYPE_ARRAY16 = 0xdc,
  ENCODING_TYPE_ARRAY32 = 0xdd,
  ENCODING_TYPE_MAP16 = 0xde,
  ENCODING_TYPE_MAP32 = 0xdf,
  ENCODING_TYPE_NEGATIVE_FIXINT = 0xe0,
  ENCODING_TYPE_NEGATIVE_FIXINT_MIN = 0xe0,
  ENCODING_TYPE_NEGATIVE_FIXINT_MAX = 0xff,
};

// Base encoding classes grouping multi-strategy encodings.
enum EncodingClass {
  ENCODING_CLASS_BOOL,
  ENCODING_CLASS_NIL,
  ENCODING_CLASS_INT,
  ENCODING_CLASS_UINT,
  ENCODING_CLASS_FLOAT,
  ENCODING_CLASS_ARRAY,
  ENCODING_CLASS_MAP,
  ENCODING_CLASS_STRING,
  ENCODING_CLASS_BINARY,
  ENCODING_CLASS_EXTENSION,
};

// Encoding prefixes are unsigned bytes.
typedef std::uint8_t EncodingType;

// Extension encoding types defined by this library.
enum EncodingExtType : int8_t {
  ENCODING_EXT_TYPE_FILE_DESCRIPTOR,
  ENCODING_EXT_TYPE_CHANNEL_HANDLE,
};

// Encoding predicates. Determines whether the given encoding is of a specific
// type.
inline constexpr bool IsFixintEncoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_NEGATIVE_FIXINT_MIN ... ENCODING_TYPE_NEGATIVE_FIXINT_MAX:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsUnsignedFixintEncoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsInt8Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_NEGATIVE_FIXINT_MIN ... ENCODING_TYPE_NEGATIVE_FIXINT_MAX:
    case ENCODING_TYPE_INT8:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsUInt8Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_UINT8:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsInt16Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_NEGATIVE_FIXINT_MIN ... ENCODING_TYPE_NEGATIVE_FIXINT_MAX:
    case ENCODING_TYPE_INT8:
    case ENCODING_TYPE_INT16:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsUInt16Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_UINT8:
    case ENCODING_TYPE_UINT16:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsInt32Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_NEGATIVE_FIXINT_MIN ... ENCODING_TYPE_NEGATIVE_FIXINT_MAX:
    case ENCODING_TYPE_INT8:
    case ENCODING_TYPE_INT16:
    case ENCODING_TYPE_INT32:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsUInt32Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_UINT8:
    case ENCODING_TYPE_UINT16:
    case ENCODING_TYPE_UINT32:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsInt64Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_NEGATIVE_FIXINT_MIN ... ENCODING_TYPE_NEGATIVE_FIXINT_MAX:
    case ENCODING_TYPE_INT8:
    case ENCODING_TYPE_INT16:
    case ENCODING_TYPE_INT32:
    case ENCODING_TYPE_INT64:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsUInt64Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_UINT8:
    case ENCODING_TYPE_UINT16:
    case ENCODING_TYPE_UINT32:
    case ENCODING_TYPE_UINT64:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsFixmapEncoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FIXMAP_MIN ... ENCODING_TYPE_FIXMAP_MAX:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsFixarrayEncoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FIXARRAY_MIN ... ENCODING_TYPE_FIXARRAY_MAX:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsFixstrEncoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FIXSTR_MIN ... ENCODING_TYPE_FIXSTR_MAX:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsFixextEncoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FIXEXT1:
    case ENCODING_TYPE_FIXEXT2:
    case ENCODING_TYPE_FIXEXT4:
    case ENCODING_TYPE_FIXEXT8:
    case ENCODING_TYPE_FIXEXT16:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsFloat32Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FLOAT32:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsFloat64Encoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FLOAT32:
    case ENCODING_TYPE_FLOAT64:
      return true;
    default:
      return false;
  }
}

inline constexpr bool IsBoolEncoding(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FALSE:
    case ENCODING_TYPE_TRUE:
      return true;
    default:
      return false;
  }
}

inline constexpr std::size_t GetFixstrSize(EncodingType encoding) {
  return encoding & ENCODING_TYPE_FIXSTR_MASK;
}

inline constexpr std::size_t GetFixarraySize(EncodingType encoding) {
  return encoding & ENCODING_TYPE_FIXARRAY_MASK;
}

inline constexpr std::size_t GetFixmapSize(EncodingType encoding) {
  return encoding & ENCODING_TYPE_FIXMAP_MASK;
}

inline constexpr std::size_t GetFixextSize(EncodingType encoding) {
  switch (encoding) {
    case ENCODING_TYPE_FIXEXT1:
      return 1;
    case ENCODING_TYPE_FIXEXT2:
      return 2;
    case ENCODING_TYPE_FIXEXT4:
      return 4;
    case ENCODING_TYPE_FIXEXT8:
      return 8;
    case ENCODING_TYPE_FIXEXT16:
      return 16;
    default:
      return 0;  // Invalid fixext size.
  }
}

// Gets the size of the encoding in bytes, not including external payload data.
inline constexpr std::size_t GetEncodingSize(EncodingType encoding) {
  switch (encoding) {
    // Encoding is fully contained within the type value.
    case ENCODING_TYPE_POSITIVE_FIXINT_MIN ... ENCODING_TYPE_POSITIVE_FIXINT_MAX:
    case ENCODING_TYPE_NEGATIVE_FIXINT_MIN ... ENCODING_TYPE_NEGATIVE_FIXINT_MAX:
    case ENCODING_TYPE_FIXMAP_MIN ... ENCODING_TYPE_FIXMAP_MAX:
    case ENCODING_TYPE_FIXARRAY_MIN ... ENCODING_TYPE_FIXARRAY_MAX:
    case ENCODING_TYPE_FIXSTR_MIN ... ENCODING_TYPE_FIXSTR_MAX:
    case ENCODING_TYPE_NIL:
    case ENCODING_TYPE_RESERVED:
    case ENCODING_TYPE_FALSE:
    case ENCODING_TYPE_TRUE:
      return 1;

    // Encoding type followed by one-byte size or immediate value.
    case ENCODING_TYPE_BIN8:
    case ENCODING_TYPE_EXT8:
    case ENCODING_TYPE_UINT8:
    case ENCODING_TYPE_INT8:
    case ENCODING_TYPE_STR8:
    // Encoding type followed by one-byte extension type.
    case ENCODING_TYPE_FIXEXT1:
    case ENCODING_TYPE_FIXEXT2:
    case ENCODING_TYPE_FIXEXT4:
    case ENCODING_TYPE_FIXEXT8:
    case ENCODING_TYPE_FIXEXT16:
      return 2;

    // Encoding type followed by two-byte size or immediate value.
    case ENCODING_TYPE_BIN16:
    case ENCODING_TYPE_EXT16:
    case ENCODING_TYPE_UINT16:
    case ENCODING_TYPE_INT16:
    case ENCODING_TYPE_STR16:
    case ENCODING_TYPE_ARRAY16:
    case ENCODING_TYPE_MAP16:
      return 3;

    // Encoding type followed by four-byte size or immediate value.
    case ENCODING_TYPE_BIN32:
    case ENCODING_TYPE_EXT32:
    case ENCODING_TYPE_FLOAT32:
    case ENCODING_TYPE_UINT32:
    case ENCODING_TYPE_INT32:
    case ENCODING_TYPE_STR32:
    case ENCODING_TYPE_ARRAY32:
    case ENCODING_TYPE_MAP32:
      return 5;

    // Encoding type followed by eight-byte immediate value.
    case ENCODING_TYPE_FLOAT64:
    case ENCODING_TYPE_UINT64:
    case ENCODING_TYPE_INT64:
      return 9;

    default:
      return 0;
  }
}

// Encoding for standard types. Each supported data type has an associated
// encoding or set of encodings. These functions determine the MessagePack
// encoding based on the data type, value, and size of their arguments.

inline constexpr EncodingType EncodeArrayType(std::size_t size) {
  if (size < (1U << 4))
    return ENCODING_TYPE_FIXARRAY | (size & ENCODING_TYPE_FIXARRAY_MASK);
  else if (size < (1U << 16))
    return ENCODING_TYPE_ARRAY16;
  else
    return ENCODING_TYPE_ARRAY32;
}

inline constexpr EncodingType EncodeMapType(std::size_t size) {
  if (size < (1U << 4))
    return ENCODING_TYPE_FIXMAP | (size & ENCODING_TYPE_FIXMAP_MASK);
  else if (size < (1U << 16))
    return ENCODING_TYPE_MAP16;
  else
    return ENCODING_TYPE_MAP32;
}

inline constexpr EncodingType EncodeStringType(std::size_t size) {
  if (size < (1U << 5))
    return ENCODING_TYPE_FIXSTR | (size & ENCODING_TYPE_FIXSTR_MASK);
  else if (size < (1U << 8))
    return ENCODING_TYPE_STR8;
  else if (size < (1U << 16))
    return ENCODING_TYPE_STR16;
  else
    return ENCODING_TYPE_STR32;
}

inline constexpr EncodingType EncodeBinType(std::size_t size) {
  if (size < (1U << 8))
    return ENCODING_TYPE_BIN8;
  else if (size < (1U << 16))
    return ENCODING_TYPE_BIN16;
  else
    return ENCODING_TYPE_BIN32;
}

inline EncodingType EncodeType(const EmptyVariant& /*empty*/) {
  return ENCODING_TYPE_NIL;
}

// Variant is encoded as a single-element map, with the type index as the key.
template <typename... Types>
inline EncodingType EncodeType(const Variant<Types...>& /*variant*/) {
  return EncodeMapType(1);
}

template <typename T>
inline constexpr EncodingType EncodeType(const StringWrapper<T>& value) {
  return EncodeStringType(value.length());
}

inline constexpr EncodingType EncodeType(const std::string& value) {
  return EncodeStringType(value.length());
}

template <typename T, std::size_t Size>
inline constexpr EncodingType EncodeType(const std::array<T, Size>& /*value*/) {
  return EncodeArrayType(Size);
}

template <typename T>
inline constexpr EncodingType EncodeType(const ArrayWrapper<T>& value) {
  return EncodeArrayType(value.size());
}

template <typename T, typename Allocator>
inline constexpr EncodingType EncodeType(
    const std::vector<T, Allocator>& value) {
  return EncodeArrayType(value.size());
}

template <typename Key, typename T, typename Compare, typename Allocator>
inline constexpr EncodingType EncodeType(
    const std::map<Key, T, Compare, Allocator>& value) {
  return EncodeMapType(value.size());
}

template <typename Key, typename T, typename Hash, typename KeyEqual,
          typename Allocator>
inline constexpr EncodingType EncodeType(
    const std::unordered_map<Key, T, Hash, KeyEqual, Allocator>& value) {
  return EncodeMapType(value.size());
}

template <typename T>
inline constexpr EncodingType EncodeType(const BufferWrapper<T>& value) {
  // BIN size is in bytes.
  return EncodeBinType(value.size() *
                       sizeof(typename BufferWrapper<T>::value_type));
}

template <typename T, typename U>
inline constexpr EncodingType EncodeType(const std::pair<T, U>& /*value*/) {
  return EncodeArrayType(2);
}

template <typename... T>
inline constexpr EncodingType EncodeType(const std::tuple<T...>& /*value*/) {
  return EncodeArrayType(sizeof...(T));
}

// FileHandle is encoded as a FIXEXT2 with a type code for "FileDescriptor"
// and a signed 16-bit index into the pushed fd array. Empty file descriptor
// have an array index of -1.
template <FileHandleMode Mode>
inline constexpr EncodingType EncodeType(const FileHandle<Mode>& /*fd*/) {
  return ENCODING_TYPE_FIXEXT2;
}

// ChannelHandle is encoded as a FIXEXT4 with a type of
// ENCODING_EXT_TYPE_CHANNEL_HANDLE and a signed 32-bit value representing
// a client channel in a remote process. Empty handle has a value of -1.
template <ChannelHandleMode Mode>
inline constexpr EncodingType EncodeType(
    const ChannelHandle<Mode>& /*handle*/) {
  return ENCODING_TYPE_FIXEXT4;
}

inline constexpr EncodingType EncodeType(const bool& value) {
  return value ? ENCODING_TYPE_TRUE : ENCODING_TYPE_FALSE;
}

// Type 'char' is a little bit special in that it is distinct from 'signed char'
// and 'unsigned char'. Treating it as an unsigned 8-bit value is safe for
// encoding purposes and nicely handles 7-bit ASCII encodings as FIXINT.
inline constexpr EncodingType EncodeType(const char& value) {
  if (value < static_cast<char>(1 << 7))
    return value;
  else
    return ENCODING_TYPE_UINT8;
}

inline constexpr EncodingType EncodeType(const uint8_t& value) {
  if (value < (1U << 7))
    return value;
  else
    return ENCODING_TYPE_UINT8;
}
inline constexpr EncodingType EncodeType(const int8_t& value) {
  if (value >= -32)
    return value;
  else
    return ENCODING_TYPE_INT8;
}
inline constexpr EncodingType EncodeType(const uint16_t& value) {
  if (value < (1U << 7))
    return static_cast<EncodingType>(value);
  else if (value < (1U << 8))
    return ENCODING_TYPE_UINT8;
  else
    return ENCODING_TYPE_UINT16;
}
inline constexpr EncodingType EncodeType(const int16_t& value) {
  if (value >= -32 && value <= 127)
    return static_cast<EncodingType>(value);
  else if (value >= -128 && value <= 127)
    return ENCODING_TYPE_INT8;
  else
    return ENCODING_TYPE_INT16;
}
inline constexpr EncodingType EncodeType(const uint32_t& value) {
  if (value < (1U << 7))
    return static_cast<EncodingType>(value);
  else if (value < (1U << 8))
    return ENCODING_TYPE_UINT8;
  else if (value < (1U << 16))
    return ENCODING_TYPE_UINT16;
  else
    return ENCODING_TYPE_UINT32;
}
inline constexpr EncodingType EncodeType(const int32_t& value) {
  if (value >= -32 && value <= 127)
    return static_cast<EncodingType>(value);
  else if (value >= -128 && value <= 127)
    return ENCODING_TYPE_INT8;
  else if (value >= -32768 && value <= 32767)
    return ENCODING_TYPE_INT16;
  else
    return ENCODING_TYPE_INT32;
}
inline constexpr EncodingType EncodeType(const uint64_t& value) {
  if (value < (1ULL << 7))
    return static_cast<EncodingType>(value);
  else if (value < (1ULL << 8))
    return ENCODING_TYPE_UINT8;
  else if (value < (1ULL << 16))
    return ENCODING_TYPE_UINT16;
  else if (value < (1ULL << 32))
    return ENCODING_TYPE_UINT32;
  else
    return ENCODING_TYPE_UINT64;
}
inline constexpr EncodingType EncodeType(const int64_t& value) {
  if (value >= -32 && value <= 127)
    return static_cast<EncodingType>(value);
  else if (value >= -128 && value <= 127)  // Effectively [-128, -32).
    return ENCODING_TYPE_INT8;
  else if (value >= -32768 && value <= 32767)
    return ENCODING_TYPE_INT16;
  else if (value >= -2147483648 && value <= 2147483647)
    return ENCODING_TYPE_INT32;
  else
    return ENCODING_TYPE_INT64;
}

inline constexpr EncodingType EncodeType(const float& /*value*/) {
  return ENCODING_TYPE_FLOAT32;
}

inline constexpr EncodingType EncodeType(const double& /*value*/) {
  return ENCODING_TYPE_FLOAT64;
}

}  // namespace rpc
}  // namespace pdx
}  // namespace android

#endif  // ANDROID_PDX_RPC_ENCODING_H_