1*890232f2SAndroid Build Coastguard Worker /*
2*890232f2SAndroid Build Coastguard Worker * Copyright 2014 Google Inc. All rights reserved.
3*890232f2SAndroid Build Coastguard Worker *
4*890232f2SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*890232f2SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*890232f2SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*890232f2SAndroid Build Coastguard Worker *
8*890232f2SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*890232f2SAndroid Build Coastguard Worker *
10*890232f2SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*890232f2SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*890232f2SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*890232f2SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*890232f2SAndroid Build Coastguard Worker * limitations under the License.
15*890232f2SAndroid Build Coastguard Worker */
16*890232f2SAndroid Build Coastguard Worker
17*890232f2SAndroid Build Coastguard Worker // independent from idl_parser, since this code is not needed for most clients
18*890232f2SAndroid Build Coastguard Worker
19*890232f2SAndroid Build Coastguard Worker #include "flatbuffers/code_generators.h"
20*890232f2SAndroid Build Coastguard Worker #include "flatbuffers/flatbuffers.h"
21*890232f2SAndroid Build Coastguard Worker #include "flatbuffers/idl.h"
22*890232f2SAndroid Build Coastguard Worker #include "flatbuffers/util.h"
23*890232f2SAndroid Build Coastguard Worker
24*890232f2SAndroid Build Coastguard Worker namespace flatbuffers {
25*890232f2SAndroid Build Coastguard Worker
GenType(const Type & type,bool underlying=false)26*890232f2SAndroid Build Coastguard Worker static std::string GenType(const Type &type, bool underlying = false) {
27*890232f2SAndroid Build Coastguard Worker switch (type.base_type) {
28*890232f2SAndroid Build Coastguard Worker case BASE_TYPE_STRUCT:
29*890232f2SAndroid Build Coastguard Worker return type.struct_def->defined_namespace->GetFullyQualifiedName(
30*890232f2SAndroid Build Coastguard Worker type.struct_def->name);
31*890232f2SAndroid Build Coastguard Worker case BASE_TYPE_VECTOR: return "[" + GenType(type.VectorType()) + "]";
32*890232f2SAndroid Build Coastguard Worker default:
33*890232f2SAndroid Build Coastguard Worker if (type.enum_def && !underlying) {
34*890232f2SAndroid Build Coastguard Worker return type.enum_def->defined_namespace->GetFullyQualifiedName(
35*890232f2SAndroid Build Coastguard Worker type.enum_def->name);
36*890232f2SAndroid Build Coastguard Worker } else {
37*890232f2SAndroid Build Coastguard Worker return kTypeNames[type.base_type];
38*890232f2SAndroid Build Coastguard Worker }
39*890232f2SAndroid Build Coastguard Worker }
40*890232f2SAndroid Build Coastguard Worker }
41*890232f2SAndroid Build Coastguard Worker
GenNameSpace(const Namespace & name_space,std::string * _schema,const Namespace ** last_namespace)42*890232f2SAndroid Build Coastguard Worker static void GenNameSpace(const Namespace &name_space, std::string *_schema,
43*890232f2SAndroid Build Coastguard Worker const Namespace **last_namespace) {
44*890232f2SAndroid Build Coastguard Worker if (*last_namespace == &name_space) return;
45*890232f2SAndroid Build Coastguard Worker *last_namespace = &name_space;
46*890232f2SAndroid Build Coastguard Worker auto &schema = *_schema;
47*890232f2SAndroid Build Coastguard Worker schema += "namespace ";
48*890232f2SAndroid Build Coastguard Worker for (auto it = name_space.components.begin();
49*890232f2SAndroid Build Coastguard Worker it != name_space.components.end(); ++it) {
50*890232f2SAndroid Build Coastguard Worker if (it != name_space.components.begin()) schema += ".";
51*890232f2SAndroid Build Coastguard Worker schema += *it;
52*890232f2SAndroid Build Coastguard Worker }
53*890232f2SAndroid Build Coastguard Worker schema += ";\n\n";
54*890232f2SAndroid Build Coastguard Worker }
55*890232f2SAndroid Build Coastguard Worker
56*890232f2SAndroid Build Coastguard Worker // Generate a flatbuffer schema from the Parser's internal representation.
GenerateFBS(const Parser & parser,const std::string & file_name)57*890232f2SAndroid Build Coastguard Worker std::string GenerateFBS(const Parser &parser, const std::string &file_name) {
58*890232f2SAndroid Build Coastguard Worker // Proto namespaces may clash with table names, escape the ones that were
59*890232f2SAndroid Build Coastguard Worker // generated from a table:
60*890232f2SAndroid Build Coastguard Worker for (auto it = parser.namespaces_.begin(); it != parser.namespaces_.end();
61*890232f2SAndroid Build Coastguard Worker ++it) {
62*890232f2SAndroid Build Coastguard Worker auto &ns = **it;
63*890232f2SAndroid Build Coastguard Worker for (size_t i = 0; i < ns.from_table; i++) {
64*890232f2SAndroid Build Coastguard Worker ns.components[ns.components.size() - 1 - i] += "_";
65*890232f2SAndroid Build Coastguard Worker }
66*890232f2SAndroid Build Coastguard Worker
67*890232f2SAndroid Build Coastguard Worker if (parser.opts.proto_mode && !parser.opts.proto_namespace_suffix.empty()) {
68*890232f2SAndroid Build Coastguard Worker // Since we know that all these namespaces come from a .proto, and all are
69*890232f2SAndroid Build Coastguard Worker // being converted, we can simply apply this suffix to all of them.
70*890232f2SAndroid Build Coastguard Worker ns.components.insert(ns.components.end() - ns.from_table,
71*890232f2SAndroid Build Coastguard Worker parser.opts.proto_namespace_suffix);
72*890232f2SAndroid Build Coastguard Worker }
73*890232f2SAndroid Build Coastguard Worker }
74*890232f2SAndroid Build Coastguard Worker
75*890232f2SAndroid Build Coastguard Worker std::string schema;
76*890232f2SAndroid Build Coastguard Worker schema += "// Generated from " + file_name + ".proto\n\n";
77*890232f2SAndroid Build Coastguard Worker if (parser.opts.include_dependence_headers) {
78*890232f2SAndroid Build Coastguard Worker // clang-format off
79*890232f2SAndroid Build Coastguard Worker int num_includes = 0;
80*890232f2SAndroid Build Coastguard Worker for (auto it = parser.included_files_.begin();
81*890232f2SAndroid Build Coastguard Worker it != parser.included_files_.end(); ++it) {
82*890232f2SAndroid Build Coastguard Worker if (it->second.empty())
83*890232f2SAndroid Build Coastguard Worker continue;
84*890232f2SAndroid Build Coastguard Worker std::string basename;
85*890232f2SAndroid Build Coastguard Worker if(parser.opts.keep_prefix) {
86*890232f2SAndroid Build Coastguard Worker basename = flatbuffers::StripExtension(it->second);
87*890232f2SAndroid Build Coastguard Worker } else {
88*890232f2SAndroid Build Coastguard Worker basename = flatbuffers::StripPath(
89*890232f2SAndroid Build Coastguard Worker flatbuffers::StripExtension(it->second));
90*890232f2SAndroid Build Coastguard Worker }
91*890232f2SAndroid Build Coastguard Worker schema += "include \"" + basename + ".fbs\";\n";
92*890232f2SAndroid Build Coastguard Worker num_includes++;
93*890232f2SAndroid Build Coastguard Worker }
94*890232f2SAndroid Build Coastguard Worker if (num_includes) schema += "\n";
95*890232f2SAndroid Build Coastguard Worker // clang-format on
96*890232f2SAndroid Build Coastguard Worker }
97*890232f2SAndroid Build Coastguard Worker // Generate code for all the enum declarations.
98*890232f2SAndroid Build Coastguard Worker const Namespace *last_namespace = nullptr;
99*890232f2SAndroid Build Coastguard Worker for (auto enum_def_it = parser.enums_.vec.begin();
100*890232f2SAndroid Build Coastguard Worker enum_def_it != parser.enums_.vec.end(); ++enum_def_it) {
101*890232f2SAndroid Build Coastguard Worker EnumDef &enum_def = **enum_def_it;
102*890232f2SAndroid Build Coastguard Worker if (parser.opts.include_dependence_headers && enum_def.generated) {
103*890232f2SAndroid Build Coastguard Worker continue;
104*890232f2SAndroid Build Coastguard Worker }
105*890232f2SAndroid Build Coastguard Worker GenNameSpace(*enum_def.defined_namespace, &schema, &last_namespace);
106*890232f2SAndroid Build Coastguard Worker GenComment(enum_def.doc_comment, &schema, nullptr);
107*890232f2SAndroid Build Coastguard Worker if (enum_def.is_union)
108*890232f2SAndroid Build Coastguard Worker schema += "union " + enum_def.name;
109*890232f2SAndroid Build Coastguard Worker else
110*890232f2SAndroid Build Coastguard Worker schema += "enum " + enum_def.name + " : ";
111*890232f2SAndroid Build Coastguard Worker schema += GenType(enum_def.underlying_type, true) + " {\n";
112*890232f2SAndroid Build Coastguard Worker for (auto it = enum_def.Vals().begin(); it != enum_def.Vals().end(); ++it) {
113*890232f2SAndroid Build Coastguard Worker auto &ev = **it;
114*890232f2SAndroid Build Coastguard Worker GenComment(ev.doc_comment, &schema, nullptr, " ");
115*890232f2SAndroid Build Coastguard Worker if (enum_def.is_union)
116*890232f2SAndroid Build Coastguard Worker schema += " " + GenType(ev.union_type) + ",\n";
117*890232f2SAndroid Build Coastguard Worker else
118*890232f2SAndroid Build Coastguard Worker schema += " " + ev.name + " = " + enum_def.ToString(ev) + ",\n";
119*890232f2SAndroid Build Coastguard Worker }
120*890232f2SAndroid Build Coastguard Worker schema += "}\n\n";
121*890232f2SAndroid Build Coastguard Worker }
122*890232f2SAndroid Build Coastguard Worker // Generate code for all structs/tables.
123*890232f2SAndroid Build Coastguard Worker for (auto it = parser.structs_.vec.begin(); it != parser.structs_.vec.end();
124*890232f2SAndroid Build Coastguard Worker ++it) {
125*890232f2SAndroid Build Coastguard Worker StructDef &struct_def = **it;
126*890232f2SAndroid Build Coastguard Worker if (parser.opts.include_dependence_headers && struct_def.generated) {
127*890232f2SAndroid Build Coastguard Worker continue;
128*890232f2SAndroid Build Coastguard Worker }
129*890232f2SAndroid Build Coastguard Worker GenNameSpace(*struct_def.defined_namespace, &schema, &last_namespace);
130*890232f2SAndroid Build Coastguard Worker GenComment(struct_def.doc_comment, &schema, nullptr);
131*890232f2SAndroid Build Coastguard Worker schema += "table " + struct_def.name + " {\n";
132*890232f2SAndroid Build Coastguard Worker for (auto field_it = struct_def.fields.vec.begin();
133*890232f2SAndroid Build Coastguard Worker field_it != struct_def.fields.vec.end(); ++field_it) {
134*890232f2SAndroid Build Coastguard Worker auto &field = **field_it;
135*890232f2SAndroid Build Coastguard Worker if (field.value.type.base_type != BASE_TYPE_UTYPE) {
136*890232f2SAndroid Build Coastguard Worker GenComment(field.doc_comment, &schema, nullptr, " ");
137*890232f2SAndroid Build Coastguard Worker schema += " " + field.name + ":" + GenType(field.value.type);
138*890232f2SAndroid Build Coastguard Worker if (field.value.constant != "0") schema += " = " + field.value.constant;
139*890232f2SAndroid Build Coastguard Worker if (field.IsRequired()) schema += " (required)";
140*890232f2SAndroid Build Coastguard Worker schema += ";\n";
141*890232f2SAndroid Build Coastguard Worker }
142*890232f2SAndroid Build Coastguard Worker }
143*890232f2SAndroid Build Coastguard Worker schema += "}\n\n";
144*890232f2SAndroid Build Coastguard Worker }
145*890232f2SAndroid Build Coastguard Worker return schema;
146*890232f2SAndroid Build Coastguard Worker }
147*890232f2SAndroid Build Coastguard Worker
GenerateFBS(const Parser & parser,const std::string & path,const std::string & file_name)148*890232f2SAndroid Build Coastguard Worker bool GenerateFBS(const Parser &parser, const std::string &path,
149*890232f2SAndroid Build Coastguard Worker const std::string &file_name) {
150*890232f2SAndroid Build Coastguard Worker return SaveFile((path + file_name + ".fbs").c_str(),
151*890232f2SAndroid Build Coastguard Worker GenerateFBS(parser, file_name), false);
152*890232f2SAndroid Build Coastguard Worker }
153*890232f2SAndroid Build Coastguard Worker
154*890232f2SAndroid Build Coastguard Worker } // namespace flatbuffers
155