mirror of
https://github.com/google/flatbuffers.git
synced 2026-06-11 07:27:27 +00:00
Union Vector
This commit is contained in:
@@ -16,7 +16,7 @@
|
||||
|
||||
../flatc --cpp --java --csharp --go --binary --python --js --php --grpc --gen-mutable --gen-object-api --no-includes monster_test.fbs monsterdata_test.json
|
||||
../flatc --cpp --java --csharp --go --binary --python --js --php --gen-mutable -o namespace_test namespace_test/namespace_test1.fbs namespace_test/namespace_test2.fbs
|
||||
../flatc --binary --schema monster_test.fbs
|
||||
../flatc --cpp -o union_vector ./union_vector/union_vector.fbs
|
||||
cd ../samples
|
||||
../flatc --cpp --gen-mutable --gen-object-api monster.fbs
|
||||
cd ../reflection
|
||||
|
||||
@@ -136,6 +136,7 @@ struct AnyUnion {
|
||||
};
|
||||
|
||||
bool VerifyAny(flatbuffers::Verifier &verifier, const void *obj, Any type);
|
||||
bool VerifyAnyVector(flatbuffers::Verifier &verifier, const flatbuffers::Vector<flatbuffers::Offset<void>> *values, const flatbuffers::Vector<uint8_t> *types);
|
||||
|
||||
MANUALLY_ALIGNED_STRUCT(2) Test FLATBUFFERS_FINAL_CLASS {
|
||||
private:
|
||||
@@ -1198,6 +1199,17 @@ inline bool VerifyAny(flatbuffers::Verifier &verifier, const void *obj, Any type
|
||||
}
|
||||
}
|
||||
|
||||
inline bool VerifyAnyVector(flatbuffers::Verifier &verifier, const flatbuffers::Vector<flatbuffers::Offset<void>> *values, const flatbuffers::Vector<uint8_t> *types) {
|
||||
if (values->size() != types->size()) return false;
|
||||
for (flatbuffers::uoffset_t i = 0; i < values->size(); ++i) {
|
||||
if (!VerifyAny(
|
||||
verifier, values->Get(i), types->GetEnum<Any>(i))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
inline flatbuffers::NativeTable *AnyUnion::UnPack(const void *obj, Any type, const flatbuffers::resolver_function_t *resolver) {
|
||||
switch (type) {
|
||||
case Any_Monster: {
|
||||
|
||||
@@ -21,6 +21,7 @@
|
||||
#include "monster_test_generated.h"
|
||||
#include "namespace_test/namespace_test1_generated.h"
|
||||
#include "namespace_test/namespace_test2_generated.h"
|
||||
#include "union_vector/union_vector_generated.h"
|
||||
|
||||
#ifndef FLATBUFFERS_CPP98_STL
|
||||
#include <random>
|
||||
@@ -954,7 +955,6 @@ void ErrorTest() {
|
||||
TestError("@", "illegal");
|
||||
TestError("table 1", "expecting");
|
||||
TestError("table X { Y:[[int]]; }", "nested vector");
|
||||
TestError("union Z { X } table X { Y:[Z]; }", "vector of union");
|
||||
TestError("table X { Y:1; }", "illegal type");
|
||||
TestError("table X { Y:int; Y:int; }", "field already");
|
||||
TestError("struct X { Y:string; }", "only scalar");
|
||||
@@ -1276,6 +1276,67 @@ void ParseUnionTest() {
|
||||
"{ e_type: N_A, e: {} }"), true);
|
||||
}
|
||||
|
||||
void UnionVectorTest() {
|
||||
// load FlatBuffer fbs schema.
|
||||
// TODO: load a JSON file with such a vector when JSON support is ready.
|
||||
std::string schemafile;
|
||||
TEST_EQ(flatbuffers::LoadFile(
|
||||
"tests/union_vector/union_vector.fbs", false, &schemafile), true);
|
||||
|
||||
// parse schema.
|
||||
flatbuffers::IDLOptions idl_opts;
|
||||
idl_opts.lang_to_generate |= flatbuffers::IDLOptions::kCpp;
|
||||
flatbuffers::Parser parser(idl_opts);
|
||||
const char *include_directories[] = { "tests/union_vector", nullptr };
|
||||
TEST_EQ(parser.Parse(schemafile.c_str(), include_directories), true);
|
||||
|
||||
flatbuffers::FlatBufferBuilder fbb;
|
||||
|
||||
// union types.
|
||||
std::vector<uint8_t> types;
|
||||
types.push_back(static_cast<uint8_t>(Character_Belle));
|
||||
types.push_back(static_cast<uint8_t>(Character_Rapunzel));
|
||||
types.push_back(static_cast<uint8_t>(Character_MuLan));
|
||||
|
||||
// union values.
|
||||
std::vector<flatbuffers::Offset<void>> characters;
|
||||
characters.push_back(CreateBelle(fbb, /*books_read=*/7).Union());
|
||||
characters.push_back(CreateRapunzel(fbb, /*hair_length=*/6).Union());
|
||||
characters.push_back(CreateMuLan(fbb, /*sward_attack_damage=*/5).Union());
|
||||
|
||||
// create Movie.
|
||||
const auto movie_offset =
|
||||
CreateMovie(fbb, fbb.CreateVector(types), fbb.CreateVector(characters));
|
||||
FinishMovieBuffer(fbb, movie_offset);
|
||||
uint8_t *buf = fbb.GetBufferPointer();
|
||||
|
||||
flatbuffers::Verifier verifier(buf, fbb.GetSize());
|
||||
TEST_EQ(VerifyMovieBuffer(verifier), true);
|
||||
|
||||
const Movie *movie = GetMovie(buf);
|
||||
TEST_EQ(movie->characters_type()->size(), 3);
|
||||
TEST_EQ(
|
||||
movie->characters_type()->GetEnum<Character>(0) == Character_Belle,
|
||||
true);
|
||||
TEST_EQ(
|
||||
movie->characters_type()->GetEnum<Character>(1) == Character_Rapunzel,
|
||||
true);
|
||||
TEST_EQ(
|
||||
movie->characters_type()->GetEnum<Character>(2) == Character_MuLan,
|
||||
true);
|
||||
|
||||
TEST_EQ(movie->characters()->size(), 3);
|
||||
const Belle *belle =
|
||||
reinterpret_cast<const Belle*>(movie->characters()->Get(0));
|
||||
TEST_EQ(belle->books_read(), 7);
|
||||
const Rapunzel *rapunzel =
|
||||
reinterpret_cast<const Rapunzel*>(movie->characters()->Get(1));
|
||||
TEST_EQ(rapunzel->hair_length(), 6);
|
||||
const MuLan *mu_lan =
|
||||
reinterpret_cast<const MuLan*>(movie->characters()->Get(2));
|
||||
TEST_EQ(mu_lan->sword_attack_damage(), 5);
|
||||
}
|
||||
|
||||
void ConformTest() {
|
||||
flatbuffers::Parser parser;
|
||||
TEST_EQ(parser.Parse("table T { A:int; } enum E:byte { A }"), true);
|
||||
@@ -1330,6 +1391,7 @@ int main(int /*argc*/, const char * /*argv*/[]) {
|
||||
InvalidUTF8Test();
|
||||
UnknownFieldsTest();
|
||||
ParseUnionTest();
|
||||
UnionVectorTest();
|
||||
ConformTest();
|
||||
|
||||
if (!testing_fails) {
|
||||
|
||||
24
tests/union_vector/union_vector.fbs
Normal file
24
tests/union_vector/union_vector.fbs
Normal file
@@ -0,0 +1,24 @@
|
||||
table MuLan {
|
||||
sword_attack_damage: int;
|
||||
}
|
||||
|
||||
table Rapunzel {
|
||||
hair_length: int;
|
||||
}
|
||||
|
||||
table Belle {
|
||||
books_read: int;
|
||||
}
|
||||
|
||||
union Character {
|
||||
MuLan,
|
||||
Rapunzel,
|
||||
Belle,
|
||||
}
|
||||
|
||||
table Movie {
|
||||
characters: [Character];
|
||||
}
|
||||
|
||||
root_type Movie;
|
||||
file_identifier "MOVI";
|
||||
300
tests/union_vector/union_vector_generated.h
Normal file
300
tests/union_vector/union_vector_generated.h
Normal file
@@ -0,0 +1,300 @@
|
||||
// automatically generated by the FlatBuffers compiler, do not modify
|
||||
|
||||
|
||||
#ifndef FLATBUFFERS_GENERATED_UNIONVECTOR_H_
|
||||
#define FLATBUFFERS_GENERATED_UNIONVECTOR_H_
|
||||
|
||||
#include "flatbuffers/flatbuffers.h"
|
||||
|
||||
struct MuLan;
|
||||
|
||||
struct Rapunzel;
|
||||
|
||||
struct Belle;
|
||||
|
||||
struct Movie;
|
||||
|
||||
enum Character {
|
||||
Character_NONE = 0,
|
||||
Character_MuLan = 1,
|
||||
Character_Rapunzel = 2,
|
||||
Character_Belle = 3,
|
||||
Character_MIN = Character_NONE,
|
||||
Character_MAX = Character_Belle
|
||||
};
|
||||
|
||||
inline const char **EnumNamesCharacter() {
|
||||
static const char *names[] = {
|
||||
"NONE",
|
||||
"MuLan",
|
||||
"Rapunzel",
|
||||
"Belle",
|
||||
nullptr
|
||||
};
|
||||
return names;
|
||||
}
|
||||
|
||||
inline const char *EnumNameCharacter(Character e) {
|
||||
const size_t index = static_cast<int>(e);
|
||||
return EnumNamesCharacter()[index];
|
||||
}
|
||||
|
||||
template<typename T> struct CharacterTraits {
|
||||
static const Character enum_value = Character_NONE;
|
||||
};
|
||||
|
||||
template<> struct CharacterTraits<MuLan> {
|
||||
static const Character enum_value = Character_MuLan;
|
||||
};
|
||||
|
||||
template<> struct CharacterTraits<Rapunzel> {
|
||||
static const Character enum_value = Character_Rapunzel;
|
||||
};
|
||||
|
||||
template<> struct CharacterTraits<Belle> {
|
||||
static const Character enum_value = Character_Belle;
|
||||
};
|
||||
|
||||
bool VerifyCharacter(flatbuffers::Verifier &verifier, const void *obj, Character type);
|
||||
bool VerifyCharacterVector(flatbuffers::Verifier &verifier, const flatbuffers::Vector<flatbuffers::Offset<void>> *values, const flatbuffers::Vector<uint8_t> *types);
|
||||
|
||||
struct MuLan FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||
enum {
|
||||
VT_SWORD_ATTACK_DAMAGE = 4
|
||||
};
|
||||
int32_t sword_attack_damage() const {
|
||||
return GetField<int32_t>(VT_SWORD_ATTACK_DAMAGE, 0);
|
||||
}
|
||||
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||
return VerifyTableStart(verifier) &&
|
||||
VerifyField<int32_t>(verifier, VT_SWORD_ATTACK_DAMAGE) &&
|
||||
verifier.EndTable();
|
||||
}
|
||||
};
|
||||
|
||||
struct MuLanBuilder {
|
||||
flatbuffers::FlatBufferBuilder &fbb_;
|
||||
flatbuffers::uoffset_t start_;
|
||||
void add_sword_attack_damage(int32_t sword_attack_damage) {
|
||||
fbb_.AddElement<int32_t>(MuLan::VT_SWORD_ATTACK_DAMAGE, sword_attack_damage, 0);
|
||||
}
|
||||
MuLanBuilder(flatbuffers::FlatBufferBuilder &_fbb)
|
||||
: fbb_(_fbb) {
|
||||
start_ = fbb_.StartTable();
|
||||
}
|
||||
MuLanBuilder &operator=(const MuLanBuilder &);
|
||||
flatbuffers::Offset<MuLan> Finish() {
|
||||
const auto end = fbb_.EndTable(start_, 1);
|
||||
auto o = flatbuffers::Offset<MuLan>(end);
|
||||
return o;
|
||||
}
|
||||
};
|
||||
|
||||
inline flatbuffers::Offset<MuLan> CreateMuLan(
|
||||
flatbuffers::FlatBufferBuilder &_fbb,
|
||||
int32_t sword_attack_damage = 0) {
|
||||
MuLanBuilder builder_(_fbb);
|
||||
builder_.add_sword_attack_damage(sword_attack_damage);
|
||||
return builder_.Finish();
|
||||
}
|
||||
|
||||
struct Rapunzel FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||
enum {
|
||||
VT_HAIR_LENGTH = 4
|
||||
};
|
||||
int32_t hair_length() const {
|
||||
return GetField<int32_t>(VT_HAIR_LENGTH, 0);
|
||||
}
|
||||
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||
return VerifyTableStart(verifier) &&
|
||||
VerifyField<int32_t>(verifier, VT_HAIR_LENGTH) &&
|
||||
verifier.EndTable();
|
||||
}
|
||||
};
|
||||
|
||||
struct RapunzelBuilder {
|
||||
flatbuffers::FlatBufferBuilder &fbb_;
|
||||
flatbuffers::uoffset_t start_;
|
||||
void add_hair_length(int32_t hair_length) {
|
||||
fbb_.AddElement<int32_t>(Rapunzel::VT_HAIR_LENGTH, hair_length, 0);
|
||||
}
|
||||
RapunzelBuilder(flatbuffers::FlatBufferBuilder &_fbb)
|
||||
: fbb_(_fbb) {
|
||||
start_ = fbb_.StartTable();
|
||||
}
|
||||
RapunzelBuilder &operator=(const RapunzelBuilder &);
|
||||
flatbuffers::Offset<Rapunzel> Finish() {
|
||||
const auto end = fbb_.EndTable(start_, 1);
|
||||
auto o = flatbuffers::Offset<Rapunzel>(end);
|
||||
return o;
|
||||
}
|
||||
};
|
||||
|
||||
inline flatbuffers::Offset<Rapunzel> CreateRapunzel(
|
||||
flatbuffers::FlatBufferBuilder &_fbb,
|
||||
int32_t hair_length = 0) {
|
||||
RapunzelBuilder builder_(_fbb);
|
||||
builder_.add_hair_length(hair_length);
|
||||
return builder_.Finish();
|
||||
}
|
||||
|
||||
struct Belle FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||
enum {
|
||||
VT_BOOKS_READ = 4
|
||||
};
|
||||
int32_t books_read() const {
|
||||
return GetField<int32_t>(VT_BOOKS_READ, 0);
|
||||
}
|
||||
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||
return VerifyTableStart(verifier) &&
|
||||
VerifyField<int32_t>(verifier, VT_BOOKS_READ) &&
|
||||
verifier.EndTable();
|
||||
}
|
||||
};
|
||||
|
||||
struct BelleBuilder {
|
||||
flatbuffers::FlatBufferBuilder &fbb_;
|
||||
flatbuffers::uoffset_t start_;
|
||||
void add_books_read(int32_t books_read) {
|
||||
fbb_.AddElement<int32_t>(Belle::VT_BOOKS_READ, books_read, 0);
|
||||
}
|
||||
BelleBuilder(flatbuffers::FlatBufferBuilder &_fbb)
|
||||
: fbb_(_fbb) {
|
||||
start_ = fbb_.StartTable();
|
||||
}
|
||||
BelleBuilder &operator=(const BelleBuilder &);
|
||||
flatbuffers::Offset<Belle> Finish() {
|
||||
const auto end = fbb_.EndTable(start_, 1);
|
||||
auto o = flatbuffers::Offset<Belle>(end);
|
||||
return o;
|
||||
}
|
||||
};
|
||||
|
||||
inline flatbuffers::Offset<Belle> CreateBelle(
|
||||
flatbuffers::FlatBufferBuilder &_fbb,
|
||||
int32_t books_read = 0) {
|
||||
BelleBuilder builder_(_fbb);
|
||||
builder_.add_books_read(books_read);
|
||||
return builder_.Finish();
|
||||
}
|
||||
|
||||
struct Movie FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
|
||||
enum {
|
||||
VT_CHARACTERS_TYPE = 4,
|
||||
VT_CHARACTERS = 6
|
||||
};
|
||||
const flatbuffers::Vector<uint8_t> *characters_type() const {
|
||||
return GetPointer<const flatbuffers::Vector<uint8_t> *>(VT_CHARACTERS_TYPE);
|
||||
}
|
||||
const flatbuffers::Vector<flatbuffers::Offset<void>> *characters() const {
|
||||
return GetPointer<const flatbuffers::Vector<flatbuffers::Offset<void>> *>(VT_CHARACTERS);
|
||||
}
|
||||
bool Verify(flatbuffers::Verifier &verifier) const {
|
||||
return VerifyTableStart(verifier) &&
|
||||
VerifyField<flatbuffers::uoffset_t>(verifier, VT_CHARACTERS_TYPE) &&
|
||||
verifier.Verify(characters_type()) &&
|
||||
VerifyField<flatbuffers::uoffset_t>(verifier, VT_CHARACTERS) &&
|
||||
verifier.Verify(characters()) &&
|
||||
VerifyCharacterVector(verifier, characters(), characters_type()) &&
|
||||
verifier.EndTable();
|
||||
}
|
||||
};
|
||||
|
||||
struct MovieBuilder {
|
||||
flatbuffers::FlatBufferBuilder &fbb_;
|
||||
flatbuffers::uoffset_t start_;
|
||||
void add_characters_type(flatbuffers::Offset<flatbuffers::Vector<uint8_t>> characters_type) {
|
||||
fbb_.AddOffset(Movie::VT_CHARACTERS_TYPE, characters_type);
|
||||
}
|
||||
void add_characters(flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<void>>> characters) {
|
||||
fbb_.AddOffset(Movie::VT_CHARACTERS, characters);
|
||||
}
|
||||
MovieBuilder(flatbuffers::FlatBufferBuilder &_fbb)
|
||||
: fbb_(_fbb) {
|
||||
start_ = fbb_.StartTable();
|
||||
}
|
||||
MovieBuilder &operator=(const MovieBuilder &);
|
||||
flatbuffers::Offset<Movie> Finish() {
|
||||
const auto end = fbb_.EndTable(start_, 2);
|
||||
auto o = flatbuffers::Offset<Movie>(end);
|
||||
return o;
|
||||
}
|
||||
};
|
||||
|
||||
inline flatbuffers::Offset<Movie> CreateMovie(
|
||||
flatbuffers::FlatBufferBuilder &_fbb,
|
||||
flatbuffers::Offset<flatbuffers::Vector<uint8_t>> characters_type = 0,
|
||||
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<void>>> characters = 0) {
|
||||
MovieBuilder builder_(_fbb);
|
||||
builder_.add_characters(characters);
|
||||
builder_.add_characters_type(characters_type);
|
||||
return builder_.Finish();
|
||||
}
|
||||
|
||||
inline flatbuffers::Offset<Movie> CreateMovieDirect(
|
||||
flatbuffers::FlatBufferBuilder &_fbb,
|
||||
const std::vector<uint8_t> *characters_type = nullptr,
|
||||
const std::vector<flatbuffers::Offset<void>> *characters = nullptr) {
|
||||
return CreateMovie(
|
||||
_fbb,
|
||||
characters_type ? _fbb.CreateVector<uint8_t>(*characters_type) : 0,
|
||||
characters ? _fbb.CreateVector<flatbuffers::Offset<void>>(*characters) : 0);
|
||||
}
|
||||
|
||||
inline bool VerifyCharacter(flatbuffers::Verifier &verifier, const void *obj, Character type) {
|
||||
switch (type) {
|
||||
case Character_NONE: {
|
||||
return true;
|
||||
}
|
||||
case Character_MuLan: {
|
||||
auto ptr = reinterpret_cast<const MuLan *>(obj);
|
||||
return verifier.VerifyTable(ptr);
|
||||
}
|
||||
case Character_Rapunzel: {
|
||||
auto ptr = reinterpret_cast<const Rapunzel *>(obj);
|
||||
return verifier.VerifyTable(ptr);
|
||||
}
|
||||
case Character_Belle: {
|
||||
auto ptr = reinterpret_cast<const Belle *>(obj);
|
||||
return verifier.VerifyTable(ptr);
|
||||
}
|
||||
default: return false;
|
||||
}
|
||||
}
|
||||
|
||||
inline bool VerifyCharacterVector(flatbuffers::Verifier &verifier, const flatbuffers::Vector<flatbuffers::Offset<void>> *values, const flatbuffers::Vector<uint8_t> *types) {
|
||||
if (values->size() != types->size()) return false;
|
||||
for (flatbuffers::uoffset_t i = 0; i < values->size(); ++i) {
|
||||
if (!VerifyCharacter(
|
||||
verifier, values->Get(i), types->GetEnum<Character>(i))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
inline const Movie *GetMovie(const void *buf) {
|
||||
return flatbuffers::GetRoot<Movie>(buf);
|
||||
}
|
||||
|
||||
inline const char *MovieIdentifier() {
|
||||
return "MOVI";
|
||||
}
|
||||
|
||||
inline bool MovieBufferHasIdentifier(const void *buf) {
|
||||
return flatbuffers::BufferHasIdentifier(
|
||||
buf, MovieIdentifier());
|
||||
}
|
||||
|
||||
inline bool VerifyMovieBuffer(
|
||||
flatbuffers::Verifier &verifier) {
|
||||
return verifier.VerifyBuffer<Movie>(MovieIdentifier());
|
||||
}
|
||||
|
||||
inline void FinishMovieBuffer(
|
||||
flatbuffers::FlatBufferBuilder &fbb,
|
||||
flatbuffers::Offset<Movie> root) {
|
||||
fbb.Finish(root, MovieIdentifier());
|
||||
}
|
||||
|
||||
#endif // FLATBUFFERS_GENERATED_UNIONVECTOR_H_
|
||||
Reference in New Issue
Block a user