MessageGenerator.cs 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885
  1. #region Copyright notice and license
  2. // Protocol Buffers - Google's data interchange format
  3. // Copyright 2008 Google Inc. All rights reserved.
  4. // http://github.com/jskeet/dotnet-protobufs/
  5. // Original C++/Java/Python code:
  6. // http://code.google.com/p/protobuf/
  7. //
  8. // Redistribution and use in source and binary forms, with or without
  9. // modification, are permitted provided that the following conditions are
  10. // met:
  11. //
  12. // * Redistributions of source code must retain the above copyright
  13. // notice, this list of conditions and the following disclaimer.
  14. // * Redistributions in binary form must reproduce the above
  15. // copyright notice, this list of conditions and the following disclaimer
  16. // in the documentation and/or other materials provided with the
  17. // distribution.
  18. // * Neither the name of Google Inc. nor the names of its
  19. // contributors may be used to endorse or promote products derived from
  20. // this software without specific prior written permission.
  21. //
  22. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  23. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  24. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  25. // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  26. // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  27. // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  28. // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  29. // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  30. // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  31. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  32. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  33. #endregion
  34. using System;
  35. using System.Collections.Generic;
  36. using Google.ProtocolBuffers.DescriptorProtos;
  37. using Google.ProtocolBuffers.Descriptors;
  38. namespace Google.ProtocolBuffers.ProtoGen
  39. {
  40. internal class MessageGenerator : SourceGeneratorBase<MessageDescriptor>, ISourceGenerator
  41. {
  42. private string[] _fieldNames;
  43. internal MessageGenerator(MessageDescriptor descriptor) : base(descriptor)
  44. {
  45. }
  46. private string ClassName
  47. {
  48. get { return Descriptor.Name; }
  49. }
  50. private string FullClassName
  51. {
  52. get { return GetClassName(Descriptor); }
  53. }
  54. /// <summary>
  55. /// Get an identifier that uniquely identifies this type within the file.
  56. /// This is used to declare static variables related to this type at the
  57. /// outermost file scope.
  58. /// </summary>
  59. private static string GetUniqueFileScopeIdentifier(IDescriptor descriptor)
  60. {
  61. return "static_" + descriptor.FullName.Replace(".", "_");
  62. }
  63. internal void GenerateStaticVariables(TextGenerator writer)
  64. {
  65. // Because descriptor.proto (Google.ProtocolBuffers.DescriptorProtos) is
  66. // used in the construction of descriptors, we have a tricky bootstrapping
  67. // problem. To help control static initialization order, we make sure all
  68. // descriptors and other static data that depends on them are members of
  69. // the proto-descriptor class. This way, they will be initialized in
  70. // a deterministic order.
  71. string identifier = GetUniqueFileScopeIdentifier(Descriptor);
  72. if (!UseLiteRuntime)
  73. {
  74. // The descriptor for this type.
  75. string access = Descriptor.File.CSharpOptions.NestClasses ? "private" : "internal";
  76. writer.WriteLine("{0} static pbd::MessageDescriptor internal__{1}__Descriptor;", access, identifier);
  77. writer.WriteLine(
  78. "{0} static pb::FieldAccess.FieldAccessorTable<{1}, {1}.Builder> internal__{2}__FieldAccessorTable;",
  79. access, FullClassName, identifier);
  80. }
  81. // Generate static members for all nested types.
  82. foreach (MessageDescriptor nestedMessage in Descriptor.NestedTypes)
  83. {
  84. new MessageGenerator(nestedMessage).GenerateStaticVariables(writer);
  85. }
  86. }
  87. internal void GenerateStaticVariableInitializers(TextGenerator writer)
  88. {
  89. string identifier = GetUniqueFileScopeIdentifier(Descriptor);
  90. if (!UseLiteRuntime)
  91. {
  92. writer.Write("internal__{0}__Descriptor = ", identifier);
  93. if (Descriptor.ContainingType == null)
  94. {
  95. writer.WriteLine("Descriptor.MessageTypes[{0}];", Descriptor.Index);
  96. }
  97. else
  98. {
  99. writer.WriteLine("internal__{0}__Descriptor.NestedTypes[{1}];",
  100. GetUniqueFileScopeIdentifier(Descriptor.ContainingType), Descriptor.Index);
  101. }
  102. writer.WriteLine("internal__{0}__FieldAccessorTable = ", identifier);
  103. writer.WriteLine(
  104. " new pb::FieldAccess.FieldAccessorTable<{1}, {1}.Builder>(internal__{0}__Descriptor,",
  105. identifier, FullClassName);
  106. writer.Print(" new string[] { ");
  107. foreach (FieldDescriptor field in Descriptor.Fields)
  108. {
  109. writer.Write("\"{0}\", ", field.CSharpOptions.PropertyName);
  110. }
  111. writer.WriteLine("});");
  112. }
  113. // Generate static member initializers for all nested types.
  114. foreach (MessageDescriptor nestedMessage in Descriptor.NestedTypes)
  115. {
  116. new MessageGenerator(nestedMessage).GenerateStaticVariableInitializers(writer);
  117. }
  118. foreach (FieldDescriptor extension in Descriptor.Extensions)
  119. {
  120. new ExtensionGenerator(extension).GenerateStaticVariableInitializers(writer);
  121. }
  122. }
  123. public string[] FieldNames
  124. {
  125. get
  126. {
  127. if (_fieldNames == null)
  128. {
  129. List<string> names = new List<string>();
  130. foreach (FieldDescriptor fieldDescriptor in Descriptor.Fields)
  131. {
  132. names.Add(fieldDescriptor.Name);
  133. }
  134. //if you change this, the search must also change in GenerateBuilderParsingMethods
  135. names.Sort(StringComparer.Ordinal);
  136. _fieldNames = names.ToArray();
  137. }
  138. return _fieldNames;
  139. }
  140. }
  141. internal int FieldOrdinal(FieldDescriptor field)
  142. {
  143. return Array.BinarySearch(FieldNames, field.Name, StringComparer.Ordinal);
  144. }
  145. private IFieldSourceGenerator CreateFieldGenerator(FieldDescriptor fieldDescriptor)
  146. {
  147. return SourceGenerators.CreateFieldGenerator(fieldDescriptor, FieldOrdinal(fieldDescriptor));
  148. }
  149. public void Generate(TextGenerator writer)
  150. {
  151. if (Descriptor.File.CSharpOptions.AddSerializable)
  152. {
  153. writer.WriteLine("[global::System.SerializableAttribute()]");
  154. }
  155. writer.WriteLine("[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]");
  156. WriteGeneratedCodeAttributes(writer);
  157. writer.WriteLine("{0} sealed partial class {1} : pb::{2}Message{3}<{1}, {1}.Builder> {{",
  158. ClassAccessLevel, ClassName,
  159. Descriptor.Proto.ExtensionRangeCount > 0 ? "Extendable" : "Generated",
  160. RuntimeSuffix);
  161. writer.Indent();
  162. if (Descriptor.File.CSharpOptions.GeneratePrivateCtor)
  163. {
  164. writer.WriteLine("private {0}() {{ }}", ClassName);
  165. }
  166. // Must call MakeReadOnly() to make sure all lists are made read-only
  167. writer.WriteLine("private static readonly {0} defaultInstance = new {0}().MakeReadOnly();", ClassName);
  168. if (OptimizeSpeed)
  169. {
  170. writer.WriteLine("private static readonly string[] _{0}FieldNames = new string[] {{ {2}{1}{2} }};",
  171. NameHelpers.UnderscoresToCamelCase(ClassName), String.Join("\", \"", FieldNames),
  172. FieldNames.Length > 0 ? "\"" : "");
  173. List<string> tags = new List<string>();
  174. foreach (string name in FieldNames)
  175. {
  176. tags.Add(WireFormat.MakeTag(Descriptor.FindFieldByName(name)).ToString());
  177. }
  178. writer.WriteLine("private static readonly uint[] _{0}FieldTags = new uint[] {{ {1} }};",
  179. NameHelpers.UnderscoresToCamelCase(ClassName), String.Join(", ", tags.ToArray()));
  180. }
  181. writer.WriteLine("public static {0} DefaultInstance {{", ClassName);
  182. writer.WriteLine(" get { return defaultInstance; }");
  183. writer.WriteLine("}");
  184. writer.WriteLine();
  185. writer.WriteLine("public override {0} DefaultInstanceForType {{", ClassName);
  186. writer.WriteLine(" get { return DefaultInstance; }");
  187. writer.WriteLine("}");
  188. writer.WriteLine();
  189. writer.WriteLine("protected override {0} ThisMessage {{", ClassName);
  190. writer.WriteLine(" get { return this; }");
  191. writer.WriteLine("}");
  192. writer.WriteLine();
  193. if (!UseLiteRuntime)
  194. {
  195. writer.WriteLine("public static pbd::MessageDescriptor Descriptor {");
  196. writer.WriteLine(" get {{ return {0}.internal__{1}__Descriptor; }}",
  197. DescriptorUtil.GetFullUmbrellaClassName(Descriptor),
  198. GetUniqueFileScopeIdentifier(Descriptor));
  199. writer.WriteLine("}");
  200. writer.WriteLine();
  201. writer.WriteLine(
  202. "protected override pb::FieldAccess.FieldAccessorTable<{0}, {0}.Builder> InternalFieldAccessors {{",
  203. ClassName);
  204. writer.WriteLine(" get {{ return {0}.internal__{1}__FieldAccessorTable; }}",
  205. DescriptorUtil.GetFullUmbrellaClassName(Descriptor),
  206. GetUniqueFileScopeIdentifier(Descriptor));
  207. writer.WriteLine("}");
  208. writer.WriteLine();
  209. }
  210. // Extensions don't need to go in an extra nested type
  211. WriteChildren(writer, null, Descriptor.Extensions);
  212. if (Descriptor.EnumTypes.Count + Descriptor.NestedTypes.Count > 0)
  213. {
  214. writer.WriteLine("#region Nested types");
  215. writer.WriteLine("[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]");
  216. WriteGeneratedCodeAttributes(writer);
  217. writer.WriteLine("public static partial class Types {");
  218. writer.Indent();
  219. WriteChildren(writer, null, Descriptor.EnumTypes);
  220. WriteChildren(writer, null, Descriptor.NestedTypes);
  221. writer.Outdent();
  222. writer.WriteLine("}");
  223. writer.WriteLine("#endregion");
  224. writer.WriteLine();
  225. }
  226. foreach (FieldDescriptor fieldDescriptor in Descriptor.Fields)
  227. {
  228. if (Descriptor.File.CSharpOptions.ClsCompliance && GetFieldConstantName(fieldDescriptor).StartsWith("_"))
  229. {
  230. writer.WriteLine("[global::System.CLSCompliant(false)]");
  231. }
  232. // Rats: we lose the debug comment here :(
  233. writer.WriteLine("public const int {0} = {1};", GetFieldConstantName(fieldDescriptor),
  234. fieldDescriptor.FieldNumber);
  235. CreateFieldGenerator(fieldDescriptor).GenerateMembers(writer);
  236. writer.WriteLine();
  237. }
  238. if (OptimizeSpeed)
  239. {
  240. GenerateIsInitialized(writer);
  241. GenerateMessageSerializationMethods(writer);
  242. }
  243. if (UseLiteRuntime)
  244. {
  245. GenerateLiteRuntimeMethods(writer);
  246. }
  247. GenerateParseFromMethods(writer);
  248. GenerateBuilder(writer);
  249. // Force the static initialization code for the file to run, since it may
  250. // initialize static variables declared in this class.
  251. writer.WriteLine("static {0}() {{", ClassName);
  252. // We call object.ReferenceEquals() just to make it a valid statement on its own.
  253. // Another option would be GetType(), but that causes problems in DescriptorProtoFile,
  254. // where the bootstrapping is somewhat recursive - type initializers call
  255. // each other, effectively. We temporarily see Descriptor as null.
  256. writer.WriteLine(" object.ReferenceEquals({0}.Descriptor, null);",
  257. DescriptorUtil.GetFullUmbrellaClassName(Descriptor));
  258. writer.WriteLine("}");
  259. writer.Outdent();
  260. writer.WriteLine("}");
  261. writer.WriteLine();
  262. }
  263. private void GenerateLiteRuntimeMethods(TextGenerator writer)
  264. {
  265. bool callbase = Descriptor.Proto.ExtensionRangeCount > 0;
  266. writer.WriteLine("#region Lite runtime methods");
  267. writer.WriteLine("public override int GetHashCode() {");
  268. writer.Indent();
  269. writer.WriteLine("int hash = GetType().GetHashCode();");
  270. foreach (FieldDescriptor fieldDescriptor in Descriptor.Fields)
  271. {
  272. CreateFieldGenerator(fieldDescriptor).WriteHash(writer);
  273. }
  274. if (callbase)
  275. {
  276. writer.WriteLine("hash ^= base.GetHashCode();");
  277. }
  278. writer.WriteLine("return hash;");
  279. writer.Outdent();
  280. writer.WriteLine("}");
  281. writer.WriteLine();
  282. writer.WriteLine("public override bool Equals(object obj) {");
  283. writer.Indent();
  284. writer.WriteLine("{0} other = obj as {0};", ClassName);
  285. writer.WriteLine("if (other == null) return false;");
  286. foreach (FieldDescriptor fieldDescriptor in Descriptor.Fields)
  287. {
  288. CreateFieldGenerator(fieldDescriptor).WriteEquals(writer);
  289. }
  290. if (callbase)
  291. {
  292. writer.WriteLine("if (!base.Equals(other)) return false;");
  293. }
  294. writer.WriteLine("return true;");
  295. writer.Outdent();
  296. writer.WriteLine("}");
  297. writer.WriteLine();
  298. writer.WriteLine("public override void PrintTo(global::System.IO.TextWriter writer) {");
  299. writer.Indent();
  300. List<FieldDescriptor> sorted = new List<FieldDescriptor>(Descriptor.Fields);
  301. sorted.Sort(
  302. new Comparison<FieldDescriptor>(
  303. delegate(FieldDescriptor a, FieldDescriptor b) { return a.FieldNumber.CompareTo(b.FieldNumber); }));
  304. foreach (FieldDescriptor fieldDescriptor in sorted)
  305. {
  306. CreateFieldGenerator(fieldDescriptor).WriteToString(writer);
  307. }
  308. if (callbase)
  309. {
  310. writer.WriteLine("base.PrintTo(writer);");
  311. }
  312. writer.Outdent();
  313. writer.WriteLine("}");
  314. writer.WriteLine("#endregion");
  315. writer.WriteLine();
  316. }
  317. private void GenerateMessageSerializationMethods(TextGenerator writer)
  318. {
  319. List<FieldDescriptor> sortedFields = new List<FieldDescriptor>(Descriptor.Fields);
  320. sortedFields.Sort((f1, f2) => f1.FieldNumber.CompareTo(f2.FieldNumber));
  321. List<DescriptorProto.Types.ExtensionRange> sortedExtensions =
  322. new List<DescriptorProto.Types.ExtensionRange>(Descriptor.Proto.ExtensionRangeList);
  323. sortedExtensions.Sort((r1, r2) => (r1.Start.CompareTo(r2.Start)));
  324. writer.WriteLine("public override void WriteTo(pb::ICodedOutputStream output) {");
  325. writer.Indent();
  326. // Make sure we've computed the serialized length, so that packed fields are generated correctly.
  327. writer.WriteLine("int size = SerializedSize;");
  328. writer.WriteLine("string[] field_names = _{0}FieldNames;", NameHelpers.UnderscoresToCamelCase(ClassName));
  329. if (Descriptor.Proto.ExtensionRangeList.Count > 0)
  330. {
  331. writer.WriteLine(
  332. "pb::ExtendableMessage{1}<{0}, {0}.Builder>.ExtensionWriter extensionWriter = CreateExtensionWriter(this);",
  333. ClassName, RuntimeSuffix);
  334. }
  335. // Merge the fields and the extension ranges, both sorted by field number.
  336. for (int i = 0, j = 0; i < Descriptor.Fields.Count || j < sortedExtensions.Count;)
  337. {
  338. if (i == Descriptor.Fields.Count)
  339. {
  340. GenerateSerializeOneExtensionRange(writer, sortedExtensions[j++]);
  341. }
  342. else if (j == sortedExtensions.Count)
  343. {
  344. GenerateSerializeOneField(writer, sortedFields[i++]);
  345. }
  346. else if (sortedFields[i].FieldNumber < sortedExtensions[j].Start)
  347. {
  348. GenerateSerializeOneField(writer, sortedFields[i++]);
  349. }
  350. else
  351. {
  352. GenerateSerializeOneExtensionRange(writer, sortedExtensions[j++]);
  353. }
  354. }
  355. if (!UseLiteRuntime)
  356. {
  357. if (Descriptor.Proto.Options.MessageSetWireFormat)
  358. {
  359. writer.WriteLine("UnknownFields.WriteAsMessageSetTo(output);");
  360. }
  361. else
  362. {
  363. writer.WriteLine("UnknownFields.WriteTo(output);");
  364. }
  365. }
  366. writer.Outdent();
  367. writer.WriteLine("}");
  368. writer.WriteLine();
  369. writer.WriteLine("private int memoizedSerializedSize = -1;");
  370. writer.WriteLine("public override int SerializedSize {");
  371. writer.Indent();
  372. writer.WriteLine("get {");
  373. writer.Indent();
  374. writer.WriteLine("int size = memoizedSerializedSize;");
  375. writer.WriteLine("if (size != -1) return size;");
  376. writer.WriteLine();
  377. writer.WriteLine("size = 0;");
  378. foreach (FieldDescriptor field in Descriptor.Fields)
  379. {
  380. CreateFieldGenerator(field).GenerateSerializedSizeCode(writer);
  381. }
  382. if (Descriptor.Proto.ExtensionRangeCount > 0)
  383. {
  384. writer.WriteLine("size += ExtensionsSerializedSize;");
  385. }
  386. if (!UseLiteRuntime)
  387. {
  388. if (Descriptor.Options.MessageSetWireFormat)
  389. {
  390. writer.WriteLine("size += UnknownFields.SerializedSizeAsMessageSet;");
  391. }
  392. else
  393. {
  394. writer.WriteLine("size += UnknownFields.SerializedSize;");
  395. }
  396. }
  397. writer.WriteLine("memoizedSerializedSize = size;");
  398. writer.WriteLine("return size;");
  399. writer.Outdent();
  400. writer.WriteLine("}");
  401. writer.Outdent();
  402. writer.WriteLine("}");
  403. writer.WriteLine();
  404. }
  405. private void GenerateSerializeOneField(TextGenerator writer, FieldDescriptor fieldDescriptor)
  406. {
  407. CreateFieldGenerator(fieldDescriptor).GenerateSerializationCode(writer);
  408. }
  409. private static void GenerateSerializeOneExtensionRange(TextGenerator writer,
  410. DescriptorProto.Types.ExtensionRange extensionRange)
  411. {
  412. writer.WriteLine("extensionWriter.WriteUntil({0}, output);", extensionRange.End);
  413. }
  414. private void GenerateParseFromMethods(TextGenerator writer)
  415. {
  416. // Note: These are separate from GenerateMessageSerializationMethods()
  417. // because they need to be generated even for messages that are optimized
  418. // for code size.
  419. writer.WriteLine("public static {0} ParseFrom(pb::ByteString data) {{", ClassName);
  420. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(data)).BuildParsed();");
  421. writer.WriteLine("}");
  422. writer.WriteLine(
  423. "public static {0} ParseFrom(pb::ByteString data, pb::ExtensionRegistry extensionRegistry) {{",
  424. ClassName);
  425. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(data, extensionRegistry)).BuildParsed();");
  426. writer.WriteLine("}");
  427. writer.WriteLine("public static {0} ParseFrom(byte[] data) {{", ClassName);
  428. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(data)).BuildParsed();");
  429. writer.WriteLine("}");
  430. writer.WriteLine("public static {0} ParseFrom(byte[] data, pb::ExtensionRegistry extensionRegistry) {{",
  431. ClassName);
  432. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(data, extensionRegistry)).BuildParsed();");
  433. writer.WriteLine("}");
  434. writer.WriteLine("public static {0} ParseFrom(global::System.IO.Stream input) {{", ClassName);
  435. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(input)).BuildParsed();");
  436. writer.WriteLine("}");
  437. writer.WriteLine(
  438. "public static {0} ParseFrom(global::System.IO.Stream input, pb::ExtensionRegistry extensionRegistry) {{",
  439. ClassName);
  440. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(input, extensionRegistry)).BuildParsed();");
  441. writer.WriteLine("}");
  442. writer.WriteLine("public static {0} ParseDelimitedFrom(global::System.IO.Stream input) {{", ClassName);
  443. writer.WriteLine(" return CreateBuilder().MergeDelimitedFrom(input).BuildParsed();");
  444. writer.WriteLine("}");
  445. writer.WriteLine(
  446. "public static {0} ParseDelimitedFrom(global::System.IO.Stream input, pb::ExtensionRegistry extensionRegistry) {{",
  447. ClassName);
  448. writer.WriteLine(" return CreateBuilder().MergeDelimitedFrom(input, extensionRegistry).BuildParsed();");
  449. writer.WriteLine("}");
  450. writer.WriteLine("public static {0} ParseFrom(pb::ICodedInputStream input) {{", ClassName);
  451. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(input)).BuildParsed();");
  452. writer.WriteLine("}");
  453. writer.WriteLine(
  454. "public static {0} ParseFrom(pb::ICodedInputStream input, pb::ExtensionRegistry extensionRegistry) {{",
  455. ClassName);
  456. writer.WriteLine(" return ((Builder) CreateBuilder().MergeFrom(input, extensionRegistry)).BuildParsed();");
  457. writer.WriteLine("}");
  458. }
  459. /// <summary>
  460. /// Returns whether or not the specified message type has any required fields.
  461. /// If it doesn't, calls to check for initialization can be optimised.
  462. /// TODO(jonskeet): Move this into MessageDescriptor?
  463. /// </summary>
  464. private static bool HasRequiredFields(MessageDescriptor descriptor,
  465. Dictionary<MessageDescriptor, object> alreadySeen)
  466. {
  467. if (alreadySeen.ContainsKey(descriptor))
  468. {
  469. // The type is already in cache. This means that either:
  470. // a. The type has no required fields.
  471. // b. We are in the midst of checking if the type has required fields,
  472. // somewhere up the stack. In this case, we know that if the type
  473. // has any required fields, they'll be found when we return to it,
  474. // and the whole call to HasRequiredFields() will return true.
  475. // Therefore, we don't have to check if this type has required fields
  476. // here.
  477. return false;
  478. }
  479. alreadySeen[descriptor] = descriptor; // Value is irrelevant
  480. // If the type has extensions, an extension with message type could contain
  481. // required fields, so we have to be conservative and assume such an
  482. // extension exists.
  483. if (descriptor.Extensions.Count > 0)
  484. {
  485. return true;
  486. }
  487. foreach (FieldDescriptor field in descriptor.Fields)
  488. {
  489. if (field.IsRequired)
  490. {
  491. return true;
  492. }
  493. // Message or group
  494. if (field.MappedType == MappedType.Message)
  495. {
  496. if (HasRequiredFields(field.MessageType, alreadySeen))
  497. {
  498. return true;
  499. }
  500. }
  501. }
  502. return false;
  503. }
  504. private void GenerateBuilder(TextGenerator writer)
  505. {
  506. writer.WriteLine("private {0} MakeReadOnly() {{", ClassName);
  507. writer.Indent();
  508. foreach (FieldDescriptor field in Descriptor.Fields)
  509. {
  510. CreateFieldGenerator(field).GenerateBuildingCode(writer);
  511. }
  512. writer.WriteLine("return this;");
  513. writer.Outdent();
  514. writer.WriteLine("}");
  515. writer.WriteLine();
  516. writer.WriteLine("public static Builder CreateBuilder() { return new Builder(); }");
  517. writer.WriteLine("public override Builder ToBuilder() { return CreateBuilder(this); }");
  518. writer.WriteLine("public override Builder CreateBuilderForType() { return new Builder(); }");
  519. writer.WriteLine("public static Builder CreateBuilder({0} prototype) {{", ClassName);
  520. writer.WriteLine(" return new Builder(prototype);");
  521. writer.WriteLine("}");
  522. writer.WriteLine();
  523. if (Descriptor.File.CSharpOptions.AddSerializable)
  524. {
  525. writer.WriteLine("[global::System.SerializableAttribute()]");
  526. }
  527. writer.WriteLine("[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]");
  528. WriteGeneratedCodeAttributes(writer);
  529. writer.WriteLine("{0} sealed partial class Builder : pb::{2}Builder{3}<{1}, Builder> {{",
  530. ClassAccessLevel, ClassName,
  531. Descriptor.Proto.ExtensionRangeCount > 0 ? "Extendable" : "Generated", RuntimeSuffix);
  532. writer.Indent();
  533. writer.WriteLine("protected override Builder ThisBuilder {");
  534. writer.WriteLine(" get { return this; }");
  535. writer.WriteLine("}");
  536. GenerateCommonBuilderMethods(writer);
  537. if (OptimizeSpeed)
  538. {
  539. GenerateBuilderParsingMethods(writer);
  540. }
  541. foreach (FieldDescriptor field in Descriptor.Fields)
  542. {
  543. writer.WriteLine();
  544. // No field comment :(
  545. CreateFieldGenerator(field).GenerateBuilderMembers(writer);
  546. }
  547. writer.Outdent();
  548. writer.WriteLine("}");
  549. }
  550. private void GenerateCommonBuilderMethods(TextGenerator writer)
  551. {
  552. //default constructor
  553. writer.WriteLine("public Builder() {");
  554. //Durring static initialization of message, DefaultInstance is expected to return null.
  555. writer.WriteLine(" result = DefaultInstance;");
  556. writer.WriteLine(" resultIsReadOnly = true;");
  557. writer.WriteLine("}");
  558. //clone constructor
  559. writer.WriteLine("internal Builder({0} cloneFrom) {{", ClassName);
  560. writer.WriteLine(" result = cloneFrom;");
  561. writer.WriteLine(" resultIsReadOnly = true;");
  562. writer.WriteLine("}");
  563. writer.WriteLine();
  564. writer.WriteLine("private bool resultIsReadOnly;");
  565. writer.WriteLine("private {0} result;", ClassName);
  566. writer.WriteLine();
  567. writer.WriteLine("private {0} PrepareBuilder() {{", ClassName);
  568. writer.WriteLine(" if (resultIsReadOnly) {");
  569. writer.WriteLine(" {0} original = result;", ClassName);
  570. writer.WriteLine(" result = new {0}();", ClassName);
  571. writer.WriteLine(" resultIsReadOnly = false;");
  572. writer.WriteLine(" MergeFrom(original);");
  573. writer.WriteLine(" }");
  574. writer.WriteLine(" return result;");
  575. writer.WriteLine("}");
  576. writer.WriteLine();
  577. writer.WriteLine("public override bool IsInitialized {");
  578. writer.WriteLine(" get { return result.IsInitialized; }");
  579. writer.WriteLine("}");
  580. writer.WriteLine();
  581. writer.WriteLine("protected override {0} MessageBeingBuilt {{", ClassName);
  582. writer.WriteLine(" get { return PrepareBuilder(); }");
  583. writer.WriteLine("}");
  584. writer.WriteLine();
  585. //Not actually expecting that DefaultInstance would ever be null here; however, we will ensure it does not break
  586. writer.WriteLine("public override Builder Clear() {");
  587. writer.WriteLine(" result = DefaultInstance;", ClassName);
  588. writer.WriteLine(" resultIsReadOnly = true;");
  589. writer.WriteLine(" return this;");
  590. writer.WriteLine("}");
  591. writer.WriteLine();
  592. writer.WriteLine("public override Builder Clone() {");
  593. writer.WriteLine(" if (resultIsReadOnly) {");
  594. writer.WriteLine(" return new Builder(result);");
  595. writer.WriteLine(" } else {");
  596. writer.WriteLine(" return new Builder().MergeFrom(result);");
  597. writer.WriteLine(" }");
  598. writer.WriteLine("}");
  599. writer.WriteLine();
  600. if (!UseLiteRuntime)
  601. {
  602. writer.WriteLine("public override pbd::MessageDescriptor DescriptorForType {");
  603. writer.WriteLine(" get {{ return {0}.Descriptor; }}", FullClassName);
  604. writer.WriteLine("}");
  605. writer.WriteLine();
  606. }
  607. writer.WriteLine("public override {0} DefaultInstanceForType {{", ClassName);
  608. writer.WriteLine(" get {{ return {0}.DefaultInstance; }}", FullClassName);
  609. writer.WriteLine("}");
  610. writer.WriteLine();
  611. writer.WriteLine("public override {0} BuildPartial() {{", ClassName);
  612. writer.Indent();
  613. writer.WriteLine("if (resultIsReadOnly) {");
  614. writer.WriteLine(" return result;");
  615. writer.WriteLine("}");
  616. writer.WriteLine("resultIsReadOnly = true;");
  617. writer.WriteLine("return result.MakeReadOnly();");
  618. writer.Outdent();
  619. writer.WriteLine("}");
  620. writer.WriteLine();
  621. if (OptimizeSpeed)
  622. {
  623. writer.WriteLine("public override Builder MergeFrom(pb::IMessage{0} other) {{", RuntimeSuffix);
  624. writer.WriteLine(" if (other is {0}) {{", ClassName);
  625. writer.WriteLine(" return MergeFrom(({0}) other);", ClassName);
  626. writer.WriteLine(" } else {");
  627. writer.WriteLine(" base.MergeFrom(other);");
  628. writer.WriteLine(" return this;");
  629. writer.WriteLine(" }");
  630. writer.WriteLine("}");
  631. writer.WriteLine();
  632. writer.WriteLine("public override Builder MergeFrom({0} other) {{", ClassName);
  633. // Optimization: If other is the default instance, we know none of its
  634. // fields are set so we can skip the merge.
  635. writer.Indent();
  636. writer.WriteLine("if (other == {0}.DefaultInstance) return this;", FullClassName);
  637. writer.WriteLine("PrepareBuilder();");
  638. foreach (FieldDescriptor field in Descriptor.Fields)
  639. {
  640. CreateFieldGenerator(field).GenerateMergingCode(writer);
  641. }
  642. // if message type has extensions
  643. if (Descriptor.Proto.ExtensionRangeCount > 0)
  644. {
  645. writer.WriteLine(" this.MergeExtensionFields(other);");
  646. }
  647. if (!UseLiteRuntime)
  648. {
  649. writer.WriteLine("this.MergeUnknownFields(other.UnknownFields);");
  650. }
  651. writer.WriteLine("return this;");
  652. writer.Outdent();
  653. writer.WriteLine("}");
  654. writer.WriteLine();
  655. }
  656. }
  657. private void GenerateBuilderParsingMethods(TextGenerator writer)
  658. {
  659. List<FieldDescriptor> sortedFields = new List<FieldDescriptor>(Descriptor.Fields);
  660. sortedFields.Sort((f1, f2) => f1.FieldNumber.CompareTo(f2.FieldNumber));
  661. writer.WriteLine("public override Builder MergeFrom(pb::ICodedInputStream input) {");
  662. writer.WriteLine(" return MergeFrom(input, pb::ExtensionRegistry.Empty);");
  663. writer.WriteLine("}");
  664. writer.WriteLine();
  665. writer.WriteLine(
  666. "public override Builder MergeFrom(pb::ICodedInputStream input, pb::ExtensionRegistry extensionRegistry) {");
  667. writer.Indent();
  668. writer.WriteLine("PrepareBuilder();");
  669. if (!UseLiteRuntime)
  670. {
  671. writer.WriteLine("pb::UnknownFieldSet.Builder unknownFields = null;");
  672. }
  673. writer.WriteLine("uint tag;");
  674. writer.WriteLine("string field_name;");
  675. writer.WriteLine("while (input.ReadTag(out tag, out field_name)) {");
  676. writer.Indent();
  677. writer.WriteLine("if(tag == 0 && field_name != null) {");
  678. writer.Indent();
  679. //if you change from StringComparer.Ordinal, the array sort in FieldNames { get; } must also change
  680. writer.WriteLine(
  681. "int field_ordinal = global::System.Array.BinarySearch(_{0}FieldNames, field_name, global::System.StringComparer.Ordinal);",
  682. NameHelpers.UnderscoresToCamelCase(ClassName));
  683. writer.WriteLine("if(field_ordinal >= 0)");
  684. writer.WriteLine(" tag = _{0}FieldTags[field_ordinal];", NameHelpers.UnderscoresToCamelCase(ClassName));
  685. writer.WriteLine("else {");
  686. if (!UseLiteRuntime)
  687. {
  688. writer.WriteLine(" if (unknownFields == null) {"); // First unknown field - create builder now
  689. writer.WriteLine(" unknownFields = pb::UnknownFieldSet.CreateBuilder(this.UnknownFields);");
  690. writer.WriteLine(" }");
  691. }
  692. writer.WriteLine(" ParseUnknownField(input, {0}extensionRegistry, tag, field_name);",
  693. UseLiteRuntime ? "" : "unknownFields, ");
  694. writer.WriteLine(" continue;");
  695. writer.WriteLine("}");
  696. writer.Outdent();
  697. writer.WriteLine("}");
  698. writer.WriteLine("switch (tag) {");
  699. writer.Indent();
  700. writer.WriteLine("case 0: {"); // 0 signals EOF / limit reached
  701. writer.WriteLine(" throw pb::InvalidProtocolBufferException.InvalidTag();");
  702. writer.WriteLine("}");
  703. writer.WriteLine("default: {");
  704. writer.WriteLine(" if (pb::WireFormat.IsEndGroupTag(tag)) {");
  705. if (!UseLiteRuntime)
  706. {
  707. writer.WriteLine(" if (unknownFields != null) {");
  708. writer.WriteLine(" this.UnknownFields = unknownFields.Build();");
  709. writer.WriteLine(" }");
  710. }
  711. writer.WriteLine(" return this;"); // it's an endgroup tag
  712. writer.WriteLine(" }");
  713. if (!UseLiteRuntime)
  714. {
  715. writer.WriteLine(" if (unknownFields == null) {"); // First unknown field - create builder now
  716. writer.WriteLine(" unknownFields = pb::UnknownFieldSet.CreateBuilder(this.UnknownFields);");
  717. writer.WriteLine(" }");
  718. }
  719. writer.WriteLine(" ParseUnknownField(input, {0}extensionRegistry, tag, field_name);",
  720. UseLiteRuntime ? "" : "unknownFields, ");
  721. writer.WriteLine(" break;");
  722. writer.WriteLine("}");
  723. foreach (FieldDescriptor field in sortedFields)
  724. {
  725. WireFormat.WireType wt = WireFormat.GetWireType(field.FieldType);
  726. uint tag = WireFormat.MakeTag(field.FieldNumber, wt);
  727. if (field.IsRepeated &&
  728. (wt == WireFormat.WireType.Varint || wt == WireFormat.WireType.Fixed32 ||
  729. wt == WireFormat.WireType.Fixed64))
  730. {
  731. writer.WriteLine("case {0}:",
  732. WireFormat.MakeTag(field.FieldNumber, WireFormat.WireType.LengthDelimited));
  733. }
  734. writer.WriteLine("case {0}: {{", tag);
  735. writer.Indent();
  736. CreateFieldGenerator(field).GenerateParsingCode(writer);
  737. writer.WriteLine("break;");
  738. writer.Outdent();
  739. writer.WriteLine("}");
  740. }
  741. writer.Outdent();
  742. writer.WriteLine("}");
  743. writer.Outdent();
  744. writer.WriteLine("}");
  745. writer.WriteLine();
  746. if (!UseLiteRuntime)
  747. {
  748. writer.WriteLine("if (unknownFields != null) {");
  749. writer.WriteLine(" this.UnknownFields = unknownFields.Build();");
  750. writer.WriteLine("}");
  751. }
  752. writer.WriteLine("return this;");
  753. writer.Outdent();
  754. writer.WriteLine("}");
  755. writer.WriteLine();
  756. }
  757. private void GenerateIsInitialized(TextGenerator writer)
  758. {
  759. writer.WriteLine("public override bool IsInitialized {");
  760. writer.Indent();
  761. writer.WriteLine("get {");
  762. writer.Indent();
  763. // Check that all required fields in this message are set.
  764. // TODO(kenton): We can optimize this when we switch to putting all the
  765. // "has" fields into a single bitfield.
  766. foreach (FieldDescriptor field in Descriptor.Fields)
  767. {
  768. if (field.IsRequired)
  769. {
  770. writer.WriteLine("if (!has{0}) return false;", field.CSharpOptions.PropertyName);
  771. }
  772. }
  773. // Now check that all embedded messages are initialized.
  774. foreach (FieldDescriptor field in Descriptor.Fields)
  775. {
  776. if (field.FieldType != FieldType.Message ||
  777. !HasRequiredFields(field.MessageType, new Dictionary<MessageDescriptor, object>()))
  778. {
  779. continue;
  780. }
  781. string propertyName = NameHelpers.UnderscoresToPascalCase(GetFieldName(field));
  782. if (field.IsRepeated)
  783. {
  784. writer.WriteLine("foreach ({0} element in {1}List) {{", GetClassName(field.MessageType),
  785. propertyName);
  786. writer.WriteLine(" if (!element.IsInitialized) return false;");
  787. writer.WriteLine("}");
  788. }
  789. else if (field.IsOptional)
  790. {
  791. writer.WriteLine("if (Has{0}) {{", propertyName);
  792. writer.WriteLine(" if (!{0}.IsInitialized) return false;", propertyName);
  793. writer.WriteLine("}");
  794. }
  795. else
  796. {
  797. writer.WriteLine("if (!{0}.IsInitialized) return false;", propertyName);
  798. }
  799. }
  800. if (Descriptor.Proto.ExtensionRangeCount > 0)
  801. {
  802. writer.WriteLine("if (!ExtensionsAreInitialized) return false;");
  803. }
  804. writer.WriteLine("return true;");
  805. writer.Outdent();
  806. writer.WriteLine("}");
  807. writer.Outdent();
  808. writer.WriteLine("}");
  809. writer.WriteLine();
  810. }
  811. internal void GenerateExtensionRegistrationCode(TextGenerator writer)
  812. {
  813. foreach (FieldDescriptor extension in Descriptor.Extensions)
  814. {
  815. new ExtensionGenerator(extension).GenerateExtensionRegistrationCode(writer);
  816. }
  817. foreach (MessageDescriptor nestedMessage in Descriptor.NestedTypes)
  818. {
  819. new MessageGenerator(nestedMessage).GenerateExtensionRegistrationCode(writer);
  820. }
  821. }
  822. }
  823. }