Skip to content

Commit 7b06041

Browse files
author
Wouter van Oortmerssen
committed
C++ generated code now has constants for vtable offsets.
Besides making the generated code looking a lot more readable, it also allows you to use these offsets in calls to Table::CheckField, to see if a field is present in a table. Change-Id: I1b4cc350c4f27c4e474c31add40c701ef4ae63b2 Tested: On Linux.
1 parent 3fc5387 commit 7b06041

File tree

3 files changed

+203
-142
lines changed

3 files changed

+203
-142
lines changed

samples/monster_generated.h

Lines changed: 32 additions & 24 deletions
Original file line numberDiff line numberDiff line change
@@ -59,41 +59,49 @@ MANUALLY_ALIGNED_STRUCT(4) Vec3 FLATBUFFERS_FINAL_CLASS {
5959
STRUCT_END(Vec3, 12);
6060

6161
struct Monster FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table {
62-
const Vec3 *pos() const { return GetStruct<const Vec3 *>(4); }
63-
Vec3 *mutable_pos() { return GetStruct<Vec3 *>(4); }
64-
int16_t mana() const { return GetField<int16_t>(6, 150); }
65-
bool mutate_mana(int16_t _mana) { return SetField(6, _mana); }
66-
int16_t hp() const { return GetField<int16_t>(8, 100); }
67-
bool mutate_hp(int16_t _hp) { return SetField(8, _hp); }
68-
const flatbuffers::String *name() const { return GetPointer<const flatbuffers::String *>(10); }
69-
flatbuffers::String *mutable_name() { return GetPointer<flatbuffers::String *>(10); }
70-
const flatbuffers::Vector<uint8_t> *inventory() const { return GetPointer<const flatbuffers::Vector<uint8_t> *>(14); }
71-
flatbuffers::Vector<uint8_t> *mutable_inventory() { return GetPointer<flatbuffers::Vector<uint8_t> *>(14); }
72-
Color color() const { return static_cast<Color>(GetField<int8_t>(16, 2)); }
73-
bool mutate_color(Color _color) { return SetField(16, static_cast<int8_t>(_color)); }
62+
enum {
63+
VT_POS = 4,
64+
VT_MANA = 6,
65+
VT_HP = 8,
66+
VT_NAME = 10,
67+
VT_INVENTORY = 14,
68+
VT_COLOR = 16,
69+
};
70+
const Vec3 *pos() const { return GetStruct<const Vec3 *>(VT_POS); }
71+
Vec3 *mutable_pos() { return GetStruct<Vec3 *>(VT_POS); }
72+
int16_t mana() const { return GetField<int16_t>(VT_MANA, 150); }
73+
bool mutate_mana(int16_t _mana) { return SetField(VT_MANA, _mana); }
74+
int16_t hp() const { return GetField<int16_t>(VT_HP, 100); }
75+
bool mutate_hp(int16_t _hp) { return SetField(VT_HP, _hp); }
76+
const flatbuffers::String *name() const { return GetPointer<const flatbuffers::String *>(VT_NAME); }
77+
flatbuffers::String *mutable_name() { return GetPointer<flatbuffers::String *>(VT_NAME); }
78+
const flatbuffers::Vector<uint8_t> *inventory() const { return GetPointer<const flatbuffers::Vector<uint8_t> *>(VT_INVENTORY); }
79+
flatbuffers::Vector<uint8_t> *mutable_inventory() { return GetPointer<flatbuffers::Vector<uint8_t> *>(VT_INVENTORY); }
80+
Color color() const { return static_cast<Color>(GetField<int8_t>(VT_COLOR, 2)); }
81+
bool mutate_color(Color _color) { return SetField(VT_COLOR, static_cast<int8_t>(_color)); }
7482
bool Verify(flatbuffers::Verifier &verifier) const {
7583
return VerifyTableStart(verifier) &&
76-
VerifyField<Vec3>(verifier, 4 /* pos */) &&
77-
VerifyField<int16_t>(verifier, 6 /* mana */) &&
78-
VerifyField<int16_t>(verifier, 8 /* hp */) &&
79-
VerifyField<flatbuffers::uoffset_t>(verifier, 10 /* name */) &&
84+
VerifyField<Vec3>(verifier, VT_POS) &&
85+
VerifyField<int16_t>(verifier, VT_MANA) &&
86+
VerifyField<int16_t>(verifier, VT_HP) &&
87+
VerifyField<flatbuffers::uoffset_t>(verifier, VT_NAME) &&
8088
verifier.Verify(name()) &&
81-
VerifyField<flatbuffers::uoffset_t>(verifier, 14 /* inventory */) &&
89+
VerifyField<flatbuffers::uoffset_t>(verifier, VT_INVENTORY) &&
8290
verifier.Verify(inventory()) &&
83-
VerifyField<int8_t>(verifier, 16 /* color */) &&
91+
VerifyField<int8_t>(verifier, VT_COLOR) &&
8492
verifier.EndTable();
8593
}
8694
};
8795

8896
struct MonsterBuilder {
8997
flatbuffers::FlatBufferBuilder &fbb_;
9098
flatbuffers::uoffset_t start_;
91-
void add_pos(const Vec3 *pos) { fbb_.AddStruct(4, pos); }
92-
void add_mana(int16_t mana) { fbb_.AddElement<int16_t>(6, mana, 150); }
93-
void add_hp(int16_t hp) { fbb_.AddElement<int16_t>(8, hp, 100); }
94-
void add_name(flatbuffers::Offset<flatbuffers::String> name) { fbb_.AddOffset(10, name); }
95-
void add_inventory(flatbuffers::Offset<flatbuffers::Vector<uint8_t>> inventory) { fbb_.AddOffset(14, inventory); }
96-
void add_color(Color color) { fbb_.AddElement<int8_t>(16, static_cast<int8_t>(color), 2); }
99+
void add_pos(const Vec3 *pos) { fbb_.AddStruct(Monster::VT_POS, pos); }
100+
void add_mana(int16_t mana) { fbb_.AddElement<int16_t>(Monster::VT_MANA, mana, 150); }
101+
void add_hp(int16_t hp) { fbb_.AddElement<int16_t>(Monster::VT_HP, hp, 100); }
102+
void add_name(flatbuffers::Offset<flatbuffers::String> name) { fbb_.AddOffset(Monster::VT_NAME, name); }
103+
void add_inventory(flatbuffers::Offset<flatbuffers::Vector<uint8_t>> inventory) { fbb_.AddOffset(Monster::VT_INVENTORY, inventory); }
104+
void add_color(Color color) { fbb_.AddElement<int8_t>(Monster::VT_COLOR, static_cast<int8_t>(color), 2); }
97105
MonsterBuilder(flatbuffers::FlatBufferBuilder &_fbb) : fbb_(_fbb) { start_ = fbb_.StartTable(); }
98106
MonsterBuilder &operator=(const MonsterBuilder &);
99107
flatbuffers::Offset<Monster> Finish() {

src/idl_gen_cpp.cpp

Lines changed: 24 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -242,6 +242,12 @@ std::string GenUnderlyingCast(const Parser &parser, const FieldDef &field,
242242
: val;
243243
}
244244

245+
std::string GenFieldOffsetName(const FieldDef &field) {
246+
std::string uname = field.name;
247+
std::transform(uname.begin(), uname.end(), uname.begin(), ::toupper);
248+
return "VT_" + uname;
249+
}
250+
245251
// Generate an accessor struct, builder structs & function for a table.
246252
static void GenTable(const Parser &parser, StructDef &struct_def,
247253
const GeneratorOptions &opts, std::string *code_ptr) {
@@ -254,6 +260,19 @@ static void GenTable(const Parser &parser, StructDef &struct_def,
254260
code += "struct " + struct_def.name;
255261
code += " FLATBUFFERS_FINAL_CLASS : private flatbuffers::Table";
256262
code += " {\n";
263+
// Generate field id constants.
264+
code += " enum {\n";
265+
for (auto it = struct_def.fields.vec.begin();
266+
it != struct_def.fields.vec.end();
267+
++it) {
268+
auto &field = **it;
269+
if (!field.deprecated) { // Deprecated fields won't be accessible.
270+
code += " " + GenFieldOffsetName(field) + " = ";
271+
code += NumToString(field.value.offset) + ",\n";
272+
}
273+
}
274+
code += " };\n";
275+
// Generate the accessors.
257276
for (auto it = struct_def.fields.vec.begin();
258277
it != struct_def.fields.vec.end();
259278
++it) {
@@ -268,7 +287,7 @@ static void GenTable(const Parser &parser, StructDef &struct_def,
268287
auto accessor = is_scalar
269288
? "GetField<"
270289
: (IsStruct(field.value.type) ? "GetStruct<" : "GetPointer<");
271-
auto offsetstr = NumToString(field.value.offset);
290+
auto offsetstr = GenFieldOffsetName(field);
272291
auto call =
273292
accessor +
274293
GenTypeGet(parser, field.value.type, "", "const ", " *", false) +
@@ -347,8 +366,7 @@ static void GenTable(const Parser &parser, StructDef &struct_def,
347366
code += prefix + "VerifyField";
348367
if (field.required) code += "Required";
349368
code += "<" + GenTypeSize(parser, field.value.type);
350-
code += ">(verifier, " + NumToString(field.value.offset);
351-
code += " /* " + field.name + " */)";
369+
code += ">(verifier, " + GenFieldOffsetName(field) + ")";
352370
switch (field.value.type.base_type) {
353371
case BASE_TYPE_UNION:
354372
code += prefix + "Verify" + field.value.type.enum_def->name;
@@ -412,7 +430,7 @@ static void GenTable(const Parser &parser, StructDef &struct_def,
412430
} else {
413431
code += "Offset";
414432
}
415-
code += "(" + NumToString(field.value.offset) + ", ";
433+
code += "(" + struct_def.name + "::" + GenFieldOffsetName(field) + ", ";
416434
code += GenUnderlyingCast(parser, field, false, field.name);
417435
if (IsScalar(field.value.type.base_type))
418436
code += ", " + field.value.constant;
@@ -433,7 +451,8 @@ static void GenTable(const Parser &parser, StructDef &struct_def,
433451
++it) {
434452
auto &field = **it;
435453
if (!field.deprecated && field.required) {
436-
code += " fbb_.Required(o, " + NumToString(field.value.offset);
454+
code += " fbb_.Required(o, ";
455+
code += struct_def.name + "::" + GenFieldOffsetName(field);
437456
code += "); // " + field.name + "\n";
438457
}
439458
}

0 commit comments

Comments
 (0)