xref: /aosp_15_r20/external/protobuf/csharp/src/Google.Protobuf.Benchmarks/WriteMessagesBenchmark.cs (revision 1b3f573f81763fcece89efc2b6a5209149e44ab8)
1*1b3f573fSAndroid Build Coastguard Worker #region Copyright notice and license
2*1b3f573fSAndroid Build Coastguard Worker // Protocol Buffers - Google's data interchange format
3*1b3f573fSAndroid Build Coastguard Worker // Copyright 2019 Google Inc.  All rights reserved.
4*1b3f573fSAndroid Build Coastguard Worker // https://github.com/protocolbuffers/protobuf
5*1b3f573fSAndroid Build Coastguard Worker //
6*1b3f573fSAndroid Build Coastguard Worker // Redistribution and use in source and binary forms, with or without
7*1b3f573fSAndroid Build Coastguard Worker // modification, are permitted provided that the following conditions are
8*1b3f573fSAndroid Build Coastguard Worker // met:
9*1b3f573fSAndroid Build Coastguard Worker //
10*1b3f573fSAndroid Build Coastguard Worker //     * Redistributions of source code must retain the above copyright
11*1b3f573fSAndroid Build Coastguard Worker // notice, this list of conditions and the following disclaimer.
12*1b3f573fSAndroid Build Coastguard Worker //     * Redistributions in binary form must reproduce the above
13*1b3f573fSAndroid Build Coastguard Worker // copyright notice, this list of conditions and the following disclaimer
14*1b3f573fSAndroid Build Coastguard Worker // in the documentation and/or other materials provided with the
15*1b3f573fSAndroid Build Coastguard Worker // distribution.
16*1b3f573fSAndroid Build Coastguard Worker //     * Neither the name of Google Inc. nor the names of its
17*1b3f573fSAndroid Build Coastguard Worker // contributors may be used to endorse or promote products derived from
18*1b3f573fSAndroid Build Coastguard Worker // this software without specific prior written permission.
19*1b3f573fSAndroid Build Coastguard Worker //
20*1b3f573fSAndroid Build Coastguard Worker // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21*1b3f573fSAndroid Build Coastguard Worker // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22*1b3f573fSAndroid Build Coastguard Worker // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
23*1b3f573fSAndroid Build Coastguard Worker // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
24*1b3f573fSAndroid Build Coastguard Worker // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
25*1b3f573fSAndroid Build Coastguard Worker // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
26*1b3f573fSAndroid Build Coastguard Worker // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27*1b3f573fSAndroid Build Coastguard Worker // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28*1b3f573fSAndroid Build Coastguard Worker // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29*1b3f573fSAndroid Build Coastguard Worker // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30*1b3f573fSAndroid Build Coastguard Worker // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31*1b3f573fSAndroid Build Coastguard Worker #endregion
32*1b3f573fSAndroid Build Coastguard Worker 
33*1b3f573fSAndroid Build Coastguard Worker using BenchmarkDotNet.Attributes;
34*1b3f573fSAndroid Build Coastguard Worker using System;
35*1b3f573fSAndroid Build Coastguard Worker using System.Collections.Generic;
36*1b3f573fSAndroid Build Coastguard Worker using System.IO;
37*1b3f573fSAndroid Build Coastguard Worker using System.Linq;
38*1b3f573fSAndroid Build Coastguard Worker using System.Buffers;
39*1b3f573fSAndroid Build Coastguard Worker using Google.Protobuf.WellKnownTypes;
40*1b3f573fSAndroid Build Coastguard Worker 
41*1b3f573fSAndroid Build Coastguard Worker namespace Google.Protobuf.Benchmarks
42*1b3f573fSAndroid Build Coastguard Worker {
43*1b3f573fSAndroid Build Coastguard Worker     /// <summary>
44*1b3f573fSAndroid Build Coastguard Worker     /// Benchmark that tests writing performance for various messages.
45*1b3f573fSAndroid Build Coastguard Worker     /// </summary>
46*1b3f573fSAndroid Build Coastguard Worker     [MemoryDiagnoser]
47*1b3f573fSAndroid Build Coastguard Worker     public class WriteMessagesBenchmark
48*1b3f573fSAndroid Build Coastguard Worker     {
49*1b3f573fSAndroid Build Coastguard Worker         const int MaxMessages = 100;
50*1b3f573fSAndroid Build Coastguard Worker 
51*1b3f573fSAndroid Build Coastguard Worker         SubTest manyWrapperFieldsTest = new SubTest(ParseMessagesBenchmark.CreateManyWrapperFieldsMessage(), MaxMessages);
52*1b3f573fSAndroid Build Coastguard Worker         SubTest manyPrimitiveFieldsTest = new SubTest(ParseMessagesBenchmark.CreateManyPrimitiveFieldsMessage(), MaxMessages);
53*1b3f573fSAndroid Build Coastguard Worker         SubTest emptyMessageTest = new SubTest(new Empty(), MaxMessages);
54*1b3f573fSAndroid Build Coastguard Worker 
55*1b3f573fSAndroid Build Coastguard Worker         public IEnumerable<int> MessageCountValues => new[] { 10, 100 };
56*1b3f573fSAndroid Build Coastguard Worker 
57*1b3f573fSAndroid Build Coastguard Worker         [GlobalSetup]
GlobalSetup()58*1b3f573fSAndroid Build Coastguard Worker         public void GlobalSetup()
59*1b3f573fSAndroid Build Coastguard Worker         {
60*1b3f573fSAndroid Build Coastguard Worker         }
61*1b3f573fSAndroid Build Coastguard Worker 
62*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
ManyWrapperFieldsMessage_ToByteArray()63*1b3f573fSAndroid Build Coastguard Worker         public byte[] ManyWrapperFieldsMessage_ToByteArray()
64*1b3f573fSAndroid Build Coastguard Worker         {
65*1b3f573fSAndroid Build Coastguard Worker             return manyWrapperFieldsTest.ToByteArray();
66*1b3f573fSAndroid Build Coastguard Worker         }
67*1b3f573fSAndroid Build Coastguard Worker 
68*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
ManyWrapperFieldsMessage_WriteToCodedOutputStream()69*1b3f573fSAndroid Build Coastguard Worker         public byte[] ManyWrapperFieldsMessage_WriteToCodedOutputStream()
70*1b3f573fSAndroid Build Coastguard Worker         {
71*1b3f573fSAndroid Build Coastguard Worker             return manyWrapperFieldsTest.WriteToCodedOutputStream_PreAllocatedBuffer();
72*1b3f573fSAndroid Build Coastguard Worker         }
73*1b3f573fSAndroid Build Coastguard Worker 
74*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
ManyWrapperFieldsMessage_WriteToSpan()75*1b3f573fSAndroid Build Coastguard Worker         public byte[] ManyWrapperFieldsMessage_WriteToSpan()
76*1b3f573fSAndroid Build Coastguard Worker         {
77*1b3f573fSAndroid Build Coastguard Worker             return manyWrapperFieldsTest.WriteToSpan_PreAllocatedBuffer();
78*1b3f573fSAndroid Build Coastguard Worker         }
79*1b3f573fSAndroid Build Coastguard Worker 
80*1b3f573fSAndroid Build Coastguard Worker 
81*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
ManyPrimitiveFieldsMessage_ToByteArray()82*1b3f573fSAndroid Build Coastguard Worker         public byte[] ManyPrimitiveFieldsMessage_ToByteArray()
83*1b3f573fSAndroid Build Coastguard Worker         {
84*1b3f573fSAndroid Build Coastguard Worker             return manyPrimitiveFieldsTest.ToByteArray();
85*1b3f573fSAndroid Build Coastguard Worker         }
86*1b3f573fSAndroid Build Coastguard Worker 
87*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
ManyPrimitiveFieldsMessage_WriteToCodedOutputStream()88*1b3f573fSAndroid Build Coastguard Worker         public byte[] ManyPrimitiveFieldsMessage_WriteToCodedOutputStream()
89*1b3f573fSAndroid Build Coastguard Worker         {
90*1b3f573fSAndroid Build Coastguard Worker             return manyPrimitiveFieldsTest.WriteToCodedOutputStream_PreAllocatedBuffer();
91*1b3f573fSAndroid Build Coastguard Worker         }
92*1b3f573fSAndroid Build Coastguard Worker 
93*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
ManyPrimitiveFieldsMessage_WriteToSpan()94*1b3f573fSAndroid Build Coastguard Worker         public byte[] ManyPrimitiveFieldsMessage_WriteToSpan()
95*1b3f573fSAndroid Build Coastguard Worker         {
96*1b3f573fSAndroid Build Coastguard Worker             return manyPrimitiveFieldsTest.WriteToSpan_PreAllocatedBuffer();
97*1b3f573fSAndroid Build Coastguard Worker         }
98*1b3f573fSAndroid Build Coastguard Worker 
99*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
EmptyMessage_ToByteArray()100*1b3f573fSAndroid Build Coastguard Worker         public byte[] EmptyMessage_ToByteArray()
101*1b3f573fSAndroid Build Coastguard Worker         {
102*1b3f573fSAndroid Build Coastguard Worker             return emptyMessageTest.ToByteArray();
103*1b3f573fSAndroid Build Coastguard Worker         }
104*1b3f573fSAndroid Build Coastguard Worker 
105*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
EmptyMessage_WriteToCodedOutputStream()106*1b3f573fSAndroid Build Coastguard Worker         public byte[] EmptyMessage_WriteToCodedOutputStream()
107*1b3f573fSAndroid Build Coastguard Worker         {
108*1b3f573fSAndroid Build Coastguard Worker             return emptyMessageTest.WriteToCodedOutputStream_PreAllocatedBuffer();
109*1b3f573fSAndroid Build Coastguard Worker         }
110*1b3f573fSAndroid Build Coastguard Worker 
111*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
EmptyMessage_WriteToSpan()112*1b3f573fSAndroid Build Coastguard Worker         public byte[] EmptyMessage_WriteToSpan()
113*1b3f573fSAndroid Build Coastguard Worker         {
114*1b3f573fSAndroid Build Coastguard Worker             return emptyMessageTest.WriteToSpan_PreAllocatedBuffer();
115*1b3f573fSAndroid Build Coastguard Worker         }
116*1b3f573fSAndroid Build Coastguard Worker 
117*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
118*1b3f573fSAndroid Build Coastguard Worker         [ArgumentsSource(nameof(MessageCountValues))]
ManyWrapperFieldsMessage_WriteDelimitedMessagesToCodedOutputStream(int messageCount)119*1b3f573fSAndroid Build Coastguard Worker         public void ManyWrapperFieldsMessage_WriteDelimitedMessagesToCodedOutputStream(int messageCount)
120*1b3f573fSAndroid Build Coastguard Worker         {
121*1b3f573fSAndroid Build Coastguard Worker             manyWrapperFieldsTest.WriteDelimitedMessagesToCodedOutputStream_PreAllocatedBuffer(messageCount);
122*1b3f573fSAndroid Build Coastguard Worker         }
123*1b3f573fSAndroid Build Coastguard Worker 
124*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
125*1b3f573fSAndroid Build Coastguard Worker         [ArgumentsSource(nameof(MessageCountValues))]
ManyWrapperFieldsMessage_WriteDelimitedMessagesToSpan(int messageCount)126*1b3f573fSAndroid Build Coastguard Worker         public void ManyWrapperFieldsMessage_WriteDelimitedMessagesToSpan(int messageCount)
127*1b3f573fSAndroid Build Coastguard Worker         {
128*1b3f573fSAndroid Build Coastguard Worker             manyWrapperFieldsTest.WriteDelimitedMessagesToSpan_PreAllocatedBuffer(messageCount);
129*1b3f573fSAndroid Build Coastguard Worker         }
130*1b3f573fSAndroid Build Coastguard Worker 
131*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
132*1b3f573fSAndroid Build Coastguard Worker         [ArgumentsSource(nameof(MessageCountValues))]
ManyPrimitiveFieldsMessage_WriteDelimitedMessagesToCodedOutputStream(int messageCount)133*1b3f573fSAndroid Build Coastguard Worker         public void ManyPrimitiveFieldsMessage_WriteDelimitedMessagesToCodedOutputStream(int messageCount)
134*1b3f573fSAndroid Build Coastguard Worker         {
135*1b3f573fSAndroid Build Coastguard Worker             manyPrimitiveFieldsTest.WriteDelimitedMessagesToCodedOutputStream_PreAllocatedBuffer(messageCount);
136*1b3f573fSAndroid Build Coastguard Worker         }
137*1b3f573fSAndroid Build Coastguard Worker 
138*1b3f573fSAndroid Build Coastguard Worker         [Benchmark]
139*1b3f573fSAndroid Build Coastguard Worker         [ArgumentsSource(nameof(MessageCountValues))]
ManyPrimitiveFieldsMessage_WriteDelimitedMessagesToSpan(int messageCount)140*1b3f573fSAndroid Build Coastguard Worker         public void ManyPrimitiveFieldsMessage_WriteDelimitedMessagesToSpan(int messageCount)
141*1b3f573fSAndroid Build Coastguard Worker         {
142*1b3f573fSAndroid Build Coastguard Worker             manyPrimitiveFieldsTest.WriteDelimitedMessagesToSpan_PreAllocatedBuffer(messageCount);
143*1b3f573fSAndroid Build Coastguard Worker         }
144*1b3f573fSAndroid Build Coastguard Worker 
145*1b3f573fSAndroid Build Coastguard Worker         private class SubTest
146*1b3f573fSAndroid Build Coastguard Worker         {
147*1b3f573fSAndroid Build Coastguard Worker             private readonly IMessage message;
148*1b3f573fSAndroid Build Coastguard Worker             private readonly byte[] outputBuffer;
149*1b3f573fSAndroid Build Coastguard Worker             private readonly byte[] multipleMessagesOutputBuffer;
150*1b3f573fSAndroid Build Coastguard Worker 
SubTest(IMessage message, int maxMessageCount)151*1b3f573fSAndroid Build Coastguard Worker             public SubTest(IMessage message, int maxMessageCount)
152*1b3f573fSAndroid Build Coastguard Worker             {
153*1b3f573fSAndroid Build Coastguard Worker                 this.message = message;
154*1b3f573fSAndroid Build Coastguard Worker 
155*1b3f573fSAndroid Build Coastguard Worker                 int messageSize = message.CalculateSize();
156*1b3f573fSAndroid Build Coastguard Worker                 this.outputBuffer = new byte[messageSize];
157*1b3f573fSAndroid Build Coastguard Worker                 this.multipleMessagesOutputBuffer = new byte[maxMessageCount * (messageSize + CodedOutputStream.ComputeLengthSize(messageSize))];
158*1b3f573fSAndroid Build Coastguard Worker             }
159*1b3f573fSAndroid Build Coastguard Worker 
ToByteArray()160*1b3f573fSAndroid Build Coastguard Worker             public byte[] ToByteArray() => message.ToByteArray();
161*1b3f573fSAndroid Build Coastguard Worker 
WriteToCodedOutputStream_PreAllocatedBuffer()162*1b3f573fSAndroid Build Coastguard Worker             public byte[] WriteToCodedOutputStream_PreAllocatedBuffer()
163*1b3f573fSAndroid Build Coastguard Worker             {
164*1b3f573fSAndroid Build Coastguard Worker                 var cos = new CodedOutputStream(outputBuffer);  // use pre-existing output buffer
165*1b3f573fSAndroid Build Coastguard Worker                 message.WriteTo(cos);
166*1b3f573fSAndroid Build Coastguard Worker                 return outputBuffer;
167*1b3f573fSAndroid Build Coastguard Worker             }
168*1b3f573fSAndroid Build Coastguard Worker 
WriteToSpan_PreAllocatedBuffer()169*1b3f573fSAndroid Build Coastguard Worker             public byte[] WriteToSpan_PreAllocatedBuffer()
170*1b3f573fSAndroid Build Coastguard Worker             {
171*1b3f573fSAndroid Build Coastguard Worker                 var span = new Span<byte>(outputBuffer);  // use pre-existing output buffer
172*1b3f573fSAndroid Build Coastguard Worker                 message.WriteTo(span);
173*1b3f573fSAndroid Build Coastguard Worker                 return outputBuffer;
174*1b3f573fSAndroid Build Coastguard Worker             }
175*1b3f573fSAndroid Build Coastguard Worker 
WriteDelimitedMessagesToCodedOutputStream_PreAllocatedBuffer(int messageCount)176*1b3f573fSAndroid Build Coastguard Worker             public byte[] WriteDelimitedMessagesToCodedOutputStream_PreAllocatedBuffer(int messageCount)
177*1b3f573fSAndroid Build Coastguard Worker             {
178*1b3f573fSAndroid Build Coastguard Worker                 var cos = new CodedOutputStream(multipleMessagesOutputBuffer);  // use pre-existing output buffer
179*1b3f573fSAndroid Build Coastguard Worker                 for (int i = 0; i < messageCount; i++)
180*1b3f573fSAndroid Build Coastguard Worker                 {
181*1b3f573fSAndroid Build Coastguard Worker                     cos.WriteMessage(message);
182*1b3f573fSAndroid Build Coastguard Worker                 }
183*1b3f573fSAndroid Build Coastguard Worker                 return multipleMessagesOutputBuffer;
184*1b3f573fSAndroid Build Coastguard Worker             }
185*1b3f573fSAndroid Build Coastguard Worker 
WriteDelimitedMessagesToSpan_PreAllocatedBuffer(int messageCount)186*1b3f573fSAndroid Build Coastguard Worker             public byte[] WriteDelimitedMessagesToSpan_PreAllocatedBuffer(int messageCount)
187*1b3f573fSAndroid Build Coastguard Worker             {
188*1b3f573fSAndroid Build Coastguard Worker                 var span = new Span<byte>(multipleMessagesOutputBuffer);  // use pre-existing output buffer
189*1b3f573fSAndroid Build Coastguard Worker                 WriteContext.Initialize(ref span, out WriteContext ctx);
190*1b3f573fSAndroid Build Coastguard Worker                 for (int i = 0; i < messageCount; i++)
191*1b3f573fSAndroid Build Coastguard Worker                 {
192*1b3f573fSAndroid Build Coastguard Worker                     ctx.WriteMessage(message);
193*1b3f573fSAndroid Build Coastguard Worker                 }
194*1b3f573fSAndroid Build Coastguard Worker                 return multipleMessagesOutputBuffer;
195*1b3f573fSAndroid Build Coastguard Worker             }
196*1b3f573fSAndroid Build Coastguard Worker         }
197*1b3f573fSAndroid Build Coastguard Worker     }
198*1b3f573fSAndroid Build Coastguard Worker }
199