mirror of
https://github.com/Mauler125/r5sdk.git
synced 2025-02-09 19:15:03 +01:00
352 lines
13 KiB
C++
352 lines
13 KiB
C++
// Protocol Buffers - Google's data interchange format
|
|
// Copyright 2008 Google Inc. All rights reserved.
|
|
// https://developers.google.com/protocol-buffers/
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are
|
|
// met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following disclaimer
|
|
// in the documentation and/or other materials provided with the
|
|
// distribution.
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
// contributors may be used to endorse or promote products derived from
|
|
// this software without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
#ifndef GOOGLE_PROTOBUF_GENERATED_MESSAGE_TABLE_DRIVEN_H__
|
|
#define GOOGLE_PROTOBUF_GENERATED_MESSAGE_TABLE_DRIVEN_H__
|
|
|
|
#include <thirdparty/protobuf/map.h>
|
|
#include <thirdparty/protobuf/map_entry_lite.h>
|
|
#include <thirdparty/protobuf/map_field_lite.h>
|
|
#include <thirdparty/protobuf/message_lite.h>
|
|
#include <thirdparty/protobuf/wire_format_lite.h>
|
|
|
|
// We require C++11 and Clang to use constexpr for variables, as GCC 4.8
|
|
// requires constexpr to be consistent between declarations of variables
|
|
// unnecessarily (see https://gcc.gnu.org/bugzilla/show_bug.cgi?id=58541).
|
|
// VS 2017 Update 3 also supports this usage of constexpr.
|
|
#if defined(__clang__) || (defined(_MSC_VER) && _MSC_VER >= 1911)
|
|
#define PROTOBUF_CONSTEXPR_VAR constexpr
|
|
#else // !__clang__
|
|
#define PROTOBUF_CONSTEXPR_VAR
|
|
#endif // !_clang
|
|
|
|
#ifdef SWIG
|
|
#error "You cannot SWIG proto headers"
|
|
#endif
|
|
|
|
#include <thirdparty/protobuf/port_def.inc>
|
|
|
|
namespace google {
|
|
namespace protobuf {
|
|
namespace internal {
|
|
|
|
// Processing-type masks.
|
|
static constexpr const unsigned char kOneofMask = 0x40;
|
|
static constexpr const unsigned char kRepeatedMask = 0x20;
|
|
// Mask for the raw type: either a WireFormatLite::FieldType or one of the
|
|
// ProcessingTypes below, without the oneof or repeated flag.
|
|
static constexpr const unsigned char kTypeMask = 0x1f;
|
|
|
|
// Wire type masks.
|
|
static constexpr const unsigned char kNotPackedMask = 0x10;
|
|
static constexpr const unsigned char kInvalidMask = 0x20;
|
|
|
|
enum ProcessingTypes {
|
|
TYPE_STRING_CORD = 19,
|
|
TYPE_STRING_STRING_PIECE = 20,
|
|
TYPE_BYTES_CORD = 21,
|
|
TYPE_BYTES_STRING_PIECE = 22,
|
|
TYPE_STRING_INLINED = 23,
|
|
TYPE_BYTES_INLINED = 24,
|
|
TYPE_MAP = 25,
|
|
};
|
|
|
|
static_assert(TYPE_MAP < kRepeatedMask, "Invalid enum");
|
|
|
|
struct PROTOBUF_EXPORT FieldMetadata {
|
|
uint32_t offset; // offset of this field in the struct
|
|
uint32_t tag; // field * 8 + wire_type
|
|
// byte offset * 8 + bit_offset;
|
|
// if the high bit is set then this is the byte offset of the oneof_case
|
|
// for this field.
|
|
uint32_t has_offset;
|
|
uint32_t type; // the type of this field.
|
|
const void* ptr; // auxiliary data
|
|
|
|
// From the serializer point of view each fundamental type can occur in
|
|
// 4 different ways. For simplicity we treat all combinations as a cartesion
|
|
// product although not all combinations are allowed.
|
|
enum FieldTypeClass {
|
|
kPresence,
|
|
kNoPresence,
|
|
kRepeated,
|
|
kPacked,
|
|
kOneOf,
|
|
kNumTypeClasses // must be last enum
|
|
};
|
|
// C++ protobuf has 20 fundamental types, were we added Cord and StringPiece
|
|
// and also distinguish the same types if they have different wire format.
|
|
enum {
|
|
kCordType = 19,
|
|
kStringPieceType = 20,
|
|
kInlinedType = 21,
|
|
kNumTypes = 21,
|
|
kSpecial = kNumTypes * kNumTypeClasses,
|
|
};
|
|
|
|
static int CalculateType(int fundamental_type, FieldTypeClass type_class);
|
|
};
|
|
|
|
// TODO(ckennelly): Add a static assertion to ensure that these masks do not
|
|
// conflict with wiretypes.
|
|
|
|
// ParseTableField is kept small to help simplify instructions for computing
|
|
// offsets, as we will always need this information to parse a field.
|
|
// Additional data, needed for some types, is stored in
|
|
// AuxiliaryParseTableField.
|
|
struct ParseTableField {
|
|
uint32_t offset;
|
|
// The presence_index ordinarily represents a has_bit index, but for fields
|
|
// inside a oneof it represents the index in _oneof_case_.
|
|
uint32_t presence_index;
|
|
unsigned char normal_wiretype;
|
|
unsigned char packed_wiretype;
|
|
|
|
// processing_type is given by:
|
|
// (FieldDescriptor->type() << 1) | FieldDescriptor->is_packed()
|
|
unsigned char processing_type;
|
|
|
|
unsigned char tag_size;
|
|
};
|
|
|
|
struct ParseTable;
|
|
|
|
union AuxiliaryParseTableField {
|
|
typedef bool (*EnumValidator)(int);
|
|
|
|
// Enums
|
|
struct enum_aux {
|
|
EnumValidator validator;
|
|
};
|
|
enum_aux enums;
|
|
// Group, messages
|
|
struct message_aux {
|
|
// ExplicitlyInitialized<T> -> T requires a reinterpret_cast, which prevents
|
|
// the tables from being constructed as a constexpr. We use void to avoid
|
|
// the cast.
|
|
const void* default_message_void;
|
|
const MessageLite* default_message() const {
|
|
return static_cast<const MessageLite*>(default_message_void);
|
|
}
|
|
};
|
|
message_aux messages;
|
|
// Strings
|
|
struct string_aux {
|
|
const void* default_ptr;
|
|
const char* field_name;
|
|
};
|
|
string_aux strings;
|
|
|
|
struct map_aux {
|
|
bool (*parse_map)(io::CodedInputStream*, void*);
|
|
};
|
|
map_aux maps;
|
|
|
|
AuxiliaryParseTableField() = default;
|
|
constexpr AuxiliaryParseTableField(AuxiliaryParseTableField::enum_aux e)
|
|
: enums(e) {}
|
|
constexpr AuxiliaryParseTableField(AuxiliaryParseTableField::message_aux m)
|
|
: messages(m) {}
|
|
constexpr AuxiliaryParseTableField(AuxiliaryParseTableField::string_aux s)
|
|
: strings(s) {}
|
|
constexpr AuxiliaryParseTableField(AuxiliaryParseTableField::map_aux m)
|
|
: maps(m) {}
|
|
};
|
|
|
|
struct ParseTable {
|
|
const ParseTableField* fields;
|
|
const AuxiliaryParseTableField* aux;
|
|
int max_field_number;
|
|
// TODO(ckennelly): Do something with this padding.
|
|
|
|
// TODO(ckennelly): Vet these for sign extension.
|
|
int64_t has_bits_offset;
|
|
int64_t oneof_case_offset;
|
|
int64_t extension_offset;
|
|
int64_t arena_offset;
|
|
|
|
// ExplicitlyInitialized<T> -> T requires a reinterpret_cast, which prevents
|
|
// the tables from being constructed as a constexpr. We use void to avoid
|
|
// the cast.
|
|
const void* default_instance_void;
|
|
const MessageLite* default_instance() const {
|
|
return static_cast<const MessageLite*>(default_instance_void);
|
|
}
|
|
|
|
bool unknown_field_set;
|
|
};
|
|
|
|
static_assert(sizeof(ParseTableField) <= 16, "ParseTableField is too large");
|
|
// The tables must be composed of POD components to ensure link-time
|
|
// initialization.
|
|
static_assert(std::is_standard_layout<ParseTableField>::value, "");
|
|
static_assert(std::is_trivial<ParseTableField>::value, "");
|
|
static_assert(std::is_standard_layout<AuxiliaryParseTableField>::value, "");
|
|
static_assert(std::is_trivial<AuxiliaryParseTableField>::value, "");
|
|
static_assert(
|
|
std::is_standard_layout<AuxiliaryParseTableField::enum_aux>::value, "");
|
|
static_assert(std::is_trivial<AuxiliaryParseTableField::enum_aux>::value, "");
|
|
static_assert(
|
|
std::is_standard_layout<AuxiliaryParseTableField::message_aux>::value, "");
|
|
static_assert(std::is_trivial<AuxiliaryParseTableField::message_aux>::value,
|
|
"");
|
|
static_assert(
|
|
std::is_standard_layout<AuxiliaryParseTableField::string_aux>::value, "");
|
|
static_assert(std::is_trivial<AuxiliaryParseTableField::string_aux>::value, "");
|
|
static_assert(std::is_standard_layout<ParseTable>::value, "");
|
|
static_assert(std::is_trivial<ParseTable>::value, "");
|
|
|
|
// TODO(ckennelly): Consolidate these implementations into a single one, using
|
|
// dynamic dispatch to the appropriate unknown field handler.
|
|
bool MergePartialFromCodedStream(MessageLite* msg, const ParseTable& table,
|
|
io::CodedInputStream* input);
|
|
bool MergePartialFromCodedStreamLite(MessageLite* msg, const ParseTable& table,
|
|
io::CodedInputStream* input);
|
|
|
|
template <typename Entry>
|
|
bool ParseMap(io::CodedInputStream* input, void* map_field) {
|
|
typedef typename MapEntryToMapField<Entry>::MapFieldType MapFieldType;
|
|
typedef Map<typename Entry::EntryKeyType, typename Entry::EntryValueType>
|
|
MapType;
|
|
typedef typename Entry::template Parser<MapFieldType, MapType> ParserType;
|
|
|
|
ParserType parser(static_cast<MapFieldType*>(map_field));
|
|
return WireFormatLite::ReadMessageNoVirtual(input, &parser);
|
|
}
|
|
|
|
struct SerializationTable {
|
|
int num_fields;
|
|
const FieldMetadata* field_table;
|
|
};
|
|
|
|
PROTOBUF_EXPORT void SerializeInternal(const uint8_t* base,
|
|
const FieldMetadata* table,
|
|
int32_t num_fields,
|
|
io::CodedOutputStream* output);
|
|
|
|
inline void TableSerialize(const MessageLite& msg,
|
|
const SerializationTable* table,
|
|
io::CodedOutputStream* output) {
|
|
const FieldMetadata* field_table = table->field_table;
|
|
int num_fields = table->num_fields - 1;
|
|
const uint8_t* base = reinterpret_cast<const uint8_t*>(&msg);
|
|
// TODO(gerbens) This skips the first test if we could use the fast
|
|
// array serialization path, we should make this
|
|
// int cached_size =
|
|
// *reinterpret_cast<const int32_t*>(base + field_table->offset);
|
|
// SerializeWithCachedSize(msg, field_table + 1, num_fields, cached_size, ...)
|
|
// But we keep conformance with the old way for now.
|
|
SerializeInternal(base, field_table + 1, num_fields, output);
|
|
}
|
|
|
|
PROTOBUF_EXPORT uint8_t* SerializeInternalToArray(const uint8_t* base,
|
|
const FieldMetadata* table,
|
|
int32_t num_fields,
|
|
bool is_deterministic,
|
|
uint8_t* buffer);
|
|
|
|
inline uint8_t* TableSerializeToArray(const MessageLite& msg,
|
|
const SerializationTable* table,
|
|
bool is_deterministic, uint8_t* buffer) {
|
|
const uint8_t* base = reinterpret_cast<const uint8_t*>(&msg);
|
|
const FieldMetadata* field_table = table->field_table + 1;
|
|
int num_fields = table->num_fields - 1;
|
|
return SerializeInternalToArray(base, field_table, num_fields,
|
|
is_deterministic, buffer);
|
|
}
|
|
|
|
template <typename T>
|
|
struct CompareHelper {
|
|
bool operator()(const T& a, const T& b) const { return a < b; }
|
|
};
|
|
|
|
template <>
|
|
struct CompareHelper<ArenaStringPtr> {
|
|
bool operator()(const ArenaStringPtr& a, const ArenaStringPtr& b) const {
|
|
return a.Get() < b.Get();
|
|
}
|
|
};
|
|
|
|
struct CompareMapKey {
|
|
template <typename T>
|
|
bool operator()(const MapEntryHelper<T>& a,
|
|
const MapEntryHelper<T>& b) const {
|
|
return Compare(a.key_, b.key_);
|
|
}
|
|
template <typename T>
|
|
bool Compare(const T& a, const T& b) const {
|
|
return CompareHelper<T>()(a, b);
|
|
}
|
|
};
|
|
|
|
template <typename MapFieldType, const SerializationTable* table>
|
|
void MapFieldSerializer(const uint8_t* base, uint32_t offset, uint32_t tag,
|
|
uint32_t has_offset, io::CodedOutputStream* output) {
|
|
typedef MapEntryHelper<typename MapFieldType::EntryTypeTrait> Entry;
|
|
typedef typename MapFieldType::MapType::const_iterator Iter;
|
|
|
|
const MapFieldType& map_field =
|
|
*reinterpret_cast<const MapFieldType*>(base + offset);
|
|
const SerializationTable* t =
|
|
table +
|
|
has_offset; // has_offset is overloaded for maps to mean table offset
|
|
if (!output->IsSerializationDeterministic()) {
|
|
for (Iter it = map_field.GetMap().begin(); it != map_field.GetMap().end();
|
|
++it) {
|
|
Entry map_entry(*it);
|
|
output->WriteVarint32(tag);
|
|
output->WriteVarint32(map_entry._cached_size_);
|
|
SerializeInternal(reinterpret_cast<const uint8_t*>(&map_entry),
|
|
t->field_table, t->num_fields, output);
|
|
}
|
|
} else {
|
|
std::vector<Entry> v;
|
|
for (Iter it = map_field.GetMap().begin(); it != map_field.GetMap().end();
|
|
++it) {
|
|
v.push_back(Entry(*it));
|
|
}
|
|
std::sort(v.begin(), v.end(), CompareMapKey());
|
|
for (int i = 0; i < v.size(); i++) {
|
|
output->WriteVarint32(tag);
|
|
output->WriteVarint32(v[i]._cached_size_);
|
|
SerializeInternal(reinterpret_cast<const uint8_t*>(&v[i]), t->field_table,
|
|
t->num_fields, output);
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace internal
|
|
} // namespace protobuf
|
|
} // namespace google
|
|
|
|
#include <thirdparty/protobuf/port_undef.inc>
|
|
|
|
#endif // GOOGLE_PROTOBUF_GENERATED_MESSAGE_TABLE_DRIVEN_H__
|