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_message_field.h>
44 #include <google/protobuf/compiler/csharp/csharp_options.h>
45 
46 namespace google {
47 namespace protobuf {
48 namespace compiler {
49 namespace csharp {
50 
MessageFieldGenerator(const FieldDescriptor * descriptor,int fieldOrdinal,const Options * options)51 MessageFieldGenerator::MessageFieldGenerator(const FieldDescriptor* descriptor,
52                                              int fieldOrdinal,
53                                              const Options *options)
54     : FieldGeneratorBase(descriptor, fieldOrdinal, options) {
55   variables_["has_property_check"] = name() + "_ != null";
56   variables_["has_not_property_check"] = name() + "_ == null";
57 }
58 
~MessageFieldGenerator()59 MessageFieldGenerator::~MessageFieldGenerator() {
60 
61 }
62 
GenerateMembers(io::Printer * printer)63 void MessageFieldGenerator::GenerateMembers(io::Printer* printer) {
64   printer->Print(
65     variables_,
66     "private $type_name$ $name$_;\n");
67   WritePropertyDocComment(printer, descriptor_);
68   AddDeprecatedFlag(printer);
69   printer->Print(
70     variables_,
71     "$access_level$ $type_name$ $property_name$ {\n"
72     "  get { return $name$_; }\n"
73     "  set {\n"
74     "    $name$_ = value;\n"
75     "  }\n"
76     "}\n");
77 }
78 
GenerateMergingCode(io::Printer * printer)79 void MessageFieldGenerator::GenerateMergingCode(io::Printer* printer) {
80   printer->Print(
81     variables_,
82     "if (other.$has_property_check$) {\n"
83     "  if ($has_not_property_check$) {\n"
84     "    $name$_ = new $type_name$();\n"
85     "  }\n"
86     "  $property_name$.MergeFrom(other.$property_name$);\n"
87     "}\n");
88 }
89 
GenerateParsingCode(io::Printer * printer)90 void MessageFieldGenerator::GenerateParsingCode(io::Printer* printer) {
91   printer->Print(
92     variables_,
93     "if ($has_not_property_check$) {\n"
94     "  $name$_ = new $type_name$();\n"
95     "}\n"
96     // TODO(jonskeet): Do we really need merging behaviour like this?
97     "input.ReadMessage($name$_);\n"); // No need to support TYPE_GROUP...
98 }
99 
GenerateSerializationCode(io::Printer * printer)100 void MessageFieldGenerator::GenerateSerializationCode(io::Printer* printer) {
101   printer->Print(
102     variables_,
103     "if ($has_property_check$) {\n"
104     "  output.WriteRawTag($tag_bytes$);\n"
105     "  output.WriteMessage($property_name$);\n"
106     "}\n");
107 }
108 
GenerateSerializedSizeCode(io::Printer * printer)109 void MessageFieldGenerator::GenerateSerializedSizeCode(io::Printer* printer) {
110   printer->Print(
111     variables_,
112     "if ($has_property_check$) {\n"
113     "  size += $tag_size$ + pb::CodedOutputStream.ComputeMessageSize($property_name$);\n"
114     "}\n");
115 }
116 
WriteHash(io::Printer * printer)117 void MessageFieldGenerator::WriteHash(io::Printer* printer) {
118   printer->Print(
119     variables_,
120     "if ($has_property_check$) hash ^= $property_name$.GetHashCode();\n");
121 }
WriteEquals(io::Printer * printer)122 void MessageFieldGenerator::WriteEquals(io::Printer* printer) {
123   printer->Print(
124     variables_,
125     "if (!object.Equals($property_name$, other.$property_name$)) return false;\n");
126 }
WriteToString(io::Printer * printer)127 void MessageFieldGenerator::WriteToString(io::Printer* printer) {
128   variables_["field_name"] = GetFieldName(descriptor_);
129   printer->Print(
130     variables_,
131     "PrintField(\"$field_name$\", has$property_name$, $name$_, writer);\n");
132 }
133 
GenerateCloningCode(io::Printer * printer)134 void MessageFieldGenerator::GenerateCloningCode(io::Printer* printer) {
135   printer->Print(variables_,
136     "$property_name$ = other.$has_property_check$ ? other.$property_name$.Clone() : null;\n");
137 }
138 
GenerateFreezingCode(io::Printer * printer)139 void MessageFieldGenerator::GenerateFreezingCode(io::Printer* printer) {
140 }
141 
GenerateCodecCode(io::Printer * printer)142 void MessageFieldGenerator::GenerateCodecCode(io::Printer* printer) {
143   printer->Print(
144     variables_,
145     "pb::FieldCodec.ForMessage($tag$, $type_name$.Parser)");
146 }
147 
MessageOneofFieldGenerator(const FieldDescriptor * descriptor,int fieldOrdinal,const Options * options)148 MessageOneofFieldGenerator::MessageOneofFieldGenerator(
149     const FieldDescriptor* descriptor,
150 	  int fieldOrdinal,
151     const Options *options)
152     : MessageFieldGenerator(descriptor, fieldOrdinal, options) {
153   SetCommonOneofFieldVariables(&variables_);
154 }
155 
~MessageOneofFieldGenerator()156 MessageOneofFieldGenerator::~MessageOneofFieldGenerator() {
157 
158 }
159 
GenerateMembers(io::Printer * printer)160 void MessageOneofFieldGenerator::GenerateMembers(io::Printer* printer) {
161   WritePropertyDocComment(printer, descriptor_);
162   AddDeprecatedFlag(printer);
163   printer->Print(
164     variables_,
165     "$access_level$ $type_name$ $property_name$ {\n"
166     "  get { return $has_property_check$ ? ($type_name$) $oneof_name$_ : null; }\n"
167     "  set {\n"
168     "    $oneof_name$_ = value;\n"
169     "    $oneof_name$Case_ = value == null ? $oneof_property_name$OneofCase.None : $oneof_property_name$OneofCase.$property_name$;\n"
170     "  }\n"
171     "}\n");
172 }
173 
GenerateParsingCode(io::Printer * printer)174 void MessageOneofFieldGenerator::GenerateParsingCode(io::Printer* printer) {
175   // TODO(jonskeet): We may be able to do better than this
176   printer->Print(
177     variables_,
178     "$type_name$ subBuilder = new $type_name$();\n"
179     "if ($has_property_check$) {\n"
180     "  subBuilder.MergeFrom($property_name$);\n"
181     "}\n"
182     "input.ReadMessage(subBuilder);\n" // No support of TYPE_GROUP
183     "$property_name$ = subBuilder;\n");
184 }
185 
WriteToString(io::Printer * printer)186 void MessageOneofFieldGenerator::WriteToString(io::Printer* printer) {
187   printer->Print(
188     variables_,
189     "PrintField(\"$descriptor_name$\", $has_property_check$, $oneof_name$_, writer);\n");
190 }
191 
GenerateCloningCode(io::Printer * printer)192 void MessageOneofFieldGenerator::GenerateCloningCode(io::Printer* printer) {
193   printer->Print(variables_,
194     "$property_name$ = other.$property_name$.Clone();\n");
195 }
196 
197 }  // namespace csharp
198 }  // namespace compiler
199 }  // namespace protobuf
200 }  // namespace google
201