SubItemSerializer.cs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. #if !NO_RUNTIME
  2. using System;
  3. using ProtoBuf.Meta;
  4. #if FEAT_COMPILER
  5. using System.Reflection.Emit;
  6. #endif
  7. namespace ProtoBuf.Serializers
  8. {
  9. sealed class SubItemSerializer : IProtoTypeSerializer
  10. {
  11. bool IProtoTypeSerializer.HasCallbacks(TypeModel.CallbackType callbackType)
  12. {
  13. return ((IProtoTypeSerializer)proxy.Serializer).HasCallbacks(callbackType);
  14. }
  15. bool IProtoTypeSerializer.CanCreateInstance()
  16. {
  17. return ((IProtoTypeSerializer)proxy.Serializer).CanCreateInstance();
  18. }
  19. #if FEAT_COMPILER
  20. void IProtoTypeSerializer.EmitCallback(Compiler.CompilerContext ctx, Compiler.Local valueFrom, TypeModel.CallbackType callbackType)
  21. {
  22. ((IProtoTypeSerializer)proxy.Serializer).EmitCallback(ctx, valueFrom, callbackType);
  23. }
  24. void IProtoTypeSerializer.EmitCreateInstance(Compiler.CompilerContext ctx)
  25. {
  26. ((IProtoTypeSerializer)proxy.Serializer).EmitCreateInstance(ctx);
  27. }
  28. #endif
  29. void IProtoTypeSerializer.Callback(object value, TypeModel.CallbackType callbackType, SerializationContext context)
  30. {
  31. ((IProtoTypeSerializer)proxy.Serializer).Callback(value, callbackType, context);
  32. }
  33. object IProtoTypeSerializer.CreateInstance(ProtoReader source)
  34. {
  35. return ((IProtoTypeSerializer)proxy.Serializer).CreateInstance(source);
  36. }
  37. private readonly int key;
  38. private readonly Type type;
  39. private readonly ISerializerProxy proxy;
  40. private readonly bool recursionCheck;
  41. public SubItemSerializer(Type type, int key, ISerializerProxy proxy, bool recursionCheck)
  42. {
  43. this.type = type ?? throw new ArgumentNullException(nameof(type));
  44. this.proxy = proxy ?? throw new ArgumentNullException(nameof(proxy));
  45. this.key = key;
  46. this.recursionCheck = recursionCheck;
  47. }
  48. Type IProtoSerializer.ExpectedType => type;
  49. bool IProtoSerializer.RequiresOldValue => true;
  50. bool IProtoSerializer.ReturnsValue => true;
  51. void IProtoSerializer.Write(object value, ProtoWriter dest)
  52. {
  53. if (recursionCheck)
  54. {
  55. ProtoWriter.WriteObject(value, key, dest);
  56. }
  57. else
  58. {
  59. ProtoWriter.WriteRecursionSafeObject(value, key, dest);
  60. }
  61. }
  62. object IProtoSerializer.Read(object value, ProtoReader source)
  63. {
  64. return ProtoReader.ReadObject(value, key, source);
  65. }
  66. #if FEAT_COMPILER
  67. bool EmitDedicatedMethod(Compiler.CompilerContext ctx, Compiler.Local valueFrom, bool read)
  68. {
  69. MethodBuilder method = ctx.GetDedicatedMethod(key, read);
  70. if (method == null) return false;
  71. using (Compiler.Local token = new ProtoBuf.Compiler.Local(ctx, ctx.MapType(typeof(SubItemToken))))
  72. {
  73. Type rwType = ctx.MapType(read ? typeof(ProtoReader) : typeof(ProtoWriter));
  74. ctx.LoadValue(valueFrom);
  75. if (!read) // write requires the object for StartSubItem; read doesn't
  76. { // (if recursion-check is disabled [subtypes] then null is fine too)
  77. if (Helpers.IsValueType(type) || !recursionCheck) { ctx.LoadNullRef(); }
  78. else { ctx.CopyValue(); }
  79. }
  80. ctx.LoadReaderWriter();
  81. ctx.EmitCall(Helpers.GetStaticMethod(rwType, "StartSubItem",
  82. read ? new Type[] { rwType } : new Type[] { ctx.MapType(typeof(object)), rwType }));
  83. ctx.StoreValue(token);
  84. // note: value already on the stack
  85. ctx.LoadReaderWriter();
  86. ctx.EmitCall(method);
  87. // handle inheritance (we will be calling the *base* version of things,
  88. // but we expect Read to return the "type" type)
  89. if (read && type != method.ReturnType) ctx.Cast(this.type);
  90. ctx.LoadValue(token);
  91. ctx.LoadReaderWriter();
  92. ctx.EmitCall(Helpers.GetStaticMethod(rwType, "EndSubItem", new Type[] { ctx.MapType(typeof(SubItemToken)), rwType }));
  93. }
  94. return true;
  95. }
  96. void IProtoSerializer.EmitWrite(Compiler.CompilerContext ctx, Compiler.Local valueFrom)
  97. {
  98. if (!EmitDedicatedMethod(ctx, valueFrom, false))
  99. {
  100. ctx.LoadValue(valueFrom);
  101. if (Helpers.IsValueType(type)) ctx.CastToObject(type);
  102. ctx.LoadValue(ctx.MapMetaKeyToCompiledKey(key)); // re-map for formality, but would expect identical, else dedicated method
  103. ctx.LoadReaderWriter();
  104. ctx.EmitCall(Helpers.GetStaticMethod(ctx.MapType(typeof(ProtoWriter)), recursionCheck ? "WriteObject" : "WriteRecursionSafeObject", new Type[] { ctx.MapType(typeof(object)), ctx.MapType(typeof(int)), ctx.MapType(typeof(ProtoWriter)) }));
  105. }
  106. }
  107. void IProtoSerializer.EmitRead(Compiler.CompilerContext ctx, Compiler.Local valueFrom)
  108. {
  109. if (!EmitDedicatedMethod(ctx, valueFrom, true))
  110. {
  111. ctx.LoadValue(valueFrom);
  112. if (Helpers.IsValueType(type)) ctx.CastToObject(type);
  113. ctx.LoadValue(ctx.MapMetaKeyToCompiledKey(key)); // re-map for formality, but would expect identical, else dedicated method
  114. ctx.LoadReaderWriter();
  115. ctx.EmitCall(Helpers.GetStaticMethod(ctx.MapType(typeof(ProtoReader)), "ReadObject"));
  116. ctx.CastFromObject(type);
  117. }
  118. }
  119. #endif
  120. }
  121. }
  122. #endif