Commit fb96026b authored by Brian Duff's avatar Brian Duff

When no clear() is generated, still initialize fields.

https://android-review.googlesource.com/#/c/67890/ removed field
initialization from the ctor, making it just call clear() instead.

When I added the generate_clear option back (as part of the reftypes
compat mode) in https://android-review.googlesource.com/#/c/109530/,
I forgot to ensure that what clear() used to do was inlined in the
constructor.

This change fixes NPEs that are happening for users of
reftypes_compat_mode who rely on unset repeated fields being empty
arrays rather than null.

Change-Id: Idb58746c60f4a4054b7ebb5c3b0e76b16ff88184
parent 9ffaa50d
...@@ -139,6 +139,15 @@ ...@@ -139,6 +139,15 @@
<arg value="--proto_path=src/test/java/com" /> <arg value="--proto_path=src/test/java/com" />
<arg value="src/test/java/com/google/protobuf/nano/unittest_reference_types_nano.proto" /> <arg value="src/test/java/com/google/protobuf/nano/unittest_reference_types_nano.proto" />
</exec> </exec>
<exec executable="../src/protoc">
<arg value="--javanano_out=
optional_field_style=reftypes_compat_mode,
generate_equals=true,
java_outer_classname=google/protobuf/nano/unittest_reference_types_nano.proto|NanoReferenceTypesCompat
:target/generated-test-sources" />
<arg value="--proto_path=src/test/java/com" />
<arg value="src/test/java/com/google/protobuf/nano/unittest_reference_types_nano.proto" />
</exec>
</tasks> </tasks>
<testSourceRoot>target/generated-test-sources</testSourceRoot> <testSourceRoot>target/generated-test-sources</testSourceRoot>
</configuration> </configuration>
......
...@@ -36,6 +36,7 @@ import com.google.protobuf.nano.NanoAccessorsOuterClass.TestNanoAccessors; ...@@ -36,6 +36,7 @@ import com.google.protobuf.nano.NanoAccessorsOuterClass.TestNanoAccessors;
import com.google.protobuf.nano.NanoHasOuterClass.TestAllTypesNanoHas; import com.google.protobuf.nano.NanoHasOuterClass.TestAllTypesNanoHas;
import com.google.protobuf.nano.NanoOuterClass.TestAllTypesNano; import com.google.protobuf.nano.NanoOuterClass.TestAllTypesNano;
import com.google.protobuf.nano.UnittestRecursiveNano.RecursiveMessageNano; import com.google.protobuf.nano.UnittestRecursiveNano.RecursiveMessageNano;
import com.google.protobuf.nano.NanoReferenceTypesCompat;
import com.google.protobuf.nano.UnittestSimpleNano.SimpleMessageNano; import com.google.protobuf.nano.UnittestSimpleNano.SimpleMessageNano;
import com.google.protobuf.nano.UnittestSingleNano.SingleMessageNano; import com.google.protobuf.nano.UnittestSingleNano.SingleMessageNano;
import com.google.protobuf.nano.testext.Extensions; import com.google.protobuf.nano.testext.Extensions;
...@@ -4381,6 +4382,11 @@ public class NanoTest extends TestCase { ...@@ -4381,6 +4382,11 @@ public class NanoTest extends TestCase {
assertMapSet(testMap.sfixed64ToSfixed64Field, int64Values, int64Values); assertMapSet(testMap.sfixed64ToSfixed64Field, int64Values, int64Values);
} }
public void testRepeatedFieldInitializedInReftypesCompatMode() {
NanoReferenceTypesCompat.TestAllTypesNano proto = new NanoReferenceTypesCompat.TestAllTypesNano();
assertNotNull(proto.repeatedString);
}
private void assertRepeatedPackablesEqual( private void assertRepeatedPackablesEqual(
NanoRepeatedPackables.NonPacked nonPacked, NanoRepeatedPackables.Packed packed) { NanoRepeatedPackables.NonPacked nonPacked, NanoRepeatedPackables.Packed packed) {
// Not using MessageNano.equals() -- that belongs to a separate test. // Not using MessageNano.equals() -- that belongs to a separate test.
......
...@@ -288,14 +288,18 @@ void MessageGenerator::Generate(io::Printer* printer) { ...@@ -288,14 +288,18 @@ void MessageGenerator::Generate(io::Printer* printer) {
} }
printer->Print("}\n"); printer->Print("}\n");
} else { } else {
printer->Print(
"\n"
"public $classname$() {\n",
"classname", descriptor_->name());
if (params_.generate_clear()) { if (params_.generate_clear()) {
printer->Print( printer->Print(" clear();\n");
"\n" } else {
"public $classname$() {\n" printer->Indent();
" clear();\n" GenerateFieldInitializers(printer);
"}\n", printer->Outdent();
"classname", descriptor_->name());
} }
printer->Print("}\n");
} }
// Other methods in this class // Other methods in this class
...@@ -495,6 +499,15 @@ void MessageGenerator::GenerateClear(io::Printer* printer) { ...@@ -495,6 +499,15 @@ void MessageGenerator::GenerateClear(io::Printer* printer) {
"classname", descriptor_->name()); "classname", descriptor_->name());
printer->Indent(); printer->Indent();
GenerateFieldInitializers(printer);
printer->Outdent();
printer->Print(
" return this;\n"
"}\n");
}
void MessageGenerator::GenerateFieldInitializers(io::Printer* printer) {
// Clear bit fields. // Clear bit fields.
int totalInts = (field_generators_.total_bits() + 31) / 32; int totalInts = (field_generators_.total_bits() + 31) / 32;
for (int i = 0; i < totalInts; i++) { for (int i = 0; i < totalInts; i++) {
...@@ -520,12 +533,7 @@ void MessageGenerator::GenerateClear(io::Printer* printer) { ...@@ -520,12 +533,7 @@ void MessageGenerator::GenerateClear(io::Printer* printer) {
if (params_.store_unknown_fields()) { if (params_.store_unknown_fields()) {
printer->Print("unknownFieldData = null;\n"); printer->Print("unknownFieldData = null;\n");
} }
printer->Print("cachedSize = -1;\n");
printer->Outdent();
printer->Print(
" cachedSize = -1;\n"
" return this;\n"
"}\n");
} }
void MessageGenerator::GenerateEquals(io::Printer* printer) { void MessageGenerator::GenerateEquals(io::Printer* printer) {
......
...@@ -77,6 +77,7 @@ class MessageGenerator { ...@@ -77,6 +77,7 @@ class MessageGenerator {
const FieldDescriptor* field); const FieldDescriptor* field);
void GenerateClear(io::Printer* printer); void GenerateClear(io::Printer* printer);
void GenerateFieldInitializers(io::Printer* printer);
void GenerateEquals(io::Printer* printer); void GenerateEquals(io::Printer* printer);
void GenerateHashCode(io::Printer* printer); void GenerateHashCode(io::Printer* printer);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment