Adding experimental runtime
This commit is contained in:
parent
787f3fb163
commit
0fdfe635e8
41 changed files with 12403 additions and 3 deletions
38
Makefile.am
38
Makefile.am
|
@ -187,6 +187,44 @@ csharp_EXTRA_DIST= \
|
|||
csharp/src/packages/repositories.config
|
||||
|
||||
java_EXTRA_DIST= \
|
||||
java/core/src/main/java/com/google/protobuf/AllocatedBuffer.java \
|
||||
java/core/src/main/java/com/google/protobuf/BinaryProtocolUtil.java \
|
||||
java/core/src/main/java/com/google/protobuf/BinaryReader.java \
|
||||
java/core/src/main/java/com/google/protobuf/BinaryWriter.java \
|
||||
java/core/src/main/java/com/google/protobuf/BufferAllocator.java \
|
||||
java/core/src/main/java/com/google/protobuf/FieldInfo.java \
|
||||
java/core/src/main/java/com/google/protobuf/Int2ObjectHashMap.java \
|
||||
java/core/src/main/java/com/google/protobuf/MessageInfo.java \
|
||||
java/core/src/main/java/com/google/protobuf/MessageInfoFactory.java \
|
||||
java/core/src/main/java/com/google/protobuf/Protobuf.java \
|
||||
java/core/src/main/java/com/google/protobuf/ProtobufLists.java \
|
||||
java/core/src/main/java/com/google/protobuf/Reader.java \
|
||||
java/core/src/main/java/com/google/protobuf/Schema.java \
|
||||
java/core/src/main/java/com/google/protobuf/SchemaFactory.java \
|
||||
java/core/src/main/java/com/google/protobuf/SchemaUtil.java \
|
||||
java/core/src/main/java/com/google/protobuf/Writer.java \
|
||||
java/core/src/main/java/com/google/protobuf/DescriptorMessageInfoFactory.java \
|
||||
java/core/src/main/java/com/google/protobuf/AbstractProto2LiteSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/AbstractProto2Schema.java \
|
||||
java/core/src/main/java/com/google/protobuf/AbstractProto2StandardSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/AbstractProto3LiteSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/AbstractProto3Schema.java \
|
||||
java/core/src/main/java/com/google/protobuf/AbstractProto3StandardSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto2LiteLookupSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto2LiteTableSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto2LookupSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto2Manifest.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto2SchemaFactory.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto2TableSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto3LiteLookupSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto3LiteTableSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto3LookupSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto3Manifest.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto3SchemaFactory.java \
|
||||
java/core/src/main/java/com/google/protobuf/Proto3TableSchema.java \
|
||||
java/core/src/main/java/com/google/protobuf/ProtoSyntax.java \
|
||||
java/core/src/main/java/com/google/protobuf/FieldType.java \
|
||||
java/core/src/main/java/com/google/protobuf/JavaType.java \
|
||||
java/README.md \
|
||||
java/core/generate-sources-build.xml \
|
||||
java/core/generate-test-sources-build.xml \
|
||||
|
|
|
@ -0,0 +1,183 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto2Manifest.FIELD_LENGTH;
|
||||
import static com.google.protobuf.Proto2Manifest.offset;
|
||||
import static com.google.protobuf.Proto2Manifest.type;
|
||||
|
||||
/** Base class for all proto2-lite-based schemas. */
|
||||
abstract class AbstractProto2LiteSchema<T> extends AbstractProto2Schema<T> {
|
||||
final Class<?> messageClass;
|
||||
|
||||
public AbstractProto2LiteSchema(Class<T> messageClass, Proto2Manifest manifest) {
|
||||
super(manifest);
|
||||
this.messageClass = messageClass;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Override
|
||||
public final T newInstance() {
|
||||
T msg = (T) UnsafeUtil.allocateInstance(messageClass);
|
||||
for (long pos = manifest.address; pos < manifest.limit; pos += FIELD_LENGTH) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
case 1: //FLOAT:
|
||||
case 2: //INT64:
|
||||
case 3: //UINT64:
|
||||
case 4: //INT32:
|
||||
case 5: //FIXED64:
|
||||
case 6: //FIXED32:
|
||||
case 7: //BOOL:
|
||||
case 9: //MESSAGE:
|
||||
case 11: //UINT32:
|
||||
case 12: //ENUM:
|
||||
case 13: //SFIXED32:
|
||||
case 14: //SFIXED64:
|
||||
case 15: //SINT32:
|
||||
case 16: //SINT64:
|
||||
// Do nothing, just use default values.
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), "");
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ByteString.EMPTY);
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), DoubleArrayList.emptyList());
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), FloatArrayList.emptyList());
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), BooleanArrayList.emptyList());
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LazyStringArrayList.EMPTY);
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ProtobufArrayList.emptyList());
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ProtobufArrayList.emptyList());
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), DoubleArrayList.emptyList());
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), FloatArrayList.emptyList());
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), BooleanArrayList.emptyList());
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case -4: //GROUP:
|
||||
break;
|
||||
case -3: //GROUP_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ProtobufArrayList.emptyList());
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Initialize the base class fields.
|
||||
SchemaUtil.initLiteBaseClassFields(msg);
|
||||
return msg;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,400 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto2Manifest.FIELD_LENGTH;
|
||||
import static com.google.protobuf.Proto2Manifest.offset;
|
||||
import static com.google.protobuf.Proto2Manifest.type;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/** Abstract base class for all proto3-based schemas. */
|
||||
abstract class AbstractProto2Schema<T> implements Schema<T> {
|
||||
protected final Proto2Manifest manifest;
|
||||
|
||||
public AbstractProto2Schema(Proto2Manifest manifest) {
|
||||
this.manifest = manifest;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Override
|
||||
public final void writeTo(T message, Writer writer) {
|
||||
for (long pos = manifest.address; pos < manifest.limit; pos += FIELD_LENGTH) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
|
||||
// Benchmarks have shown that switching on a byte is faster than an enum.
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeDouble(
|
||||
manifest.numberAt(pos), UnsafeUtil.getDouble(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 1: //FLOAT:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeFloat(
|
||||
manifest.numberAt(pos), UnsafeUtil.getFloat(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 2: //INT64:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeInt64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 3: //UINT64:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeUInt64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 4: //INT32:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeInt32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 5: //FIXED64:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeFixed64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 6: //FIXED32:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeFixed32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 7: //BOOL:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeBool(
|
||||
manifest.numberAt(pos), UnsafeUtil.getBoolean(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 8: //STRING:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writeString(
|
||||
manifest.numberAt(pos),
|
||||
UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
}
|
||||
break;
|
||||
case 9: //MESSAGE:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeMessage(
|
||||
manifest.numberAt(pos), UnsafeUtil.getObject(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeBytes(
|
||||
manifest.numberAt(pos),
|
||||
(ByteString) UnsafeUtil.getObject(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 11: //UINT32:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeUInt32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 12: //ENUM:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeEnum(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 13: //SFIXED32:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeSFixed32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 14: //SFIXED64:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeSFixed64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 15: //SINT32:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeSInt32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 16: //SINT64:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeSInt64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
SchemaUtil.writeDoubleList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Double>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
SchemaUtil.writeFloatList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Float>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
SchemaUtil.writeInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
SchemaUtil.writeUInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
SchemaUtil.writeInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
SchemaUtil.writeFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
SchemaUtil.writeFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
SchemaUtil.writeBoolList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Boolean>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
SchemaUtil.writeStringList(
|
||||
manifest.numberAt(pos),
|
||||
(List<String>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.writeMessageList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Double>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
SchemaUtil.writeBytesList(
|
||||
manifest.numberAt(pos),
|
||||
(List<ByteString>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
SchemaUtil.writeUInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
SchemaUtil.writeEnumList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
SchemaUtil.writeSFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
SchemaUtil.writeSFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
SchemaUtil.writeSInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
SchemaUtil.writeSInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
SchemaUtil.writeDoubleList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Double>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
SchemaUtil.writeFloatList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Float>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
SchemaUtil.writeInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
SchemaUtil.writeUInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
SchemaUtil.writeInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
SchemaUtil.writeFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
SchemaUtil.writeFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
SchemaUtil.writeBoolList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Boolean>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
SchemaUtil.writeUInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
SchemaUtil.writeEnumList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
SchemaUtil.writeSFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
SchemaUtil.writeSFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
SchemaUtil.writeSInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
SchemaUtil.writeSInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case -4: //GROUP:
|
||||
if (manifest.isFieldPresent(message, pos)) {
|
||||
writer.writeGroup(
|
||||
manifest.numberAt(pos), UnsafeUtil.getObject(message, offset(typeAndOffset)));
|
||||
}
|
||||
break;
|
||||
case -3: //GROUP_LIST:
|
||||
SchemaUtil.writeGroupList(
|
||||
manifest.numberAt(pos),
|
||||
(List<?>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
default:
|
||||
// Assume it's an empty entry - just go to the next entry.
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void writeString(int fieldNumber, Object value, Writer writer) {
|
||||
if (value instanceof String) {
|
||||
writer.writeString(fieldNumber, (String) value);
|
||||
} else {
|
||||
writer.writeBytes(fieldNumber, (ByteString) value);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,127 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto2Manifest.FIELD_LENGTH;
|
||||
import static com.google.protobuf.Proto2Manifest.offset;
|
||||
import static com.google.protobuf.Proto2Manifest.type;
|
||||
|
||||
import java.util.Collections;
|
||||
|
||||
/** Abtract base class for standard (i.e. non-lite) proto2 schemas. */
|
||||
abstract class AbstractProto2StandardSchema<T> extends AbstractProto2Schema<T> {
|
||||
final Class<?> messageClass;
|
||||
|
||||
public AbstractProto2StandardSchema(Class<T> messageClass, Proto2Manifest manifest) {
|
||||
super(manifest);
|
||||
this.messageClass = messageClass;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Override
|
||||
public final T newInstance() {
|
||||
T msg = (T) UnsafeUtil.allocateInstance(messageClass);
|
||||
for (long pos = manifest.address; pos < manifest.limit; pos += FIELD_LENGTH) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
case 1: //FLOAT:
|
||||
case 2: //INT64:
|
||||
case 3: //UINT64:
|
||||
case 4: //INT32:
|
||||
case 5: //FIXED64:
|
||||
case 6: //FIXED32:
|
||||
case 7: //BOOL:
|
||||
case 9: //MESSAGE:
|
||||
case 11: //UINT32:
|
||||
case 12: //ENUM:
|
||||
case 13: //SFIXED32:
|
||||
case 14: //SFIXED64:
|
||||
case 15: //SINT32:
|
||||
case 16: //SINT64:
|
||||
// Do nothing, just use default values.
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), "");
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ByteString.EMPTY);
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
case 18: //FLOAT_LIST:
|
||||
case 19: //INT64_LIST:
|
||||
case 20: //UINT64_LIST:
|
||||
case 21: //INT32_LIST:
|
||||
case 22: //FIXED64_LIST:
|
||||
case 23: //FIXED32_LIST:
|
||||
case 24: //BOOL_LIST:
|
||||
case 26: //MESSAGE_LIST:
|
||||
case 27: //BYTES_LIST:
|
||||
case 28: //UINT32_LIST:
|
||||
case 29: //ENUM_LIST:
|
||||
case 30: //SFIXED32_LIST:
|
||||
case 31: //SFIXED64_LIST:
|
||||
case 32: //SINT32_LIST:
|
||||
case 33: //SINT64_LIST:
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), Collections.emptyList());
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LazyStringArrayList.EMPTY);
|
||||
break;
|
||||
case -4: //GROUP
|
||||
break;
|
||||
case -3: //GROUP_LIST
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), Collections.emptyList());
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Initialize the base class fields.
|
||||
SchemaUtil.initBaseClassFields(msg);
|
||||
return msg;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,178 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto3Manifest.FIELD_LENGTH;
|
||||
import static com.google.protobuf.Proto3Manifest.offset;
|
||||
import static com.google.protobuf.Proto3Manifest.type;
|
||||
|
||||
/** Base class for all proto3-lite-based schemas. */
|
||||
abstract class AbstractProto3LiteSchema<T> extends AbstractProto3Schema<T> {
|
||||
final Class<?> messageClass;
|
||||
|
||||
public AbstractProto3LiteSchema(Class<T> messageClass, Proto3Manifest manifest) {
|
||||
super(manifest);
|
||||
this.messageClass = messageClass;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Override
|
||||
public final T newInstance() {
|
||||
T msg = (T) UnsafeUtil.allocateInstance(messageClass);
|
||||
for (long pos = manifest.address; pos < manifest.limit; pos += FIELD_LENGTH) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
case 1: //FLOAT:
|
||||
case 2: //INT64:
|
||||
case 3: //UINT64:
|
||||
case 4: //INT32:
|
||||
case 5: //FIXED64:
|
||||
case 6: //FIXED32:
|
||||
case 7: //BOOL:
|
||||
case 9: //MESSAGE:
|
||||
case 11: //UINT32:
|
||||
case 12: //ENUM:
|
||||
case 13: //SFIXED32:
|
||||
case 14: //SFIXED64:
|
||||
case 15: //SINT32:
|
||||
case 16: //SINT64:
|
||||
// Do nothing, just use default values.
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), "");
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ByteString.EMPTY);
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), DoubleArrayList.emptyList());
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), FloatArrayList.emptyList());
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), BooleanArrayList.emptyList());
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LazyStringArrayList.EMPTY);
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ProtobufArrayList.emptyList());
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ProtobufArrayList.emptyList());
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), DoubleArrayList.emptyList());
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), FloatArrayList.emptyList());
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), BooleanArrayList.emptyList());
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), IntArrayList.emptyList());
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LongArrayList.emptyList());
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Initialize the base class fields.
|
||||
SchemaUtil.initLiteBaseClassFields(msg);
|
||||
return msg;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,347 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto3Manifest.FIELD_LENGTH;
|
||||
import static com.google.protobuf.Proto3Manifest.offset;
|
||||
import static com.google.protobuf.Proto3Manifest.type;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/** Abstract base class for all proto3-based schemas. */
|
||||
abstract class AbstractProto3Schema<T> implements Schema<T> {
|
||||
protected final Proto3Manifest manifest;
|
||||
|
||||
public AbstractProto3Schema(Proto3Manifest manifest) {
|
||||
this.manifest = manifest;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Override
|
||||
public final void writeTo(T message, Writer writer) {
|
||||
for (long pos = manifest.address; pos < manifest.limit; pos += FIELD_LENGTH) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
|
||||
// Benchmarks have shown that switching on a byte is faster than an enum.
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
SchemaUtil.writeDouble(
|
||||
manifest.numberAt(pos), UnsafeUtil.getDouble(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 1: //FLOAT:
|
||||
SchemaUtil.writeFloat(
|
||||
manifest.numberAt(pos), UnsafeUtil.getFloat(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 2: //INT64:
|
||||
SchemaUtil.writeInt64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 3: //UINT64:
|
||||
SchemaUtil.writeUInt64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 4: //INT32:
|
||||
SchemaUtil.writeInt32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 5: //FIXED64:
|
||||
SchemaUtil.writeFixed64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 6: //FIXED32:
|
||||
SchemaUtil.writeFixed32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 7: //BOOL:
|
||||
SchemaUtil.writeBool(
|
||||
manifest.numberAt(pos),
|
||||
UnsafeUtil.getBoolean(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 8: //STRING:
|
||||
SchemaUtil.writeString(
|
||||
manifest.numberAt(pos), UnsafeUtil.getObject(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 9: //MESSAGE:
|
||||
SchemaUtil.writeMessage(
|
||||
manifest.numberAt(pos), UnsafeUtil.getObject(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
SchemaUtil.writeBytes(
|
||||
manifest.numberAt(pos),
|
||||
(ByteString) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 11: //UINT32:
|
||||
SchemaUtil.writeUInt32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 12: //ENUM:
|
||||
SchemaUtil.writeEnum(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 13: //SFIXED32:
|
||||
SchemaUtil.writeSFixed32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 14: //SFIXED64:
|
||||
SchemaUtil.writeSFixed64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 15: //SINT32:
|
||||
SchemaUtil.writeSInt32(
|
||||
manifest.numberAt(pos), UnsafeUtil.getInt(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 16: //SINT64:
|
||||
SchemaUtil.writeSInt64(
|
||||
manifest.numberAt(pos), UnsafeUtil.getLong(message, offset(typeAndOffset)), writer);
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
SchemaUtil.writeDoubleList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Double>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
SchemaUtil.writeFloatList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Float>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
SchemaUtil.writeInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
SchemaUtil.writeUInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
SchemaUtil.writeInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
SchemaUtil.writeFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
SchemaUtil.writeFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
SchemaUtil.writeBoolList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Boolean>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
SchemaUtil.writeStringList(
|
||||
manifest.numberAt(pos),
|
||||
(List<String>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.writeMessageList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Double>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
SchemaUtil.writeBytesList(
|
||||
manifest.numberAt(pos),
|
||||
(List<ByteString>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer);
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
SchemaUtil.writeUInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
SchemaUtil.writeEnumList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
SchemaUtil.writeSFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
SchemaUtil.writeSFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
SchemaUtil.writeSInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
SchemaUtil.writeSInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
false);
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
SchemaUtil.writeDoubleList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Double>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
SchemaUtil.writeFloatList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Float>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
SchemaUtil.writeInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
SchemaUtil.writeUInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
SchemaUtil.writeInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
SchemaUtil.writeFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
SchemaUtil.writeFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
SchemaUtil.writeBoolList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Boolean>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
SchemaUtil.writeUInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
SchemaUtil.writeEnumList(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
SchemaUtil.writeSFixed32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
SchemaUtil.writeSFixed64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
SchemaUtil.writeSInt32List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Integer>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
SchemaUtil.writeSInt64List(
|
||||
manifest.numberAt(pos),
|
||||
(List<Long>) UnsafeUtil.getObject(message, offset(typeAndOffset)),
|
||||
writer,
|
||||
true);
|
||||
break;
|
||||
default:
|
||||
// Assume it's an empty entry - just go to the next entry.
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,124 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto3Manifest.FIELD_LENGTH;
|
||||
import static com.google.protobuf.Proto3Manifest.offset;
|
||||
import static com.google.protobuf.Proto3Manifest.type;
|
||||
|
||||
import java.util.Collections;
|
||||
|
||||
/**
|
||||
* Abtract base class for standard (i.e. non-lite) proto3 schemas.
|
||||
*/
|
||||
abstract class AbstractProto3StandardSchema<T> extends AbstractProto3Schema<T> {
|
||||
final Class<?> messageClass;
|
||||
|
||||
public AbstractProto3StandardSchema(Class<T> messageClass, Proto3Manifest manifest) {
|
||||
super(manifest);
|
||||
this.messageClass = messageClass;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Override
|
||||
public final T newInstance() {
|
||||
T msg = (T) UnsafeUtil.allocateInstance(messageClass);
|
||||
for (long pos = manifest.address; pos < manifest.limit; pos += FIELD_LENGTH) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
case 1: //FLOAT:
|
||||
case 2: //INT64:
|
||||
case 3: //UINT64:
|
||||
case 4: //INT32:
|
||||
case 5: //FIXED64:
|
||||
case 6: //FIXED32:
|
||||
case 7: //BOOL:
|
||||
case 9: //MESSAGE:
|
||||
case 11: //UINT32:
|
||||
case 12: //ENUM:
|
||||
case 13: //SFIXED32:
|
||||
case 14: //SFIXED64:
|
||||
case 15: //SINT32:
|
||||
case 16: //SINT64:
|
||||
// Do nothing, just use default values.
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), "");
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), ByteString.EMPTY);
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
case 18: //FLOAT_LIST:
|
||||
case 19: //INT64_LIST:
|
||||
case 20: //UINT64_LIST:
|
||||
case 21: //INT32_LIST:
|
||||
case 22: //FIXED64_LIST:
|
||||
case 23: //FIXED32_LIST:
|
||||
case 24: //BOOL_LIST:
|
||||
case 26: //MESSAGE_LIST:
|
||||
case 27: //BYTES_LIST:
|
||||
case 28: //UINT32_LIST:
|
||||
case 29: //ENUM_LIST:
|
||||
case 30: //SFIXED32_LIST:
|
||||
case 31: //SFIXED64_LIST:
|
||||
case 32: //SINT32_LIST:
|
||||
case 33: //SINT64_LIST:
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), Collections.emptyList());
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
UnsafeUtil.putObject(msg, offset(typeAndOffset), LazyStringArrayList.EMPTY);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Initialize the base class fields.
|
||||
SchemaUtil.initBaseClassFields(msg);
|
||||
return msg;
|
||||
}
|
||||
}
|
263
java/core/src/main/java/com/google/protobuf/AllocatedBuffer.java
Normal file
263
java/core/src/main/java/com/google/protobuf/AllocatedBuffer.java
Normal file
|
@ -0,0 +1,263 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Internal.checkNotNull;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
|
||||
/**
|
||||
* A buffer that was allocated by a {@link BufferAllocator}. For every buffer, it is guaranteed that
|
||||
* at least one of {@link #hasArray()} or {@link #hasNioBuffer()} will be {@code true}.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public abstract class AllocatedBuffer {
|
||||
/**
|
||||
* Indicates whether this buffer contains a backing {@link ByteBuffer} (i.e. it is safe to call
|
||||
* {@link #nioBuffer()}).
|
||||
*/
|
||||
public abstract boolean hasNioBuffer();
|
||||
|
||||
/**
|
||||
* Indicates whether this buffer contains a backing array (i.e. it is safe to call {@link
|
||||
* #array()}).
|
||||
*/
|
||||
public abstract boolean hasArray();
|
||||
|
||||
/**
|
||||
* Returns the {@link ByteBuffer} that backs this buffer <i>(optional operation)</i>.
|
||||
*
|
||||
* <p>Call {@link #hasNioBuffer()} before invoking this method in order to ensure that this buffer
|
||||
* has a backing {@link ByteBuffer}.
|
||||
*
|
||||
* @return The {@link ByteBuffer} that backs this buffer
|
||||
* @throws UnsupportedOperationException If this buffer is not backed by a {@link ByteBuffer}.
|
||||
*/
|
||||
public abstract ByteBuffer nioBuffer();
|
||||
|
||||
/**
|
||||
* Returns the byte array that backs this buffer <i>(optional operation)</i>.
|
||||
*
|
||||
* <p>Call {@link #hasArray()} before invoking this method in order to ensure that this buffer has
|
||||
* an accessible backing array.
|
||||
*
|
||||
* @return The array that backs this buffer
|
||||
* @throws java.nio.ReadOnlyBufferException If this buffer is backed by an array but is read-only
|
||||
* @throws UnsupportedOperationException If this buffer is not backed by an accessible array
|
||||
*/
|
||||
public abstract byte[] array();
|
||||
|
||||
/**
|
||||
* Returns the offset within this buffer's backing array of the first element of the buffer
|
||||
* <i>(optional operation)</i>.
|
||||
*
|
||||
* <p>If this buffer is backed by an array then {@link #position()} corresponds to the array index
|
||||
* {@link #position()} {@code +} {@link #arrayOffset()}.
|
||||
*
|
||||
* <p>Invoke the {@link #hasArray hasArray} method before invoking this method in order to ensure
|
||||
* that this buffer has an accessible backing array.
|
||||
*
|
||||
* @return The offset within this buffer's array of the first element of the buffer
|
||||
* @throws java.nio.ReadOnlyBufferException If this buffer is backed by an array but is read-only
|
||||
* @throws UnsupportedOperationException If this buffer is not backed by an accessible array
|
||||
*/
|
||||
public abstract int arrayOffset();
|
||||
|
||||
/**
|
||||
* Returns this buffer's position.
|
||||
*
|
||||
* @return The position of this buffer
|
||||
*/
|
||||
public abstract int position();
|
||||
|
||||
/**
|
||||
* Sets this buffer's position.
|
||||
*
|
||||
* @param position The new position value; must be non-negative and no larger than the current
|
||||
* limit
|
||||
* @return This buffer
|
||||
* @throws IllegalArgumentException If the preconditions on {@code position} do not hold
|
||||
*/
|
||||
public abstract AllocatedBuffer position(int position);
|
||||
|
||||
/**
|
||||
* Returns this buffer's limit.
|
||||
*
|
||||
* @return The limit of this buffer
|
||||
*/
|
||||
public abstract int limit();
|
||||
|
||||
/**
|
||||
* Returns the number of elements between the current {@link #position()} and the {@link #limit()}
|
||||
* .
|
||||
*
|
||||
* @return The number of elements remaining in this buffer
|
||||
*/
|
||||
public abstract int remaining();
|
||||
|
||||
/**
|
||||
* Creates a new {@link AllocatedBuffer} that is backed by the given array. The returned buffer
|
||||
* will have {@link #hasArray} == {@code true}, {@link #arrayOffset()} == {@code 0}, {@link
|
||||
* #position()} == {@code 0} and {@link #limit()} equal to the length of {@code bytes}.
|
||||
*/
|
||||
public static AllocatedBuffer wrap(byte[] bytes) {
|
||||
return wrapNoCheck(bytes, 0, bytes.length);
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates a new {@link AllocatedBuffer} that is backed by the given array. The returned buffer
|
||||
* will have {@link #hasArray} == {@code true}, {@link #arrayOffset()} == {@code offset}, {@link
|
||||
* #position()} == {@code 0} and {@link #limit()} == {@code length}.
|
||||
*/
|
||||
public static AllocatedBuffer wrap(final byte[] bytes, final int offset, final int length) {
|
||||
if (offset < 0 || length < 0 || (offset + length) > bytes.length) {
|
||||
throw new IndexOutOfBoundsException(
|
||||
String.format("bytes.length=%d, offset=%d, lenght=%d", bytes.length, offset, length));
|
||||
}
|
||||
|
||||
return wrapNoCheck(bytes, offset, length);
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates a new {@link AllocatedBuffer} that is backed by the given {@link ByteBuffer}. The
|
||||
* returned buffer will have {@link #hasNioBuffer} == {@code true}.
|
||||
*/
|
||||
public static AllocatedBuffer wrap(final ByteBuffer buffer) {
|
||||
checkNotNull(buffer, "buffer");
|
||||
|
||||
return new AllocatedBuffer() {
|
||||
|
||||
@Override
|
||||
public boolean hasNioBuffer() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ByteBuffer nioBuffer() {
|
||||
return buffer;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean hasArray() {
|
||||
return buffer.hasArray();
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] array() {
|
||||
return buffer.array();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int arrayOffset() {
|
||||
return buffer.arrayOffset();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int position() {
|
||||
return buffer.position();
|
||||
}
|
||||
|
||||
@Override
|
||||
public AllocatedBuffer position(int position) {
|
||||
buffer.position(position);
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int limit() {
|
||||
return buffer.limit();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int remaining() {
|
||||
return buffer.remaining();
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private static AllocatedBuffer wrapNoCheck(
|
||||
final byte[] bytes, final int offset, final int length) {
|
||||
return new AllocatedBuffer() {
|
||||
// Relative to offset.
|
||||
private int position;
|
||||
|
||||
@Override
|
||||
public boolean hasNioBuffer() {
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ByteBuffer nioBuffer() {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean hasArray() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public byte[] array() {
|
||||
return bytes;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int arrayOffset() {
|
||||
return offset;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int position() {
|
||||
return position;
|
||||
}
|
||||
|
||||
@Override
|
||||
public AllocatedBuffer position(int position) {
|
||||
if (position < 0 || position > length) {
|
||||
throw new IllegalArgumentException("Invalid position: " + position);
|
||||
}
|
||||
this.position = position;
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int limit() {
|
||||
// Relative to offset.
|
||||
return length;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int remaining() {
|
||||
return length - position;
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
|
@ -0,0 +1,117 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
/** Common methods and constants for the binary protobuf protocol. */
|
||||
@ExperimentalApi
|
||||
final class BinaryProtocolUtil {
|
||||
private BinaryProtocolUtil() {}
|
||||
|
||||
private static final int TAG_TYPE_BITS = 3;
|
||||
private static final int TAG_TYPE_MASK = (1 << TAG_TYPE_BITS) - 1;
|
||||
|
||||
static final byte WIRETYPE_VARINT = 0;
|
||||
static final byte WIRETYPE_FIXED64 = 1;
|
||||
static final byte WIRETYPE_LENGTH_DELIMITED = 2;
|
||||
static final byte WIRETYPE_START_GROUP = 3;
|
||||
static final byte WIRETYPE_END_GROUP = 4;
|
||||
static final byte WIRETYPE_FIXED32 = 5;
|
||||
static final byte MAX_VARINT32_SIZE = 5;
|
||||
static final byte MAX_VARINT64_SIZE = 10;
|
||||
static final byte FIXED32_SIZE = 4;
|
||||
static final byte FIXED64_SIZE = 8;
|
||||
|
||||
static int tagFor(int fieldNumber, byte wireType) {
|
||||
return (fieldNumber << TAG_TYPE_BITS) | wireType;
|
||||
}
|
||||
|
||||
static int getFieldNumber(final int tag) {
|
||||
return tag >>> TAG_TYPE_BITS;
|
||||
}
|
||||
|
||||
static int getWireType(final int tag) {
|
||||
return tag & TAG_TYPE_MASK;
|
||||
}
|
||||
|
||||
/**
|
||||
* Decode a ZigZag-encoded 32-bit value. ZigZag encodes signed integers into values that can be
|
||||
* efficiently encoded with varint. (Otherwise, negative values must be sign-extended to 64 bits
|
||||
* to be varint encoded, thus always taking 10 bytes on the wire.)
|
||||
*
|
||||
* @param n An unsigned 32-bit integer, stored in a signed int because Java has no explicit
|
||||
* unsigned support.
|
||||
* @return A signed 32-bit integer.
|
||||
*/
|
||||
static int decodeZigZag32(final int n) {
|
||||
return (n >>> 1) ^ -(n & 1);
|
||||
}
|
||||
|
||||
/**
|
||||
* Decode a ZigZag-encoded 64-bit value. ZigZag encodes signed integers into values that can be
|
||||
* efficiently encoded with varint. (Otherwise, negative values must be sign-extended to 64 bits
|
||||
* to be varint encoded, thus always taking 10 bytes on the wire.)
|
||||
*
|
||||
* @param n An unsigned 64-bit integer, stored in a signed int because Java has no explicit
|
||||
* unsigned support.
|
||||
* @return A signed 64-bit integer.
|
||||
*/
|
||||
static long decodeZigZag64(final long n) {
|
||||
return (n >>> 1) ^ -(n & 1);
|
||||
}
|
||||
|
||||
/**
|
||||
* Encode a ZigZag-encoded 32-bit value. ZigZag encodes signed integers into values that can be
|
||||
* efficiently encoded with varint. (Otherwise, negative values must be sign-extended to 64 bits
|
||||
* to be varint encoded, thus always taking 10 bytes on the wire.)
|
||||
*
|
||||
* @param n A signed 32-bit integer.
|
||||
* @return An unsigned 32-bit integer, stored in a signed int because Java has no explicit
|
||||
* unsigned support.
|
||||
*/
|
||||
static int encodeZigZag32(final int n) {
|
||||
// Note: the right-shift must be arithmetic
|
||||
return (n << 1) ^ (n >> 31);
|
||||
}
|
||||
|
||||
/**
|
||||
* Encode a ZigZag-encoded 64-bit value. ZigZag encodes signed integers into values that can be
|
||||
* efficiently encoded with varint. (Otherwise, negative values must be sign-extended to 64 bits
|
||||
* to be varint encoded, thus always taking 10 bytes on the wire.)
|
||||
*
|
||||
* @param n A signed 64-bit integer.
|
||||
* @return An unsigned 64-bit integer, stored in a signed int because Java has no explicit
|
||||
* unsigned support.
|
||||
*/
|
||||
static long encodeZigZag64(final long n) {
|
||||
// Note: the right-shift must be arithmetic
|
||||
return (n << 1) ^ (n >> 63);
|
||||
}
|
||||
}
|
1516
java/core/src/main/java/com/google/protobuf/BinaryReader.java
Normal file
1516
java/core/src/main/java/com/google/protobuf/BinaryReader.java
Normal file
File diff suppressed because it is too large
Load diff
2804
java/core/src/main/java/com/google/protobuf/BinaryWriter.java
Normal file
2804
java/core/src/main/java/com/google/protobuf/BinaryWriter.java
Normal file
File diff suppressed because it is too large
Load diff
|
@ -0,0 +1,71 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
|
||||
/**
|
||||
* An object responsible for allocation of buffers. This is an extension point to enable buffer
|
||||
* pooling within an application.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public abstract class BufferAllocator {
|
||||
private static final BufferAllocator UNPOOLED =
|
||||
new BufferAllocator() {
|
||||
@Override
|
||||
public AllocatedBuffer allocateHeapBuffer(int capacity) {
|
||||
return AllocatedBuffer.wrap(new byte[capacity]);
|
||||
}
|
||||
|
||||
@Override
|
||||
public AllocatedBuffer allocateDirectBuffer(int capacity) {
|
||||
return AllocatedBuffer.wrap(ByteBuffer.allocateDirect(capacity));
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Returns an unpooled buffer allocator, which will create a new buffer for each request.
|
||||
*/
|
||||
public static BufferAllocator unpooled() {
|
||||
return UNPOOLED;
|
||||
}
|
||||
|
||||
/**
|
||||
* Allocates a buffer with the given capacity that is backed by an array on the heap.
|
||||
*/
|
||||
public abstract AllocatedBuffer allocateHeapBuffer(int capacity);
|
||||
|
||||
/**
|
||||
* Allocates a direct (i.e. non-heap) buffer with the given capacity.
|
||||
*/
|
||||
public abstract AllocatedBuffer allocateDirectBuffer(int capacity);
|
||||
}
|
||||
|
|
@ -28,6 +28,7 @@
|
|||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.io.ByteArrayInputStream;
|
||||
|
@ -52,9 +53,9 @@ import java.util.NoSuchElementException;
|
|||
|
||||
/**
|
||||
* Immutable sequence of bytes. Substring is supported by sharing the reference
|
||||
* to the immutable underlying bytes. Concatenation is likewise supported
|
||||
* without copying (long strings) by building a tree of pieces in
|
||||
* {@link RopeByteString}.
|
||||
* to the immutable underlying bytes, as with {@link String}. Concatenation is
|
||||
* likewise supported without copying (long strings) by building a tree of
|
||||
* pieces in {@link RopeByteString}.
|
||||
* <p>
|
||||
* Like {@link String}, the contents of a {@link ByteString} can never be
|
||||
* observed to change, not even in the presence of a data race or incorrect
|
||||
|
@ -691,6 +692,16 @@ public abstract class ByteString implements Iterable<Byte>, Serializable {
|
|||
*/
|
||||
abstract void writeTo(ByteOutput byteOutput) throws IOException;
|
||||
|
||||
/**
|
||||
* This method behaves exactly the same as {@link #writeTo(ByteOutput)} unless the {@link
|
||||
* ByteString} is a rope. For ropes, the leaf nodes are written in reverse order to the {@code
|
||||
* byteOutput}.
|
||||
*
|
||||
* @param byteOutput the output target to receive the bytes
|
||||
* @throws IOException if an I/O error occurs
|
||||
* @see UnsafeByteOperations#unsafeWriteToReverse(ByteString, ByteOutput)
|
||||
*/
|
||||
abstract void writeToReverse(ByteOutput byteOutput) throws IOException;
|
||||
|
||||
/**
|
||||
* Constructs a read-only {@code java.nio.ByteBuffer} whose content
|
||||
|
@ -833,6 +844,10 @@ public abstract class ByteString implements Iterable<Byte>, Serializable {
|
|||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
void writeToReverse(ByteOutput byteOutput) throws IOException {
|
||||
writeTo(byteOutput);
|
||||
}
|
||||
|
||||
/**
|
||||
* Check equality of the substring of given length of this object starting at
|
||||
|
|
|
@ -0,0 +1,264 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.FieldInfo.forField;
|
||||
import static com.google.protobuf.FieldInfo.forPresenceCheckedField;
|
||||
|
||||
import com.google.protobuf.Descriptors.Descriptor;
|
||||
import com.google.protobuf.Descriptors.FieldDescriptor;
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.Method;
|
||||
import java.util.List;
|
||||
|
||||
/** A factory for message info based on protobuf descriptors for a {@link GeneratedMessage}. */
|
||||
@ExperimentalApi
|
||||
public final class DescriptorMessageInfoFactory implements MessageInfoFactory {
|
||||
private static final String GET_DEFAULT_INSTANCE_METHOD_NAME = "getDefaultInstance";
|
||||
private static final DescriptorMessageInfoFactory instance = new DescriptorMessageInfoFactory();
|
||||
|
||||
// Disallow construction - it's a singleton.
|
||||
private DescriptorMessageInfoFactory() {
|
||||
}
|
||||
|
||||
public static DescriptorMessageInfoFactory getInstance() {
|
||||
return instance;
|
||||
}
|
||||
|
||||
@Override
|
||||
public MessageInfo messageInfoFor(Class<?> messageType) {
|
||||
if (!GeneratedMessage.class.isAssignableFrom(messageType)) {
|
||||
throw new IllegalArgumentException("Only generated protobuf messages are supported");
|
||||
}
|
||||
|
||||
return convert(messageType, descriptorForType(messageType));
|
||||
}
|
||||
|
||||
private static Descriptor descriptorForType(Class<?> messageType) {
|
||||
try {
|
||||
Method method = messageType.getDeclaredMethod(GET_DEFAULT_INSTANCE_METHOD_NAME);
|
||||
GeneratedMessage message = (GeneratedMessage) method.invoke(null);
|
||||
return message.getDescriptorForType();
|
||||
} catch (Exception e) {
|
||||
throw new IllegalArgumentException(
|
||||
"Unable to get default instance for message class " + messageType.getName(), e);
|
||||
}
|
||||
}
|
||||
|
||||
private static MessageInfo convert(Class<?> messageType, Descriptor desc) {
|
||||
switch (desc.getFile().getSyntax()) {
|
||||
case PROTO2:
|
||||
return convertProto2(messageType, desc);
|
||||
case PROTO3:
|
||||
return convertProto3(messageType, desc);
|
||||
default:
|
||||
throw new IllegalArgumentException("Unsupported syntax: " + desc.getFile().getSyntax());
|
||||
}
|
||||
}
|
||||
|
||||
private static MessageInfo convertProto2(Class<?> messageType, Descriptor desc) {
|
||||
List<FieldDescriptor> fieldDescriptors = desc.getFields();
|
||||
MessageInfo.Builder builder = MessageInfo.newBuilder(fieldDescriptors.size());
|
||||
builder.withSyntax(ProtoSyntax.PROTO2);
|
||||
|
||||
int bitFieldIndex = 0;
|
||||
int presenceMask = 1;
|
||||
Field bitField = bitField(messageType, bitFieldIndex++);
|
||||
|
||||
// Fields in the descriptor are ordered by the index position in which they appear in the
|
||||
// proto file. This is the same order used to determine the presence mask used in the
|
||||
// bitFields. So to determine the appropriate presence mask to be used for a field, we simply
|
||||
// need to shift the presence mask whenever a presence-checked field is encountered.
|
||||
for (int i = 0; i < fieldDescriptors.size(); ++i) {
|
||||
FieldDescriptor fd = fieldDescriptors.get(i);
|
||||
Field field = field(messageType, fd);
|
||||
int number = fd.getNumber();
|
||||
FieldType type = getFieldType(fd);
|
||||
|
||||
if (fd.isRepeated()) {
|
||||
// Repeated fields are not presence-checked.
|
||||
builder.add(forField(field, number, type));
|
||||
continue;
|
||||
}
|
||||
|
||||
// It's a presence-checked field.
|
||||
builder.add(forPresenceCheckedField(field, number, type, bitField, presenceMask));
|
||||
|
||||
// Update the presence mask/bitfield
|
||||
presenceMask <<= 1;
|
||||
if (presenceMask == 0) {
|
||||
// We've assigned all of the bits in the current bitField. Advance to the next one.
|
||||
bitField = bitField(messageType, bitFieldIndex++);
|
||||
}
|
||||
}
|
||||
|
||||
return builder.build();
|
||||
}
|
||||
|
||||
private static MessageInfo convertProto3(Class<?> messageType, Descriptor desc) {
|
||||
List<FieldDescriptor> fieldDescriptors = desc.getFields();
|
||||
MessageInfo.Builder builder = MessageInfo.newBuilder(fieldDescriptors.size());
|
||||
builder.withSyntax(ProtoSyntax.PROTO3);
|
||||
for (int i = 0; i < fieldDescriptors.size(); ++i) {
|
||||
FieldDescriptor fd = fieldDescriptors.get(i);
|
||||
builder.add(forField(field(messageType, fd), fd.getNumber(), getFieldType(fd)));
|
||||
}
|
||||
|
||||
return builder.build();
|
||||
}
|
||||
|
||||
private static FieldType getFieldType(FieldDescriptor fd) {
|
||||
switch (fd.getType()) {
|
||||
case BOOL:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.BOOL;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.BOOL_LIST_PACKED : FieldType.BOOL_LIST;
|
||||
case BYTES:
|
||||
return fd.isRepeated() ? FieldType.BYTES_LIST : FieldType.BYTES;
|
||||
case DOUBLE:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.DOUBLE;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.DOUBLE_LIST_PACKED : FieldType.DOUBLE_LIST;
|
||||
case ENUM:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.ENUM;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.ENUM_LIST_PACKED : FieldType.ENUM_LIST;
|
||||
case FIXED32:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.FIXED32;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.FIXED32_LIST_PACKED : FieldType.FIXED32_LIST;
|
||||
case FIXED64:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.FIXED64;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.FIXED64_LIST_PACKED : FieldType.FIXED64_LIST;
|
||||
case FLOAT:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.FLOAT;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.FLOAT_LIST_PACKED : FieldType.FLOAT_LIST;
|
||||
case GROUP:
|
||||
return fd.isRepeated() ? FieldType.GROUP_LIST : FieldType.GROUP;
|
||||
case INT32:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.INT32;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.INT32_LIST_PACKED : FieldType.INT32_LIST;
|
||||
case INT64:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.INT64;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.INT64_LIST_PACKED : FieldType.INT64_LIST;
|
||||
case MESSAGE:
|
||||
// TODO(nathanmittler): Add support for maps.
|
||||
return fd.isRepeated() ? FieldType.MESSAGE_LIST : FieldType.MESSAGE;
|
||||
case SFIXED32:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.SFIXED32;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.SFIXED32_LIST_PACKED : FieldType.SFIXED32_LIST;
|
||||
case SFIXED64:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.SFIXED64;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.SFIXED64_LIST_PACKED : FieldType.SFIXED64_LIST;
|
||||
case SINT32:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.SINT32;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.SINT32_LIST_PACKED : FieldType.SINT32_LIST;
|
||||
case SINT64:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.SINT64;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.SINT64_LIST_PACKED : FieldType.SINT64_LIST;
|
||||
case STRING:
|
||||
return fd.isRepeated() ? FieldType.STRING_LIST : FieldType.STRING;
|
||||
case UINT32:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.UINT32;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.UINT32_LIST_PACKED : FieldType.UINT32_LIST;
|
||||
case UINT64:
|
||||
if (!fd.isRepeated()) {
|
||||
return FieldType.UINT64;
|
||||
}
|
||||
return fd.isPacked() ? FieldType.UINT64_LIST_PACKED : FieldType.UINT64_LIST;
|
||||
default:
|
||||
throw new IllegalArgumentException("Unsupported field type: " + fd.getType());
|
||||
}
|
||||
}
|
||||
|
||||
private static Field bitField(Class<?> messageType, int index) {
|
||||
return field(messageType, "bitField" + index + "_");
|
||||
}
|
||||
|
||||
private static Field field(Class<?> messageType, FieldDescriptor fd) {
|
||||
return field(messageType, getFieldName(fd));
|
||||
}
|
||||
|
||||
private static Field field(Class<?> messageType, String fieldName) {
|
||||
try {
|
||||
return messageType.getDeclaredField(fieldName);
|
||||
} catch (Exception e) {
|
||||
throw new IllegalArgumentException(
|
||||
"Unable to find field " + fieldName + " in message class " + messageType.getName());
|
||||
}
|
||||
}
|
||||
|
||||
// This method must match exactly with the corresponding function in protocol compiler.
|
||||
// See: https://github.com/google/protobuf/blob/v3.0.0/src/google/protobuf/compiler/java/java_helpers.cc#L153
|
||||
private static String getFieldName(FieldDescriptor fd) {
|
||||
String snakeCase =
|
||||
fd.getType() == FieldDescriptor.Type.GROUP ? fd.getMessageType().getName() : fd.getName();
|
||||
StringBuilder sb = new StringBuilder(snakeCase.length() + 1);
|
||||
boolean capNext = false;
|
||||
for (int ctr = 0; ctr < snakeCase.length(); ctr++) {
|
||||
char next = snakeCase.charAt(ctr);
|
||||
if (next == '_') {
|
||||
capNext = true;
|
||||
} else if (capNext) {
|
||||
sb.append(Character.toUpperCase(next));
|
||||
capNext = false;
|
||||
} else if (ctr == 0) {
|
||||
sb.append(Character.toLowerCase(next));
|
||||
} else {
|
||||
sb.append(next);
|
||||
}
|
||||
}
|
||||
sb.append('_');
|
||||
return sb.toString();
|
||||
}
|
||||
}
|
175
java/core/src/main/java/com/google/protobuf/FieldInfo.java
Normal file
175
java/core/src/main/java/com/google/protobuf/FieldInfo.java
Normal file
|
@ -0,0 +1,175 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Internal.checkNotNull;
|
||||
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.ParameterizedType;
|
||||
import java.lang.reflect.Type;
|
||||
|
||||
/** Information for a single field in a protobuf message class. */
|
||||
@ExperimentalApi
|
||||
public final class FieldInfo implements Comparable<FieldInfo> {
|
||||
private final Field field;
|
||||
private final FieldType type;
|
||||
private final int fieldNumber;
|
||||
private final Field presenceField;
|
||||
private final int presenceMask;
|
||||
|
||||
/** Constructs a new descriptor for a field. */
|
||||
public static FieldInfo forField(Field field, int fieldNumber, FieldType fieldType) {
|
||||
return new FieldInfo(field, fieldNumber, fieldType, null, 0);
|
||||
}
|
||||
|
||||
/** Constructor for a field that uses a presence bit field (i.e. proto2 only). */
|
||||
public static FieldInfo forPresenceCheckedField(
|
||||
Field field, int fieldNumber, FieldType fieldType, Field presenceField, int presenceMask) {
|
||||
checkNotNull(presenceField, "presenceField");
|
||||
return new FieldInfo(field, fieldNumber, fieldType, presenceField, presenceMask);
|
||||
}
|
||||
|
||||
private FieldInfo(
|
||||
Field field, int fieldNumber, FieldType type, Field presenceField, int presenceMask) {
|
||||
if (fieldNumber <= 0) {
|
||||
throw new IllegalArgumentException("fieldNumber must be positive: " + fieldNumber);
|
||||
}
|
||||
if (presenceField != null && !isExactlyOneBitSet(presenceMask)) {
|
||||
throw new IllegalArgumentException(
|
||||
"presenceMask must have exactly one bit set: " + presenceMask);
|
||||
}
|
||||
this.field = checkNotNull(field, "field");
|
||||
this.type = checkNotNull(type, "type");
|
||||
this.fieldNumber = fieldNumber;
|
||||
this.presenceField = presenceField;
|
||||
this.presenceMask = presenceMask;
|
||||
}
|
||||
|
||||
/** Gets the subject {@link Field} of this descriptor. */
|
||||
public Field getField() {
|
||||
return field;
|
||||
}
|
||||
|
||||
/** Gets the type information for the field. */
|
||||
public FieldType getType() {
|
||||
return type;
|
||||
}
|
||||
|
||||
/** Gets the field number for the field. */
|
||||
public int getFieldNumber() {
|
||||
return fieldNumber;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int compareTo(FieldInfo o) {
|
||||
return fieldNumber - o.fieldNumber;
|
||||
}
|
||||
|
||||
/**
|
||||
* For list fields, returns the generic argument that represents the type stored in the list. For
|
||||
* non-list fields, returns {@code null}.
|
||||
*/
|
||||
public Class<?> getListElementType() {
|
||||
if (!type.isList()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Type genericType = field.getGenericType();
|
||||
if (!(genericType instanceof ParameterizedType)) {
|
||||
throw new IllegalStateException(
|
||||
"Cannot determine parameterized type for list field " + fieldNumber);
|
||||
}
|
||||
|
||||
Type type = ((ParameterizedType) field.getGenericType()).getActualTypeArguments()[0];
|
||||
return (Class<?>) type;
|
||||
}
|
||||
|
||||
/** Gets the presence bit field. Only valid for unary fields. For lists, returns {@code null}. */
|
||||
public Field getPresenceField() {
|
||||
return presenceField;
|
||||
}
|
||||
|
||||
/**
|
||||
* If {@link #getPresenceField()} is non-{@code null}, returns the mask used to identify the
|
||||
* presence bit for this field in the message.
|
||||
*/
|
||||
public int getPresenceMask() {
|
||||
return presenceMask;
|
||||
}
|
||||
|
||||
public static Builder newBuilder() {
|
||||
return new Builder();
|
||||
}
|
||||
|
||||
/** A builder for {@link FieldInfo} instances. */
|
||||
public static final class Builder {
|
||||
private Field field;
|
||||
private FieldType type;
|
||||
private int fieldNumber;
|
||||
private Field presenceField;
|
||||
private int presenceMask;
|
||||
|
||||
private Builder() {}
|
||||
|
||||
public Builder withField(Field field) {
|
||||
this.field = field;
|
||||
return this;
|
||||
}
|
||||
|
||||
public Builder withType(FieldType type) {
|
||||
this.type = type;
|
||||
return this;
|
||||
}
|
||||
|
||||
public Builder withFieldNumber(int fieldNumber) {
|
||||
this.fieldNumber = fieldNumber;
|
||||
return this;
|
||||
}
|
||||
|
||||
public Builder withPresenceField(Field presenceField) {
|
||||
this.presenceField = checkNotNull(presenceField, "presenceField");
|
||||
return this;
|
||||
}
|
||||
|
||||
public Builder withPresenceMask(int presenceMask) {
|
||||
this.presenceMask = presenceMask;
|
||||
return this;
|
||||
}
|
||||
|
||||
public FieldInfo build() {
|
||||
return new FieldInfo(field, fieldNumber, type, presenceField, presenceMask);
|
||||
}
|
||||
}
|
||||
|
||||
private static boolean isExactlyOneBitSet(int value) {
|
||||
return value != 0 && (value & (value - 1)) == 0;
|
||||
}
|
||||
}
|
535
java/core/src/main/java/com/google/protobuf/FieldType.java
Normal file
535
java/core/src/main/java/com/google/protobuf/FieldType.java
Normal file
|
@ -0,0 +1,535 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.ParameterizedType;
|
||||
import java.lang.reflect.Type;
|
||||
import java.lang.reflect.TypeVariable;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Enumeration identifying all relevant type information for a protobuf field.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public enum FieldType {
|
||||
DOUBLE(0, Collection.SCALAR, JavaType.DOUBLE, JavaType.VOID),
|
||||
FLOAT(1, Collection.SCALAR, JavaType.FLOAT, JavaType.VOID),
|
||||
INT64(2, Collection.SCALAR, JavaType.LONG, JavaType.VOID),
|
||||
UINT64(3, Collection.SCALAR, JavaType.LONG, JavaType.VOID),
|
||||
INT32(4, Collection.SCALAR, JavaType.INT, JavaType.VOID),
|
||||
FIXED64(5, Collection.SCALAR, JavaType.LONG, JavaType.VOID),
|
||||
FIXED32(6, Collection.SCALAR, JavaType.INT, JavaType.VOID),
|
||||
BOOL(7, Collection.SCALAR, JavaType.BOOLEAN, JavaType.VOID),
|
||||
STRING(8, Collection.SCALAR, JavaType.STRING, JavaType.VOID),
|
||||
MESSAGE(9, Collection.SCALAR, JavaType.MESSAGE, JavaType.VOID),
|
||||
BYTES(10, Collection.SCALAR, JavaType.BYTE_STRING, JavaType.VOID),
|
||||
UINT32(11, Collection.SCALAR, JavaType.INT, JavaType.VOID),
|
||||
ENUM(12, Collection.SCALAR, JavaType.ENUM, JavaType.VOID),
|
||||
SFIXED32(13, Collection.SCALAR, JavaType.INT, JavaType.VOID),
|
||||
SFIXED64(14, Collection.SCALAR, JavaType.LONG, JavaType.VOID),
|
||||
SINT32(15, Collection.SCALAR, JavaType.INT, JavaType.VOID),
|
||||
SINT64(16, Collection.SCALAR, JavaType.LONG, JavaType.VOID),
|
||||
DOUBLE_LIST(17, Collection.VECTOR, JavaType.DOUBLE, JavaType.VOID),
|
||||
FLOAT_LIST(18, Collection.VECTOR, JavaType.FLOAT, JavaType.VOID),
|
||||
INT64_LIST(19, Collection.VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
UINT64_LIST(20, Collection.VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
INT32_LIST(21, Collection.VECTOR, JavaType.INT, JavaType.VOID),
|
||||
FIXED64_LIST(22, Collection.VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
FIXED32_LIST(23, Collection.VECTOR, JavaType.INT, JavaType.VOID),
|
||||
BOOL_LIST(24, Collection.VECTOR, JavaType.BOOLEAN, JavaType.VOID),
|
||||
STRING_LIST(25, Collection.VECTOR, JavaType.STRING, JavaType.VOID),
|
||||
MESSAGE_LIST(26, Collection.VECTOR, JavaType.MESSAGE, JavaType.VOID),
|
||||
BYTES_LIST(27, Collection.VECTOR, JavaType.BYTE_STRING, JavaType.VOID),
|
||||
UINT32_LIST(28, Collection.VECTOR, JavaType.INT, JavaType.VOID),
|
||||
ENUM_LIST(29, Collection.VECTOR, JavaType.ENUM, JavaType.VOID),
|
||||
SFIXED32_LIST(30, Collection.VECTOR, JavaType.INT, JavaType.VOID),
|
||||
SFIXED64_LIST(31, Collection.VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
SINT32_LIST(32, Collection.VECTOR, JavaType.INT, JavaType.VOID),
|
||||
SINT64_LIST(33, Collection.VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
DOUBLE_LIST_PACKED(34, Collection.PACKED_VECTOR, JavaType.DOUBLE, JavaType.VOID),
|
||||
FLOAT_LIST_PACKED(35, Collection.PACKED_VECTOR, JavaType.FLOAT, JavaType.VOID),
|
||||
INT64_LIST_PACKED(36, Collection.PACKED_VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
UINT64_LIST_PACKED(37, Collection.PACKED_VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
INT32_LIST_PACKED(38, Collection.PACKED_VECTOR, JavaType.INT, JavaType.VOID),
|
||||
FIXED64_LIST_PACKED(39, Collection.PACKED_VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
FIXED32_LIST_PACKED(40, Collection.PACKED_VECTOR, JavaType.INT, JavaType.VOID),
|
||||
BOOL_LIST_PACKED(41, Collection.PACKED_VECTOR, JavaType.BOOLEAN, JavaType.VOID),
|
||||
UINT32_LIST_PACKED(42, Collection.PACKED_VECTOR, JavaType.INT, JavaType.VOID),
|
||||
ENUM_LIST_PACKED(43, Collection.PACKED_VECTOR, JavaType.ENUM, JavaType.VOID),
|
||||
SFIXED32_LIST_PACKED(44, Collection.PACKED_VECTOR, JavaType.INT, JavaType.VOID),
|
||||
SFIXED64_LIST_PACKED(45, Collection.PACKED_VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
SINT32_LIST_PACKED(46, Collection.PACKED_VECTOR, JavaType.INT, JavaType.VOID),
|
||||
SINT64_LIST_PACKED(47, Collection.PACKED_VECTOR, JavaType.LONG, JavaType.VOID),
|
||||
INT32_TO_INT32_MAP(48, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
INT32_TO_INT64_MAP(49, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
INT32_TO_UINT32_MAP(50, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
INT32_TO_UINT64_MAP(51, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
INT32_TO_SINT32_MAP(52, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
INT32_TO_SINT64_MAP(53, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
INT32_TO_FIXED32_MAP(54, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
INT32_TO_FIXED64_MAP(55, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
INT32_TO_SFIXED32_MAP(56, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
INT32_TO_SFIXED64_MAP(57, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
INT32_TO_BOOL_MAP(58, Collection.MAP, JavaType.INT, JavaType.BOOLEAN),
|
||||
INT32_TO_STRING_MAP(59, Collection.MAP, JavaType.INT, JavaType.STRING),
|
||||
INT32_TO_ENUM_MAP(60, Collection.MAP, JavaType.INT, JavaType.ENUM),
|
||||
INT32_TO_MESSAGE_MAP(61, Collection.MAP, JavaType.INT, JavaType.MESSAGE),
|
||||
INT32_TO_BYTES_MAP(62, Collection.MAP, JavaType.INT, JavaType.BYTE_STRING),
|
||||
INT32_TO_DOUBLE_MAP(63, Collection.MAP, JavaType.INT, JavaType.DOUBLE),
|
||||
INT32_TO_FLOAT_MAP(64, Collection.MAP, JavaType.INT, JavaType.FLOAT),
|
||||
INT64_TO_INT32_MAP(65, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
INT64_TO_INT64_MAP(66, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
INT64_TO_UINT32_MAP(67, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
INT64_TO_UINT64_MAP(68, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
INT64_TO_SINT32_MAP(69, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
INT64_TO_SINT64_MAP(70, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
INT64_TO_FIXED32_MAP(71, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
INT64_TO_FIXED64_MAP(72, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
INT64_TO_SFIXED32_MAP(73, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
INT64_TO_SFIXED64_MAP(74, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
INT64_TO_BOOL_MAP(75, Collection.MAP, JavaType.LONG, JavaType.BOOLEAN),
|
||||
INT64_TO_STRING_MAP(76, Collection.MAP, JavaType.LONG, JavaType.STRING),
|
||||
INT64_TO_ENUM_MAP(77, Collection.MAP, JavaType.LONG, JavaType.ENUM),
|
||||
INT64_TO_MESSAGE_MAP(78, Collection.MAP, JavaType.LONG, JavaType.MESSAGE),
|
||||
INT64_TO_BYTES_MAP(79, Collection.MAP, JavaType.LONG, JavaType.BYTE_STRING),
|
||||
INT64_TO_DOUBLE_MAP(80, Collection.MAP, JavaType.LONG, JavaType.DOUBLE),
|
||||
INT64_TO_FLOAT_MAP(81, Collection.MAP, JavaType.LONG, JavaType.FLOAT),
|
||||
UINT32_TO_INT32_MAP(82, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
UINT32_TO_INT64_MAP(83, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
UINT32_TO_UINT32_MAP(84, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
UINT32_TO_UINT64_MAP(85, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
UINT32_TO_SINT32_MAP(86, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
UINT32_TO_SINT64_MAP(87, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
UINT32_TO_FIXED32_MAP(88, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
UINT32_TO_FIXED64_MAP(89, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
UINT32_TO_SFIXED32_MAP(90, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
UINT32_TO_SFIXED64_MAP(91, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
UINT32_TO_BOOL_MAP(92, Collection.MAP, JavaType.INT, JavaType.BOOLEAN),
|
||||
UINT32_TO_STRING_MAP(93, Collection.MAP, JavaType.INT, JavaType.STRING),
|
||||
UINT32_TO_ENUM_MAP(94, Collection.MAP, JavaType.INT, JavaType.ENUM),
|
||||
UINT32_TO_MESSAGE_MAP(95, Collection.MAP, JavaType.INT, JavaType.MESSAGE),
|
||||
UINT32_TO_BYTES_MAP(96, Collection.MAP, JavaType.INT, JavaType.BYTE_STRING),
|
||||
UINT32_TO_DOUBLE_MAP(97, Collection.MAP, JavaType.INT, JavaType.DOUBLE),
|
||||
UINT32_TO_FLOAT_MAP(98, Collection.MAP, JavaType.INT, JavaType.FLOAT),
|
||||
UINT64_TO_INT32_MAP(99, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
UINT64_TO_INT64_MAP(100, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
UINT64_TO_UINT32_MAP(101, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
UINT64_TO_UINT64_MAP(102, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
UINT64_TO_SINT32_MAP(103, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
UINT64_TO_SINT64_MAP(104, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
UINT64_TO_FIXED32_MAP(105, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
UINT64_TO_FIXED64_MAP(106, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
UINT64_TO_SFIXED32_MAP(107, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
UINT64_TO_SFIXED64_MAP(108, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
UINT64_TO_BOOL_MAP(109, Collection.MAP, JavaType.LONG, JavaType.BOOLEAN),
|
||||
UINT64_TO_STRING_MAP(110, Collection.MAP, JavaType.LONG, JavaType.STRING),
|
||||
UINT64_TO_ENUM_MAP(111, Collection.MAP, JavaType.LONG, JavaType.ENUM),
|
||||
UINT64_TO_MESSAGE_MAP(112, Collection.MAP, JavaType.LONG, JavaType.MESSAGE),
|
||||
UINT64_TO_BYTES_MAP(113, Collection.MAP, JavaType.LONG, JavaType.BYTE_STRING),
|
||||
UINT64_TO_DOUBLE_MAP(114, Collection.MAP, JavaType.LONG, JavaType.DOUBLE),
|
||||
UINT64_TO_FLOAT_MAP(115, Collection.MAP, JavaType.LONG, JavaType.FLOAT),
|
||||
SINT32_TO_INT32_MAP(116, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SINT32_TO_INT64_MAP(117, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SINT32_TO_UINT32_MAP(118, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SINT32_TO_UINT64_MAP(119, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SINT32_TO_SINT32_MAP(120, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SINT32_TO_SINT64_MAP(121, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SINT32_TO_FIXED32_MAP(122, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SINT32_TO_FIXED64_MAP(123, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SINT32_TO_SFIXED32_MAP(124, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SINT32_TO_SFIXED64_MAP(125, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SINT32_TO_BOOL_MAP(126, Collection.MAP, JavaType.INT, JavaType.BOOLEAN),
|
||||
SINT32_TO_STRING_MAP(127, Collection.MAP, JavaType.INT, JavaType.STRING),
|
||||
SINT32_TO_ENUM_MAP(128, Collection.MAP, JavaType.INT, JavaType.ENUM),
|
||||
SINT32_TO_MESSAGE_MAP(129, Collection.MAP, JavaType.INT, JavaType.MESSAGE),
|
||||
SINT32_TO_BYTES_MAP(130, Collection.MAP, JavaType.INT, JavaType.BYTE_STRING),
|
||||
SINT32_TO_DOUBLE_MAP(131, Collection.MAP, JavaType.INT, JavaType.DOUBLE),
|
||||
SINT32_TO_FLOAT_MAP(132, Collection.MAP, JavaType.INT, JavaType.FLOAT),
|
||||
SINT64_TO_INT32_MAP(133, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SINT64_TO_INT64_MAP(134, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SINT64_TO_UINT32_MAP(135, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SINT64_TO_UINT64_MAP(136, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SINT64_TO_SINT32_MAP(137, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SINT64_TO_SINT64_MAP(138, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SINT64_TO_FIXED32_MAP(139, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SINT64_TO_FIXED64_MAP(140, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SINT64_TO_SFIXED32_MAP(141, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SINT64_TO_SFIXED64_MAP(142, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SINT64_TO_BOOL_MAP(143, Collection.MAP, JavaType.LONG, JavaType.BOOLEAN),
|
||||
SINT64_TO_STRING_MAP(144, Collection.MAP, JavaType.LONG, JavaType.STRING),
|
||||
SINT64_TO_ENUM_MAP(145, Collection.MAP, JavaType.LONG, JavaType.ENUM),
|
||||
SINT64_TO_MESSAGE_MAP(146, Collection.MAP, JavaType.LONG, JavaType.MESSAGE),
|
||||
SINT64_TO_BYTES_MAP(147, Collection.MAP, JavaType.LONG, JavaType.BYTE_STRING),
|
||||
SINT64_TO_DOUBLE_MAP(148, Collection.MAP, JavaType.LONG, JavaType.DOUBLE),
|
||||
SINT64_TO_FLOAT_MAP(149, Collection.MAP, JavaType.LONG, JavaType.FLOAT),
|
||||
FIXED32_TO_INT32_MAP(150, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
FIXED32_TO_INT64_MAP(151, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
FIXED32_TO_UINT32_MAP(152, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
FIXED32_TO_UINT64_MAP(153, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
FIXED32_TO_SINT32_MAP(154, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
FIXED32_TO_SINT64_MAP(155, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
FIXED32_TO_FIXED32_MAP(156, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
FIXED32_TO_FIXED64_MAP(157, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
FIXED32_TO_SFIXED32_MAP(158, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
FIXED32_TO_SFIXED64_MAP(159, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
FIXED32_TO_BOOL_MAP(160, Collection.MAP, JavaType.INT, JavaType.BOOLEAN),
|
||||
FIXED32_TO_STRING_MAP(161, Collection.MAP, JavaType.INT, JavaType.STRING),
|
||||
FIXED32_TO_ENUM_MAP(162, Collection.MAP, JavaType.INT, JavaType.ENUM),
|
||||
FIXED32_TO_MESSAGE_MAP(163, Collection.MAP, JavaType.INT, JavaType.MESSAGE),
|
||||
FIXED32_TO_BYTES_MAP(164, Collection.MAP, JavaType.INT, JavaType.BYTE_STRING),
|
||||
FIXED32_TO_DOUBLE_MAP(165, Collection.MAP, JavaType.INT, JavaType.DOUBLE),
|
||||
FIXED32_TO_FLOAT_MAP(166, Collection.MAP, JavaType.INT, JavaType.FLOAT),
|
||||
FIXED64_TO_INT32_MAP(167, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
FIXED64_TO_INT64_MAP(168, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
FIXED64_TO_UINT32_MAP(169, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
FIXED64_TO_UINT64_MAP(170, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
FIXED64_TO_SINT32_MAP(171, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
FIXED64_TO_SINT64_MAP(172, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
FIXED64_TO_FIXED32_MAP(173, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
FIXED64_TO_FIXED64_MAP(174, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
FIXED64_TO_SFIXED32_MAP(175, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
FIXED64_TO_SFIXED64_MAP(176, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
FIXED64_TO_BOOL_MAP(177, Collection.MAP, JavaType.LONG, JavaType.BOOLEAN),
|
||||
FIXED64_TO_STRING_MAP(178, Collection.MAP, JavaType.LONG, JavaType.STRING),
|
||||
FIXED64_TO_ENUM_MAP(179, Collection.MAP, JavaType.LONG, JavaType.ENUM),
|
||||
FIXED64_TO_MESSAGE_MAP(180, Collection.MAP, JavaType.LONG, JavaType.MESSAGE),
|
||||
FIXED64_TO_BYTES_MAP(181, Collection.MAP, JavaType.LONG, JavaType.BYTE_STRING),
|
||||
FIXED64_TO_DOUBLE_MAP(182, Collection.MAP, JavaType.LONG, JavaType.DOUBLE),
|
||||
FIXED64_TO_FLOAT_MAP(183, Collection.MAP, JavaType.LONG, JavaType.FLOAT),
|
||||
SFIXED32_TO_INT32_MAP(184, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SFIXED32_TO_INT64_MAP(185, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SFIXED32_TO_UINT32_MAP(186, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SFIXED32_TO_UINT64_MAP(187, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SFIXED32_TO_SINT32_MAP(188, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SFIXED32_TO_SINT64_MAP(189, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SFIXED32_TO_FIXED32_MAP(190, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SFIXED32_TO_FIXED64_MAP(191, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SFIXED32_TO_SFIXED32_MAP(192, Collection.MAP, JavaType.INT, JavaType.INT),
|
||||
SFIXED32_TO_SFIXED64_MAP(193, Collection.MAP, JavaType.INT, JavaType.LONG),
|
||||
SFIXED32_TO_BOOL_MAP(194, Collection.MAP, JavaType.INT, JavaType.BOOLEAN),
|
||||
SFIXED32_TO_STRING_MAP(195, Collection.MAP, JavaType.INT, JavaType.STRING),
|
||||
SFIXED32_TO_ENUM_MAP(196, Collection.MAP, JavaType.INT, JavaType.ENUM),
|
||||
SFIXED32_TO_MESSAGE_MAP(197, Collection.MAP, JavaType.INT, JavaType.MESSAGE),
|
||||
SFIXED32_TO_BYTES_MAP(198, Collection.MAP, JavaType.INT, JavaType.BYTE_STRING),
|
||||
SFIXED32_TO_DOUBLE_MAP(199, Collection.MAP, JavaType.INT, JavaType.DOUBLE),
|
||||
SFIXED32_TO_FLOAT_MAP(200, Collection.MAP, JavaType.INT, JavaType.FLOAT),
|
||||
SFIXED64_TO_INT32_MAP(201, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SFIXED64_TO_INT64_MAP(202, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SFIXED64_TO_UINT32_MAP(203, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SFIXED64_TO_UINT64_MAP(204, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SFIXED64_TO_SINT32_MAP(205, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SFIXED64_TO_SINT64_MAP(206, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SFIXED64_TO_FIXED32_MAP(207, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SFIXED64_TO_FIXED64_MAP(208, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SFIXED64_TO_SFIXED32_MAP(209, Collection.MAP, JavaType.LONG, JavaType.INT),
|
||||
SFIXED64_TO_SFIXED64_MAP(210, Collection.MAP, JavaType.LONG, JavaType.LONG),
|
||||
SFIXED64_TO_BOOL_MAP(211, Collection.MAP, JavaType.LONG, JavaType.BOOLEAN),
|
||||
SFIXED64_TO_STRING_MAP(212, Collection.MAP, JavaType.LONG, JavaType.STRING),
|
||||
SFIXED64_TO_ENUM_MAP(213, Collection.MAP, JavaType.LONG, JavaType.ENUM),
|
||||
SFIXED64_TO_MESSAGE_MAP(214, Collection.MAP, JavaType.LONG, JavaType.MESSAGE),
|
||||
SFIXED64_TO_BYTES_MAP(215, Collection.MAP, JavaType.LONG, JavaType.BYTE_STRING),
|
||||
SFIXED64_TO_DOUBLE_MAP(216, Collection.MAP, JavaType.LONG, JavaType.DOUBLE),
|
||||
SFIXED64_TO_FLOAT_MAP(217, Collection.MAP, JavaType.LONG, JavaType.FLOAT),
|
||||
BOOL_TO_INT32_MAP(218, Collection.MAP, JavaType.BOOLEAN, JavaType.INT),
|
||||
BOOL_TO_INT64_MAP(219, Collection.MAP, JavaType.BOOLEAN, JavaType.LONG),
|
||||
BOOL_TO_UINT32_MAP(220, Collection.MAP, JavaType.BOOLEAN, JavaType.INT),
|
||||
BOOL_TO_UINT64_MAP(221, Collection.MAP, JavaType.BOOLEAN, JavaType.LONG),
|
||||
BOOL_TO_SINT32_MAP(222, Collection.MAP, JavaType.BOOLEAN, JavaType.INT),
|
||||
BOOL_TO_SINT64_MAP(223, Collection.MAP, JavaType.BOOLEAN, JavaType.LONG),
|
||||
BOOL_TO_FIXED32_MAP(224, Collection.MAP, JavaType.BOOLEAN, JavaType.INT),
|
||||
BOOL_TO_FIXED64_MAP(225, Collection.MAP, JavaType.BOOLEAN, JavaType.LONG),
|
||||
BOOL_TO_SFIXED32_MAP(226, Collection.MAP, JavaType.BOOLEAN, JavaType.INT),
|
||||
BOOL_TO_SFIXED64_MAP(227, Collection.MAP, JavaType.BOOLEAN, JavaType.LONG),
|
||||
BOOL_TO_BOOL_MAP(228, Collection.MAP, JavaType.BOOLEAN, JavaType.BOOLEAN),
|
||||
BOOL_TO_STRING_MAP(229, Collection.MAP, JavaType.BOOLEAN, JavaType.STRING),
|
||||
BOOL_TO_ENUM_MAP(230, Collection.MAP, JavaType.BOOLEAN, JavaType.ENUM),
|
||||
BOOL_TO_MESSAGE_MAP(231, Collection.MAP, JavaType.BOOLEAN, JavaType.MESSAGE),
|
||||
BOOL_TO_BYTES_MAP(232, Collection.MAP, JavaType.BOOLEAN, JavaType.BYTE_STRING),
|
||||
BOOL_TO_DOUBLE_MAP(233, Collection.MAP, JavaType.BOOLEAN, JavaType.DOUBLE),
|
||||
BOOL_TO_FLOAT_MAP(234, Collection.MAP, JavaType.BOOLEAN, JavaType.FLOAT),
|
||||
STRING_TO_INT32_MAP(235, Collection.MAP, JavaType.STRING, JavaType.INT),
|
||||
STRING_TO_INT64_MAP(236, Collection.MAP, JavaType.STRING, JavaType.LONG),
|
||||
STRING_TO_UINT32_MAP(237, Collection.MAP, JavaType.STRING, JavaType.INT),
|
||||
STRING_TO_UINT64_MAP(238, Collection.MAP, JavaType.STRING, JavaType.LONG),
|
||||
STRING_TO_SINT32_MAP(239, Collection.MAP, JavaType.STRING, JavaType.INT),
|
||||
STRING_TO_SINT64_MAP(240, Collection.MAP, JavaType.STRING, JavaType.LONG),
|
||||
STRING_TO_FIXED32_MAP(241, Collection.MAP, JavaType.STRING, JavaType.INT),
|
||||
STRING_TO_FIXED64_MAP(242, Collection.MAP, JavaType.STRING, JavaType.LONG),
|
||||
STRING_TO_SFIXED32_MAP(243, Collection.MAP, JavaType.STRING, JavaType.INT),
|
||||
STRING_TO_SFIXED64_MAP(244, Collection.MAP, JavaType.STRING, JavaType.LONG),
|
||||
STRING_TO_BOOL_MAP(245, Collection.MAP, JavaType.STRING, JavaType.BOOLEAN),
|
||||
STRING_TO_STRING_MAP(246, Collection.MAP, JavaType.STRING, JavaType.STRING),
|
||||
STRING_TO_ENUM_MAP(247, Collection.MAP, JavaType.STRING, JavaType.ENUM),
|
||||
STRING_TO_MESSAGE_MAP(248, Collection.MAP, JavaType.STRING, JavaType.MESSAGE),
|
||||
STRING_TO_BYTES_MAP(249, Collection.MAP, JavaType.STRING, JavaType.BYTE_STRING),
|
||||
STRING_TO_DOUBLE_MAP(250, Collection.MAP, JavaType.STRING, JavaType.DOUBLE),
|
||||
STRING_TO_FLOAT_MAP(251, Collection.MAP, JavaType.STRING, JavaType.FLOAT),
|
||||
GROUP(252, Collection.SCALAR, JavaType.MESSAGE, JavaType.VOID),
|
||||
GROUP_LIST(253, Collection.VECTOR, JavaType.MESSAGE, JavaType.VOID),
|
||||
GROUP_LIST_PACKED(254, Collection.PACKED_VECTOR, JavaType.MESSAGE, JavaType.VOID);
|
||||
|
||||
private final JavaType javaType1;
|
||||
private final JavaType javaType2;
|
||||
private final int id;
|
||||
private final Collection collection;
|
||||
private final Class<?> elementType1;
|
||||
private final Class<?> elementType2;
|
||||
|
||||
FieldType(int id, Collection collection, JavaType javaType1, JavaType javaType2) {
|
||||
this.id = id;
|
||||
this.collection = collection;
|
||||
this.javaType1 = javaType1;
|
||||
this.javaType2 = javaType2;
|
||||
|
||||
switch (collection) {
|
||||
case MAP:
|
||||
elementType1 = javaType1.getBoxedType();
|
||||
elementType2 = javaType2.getBoxedType();
|
||||
break;
|
||||
case VECTOR:
|
||||
elementType1 = javaType1.getBoxedType();
|
||||
elementType2 = null;
|
||||
break;
|
||||
case SCALAR:
|
||||
default:
|
||||
elementType1 = null;
|
||||
elementType2 = null;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* A reliable unique identifier for this type.
|
||||
*/
|
||||
public int id() {
|
||||
return id;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the {@link JavaType} for this field. For lists, this identifies the type of the elements
|
||||
* contained within the list.
|
||||
*/
|
||||
public JavaType getJavaType() {
|
||||
return javaType1;
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether a list field should be represented on the wire in packed form.
|
||||
*/
|
||||
public boolean isPacked() {
|
||||
return Collection.PACKED_VECTOR.equals(collection);
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether this field represents a list of values.
|
||||
*/
|
||||
public boolean isList() {
|
||||
return collection.isList();
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether or not this {@link FieldType} can be applied to the given {@link Field}.
|
||||
*/
|
||||
public boolean isValidForField(Field field) {
|
||||
if (Collection.VECTOR.equals(collection)) {
|
||||
return isValidForList(field);
|
||||
} else {
|
||||
return javaType1.getType().isAssignableFrom(field.getType());
|
||||
}
|
||||
}
|
||||
|
||||
private boolean isValidForList(Field field) {
|
||||
Class<?> clazz = field.getType();
|
||||
if (!javaType1.getType().isAssignableFrom(clazz)) {
|
||||
// The field isn't a List type.
|
||||
return false;
|
||||
}
|
||||
Type[] types = EMPTY_TYPES;
|
||||
Type genericType = field.getGenericType();
|
||||
if (genericType instanceof ParameterizedType) {
|
||||
types = ((ParameterizedType) field.getGenericType()).getActualTypeArguments();
|
||||
}
|
||||
Type listParameter = getListParameter(clazz, types);
|
||||
if (!(listParameter instanceof Class)) {
|
||||
// It's a wildcard, we should allow anything in the list.
|
||||
return true;
|
||||
}
|
||||
return elementType1.isAssignableFrom((Class<?>) listParameter);
|
||||
}
|
||||
|
||||
/**
|
||||
* Looks up the appropriate {@link FieldType} by it's identifier.
|
||||
*
|
||||
* @return the {@link FieldType} or {@code null} if not found.
|
||||
*/
|
||||
/* @Nullable */
|
||||
public static FieldType forId(byte id) {
|
||||
if (id < 0 || id >= VALUES.length) {
|
||||
return null;
|
||||
}
|
||||
return VALUES[id];
|
||||
}
|
||||
|
||||
private static final FieldType[] VALUES;
|
||||
private static final Type[] EMPTY_TYPES = new Type[0];
|
||||
|
||||
static {
|
||||
FieldType[] values = values();
|
||||
VALUES = new FieldType[values.length];
|
||||
for (FieldType type : values) {
|
||||
VALUES[type.id] = type;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Given a class, finds a generic super class or interface that extends {@link List}.
|
||||
*
|
||||
* @return the generic super class/interface, or {@code null} if not found.
|
||||
*/
|
||||
/* @Nullable */
|
||||
private static Type getGenericSuperList(Class<?> clazz) {
|
||||
// First look at interfaces.
|
||||
Type[] genericInterfaces = clazz.getGenericInterfaces();
|
||||
for (Type genericInterface : genericInterfaces) {
|
||||
if (genericInterface instanceof ParameterizedType) {
|
||||
ParameterizedType parameterizedType = (ParameterizedType) genericInterface;
|
||||
Class<?> rawType = (Class<?>) parameterizedType.getRawType();
|
||||
if (List.class.isAssignableFrom(rawType)) {
|
||||
return genericInterface;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Try the subclass
|
||||
Type type = clazz.getGenericSuperclass();
|
||||
if (type instanceof ParameterizedType) {
|
||||
ParameterizedType parameterizedType = (ParameterizedType) type;
|
||||
Class<?> rawType = (Class<?>) parameterizedType.getRawType();
|
||||
if (List.class.isAssignableFrom(rawType)) {
|
||||
return type;
|
||||
}
|
||||
}
|
||||
|
||||
// No super class/interface extends List.
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Inspects the inheritance hierarchy for the given class and finds the generic type parameter
|
||||
* for {@link List}.
|
||||
*
|
||||
* @param clazz the class to begin the search.
|
||||
* @param realTypes the array of actual type parameters for {@code clazz}. These will be used to
|
||||
* substitute generic parameters up the inheritance hierarchy. If {@code clazz} does not have any
|
||||
* generic parameters, this list should be empty.
|
||||
* @return the {@link List} parameter.
|
||||
*/
|
||||
private static Type getListParameter(Class<?> clazz, Type[] realTypes) {
|
||||
top:
|
||||
while (clazz != List.class) {
|
||||
// First look at generic subclass and interfaces.
|
||||
Type genericType = getGenericSuperList(clazz);
|
||||
if (genericType instanceof ParameterizedType) {
|
||||
// Replace any generic parameters with the real values.
|
||||
ParameterizedType parameterizedType = (ParameterizedType) genericType;
|
||||
Type[] superArgs = parameterizedType.getActualTypeArguments();
|
||||
for (int i = 0; i < superArgs.length; ++i) {
|
||||
Type superArg = superArgs[i];
|
||||
if (superArg instanceof TypeVariable) {
|
||||
// Get the type variables for this class so that we can match them to the variables
|
||||
// used on the super class.
|
||||
TypeVariable<?>[] clazzParams = clazz.getTypeParameters();
|
||||
if (realTypes.length != clazzParams.length) {
|
||||
throw new RuntimeException("Type array mismatch");
|
||||
}
|
||||
|
||||
// Replace the variable parameter with the real type.
|
||||
boolean foundReplacement = false;
|
||||
for (int j = 0; j < clazzParams.length; ++j) {
|
||||
if (superArg == clazzParams[j]) {
|
||||
Type realType = realTypes[j];
|
||||
superArgs[i] = realType;
|
||||
foundReplacement = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!foundReplacement) {
|
||||
throw new RuntimeException("Unable to find replacement for " + superArg);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Class<?> parent = (Class<?>) parameterizedType.getRawType();
|
||||
|
||||
realTypes = superArgs;
|
||||
clazz = parent;
|
||||
continue;
|
||||
}
|
||||
|
||||
// None of the parameterized types inherit List. Just continue up the inheritance hierarchy
|
||||
// toward the List interface until we can identify the parameters.
|
||||
realTypes = EMPTY_TYPES;
|
||||
for (Class<?> iface : clazz.getInterfaces()) {
|
||||
if (List.class.isAssignableFrom(iface)) {
|
||||
clazz = iface;
|
||||
continue top;
|
||||
}
|
||||
}
|
||||
clazz = clazz.getSuperclass();
|
||||
}
|
||||
|
||||
if (realTypes.length != 1) {
|
||||
throw new RuntimeException("Unable to identify parameter type for List<T>");
|
||||
}
|
||||
return realTypes[0];
|
||||
}
|
||||
|
||||
enum Collection {
|
||||
SCALAR(false),
|
||||
VECTOR(true),
|
||||
PACKED_VECTOR(true),
|
||||
MAP(false);
|
||||
|
||||
private final boolean isList;
|
||||
|
||||
Collection(boolean isList) {
|
||||
this.isList = isList;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return the isList
|
||||
*/
|
||||
public boolean isList() {
|
||||
return isList;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,754 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.util.AbstractCollection;
|
||||
import java.util.AbstractSet;
|
||||
import java.util.Arrays;
|
||||
import java.util.Collection;
|
||||
import java.util.Iterator;
|
||||
import java.util.Map;
|
||||
import java.util.NoSuchElementException;
|
||||
import java.util.Set;
|
||||
|
||||
/**
|
||||
* A hash map that uses primitive integers as keys with open addressing. To minimize the memory
|
||||
* footprint, this class uses open addressing rather than chaining. Collisions are resolved using
|
||||
* linear probing. Deletions implement compaction, so cost of remove can approach O(N) for full
|
||||
* maps, which makes a small loadFactor recommended.
|
||||
*
|
||||
* @param <V> The value type stored in the map.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public final class Int2ObjectHashMap<V> implements Map<Integer, V> {
|
||||
/**
|
||||
* A primitive entry in the map, provided by the iterator from {@link #entries()}
|
||||
*
|
||||
* @param <V> the value type stored in the map.
|
||||
*/
|
||||
public interface PrimitiveEntry<V> {
|
||||
/** Gets the key for this entry. */
|
||||
int key();
|
||||
|
||||
/** Gets the value for this entry. */
|
||||
V value();
|
||||
|
||||
/** Sets the value for this entry. */
|
||||
void setValue(V value);
|
||||
}
|
||||
|
||||
/** Default initial capacity. Used if not specified in the constructor */
|
||||
public static final int DEFAULT_CAPACITY = 8;
|
||||
|
||||
/** Default load factor. Used if not specified in the constructor */
|
||||
public static final float DEFAULT_LOAD_FACTOR = 0.5f;
|
||||
|
||||
/**
|
||||
* Placeholder for null values, so we can use the actual null to mean available. (Better than
|
||||
* using a placeholder for available: less references for GC processing.)
|
||||
*/
|
||||
private static final Object NULL_VALUE = new Object();
|
||||
|
||||
/** The maximum number of elements allowed without allocating more space. */
|
||||
private int maxSize;
|
||||
|
||||
/** The load factor for the map. Used to calculate {@link #maxSize}. */
|
||||
private final float loadFactor;
|
||||
|
||||
private int[] keys;
|
||||
private V[] values;
|
||||
private int size;
|
||||
private int mask;
|
||||
|
||||
private final Set<Integer> keySet = new KeySet();
|
||||
private final Set<Entry<Integer, V>> entrySet = new EntrySet();
|
||||
private final Iterable<PrimitiveEntry<V>> entries =
|
||||
new Iterable<PrimitiveEntry<V>>() {
|
||||
@Override
|
||||
public Iterator<PrimitiveEntry<V>> iterator() {
|
||||
return new PrimitiveIterator();
|
||||
}
|
||||
};
|
||||
|
||||
public Int2ObjectHashMap() {
|
||||
this(DEFAULT_CAPACITY, DEFAULT_LOAD_FACTOR);
|
||||
}
|
||||
|
||||
public Int2ObjectHashMap(int initialCapacity) {
|
||||
this(initialCapacity, DEFAULT_LOAD_FACTOR);
|
||||
}
|
||||
|
||||
public Int2ObjectHashMap(int initialCapacity, float loadFactor) {
|
||||
if (initialCapacity < 1) {
|
||||
throw new IllegalArgumentException("initialCapacity must be >= 1");
|
||||
}
|
||||
if (loadFactor <= 0.0f || loadFactor > 1.0f) {
|
||||
// Cannot exceed 1 because we can never store more than capacity elements;
|
||||
// using a bigger loadFactor would trigger rehashing before the desired load is reached.
|
||||
throw new IllegalArgumentException("loadFactor must be > 0 and <= 1");
|
||||
}
|
||||
|
||||
this.loadFactor = loadFactor;
|
||||
|
||||
// Adjust the initial capacity if necessary.
|
||||
int capacity = findNextPositivePowerOfTwo(initialCapacity);
|
||||
mask = capacity - 1;
|
||||
|
||||
// Allocate the arrays.
|
||||
keys = new int[capacity];
|
||||
@SuppressWarnings({"unchecked", "SuspiciousArrayCast"})
|
||||
V[] temp = (V[]) new Object[capacity];
|
||||
values = temp;
|
||||
|
||||
// Initialize the maximum size value.
|
||||
maxSize = calcMaxSize(capacity);
|
||||
}
|
||||
|
||||
private static <T> T toExternal(T value) {
|
||||
return value == NULL_VALUE ? null : value;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
private static <T> T toInternal(T value) {
|
||||
return value == null ? (T) NULL_VALUE : value;
|
||||
}
|
||||
|
||||
public V get(int key) {
|
||||
int index = indexOf(key);
|
||||
return index == -1 ? null : toExternal(values[index]);
|
||||
}
|
||||
|
||||
public V put(int key, V value) {
|
||||
int startIndex = hashIndex(key);
|
||||
int index = startIndex;
|
||||
|
||||
for (; ; ) {
|
||||
if (values[index] == null) {
|
||||
// Found empty slot, use it.
|
||||
keys[index] = key;
|
||||
values[index] = toInternal(value);
|
||||
growSize();
|
||||
return null;
|
||||
}
|
||||
if (keys[index] == key) {
|
||||
// Found existing entry with this key, just replace the value.
|
||||
V previousValue = values[index];
|
||||
values[index] = toInternal(value);
|
||||
return toExternal(previousValue);
|
||||
}
|
||||
|
||||
// Conflict, keep probing ...
|
||||
if ((index = probeNext(index)) == startIndex) {
|
||||
// Can only happen if the map was full at MAX_ARRAY_SIZE and couldn't grow.
|
||||
throw new IllegalStateException("Unable to insert");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void putAll(Map<? extends Integer, ? extends V> sourceMap) {
|
||||
if (sourceMap instanceof Int2ObjectHashMap) {
|
||||
// Optimization - iterate through the arrays.
|
||||
@SuppressWarnings("unchecked")
|
||||
Int2ObjectHashMap<V> source = (Int2ObjectHashMap<V>) sourceMap;
|
||||
for (int i = 0; i < source.values.length; ++i) {
|
||||
V sourceValue = source.values[i];
|
||||
if (sourceValue != null) {
|
||||
put(source.keys[i], sourceValue);
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// Otherwise, just add each entry.
|
||||
for (Entry<? extends Integer, ? extends V> entry : sourceMap.entrySet()) {
|
||||
put(entry.getKey(), entry.getValue());
|
||||
}
|
||||
}
|
||||
|
||||
public V remove(int key) {
|
||||
int index = indexOf(key);
|
||||
if (index == -1) {
|
||||
return null;
|
||||
}
|
||||
|
||||
V prev = values[index];
|
||||
removeAt(index);
|
||||
return toExternal(prev);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int size() {
|
||||
return size;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isEmpty() {
|
||||
return size == 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void clear() {
|
||||
Arrays.fill(keys, 0);
|
||||
Arrays.fill(values, null);
|
||||
size = 0;
|
||||
}
|
||||
|
||||
public boolean containsKey(int key) {
|
||||
return indexOf(key) >= 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean containsValue(Object value) {
|
||||
@SuppressWarnings("unchecked")
|
||||
V v1 = toInternal((V) value);
|
||||
for (V v2 : values) {
|
||||
// The map supports null values; this will be matched as NULL_VALUE.equals(NULL_VALUE).
|
||||
if (v2 != null && v2.equals(v1)) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public Iterable<PrimitiveEntry<V>> entries() {
|
||||
return entries;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Collection<V> values() {
|
||||
return new AbstractCollection<V>() {
|
||||
@Override
|
||||
public Iterator<V> iterator() {
|
||||
return new Iterator<V>() {
|
||||
final PrimitiveIterator iter = new PrimitiveIterator();
|
||||
|
||||
@Override
|
||||
public boolean hasNext() {
|
||||
return iter.hasNext();
|
||||
}
|
||||
|
||||
@Override
|
||||
public V next() {
|
||||
return iter.next().value();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void remove() {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
@Override
|
||||
public int size() {
|
||||
return size;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
// Hashcode is based on all non-zero, valid keys. We have to scan the whole keys
|
||||
// array, which may have different lengths for two maps of same size(), so the
|
||||
// capacity cannot be used as input for hashing but the size can.
|
||||
int hash = size;
|
||||
for (int key : keys) {
|
||||
// 0 can be a valid key or unused slot, but won't impact the hashcode in either case.
|
||||
// This way we can use a cheap loop without conditionals, or hard-to-unroll operations,
|
||||
// or the devastatingly bad memory locality of visiting value objects.
|
||||
// Also, it's important to use a hash function that does not depend on the ordering
|
||||
// of terms, only their values; since the map is an unordered collection and
|
||||
// entries can end up in different positions in different maps that have the same
|
||||
// elements, but with different history of puts/removes, due to conflicts.
|
||||
hash ^= hashCode(key);
|
||||
}
|
||||
return hash;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (this == obj) {
|
||||
return true;
|
||||
}
|
||||
if (!(obj instanceof Int2ObjectHashMap)) {
|
||||
return false;
|
||||
}
|
||||
@SuppressWarnings("rawtypes")
|
||||
Int2ObjectHashMap other = (Int2ObjectHashMap) obj;
|
||||
if (size != other.size()) {
|
||||
return false;
|
||||
}
|
||||
for (int i = 0; i < values.length; ++i) {
|
||||
V value = values[i];
|
||||
if (value != null) {
|
||||
int key = keys[i];
|
||||
Object otherValue = other.get(key);
|
||||
if (value == NULL_VALUE) {
|
||||
if (otherValue != null) {
|
||||
return false;
|
||||
}
|
||||
} else if (!value.equals(otherValue)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean containsKey(Object key) {
|
||||
return containsKey(objectToKey(key));
|
||||
}
|
||||
|
||||
@Override
|
||||
public V get(Object key) {
|
||||
return get(objectToKey(key));
|
||||
}
|
||||
|
||||
@Override
|
||||
public V put(Integer key, V value) {
|
||||
return put(objectToKey(key), value);
|
||||
}
|
||||
|
||||
@Override
|
||||
public V remove(Object key) {
|
||||
return remove(objectToKey(key));
|
||||
}
|
||||
|
||||
@Override
|
||||
public Set<Integer> keySet() {
|
||||
return keySet;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Set<Entry<Integer, V>> entrySet() {
|
||||
return entrySet;
|
||||
}
|
||||
|
||||
private int objectToKey(Object key) {
|
||||
return ((Integer) key).intValue();
|
||||
}
|
||||
|
||||
/**
|
||||
* Locates the index for the given key. This method probes using double hashing.
|
||||
*
|
||||
* @param key the key for an entry in the map.
|
||||
* @return the index where the key was found, or {@code -1} if no entry is found for that key.
|
||||
*/
|
||||
private int indexOf(int key) {
|
||||
int startIndex = hashIndex(key);
|
||||
int index = startIndex;
|
||||
|
||||
for (; ; ) {
|
||||
if (values[index] == null) {
|
||||
// It's available, so no chance that this value exists anywhere in the map.
|
||||
return -1;
|
||||
}
|
||||
if (key == keys[index]) {
|
||||
return index;
|
||||
}
|
||||
|
||||
// Conflict, keep probing ...
|
||||
if ((index = probeNext(index)) == startIndex) {
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/** Returns the hashed index for the given key. */
|
||||
private int hashIndex(int key) {
|
||||
// The array lengths are always a power of two, so we can use a bitmask to stay inside the
|
||||
// array bounds.
|
||||
return hashCode(key) & mask;
|
||||
}
|
||||
|
||||
/** Returns the hash code for the key. */
|
||||
private static int hashCode(int key) {
|
||||
return key;
|
||||
}
|
||||
|
||||
/** Get the next sequential index after {@code index} and wraps if necessary. */
|
||||
private int probeNext(int index) {
|
||||
// The array lengths are always a power of two, so we can use a bitmask to stay inside the
|
||||
// array bounds.
|
||||
return (index + 1) & mask;
|
||||
}
|
||||
|
||||
/** Grows the map size after an insertion. If necessary, performs a rehash of the map. */
|
||||
private void growSize() {
|
||||
size++;
|
||||
|
||||
if (size > maxSize) {
|
||||
if (keys.length == Integer.MAX_VALUE) {
|
||||
throw new IllegalStateException("Max capacity reached at size=" + size);
|
||||
}
|
||||
|
||||
// Double the capacity.
|
||||
rehash(keys.length << 1);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Removes entry at the given index position. Also performs opportunistic, incremental rehashing
|
||||
* if necessary to not break conflict chains.
|
||||
*
|
||||
* @param index the index position of the element to remove.
|
||||
* @return {@code true} if the next item was moved back. {@code false} otherwise.
|
||||
*/
|
||||
private boolean removeAt(final int index) {
|
||||
--size;
|
||||
// Clearing the key is not strictly necessary (for GC like in a regular collection),
|
||||
// but recommended for security. The memory location is still fresh in the cache anyway.
|
||||
keys[index] = 0;
|
||||
values[index] = null;
|
||||
|
||||
// In the interval from index to the next available entry, the arrays may have entries
|
||||
// that are displaced from their base position due to prior conflicts. Iterate these
|
||||
// entries and move them back if possible, optimizing future lookups.
|
||||
// Knuth Section 6.4 Algorithm R, also used by the JDK's IdentityHashMap.
|
||||
|
||||
boolean movedBack = false;
|
||||
int nextFree = index;
|
||||
for (int i = probeNext(index); values[i] != null; i = probeNext(i)) {
|
||||
int bucket = hashIndex(keys[i]);
|
||||
if ((i < bucket && (bucket <= nextFree || nextFree <= i))
|
||||
|| (bucket <= nextFree && nextFree <= i)) {
|
||||
// Move the displaced entry "back" to the first available position.
|
||||
keys[nextFree] = keys[i];
|
||||
values[nextFree] = values[i];
|
||||
movedBack = true;
|
||||
// Put the first entry after the displaced entry
|
||||
keys[i] = 0;
|
||||
values[i] = null;
|
||||
nextFree = i;
|
||||
}
|
||||
}
|
||||
return movedBack;
|
||||
}
|
||||
|
||||
/** Calculates the maximum size allowed before rehashing. */
|
||||
private int calcMaxSize(int capacity) {
|
||||
// Clip the upper bound so that there will always be at least one available slot.
|
||||
int upperBound = capacity - 1;
|
||||
return Math.min(upperBound, (int) (capacity * loadFactor));
|
||||
}
|
||||
|
||||
/**
|
||||
* Rehashes the map for the given capacity.
|
||||
*
|
||||
* @param newCapacity the new capacity for the map.
|
||||
*/
|
||||
private void rehash(int newCapacity) {
|
||||
int[] oldKeys = keys;
|
||||
V[] oldVals = values;
|
||||
|
||||
keys = new int[newCapacity];
|
||||
@SuppressWarnings({"unchecked", "SuspiciousArrayCast"})
|
||||
V[] temp = (V[]) new Object[newCapacity];
|
||||
values = temp;
|
||||
|
||||
maxSize = calcMaxSize(newCapacity);
|
||||
mask = newCapacity - 1;
|
||||
|
||||
// Insert to the new arrays.
|
||||
for (int i = 0; i < oldVals.length; ++i) {
|
||||
V oldVal = oldVals[i];
|
||||
if (oldVal != null) {
|
||||
// Inlined put(), but much simpler: we don't need to worry about
|
||||
// duplicated keys, growing/rehashing, or failing to insert.
|
||||
int oldKey = oldKeys[i];
|
||||
int index = hashIndex(oldKey);
|
||||
|
||||
for (; ; ) {
|
||||
if (values[index] == null) {
|
||||
keys[index] = oldKey;
|
||||
values[index] = oldVal;
|
||||
break;
|
||||
}
|
||||
|
||||
// Conflict, keep probing. Can wrap around, but never reaches startIndex again.
|
||||
index = probeNext(index);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
if (isEmpty()) {
|
||||
return "{}";
|
||||
}
|
||||
StringBuilder sb = new StringBuilder(4 * size);
|
||||
sb.append('{');
|
||||
boolean first = true;
|
||||
for (int i = 0; i < values.length; ++i) {
|
||||
V value = values[i];
|
||||
if (value != null) {
|
||||
if (!first) {
|
||||
sb.append(", ");
|
||||
}
|
||||
sb.append(keyToString(keys[i]))
|
||||
.append('=')
|
||||
.append(value == this ? "(this Map)" : toExternal(value));
|
||||
first = false;
|
||||
}
|
||||
}
|
||||
return sb.append('}').toString();
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper method called by {@link #toString()} in order to convert a single map key into a string.
|
||||
* This is protected to allow subclasses to override the appearance of a given key.
|
||||
*/
|
||||
protected String keyToString(int key) {
|
||||
return Integer.toString(key);
|
||||
}
|
||||
|
||||
/** Set implementation for iterating over the entries of the map. */
|
||||
private final class EntrySet extends AbstractSet<Entry<Integer, V>> {
|
||||
@Override
|
||||
public Iterator<Entry<Integer, V>> iterator() {
|
||||
return new MapIterator();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int size() {
|
||||
return Int2ObjectHashMap.this.size();
|
||||
}
|
||||
}
|
||||
|
||||
/** Set implementation for iterating over the keys. */
|
||||
private final class KeySet extends AbstractSet<Integer> {
|
||||
@Override
|
||||
public int size() {
|
||||
return Int2ObjectHashMap.this.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean contains(Object o) {
|
||||
return Int2ObjectHashMap.this.containsKey(o);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean remove(Object o) {
|
||||
return Int2ObjectHashMap.this.remove(o) != null;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean retainAll(Collection<?> retainedKeys) {
|
||||
boolean changed = false;
|
||||
for (Iterator<PrimitiveEntry<V>> iter = entries().iterator(); iter.hasNext(); ) {
|
||||
PrimitiveEntry<V> entry = iter.next();
|
||||
if (!retainedKeys.contains(entry.key())) {
|
||||
changed = true;
|
||||
iter.remove();
|
||||
}
|
||||
}
|
||||
return changed;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void clear() {
|
||||
Int2ObjectHashMap.this.clear();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Iterator<Integer> iterator() {
|
||||
return new Iterator<Integer>() {
|
||||
private final Iterator<Entry<Integer, V>> iter = entrySet.iterator();
|
||||
|
||||
@Override
|
||||
public boolean hasNext() {
|
||||
return iter.hasNext();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Integer next() {
|
||||
return iter.next().getKey();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void remove() {
|
||||
iter.remove();
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Iterator over primitive entries. Entry key/values are overwritten by each call to {@link
|
||||
* #next()}.
|
||||
*/
|
||||
private final class PrimitiveIterator implements Iterator<PrimitiveEntry<V>>, PrimitiveEntry<V> {
|
||||
private int prevIndex = -1;
|
||||
private int nextIndex = -1;
|
||||
private int entryIndex = -1;
|
||||
|
||||
private void scanNext() {
|
||||
for (; ; ) {
|
||||
if (++nextIndex == values.length || values[nextIndex] != null) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean hasNext() {
|
||||
if (nextIndex == -1) {
|
||||
scanNext();
|
||||
}
|
||||
return nextIndex < keys.length;
|
||||
}
|
||||
|
||||
@Override
|
||||
public PrimitiveEntry<V> next() {
|
||||
if (!hasNext()) {
|
||||
throw new NoSuchElementException();
|
||||
}
|
||||
|
||||
prevIndex = nextIndex;
|
||||
scanNext();
|
||||
|
||||
// Always return the same Entry object, just change its index each time.
|
||||
entryIndex = prevIndex;
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void remove() {
|
||||
if (prevIndex < 0) {
|
||||
throw new IllegalStateException("next must be called before each remove.");
|
||||
}
|
||||
if (removeAt(prevIndex)) {
|
||||
// removeAt may move elements "back" in the array if they have been displaced because
|
||||
// their spot in the array was occupied when they were inserted. If this occurs then the
|
||||
// nextIndex is now invalid and should instead point to the prevIndex which now holds an
|
||||
// element which was "moved back".
|
||||
nextIndex = prevIndex;
|
||||
}
|
||||
prevIndex = -1;
|
||||
}
|
||||
|
||||
// Entry implementation. Since this implementation uses a single Entry, we coalesce that
|
||||
// into the Iterator object (potentially making loop optimization much easier).
|
||||
|
||||
@Override
|
||||
public int key() {
|
||||
return keys[entryIndex];
|
||||
}
|
||||
|
||||
@Override
|
||||
public V value() {
|
||||
return toExternal(values[entryIndex]);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setValue(V value) {
|
||||
values[entryIndex] = toInternal(value);
|
||||
}
|
||||
}
|
||||
|
||||
/** Iterator used by the {@link Map} interface. */
|
||||
private final class MapIterator implements Iterator<Entry<Integer, V>> {
|
||||
private final PrimitiveIterator iter = new PrimitiveIterator();
|
||||
|
||||
@Override
|
||||
public boolean hasNext() {
|
||||
return iter.hasNext();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Entry<Integer, V> next() {
|
||||
if (!hasNext()) {
|
||||
throw new NoSuchElementException();
|
||||
}
|
||||
|
||||
iter.next();
|
||||
|
||||
return new MapEntry(iter.entryIndex);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void remove() {
|
||||
iter.remove();
|
||||
}
|
||||
}
|
||||
|
||||
/** A single entry in the map. */
|
||||
final class MapEntry implements Entry<Integer, V> {
|
||||
private final int entryIndex;
|
||||
|
||||
MapEntry(int entryIndex) {
|
||||
this.entryIndex = entryIndex;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Integer getKey() {
|
||||
verifyExists();
|
||||
return keys[entryIndex];
|
||||
}
|
||||
|
||||
@Override
|
||||
public V getValue() {
|
||||
verifyExists();
|
||||
return toExternal(values[entryIndex]);
|
||||
}
|
||||
|
||||
@Override
|
||||
public V setValue(V value) {
|
||||
verifyExists();
|
||||
V prevValue = toExternal(values[entryIndex]);
|
||||
values[entryIndex] = toInternal(value);
|
||||
return prevValue;
|
||||
}
|
||||
|
||||
private void verifyExists() {
|
||||
if (values[entryIndex] == null) {
|
||||
throw new IllegalStateException("The map entry has been removed");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Fast method of finding the next power of 2 greater than or equal to the supplied value.
|
||||
*
|
||||
* <p>If the value is {@code <= 0} then 1 will be returned. This method is not suitable for {@link
|
||||
* Integer#MIN_VALUE} or numbers greater than 2^30.
|
||||
*
|
||||
* @param value from which to search for next power of 2
|
||||
* @return The next power of 2 or the value itself if it is a power of 2
|
||||
*/
|
||||
private static int findNextPositivePowerOfTwo(final int value) {
|
||||
assert value > Integer.MIN_VALUE && value < 0x40000000;
|
||||
return 1 << (32 - Integer.numberOfLeadingZeros(value - 1));
|
||||
}
|
||||
}
|
87
java/core/src/main/java/com/google/protobuf/JavaType.java
Normal file
87
java/core/src/main/java/com/google/protobuf/JavaType.java
Normal file
|
@ -0,0 +1,87 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
/**
|
||||
* Enum that identifies the Java types required to store protobuf fields.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public enum JavaType {
|
||||
VOID(Void.class, Void.class, null),
|
||||
INT(int.class, Integer.class, 0),
|
||||
LONG(long.class, Long.class, 0L),
|
||||
FLOAT(float.class, Float.class, 0F),
|
||||
DOUBLE(double.class, Double.class, 0D),
|
||||
BOOLEAN(boolean.class, Boolean.class, false),
|
||||
STRING(String.class, String.class, ""),
|
||||
BYTE_STRING(ByteString.class, ByteString.class, ByteString.EMPTY),
|
||||
ENUM(int.class, Integer.class, null),
|
||||
MESSAGE(Object.class, Object.class, null);
|
||||
|
||||
private final Class<?> type;
|
||||
private final Class<?> boxedType;
|
||||
private final Object defaultDefault;
|
||||
|
||||
JavaType(Class<?> type, Class<?> boxedType, Object defaultDefault) {
|
||||
this.type = type;
|
||||
this.boxedType = boxedType;
|
||||
this.defaultDefault = defaultDefault;
|
||||
}
|
||||
|
||||
/**
|
||||
* The default default value for fields of this type, if it's a primitive
|
||||
* type.
|
||||
*/
|
||||
public Object getDefaultDefault() {
|
||||
return defaultDefault;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the required type for a field that would hold a value of this type.
|
||||
*/
|
||||
public Class<?> getType() {
|
||||
return type;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return the boxedType
|
||||
*/
|
||||
public Class<?> getBoxedType() {
|
||||
return boxedType;
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether or not this {@link JavaType} can be applied to a field of the given type.
|
||||
*/
|
||||
public boolean isValidType(Class<?> t) {
|
||||
return type.isAssignableFrom(t);
|
||||
}
|
||||
}
|
158
java/core/src/main/java/com/google/protobuf/MessageInfo.java
Normal file
158
java/core/src/main/java/com/google/protobuf/MessageInfo.java
Normal file
|
@ -0,0 +1,158 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Internal.checkNotNull;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Information for the layout of a protobuf message class. This describes all of the fields
|
||||
* contained within a message.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public final class MessageInfo {
|
||||
private final ProtoSyntax syntax;
|
||||
private final boolean messageSetWireFormat;
|
||||
private final List<FieldInfo> fields;
|
||||
|
||||
/**
|
||||
* Constructor.
|
||||
*
|
||||
* @param fields the set of fields for the message.
|
||||
*/
|
||||
private MessageInfo(ProtoSyntax syntax, boolean messageSetWireFormat, List<FieldInfo> fields) {
|
||||
this.syntax = syntax;
|
||||
this.messageSetWireFormat = messageSetWireFormat;
|
||||
this.fields = fields;
|
||||
}
|
||||
|
||||
/** Gets the syntax for the message (e.g. PROTO2, PROTO3). */
|
||||
public ProtoSyntax getSyntax() {
|
||||
return syntax;
|
||||
}
|
||||
|
||||
/** Indicates whether or not the message should be represented with message set wire format. */
|
||||
public boolean isMessageSetWireFormat() {
|
||||
return messageSetWireFormat;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the information for all fields within this message, sorted in ascending order by their
|
||||
* field number.
|
||||
*/
|
||||
public List<FieldInfo> getFields() {
|
||||
return fields;
|
||||
}
|
||||
|
||||
/** Creates a new map of field number to message class for message fields. */
|
||||
public Int2ObjectHashMap<Class<?>> messageFieldClassMap() {
|
||||
Int2ObjectHashMap<Class<?>> classMap = new Int2ObjectHashMap<Class<?>>();
|
||||
for (int i = 0; i < fields.size(); ++i) {
|
||||
FieldInfo fd = fields.get(i);
|
||||
int fieldNumber = fd.getFieldNumber();
|
||||
|
||||
// Configure messages
|
||||
switch (fd.getType()) {
|
||||
case MESSAGE:
|
||||
classMap.put(fieldNumber, fd.getField().getType());
|
||||
break;
|
||||
case MESSAGE_LIST:
|
||||
classMap.put(fieldNumber, fd.getListElementType());
|
||||
break;
|
||||
case GROUP:
|
||||
classMap.put(fieldNumber, fd.getField().getType());
|
||||
break;
|
||||
case GROUP_LIST:
|
||||
classMap.put(fieldNumber, fd.getListElementType());
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
return classMap;
|
||||
}
|
||||
|
||||
/** Helper method for creating a new builder for {@link MessageInfo}. */
|
||||
public static Builder newBuilder() {
|
||||
return new Builder();
|
||||
}
|
||||
|
||||
/** Helper method for creating a new builder for {@link MessageInfo}. */
|
||||
public static Builder newBuilder(int numFields) {
|
||||
return new Builder(numFields);
|
||||
}
|
||||
|
||||
/** A builder of {@link MessageInfo} instances. */
|
||||
public static final class Builder {
|
||||
private final ArrayList<FieldInfo> fields;
|
||||
private ProtoSyntax syntax;
|
||||
private boolean wasBuilt;
|
||||
private boolean messageSetWireFormat;
|
||||
|
||||
public Builder() {
|
||||
fields = new ArrayList<FieldInfo>();
|
||||
}
|
||||
|
||||
public Builder(int numFields) {
|
||||
fields = new ArrayList<FieldInfo>(numFields);
|
||||
}
|
||||
|
||||
public void withSyntax(ProtoSyntax syntax) {
|
||||
this.syntax = checkNotNull(syntax, "syntax");
|
||||
}
|
||||
|
||||
public void withMessageSetWireFormat(boolean messageSetWireFormat) {
|
||||
this.messageSetWireFormat = messageSetWireFormat;
|
||||
}
|
||||
|
||||
public void add(FieldInfo field) {
|
||||
if (wasBuilt) {
|
||||
throw new IllegalStateException("Builder can only build once");
|
||||
}
|
||||
fields.add(field);
|
||||
}
|
||||
|
||||
public MessageInfo build() {
|
||||
if (wasBuilt) {
|
||||
throw new IllegalStateException("Builder can only build once");
|
||||
}
|
||||
if (syntax == null) {
|
||||
throw new IllegalStateException("Must specify a proto syntax");
|
||||
}
|
||||
wasBuilt = true;
|
||||
Collections.sort(fields);
|
||||
return new MessageInfo(syntax, messageSetWireFormat, Collections.unmodifiableList(fields));
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,38 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
/** A factory that creates {@link MessageInfo} instances for message types. */
|
||||
@ExperimentalApi
|
||||
public interface MessageInfoFactory {
|
||||
/** Returns a information of the message class. */
|
||||
MessageInfo messageInfoFor(Class<?> clazz);
|
||||
}
|
|
@ -0,0 +1,281 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto2Manifest.offset;
|
||||
import static com.google.protobuf.Proto2Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, lookup-based schema that can be used with any proto3-lite message class. The message
|
||||
* class must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto2LiteLookupSchema<T> extends AbstractProto2LiteSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto2LiteLookupSchema(Class<T> messageClass, MessageInfo msgInfo) {
|
||||
super(messageClass, Proto2Manifest.newLookupManifest(msgInfo));
|
||||
this.messageFieldClassMap = msgInfo.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.lookupPositionForFieldNumber(fieldNumber);
|
||||
if (pos >= 0L) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(
|
||||
SchemaUtil.<String>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readProtobufMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
continue;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case -4: //GROUP (actually should be 252, but byte is [-128, 127])
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readGroup(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case -3: //GROUP_LIST (actually should be 253, but byte is [-128, 127])
|
||||
SchemaUtil.readGroupList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
break;
|
||||
default:
|
||||
// Unknown field type - break out of loop and skip the field.
|
||||
break;
|
||||
}
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat it as an unknown field - same as the default case.
|
||||
}
|
||||
}
|
||||
|
||||
// Unknown field.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,291 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto2Manifest.offset;
|
||||
import static com.google.protobuf.Proto2Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, table-based schema that can be used with any proto3 lite message class. The message
|
||||
* class must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto2LiteTableSchema<T> extends AbstractProto2LiteSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto2LiteTableSchema(Class<T> messageClass, MessageInfo descriptor) {
|
||||
super(messageClass, Proto2Manifest.newTableManfiest(descriptor));
|
||||
this.messageFieldClassMap = descriptor.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.tablePositionForFieldNumber(fieldNumber);
|
||||
if (pos < 0) {
|
||||
// Unknown field.
|
||||
if (reader.skipField()) {
|
||||
continue;
|
||||
}
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
|
||||
// Benchmarks have shown that switching on a byte is faster than an enum.
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(
|
||||
SchemaUtil.<String>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readProtobufMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case -4: //GROUP (actually should be 252, but byte is [-128, 127])
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readGroup(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case -3: //GROUP_LIST (actually should be 253, but byte is [-128, 127])
|
||||
SchemaUtil.readGroupList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
break;
|
||||
default:
|
||||
// Assume we've landed on an empty entry. Treat it as an unknown field - just skip it.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
break;
|
||||
}
|
||||
// TODO(nathanmittler): Do we need to make lists immutable?
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat fields with an invalid wire type as unknown fields (i.e. same as the default case).
|
||||
if (!reader.skipField()) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,273 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto2Manifest.offset;
|
||||
import static com.google.protobuf.Proto2Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, lookup-based schema that can be used with any standard (i.e. non-lite) proto3 message
|
||||
* class. The message class must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto2LookupSchema<T> extends AbstractProto2StandardSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto2LookupSchema(Class<T> messageClass, MessageInfo descriptor) {
|
||||
super(messageClass, Proto2Manifest.newLookupManifest(descriptor));
|
||||
this.messageFieldClassMap = descriptor.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.lookupPositionForFieldNumber(fieldNumber);
|
||||
if (pos >= 0L) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(
|
||||
SchemaUtil.<String>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
continue;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case -4: //GROUP (actually should be 252, but byte is [-128, 127])
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readGroup(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
continue;
|
||||
case -3: //GROUP_LIST (actually should be 253, but byte is [-128, 127])
|
||||
SchemaUtil.readGroupList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
continue;
|
||||
default:
|
||||
// Unknown field type - break out of loop and skip the field.
|
||||
break;
|
||||
}
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat it as an unknown field - same as the default case.
|
||||
}
|
||||
}
|
||||
|
||||
// Unknown field.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
261
java/core/src/main/java/com/google/protobuf/Proto2Manifest.java
Normal file
261
java/core/src/main/java/com/google/protobuf/Proto2Manifest.java
Normal file
|
@ -0,0 +1,261 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.List;
|
||||
|
||||
/** Container for the field metadata of a single proto2 schema. */
|
||||
final class Proto2Manifest {
|
||||
static final int INT_LENGTH = 4;
|
||||
static final int LONG_LENGTH = INT_LENGTH * 2;
|
||||
static final int LONGS_PER_FIELD = 2;
|
||||
/**
|
||||
* Note that field length is always a power of two so that we can use bit shifting (rather than
|
||||
* division) to find the location of a field when parsing.
|
||||
*/
|
||||
static final int FIELD_LENGTH = LONGS_PER_FIELD * LONG_LENGTH;
|
||||
|
||||
static final int FIELD_SHIFT = 4 /* 2^4 = 16 */;
|
||||
static final int OFFSET_BITS = 20;
|
||||
static final int OFFSET_MASK = 0XFFFFF;
|
||||
static final long EMPTY_LONG = 0xFFFFFFFFFFFFFFFFL;
|
||||
|
||||
/**
|
||||
* Holds all information for accessing the message fields. The layout is as follows (field
|
||||
* positions are relative to the offset of the start of the field in the buffer):
|
||||
*
|
||||
* <p>
|
||||
*
|
||||
* <pre>
|
||||
* [ 0 - 3] unused
|
||||
* [ 4 - 31] field number
|
||||
* [32 - 37] unused
|
||||
* [38 - 43] field type
|
||||
* [44 - 63] field offset
|
||||
* [64 - 69] unused
|
||||
* [70 - 75] field presence mask shift
|
||||
* [76 - 95] presence field offset
|
||||
* [96 - 127] unused
|
||||
* </pre>
|
||||
*/
|
||||
final ByteBuffer buffer;
|
||||
|
||||
final long address;
|
||||
final long limit;
|
||||
final int numFields;
|
||||
|
||||
final int minFieldNumber;
|
||||
final int maxFieldNumber;
|
||||
|
||||
private Proto2Manifest(
|
||||
ByteBuffer buffer,
|
||||
long address,
|
||||
long limit,
|
||||
int numFields,
|
||||
int minFieldNumber,
|
||||
int maxFieldNumber) {
|
||||
this.buffer = buffer;
|
||||
this.address = address;
|
||||
this.limit = limit;
|
||||
this.numFields = numFields;
|
||||
this.minFieldNumber = minFieldNumber;
|
||||
this.maxFieldNumber = maxFieldNumber;
|
||||
}
|
||||
|
||||
boolean isFieldInRange(int fieldNumber) {
|
||||
return fieldNumber >= minFieldNumber && fieldNumber <= maxFieldNumber;
|
||||
}
|
||||
|
||||
long tablePositionForFieldNumber(int fieldNumber) {
|
||||
if (fieldNumber < minFieldNumber || fieldNumber > maxFieldNumber) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
return indexToAddress(fieldNumber - minFieldNumber);
|
||||
}
|
||||
|
||||
<T> boolean isFieldPresent(T message, long pos) {
|
||||
int maskShiftAndOffset = UnsafeUtil.getInt(pos + LONG_LENGTH);
|
||||
long offset = maskShiftAndOffset & OFFSET_MASK;
|
||||
int mask = 1 << (maskShiftAndOffset >>> OFFSET_BITS);
|
||||
return (UnsafeUtil.getInt(message, offset) & mask) != 0;
|
||||
}
|
||||
|
||||
<T> void setFieldPresent(T message, long pos) {
|
||||
int maskShiftAndOffset = UnsafeUtil.getInt(pos + LONG_LENGTH);
|
||||
long offset = maskShiftAndOffset & OFFSET_MASK;
|
||||
int mask = 1 << (maskShiftAndOffset >>> OFFSET_BITS);
|
||||
UnsafeUtil.putInt(message, offset, UnsafeUtil.getInt(message, offset) | mask);
|
||||
}
|
||||
|
||||
long lookupPositionForFieldNumber(int fieldNumber) {
|
||||
int min = 0;
|
||||
int max = numFields - 1;
|
||||
while (min <= max) {
|
||||
// Find the midpoint address.
|
||||
int mid = (max + min) >>> 1;
|
||||
long midAddress = indexToAddress(mid);
|
||||
int midFieldNumber = numberAt(midAddress);
|
||||
if (fieldNumber == midFieldNumber) {
|
||||
// Found the field.
|
||||
return midAddress;
|
||||
}
|
||||
if (fieldNumber < midFieldNumber) {
|
||||
// Search the lower half.
|
||||
max = mid - 1;
|
||||
} else {
|
||||
// Search the upper half.
|
||||
min = mid + 1;
|
||||
}
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
int numberAt(long pos) {
|
||||
return UnsafeUtil.getInt(pos);
|
||||
}
|
||||
|
||||
int typeAndOffsetAt(long pos) {
|
||||
return UnsafeUtil.getInt(pos + INT_LENGTH);
|
||||
}
|
||||
|
||||
private long indexToAddress(int index) {
|
||||
return address + (index << FIELD_SHIFT);
|
||||
}
|
||||
|
||||
static byte type(int value) {
|
||||
return (byte) (value >>> OFFSET_BITS);
|
||||
}
|
||||
|
||||
static long offset(int value) {
|
||||
return value & OFFSET_MASK;
|
||||
}
|
||||
|
||||
static Proto2Manifest newTableManfiest(MessageInfo descriptor) {
|
||||
List<FieldInfo> fds = descriptor.getFields();
|
||||
if (fds.isEmpty()) {
|
||||
throw new IllegalArgumentException("Table-based schema requires at least one field");
|
||||
}
|
||||
|
||||
// Set up the buffer for direct indexing by field number.
|
||||
final int minFieldNumber = fds.get(0).getFieldNumber();
|
||||
final int maxFieldNumber = fds.get(fds.size() - 1).getFieldNumber();
|
||||
final int numEntries = (maxFieldNumber - minFieldNumber) + 1;
|
||||
|
||||
int bufferLength = numEntries * FIELD_LENGTH;
|
||||
ByteBuffer buffer = ByteBuffer.allocateDirect(bufferLength + LONG_LENGTH);
|
||||
long tempAddress = UnsafeUtil.addressOffset(buffer);
|
||||
if ((tempAddress & 7L) != 0) {
|
||||
// Make sure that the memory address is 8-byte aligned.
|
||||
tempAddress = (tempAddress & ~7L) + LONG_LENGTH;
|
||||
}
|
||||
final long address = tempAddress;
|
||||
final long limit = address + bufferLength;
|
||||
|
||||
// Fill in the manifest data from the descriptors.
|
||||
int fieldIndex = 0;
|
||||
FieldInfo fd = fds.get(fieldIndex++);
|
||||
for (int bufferIndex = 0; bufferIndex < bufferLength; bufferIndex += FIELD_LENGTH) {
|
||||
final int fieldNumber = fd.getFieldNumber();
|
||||
if (bufferIndex < ((fieldNumber - minFieldNumber) << FIELD_SHIFT)) {
|
||||
// Mark this entry as "empty".
|
||||
long skipLimit = address + bufferIndex + FIELD_LENGTH;
|
||||
for (long skipPos = address + bufferIndex; skipPos < skipLimit; skipPos += LONG_LENGTH) {
|
||||
UnsafeUtil.putLong(skipPos, EMPTY_LONG);
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
// We found the entry for the next field. Store the entry in the manifest for
|
||||
// this field and increment the field index.
|
||||
FieldType type = fd.getType();
|
||||
long pos = address + bufferIndex;
|
||||
UnsafeUtil.putInt(pos, fieldNumber);
|
||||
UnsafeUtil.putInt(
|
||||
pos + INT_LENGTH,
|
||||
(type.id() << OFFSET_BITS) | (int) UnsafeUtil.objectFieldOffset(fd.getField()));
|
||||
if (!type.isList()) {
|
||||
int presenceOffset = (int) UnsafeUtil.objectFieldOffset(fd.getPresenceField());
|
||||
int maskShift = Integer.numberOfTrailingZeros(fd.getPresenceMask());
|
||||
UnsafeUtil.putInt(pos + LONG_LENGTH, maskShift << OFFSET_BITS | presenceOffset);
|
||||
}
|
||||
|
||||
// Advance to the next field, unless we're at the end.
|
||||
if (fieldIndex < fds.size()) {
|
||||
fd = fds.get(fieldIndex++);
|
||||
}
|
||||
}
|
||||
|
||||
return new Proto2Manifest(buffer, address, limit, fds.size(), minFieldNumber, maxFieldNumber);
|
||||
}
|
||||
|
||||
static Proto2Manifest newLookupManifest(MessageInfo descriptor) {
|
||||
List<FieldInfo> fds = descriptor.getFields();
|
||||
|
||||
final int numFields = fds.size();
|
||||
int bufferLength = numFields * FIELD_LENGTH;
|
||||
final ByteBuffer buffer = ByteBuffer.allocateDirect(bufferLength + LONG_LENGTH);
|
||||
long tempAddress = UnsafeUtil.addressOffset(buffer);
|
||||
if ((tempAddress & 7L) != 0) {
|
||||
// Make sure that the memory address is 8-byte aligned.
|
||||
tempAddress = (tempAddress & ~7L) + LONG_LENGTH;
|
||||
}
|
||||
final long address = tempAddress;
|
||||
final long limit = address + bufferLength;
|
||||
|
||||
// Allocate and populate the data buffer.
|
||||
long pos = address;
|
||||
for (int i = 0; i < fds.size(); ++i, pos += FIELD_LENGTH) {
|
||||
FieldInfo fd = fds.get(i);
|
||||
UnsafeUtil.putInt(pos, fd.getFieldNumber());
|
||||
UnsafeUtil.putInt(
|
||||
pos + INT_LENGTH,
|
||||
(fd.getType().id() << OFFSET_BITS) | (int) UnsafeUtil.objectFieldOffset(fd.getField()));
|
||||
if (!fd.getType().isList()) {
|
||||
int presenceOffset = (int) UnsafeUtil.objectFieldOffset(fd.getPresenceField());
|
||||
int maskShift = Integer.numberOfTrailingZeros(fd.getPresenceMask());
|
||||
UnsafeUtil.putInt(pos + LONG_LENGTH, maskShift << OFFSET_BITS | presenceOffset);
|
||||
}
|
||||
}
|
||||
|
||||
if (numFields > 0) {
|
||||
return new Proto2Manifest(
|
||||
buffer,
|
||||
address,
|
||||
limit,
|
||||
numFields,
|
||||
fds.get(0).getFieldNumber(),
|
||||
fds.get(numFields - 1).getFieldNumber());
|
||||
}
|
||||
return new Proto2Manifest(buffer, address, limit, numFields, -1, -1);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,112 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Internal.checkNotNull;
|
||||
|
||||
/** Manufactures instances of {@link Proto3TableSchema}. */
|
||||
@ExperimentalApi
|
||||
public final class Proto2SchemaFactory implements SchemaFactory {
|
||||
/**
|
||||
* The mode with which to generate schemas.
|
||||
*
|
||||
* <p>For testing purposes only.
|
||||
*/
|
||||
public enum Mode {
|
||||
/** Always use a table-based indexing of fields. */
|
||||
TABLE,
|
||||
|
||||
/** Always used lookup-based (i.e. binary search) indexing of fields. */
|
||||
LOOKUP,
|
||||
|
||||
/**
|
||||
* Default. Determine the appropriate field indexing mode based on how sparse the field numbers
|
||||
* are for the message.
|
||||
*/
|
||||
DYNAMIC
|
||||
}
|
||||
|
||||
private final MessageInfoFactory messageDescriptorFactory;
|
||||
private final Mode mode;
|
||||
|
||||
public Proto2SchemaFactory() {
|
||||
this(DescriptorMessageInfoFactory.getInstance());
|
||||
}
|
||||
|
||||
public Proto2SchemaFactory(MessageInfoFactory messageDescriptorFactory) {
|
||||
this(messageDescriptorFactory, Mode.DYNAMIC);
|
||||
}
|
||||
|
||||
/** For testing purposes only. Allows specification of {@link Mode}. */
|
||||
public Proto2SchemaFactory(MessageInfoFactory messageDescriptorFactory, Mode mode) {
|
||||
if (!isSupported()) {
|
||||
throw new IllegalStateException("Schema factory is unsupported on this platform");
|
||||
}
|
||||
this.messageDescriptorFactory =
|
||||
checkNotNull(messageDescriptorFactory, "messageDescriptorFactory");
|
||||
this.mode = checkNotNull(mode, "mode");
|
||||
}
|
||||
|
||||
public static boolean isSupported() {
|
||||
return UnsafeUtil.hasUnsafeArrayOperations() && UnsafeUtil.hasUnsafeByteBufferOperations();
|
||||
}
|
||||
|
||||
@Override
|
||||
public <T> Schema<T> createSchema(Class<T> messageType) {
|
||||
SchemaUtil.requireGeneratedMessage(messageType);
|
||||
|
||||
MessageInfo descriptor = messageDescriptorFactory.messageInfoFor(messageType);
|
||||
switch (mode) {
|
||||
case TABLE:
|
||||
return newTableSchema(messageType, descriptor);
|
||||
case LOOKUP:
|
||||
return newLookupSchema(messageType, descriptor);
|
||||
default:
|
||||
return SchemaUtil.shouldUseTableSwitch(descriptor.getFields())
|
||||
? newTableSchema(messageType, descriptor)
|
||||
: newLookupSchema(messageType, descriptor);
|
||||
}
|
||||
}
|
||||
|
||||
private <T> Schema<T> newTableSchema(Class<T> messageType, MessageInfo descriptor) {
|
||||
if (GeneratedMessageLite.class.isAssignableFrom(messageType)) {
|
||||
return new Proto2LiteTableSchema<T>(messageType, descriptor);
|
||||
}
|
||||
return new Proto2TableSchema<T>(messageType, descriptor);
|
||||
}
|
||||
|
||||
private <T> Schema<T> newLookupSchema(Class<T> messageType, MessageInfo descriptor) {
|
||||
if (GeneratedMessageLite.class.isAssignableFrom(messageType)) {
|
||||
return new Proto2LiteLookupSchema<T>(messageType, descriptor);
|
||||
}
|
||||
return new Proto2LookupSchema<T>(messageType, descriptor);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,270 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto2Manifest.offset;
|
||||
import static com.google.protobuf.Proto2Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, table-based schema that can be used with any proto2 message class. The message class
|
||||
* must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto2TableSchema<T> extends AbstractProto2StandardSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto2TableSchema(Class<T> messageClass, MessageInfo descriptor) {
|
||||
super(messageClass, Proto2Manifest.newTableManfiest(descriptor));
|
||||
this.messageFieldClassMap = descriptor.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.tablePositionForFieldNumber(fieldNumber);
|
||||
if (pos < 0) {
|
||||
// Unknown field.
|
||||
if (reader.skipField()) {
|
||||
continue;
|
||||
}
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
|
||||
// Benchmarks have shown that switching on a byte is faster than an enum.
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(SchemaUtil.<String>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case -4: //GROUP (actually should be 252, but byte is [-128, 127])
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readGroup(messageFieldClassMap.get(fieldNumber)));
|
||||
manifest.setFieldPresent(message, pos);
|
||||
break;
|
||||
case -3: //GROUP_LIST (actually should be 253, but byte is [-128, 127])
|
||||
SchemaUtil.readGroupList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
break;
|
||||
default:
|
||||
// Assume we've landed on an empty entry. Treat it as an unknown field - just skip it.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
break;
|
||||
}
|
||||
// TODO(nathanmittler): Do we need to make lists immutable?
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat fields with an invalid wire type as unknown fields (i.e. same as the default case).
|
||||
if (!reader.skipField()) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,253 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto3Manifest.offset;
|
||||
import static com.google.protobuf.Proto3Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, lookup-based schema that can be used with any proto3-lite message class. The message
|
||||
* class must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto3LiteLookupSchema<T> extends AbstractProto3LiteSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto3LiteLookupSchema(Class<T> messageClass, MessageInfo descriptor) {
|
||||
super(messageClass, Proto3Manifest.newLookupManifest(descriptor));
|
||||
this.messageFieldClassMap = descriptor.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.lookupPositionForFieldNumber(fieldNumber);
|
||||
if (pos >= 0L) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
continue;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
continue;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
continue;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
continue;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
continue;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
continue;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
continue;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
continue;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
continue;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
continue;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
continue;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
continue;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
continue;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
continue;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
continue;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
continue;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
continue;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(
|
||||
SchemaUtil.<String>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readProtobufMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
continue;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
default:
|
||||
// Unknown field type - break out of loop and skip the field.
|
||||
break;
|
||||
}
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat it as an unknown field - same as the default case.
|
||||
}
|
||||
}
|
||||
|
||||
// Unknown field.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,263 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto3Manifest.offset;
|
||||
import static com.google.protobuf.Proto3Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, table-based schema that can be used with any proto3 lite message class. The message
|
||||
* class must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto3LiteTableSchema<T> extends AbstractProto3LiteSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto3LiteTableSchema(Class<T> messageClass, MessageInfo descriptor) {
|
||||
super(messageClass, Proto3Manifest.newTableManfiest(descriptor));
|
||||
this.messageFieldClassMap = descriptor.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.tablePositionForFieldNumber(fieldNumber);
|
||||
if (pos < 0) {
|
||||
// Unknown field.
|
||||
if (reader.skipField()) {
|
||||
continue;
|
||||
}
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
|
||||
// Benchmarks have shown that switching on a byte is faster than an enum.
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
break;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
break;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
break;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
break;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
break;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
break;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
break;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
break;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
break;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
break;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
break;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
break;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
break;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
break;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(
|
||||
SchemaUtil.<String>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readProtobufMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(
|
||||
SchemaUtil.<Float>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(
|
||||
SchemaUtil.<Long>mutableProtobufListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
default:
|
||||
// Assume we've landed on an empty entry. Treat it as an unknown field - just skip it.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
break;
|
||||
}
|
||||
// TODO(nathanmittler): Do we need to make lists immutable?
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat fields with an invalid wire type as unknown fields (i.e. same as the default case).
|
||||
if (!reader.skipField()) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,245 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto3Manifest.offset;
|
||||
import static com.google.protobuf.Proto3Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, lookup-based schema that can be used with any standard (i.e. non-lite) proto3 message
|
||||
* class. The message class must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto3LookupSchema<T> extends AbstractProto3StandardSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto3LookupSchema(Class<T> messageClass, MessageInfo descriptor) {
|
||||
super(messageClass, Proto3Manifest.newTableManfiest(descriptor));
|
||||
this.messageFieldClassMap = descriptor.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.lookupPositionForFieldNumber(fieldNumber);
|
||||
if (pos >= 0L) {
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
continue;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
continue;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
continue;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
continue;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
continue;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
continue;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
continue;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
continue;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
continue;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
continue;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
continue;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
continue;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
continue;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
continue;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
continue;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
continue;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
continue;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(
|
||||
SchemaUtil.<String>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
continue;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(
|
||||
SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(
|
||||
SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(
|
||||
SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
continue;
|
||||
default:
|
||||
// Unknown field type - break out of loop and skip the field.
|
||||
break;
|
||||
}
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat it as an unknown field - same as the default case.
|
||||
}
|
||||
}
|
||||
|
||||
// Unknown field.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
233
java/core/src/main/java/com/google/protobuf/Proto3Manifest.java
Normal file
233
java/core/src/main/java/com/google/protobuf/Proto3Manifest.java
Normal file
|
@ -0,0 +1,233 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.List;
|
||||
|
||||
/** Container for the field metadata of a single proto3 schema. */
|
||||
final class Proto3Manifest {
|
||||
static final int INT_LENGTH = 4;
|
||||
static final int LONG_LENGTH = INT_LENGTH * 2;
|
||||
static final int LONGS_PER_FIELD = 2;
|
||||
/**
|
||||
* Note that field length is always a power of two so that we can use bit shifting (rather than
|
||||
* division) to find the location of a field when parsing.
|
||||
*/
|
||||
static final int FIELD_LENGTH = LONGS_PER_FIELD * LONG_LENGTH;
|
||||
|
||||
static final int FIELD_SHIFT = 4 /* 2^4 = 16 */;
|
||||
static final int OFFSET_BITS = 20;
|
||||
static final int OFFSET_MASK = 0XFFFFF;
|
||||
static final long EMPTY_LONG = 0xFFFFFFFFFFFFFFFFL;
|
||||
|
||||
/**
|
||||
* Holds all information for accessing the message fields. The layout is as follows (field
|
||||
* positions are relative to the offset of the start of the field in the buffer):
|
||||
*
|
||||
* <p>
|
||||
*
|
||||
* <pre>
|
||||
* [ 0 - 3] unused
|
||||
* [ 4 - 31] field number
|
||||
* [32 - 37] unused
|
||||
* [38 - 43] field type
|
||||
* [44 - 63] field offset
|
||||
* [64 - 127] unused
|
||||
* </pre>
|
||||
*/
|
||||
final ByteBuffer buffer;
|
||||
|
||||
final long address;
|
||||
final long limit;
|
||||
final int numFields;
|
||||
|
||||
final int minFieldNumber;
|
||||
final int maxFieldNumber;
|
||||
|
||||
private Proto3Manifest(
|
||||
ByteBuffer buffer,
|
||||
long address,
|
||||
long limit,
|
||||
int numFields,
|
||||
int minFieldNumber,
|
||||
int maxFieldNumber) {
|
||||
this.buffer = buffer;
|
||||
this.address = address;
|
||||
this.limit = limit;
|
||||
this.numFields = numFields;
|
||||
this.minFieldNumber = minFieldNumber;
|
||||
this.maxFieldNumber = maxFieldNumber;
|
||||
}
|
||||
|
||||
boolean isFieldInRange(int fieldNumber) {
|
||||
return fieldNumber >= minFieldNumber && fieldNumber <= maxFieldNumber;
|
||||
}
|
||||
|
||||
long tablePositionForFieldNumber(int fieldNumber) {
|
||||
if (fieldNumber < minFieldNumber || fieldNumber > maxFieldNumber) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
return indexToAddress(fieldNumber - minFieldNumber);
|
||||
}
|
||||
|
||||
long lookupPositionForFieldNumber(int fieldNumber) {
|
||||
int min = 0;
|
||||
int max = numFields - 1;
|
||||
while (min <= max) {
|
||||
// Find the midpoint address.
|
||||
int mid = (max + min) >>> 1;
|
||||
long midAddress = indexToAddress(mid);
|
||||
int midFieldNumber = numberAt(midAddress);
|
||||
if (fieldNumber == midFieldNumber) {
|
||||
// Found the field.
|
||||
return midAddress;
|
||||
}
|
||||
if (fieldNumber < midFieldNumber) {
|
||||
// Search the lower half.
|
||||
max = mid - 1;
|
||||
} else {
|
||||
// Search the upper half.
|
||||
min = mid + 1;
|
||||
}
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
int numberAt(long pos) {
|
||||
return UnsafeUtil.getInt(pos);
|
||||
}
|
||||
|
||||
int typeAndOffsetAt(long pos) {
|
||||
return UnsafeUtil.getInt(pos + INT_LENGTH);
|
||||
}
|
||||
|
||||
private long indexToAddress(int index) {
|
||||
return address + (index << FIELD_SHIFT);
|
||||
}
|
||||
|
||||
static byte type(int value) {
|
||||
return (byte) (value >>> OFFSET_BITS);
|
||||
}
|
||||
|
||||
static long offset(int value) {
|
||||
return value & OFFSET_MASK;
|
||||
}
|
||||
|
||||
static Proto3Manifest newTableManfiest(MessageInfo descriptor) {
|
||||
List<FieldInfo> fds = descriptor.getFields();
|
||||
if (fds.isEmpty()) {
|
||||
throw new IllegalArgumentException("Table-based schema requires at least one field");
|
||||
}
|
||||
|
||||
// Set up the buffer for direct indexing by field number.
|
||||
final int minFieldNumber = fds.get(0).getFieldNumber();
|
||||
final int maxFieldNumber = fds.get(fds.size() - 1).getFieldNumber();
|
||||
final int numEntries = (maxFieldNumber - minFieldNumber) + 1;
|
||||
|
||||
int bufferLength = numEntries * FIELD_LENGTH;
|
||||
ByteBuffer buffer = ByteBuffer.allocateDirect(bufferLength + LONG_LENGTH);
|
||||
long tempAddress = UnsafeUtil.addressOffset(buffer);
|
||||
if ((tempAddress & 7L) != 0) {
|
||||
// Make sure that the memory address is 8-byte aligned.
|
||||
tempAddress = (tempAddress & ~7L) + LONG_LENGTH;
|
||||
}
|
||||
final long address = tempAddress;
|
||||
final long limit = address + bufferLength;
|
||||
|
||||
// Fill in the manifest data from the descriptors.
|
||||
int fieldIndex = 0;
|
||||
FieldInfo fd = fds.get(fieldIndex++);
|
||||
for (int bufferIndex = 0; bufferIndex < bufferLength; bufferIndex += FIELD_LENGTH) {
|
||||
final int fieldNumber = fd.getFieldNumber();
|
||||
if (bufferIndex < ((fieldNumber - minFieldNumber) << FIELD_SHIFT)) {
|
||||
// Mark this entry as "empty".
|
||||
long skipLimit = address + bufferIndex + FIELD_LENGTH;
|
||||
for (long skipPos = address + bufferIndex; skipPos < skipLimit; skipPos += LONG_LENGTH) {
|
||||
UnsafeUtil.putLong(skipPos, EMPTY_LONG);
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
// We found the entry for the next field. Store the entry in the manifest for
|
||||
// this field and increment the field index.
|
||||
FieldType type = fd.getType();
|
||||
UnsafeUtil.putInt(address + bufferIndex, fieldNumber);
|
||||
UnsafeUtil.putInt(
|
||||
address + bufferIndex + INT_LENGTH,
|
||||
(type.id() << OFFSET_BITS) | (int) UnsafeUtil.objectFieldOffset(fd.getField()));
|
||||
|
||||
// Advance to the next field, unless we're at the end.
|
||||
if (fieldIndex < fds.size()) {
|
||||
fd = fds.get(fieldIndex++);
|
||||
}
|
||||
}
|
||||
|
||||
return new Proto3Manifest(buffer, address, limit, fds.size(), minFieldNumber, maxFieldNumber);
|
||||
}
|
||||
|
||||
static Proto3Manifest newLookupManifest(MessageInfo descriptor) {
|
||||
List<FieldInfo> fds = descriptor.getFields();
|
||||
|
||||
final int numFields = fds.size();
|
||||
int bufferLength = numFields * FIELD_LENGTH;
|
||||
final ByteBuffer buffer = ByteBuffer.allocateDirect(bufferLength + LONG_LENGTH);
|
||||
long tempAddress = UnsafeUtil.addressOffset(buffer);
|
||||
if ((tempAddress & 7L) != 0) {
|
||||
// Make sure that the memory address is 8-byte aligned.
|
||||
tempAddress = (tempAddress & ~7L) + LONG_LENGTH;
|
||||
}
|
||||
final long address = tempAddress;
|
||||
final long limit = address + bufferLength;
|
||||
|
||||
// Allocate and populate the data buffer.
|
||||
long pos = address;
|
||||
for (int i = 0; i < fds.size(); ++i, pos += FIELD_LENGTH) {
|
||||
FieldInfo fd = fds.get(i);
|
||||
UnsafeUtil.putInt(pos, fd.getFieldNumber());
|
||||
UnsafeUtil.putInt(
|
||||
pos + INT_LENGTH,
|
||||
(fd.getType().id() << OFFSET_BITS) | (int) UnsafeUtil.objectFieldOffset(fd.getField()));
|
||||
}
|
||||
|
||||
if (numFields > 0) {
|
||||
return new Proto3Manifest(
|
||||
buffer,
|
||||
address,
|
||||
limit,
|
||||
numFields,
|
||||
fds.get(0).getFieldNumber(),
|
||||
fds.get(numFields - 1).getFieldNumber());
|
||||
}
|
||||
return new Proto3Manifest(buffer, address, limit, numFields, -1, -1);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,115 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Internal.checkNotNull;
|
||||
|
||||
/**
|
||||
* Manufactures schemas for proto3 messages. Message classes must extend {@link
|
||||
* com.google.protobuf.GeneratedMessage} or {@link com.google.protobuf.GeneratedMessageLite}.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public final class Proto3SchemaFactory implements SchemaFactory {
|
||||
/**
|
||||
* The mode with which to generate schemas.
|
||||
*
|
||||
* <p>For testing purposes only.
|
||||
*/
|
||||
public enum Mode {
|
||||
/** Always use a table-based indexing of fields. */
|
||||
TABLE,
|
||||
|
||||
/** Always used lookup-based (i.e. binary search) indexing of fields. */
|
||||
LOOKUP,
|
||||
|
||||
/**
|
||||
* Default. Determine the appropriate field indexing mode based on how sparse the field numbers
|
||||
* are for the message.
|
||||
*/
|
||||
DYNAMIC
|
||||
}
|
||||
|
||||
private final MessageInfoFactory messageDescriptorFactory;
|
||||
private final Mode mode;
|
||||
|
||||
public Proto3SchemaFactory() {
|
||||
this(DescriptorMessageInfoFactory.getInstance());
|
||||
}
|
||||
|
||||
public Proto3SchemaFactory(MessageInfoFactory messageDescriptorFactory) {
|
||||
this(messageDescriptorFactory, Mode.DYNAMIC);
|
||||
}
|
||||
|
||||
/** For testing purposes only. Allows specification of {@link Mode}. */
|
||||
public Proto3SchemaFactory(MessageInfoFactory messageDescriptorFactory, Mode mode) {
|
||||
if (!isSupported()) {
|
||||
throw new IllegalStateException("Schema factory is unsupported on this platform");
|
||||
}
|
||||
this.messageDescriptorFactory =
|
||||
checkNotNull(messageDescriptorFactory, "messageDescriptorFactory");
|
||||
this.mode = checkNotNull(mode, "mode");
|
||||
}
|
||||
|
||||
public static boolean isSupported() {
|
||||
return UnsafeUtil.hasUnsafeArrayOperations() && UnsafeUtil.hasUnsafeByteBufferOperations();
|
||||
}
|
||||
|
||||
@Override
|
||||
public <T> Schema<T> createSchema(Class<T> messageType) {
|
||||
SchemaUtil.requireGeneratedMessage(messageType);
|
||||
|
||||
MessageInfo descriptor = messageDescriptorFactory.messageInfoFor(messageType);
|
||||
switch (mode) {
|
||||
case TABLE:
|
||||
return newTableSchema(messageType, descriptor);
|
||||
case LOOKUP:
|
||||
return newLookupSchema(messageType, descriptor);
|
||||
default:
|
||||
return SchemaUtil.shouldUseTableSwitch(descriptor.getFields())
|
||||
? newTableSchema(messageType, descriptor)
|
||||
: newLookupSchema(messageType, descriptor);
|
||||
}
|
||||
}
|
||||
|
||||
private <T> Schema<T> newTableSchema(Class<T> messageType, MessageInfo descriptor) {
|
||||
if (GeneratedMessageLite.class.isAssignableFrom(messageType)) {
|
||||
return new Proto3LiteTableSchema<T>(messageType, descriptor);
|
||||
}
|
||||
return new Proto3TableSchema<T>(messageType, descriptor);
|
||||
}
|
||||
|
||||
private <T> Schema<T> newLookupSchema(Class<T> messageType, MessageInfo descriptor) {
|
||||
if (GeneratedMessageLite.class.isAssignableFrom(messageType)) {
|
||||
return new Proto3LiteLookupSchema<T>(messageType, descriptor);
|
||||
}
|
||||
return new Proto3LookupSchema<T>(messageType, descriptor);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,242 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Proto3Manifest.offset;
|
||||
import static com.google.protobuf.Proto3Manifest.type;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A generic, table-based schema that can be used with any proto3 message class. The message class
|
||||
* must extend {@link GeneratedMessage}.
|
||||
*/
|
||||
final class Proto3TableSchema<T> extends AbstractProto3StandardSchema<T> {
|
||||
private final Int2ObjectHashMap<Class<?>> messageFieldClassMap;
|
||||
|
||||
Proto3TableSchema(Class<T> messageClass, MessageInfo descriptor) {
|
||||
super(messageClass, Proto3Manifest.newTableManfiest(descriptor));
|
||||
this.messageFieldClassMap = descriptor.messageFieldClassMap();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void mergeFrom(T message, Reader reader) throws IOException {
|
||||
while (true) {
|
||||
final int fieldNumber = reader.getFieldNumber();
|
||||
final long pos = manifest.tablePositionForFieldNumber(fieldNumber);
|
||||
if (pos < 0) {
|
||||
// Unknown field.
|
||||
if (reader.skipField()) {
|
||||
continue;
|
||||
}
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
final int typeAndOffset = manifest.typeAndOffsetAt(pos);
|
||||
|
||||
// Benchmarks have shown that switching on a byte is faster than an enum.
|
||||
try {
|
||||
switch (type(typeAndOffset)) {
|
||||
case 0: //DOUBLE:
|
||||
UnsafeUtil.putDouble(message, offset(typeAndOffset), reader.readDouble());
|
||||
break;
|
||||
case 1: //FLOAT:
|
||||
UnsafeUtil.putFloat(message, offset(typeAndOffset), reader.readFloat());
|
||||
break;
|
||||
case 2: //INT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readInt64());
|
||||
break;
|
||||
case 3: //UINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readUInt64());
|
||||
break;
|
||||
case 4: //INT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readInt32());
|
||||
break;
|
||||
case 5: //FIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readFixed64());
|
||||
break;
|
||||
case 6: //FIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readFixed32());
|
||||
break;
|
||||
case 7: //BOOL:
|
||||
UnsafeUtil.putBoolean(message, offset(typeAndOffset), reader.readBool());
|
||||
break;
|
||||
case 8: //STRING:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readString());
|
||||
break;
|
||||
case 9: //MESSAGE:
|
||||
UnsafeUtil.putObject(
|
||||
message,
|
||||
offset(typeAndOffset),
|
||||
reader.readMessage(messageFieldClassMap.get(fieldNumber)));
|
||||
break;
|
||||
case 10: //BYTES:
|
||||
UnsafeUtil.putObject(message, offset(typeAndOffset), reader.readBytes());
|
||||
break;
|
||||
case 11: //UINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readUInt32());
|
||||
break;
|
||||
case 12: //ENUM:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readEnum());
|
||||
break;
|
||||
case 13: //SFIXED32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSFixed32());
|
||||
break;
|
||||
case 14: //SFIXED64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSFixed64());
|
||||
break;
|
||||
case 15: //SINT32:
|
||||
UnsafeUtil.putInt(message, offset(typeAndOffset), reader.readSInt32());
|
||||
break;
|
||||
case 16: //SINT64:
|
||||
UnsafeUtil.putLong(message, offset(typeAndOffset), reader.readSInt64());
|
||||
break;
|
||||
case 17: //DOUBLE_LIST:
|
||||
reader.readDoubleList(SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 18: //FLOAT_LIST:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 19: //INT64_LIST:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 20: //UINT64_LIST:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 21: //INT32_LIST:
|
||||
reader.readInt32List(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 22: //FIXED64_LIST:
|
||||
reader.readFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 23: //FIXED32_LIST:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 24: //BOOL_LIST:
|
||||
reader.readBoolList(SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 25: //STRING_LIST:
|
||||
reader.readStringList(SchemaUtil.<String>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 26: //MESSAGE_LIST:
|
||||
SchemaUtil.readMessageList(
|
||||
message, offset(typeAndOffset), reader, messageFieldClassMap.get(fieldNumber));
|
||||
break;
|
||||
case 27: //BYTES_LIST:
|
||||
reader.readBytesList(
|
||||
SchemaUtil.<ByteString>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 28: //UINT32_LIST:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 29: //ENUM_LIST:
|
||||
reader.readEnumList(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 30: //SFIXED32_LIST:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 31: //SFIXED64_LIST:
|
||||
reader.readSFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 32: //SINT32_LIST:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 33: //SINT64_LIST:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 34: //DOUBLE_LIST_PACKED:
|
||||
reader.readDoubleList(SchemaUtil.<Double>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 35: //FLOAT_LIST_PACKED:
|
||||
reader.readFloatList(SchemaUtil.<Float>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 36: //INT64_LIST_PACKED:
|
||||
reader.readInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 37: //UINT64_LIST_PACKED:
|
||||
reader.readUInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 38: //INT32_LIST_PACKED:
|
||||
reader.readInt32List(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 39: //FIXED64_LIST_PACKED:
|
||||
reader.readFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 40: //FIXED32_LIST_PACKED:
|
||||
reader.readFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 41: //BOOL_LIST_PACKED:
|
||||
reader.readBoolList(SchemaUtil.<Boolean>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 42: //UINT32_LIST_PACKED:
|
||||
reader.readUInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 43: //ENUM_LIST_PACKED:
|
||||
reader.readEnumList(SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 44: //SFIXED32_LIST_PACKED:
|
||||
reader.readSFixed32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 45: //SFIXED64_LIST_PACKED:
|
||||
reader.readSFixed64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 46: //SINT32_LIST_PACKED:
|
||||
reader.readSInt32List(
|
||||
SchemaUtil.<Integer>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
case 47: //SINT64_LIST_PACKED:
|
||||
reader.readSInt64List(SchemaUtil.<Long>mutableListAt(message, offset(typeAndOffset)));
|
||||
break;
|
||||
default:
|
||||
// Assume we've landed on an empty entry. Treat it as an unknown field - just skip it.
|
||||
if (!reader.skipField()) {
|
||||
// Done reading.
|
||||
return;
|
||||
}
|
||||
break;
|
||||
}
|
||||
// TODO(nathanmittler): Do we need to make lists immutable?
|
||||
} catch (InvalidProtocolBufferException.InvalidWireTypeException e) {
|
||||
// Treat fields with an invalid wire type as unknown fields (i.e. same as the default case).
|
||||
if (!reader.skipField()) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
38
java/core/src/main/java/com/google/protobuf/ProtoSyntax.java
Normal file
38
java/core/src/main/java/com/google/protobuf/ProtoSyntax.java
Normal file
|
@ -0,0 +1,38 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
/** Represents the syntax version of the message. */
|
||||
@ExperimentalApi
|
||||
public enum ProtoSyntax {
|
||||
PROTO2,
|
||||
PROTO3;
|
||||
}
|
148
java/core/src/main/java/com/google/protobuf/Protobuf.java
Normal file
148
java/core/src/main/java/com/google/protobuf/Protobuf.java
Normal file
|
@ -0,0 +1,148 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import static com.google.protobuf.Internal.checkNotNull;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.concurrent.ConcurrentHashMap;
|
||||
import java.util.concurrent.ConcurrentMap;
|
||||
|
||||
/**
|
||||
* Main runtime interface for protobuf. Applications should interact with this interface (rather
|
||||
* than directly accessing internal APIs) in order to perform operations on protobuf messages.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public final class Protobuf {
|
||||
private static final Protobuf INSTANCE = new Protobuf();
|
||||
|
||||
private final SchemaFactory schemaFactory;
|
||||
|
||||
// TODO(nathanmittler): Consider using ClassValue instead.
|
||||
private final ConcurrentMap<Class<?>, Schema<?>> schemaCache =
|
||||
new ConcurrentHashMap<Class<?>, Schema<?>>();
|
||||
|
||||
/** Gets the singleton instance of the Protobuf runtime. */
|
||||
public static Protobuf getInstance() {
|
||||
return INSTANCE;
|
||||
}
|
||||
|
||||
/** Writes the given message to the target {@link Writer}. */
|
||||
public <T> void writeTo(T message, Writer writer) {
|
||||
schemaFor(message).writeTo(message, writer);
|
||||
}
|
||||
|
||||
/** Reads fields from the given {@link Reader} and merges them into the message. */
|
||||
public <T> void mergeFrom(T message, Reader reader) throws IOException {
|
||||
schemaFor(message).mergeFrom(message, reader);
|
||||
}
|
||||
|
||||
/** Gets the schema for the given message type. */
|
||||
public <T> Schema<T> schemaFor(Class<T> messageType) {
|
||||
checkNotNull(messageType, "messageType");
|
||||
@SuppressWarnings("unchecked")
|
||||
Schema<T> schema = (Schema<T>) schemaCache.get(messageType);
|
||||
if (schema == null) {
|
||||
schema = schemaFactory.createSchema(messageType);
|
||||
@SuppressWarnings("unchecked")
|
||||
Schema<T> previous = (Schema<T>) registerSchema(messageType, schema);
|
||||
if (previous != null) {
|
||||
// A new schema was registered by another thread.
|
||||
schema = previous;
|
||||
}
|
||||
}
|
||||
return schema;
|
||||
}
|
||||
|
||||
/** Gets the schema for the given message. */
|
||||
@SuppressWarnings("unchecked")
|
||||
public <T> Schema<T> schemaFor(T message) {
|
||||
return schemaFor((Class<T>) message.getClass());
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers the given schema for the message type only if a schema was not already registered.
|
||||
*
|
||||
* @param messageType the type of message on which the schema operates.
|
||||
* @param schema the schema for the message type.
|
||||
* @return the previously registered schema, or {@code null} if the given schema was successfully
|
||||
* registered.
|
||||
*/
|
||||
public Schema<?> registerSchema(Class<?> messageType, Schema<?> schema) {
|
||||
checkNotNull(messageType, "messageType");
|
||||
checkNotNull(schema, "schema");
|
||||
return schemaCache.putIfAbsent(messageType, schema);
|
||||
}
|
||||
|
||||
/**
|
||||
* Visible for testing only. Registers the given schema for the message type. If a schema was
|
||||
* previously registered, it will be replaced by the provided schema.
|
||||
*
|
||||
* @param messageType the type of message on which the schema operates.
|
||||
* @param schema the schema for the message type.
|
||||
* @return the previously registered schema, or {@code null} if no schema was registered
|
||||
* previously.
|
||||
*/
|
||||
public Schema<?> registerSchemaOverride(Class<?> messageType, Schema<?> schema) {
|
||||
checkNotNull(messageType, "messageType");
|
||||
checkNotNull(schema, "schema");
|
||||
return schemaCache.put(messageType, schema);
|
||||
}
|
||||
|
||||
private Protobuf() {
|
||||
// TODO(nathanmittler): Detect the proper factory for the platform.
|
||||
SchemaFactory factory = null;
|
||||
for (String className :
|
||||
new String[] {
|
||||
"com.google.frameworks.protobuf.experimental.android.schema.AndroidProto3SchemaFactory",
|
||||
// TODO(nathanmittler): Remove this once the code has been completely ported over.
|
||||
"com.google.frameworks.protobuf.experimental.schema.generic.Proto3SchemaFactory",
|
||||
"com.google.protobuf.Proto3SchemaFactory"
|
||||
}) {
|
||||
factory = newSchemaFactory(className);
|
||||
if (factory != null) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (factory == null) {
|
||||
throw new IllegalStateException("Unable to locate a default SchemaFactory. Check classpath.");
|
||||
}
|
||||
schemaFactory = factory;
|
||||
}
|
||||
|
||||
private static SchemaFactory newSchemaFactory(String className) {
|
||||
try {
|
||||
return (SchemaFactory) Class.forName(className).getConstructor().newInstance();
|
||||
} catch (Throwable e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,96 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import com.google.protobuf.Internal.BooleanList;
|
||||
import com.google.protobuf.Internal.DoubleList;
|
||||
import com.google.protobuf.Internal.FloatList;
|
||||
import com.google.protobuf.Internal.IntList;
|
||||
import com.google.protobuf.Internal.LongList;
|
||||
import com.google.protobuf.Internal.ProtobufList;
|
||||
|
||||
/**
|
||||
* Utility class for construction of lists that extend {@link ProtobufList}.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public final class ProtobufLists {
|
||||
private ProtobufLists() {}
|
||||
|
||||
public static <E> ProtobufList<E> emptyProtobufList() {
|
||||
return ProtobufArrayList.emptyList();
|
||||
}
|
||||
|
||||
public static <E> ProtobufList<E> mutableCopy(ProtobufList<E> list) {
|
||||
int size = list.size();
|
||||
return list.mutableCopyWithCapacity(
|
||||
size == 0 ? AbstractProtobufList.DEFAULT_CAPACITY : size * 2);
|
||||
}
|
||||
|
||||
public static BooleanList emptyBooleanList() {
|
||||
return BooleanArrayList.emptyList();
|
||||
}
|
||||
|
||||
public static BooleanList newBooleanList() {
|
||||
return new BooleanArrayList();
|
||||
}
|
||||
|
||||
public static IntList emptyIntList() {
|
||||
return IntArrayList.emptyList();
|
||||
}
|
||||
|
||||
public static IntList newIntList() {
|
||||
return new IntArrayList();
|
||||
}
|
||||
|
||||
public static LongList emptyLongList() {
|
||||
return LongArrayList.emptyList();
|
||||
}
|
||||
|
||||
public static LongList newLongList() {
|
||||
return new LongArrayList();
|
||||
}
|
||||
|
||||
public static FloatList emptyFloatList() {
|
||||
return FloatArrayList.emptyList();
|
||||
}
|
||||
|
||||
public static FloatList newFloatList() {
|
||||
return new FloatArrayList();
|
||||
}
|
||||
|
||||
public static DoubleList emptyDoubleList() {
|
||||
return DoubleArrayList.emptyList();
|
||||
}
|
||||
|
||||
public static DoubleList newDoubleList() {
|
||||
return new DoubleArrayList();
|
||||
}
|
||||
}
|
312
java/core/src/main/java/com/google/protobuf/Reader.java
Normal file
312
java/core/src/main/java/com/google/protobuf/Reader.java
Normal file
|
@ -0,0 +1,312 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.List;
|
||||
|
||||
/** A reader of fields from a serialized protobuf message. */
|
||||
// TODO(nathanmittler): Refactor to allow the reader to allocate properly sized lists.
|
||||
@ExperimentalApi
|
||||
public interface Reader {
|
||||
/** Value used to indicate that the end of input has been reached. */
|
||||
int READ_DONE = Integer.MAX_VALUE;
|
||||
|
||||
/**
|
||||
* Gets the field number for the current field being read.
|
||||
*
|
||||
* @return the current field number or {@link #READ_DONE} if the end of input has been reached.
|
||||
*/
|
||||
int getFieldNumber() throws IOException;
|
||||
|
||||
/**
|
||||
* Skips the current field and advances the reader to the next field.
|
||||
*
|
||||
* @return {@code true} if there are more fields or {@link #READ_DONE} if the end of input has
|
||||
* been reached.
|
||||
*/
|
||||
boolean skipField() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code DOUBLE} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
double readDouble() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code FLOAT} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
float readFloat() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code UINT64} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
long readUInt64() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code INT64} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
long readInt64() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code INT32} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
int readInt32() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code FIXED64} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
long readFixed64() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code FIXED32} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
int readFixed32() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code BOOL} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
boolean readBool() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code STRING} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
String readString() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code MESSAGE} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
<T> T readMessage(Class<T> clazz) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code GROUP} and advances the reader to the next
|
||||
* field.
|
||||
*
|
||||
* @deprecated groups fields are deprecated.
|
||||
*/
|
||||
@Deprecated
|
||||
<T> T readGroup(Class<T> clazz) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code BYTES} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
ByteString readBytes() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code UINT32} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
int readUInt32() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code ENUM} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
int readEnum() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code SFIXED32} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
int readSFixed32() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code SFIXED64} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
long readSFixed64() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code SINT32} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
int readSInt32() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads and returns the next field of type {@code SINT64} and advances the reader to the next
|
||||
* field.
|
||||
*/
|
||||
long readSInt64() throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code DOUBLE_LIST} or {@code DOUBLE_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readDoubleList(List<Double> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code FLOAT_LIST} or {@code FLOAT_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readFloatList(List<Float> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code UINT64_LIST} or {@code UINT64_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readUInt64List(List<Long> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code INT64_LIST} or {@code INT64_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readInt64List(List<Long> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code INT32_LIST} or {@code INT32_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readInt32List(List<Integer> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code FIXED64_LIST} or {@code FIXED64_LIST_PACKED} and advances
|
||||
* the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readFixed64List(List<Long> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code FIXED32_LIST} or {@code FIXED32_LIST_PACKED} and advances
|
||||
* the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readFixed32List(List<Integer> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code BOOL_LIST} or {@code BOOL_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readBoolList(List<Boolean> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code STRING_LIST} and advances the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readStringList(List<String> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code MESSAGE_LIST} and advances the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
* @param targetType the type of the elements stored in the {@code target} list.
|
||||
*/
|
||||
<T> void readMessageList(List<T> target, Class<T> targetType) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code GROUP_LIST} and advances the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
* @param targetType the type of the elements stored in the {@code target} list.
|
||||
* @deprecated groups fields are deprecated.
|
||||
*/
|
||||
@Deprecated
|
||||
<T> void readGroupList(List<T> target, Class<T> targetType) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code BYTES_LIST} and advances the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readBytesList(List<ByteString> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code UINT32_LIST} or {@code UINT32_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readUInt32List(List<Integer> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code ENUM_LIST} or {@code ENUM_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readEnumList(List<Integer> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code SFIXED32_LIST} or {@code SFIXED32_LIST_PACKED} and advances
|
||||
* the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readSFixed32List(List<Integer> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code SFIXED64_LIST} or {@code SFIXED64_LIST_PACKED} and advances
|
||||
* the reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readSFixed64List(List<Long> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code SINT32_LIST} or {@code SINT32_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readSInt32List(List<Integer> target) throws IOException;
|
||||
|
||||
/**
|
||||
* Reads the next field of type {@code SINT64_LIST} or {@code SINT64_LIST_PACKED} and advances the
|
||||
* reader to the next field.
|
||||
*
|
||||
* @param target the list that will receive the read values.
|
||||
*/
|
||||
void readSInt64List(List<Long> target) throws IOException;
|
||||
}
|
|
@ -406,6 +406,11 @@ final class RopeByteString extends ByteString {
|
|||
right.writeTo(output);
|
||||
}
|
||||
|
||||
@Override
|
||||
void writeToReverse(ByteOutput output) throws IOException {
|
||||
right.writeToReverse(output);
|
||||
left.writeToReverse(output);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected String toStringInternal(Charset charset) {
|
||||
|
|
55
java/core/src/main/java/com/google/protobuf/Schema.java
Normal file
55
java/core/src/main/java/com/google/protobuf/Schema.java
Normal file
|
@ -0,0 +1,55 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.io.IOException;
|
||||
|
||||
/**
|
||||
* A runtime schema for a single protobuf message. A schema provides operations on message
|
||||
* instances such as serialization/deserialization.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public interface Schema<T> {
|
||||
/**
|
||||
* Writes the given message to the target {@link Writer}.
|
||||
*/
|
||||
void writeTo(T message, Writer writer);
|
||||
|
||||
/**
|
||||
* Reads fields from the given {@link Reader} and merges them into the message.
|
||||
*/
|
||||
void mergeFrom(T message, Reader reader) throws IOException;
|
||||
|
||||
/**
|
||||
* Creates a new instance of the message class.
|
||||
*/
|
||||
T newInstance();
|
||||
}
|
|
@ -0,0 +1,42 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
/**
|
||||
* A factory that manufactures {@link Schema} instances for protobuf messages.
|
||||
*/
|
||||
@ExperimentalApi
|
||||
public interface SchemaFactory {
|
||||
/**
|
||||
* Creates a schema instance for the given protobuf message type.
|
||||
*/
|
||||
<T> Schema<T> createSchema(Class<T> messageType);
|
||||
}
|
516
java/core/src/main/java/com/google/protobuf/SchemaUtil.java
Normal file
516
java/core/src/main/java/com/google/protobuf/SchemaUtil.java
Normal file
|
@ -0,0 +1,516 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import com.google.protobuf.Internal.ProtobufList;
|
||||
import java.io.IOException;
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.Method;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
|
||||
/** Helper methods used by schemas. */
|
||||
@ExperimentalApi
|
||||
public final class SchemaUtil {
|
||||
private static final Class<?> UNMODIFIABLE_LIST_CLASS =
|
||||
Collections.unmodifiableList(new ArrayList<Integer>()).getClass();
|
||||
private static final Class<?> ABSTRACT_MESSAGE_CLASS = getAbstractMessageClass();
|
||||
private static final Class<?> GENERATED_MESSAGE_CLASS = getGeneratedMessageClass();
|
||||
private static final Class<?> UNKNOWN_FIELD_SET_CLASS = getUnknownFieldSetClass();
|
||||
private static final long UNKNOWN_FIELD_OFFSET = unknownFieldOffset();
|
||||
private static final long LITE_UNKNOWN_FIELD_OFFSET = unknownFieldLiteOffset();
|
||||
private static final Object DEFAULT_UNKNOWN_FIELD_SET = defaultUnknownFieldSet();
|
||||
private static final long MEMOIZED_SIZE_FIELD_OFFSET = getMemoizedSizeFieldOffset();
|
||||
private static final long LITE_MEMOIZED_SIZE_FIELD_OFFSET = getLiteMemoizedSizeFieldOffset();
|
||||
|
||||
private SchemaUtil() {}
|
||||
|
||||
/**
|
||||
* Initializes all of the base class fields for the given {@link GeneratedMessageLite} to their
|
||||
* default values.
|
||||
*/
|
||||
public static void initLiteBaseClassFields(Object msg) {
|
||||
UnsafeUtil.putObject(msg, LITE_UNKNOWN_FIELD_OFFSET, UnknownFieldSetLite.getDefaultInstance());
|
||||
UnsafeUtil.putInt(msg, LITE_MEMOIZED_SIZE_FIELD_OFFSET, -1);
|
||||
}
|
||||
|
||||
/**
|
||||
* Initializes all of the base class fields for the given {@link
|
||||
* com.google.protobuf.GeneratedMessage} to their default values.
|
||||
*/
|
||||
public static void initBaseClassFields(Object msg) {
|
||||
UnsafeUtil.putObject(msg, UNKNOWN_FIELD_OFFSET, DEFAULT_UNKNOWN_FIELD_SET);
|
||||
UnsafeUtil.putInt(msg, MEMOIZED_SIZE_FIELD_OFFSET, -1);
|
||||
}
|
||||
|
||||
/**
|
||||
* Requires that the given message extend {@link com.google.protobuf.GeneratedMessage} or {@link
|
||||
* GeneratedMessageLite}.
|
||||
*/
|
||||
public static void requireGeneratedMessage(Class<?> messageType) {
|
||||
if (!GeneratedMessageLite.class.isAssignableFrom(messageType)
|
||||
&& !GENERATED_MESSAGE_CLASS.isAssignableFrom(messageType)) {
|
||||
throw new IllegalArgumentException(
|
||||
"Message classes must extend GeneratedMessage or GeneratedMessageLite");
|
||||
}
|
||||
}
|
||||
|
||||
/** Initializes the message's memoized size to the default value. */
|
||||
public static void initMemoizedSize(Object msg) {
|
||||
final long fieldOffset =
|
||||
(msg instanceof GeneratedMessageLite)
|
||||
? LITE_MEMOIZED_SIZE_FIELD_OFFSET
|
||||
: MEMOIZED_SIZE_FIELD_OFFSET;
|
||||
|
||||
if (fieldOffset < 0) {
|
||||
throw new IllegalArgumentException(
|
||||
"Unable to identify memoizedSize field offset for message of type: "
|
||||
+ msg.getClass().getName());
|
||||
}
|
||||
|
||||
UnsafeUtil.putInt(msg, fieldOffset, -1);
|
||||
}
|
||||
|
||||
public static void writeDouble(int fieldNumber, double value, Writer writer) {
|
||||
if (Double.compare(value, 0.0) != 0) {
|
||||
writer.writeDouble(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeFloat(int fieldNumber, float value, Writer writer) {
|
||||
if (Float.compare(value, 0.0f) != 0) {
|
||||
writer.writeFloat(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeInt64(int fieldNumber, long value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeInt64(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeUInt64(int fieldNumber, long value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeUInt64(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSInt64(int fieldNumber, long value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeSInt64(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeFixed64(int fieldNumber, long value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeFixed64(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSFixed64(int fieldNumber, long value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeSFixed64(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeInt32(int fieldNumber, int value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeInt32(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeUInt32(int fieldNumber, int value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeUInt32(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSInt32(int fieldNumber, int value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeSInt32(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeFixed32(int fieldNumber, int value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeFixed32(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSFixed32(int fieldNumber, int value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeSFixed32(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeEnum(int fieldNumber, int value, Writer writer) {
|
||||
if (value != 0) {
|
||||
writer.writeEnum(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeBool(int fieldNumber, boolean value, Writer writer) {
|
||||
if (value) {
|
||||
writer.writeBool(fieldNumber, true);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeString(int fieldNumber, Object value, Writer writer) {
|
||||
if (value instanceof String) {
|
||||
writeStringInternal(fieldNumber, (String) value, writer);
|
||||
} else {
|
||||
writeBytes(fieldNumber, (ByteString) value, writer);
|
||||
}
|
||||
}
|
||||
|
||||
private static void writeStringInternal(int fieldNumber, String value, Writer writer) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeString(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeBytes(int fieldNumber, ByteString value, Writer writer) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeBytes(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeMessage(int fieldNumber, Object value, Writer writer) {
|
||||
if (value != null) {
|
||||
writer.writeMessage(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeDoubleList(
|
||||
int fieldNumber, List<Double> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeDoubleList(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeFloatList(
|
||||
int fieldNumber, List<Float> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeFloatList(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeInt64List(
|
||||
int fieldNumber, List<Long> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeInt64List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeUInt64List(
|
||||
int fieldNumber, List<Long> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeUInt64List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSInt64List(
|
||||
int fieldNumber, List<Long> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeSInt64List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeFixed64List(
|
||||
int fieldNumber, List<Long> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeFixed64List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSFixed64List(
|
||||
int fieldNumber, List<Long> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeSFixed64List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeInt32List(
|
||||
int fieldNumber, List<Integer> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeInt32List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeUInt32List(
|
||||
int fieldNumber, List<Integer> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeUInt32List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSInt32List(
|
||||
int fieldNumber, List<Integer> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeSInt32List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeFixed32List(
|
||||
int fieldNumber, List<Integer> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeFixed32List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeSFixed32List(
|
||||
int fieldNumber, List<Integer> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeSFixed32List(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeEnumList(
|
||||
int fieldNumber, List<Integer> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeEnumList(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeBoolList(
|
||||
int fieldNumber, List<Boolean> value, Writer writer, boolean packed) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeBoolList(fieldNumber, value, packed);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeStringList(int fieldNumber, List<String> value, Writer writer) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeStringList(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeBytesList(int fieldNumber, List<ByteString> value, Writer writer) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeBytesList(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeMessageList(int fieldNumber, List<?> value, Writer writer) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeMessageList(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
public static void writeGroupList(int fieldNumber, List<?> value, Writer writer) {
|
||||
if (value != null && !value.isEmpty()) {
|
||||
writer.writeGroupList(fieldNumber, value);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Used for protobuf-lite (i.e. messages that extend {@link GeneratedMessageLite}). Reads in a
|
||||
* list of messages into a target {@link ProtobufList} field in the message.
|
||||
*/
|
||||
public static final <T> void readProtobufMessageList(
|
||||
Object message, long offset, Reader reader, Class<T> targetType) throws IOException {
|
||||
reader.readMessageList(SchemaUtil.<T>mutableProtobufListAt(message, offset), targetType);
|
||||
}
|
||||
|
||||
/**
|
||||
* Used for standard protobuf messages (i.e. messages that extend {@link
|
||||
* com.google.protobuf.GeneratedMessage}). Reads in a list of messages into a target list field in
|
||||
* the message.
|
||||
*/
|
||||
public static <T> void readMessageList(
|
||||
Object message, long offset, Reader reader, Class<T> targetType) throws IOException {
|
||||
reader.readMessageList(SchemaUtil.<T>mutableListAt(message, offset), targetType);
|
||||
}
|
||||
|
||||
/**
|
||||
* Used for group types. Reads in a list of group messages into a target list field in the
|
||||
* message.
|
||||
*/
|
||||
public static <T> void readGroupList(
|
||||
Object message, long offset, Reader reader, Class<T> targetType) throws IOException {
|
||||
reader.readGroupList(SchemaUtil.<T>mutableListAt(message, offset), targetType);
|
||||
}
|
||||
|
||||
/**
|
||||
* Used for protobuf-lite (i.e. messages that extend GeneratedMessageLite). Converts the {@link
|
||||
* ProtobufList} at the given field position in the message into a mutable list (if it isn't
|
||||
* already).
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
public static <L> ProtobufList<L> mutableProtobufListAt(Object message, long pos) {
|
||||
ProtobufList<L> list = (ProtobufList<L>) UnsafeUtil.getObject(message, pos);
|
||||
if (!list.isModifiable()) {
|
||||
int size = list.size();
|
||||
list = list.mutableCopyWithCapacity(
|
||||
size == 0 ? AbstractProtobufList.DEFAULT_CAPACITY : size * 2);
|
||||
UnsafeUtil.putObject(message, pos, list);
|
||||
}
|
||||
return list;
|
||||
}
|
||||
|
||||
/** Converts the list at the given field location in the message into a mutable list. */
|
||||
@SuppressWarnings("unchecked")
|
||||
public static <L> List<L> mutableListAt(Object message, long pos) {
|
||||
List<L> list = (List<L>) UnsafeUtil.getObject(message, pos);
|
||||
if (list.isEmpty()) {
|
||||
list =
|
||||
(list instanceof LazyStringList)
|
||||
? (List<L>) new LazyStringArrayList()
|
||||
: new ArrayList<L>();
|
||||
UnsafeUtil.putObject(message, pos, list);
|
||||
} else if (UNMODIFIABLE_LIST_CLASS.isAssignableFrom(list.getClass())) {
|
||||
list = new ArrayList<L>(list);
|
||||
UnsafeUtil.putObject(message, pos, list);
|
||||
} else if (list instanceof UnmodifiableLazyStringList) {
|
||||
// Convert the list to a mutable list.
|
||||
list = (List<L>) new LazyStringArrayList((List<String>) list);
|
||||
UnsafeUtil.putObject(message, pos, list);
|
||||
}
|
||||
return list;
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines whether to issue tableswitch or lookupswitch for the mergeFrom method.
|
||||
*
|
||||
* @see #shouldUseTableSwitch(int, int, int)
|
||||
*/
|
||||
public static boolean shouldUseTableSwitch(List<FieldInfo> fields) {
|
||||
// Determine whether to issue a tableswitch or a lookupswitch
|
||||
// instruction.
|
||||
if (fields.isEmpty()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
int lo = fields.get(0).getFieldNumber();
|
||||
int hi = fields.get(fields.size() - 1).getFieldNumber();
|
||||
return shouldUseTableSwitch(lo, hi, fields.size());
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines whether to issue tableswitch or lookupswitch for the mergeFrom method. This is based
|
||||
* on the <a href=
|
||||
* "http://hg.openjdk.java.net/jdk8/jdk8/langtools/file/30db5e0aaf83/src/share/classes/com/sun/tools/javac/jvm/Gen.java#l1159">
|
||||
* logic in the JDK</a>.
|
||||
*
|
||||
* @param lo the lowest fieldNumber contained within the message.
|
||||
* @param hi the higest fieldNumber contained within the message.
|
||||
* @param numFields the total number of fields in the message.
|
||||
* @return {@code true} if tableswitch should be used, rather than lookupswitch.
|
||||
*/
|
||||
public static boolean shouldUseTableSwitch(int lo, int hi, int numFields) {
|
||||
long tableSpaceCost = 4 + ((long) hi - lo + 1); // words
|
||||
long tableTimeCost = 3; // comparisons
|
||||
long lookupSpaceCost = 3 + 2 * (long) numFields;
|
||||
long lookupTimeCost = numFields;
|
||||
return tableSpaceCost + 3 * tableTimeCost <= lookupSpaceCost + 3 * lookupTimeCost;
|
||||
}
|
||||
|
||||
private static Class<?> getAbstractMessageClass() {
|
||||
try {
|
||||
return Class.forName("com.google.protobuf.AbstractMessage");
|
||||
} catch (Throwable e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
private static Class<?> getGeneratedMessageClass() {
|
||||
try {
|
||||
return Class.forName("com.google.protobuf.GeneratedMessage");
|
||||
} catch (Throwable e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
private static Class<?> getUnknownFieldSetClass() {
|
||||
try {
|
||||
return Class.forName("com.google.protobuf.UnknownFieldSet");
|
||||
} catch (Throwable e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
private static long unknownFieldOffset() {
|
||||
try {
|
||||
if (GENERATED_MESSAGE_CLASS != null) {
|
||||
Field field = GENERATED_MESSAGE_CLASS.getDeclaredField("unknownFields");
|
||||
return UnsafeUtil.objectFieldOffset(field);
|
||||
}
|
||||
} catch (Throwable e) {
|
||||
// Do nothing.
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
private static long unknownFieldLiteOffset() {
|
||||
try {
|
||||
Field field = GeneratedMessageLite.class.getDeclaredField("unknownFields");
|
||||
return UnsafeUtil.objectFieldOffset(field);
|
||||
} catch (Throwable e) {
|
||||
// Do nothing.
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
private static Object defaultUnknownFieldSet() {
|
||||
try {
|
||||
if (UNKNOWN_FIELD_SET_CLASS != null) {
|
||||
Method method = UNKNOWN_FIELD_SET_CLASS.getDeclaredMethod("getDefaultInstance");
|
||||
return method.invoke(null);
|
||||
}
|
||||
} catch (Throwable e) {
|
||||
// Do nothing.
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
private static long getMemoizedSizeFieldOffset() {
|
||||
try {
|
||||
if (ABSTRACT_MESSAGE_CLASS != null) {
|
||||
Field field = ABSTRACT_MESSAGE_CLASS.getDeclaredField("memoizedSize");
|
||||
return UnsafeUtil.objectFieldOffset(field);
|
||||
}
|
||||
} catch (Throwable e) {
|
||||
// Do nothing.
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
private static long getLiteMemoizedSizeFieldOffset() {
|
||||
try {
|
||||
Field field = GeneratedMessageLite.class.getDeclaredField("memoizedSerializedSize");
|
||||
return UnsafeUtil.objectFieldOffset(field);
|
||||
} catch (Throwable e) {
|
||||
// Do nothing.
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
}
|
155
java/core/src/main/java/com/google/protobuf/Writer.java
Normal file
155
java/core/src/main/java/com/google/protobuf/Writer.java
Normal file
|
@ -0,0 +1,155 @@
|
|||
// Protocol Buffers - Google's data interchange format
|
||||
// Copyright 2008 Google Inc. All rights reserved.
|
||||
// https://developers.google.com/protocol-buffers/
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package com.google.protobuf;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/** A writer that performs serialization of protobuf message fields. */
|
||||
@ExperimentalApi
|
||||
public interface Writer {
|
||||
/** Writes a field of type {@link FieldType#SFIXED32}. */
|
||||
void writeSFixed32(int fieldNumber, int value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#INT64}. */
|
||||
void writeInt64(int fieldNumber, long value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#SFIXED64}. */
|
||||
void writeSFixed64(int fieldNumber, long value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#FLOAT}. */
|
||||
void writeFloat(int fieldNumber, float value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#DOUBLE}. */
|
||||
void writeDouble(int fieldNumber, double value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#ENUM}. */
|
||||
void writeEnum(int fieldNumber, int value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#UINT64}. */
|
||||
void writeUInt64(int fieldNumber, long value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#INT32}. */
|
||||
void writeInt32(int fieldNumber, int value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#FIXED64}. */
|
||||
void writeFixed64(int fieldNumber, long value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#FIXED32}. */
|
||||
void writeFixed32(int fieldNumber, int value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#BOOL}. */
|
||||
void writeBool(int fieldNumber, boolean value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#STRING}. */
|
||||
void writeString(int fieldNumber, String value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#BYTES}. */
|
||||
void writeBytes(int fieldNumber, ByteString value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#UINT32}. */
|
||||
void writeUInt32(int fieldNumber, int value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#SINT32}. */
|
||||
void writeSInt32(int fieldNumber, int value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#SINT64}. */
|
||||
void writeSInt64(int fieldNumber, long value);
|
||||
|
||||
/** Writes a field of type {@link FieldType#MESSAGE}. */
|
||||
void writeMessage(int fieldNumber, Object value);
|
||||
|
||||
/**
|
||||
* Writes a field of type {@link FieldType#GROUP}.
|
||||
*
|
||||
* @deprecated groups fields are deprecated.
|
||||
*/
|
||||
@Deprecated
|
||||
void writeGroup(int fieldNumber, Object value);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#INT32}. */
|
||||
void writeInt32List(int fieldNumber, List<Integer> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#FIXED32}. */
|
||||
void writeFixed32List(int fieldNumber, List<Integer> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#INT64}. */
|
||||
void writeInt64List(int fieldNumber, List<Long> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#UINT64}. */
|
||||
void writeUInt64List(int fieldNumber, List<Long> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#FIXED64}. */
|
||||
void writeFixed64List(int fieldNumber, List<Long> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#FLOAT}. */
|
||||
void writeFloatList(int fieldNumber, List<Float> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#DOUBLE}. */
|
||||
void writeDoubleList(int fieldNumber, List<Double> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#ENUM}. */
|
||||
void writeEnumList(int fieldNumber, List<Integer> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#BOOL}. */
|
||||
void writeBoolList(int fieldNumber, List<Boolean> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#STRING}. */
|
||||
void writeStringList(int fieldNumber, List<String> value);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#BYTES}. */
|
||||
void writeBytesList(int fieldNumber, List<ByteString> value);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#UINT32}. */
|
||||
void writeUInt32List(int fieldNumber, List<Integer> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#SFIXED32}. */
|
||||
void writeSFixed32List(int fieldNumber, List<Integer> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#SFIXED64}. */
|
||||
void writeSFixed64List(int fieldNumber, List<Long> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#SINT32}. */
|
||||
void writeSInt32List(int fieldNumber, List<Integer> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#SINT64}. */
|
||||
void writeSInt64List(int fieldNumber, List<Long> value, boolean packed);
|
||||
|
||||
/** Writes a list field of type {@link FieldType#MESSAGE}. */
|
||||
void writeMessageList(int fieldNumber, List<?> value);
|
||||
|
||||
/**
|
||||
* Writes a list field of type {@link FieldType#GROUP}.
|
||||
*
|
||||
* @deprecated groups fields are deprecated.
|
||||
*/
|
||||
@Deprecated
|
||||
void writeGroupList(int fieldNumber, List<?> value);
|
||||
}
|
Loading…
Add table
Reference in a new issue