Nav apraksta
Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

CecilExtensions.cs 6.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using Mono.Cecil;
  5. namespace zzzUnity.Burst.CodeGen
  6. {
  7. /// <summary>
  8. /// Provides some Cecil Extensions.
  9. /// </summary>
  10. #if BURST_COMPILER_SHARED
  11. public
  12. #else
  13. internal
  14. #endif
  15. static class CecilExtensions
  16. {
  17. public static void BuildAssemblyQualifiedName(this TypeReference type, StringBuilder builder)
  18. {
  19. if (type == null) throw new ArgumentNullException(nameof(type));
  20. TypeReference elementType;
  21. type.BuildReflectionFullName(builder, out elementType, assemblyQualified: true);
  22. if (!(elementType is GenericParameter))
  23. {
  24. // Recover assembly reference from scope first (e.g for types imported), otherwise from Module.Assembly
  25. var assemblyReference = elementType.Scope as AssemblyNameReference ?? elementType.Module?.Assembly?.Name;
  26. if (assemblyReference != null)
  27. {
  28. builder.Append(", ").Append(assemblyReference);
  29. }
  30. }
  31. }
  32. public static void BuildReflectionFullName(this TypeReference type, StringBuilder builder, bool assemblyQualified)
  33. {
  34. BuildReflectionFullName(type, builder, out _, assemblyQualified);
  35. }
  36. public static void BuildReflectionFullName(this TypeReference type, StringBuilder builder, out TypeReference elementType, bool assemblyQualified)
  37. {
  38. if (type == null) throw new ArgumentNullException(nameof(type));
  39. if (type is PointerType pointerType)
  40. {
  41. pointerType.ElementType.BuildReflectionFullName(builder, out elementType, assemblyQualified);
  42. builder.Append("*");
  43. }
  44. else if (type is PinnedType pinnedType)
  45. {
  46. pinnedType.ElementType.BuildReflectionFullName(builder, out elementType, assemblyQualified);
  47. builder.Append(" pinned");
  48. }
  49. else if (type is ByReferenceType byReferenceType)
  50. {
  51. byReferenceType.ElementType.BuildReflectionFullName(builder, out elementType, assemblyQualified);
  52. builder.Append("&");
  53. }
  54. else if (type is ArrayType arrayType)
  55. {
  56. arrayType.ElementType.BuildReflectionFullName(builder, out elementType, assemblyQualified);
  57. builder.Append("[]");
  58. }
  59. else if (type is GenericParameter genericParameter)
  60. {
  61. elementType = type;
  62. builder.Append(genericParameter.Type == GenericParameterType.Method ? "!!" : "!");
  63. builder.Append(genericParameter.Position);
  64. }
  65. else if (type is FunctionPointerType functionPointerType)
  66. {
  67. elementType = type;
  68. builder.Append("delegate* ");
  69. builder.Append(functionPointerType.CallingConvention switch
  70. {
  71. MethodCallingConvention.Default => "managed",
  72. MethodCallingConvention.Unmanaged => "unmanaged",
  73. MethodCallingConvention.C => "unmanaged[Cdecl]",
  74. MethodCallingConvention.FastCall => "unmanaged[Fastcall]",
  75. MethodCallingConvention.ThisCall => "unmanaged[Thiscall]",
  76. MethodCallingConvention.StdCall => "unmanaged[Stdcall]",
  77. MethodCallingConvention.Generic => "generic",
  78. MethodCallingConvention.VarArg => "vararg",
  79. var conv => $"<unknown calling conv: {(int)conv}>",
  80. });
  81. builder.Append("<");
  82. for (var i = 0; i < functionPointerType.Parameters.Count; i++)
  83. {
  84. var param = functionPointerType.Parameters[i];
  85. param.ParameterType.BuildAssemblyQualifiedName(builder);
  86. builder.Append(", ");
  87. }
  88. functionPointerType.MethodReturnType.ReturnType.BuildAssemblyQualifiedName(builder);
  89. builder.Append(">");
  90. }
  91. else
  92. {
  93. elementType = type;
  94. var types = new List<TypeReference>();
  95. var declaringType = type;
  96. while (declaringType != null)
  97. {
  98. types.Add(declaringType);
  99. declaringType = declaringType.DeclaringType;
  100. }
  101. var baseType = types[types.Count - 1];
  102. if (!string.IsNullOrEmpty(baseType.Namespace))
  103. {
  104. builder.Append(baseType.Namespace);
  105. builder.Append(".");
  106. }
  107. builder.Append(baseType.Name);
  108. for (int i = types.Count - 2; i >= 0; i--)
  109. {
  110. var nestedType = types[i];
  111. builder.Append("+").Append(nestedType.Name);
  112. }
  113. if (elementType is GenericInstanceType genericInstanceType && genericInstanceType.HasGenericArguments)
  114. {
  115. builder.Append("[");
  116. for (var i = 0; i < genericInstanceType.GenericArguments.Count; i++)
  117. {
  118. var genericArgument = genericInstanceType.GenericArguments[i];
  119. if (i > 0)
  120. {
  121. builder.Append(",");
  122. }
  123. if (assemblyQualified)
  124. {
  125. builder.Append("[");
  126. genericArgument.BuildAssemblyQualifiedName(builder);
  127. builder.Append("]");
  128. }
  129. else
  130. {
  131. genericArgument.BuildReflectionFullName(builder, out var _, assemblyQualified: true);
  132. }
  133. }
  134. builder.Append("]");
  135. }
  136. }
  137. }
  138. }
  139. }