1 #include "Errors.h"
2 #include "stream_proto_utils.h"
3 #include "string_utils.h"
4 
5 #include <iomanip>
6 #include <iostream>
7 #include <sstream>
8 
9 using namespace android::stream_proto;
10 using namespace google::protobuf::io;
11 using namespace std;
12 
13 const bool GENERATE_MAPPING = true;
14 
15 static string
make_filename(const FileDescriptorProto & file_descriptor)16 make_filename(const FileDescriptorProto& file_descriptor)
17 {
18     return file_descriptor.name() + ".h";
19 }
20 
21 static void
write_enum(stringstream & text,const EnumDescriptorProto & enu,const string & indent)22 write_enum(stringstream& text, const EnumDescriptorProto& enu, const string& indent)
23 {
24     const int N = enu.value_size();
25     text << indent << "// enum " << enu.name() << endl;
26     for (int i=0; i<N; i++) {
27         const EnumValueDescriptorProto& value = enu.value(i);
28         text << indent << "const int "
29                 << make_constant_name(value.name())
30                 << " = " << value.number() << ";" << endl;
31     }
32 
33     if (GENERATE_MAPPING) {
34         string name = make_constant_name(enu.name());
35         string prefix = name + "_";
36         text << indent << "const int _ENUM_" << name << "_COUNT = " << N << ";" << endl;
37         text << indent << "const char* _ENUM_" << name << "_NAMES[" << N << "] = {" << endl;
38         for (int i=0; i<N; i++) {
39             text << indent << INDENT << "\"" << stripPrefix(enu.value(i).name(), prefix) << "\"," << endl;
40         }
41         text << indent << "};" << endl;
42         text << indent << "const int _ENUM_" << name << "_VALUES[" << N << "] = {" << endl;
43         for (int i=0; i<N; i++) {
44             text << indent << INDENT << make_constant_name(enu.value(i).name()) << "," << endl;
45         }
46         text << indent << "};" << endl;
47     }
48 
49     text << endl;
50 }
51 
52 static void
write_field(stringstream & text,const FieldDescriptorProto & field,const string & indent)53 write_field(stringstream& text, const FieldDescriptorProto& field, const string& indent)
54 {
55     string optional_comment = field.label() == FieldDescriptorProto::LABEL_OPTIONAL
56             ? "optional " : "";
57     string repeated_comment = field.label() == FieldDescriptorProto::LABEL_REPEATED
58             ? "repeated " : "";
59     string proto_type = get_proto_type(field);
60     string packed_comment = field.options().packed()
61             ? " [packed=true]" : "";
62     text << indent << "// " << optional_comment << repeated_comment << proto_type << ' '
63             << field.name() << " = " << field.number() << packed_comment << ';' << endl;
64 
65     text << indent << "const uint64_t " << make_constant_name(field.name()) << " = 0x";
66 
67     ios::fmtflags fmt(text.flags());
68     text << setfill('0') << setw(16) << hex << get_field_id(field);
69     text.flags(fmt);
70 
71     text << "LL;" << endl;
72 
73     text << endl;
74 }
75 
76 static void
write_message(stringstream & text,const DescriptorProto & message,const string & indent)77 write_message(stringstream& text, const DescriptorProto& message, const string& indent)
78 {
79     int N;
80     const string indented = indent + INDENT;
81 
82     text << indent << "// message " << message.name() << endl;
83     text << indent << "namespace " << message.name() << " {" << endl;
84 
85     // Enums
86     N = message.enum_type_size();
87     for (int i=0; i<N; i++) {
88         write_enum(text, message.enum_type(i), indented);
89     }
90 
91     // Nested classes
92     N = message.nested_type_size();
93     for (int i=0; i<N; i++) {
94         write_message(text, message.nested_type(i), indented);
95     }
96 
97     // Fields
98     N = message.field_size();
99     for (int i=0; i<N; i++) {
100         write_field(text, message.field(i), indented);
101     }
102 
103     if (GENERATE_MAPPING) {
104         N = message.field_size();
105         text << indented << "const int _FIELD_COUNT = " << N << ";" << endl;
106         text << indented << "const char* _FIELD_NAMES[" << N << "] = {" << endl;
107         for (int i=0; i<N; i++) {
108             text << indented << INDENT << "\"" << message.field(i).name() << "\"," << endl;
109         }
110         text << indented << "};" << endl;
111         text << indented << "const uint64_t _FIELD_IDS[" << N << "] = {" << endl;
112         for (int i=0; i<N; i++) {
113             text << indented << INDENT << make_constant_name(message.field(i).name()) << "," << endl;
114         }
115         text << indented << "};" << endl << endl;
116     }
117 
118     text << indent << "} //" << message.name() << endl;
119     text << endl;
120 }
121 
122 static void
write_header_file(CodeGeneratorResponse * response,const FileDescriptorProto & file_descriptor)123 write_header_file(CodeGeneratorResponse* response, const FileDescriptorProto& file_descriptor)
124 {
125     stringstream text;
126 
127     text << "// Generated by protoc-gen-cppstream. DO NOT MODIFY." << endl;
128     text << "// source: " << file_descriptor.name() << endl << endl;
129 
130     string header = "ANDROID_" + replace_string(file_descriptor.name(), '/', '_');
131     header = replace_string(header, '.', '_') + "_stream_h";
132     header = make_constant_name(header);
133 
134     text << "#ifndef " << header << endl;
135     text << "#define " << header << endl;
136     text << endl;
137 
138     vector<string> namespaces = split(file_descriptor.package(), '.');
139     for (vector<string>::iterator it = namespaces.begin(); it != namespaces.end(); it++) {
140         text << "namespace " << *it << " {" << endl;
141     }
142     text << endl;
143 
144     size_t N;
145     N = file_descriptor.enum_type_size();
146     for (size_t i=0; i<N; i++) {
147         write_enum(text, file_descriptor.enum_type(i), "");
148     }
149 
150     N = file_descriptor.message_type_size();
151     for (size_t i=0; i<N; i++) {
152         write_message(text, file_descriptor.message_type(i), "");
153     }
154 
155     for (vector<string>::iterator it = namespaces.begin(); it != namespaces.end(); it++) {
156         text << "} // " << *it << endl;
157     }
158 
159     text << endl;
160     text << "#endif // " << header << endl;
161 
162     CodeGeneratorResponse::File* file_response = response->add_file();
163     file_response->set_name(make_filename(file_descriptor));
164     file_response->set_content(text.str());
165 }
166 
main(int argc,char const * argv[])167 int main(int argc, char const *argv[])
168 {
169     (void)argc;
170     (void)argv;
171 
172     GOOGLE_PROTOBUF_VERIFY_VERSION;
173 
174     CodeGeneratorRequest request;
175     CodeGeneratorResponse response;
176 
177     // Read the request
178     request.ParseFromIstream(&cin);
179 
180     // Build the files we need.
181     const int N = request.proto_file_size();
182     for (int i=0; i<N; i++) {
183         const FileDescriptorProto& file_descriptor = request.proto_file(i);
184         if (should_generate_for_file(request, file_descriptor.name())) {
185             write_header_file(&response, file_descriptor);
186         }
187     }
188 
189     // If we had errors, don't write the response. Print the errors and exit.
190     if (ERRORS.HasErrors()) {
191         ERRORS.Print();
192         return 1;
193     }
194 
195     // If we didn't have errors, write the response and exit happily.
196     response.SerializeToOstream(&cout);
197 
198     /* code */
199     return 0;
200 }
201