xref: /aosp_15_r20/external/cronet/third_party/protobuf/src/google/protobuf/compiler/java/extension.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
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: [email protected] (Kenton Varda)
32 //  Based on original Protocol Buffers design by
33 //  Sanjay Ghemawat, Jeff Dean, and others.
34 
35 #include <google/protobuf/compiler/java/extension.h>
36 
37 #include <google/protobuf/io/printer.h>
38 #include <google/protobuf/stubs/strutil.h>
39 #include <google/protobuf/compiler/java/context.h>
40 #include <google/protobuf/compiler/java/doc_comment.h>
41 #include <google/protobuf/compiler/java/helpers.h>
42 #include <google/protobuf/compiler/java/name_resolver.h>
43 
44 // Must be last.
45 #include <google/protobuf/port_def.inc>
46 
47 namespace google {
48 namespace protobuf {
49 namespace compiler {
50 namespace java {
51 
ImmutableExtensionGenerator(const FieldDescriptor * descriptor,Context * context)52 ImmutableExtensionGenerator::ImmutableExtensionGenerator(
53     const FieldDescriptor* descriptor, Context* context)
54     : descriptor_(descriptor), name_resolver_(context->GetNameResolver()) {
55   if (descriptor_->extension_scope() != NULL) {
56     scope_ =
57         name_resolver_->GetImmutableClassName(descriptor_->extension_scope());
58   } else {
59     scope_ = name_resolver_->GetImmutableClassName(descriptor_->file());
60   }
61 }
62 
~ImmutableExtensionGenerator()63 ImmutableExtensionGenerator::~ImmutableExtensionGenerator() {}
64 
65 // Initializes the vars referenced in the generated code templates.
InitTemplateVars(const FieldDescriptor * descriptor,const std::string & scope,bool immutable,ClassNameResolver * name_resolver,std::map<std::string,std::string> * vars_pointer)66 void ExtensionGenerator::InitTemplateVars(
67     const FieldDescriptor* descriptor, const std::string& scope, bool immutable,
68     ClassNameResolver* name_resolver,
69     std::map<std::string, std::string>* vars_pointer) {
70   std::map<std::string, std::string>& vars = *vars_pointer;
71   vars["scope"] = scope;
72   vars["name"] = UnderscoresToCamelCaseCheckReserved(descriptor);
73   vars["containing_type"] =
74       name_resolver->GetClassName(descriptor->containing_type(), immutable);
75   vars["number"] = StrCat(descriptor->number());
76   vars["constant_name"] = FieldConstantName(descriptor);
77   vars["index"] = StrCat(descriptor->index());
78   vars["default"] = descriptor->is_repeated()
79                         ? ""
80                         : DefaultValue(descriptor, immutable, name_resolver);
81   vars["type_constant"] = FieldTypeName(GetType(descriptor));
82   vars["packed"] = descriptor->is_packed() ? "true" : "false";
83   vars["enum_map"] = "null";
84   vars["prototype"] = "null";
85 
86   JavaType java_type = GetJavaType(descriptor);
87   std::string singular_type;
88   switch (java_type) {
89     case JAVATYPE_MESSAGE:
90       singular_type =
91           name_resolver->GetClassName(descriptor->message_type(), immutable);
92       vars["prototype"] = singular_type + ".getDefaultInstance()";
93       break;
94     case JAVATYPE_ENUM:
95       singular_type =
96           name_resolver->GetClassName(descriptor->enum_type(), immutable);
97       vars["enum_map"] = singular_type + ".internalGetValueMap()";
98       break;
99     case JAVATYPE_STRING:
100       singular_type = "java.lang.String";
101       break;
102     case JAVATYPE_BYTES:
103       singular_type = immutable ? "com.google.protobuf.ByteString" : "byte[]";
104       break;
105     default:
106       singular_type = BoxedPrimitiveTypeName(java_type);
107       break;
108   }
109   vars["type"] = descriptor->is_repeated()
110                      ? "java.util.List<" + singular_type + ">"
111                      : singular_type;
112   vars["singular_type"] = singular_type;
113 }
114 
Generate(io::Printer * printer)115 void ImmutableExtensionGenerator::Generate(io::Printer* printer) {
116   std::map<std::string, std::string> vars;
117   const bool kUseImmutableNames = true;
118   InitTemplateVars(descriptor_, scope_, kUseImmutableNames, name_resolver_,
119                    &vars);
120   printer->Print(vars, "public static final int $constant_name$ = $number$;\n");
121 
122   WriteFieldDocComment(printer, descriptor_);
123   if (descriptor_->extension_scope() == NULL) {
124     // Non-nested
125     printer->Print(
126         vars,
127         "public static final\n"
128         "  com.google.protobuf.GeneratedMessage.GeneratedExtension<\n"
129         "    $containing_type$,\n"
130         "    $type$> $name$ = com.google.protobuf.GeneratedMessage\n"
131         "        .newFileScopedGeneratedExtension(\n"
132         "      $singular_type$.class,\n"
133         "      $prototype$);\n");
134   } else {
135     // Nested
136     printer->Print(
137         vars,
138         "public static final\n"
139         "  com.google.protobuf.GeneratedMessage.GeneratedExtension<\n"
140         "    $containing_type$,\n"
141         "    $type$> $name$ = com.google.protobuf.GeneratedMessage\n"
142         "        .newMessageScopedGeneratedExtension(\n"
143         "      $scope$.getDefaultInstance(),\n"
144         "      $index$,\n"
145         "      $singular_type$.class,\n"
146         "      $prototype$);\n");
147   }
148   printer->Annotate("name", descriptor_);
149 }
150 
GenerateNonNestedInitializationCode(io::Printer * printer)151 int ImmutableExtensionGenerator::GenerateNonNestedInitializationCode(
152     io::Printer* printer) {
153   int bytecode_estimate = 0;
154   if (descriptor_->extension_scope() == NULL) {
155     // Only applies to non-nested extensions.
156     printer->Print(
157         "$name$.internalInit(descriptor.getExtensions().get($index$));\n",
158         "name", UnderscoresToCamelCaseCheckReserved(descriptor_), "index",
159         StrCat(descriptor_->index()));
160     bytecode_estimate += 21;
161   }
162   return bytecode_estimate;
163 }
164 
GenerateRegistrationCode(io::Printer * printer)165 int ImmutableExtensionGenerator::GenerateRegistrationCode(
166     io::Printer* printer) {
167   printer->Print("registry.add($scope$.$name$);\n", "scope", scope_, "name",
168                  UnderscoresToCamelCaseCheckReserved(descriptor_));
169   return 7;
170 }
171 
172 }  // namespace java
173 }  // namespace compiler
174 }  // namespace protobuf
175 }  // namespace google
176 
177 #include <google/protobuf/port_undef.inc>
178