xref: /aosp_15_r20/external/protobuf/csharp/src/Google.Protobuf/UnknownFieldSet.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 2015 Google Inc.  All rights reserved.
4*1b3f573fSAndroid Build Coastguard Worker // https://developers.google.com/protocol-buffers/
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 System;
34*1b3f573fSAndroid Build Coastguard Worker using System.Collections.Generic;
35*1b3f573fSAndroid Build Coastguard Worker using System.IO;
36*1b3f573fSAndroid Build Coastguard Worker using System.Security;
37*1b3f573fSAndroid Build Coastguard Worker using Google.Protobuf.Reflection;
38*1b3f573fSAndroid Build Coastguard Worker 
39*1b3f573fSAndroid Build Coastguard Worker namespace Google.Protobuf
40*1b3f573fSAndroid Build Coastguard Worker {
41*1b3f573fSAndroid Build Coastguard Worker     /// <summary>
42*1b3f573fSAndroid Build Coastguard Worker     /// Used to keep track of fields which were seen when parsing a protocol message
43*1b3f573fSAndroid Build Coastguard Worker     /// but whose field numbers or types are unrecognized. This most frequently
44*1b3f573fSAndroid Build Coastguard Worker     /// occurs when new fields are added to a message type and then messages containing
45*1b3f573fSAndroid Build Coastguard Worker     /// those fields are read by old software that was built before the new types were
46*1b3f573fSAndroid Build Coastguard Worker     /// added.
47*1b3f573fSAndroid Build Coastguard Worker     ///
48*1b3f573fSAndroid Build Coastguard Worker     /// Most users will never need to use this class directly.
49*1b3f573fSAndroid Build Coastguard Worker     /// </summary>
50*1b3f573fSAndroid Build Coastguard Worker     public sealed partial class UnknownFieldSet
51*1b3f573fSAndroid Build Coastguard Worker     {
52*1b3f573fSAndroid Build Coastguard Worker         private readonly IDictionary<int, UnknownField> fields;
53*1b3f573fSAndroid Build Coastguard Worker 
54*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
55*1b3f573fSAndroid Build Coastguard Worker         /// Creates a new UnknownFieldSet.
56*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
UnknownFieldSet()57*1b3f573fSAndroid Build Coastguard Worker         internal UnknownFieldSet()
58*1b3f573fSAndroid Build Coastguard Worker         {
59*1b3f573fSAndroid Build Coastguard Worker             this.fields = new Dictionary<int, UnknownField>();
60*1b3f573fSAndroid Build Coastguard Worker         }
61*1b3f573fSAndroid Build Coastguard Worker 
62*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
63*1b3f573fSAndroid Build Coastguard Worker         /// Checks whether or not the given field number is present in the set.
64*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
HasField(int field)65*1b3f573fSAndroid Build Coastguard Worker         internal bool HasField(int field)
66*1b3f573fSAndroid Build Coastguard Worker         {
67*1b3f573fSAndroid Build Coastguard Worker             return fields.ContainsKey(field);
68*1b3f573fSAndroid Build Coastguard Worker         }
69*1b3f573fSAndroid Build Coastguard Worker 
70*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
71*1b3f573fSAndroid Build Coastguard Worker         /// Serializes the set and writes it to <paramref name="output"/>.
72*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
WriteTo(CodedOutputStream output)73*1b3f573fSAndroid Build Coastguard Worker         public void WriteTo(CodedOutputStream output)
74*1b3f573fSAndroid Build Coastguard Worker         {
75*1b3f573fSAndroid Build Coastguard Worker             WriteContext.Initialize(output, out WriteContext ctx);
76*1b3f573fSAndroid Build Coastguard Worker             try
77*1b3f573fSAndroid Build Coastguard Worker             {
78*1b3f573fSAndroid Build Coastguard Worker                 WriteTo(ref ctx);
79*1b3f573fSAndroid Build Coastguard Worker             }
80*1b3f573fSAndroid Build Coastguard Worker             finally
81*1b3f573fSAndroid Build Coastguard Worker             {
82*1b3f573fSAndroid Build Coastguard Worker                 ctx.CopyStateTo(output);
83*1b3f573fSAndroid Build Coastguard Worker             }
84*1b3f573fSAndroid Build Coastguard Worker         }
85*1b3f573fSAndroid Build Coastguard Worker 
86*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
87*1b3f573fSAndroid Build Coastguard Worker         /// Serializes the set and writes it to <paramref name="ctx"/>.
88*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
89*1b3f573fSAndroid Build Coastguard Worker         [SecuritySafeCritical]
WriteTo(ref WriteContext ctx)90*1b3f573fSAndroid Build Coastguard Worker         public void WriteTo(ref WriteContext ctx)
91*1b3f573fSAndroid Build Coastguard Worker         {
92*1b3f573fSAndroid Build Coastguard Worker             foreach (KeyValuePair<int, UnknownField> entry in fields)
93*1b3f573fSAndroid Build Coastguard Worker             {
94*1b3f573fSAndroid Build Coastguard Worker                 entry.Value.WriteTo(entry.Key, ref ctx);
95*1b3f573fSAndroid Build Coastguard Worker             }
96*1b3f573fSAndroid Build Coastguard Worker         }
97*1b3f573fSAndroid Build Coastguard Worker 
98*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
99*1b3f573fSAndroid Build Coastguard Worker         /// Gets the number of bytes required to encode this set.
100*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
CalculateSize()101*1b3f573fSAndroid Build Coastguard Worker         public int CalculateSize()
102*1b3f573fSAndroid Build Coastguard Worker         {
103*1b3f573fSAndroid Build Coastguard Worker             int result = 0;
104*1b3f573fSAndroid Build Coastguard Worker             foreach (KeyValuePair<int, UnknownField> entry in fields)
105*1b3f573fSAndroid Build Coastguard Worker             {
106*1b3f573fSAndroid Build Coastguard Worker                 result += entry.Value.GetSerializedSize(entry.Key);
107*1b3f573fSAndroid Build Coastguard Worker             }
108*1b3f573fSAndroid Build Coastguard Worker             return result;
109*1b3f573fSAndroid Build Coastguard Worker         }
110*1b3f573fSAndroid Build Coastguard Worker 
111*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
112*1b3f573fSAndroid Build Coastguard Worker         /// Checks if two unknown field sets are equal.
113*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
Equals(object other)114*1b3f573fSAndroid Build Coastguard Worker         public override bool Equals(object other)
115*1b3f573fSAndroid Build Coastguard Worker         {
116*1b3f573fSAndroid Build Coastguard Worker             if (ReferenceEquals(this, other))
117*1b3f573fSAndroid Build Coastguard Worker             {
118*1b3f573fSAndroid Build Coastguard Worker                 return true;
119*1b3f573fSAndroid Build Coastguard Worker             }
120*1b3f573fSAndroid Build Coastguard Worker             UnknownFieldSet otherSet = other as UnknownFieldSet;
121*1b3f573fSAndroid Build Coastguard Worker             IDictionary<int, UnknownField> otherFields = otherSet.fields;
122*1b3f573fSAndroid Build Coastguard Worker             if (fields.Count  != otherFields.Count)
123*1b3f573fSAndroid Build Coastguard Worker             {
124*1b3f573fSAndroid Build Coastguard Worker                 return false;
125*1b3f573fSAndroid Build Coastguard Worker             }
126*1b3f573fSAndroid Build Coastguard Worker             foreach (KeyValuePair<int, UnknownField> leftEntry in fields)
127*1b3f573fSAndroid Build Coastguard Worker             {
128*1b3f573fSAndroid Build Coastguard Worker                 UnknownField rightValue;
129*1b3f573fSAndroid Build Coastguard Worker                 if (!otherFields.TryGetValue(leftEntry.Key, out rightValue))
130*1b3f573fSAndroid Build Coastguard Worker                 {
131*1b3f573fSAndroid Build Coastguard Worker                     return false;
132*1b3f573fSAndroid Build Coastguard Worker                 }
133*1b3f573fSAndroid Build Coastguard Worker                 if (!leftEntry.Value.Equals(rightValue))
134*1b3f573fSAndroid Build Coastguard Worker                 {
135*1b3f573fSAndroid Build Coastguard Worker                     return false;
136*1b3f573fSAndroid Build Coastguard Worker                 }
137*1b3f573fSAndroid Build Coastguard Worker             }
138*1b3f573fSAndroid Build Coastguard Worker             return true;
139*1b3f573fSAndroid Build Coastguard Worker         }
140*1b3f573fSAndroid Build Coastguard Worker 
141*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
142*1b3f573fSAndroid Build Coastguard Worker         /// Gets the unknown field set's hash code.
143*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
GetHashCode()144*1b3f573fSAndroid Build Coastguard Worker         public override int GetHashCode()
145*1b3f573fSAndroid Build Coastguard Worker         {
146*1b3f573fSAndroid Build Coastguard Worker             int ret = 1;
147*1b3f573fSAndroid Build Coastguard Worker             foreach (KeyValuePair<int, UnknownField> field in fields)
148*1b3f573fSAndroid Build Coastguard Worker             {
149*1b3f573fSAndroid Build Coastguard Worker                 // Use ^ here to make the field order irrelevant.
150*1b3f573fSAndroid Build Coastguard Worker                 int hash = field.Key.GetHashCode() ^ field.Value.GetHashCode();
151*1b3f573fSAndroid Build Coastguard Worker                 ret ^= hash;
152*1b3f573fSAndroid Build Coastguard Worker             }
153*1b3f573fSAndroid Build Coastguard Worker             return ret;
154*1b3f573fSAndroid Build Coastguard Worker         }
155*1b3f573fSAndroid Build Coastguard Worker 
156*1b3f573fSAndroid Build Coastguard Worker         // Optimization:  We keep around the last field that was
157*1b3f573fSAndroid Build Coastguard Worker         // modified so that we can efficiently add to it multiple times in a
158*1b3f573fSAndroid Build Coastguard Worker         // row (important when parsing an unknown repeated field).
159*1b3f573fSAndroid Build Coastguard Worker         private int lastFieldNumber;
160*1b3f573fSAndroid Build Coastguard Worker         private UnknownField lastField;
161*1b3f573fSAndroid Build Coastguard Worker 
GetOrAddField(int number)162*1b3f573fSAndroid Build Coastguard Worker         private UnknownField GetOrAddField(int number)
163*1b3f573fSAndroid Build Coastguard Worker         {
164*1b3f573fSAndroid Build Coastguard Worker             if (lastField != null && number == lastFieldNumber)
165*1b3f573fSAndroid Build Coastguard Worker             {
166*1b3f573fSAndroid Build Coastguard Worker                 return lastField;
167*1b3f573fSAndroid Build Coastguard Worker             }
168*1b3f573fSAndroid Build Coastguard Worker             if (number == 0)
169*1b3f573fSAndroid Build Coastguard Worker             {
170*1b3f573fSAndroid Build Coastguard Worker                 return null;
171*1b3f573fSAndroid Build Coastguard Worker             }
172*1b3f573fSAndroid Build Coastguard Worker 
173*1b3f573fSAndroid Build Coastguard Worker             UnknownField existing;
174*1b3f573fSAndroid Build Coastguard Worker             if (fields.TryGetValue(number, out existing))
175*1b3f573fSAndroid Build Coastguard Worker             {
176*1b3f573fSAndroid Build Coastguard Worker                 return existing;
177*1b3f573fSAndroid Build Coastguard Worker             }
178*1b3f573fSAndroid Build Coastguard Worker             lastField = new UnknownField();
179*1b3f573fSAndroid Build Coastguard Worker             AddOrReplaceField(number, lastField);
180*1b3f573fSAndroid Build Coastguard Worker             lastFieldNumber = number;
181*1b3f573fSAndroid Build Coastguard Worker             return lastField;
182*1b3f573fSAndroid Build Coastguard Worker         }
183*1b3f573fSAndroid Build Coastguard Worker 
184*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
185*1b3f573fSAndroid Build Coastguard Worker         /// Adds a field to the set. If a field with the same number already exists, it
186*1b3f573fSAndroid Build Coastguard Worker         /// is replaced.
187*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
AddOrReplaceField(int number, UnknownField field)188*1b3f573fSAndroid Build Coastguard Worker         internal UnknownFieldSet AddOrReplaceField(int number, UnknownField field)
189*1b3f573fSAndroid Build Coastguard Worker         {
190*1b3f573fSAndroid Build Coastguard Worker             if (number == 0)
191*1b3f573fSAndroid Build Coastguard Worker             {
192*1b3f573fSAndroid Build Coastguard Worker                 throw new ArgumentOutOfRangeException("number", "Zero is not a valid field number.");
193*1b3f573fSAndroid Build Coastguard Worker             }
194*1b3f573fSAndroid Build Coastguard Worker             fields[number] = field;
195*1b3f573fSAndroid Build Coastguard Worker             return this;
196*1b3f573fSAndroid Build Coastguard Worker         }
197*1b3f573fSAndroid Build Coastguard Worker 
198*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
199*1b3f573fSAndroid Build Coastguard Worker         /// Parse a single field from <paramref name="ctx"/> and merge it
200*1b3f573fSAndroid Build Coastguard Worker         /// into this set.
201*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
202*1b3f573fSAndroid Build Coastguard Worker         /// <param name="ctx">The parse context from which to read the field</param>
203*1b3f573fSAndroid Build Coastguard Worker         /// <returns>false if the tag is an "end group" tag, true otherwise</returns>
MergeFieldFrom(ref ParseContext ctx)204*1b3f573fSAndroid Build Coastguard Worker         private bool MergeFieldFrom(ref ParseContext ctx)
205*1b3f573fSAndroid Build Coastguard Worker         {
206*1b3f573fSAndroid Build Coastguard Worker             uint tag = ctx.LastTag;
207*1b3f573fSAndroid Build Coastguard Worker             int number = WireFormat.GetTagFieldNumber(tag);
208*1b3f573fSAndroid Build Coastguard Worker             switch (WireFormat.GetTagWireType(tag))
209*1b3f573fSAndroid Build Coastguard Worker             {
210*1b3f573fSAndroid Build Coastguard Worker                 case WireFormat.WireType.Varint:
211*1b3f573fSAndroid Build Coastguard Worker                     {
212*1b3f573fSAndroid Build Coastguard Worker                         ulong uint64 = ctx.ReadUInt64();
213*1b3f573fSAndroid Build Coastguard Worker                         GetOrAddField(number).AddVarint(uint64);
214*1b3f573fSAndroid Build Coastguard Worker                         return true;
215*1b3f573fSAndroid Build Coastguard Worker                     }
216*1b3f573fSAndroid Build Coastguard Worker                 case WireFormat.WireType.Fixed32:
217*1b3f573fSAndroid Build Coastguard Worker                     {
218*1b3f573fSAndroid Build Coastguard Worker                         uint uint32 = ctx.ReadFixed32();
219*1b3f573fSAndroid Build Coastguard Worker                         GetOrAddField(number).AddFixed32(uint32);
220*1b3f573fSAndroid Build Coastguard Worker                         return true;
221*1b3f573fSAndroid Build Coastguard Worker                     }
222*1b3f573fSAndroid Build Coastguard Worker                 case WireFormat.WireType.Fixed64:
223*1b3f573fSAndroid Build Coastguard Worker                     {
224*1b3f573fSAndroid Build Coastguard Worker                         ulong uint64 = ctx.ReadFixed64();
225*1b3f573fSAndroid Build Coastguard Worker                         GetOrAddField(number).AddFixed64(uint64);
226*1b3f573fSAndroid Build Coastguard Worker                         return true;
227*1b3f573fSAndroid Build Coastguard Worker                     }
228*1b3f573fSAndroid Build Coastguard Worker                 case WireFormat.WireType.LengthDelimited:
229*1b3f573fSAndroid Build Coastguard Worker                     {
230*1b3f573fSAndroid Build Coastguard Worker                         ByteString bytes = ctx.ReadBytes();
231*1b3f573fSAndroid Build Coastguard Worker                         GetOrAddField(number).AddLengthDelimited(bytes);
232*1b3f573fSAndroid Build Coastguard Worker                         return true;
233*1b3f573fSAndroid Build Coastguard Worker                     }
234*1b3f573fSAndroid Build Coastguard Worker                 case WireFormat.WireType.StartGroup:
235*1b3f573fSAndroid Build Coastguard Worker                     {
236*1b3f573fSAndroid Build Coastguard Worker                         UnknownFieldSet set = new UnknownFieldSet();
237*1b3f573fSAndroid Build Coastguard Worker                         ParsingPrimitivesMessages.ReadGroup(ref ctx, number, set);
238*1b3f573fSAndroid Build Coastguard Worker                         GetOrAddField(number).AddGroup(set);
239*1b3f573fSAndroid Build Coastguard Worker                         return true;
240*1b3f573fSAndroid Build Coastguard Worker                     }
241*1b3f573fSAndroid Build Coastguard Worker                 case WireFormat.WireType.EndGroup:
242*1b3f573fSAndroid Build Coastguard Worker                     {
243*1b3f573fSAndroid Build Coastguard Worker                         return false;
244*1b3f573fSAndroid Build Coastguard Worker                     }
245*1b3f573fSAndroid Build Coastguard Worker                 default:
246*1b3f573fSAndroid Build Coastguard Worker                     throw InvalidProtocolBufferException.InvalidWireType();
247*1b3f573fSAndroid Build Coastguard Worker             }
248*1b3f573fSAndroid Build Coastguard Worker         }
249*1b3f573fSAndroid Build Coastguard Worker 
MergeGroupFrom(ref ParseContext ctx)250*1b3f573fSAndroid Build Coastguard Worker         internal void MergeGroupFrom(ref ParseContext ctx)
251*1b3f573fSAndroid Build Coastguard Worker         {
252*1b3f573fSAndroid Build Coastguard Worker             while (true)
253*1b3f573fSAndroid Build Coastguard Worker             {
254*1b3f573fSAndroid Build Coastguard Worker                 uint tag = ctx.ReadTag();
255*1b3f573fSAndroid Build Coastguard Worker                 if (tag == 0)
256*1b3f573fSAndroid Build Coastguard Worker                 {
257*1b3f573fSAndroid Build Coastguard Worker                     break;
258*1b3f573fSAndroid Build Coastguard Worker                 }
259*1b3f573fSAndroid Build Coastguard Worker                 if (!MergeFieldFrom(ref ctx))
260*1b3f573fSAndroid Build Coastguard Worker                 {
261*1b3f573fSAndroid Build Coastguard Worker                     break;
262*1b3f573fSAndroid Build Coastguard Worker                 }
263*1b3f573fSAndroid Build Coastguard Worker             }
264*1b3f573fSAndroid Build Coastguard Worker         }
265*1b3f573fSAndroid Build Coastguard Worker 
266*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
267*1b3f573fSAndroid Build Coastguard Worker         /// Create a new UnknownFieldSet if unknownFields is null.
268*1b3f573fSAndroid Build Coastguard Worker         /// Parse a single field from <paramref name="input"/> and merge it
269*1b3f573fSAndroid Build Coastguard Worker         /// into unknownFields. If <paramref name="input"/> is configured to discard unknown fields,
270*1b3f573fSAndroid Build Coastguard Worker         /// <paramref name="unknownFields"/> will be returned as-is and the field will be skipped.
271*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
272*1b3f573fSAndroid Build Coastguard Worker         /// <param name="unknownFields">The UnknownFieldSet which need to be merged</param>
273*1b3f573fSAndroid Build Coastguard Worker         /// <param name="input">The coded input stream containing the field</param>
274*1b3f573fSAndroid Build Coastguard Worker         /// <returns>The merged UnknownFieldSet</returns>
MergeFieldFrom(UnknownFieldSet unknownFields, CodedInputStream input)275*1b3f573fSAndroid Build Coastguard Worker         public static UnknownFieldSet MergeFieldFrom(UnknownFieldSet unknownFields,
276*1b3f573fSAndroid Build Coastguard Worker                                                      CodedInputStream input)
277*1b3f573fSAndroid Build Coastguard Worker         {
278*1b3f573fSAndroid Build Coastguard Worker             ParseContext.Initialize(input, out ParseContext ctx);
279*1b3f573fSAndroid Build Coastguard Worker             try
280*1b3f573fSAndroid Build Coastguard Worker             {
281*1b3f573fSAndroid Build Coastguard Worker                 return MergeFieldFrom(unknownFields, ref ctx);
282*1b3f573fSAndroid Build Coastguard Worker             }
283*1b3f573fSAndroid Build Coastguard Worker             finally
284*1b3f573fSAndroid Build Coastguard Worker             {
285*1b3f573fSAndroid Build Coastguard Worker                 ctx.CopyStateTo(input);
286*1b3f573fSAndroid Build Coastguard Worker             }
287*1b3f573fSAndroid Build Coastguard Worker         }
288*1b3f573fSAndroid Build Coastguard Worker 
289*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
290*1b3f573fSAndroid Build Coastguard Worker         /// Create a new UnknownFieldSet if unknownFields is null.
291*1b3f573fSAndroid Build Coastguard Worker         /// Parse a single field from <paramref name="ctx"/> and merge it
292*1b3f573fSAndroid Build Coastguard Worker         /// into unknownFields. If <paramref name="ctx"/> is configured to discard unknown fields,
293*1b3f573fSAndroid Build Coastguard Worker         /// <paramref name="unknownFields"/> will be returned as-is and the field will be skipped.
294*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
295*1b3f573fSAndroid Build Coastguard Worker         /// <param name="unknownFields">The UnknownFieldSet which need to be merged</param>
296*1b3f573fSAndroid Build Coastguard Worker         /// <param name="ctx">The parse context from which to read the field</param>
297*1b3f573fSAndroid Build Coastguard Worker         /// <returns>The merged UnknownFieldSet</returns>
298*1b3f573fSAndroid Build Coastguard Worker         [SecuritySafeCritical]
MergeFieldFrom(UnknownFieldSet unknownFields, ref ParseContext ctx)299*1b3f573fSAndroid Build Coastguard Worker         public static UnknownFieldSet MergeFieldFrom(UnknownFieldSet unknownFields,
300*1b3f573fSAndroid Build Coastguard Worker                                                      ref ParseContext ctx)
301*1b3f573fSAndroid Build Coastguard Worker         {
302*1b3f573fSAndroid Build Coastguard Worker             if (ctx.DiscardUnknownFields)
303*1b3f573fSAndroid Build Coastguard Worker             {
304*1b3f573fSAndroid Build Coastguard Worker                 ParsingPrimitivesMessages.SkipLastField(ref ctx.buffer, ref ctx.state);
305*1b3f573fSAndroid Build Coastguard Worker                 return unknownFields;
306*1b3f573fSAndroid Build Coastguard Worker             }
307*1b3f573fSAndroid Build Coastguard Worker             if (unknownFields == null)
308*1b3f573fSAndroid Build Coastguard Worker             {
309*1b3f573fSAndroid Build Coastguard Worker                 unknownFields = new UnknownFieldSet();
310*1b3f573fSAndroid Build Coastguard Worker             }
311*1b3f573fSAndroid Build Coastguard Worker             if (!unknownFields.MergeFieldFrom(ref ctx))
312*1b3f573fSAndroid Build Coastguard Worker             {
313*1b3f573fSAndroid Build Coastguard Worker                 throw new InvalidProtocolBufferException("Merge an unknown field of end-group tag, indicating that the corresponding start-group was missing."); // match the old code-gen
314*1b3f573fSAndroid Build Coastguard Worker             }
315*1b3f573fSAndroid Build Coastguard Worker             return unknownFields;
316*1b3f573fSAndroid Build Coastguard Worker         }
317*1b3f573fSAndroid Build Coastguard Worker 
318*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
319*1b3f573fSAndroid Build Coastguard Worker         /// Merges the fields from <paramref name="other"/> into this set.
320*1b3f573fSAndroid Build Coastguard Worker         /// If a field number exists in both sets, the values in <paramref name="other"/>
321*1b3f573fSAndroid Build Coastguard Worker         /// will be appended to the values in this set.
322*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
MergeFrom(UnknownFieldSet other)323*1b3f573fSAndroid Build Coastguard Worker         private UnknownFieldSet MergeFrom(UnknownFieldSet other)
324*1b3f573fSAndroid Build Coastguard Worker         {
325*1b3f573fSAndroid Build Coastguard Worker             if (other != null)
326*1b3f573fSAndroid Build Coastguard Worker             {
327*1b3f573fSAndroid Build Coastguard Worker                 foreach (KeyValuePair<int, UnknownField> entry in other.fields)
328*1b3f573fSAndroid Build Coastguard Worker                 {
329*1b3f573fSAndroid Build Coastguard Worker                     MergeField(entry.Key, entry.Value);
330*1b3f573fSAndroid Build Coastguard Worker                 }
331*1b3f573fSAndroid Build Coastguard Worker             }
332*1b3f573fSAndroid Build Coastguard Worker             return this;
333*1b3f573fSAndroid Build Coastguard Worker         }
334*1b3f573fSAndroid Build Coastguard Worker 
335*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
336*1b3f573fSAndroid Build Coastguard Worker         /// Created a new UnknownFieldSet to <paramref name="unknownFields"/> if
337*1b3f573fSAndroid Build Coastguard Worker         /// needed and merges the fields from <paramref name="other"/> into the first set.
338*1b3f573fSAndroid Build Coastguard Worker         /// If a field number exists in both sets, the values in <paramref name="other"/>
339*1b3f573fSAndroid Build Coastguard Worker         /// will be appended to the values in this set.
340*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
MergeFrom(UnknownFieldSet unknownFields, UnknownFieldSet other)341*1b3f573fSAndroid Build Coastguard Worker         public static UnknownFieldSet MergeFrom(UnknownFieldSet unknownFields,
342*1b3f573fSAndroid Build Coastguard Worker                                                 UnknownFieldSet other)
343*1b3f573fSAndroid Build Coastguard Worker         {
344*1b3f573fSAndroid Build Coastguard Worker             if (other == null)
345*1b3f573fSAndroid Build Coastguard Worker             {
346*1b3f573fSAndroid Build Coastguard Worker                 return unknownFields;
347*1b3f573fSAndroid Build Coastguard Worker             }
348*1b3f573fSAndroid Build Coastguard Worker             if (unknownFields == null)
349*1b3f573fSAndroid Build Coastguard Worker             {
350*1b3f573fSAndroid Build Coastguard Worker                 unknownFields = new UnknownFieldSet();
351*1b3f573fSAndroid Build Coastguard Worker             }
352*1b3f573fSAndroid Build Coastguard Worker             unknownFields.MergeFrom(other);
353*1b3f573fSAndroid Build Coastguard Worker             return unknownFields;
354*1b3f573fSAndroid Build Coastguard Worker         }
355*1b3f573fSAndroid Build Coastguard Worker 
356*1b3f573fSAndroid Build Coastguard Worker 
357*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
358*1b3f573fSAndroid Build Coastguard Worker         /// Adds a field to the unknown field set. If a field with the same
359*1b3f573fSAndroid Build Coastguard Worker         /// number already exists, the two are merged.
360*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
MergeField(int number, UnknownField field)361*1b3f573fSAndroid Build Coastguard Worker         private UnknownFieldSet MergeField(int number, UnknownField field)
362*1b3f573fSAndroid Build Coastguard Worker         {
363*1b3f573fSAndroid Build Coastguard Worker             if (number == 0)
364*1b3f573fSAndroid Build Coastguard Worker             {
365*1b3f573fSAndroid Build Coastguard Worker                 throw new ArgumentOutOfRangeException("number", "Zero is not a valid field number.");
366*1b3f573fSAndroid Build Coastguard Worker             }
367*1b3f573fSAndroid Build Coastguard Worker             if (HasField(number))
368*1b3f573fSAndroid Build Coastguard Worker             {
369*1b3f573fSAndroid Build Coastguard Worker                 GetOrAddField(number).MergeFrom(field);
370*1b3f573fSAndroid Build Coastguard Worker             }
371*1b3f573fSAndroid Build Coastguard Worker             else
372*1b3f573fSAndroid Build Coastguard Worker             {
373*1b3f573fSAndroid Build Coastguard Worker                 AddOrReplaceField(number, field);
374*1b3f573fSAndroid Build Coastguard Worker             }
375*1b3f573fSAndroid Build Coastguard Worker             return this;
376*1b3f573fSAndroid Build Coastguard Worker         }
377*1b3f573fSAndroid Build Coastguard Worker 
378*1b3f573fSAndroid Build Coastguard Worker         /// <summary>
379*1b3f573fSAndroid Build Coastguard Worker         /// Clone an unknown field set from <paramref name="other"/>.
380*1b3f573fSAndroid Build Coastguard Worker         /// </summary>
Clone(UnknownFieldSet other)381*1b3f573fSAndroid Build Coastguard Worker         public static UnknownFieldSet Clone(UnknownFieldSet other)
382*1b3f573fSAndroid Build Coastguard Worker         {
383*1b3f573fSAndroid Build Coastguard Worker             if (other == null)
384*1b3f573fSAndroid Build Coastguard Worker             {
385*1b3f573fSAndroid Build Coastguard Worker                 return null;
386*1b3f573fSAndroid Build Coastguard Worker             }
387*1b3f573fSAndroid Build Coastguard Worker             UnknownFieldSet unknownFields = new UnknownFieldSet();
388*1b3f573fSAndroid Build Coastguard Worker             unknownFields.MergeFrom(other);
389*1b3f573fSAndroid Build Coastguard Worker             return unknownFields;
390*1b3f573fSAndroid Build Coastguard Worker         }
391*1b3f573fSAndroid Build Coastguard Worker     }
392*1b3f573fSAndroid Build Coastguard Worker }
393*1b3f573fSAndroid Build Coastguard Worker 
394