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 #include <iostream>
32 
33 #include <google/protobuf/compiler/objectivec/objectivec_field.h>
34 #include <google/protobuf/compiler/objectivec/objectivec_helpers.h>
35 #include <google/protobuf/compiler/objectivec/objectivec_enum_field.h>
36 #include <google/protobuf/compiler/objectivec/objectivec_map_field.h>
37 #include <google/protobuf/compiler/objectivec/objectivec_message_field.h>
38 #include <google/protobuf/compiler/objectivec/objectivec_primitive_field.h>
39 #include <google/protobuf/io/printer.h>
40 #include <google/protobuf/wire_format.h>
41 #include <google/protobuf/stubs/common.h>
42 #include <google/protobuf/stubs/strutil.h>
43 
44 namespace google {
45 namespace protobuf {
46 namespace compiler {
47 namespace objectivec {
48 
49 namespace {
50 
SetCommonFieldVariables(const FieldDescriptor * descriptor,map<string,string> * variables)51 void SetCommonFieldVariables(const FieldDescriptor* descriptor,
52                              map<string, string>* variables) {
53   string camel_case_name = FieldName(descriptor);
54   string raw_field_name;
55   if (descriptor->type() == FieldDescriptor::TYPE_GROUP) {
56     raw_field_name = descriptor->message_type()->name();
57   } else {
58     raw_field_name = descriptor->name();
59   }
60   // The logic here has to match -[GGPBFieldDescriptor textFormatName].
61   const string un_camel_case_name(
62       UnCamelCaseFieldName(camel_case_name, descriptor));
63   const bool needs_custom_name = (raw_field_name != un_camel_case_name);
64 
65   SourceLocation location;
66   if (descriptor->GetSourceLocation(&location)) {
67     (*variables)["comments"] = BuildCommentsString(location);
68   } else {
69     (*variables)["comments"] = "\n";
70   }
71   const string& classname = ClassName(descriptor->containing_type());
72   (*variables)["classname"] = classname;
73   (*variables)["name"] = camel_case_name;
74   const string& capitalized_name = FieldNameCapitalized(descriptor);
75   (*variables)["capitalized_name"] = capitalized_name;
76   (*variables)["raw_field_name"] = raw_field_name;
77   (*variables)["field_number_name"] =
78       classname + "_FieldNumber_" + capitalized_name;
79   (*variables)["field_number"] = SimpleItoa(descriptor->number());
80   (*variables)["field_type"] = GetCapitalizedType(descriptor);
81   (*variables)["deprecated_attribute"] = GetOptionalDeprecatedAttribute(descriptor);
82   std::vector<string> field_flags;
83   if (descriptor->is_repeated()) field_flags.push_back("GPBFieldRepeated");
84   if (descriptor->is_required()) field_flags.push_back("GPBFieldRequired");
85   if (descriptor->is_optional()) field_flags.push_back("GPBFieldOptional");
86   if (descriptor->is_packed()) field_flags.push_back("GPBFieldPacked");
87 
88   // ObjC custom flags.
89   if (descriptor->has_default_value())
90     field_flags.push_back("GPBFieldHasDefaultValue");
91   if (needs_custom_name) field_flags.push_back("GPBFieldTextFormatNameCustom");
92   if (descriptor->type() == FieldDescriptor::TYPE_ENUM) {
93     field_flags.push_back("GPBFieldHasEnumDescriptor");
94   }
95 
96   (*variables)["fieldflags"] = BuildFlagsString(field_flags);
97 
98   (*variables)["default"] = DefaultValue(descriptor);
99   (*variables)["default_name"] = GPBGenericValueFieldName(descriptor);
100 
101   (*variables)["dataTypeSpecific_name"] = "className";
102   (*variables)["dataTypeSpecific_value"] = "NULL";
103 
104   (*variables)["storage_offset_value"] =
105       "(uint32_t)offsetof(" + classname + "__storage_, " + camel_case_name + ")";
106   (*variables)["storage_offset_comment"] = "";
107 
108   // Clear some common things so they can be set just when needed.
109   (*variables)["storage_attribute"] = "";
110 }
111 
112 }  // namespace
113 
Make(const FieldDescriptor * field,const Options & options)114 FieldGenerator* FieldGenerator::Make(const FieldDescriptor* field,
115                                      const Options& options) {
116   FieldGenerator* result = NULL;
117   if (field->is_repeated()) {
118     switch (GetObjectiveCType(field)) {
119       case OBJECTIVECTYPE_MESSAGE: {
120         if (field->is_map()) {
121           result = new MapFieldGenerator(field, options);
122         } else {
123           result = new RepeatedMessageFieldGenerator(field, options);
124         }
125         break;
126       }
127       case OBJECTIVECTYPE_ENUM:
128         result = new RepeatedEnumFieldGenerator(field, options);
129         break;
130       default:
131         result = new RepeatedPrimitiveFieldGenerator(field, options);
132         break;
133     }
134   } else {
135     switch (GetObjectiveCType(field)) {
136       case OBJECTIVECTYPE_MESSAGE: {
137         result = new MessageFieldGenerator(field, options);
138         break;
139       }
140       case OBJECTIVECTYPE_ENUM:
141         result = new EnumFieldGenerator(field, options);
142         break;
143       default:
144         if (IsReferenceType(field)) {
145           result = new PrimitiveObjFieldGenerator(field, options);
146         } else {
147           result = new PrimitiveFieldGenerator(field, options);
148         }
149         break;
150     }
151   }
152   result->FinishInitialization();
153   return result;
154 }
155 
FieldGenerator(const FieldDescriptor * descriptor,const Options & options)156 FieldGenerator::FieldGenerator(const FieldDescriptor* descriptor,
157                                const Options& options)
158     : descriptor_(descriptor) {
159   SetCommonFieldVariables(descriptor, &variables_);
160 }
161 
~FieldGenerator()162 FieldGenerator::~FieldGenerator() {}
163 
GenerateFieldNumberConstant(io::Printer * printer) const164 void FieldGenerator::GenerateFieldNumberConstant(io::Printer* printer) const {
165   printer->Print(
166       variables_,
167       "$field_number_name$ = $field_number$,\n");
168 }
169 
GenerateCFunctionDeclarations(io::Printer * printer) const170 void FieldGenerator::GenerateCFunctionDeclarations(
171     io::Printer* printer) const {
172   // Nothing
173 }
174 
GenerateCFunctionImplementations(io::Printer * printer) const175 void FieldGenerator::GenerateCFunctionImplementations(
176     io::Printer* printer) const {
177   // Nothing
178 }
179 
DetermineForwardDeclarations(set<string> * fwd_decls) const180 void FieldGenerator::DetermineForwardDeclarations(
181     set<string>* fwd_decls) const {
182   // Nothing
183 }
184 
GenerateFieldDescription(io::Printer * printer,bool include_default) const185 void FieldGenerator::GenerateFieldDescription(
186     io::Printer* printer, bool include_default) const {
187   // Printed in the same order as the structure decl.
188   if (include_default) {
189     printer->Print(
190         variables_,
191         "{\n"
192         "  .defaultValue.$default_name$ = $default$,\n"
193         "  .core.name = \"$name$\",\n"
194         "  .core.dataTypeSpecific.$dataTypeSpecific_name$ = $dataTypeSpecific_value$,\n"
195         "  .core.number = $field_number_name$,\n"
196         "  .core.hasIndex = $has_index$,\n"
197         "  .core.offset = $storage_offset_value$,$storage_offset_comment$\n"
198         "  .core.flags = $fieldflags$,\n"
199         "  .core.dataType = GPBDataType$field_type$,\n"
200         "},\n");
201   } else {
202     printer->Print(
203         variables_,
204         "{\n"
205         "  .name = \"$name$\",\n"
206         "  .dataTypeSpecific.$dataTypeSpecific_name$ = $dataTypeSpecific_value$,\n"
207         "  .number = $field_number_name$,\n"
208         "  .hasIndex = $has_index$,\n"
209         "  .offset = $storage_offset_value$,$storage_offset_comment$\n"
210         "  .flags = $fieldflags$,\n"
211         "  .dataType = GPBDataType$field_type$,\n"
212         "},\n");
213   }
214 }
215 
SetRuntimeHasBit(int has_index)216 void FieldGenerator::SetRuntimeHasBit(int has_index) {
217   variables_["has_index"] = SimpleItoa(has_index);
218 }
219 
SetNoHasBit(void)220 void FieldGenerator::SetNoHasBit(void) {
221   variables_["has_index"] = "GPBNoHasBit";
222 }
223 
ExtraRuntimeHasBitsNeeded(void) const224 int FieldGenerator::ExtraRuntimeHasBitsNeeded(void) const {
225   return 0;
226 }
227 
SetExtraRuntimeHasBitsBase(int index_base)228 void FieldGenerator::SetExtraRuntimeHasBitsBase(int index_base) {
229   // NOTE: src/google/protobuf/compiler/plugin.cc makes use of cerr for some
230   // error cases, so it seems to be ok to use as a back door for errors.
231   cerr << "Error: should have overridden SetExtraRuntimeHasBitsBase()." << endl;
232   cerr.flush();
233   abort();
234 }
235 
SetOneofIndexBase(int index_base)236 void FieldGenerator::SetOneofIndexBase(int index_base) {
237   if (descriptor_->containing_oneof() != NULL) {
238     int index = descriptor_->containing_oneof()->index() + index_base;
239     // Flip the sign to mark it as a oneof.
240     variables_["has_index"] = SimpleItoa(-index);
241   }
242 }
243 
FinishInitialization(void)244 void FieldGenerator::FinishInitialization(void) {
245   // If "property_type" wasn't set, make it "storage_type".
246   if ((variables_.find("property_type") == variables_.end()) &&
247       (variables_.find("storage_type") != variables_.end())) {
248     variables_["property_type"] = variable("storage_type");
249   }
250 }
251 
SingleFieldGenerator(const FieldDescriptor * descriptor,const Options & options)252 SingleFieldGenerator::SingleFieldGenerator(const FieldDescriptor* descriptor,
253                                            const Options& options)
254     : FieldGenerator(descriptor, options) {
255   // Nothing
256 }
257 
~SingleFieldGenerator()258 SingleFieldGenerator::~SingleFieldGenerator() {}
259 
GenerateFieldStorageDeclaration(io::Printer * printer) const260 void SingleFieldGenerator::GenerateFieldStorageDeclaration(
261     io::Printer* printer) const {
262   printer->Print(variables_, "$storage_type$ $name$;\n");
263 }
264 
GeneratePropertyDeclaration(io::Printer * printer) const265 void SingleFieldGenerator::GeneratePropertyDeclaration(
266     io::Printer* printer) const {
267   printer->Print(variables_, "$comments$");
268   printer->Print(
269       variables_,
270       "@property(nonatomic, readwrite) $property_type$ $name$$deprecated_attribute$;\n"
271       "\n");
272   if (WantsHasProperty()) {
273     printer->Print(
274         variables_,
275         "@property(nonatomic, readwrite) BOOL has$capitalized_name$$deprecated_attribute$;\n");
276   }
277 }
278 
GeneratePropertyImplementation(io::Printer * printer) const279 void SingleFieldGenerator::GeneratePropertyImplementation(
280     io::Printer* printer) const {
281   if (WantsHasProperty()) {
282     printer->Print(variables_, "@dynamic has$capitalized_name$, $name$;\n");
283   } else {
284     printer->Print(variables_, "@dynamic $name$;\n");
285   }
286 }
287 
WantsHasProperty(void) const288 bool SingleFieldGenerator::WantsHasProperty(void) const {
289   if (descriptor_->containing_oneof() != NULL) {
290     // If in a oneof, it uses the oneofcase instead of a has bit.
291     return false;
292   }
293   if (HasFieldPresence(descriptor_->file())) {
294     // In proto1/proto2, every field has a has_$name$() method.
295     return true;
296   }
297   return false;
298 }
299 
RuntimeUsesHasBit(void) const300 bool SingleFieldGenerator::RuntimeUsesHasBit(void) const {
301   if (descriptor_->containing_oneof() != NULL) {
302     // The oneof tracks what is set instead.
303     return false;
304   }
305   return true;
306 }
307 
ObjCObjFieldGenerator(const FieldDescriptor * descriptor,const Options & options)308 ObjCObjFieldGenerator::ObjCObjFieldGenerator(const FieldDescriptor* descriptor,
309                                              const Options& options)
310     : SingleFieldGenerator(descriptor, options) {
311   variables_["property_storage_attribute"] = "strong";
312   if (IsRetainedName(variables_["name"])) {
313     variables_["storage_attribute"] = " NS_RETURNS_NOT_RETAINED";
314   }
315 }
316 
~ObjCObjFieldGenerator()317 ObjCObjFieldGenerator::~ObjCObjFieldGenerator() {}
318 
GenerateFieldStorageDeclaration(io::Printer * printer) const319 void ObjCObjFieldGenerator::GenerateFieldStorageDeclaration(
320     io::Printer* printer) const {
321   printer->Print(variables_, "$storage_type$ *$name$;\n");
322 }
323 
GeneratePropertyDeclaration(io::Printer * printer) const324 void ObjCObjFieldGenerator::GeneratePropertyDeclaration(
325     io::Printer* printer) const {
326 
327   // Differs from SingleFieldGenerator::GeneratePropertyDeclaration() in that
328   // it uses pointers and deals with Objective C's rules around storage name
329   // conventions (init*, new*, etc.)
330 
331   printer->Print(variables_, "$comments$");
332   printer->Print(
333       variables_,
334       "@property(nonatomic, readwrite, $property_storage_attribute$, null_resettable) $property_type$ *$name$$storage_attribute$$deprecated_attribute$;\n");
335   if (WantsHasProperty()) {
336     printer->Print(
337         variables_,
338         "/// Test to see if @c $name$ has been set.\n"
339         "@property(nonatomic, readwrite) BOOL has$capitalized_name$$deprecated_attribute$;\n");
340   }
341   if (IsInitName(variables_.find("name")->second)) {
342     // If property name starts with init we need to annotate it to get past ARC.
343     // http://stackoverflow.com/questions/18723226/how-do-i-annotate-an-objective-c-property-with-an-objc-method-family/18723227#18723227
344     printer->Print(variables_,
345                    "- ($property_type$ *)$name$ GPB_METHOD_FAMILY_NONE$deprecated_attribute$;\n");
346   }
347   printer->Print("\n");
348 }
349 
RepeatedFieldGenerator(const FieldDescriptor * descriptor,const Options & options)350 RepeatedFieldGenerator::RepeatedFieldGenerator(
351     const FieldDescriptor* descriptor, const Options& options)
352     : ObjCObjFieldGenerator(descriptor, options) {
353   // Default to no comment and let the cases needing it fill it in.
354   variables_["array_comment"] = "";
355 }
356 
~RepeatedFieldGenerator()357 RepeatedFieldGenerator::~RepeatedFieldGenerator() {}
358 
FinishInitialization(void)359 void RepeatedFieldGenerator::FinishInitialization(void) {
360   FieldGenerator::FinishInitialization();
361   if (variables_.find("array_property_type") == variables_.end()) {
362     variables_["array_property_type"] = variable("array_storage_type");
363   }
364 }
365 
GenerateFieldStorageDeclaration(io::Printer * printer) const366 void RepeatedFieldGenerator::GenerateFieldStorageDeclaration(
367     io::Printer* printer) const {
368   printer->Print(variables_, "$array_storage_type$ *$name$;\n");
369 }
370 
GeneratePropertyImplementation(io::Printer * printer) const371 void RepeatedFieldGenerator::GeneratePropertyImplementation(
372     io::Printer* printer) const {
373   printer->Print(variables_, "@dynamic $name$, $name$_Count;\n");
374 }
375 
GeneratePropertyDeclaration(io::Printer * printer) const376 void RepeatedFieldGenerator::GeneratePropertyDeclaration(
377     io::Printer* printer) const {
378 
379   // Repeated fields don't need the has* properties, but they do expose a
380   // *Count (to check without autocreation).  So for the field property we need
381   // the same logic as ObjCObjFieldGenerator::GeneratePropertyDeclaration() for
382   // dealing with needing Objective C's rules around storage name conventions
383   // (init*, new*, etc.)
384 
385   printer->Print(
386       variables_,
387       "$comments$"
388       "$array_comment$"
389       "@property(nonatomic, readwrite, strong, null_resettable) $array_property_type$ *$name$$storage_attribute$$deprecated_attribute$;\n"
390       "/// The number of items in @c $name$ without causing the array to be created.\n"
391       "@property(nonatomic, readonly) NSUInteger $name$_Count$deprecated_attribute$;\n");
392   if (IsInitName(variables_.find("name")->second)) {
393     // If property name starts with init we need to annotate it to get past ARC.
394     // http://stackoverflow.com/questions/18723226/how-do-i-annotate-an-objective-c-property-with-an-objc-method-family/18723227#18723227
395     printer->Print(variables_,
396                    "- ($array_property_type$ *)$name$ GPB_METHOD_FAMILY_NONE$deprecated_attribute$;\n");
397   }
398   printer->Print("\n");
399 }
400 
WantsHasProperty(void) const401 bool RepeatedFieldGenerator::WantsHasProperty(void) const {
402   // Consumer check the array size/existance rather than a has bit.
403   return false;
404 }
405 
RuntimeUsesHasBit(void) const406 bool RepeatedFieldGenerator::RuntimeUsesHasBit(void) const {
407   return false;  // The array having anything is what is used.
408 }
409 
FieldGeneratorMap(const Descriptor * descriptor,const Options & options)410 FieldGeneratorMap::FieldGeneratorMap(const Descriptor* descriptor,
411                                      const Options& options)
412     : descriptor_(descriptor),
413       field_generators_(
414           new scoped_ptr<FieldGenerator>[descriptor->field_count()]),
415       extension_generators_(
416           new scoped_ptr<FieldGenerator>[descriptor->extension_count()]) {
417   // Construct all the FieldGenerators.
418   for (int i = 0; i < descriptor->field_count(); i++) {
419     field_generators_[i].reset(
420         FieldGenerator::Make(descriptor->field(i), options));
421   }
422   for (int i = 0; i < descriptor->extension_count(); i++) {
423     extension_generators_[i].reset(
424         FieldGenerator::Make(descriptor->extension(i), options));
425   }
426 }
427 
~FieldGeneratorMap()428 FieldGeneratorMap::~FieldGeneratorMap() {}
429 
get(const FieldDescriptor * field) const430 const FieldGenerator& FieldGeneratorMap::get(
431     const FieldDescriptor* field) const {
432   GOOGLE_CHECK_EQ(field->containing_type(), descriptor_);
433   return *field_generators_[field->index()];
434 }
435 
get_extension(int index) const436 const FieldGenerator& FieldGeneratorMap::get_extension(int index) const {
437   return *extension_generators_[index];
438 }
439 
CalculateHasBits(void)440 int FieldGeneratorMap::CalculateHasBits(void) {
441   int total_bits = 0;
442   for (int i = 0; i < descriptor_->field_count(); i++) {
443     if (field_generators_[i]->RuntimeUsesHasBit()) {
444       field_generators_[i]->SetRuntimeHasBit(total_bits);
445       ++total_bits;
446     } else {
447       field_generators_[i]->SetNoHasBit();
448     }
449     int extra_bits = field_generators_[i]->ExtraRuntimeHasBitsNeeded();
450     if (extra_bits) {
451       field_generators_[i]->SetExtraRuntimeHasBitsBase(total_bits);
452       total_bits += extra_bits;
453     }
454   }
455   return total_bits;
456 }
457 
SetOneofIndexBase(int index_base)458 void FieldGeneratorMap::SetOneofIndexBase(int index_base) {
459   for (int i = 0; i < descriptor_->field_count(); i++) {
460     field_generators_[i]->SetOneofIndexBase(index_base);
461   }
462 }
463 
DoesAnyFieldHaveNonZeroDefault(void) const464 bool FieldGeneratorMap::DoesAnyFieldHaveNonZeroDefault(void) const {
465   for (int i = 0; i < descriptor_->field_count(); i++) {
466     if (HasNonZeroDefaultValue(descriptor_->field(i))) {
467       return true;
468     }
469   }
470 
471   return false;
472 }
473 
474 }  // namespace objectivec
475 }  // namespace compiler
476 }  // namespace protobuf
477 }  // namespace google
478