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 <sstream>
32 
33 #include <google/protobuf/compiler/code_generator.h>
34 #include <google/protobuf/compiler/plugin.h>
35 #include <google/protobuf/descriptor.h>
36 #include <google/protobuf/descriptor.pb.h>
37 #include <google/protobuf/io/printer.h>
38 #include <google/protobuf/io/zero_copy_stream.h>
39 #include <google/protobuf/stubs/strutil.h>
40 
41 #include <google/protobuf/compiler/csharp/csharp_doc_comment.h>
42 #include <google/protobuf/compiler/csharp/csharp_helpers.h>
43 #include <google/protobuf/compiler/csharp/csharp_options.h>
44 #include <google/protobuf/compiler/csharp/csharp_primitive_field.h>
45 
46 namespace google {
47 namespace protobuf {
48 namespace compiler {
49 namespace csharp {
50 
PrimitiveFieldGenerator(const FieldDescriptor * descriptor,int fieldOrdinal,const Options * options)51 PrimitiveFieldGenerator::PrimitiveFieldGenerator(
52     const FieldDescriptor* descriptor, int fieldOrdinal, const Options *options)
53     : FieldGeneratorBase(descriptor, fieldOrdinal, options) {
54   // TODO(jonskeet): Make this cleaner...
55   is_value_type = descriptor->type() != FieldDescriptor::TYPE_STRING
56       && descriptor->type() != FieldDescriptor::TYPE_BYTES;
57   if (!is_value_type) {
58     variables_["has_property_check"] = variables_["property_name"] + ".Length != 0";
59     variables_["other_has_property_check"] = "other." + variables_["property_name"] + ".Length != 0";
60   }
61 }
62 
~PrimitiveFieldGenerator()63 PrimitiveFieldGenerator::~PrimitiveFieldGenerator() {
64 }
65 
GenerateMembers(io::Printer * printer)66 void PrimitiveFieldGenerator::GenerateMembers(io::Printer* printer) {
67   // TODO(jonskeet): Work out whether we want to prevent the fields from ever being
68   // null, or whether we just handle it, in the cases of bytes and string.
69   // (Basically, should null-handling code be in the getter or the setter?)
70   printer->Print(
71     variables_,
72     "private $type_name$ $name_def_message$;\n");
73   WritePropertyDocComment(printer, descriptor_);
74   AddDeprecatedFlag(printer);
75   printer->Print(
76     variables_,
77     "$access_level$ $type_name$ $property_name$ {\n"
78     "  get { return $name$_; }\n"
79     "  set {\n");
80   if (is_value_type) {
81     printer->Print(
82       variables_,
83       "    $name$_ = value;\n");
84   } else {
85     printer->Print(
86       variables_,
87       "    $name$_ = pb::ProtoPreconditions.CheckNotNull(value, \"value\");\n");
88   }
89   printer->Print(
90     "  }\n"
91     "}\n");
92 }
93 
GenerateMergingCode(io::Printer * printer)94 void PrimitiveFieldGenerator::GenerateMergingCode(io::Printer* printer) {
95   printer->Print(
96     variables_,
97     "if ($other_has_property_check$) {\n"
98     "  $property_name$ = other.$property_name$;\n"
99     "}\n");
100 }
101 
GenerateParsingCode(io::Printer * printer)102 void PrimitiveFieldGenerator::GenerateParsingCode(io::Printer* printer) {
103   // Note: invoke the property setter rather than writing straight to the field,
104   // so that we can normalize "null to empty" for strings and bytes.
105   printer->Print(
106     variables_,
107     "$property_name$ = input.Read$capitalized_type_name$();\n");
108 }
109 
GenerateSerializationCode(io::Printer * printer)110 void PrimitiveFieldGenerator::GenerateSerializationCode(io::Printer* printer) {
111   printer->Print(
112     variables_,
113     "if ($has_property_check$) {\n"
114     "  output.WriteRawTag($tag_bytes$);\n"
115     "  output.Write$capitalized_type_name$($property_name$);\n"
116     "}\n");
117 }
118 
GenerateSerializedSizeCode(io::Printer * printer)119 void PrimitiveFieldGenerator::GenerateSerializedSizeCode(io::Printer* printer) {
120   printer->Print(
121     variables_,
122     "if ($has_property_check$) {\n");
123   printer->Indent();
124   int fixedSize = GetFixedSize(descriptor_->type());
125   if (fixedSize == -1) {
126     printer->Print(
127       variables_,
128       "size += $tag_size$ + pb::CodedOutputStream.Compute$capitalized_type_name$Size($property_name$);\n");
129   } else {
130     printer->Print(
131       "size += $tag_size$ + $fixed_size$;\n",
132       "fixed_size", SimpleItoa(fixedSize),
133       "tag_size", variables_["tag_size"]);
134   }
135   printer->Outdent();
136   printer->Print("}\n");
137 }
138 
WriteHash(io::Printer * printer)139 void PrimitiveFieldGenerator::WriteHash(io::Printer* printer) {
140   printer->Print(
141     variables_,
142     "if ($has_property_check$) hash ^= $property_name$.GetHashCode();\n");
143 }
WriteEquals(io::Printer * printer)144 void PrimitiveFieldGenerator::WriteEquals(io::Printer* printer) {
145   printer->Print(
146     variables_,
147     "if ($property_name$ != other.$property_name$) return false;\n");
148 }
WriteToString(io::Printer * printer)149 void PrimitiveFieldGenerator::WriteToString(io::Printer* printer) {
150   printer->Print(
151     variables_,
152     "PrintField(\"$descriptor_name$\", $has_property_check$, $property_name$, writer);\n");
153 }
154 
GenerateCloningCode(io::Printer * printer)155 void PrimitiveFieldGenerator::GenerateCloningCode(io::Printer* printer) {
156   printer->Print(variables_,
157     "$name$_ = other.$name$_;\n");
158 }
159 
GenerateCodecCode(io::Printer * printer)160 void PrimitiveFieldGenerator::GenerateCodecCode(io::Printer* printer) {
161   printer->Print(
162     variables_,
163     "pb::FieldCodec.For$capitalized_type_name$($tag$)");
164 }
165 
PrimitiveOneofFieldGenerator(const FieldDescriptor * descriptor,int fieldOrdinal,const Options * options)166 PrimitiveOneofFieldGenerator::PrimitiveOneofFieldGenerator(
167     const FieldDescriptor* descriptor, int fieldOrdinal, const Options *options)
168     : PrimitiveFieldGenerator(descriptor, fieldOrdinal, options) {
169   SetCommonOneofFieldVariables(&variables_);
170 }
171 
~PrimitiveOneofFieldGenerator()172 PrimitiveOneofFieldGenerator::~PrimitiveOneofFieldGenerator() {
173 }
174 
GenerateMembers(io::Printer * printer)175 void PrimitiveOneofFieldGenerator::GenerateMembers(io::Printer* printer) {
176   WritePropertyDocComment(printer, descriptor_);
177   AddDeprecatedFlag(printer);
178   printer->Print(
179     variables_,
180     "$access_level$ $type_name$ $property_name$ {\n"
181     "  get { return $has_property_check$ ? ($type_name$) $oneof_name$_ : $default_value$; }\n"
182     "  set {\n");
183     if (is_value_type) {
184       printer->Print(
185         variables_,
186         "    $oneof_name$_ = value;\n");
187     } else {
188       printer->Print(
189         variables_,
190         "    $oneof_name$_ = pb::ProtoPreconditions.CheckNotNull(value, \"value\");\n");
191     }
192     printer->Print(
193       variables_,
194       "    $oneof_name$Case_ = $oneof_property_name$OneofCase.$property_name$;\n"
195       "  }\n"
196       "}\n");
197 }
198 
WriteToString(io::Printer * printer)199 void PrimitiveOneofFieldGenerator::WriteToString(io::Printer* printer) {
200   printer->Print(variables_,
201     "PrintField(\"$descriptor_name$\", $has_property_check$, $oneof_name$_, writer);\n");
202 }
203 
GenerateParsingCode(io::Printer * printer)204 void PrimitiveOneofFieldGenerator::GenerateParsingCode(io::Printer* printer) {
205     printer->Print(
206       variables_,
207       "$property_name$ = input.Read$capitalized_type_name$();\n");
208 }
209 
GenerateCloningCode(io::Printer * printer)210 void PrimitiveOneofFieldGenerator::GenerateCloningCode(io::Printer* printer) {
211   printer->Print(variables_,
212     "$property_name$ = other.$property_name$;\n");
213 }
214 
215 }  // namespace csharp
216 }  // namespace compiler
217 }  // namespace protobuf
218 }  // namespace google
219