1 // Protocol Buffers - Google's data interchange format
2 // Copyright 2008 Google Inc. All rights reserved.
3 // https://developers.google.com/protocol-buffers/
4 //
5 // Redistribution and use in source and binary forms, with or without
6 // modification, are permitted provided that the following conditions are
7 // met:
8 //
9 // * Redistributions of source code must retain the above copyright
10 // notice, this list of conditions and the following disclaimer.
11 // * Redistributions in binary form must reproduce the above
12 // copyright notice, this list of conditions and the following disclaimer
13 // in the documentation and/or other materials provided with the
14 // distribution.
15 // * Neither the name of Google Inc. nor the names of its
16 // contributors may be used to endorse or promote products derived from
17 // this software without specific prior written permission.
18 //
19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31 // Author: kenton@google.com (Kenton Varda)
32 // Based on original Protocol Buffers design by
33 // Sanjay Ghemawat, Jeff Dean, and others.
34
35 #include <google/protobuf/compiler/cpp/cpp_enum_field.h>
36 #include <google/protobuf/compiler/cpp/cpp_helpers.h>
37 #include <google/protobuf/io/printer.h>
38 #include <google/protobuf/wire_format.h>
39 #include <google/protobuf/stubs/strutil.h>
40
41 namespace google {
42 namespace protobuf {
43 namespace compiler {
44 namespace cpp {
45
46 namespace {
47
SetEnumVariables(const FieldDescriptor * descriptor,map<string,string> * variables,const Options & options)48 void SetEnumVariables(const FieldDescriptor* descriptor,
49 map<string, string>* variables,
50 const Options& options) {
51 SetCommonFieldVariables(descriptor, variables, options);
52 const EnumValueDescriptor* default_value = descriptor->default_value_enum();
53 (*variables)["type"] = ClassName(descriptor->enum_type(), true);
54 (*variables)["default"] = Int32ToString(default_value->number());
55 (*variables)["full_name"] = descriptor->full_name();
56 }
57
58 } // namespace
59
60 // ===================================================================
61
EnumFieldGenerator(const FieldDescriptor * descriptor,const Options & options)62 EnumFieldGenerator::EnumFieldGenerator(const FieldDescriptor* descriptor,
63 const Options& options)
64 : FieldGenerator(options), descriptor_(descriptor) {
65 SetEnumVariables(descriptor, &variables_, options);
66 }
67
~EnumFieldGenerator()68 EnumFieldGenerator::~EnumFieldGenerator() {}
69
70 void EnumFieldGenerator::
GeneratePrivateMembers(io::Printer * printer) const71 GeneratePrivateMembers(io::Printer* printer) const {
72 printer->Print(variables_, "int $name$_;\n");
73 }
74
75 void EnumFieldGenerator::
GenerateAccessorDeclarations(io::Printer * printer) const76 GenerateAccessorDeclarations(io::Printer* printer) const {
77 printer->Print(variables_,
78 "$deprecated_attr$$type$ $name$() const;\n"
79 "$deprecated_attr$void set_$name$($type$ value);\n");
80 }
81
82 void EnumFieldGenerator::
GenerateInlineAccessorDefinitions(io::Printer * printer,bool is_inline) const83 GenerateInlineAccessorDefinitions(io::Printer* printer,
84 bool is_inline) const {
85 map<string, string> variables(variables_);
86 variables["inline"] = is_inline ? "inline" : "";
87 printer->Print(variables,
88 "$inline$ $type$ $classname$::$name$() const {\n"
89 " // @@protoc_insertion_point(field_get:$full_name$)\n"
90 " return static_cast< $type$ >($name$_);\n"
91 "}\n"
92 "$inline$ void $classname$::set_$name$($type$ value) {\n");
93 if (!HasPreservingUnknownEnumSemantics(descriptor_->file())) {
94 printer->Print(variables,
95 " assert($type$_IsValid(value));\n");
96 }
97 printer->Print(variables,
98 " $set_hasbit$\n"
99 " $name$_ = value;\n"
100 " // @@protoc_insertion_point(field_set:$full_name$)\n"
101 "}\n");
102 }
103
104 void EnumFieldGenerator::
GenerateClearingCode(io::Printer * printer) const105 GenerateClearingCode(io::Printer* printer) const {
106 printer->Print(variables_, "$name$_ = $default$;\n");
107 }
108
109 void EnumFieldGenerator::
GenerateMergingCode(io::Printer * printer) const110 GenerateMergingCode(io::Printer* printer) const {
111 printer->Print(variables_, "set_$name$(from.$name$());\n");
112 }
113
114 void EnumFieldGenerator::
GenerateSwappingCode(io::Printer * printer) const115 GenerateSwappingCode(io::Printer* printer) const {
116 printer->Print(variables_, "std::swap($name$_, other->$name$_);\n");
117 }
118
119 void EnumFieldGenerator::
GenerateConstructorCode(io::Printer * printer) const120 GenerateConstructorCode(io::Printer* printer) const {
121 printer->Print(variables_, "$name$_ = $default$;\n");
122 }
123
124 void EnumFieldGenerator::
GenerateMergeFromCodedStream(io::Printer * printer) const125 GenerateMergeFromCodedStream(io::Printer* printer) const {
126 printer->Print(variables_,
127 "int value;\n"
128 "DO_((::google::protobuf::internal::WireFormatLite::ReadPrimitive<\n"
129 " int, ::google::protobuf::internal::WireFormatLite::TYPE_ENUM>(\n"
130 " input, &value)));\n");
131 if (HasPreservingUnknownEnumSemantics(descriptor_->file())) {
132 printer->Print(variables_,
133 "set_$name$(static_cast< $type$ >(value));\n");
134 } else {
135 printer->Print(variables_,
136 "if ($type$_IsValid(value)) {\n"
137 " set_$name$(static_cast< $type$ >(value));\n");
138 if (UseUnknownFieldSet(descriptor_->file(), options_)) {
139 printer->Print(variables_,
140 "} else {\n"
141 " mutable_unknown_fields()->AddVarint($number$, value);\n");
142 } else {
143 printer->Print(
144 "} else {\n"
145 " unknown_fields_stream.WriteVarint32($tag$);\n"
146 " unknown_fields_stream.WriteVarint32(value);\n",
147 "tag", SimpleItoa(internal::WireFormat::MakeTag(descriptor_)));
148 }
149 printer->Print(variables_,
150 "}\n");
151 }
152 }
153
154 void EnumFieldGenerator::
GenerateSerializeWithCachedSizes(io::Printer * printer) const155 GenerateSerializeWithCachedSizes(io::Printer* printer) const {
156 printer->Print(variables_,
157 "::google::protobuf::internal::WireFormatLite::WriteEnum(\n"
158 " $number$, this->$name$(), output);\n");
159 }
160
161 void EnumFieldGenerator::
GenerateSerializeWithCachedSizesToArray(io::Printer * printer) const162 GenerateSerializeWithCachedSizesToArray(io::Printer* printer) const {
163 printer->Print(variables_,
164 "target = ::google::protobuf::internal::WireFormatLite::WriteEnumToArray(\n"
165 " $number$, this->$name$(), target);\n");
166 }
167
168 void EnumFieldGenerator::
GenerateByteSize(io::Printer * printer) const169 GenerateByteSize(io::Printer* printer) const {
170 printer->Print(variables_,
171 "total_size += $tag_size$ +\n"
172 " ::google::protobuf::internal::WireFormatLite::EnumSize(this->$name$());\n");
173 }
174
175 // ===================================================================
176
177 EnumOneofFieldGenerator::
EnumOneofFieldGenerator(const FieldDescriptor * descriptor,const Options & options)178 EnumOneofFieldGenerator(const FieldDescriptor* descriptor,
179 const Options& options)
180 : EnumFieldGenerator(descriptor, options) {
181 SetCommonOneofFieldVariables(descriptor, &variables_);
182 }
183
~EnumOneofFieldGenerator()184 EnumOneofFieldGenerator::~EnumOneofFieldGenerator() {}
185
186 void EnumOneofFieldGenerator::
GenerateInlineAccessorDefinitions(io::Printer * printer,bool is_inline) const187 GenerateInlineAccessorDefinitions(io::Printer* printer,
188 bool is_inline) const {
189 map<string, string> variables(variables_);
190 variables["inline"] = is_inline ? "inline" : "";
191 printer->Print(variables,
192 "$inline$ $type$ $classname$::$name$() const {\n"
193 " // @@protoc_insertion_point(field_get:$full_name$)\n"
194 " if (has_$name$()) {\n"
195 " return static_cast< $type$ >($oneof_prefix$$name$_);\n"
196 " }\n"
197 " return static_cast< $type$ >($default$);\n"
198 "}\n"
199 "$inline$ void $classname$::set_$name$($type$ value) {\n");
200 if (!HasPreservingUnknownEnumSemantics(descriptor_->file())) {
201 printer->Print(variables,
202 " assert($type$_IsValid(value));\n");
203 }
204 printer->Print(variables,
205 " if (!has_$name$()) {\n"
206 " clear_$oneof_name$();\n"
207 " set_has_$name$();\n"
208 " }\n"
209 " $oneof_prefix$$name$_ = value;\n"
210 " // @@protoc_insertion_point(field_set:$full_name$)\n"
211 "}\n");
212 }
213
214 void EnumOneofFieldGenerator::
GenerateClearingCode(io::Printer * printer) const215 GenerateClearingCode(io::Printer* printer) const {
216 printer->Print(variables_, "$oneof_prefix$$name$_ = $default$;\n");
217 }
218
219 void EnumOneofFieldGenerator::
GenerateSwappingCode(io::Printer * printer) const220 GenerateSwappingCode(io::Printer* printer) const {
221 // Don't print any swapping code. Swapping the union will swap this field.
222 }
223
224 void EnumOneofFieldGenerator::
GenerateConstructorCode(io::Printer * printer) const225 GenerateConstructorCode(io::Printer* printer) const {
226 printer->Print(variables_,
227 " $classname$_default_oneof_instance_->$name$_ = $default$;\n");
228 }
229
230 // ===================================================================
231
RepeatedEnumFieldGenerator(const FieldDescriptor * descriptor,const Options & options)232 RepeatedEnumFieldGenerator::RepeatedEnumFieldGenerator(
233 const FieldDescriptor* descriptor, const Options& options)
234 : FieldGenerator(options), descriptor_(descriptor) {
235 SetEnumVariables(descriptor, &variables_, options);
236 }
237
~RepeatedEnumFieldGenerator()238 RepeatedEnumFieldGenerator::~RepeatedEnumFieldGenerator() {}
239
240 void RepeatedEnumFieldGenerator::
GeneratePrivateMembers(io::Printer * printer) const241 GeneratePrivateMembers(io::Printer* printer) const {
242 printer->Print(variables_,
243 "::google::protobuf::RepeatedField<int> $name$_;\n");
244 if (descriptor_->is_packed() &&
245 HasGeneratedMethods(descriptor_->file(), options_)) {
246 printer->Print(variables_,
247 "mutable int _$name$_cached_byte_size_;\n");
248 }
249 }
250
251 void RepeatedEnumFieldGenerator::
GenerateAccessorDeclarations(io::Printer * printer) const252 GenerateAccessorDeclarations(io::Printer* printer) const {
253 printer->Print(variables_,
254 "$deprecated_attr$$type$ $name$(int index) const;\n"
255 "$deprecated_attr$void set_$name$(int index, $type$ value);\n"
256 "$deprecated_attr$void add_$name$($type$ value);\n");
257 printer->Print(variables_,
258 "$deprecated_attr$const ::google::protobuf::RepeatedField<int>& $name$() const;\n"
259 "$deprecated_attr$::google::protobuf::RepeatedField<int>* mutable_$name$();\n");
260 }
261
262 void RepeatedEnumFieldGenerator::
GenerateInlineAccessorDefinitions(io::Printer * printer,bool is_inline) const263 GenerateInlineAccessorDefinitions(io::Printer* printer,
264 bool is_inline) const {
265 map<string, string> variables(variables_);
266 variables["inline"] = is_inline ? "inline" : "";
267 printer->Print(variables,
268 "$inline$ $type$ $classname$::$name$(int index) const {\n"
269 " // @@protoc_insertion_point(field_get:$full_name$)\n"
270 " return static_cast< $type$ >($name$_.Get(index));\n"
271 "}\n"
272 "$inline$ void $classname$::set_$name$(int index, $type$ value) {\n");
273 if (!HasPreservingUnknownEnumSemantics(descriptor_->file())) {
274 printer->Print(variables,
275 " assert($type$_IsValid(value));\n");
276 }
277 printer->Print(variables,
278 " $name$_.Set(index, value);\n"
279 " // @@protoc_insertion_point(field_set:$full_name$)\n"
280 "}\n"
281 "$inline$ void $classname$::add_$name$($type$ value) {\n");
282 if (!HasPreservingUnknownEnumSemantics(descriptor_->file())) {
283 printer->Print(variables,
284 " assert($type$_IsValid(value));\n");
285 }
286 printer->Print(variables,
287 " $name$_.Add(value);\n"
288 " // @@protoc_insertion_point(field_add:$full_name$)\n"
289 "}\n");
290 printer->Print(variables,
291 "$inline$ const ::google::protobuf::RepeatedField<int>&\n"
292 "$classname$::$name$() const {\n"
293 " // @@protoc_insertion_point(field_list:$full_name$)\n"
294 " return $name$_;\n"
295 "}\n"
296 "$inline$ ::google::protobuf::RepeatedField<int>*\n"
297 "$classname$::mutable_$name$() {\n"
298 " // @@protoc_insertion_point(field_mutable_list:$full_name$)\n"
299 " return &$name$_;\n"
300 "}\n");
301 }
302
303 void RepeatedEnumFieldGenerator::
GenerateClearingCode(io::Printer * printer) const304 GenerateClearingCode(io::Printer* printer) const {
305 printer->Print(variables_, "$name$_.Clear();\n");
306 }
307
308 void RepeatedEnumFieldGenerator::
GenerateMergingCode(io::Printer * printer) const309 GenerateMergingCode(io::Printer* printer) const {
310 printer->Print(variables_, "$name$_.MergeFrom(from.$name$_);\n");
311 }
312
313 void RepeatedEnumFieldGenerator::
GenerateSwappingCode(io::Printer * printer) const314 GenerateSwappingCode(io::Printer* printer) const {
315 printer->Print(variables_, "$name$_.UnsafeArenaSwap(&other->$name$_);\n");
316 }
317
318 void RepeatedEnumFieldGenerator::
GenerateConstructorCode(io::Printer * printer) const319 GenerateConstructorCode(io::Printer* printer) const {
320 // Not needed for repeated fields.
321 }
322
323 void RepeatedEnumFieldGenerator::
GenerateMergeFromCodedStream(io::Printer * printer) const324 GenerateMergeFromCodedStream(io::Printer* printer) const {
325 // Don't use ReadRepeatedPrimitive here so that the enum can be validated.
326 printer->Print(variables_,
327 "int value;\n"
328 "DO_((::google::protobuf::internal::WireFormatLite::ReadPrimitive<\n"
329 " int, ::google::protobuf::internal::WireFormatLite::TYPE_ENUM>(\n"
330 " input, &value)));\n");
331 if (HasPreservingUnknownEnumSemantics(descriptor_->file())) {
332 printer->Print(variables_,
333 "add_$name$(static_cast< $type$ >(value));\n");
334 } else {
335 printer->Print(variables_,
336 "if ($type$_IsValid(value)) {\n"
337 " add_$name$(static_cast< $type$ >(value));\n");
338 if (UseUnknownFieldSet(descriptor_->file(), options_)) {
339 printer->Print(variables_,
340 "} else {\n"
341 " mutable_unknown_fields()->AddVarint($number$, value);\n");
342 } else {
343 printer->Print(
344 "} else {\n"
345 " unknown_fields_stream.WriteVarint32(tag);\n"
346 " unknown_fields_stream.WriteVarint32(value);\n");
347 }
348 printer->Print("}\n");
349 }
350 }
351
352 void RepeatedEnumFieldGenerator::
GenerateMergeFromCodedStreamWithPacking(io::Printer * printer) const353 GenerateMergeFromCodedStreamWithPacking(io::Printer* printer) const {
354 if (!descriptor_->is_packed()) {
355 // This path is rarely executed, so we use a non-inlined implementation.
356 if (HasPreservingUnknownEnumSemantics(descriptor_->file())) {
357 printer->Print(variables_,
358 "DO_((::google::protobuf::internal::"
359 "WireFormatLite::ReadPackedEnumPreserveUnknowns(\n"
360 " input,\n"
361 " $number$,\n"
362 " NULL,\n"
363 " NULL,\n"
364 " this->mutable_$name$())));\n");
365 } else if (UseUnknownFieldSet(descriptor_->file(), options_)) {
366 printer->Print(variables_,
367 "DO_((::google::protobuf::internal::WireFormat::ReadPackedEnumPreserveUnknowns(\n"
368 " input,\n"
369 " $number$,\n"
370 " $type$_IsValid,\n"
371 " mutable_unknown_fields(),\n"
372 " this->mutable_$name$())));\n");
373 } else {
374 printer->Print(variables_,
375 "DO_((::google::protobuf::internal::"
376 "WireFormatLite::ReadPackedEnumPreserveUnknowns(\n"
377 " input,\n"
378 " $number$,\n"
379 " $type$_IsValid,\n"
380 " &unknown_fields_stream,\n"
381 " this->mutable_$name$())));\n");
382 }
383 } else {
384 printer->Print(variables_,
385 "::google::protobuf::uint32 length;\n"
386 "DO_(input->ReadVarint32(&length));\n"
387 "::google::protobuf::io::CodedInputStream::Limit limit = "
388 "input->PushLimit(length);\n"
389 "while (input->BytesUntilLimit() > 0) {\n"
390 " int value;\n"
391 " DO_((::google::protobuf::internal::WireFormatLite::ReadPrimitive<\n"
392 " int, ::google::protobuf::internal::WireFormatLite::TYPE_ENUM>(\n"
393 " input, &value)));\n");
394 if (HasPreservingUnknownEnumSemantics(descriptor_->file())) {
395 printer->Print(variables_,
396 " add_$name$(static_cast< $type$ >(value));\n");
397 } else {
398 printer->Print(variables_,
399 " if ($type$_IsValid(value)) {\n"
400 " add_$name$(static_cast< $type$ >(value));\n"
401 " } else {\n");
402 if (UseUnknownFieldSet(descriptor_->file(), options_)) {
403 printer->Print(variables_,
404 " mutable_unknown_fields()->AddVarint($number$, value);\n");
405 } else {
406 printer->Print(variables_,
407 " unknown_fields_stream.WriteVarint32(tag);\n"
408 " unknown_fields_stream.WriteVarint32(value);\n");
409 }
410 printer->Print(
411 " }\n");
412 }
413 printer->Print(variables_,
414 "}\n"
415 "input->PopLimit(limit);\n");
416 }
417 }
418
419 void RepeatedEnumFieldGenerator::
GenerateSerializeWithCachedSizes(io::Printer * printer) const420 GenerateSerializeWithCachedSizes(io::Printer* printer) const {
421 if (descriptor_->is_packed()) {
422 // Write the tag and the size.
423 printer->Print(variables_,
424 "if (this->$name$_size() > 0) {\n"
425 " ::google::protobuf::internal::WireFormatLite::WriteTag(\n"
426 " $number$,\n"
427 " ::google::protobuf::internal::WireFormatLite::WIRETYPE_LENGTH_DELIMITED,\n"
428 " output);\n"
429 " output->WriteVarint32(_$name$_cached_byte_size_);\n"
430 "}\n");
431 }
432 printer->Print(variables_,
433 "for (int i = 0; i < this->$name$_size(); i++) {\n");
434 if (descriptor_->is_packed()) {
435 printer->Print(variables_,
436 " ::google::protobuf::internal::WireFormatLite::WriteEnumNoTag(\n"
437 " this->$name$(i), output);\n");
438 } else {
439 printer->Print(variables_,
440 " ::google::protobuf::internal::WireFormatLite::WriteEnum(\n"
441 " $number$, this->$name$(i), output);\n");
442 }
443 printer->Print("}\n");
444 }
445
446 void RepeatedEnumFieldGenerator::
GenerateSerializeWithCachedSizesToArray(io::Printer * printer) const447 GenerateSerializeWithCachedSizesToArray(io::Printer* printer) const {
448 if (descriptor_->is_packed()) {
449 // Write the tag and the size.
450 printer->Print(variables_,
451 "if (this->$name$_size() > 0) {\n"
452 " target = ::google::protobuf::internal::WireFormatLite::WriteTagToArray(\n"
453 " $number$,\n"
454 " ::google::protobuf::internal::WireFormatLite::WIRETYPE_LENGTH_DELIMITED,\n"
455 " target);\n"
456 " target = ::google::protobuf::io::CodedOutputStream::WriteVarint32ToArray("
457 " _$name$_cached_byte_size_, target);\n"
458 "}\n");
459 }
460 printer->Print(variables_,
461 "for (int i = 0; i < this->$name$_size(); i++) {\n");
462 if (descriptor_->is_packed()) {
463 printer->Print(variables_,
464 " target = ::google::protobuf::internal::WireFormatLite::WriteEnumNoTagToArray(\n"
465 " this->$name$(i), target);\n");
466 } else {
467 printer->Print(variables_,
468 " target = ::google::protobuf::internal::WireFormatLite::WriteEnumToArray(\n"
469 " $number$, this->$name$(i), target);\n");
470 }
471 printer->Print("}\n");
472 }
473
474 void RepeatedEnumFieldGenerator::
GenerateByteSize(io::Printer * printer) const475 GenerateByteSize(io::Printer* printer) const {
476 printer->Print(variables_,
477 "{\n"
478 " int data_size = 0;\n");
479 printer->Indent();
480 printer->Print(variables_,
481 "for (int i = 0; i < this->$name$_size(); i++) {\n"
482 " data_size += ::google::protobuf::internal::WireFormatLite::EnumSize(\n"
483 " this->$name$(i));\n"
484 "}\n");
485
486 if (descriptor_->is_packed()) {
487 printer->Print(variables_,
488 "if (data_size > 0) {\n"
489 " total_size += $tag_size$ +\n"
490 " ::google::protobuf::internal::WireFormatLite::Int32Size(data_size);\n"
491 "}\n"
492 "GOOGLE_SAFE_CONCURRENT_WRITES_BEGIN();\n"
493 "_$name$_cached_byte_size_ = data_size;\n"
494 "GOOGLE_SAFE_CONCURRENT_WRITES_END();\n"
495 "total_size += data_size;\n");
496 } else {
497 printer->Print(variables_,
498 "total_size += $tag_size$ * this->$name$_size() + data_size;\n");
499 }
500 printer->Outdent();
501 printer->Print("}\n");
502 }
503
504 } // namespace cpp
505 } // namespace compiler
506 } // namespace protobuf
507 } // namespace google
508