Compare commits

...
Sign in to create a new pull request.

1 commit

Author SHA1 Message Date
Feng Xiao
92f30880c5 Add a Java JMH micro-benchmark for proto3 messages. 2017-01-12 14:52:14 -08:00
3 changed files with 335 additions and 0 deletions

147
java/benchmark/pom.xml Normal file
View file

@ -0,0 +1,147 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.google.protobuf</groupId>
<artifactId>benchmark</artifactId>
<version>0.1</version>
<packaging>jar</packaging>
<name>Protocol Buffers [Benchmark]</name>
<dependencies>
<dependency>
<groupId>org.openjdk.jmh</groupId>
<artifactId>jmh-core</artifactId>
<version>1.17.3</version>
</dependency>
<dependency>
<groupId>org.openjdk.jmh</groupId>
<artifactId>jmh-generator-annprocess</artifactId>
<version>1.17.3</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>com.google.protobuf</groupId>
<artifactId>protobuf-java</artifactId>
<version>3.1.0</version>
</dependency>
</dependencies>
<properties>
<uberjar.name>benchmarks</uberjar.name>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<protobuf.basedir>${project.basedir}/../..</protobuf.basedir>
<protobuf.source.dir>${protobuf.basedir}/src</protobuf.source.dir>
<protoc>${protobuf.source.dir}/protoc</protoc>
<generated.sources.dir>${project.build.directory}/generated-sources</generated.sources.dir>
</properties>
<build>
<plugins>
<plugin>
<artifactId>maven-antrun-plugin</artifactId>
<executions>
<execution>
<id>generate-sources</id>
<phase>generate-sources</phase>
<configuration>
<tasks>
<mkdir dir="${generated.sources.dir}" />
<exec executable="${protoc}">
<arg value="--java_out=${generated.sources.dir}" />
<arg value="--proto_path=${protobuf.source.dir}" />
<arg value="${protobuf.source.dir}/google/protobuf/unittest_proto3.proto"/>
<arg value="${protobuf.source.dir}/google/protobuf/unittest_import_proto3.proto"/>
<arg value="${protobuf.source.dir}/google/protobuf/unittest_import_public_proto3.proto"/>
</exec>
</tasks>
</configuration>
<goals>
<goal>run</goal>
</goals>
</execution>
<!-- Generate the test protos -->
<execution>
<id>generate-test-sources</id>
<phase>generate-test-sources</phase>
<configuration>
<target>
<ant antfile="generate-test-sources-build.xml"/>
</target>
</configuration>
<goals>
<goal>run</goal>
</goals>
</execution>
</executions>
</plugin>
<!-- Add generated sources to the build -->
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>build-helper-maven-plugin</artifactId>
<executions>
<execution>
<id>add-generated-sources</id>
<phase>generate-sources</phase>
<goals>
<goal>add-source</goal>
</goals>
<configuration>
<sources>
<source>${generated.sources.dir}</source>
</sources>
</configuration>
</execution>
</executions>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.1</version>
<configuration>
<compilerVersion>1.6</compilerVersion>
<source>1.6</source>
<target>1.6</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<version>2.2</version>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<finalName>${uberjar.name}</finalName>
<transformers>
<transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
<mainClass>org.openjdk.jmh.Main</mainClass>
</transformer>
</transformers>
<filters>
<filter>
<artifact>*:*</artifact>
<excludes>
<exclude>META-INF/*.SF</exclude>
<exclude>META-INF/*.DSA</exclude>
<exclude>META-INF/*.RSA</exclude>
</excludes>
</filter>
</filters>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
</project>

View file

@ -0,0 +1,186 @@
package com.google.protobuf;
import com.google.protobuf.test.UnittestImportProto3.ImportEnum;
import com.google.protobuf.test.UnittestImportProto3.ImportMessage;
import com.google.protobuf.test.UnittestImportPublicProto3.PublicImportMessage;
import java.io.ByteArrayOutputStream;
import org.openjdk.jmh.annotations.Benchmark;
import org.openjdk.jmh.annotations.Level;
import org.openjdk.jmh.annotations.Setup;
import org.openjdk.jmh.annotations.Scope;
import org.openjdk.jmh.annotations.State;
import protobuf_unittest.UnittestProto.ForeignEnum;
import protobuf_unittest.UnittestProto.ForeignMessage;
import protobuf_unittest.UnittestProto.TestAllTypes;
@State(Scope.Benchmark)
public class BenchmarkSuite {
private static TestAllTypes proto3TestAllTypes =
TestDataGenerator.getProto3TestAllTypes();
private static TestAllTypes largeTestAllTypes =
TestDataGenerator.getLargeTestAllTypes();
static {
System.out.println("TestAllTypes: " + proto3TestAllTypes.getSerializedSize());
System.out.println("large TestAllTypes: " + largeTestAllTypes.getSerializedSize());
}
private static long writeByteArray(Message prototype) throws Exception {
return prototype.toByteArray().length;
}
private static long writeByteString(Message prototype) throws Exception {
return prototype.toByteString().size();
}
private static long writeByteArrayStream(Message prototype) throws Exception {
ByteArrayOutputStream output = new ByteArrayOutputStream(
Math.min(prototype.getSerializedSize(),
CodedOutputStream.DEFAULT_BUFFER_SIZE));
prototype.writeTo(output);
return output.size();
}
@Benchmark
public long proto3TestAllTypesWriteByteString() throws Exception {
return writeByteString(proto3TestAllTypes);
}
@Benchmark
public long proto3TestAllTypesWriteByteArray() throws Exception {
return writeByteArray(proto3TestAllTypes);
}
@Benchmark
public long proto3TestAllTypesWriteByteArrayStream() throws Exception {
return writeByteArrayStream(proto3TestAllTypes);
}
@Benchmark
public long largeTestAllTypesWriteByteString() throws Exception {
return writeByteString(largeTestAllTypes);
}
@Benchmark
public long largeTestAllTypesWriteByteArray() throws Exception {
return writeByteArray(largeTestAllTypes);
}
@Benchmark
public long largeTestAllTypesWriteByteArrayStream() throws Exception {
return writeByteArrayStream(largeTestAllTypes);
}
}
class TestDataGenerator {
static TestAllTypes getProto3TestAllTypes() {
TestAllTypes.Builder builder = TestAllTypes.newBuilder();
setAllFields(builder);
return builder.build();
}
static TestAllTypes getLargeTestAllTypes() {
TestAllTypes.Builder builder = TestAllTypes.newBuilder();
for (int i = 0; i < 100; ++i) {
builder.addChildren(getProto3TestAllTypes());
}
return builder.build();
}
private static ByteString toBytes(String str) {
try {
return ByteString.copyFrom(str.getBytes("UTF-8"));
} catch (Exception e) {
throw new RuntimeException("unexpected exception: " + e.getMessage());
}
}
private static void setAllFields(TestAllTypes.Builder message) {
message.setSingleInt32 (101);
message.setSingleInt64 (102);
message.setSingleUint32 (103);
message.setSingleUint64 (104);
message.setSingleSint32 (105);
message.setSingleSint64 (106);
message.setSingleFixed32 (107);
message.setSingleFixed64 (108);
message.setSingleSfixed32(109);
message.setSingleSfixed64(110);
message.setSingleFloat (111);
message.setSingleDouble (112);
message.setSingleBool (true);
message.setSingleString ("115");
message.setSingleBytes (toBytes("116"));
message.setSingleNestedMessage(
TestAllTypes.NestedMessage.newBuilder().setBb(118).build());
message.setSingleForeignMessage(
ForeignMessage.newBuilder().setC(119).build());
message.setSingleImportMessage(
ImportMessage.newBuilder().setD(120).build());
message.setSinglePublicImportMessage(
PublicImportMessage.newBuilder().setE(126).build());
message.setSingleNestedEnum(TestAllTypes.NestedEnum.BAZ);
message.setSingleForeignEnum(ForeignEnum.FOREIGN_BAZ);
message.setSingleImportEnum(ImportEnum.IMPORT_BAZ);
// -----------------------------------------------------------------
message.addRepeatedInt32 (201);
message.addRepeatedInt64 (202);
message.addRepeatedUint32 (203);
message.addRepeatedUint64 (204);
message.addRepeatedSint32 (205);
message.addRepeatedSint64 (206);
message.addRepeatedFixed32 (207);
message.addRepeatedFixed64 (208);
message.addRepeatedSfixed32(209);
message.addRepeatedSfixed64(210);
message.addRepeatedFloat (211);
message.addRepeatedDouble (212);
message.addRepeatedBool (true);
message.addRepeatedString ("215");
message.addRepeatedBytes (toBytes("216"));
message.addRepeatedNestedMessage(
TestAllTypes.NestedMessage.newBuilder().setBb(218).build());
message.addRepeatedForeignMessage(
ForeignMessage.newBuilder().setC(219).build());
message.addRepeatedImportMessage(
ImportMessage.newBuilder().setD(220).build());
message.addRepeatedNestedEnum(TestAllTypes.NestedEnum.BAR);
message.addRepeatedForeignEnum(ForeignEnum.FOREIGN_BAR);
message.addRepeatedImportEnum(ImportEnum.IMPORT_BAR);
// Add a second one of each field.
message.addRepeatedInt32 (301);
message.addRepeatedInt64 (302);
message.addRepeatedUint32 (303);
message.addRepeatedUint64 (304);
message.addRepeatedSint32 (305);
message.addRepeatedSint64 (306);
message.addRepeatedFixed32 (307);
message.addRepeatedFixed64 (308);
message.addRepeatedSfixed32(309);
message.addRepeatedSfixed64(310);
message.addRepeatedFloat (311);
message.addRepeatedDouble (312);
message.addRepeatedBool (false);
message.addRepeatedString ("315");
message.addRepeatedBytes (toBytes("316"));
message.addRepeatedNestedMessage(
TestAllTypes.NestedMessage.newBuilder().setBb(318).build());
message.addRepeatedForeignMessage(
ForeignMessage.newBuilder().setC(319).build());
message.addRepeatedImportMessage(
ImportMessage.newBuilder().setD(320).build());
message.addRepeatedNestedEnum(TestAllTypes.NestedEnum.BAZ);
message.addRepeatedForeignEnum(ForeignEnum.FOREIGN_BAZ);
message.addRepeatedImportEnum(ImportEnum.IMPORT_BAZ);
}
}

View file

@ -133,6 +133,8 @@ message TestAllTypes {
repeated protobuf_unittest_import.PublicImportMessage
repeated_public_import_message = 54;
repeated TestAllTypes children = 55;
// For oneof test
oneof oneof_field {
uint32 oneof_uint32 = 111;