Extensions.cs 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using ILRuntime.CLR.TypeSystem;
  6. using ILRuntime.CLR.Method;
  7. using ILRuntime.Other;
  8. using Mono.Cecil;
  9. using ILRuntime.Runtime.Intepreter;
  10. namespace ILRuntime.CLR.Utils
  11. {
  12. public delegate TResult Func<T1, T2, T3, T4, T5, TResult>(T1 a1, T2 a2, T3 a3, T4 a4, T5 a5);
  13. public static class Extensions
  14. {
  15. public static List<IType> EmptyParamList = new List<IType>();
  16. public static List<IType> GetParamList(this MethodReference def, ILRuntime.Runtime.Enviorment.AppDomain appdomain, IType contextType, IMethod contextMethod, IType[] genericArguments)
  17. {
  18. if (def.HasParameters)
  19. {
  20. List<IType> param = new List<IType>();
  21. var dt = appdomain.GetType(def.DeclaringType, contextType, contextMethod);
  22. foreach (var i in def.Parameters)
  23. {
  24. IType t = null;
  25. t = appdomain.GetType(i.ParameterType, dt, null);
  26. if (t == null && def.IsGenericInstance)
  27. {
  28. GenericInstanceMethod gim = (GenericInstanceMethod)def;
  29. string name = i.ParameterType.IsByReference ? i.ParameterType.GetElementType().FullName : i.ParameterType.FullName;
  30. for (int j = 0; j < gim.GenericArguments.Count; j++)
  31. {
  32. var gp = gim.ElementMethod.GenericParameters[j];
  33. var ga = gim.GenericArguments[j];
  34. if (name == gp.Name)
  35. {
  36. t = appdomain.GetType(ga, contextType, contextMethod);
  37. if (t == null && genericArguments != null)
  38. t = genericArguments[j];
  39. break;
  40. }
  41. else if (name.Contains(gp.Name))
  42. {
  43. if (name == gp.Name)
  44. {
  45. name = ga.FullName;
  46. }
  47. else if (name == gp.Name + "[]")
  48. {
  49. name = ga.FullName + "[]";
  50. }
  51. else
  52. {
  53. /*name = name.Replace("<" + gp.Name + ">", "<" + ga.FullName + ">");
  54. name = name.Replace("<" + gp.Name + "[", "<" + ga.FullName + "[");
  55. name = name.Replace("<" + gp.Name + ",", "<" + ga.FullName + ",");
  56. name = name.Replace("," + gp.Name + ">", "," + ga.FullName + ">");
  57. name = name.Replace("," + gp.Name + "[", "," + ga.FullName + "[");
  58. name = name.Replace("," + gp.Name + ",", "," + ga.FullName + ",");
  59. name = name.Replace("," + gp.Name + "[", "," + ga.FullName + "[");*/
  60. name = ReplaceGenericArgument(name, gp.Name, ga.FullName);
  61. }
  62. }
  63. }
  64. if (t == null)
  65. t = appdomain.GetType(name);
  66. }
  67. param.Add(t);
  68. }
  69. return param;
  70. }
  71. else
  72. return EmptyParamList;
  73. }
  74. static string ReplaceGenericArgument(string typename, string argumentName, string argumentType)
  75. {
  76. string baseType;
  77. StringBuilder sb = new StringBuilder();
  78. List<string> ga;
  79. bool isArray;
  80. Runtime.Enviorment.AppDomain.ParseGenericType(typename, out baseType, out ga, out isArray);
  81. if (baseType == argumentName)
  82. sb.Append(argumentType);
  83. else
  84. sb.Append(baseType);
  85. if (ga != null && ga.Count > 0)
  86. {
  87. sb.Append("<");
  88. bool isFirst = true;
  89. foreach(var i in ga)
  90. {
  91. if (isFirst)
  92. isFirst = false;
  93. else
  94. sb.Append(",");
  95. sb.Append(ReplaceGenericArgument(i, argumentName, argumentType));
  96. }
  97. sb.Append(">");
  98. }
  99. if (isArray)
  100. sb.Append("[]");
  101. return sb.ToString();
  102. }
  103. [Flags]
  104. public enum TypeFlags
  105. {
  106. Default = 0,
  107. IsPrimitive = 0x1,
  108. IsByRef = 0x2,
  109. IsEnum = 0x4,
  110. IsDelegate = 0x8,
  111. IsValueType = 0x10,
  112. }
  113. private static readonly Dictionary<Type, TypeFlags> typeFlags = new Dictionary<Type, TypeFlags>(new ByReferenceKeyComparer<Type>());
  114. public static bool FastIsEnum(this Type pt)
  115. {
  116. return (pt.GetTypeFlags() & TypeFlags.IsEnum) != 0;
  117. }
  118. public static bool FastIsByRef(this Type pt)
  119. {
  120. return (pt.GetTypeFlags() & TypeFlags.IsByRef) != 0;
  121. }
  122. public static bool FastIsPrimitive(this Type pt)
  123. {
  124. return (pt.GetTypeFlags() & TypeFlags.IsPrimitive) != 0;
  125. }
  126. public static bool FastIsValueType(this Type pt)
  127. {
  128. return (pt.GetTypeFlags() & TypeFlags.IsValueType) != 0;
  129. }
  130. public static TypeFlags GetTypeFlags(this Type pt)
  131. {
  132. var result = TypeFlags.Default;
  133. if (!typeFlags.TryGetValue(pt, out result))
  134. {
  135. if (pt.IsPrimitive)
  136. {
  137. result |= TypeFlags.IsPrimitive;
  138. }
  139. if (pt == typeof(Delegate) || pt.IsSubclassOf(typeof(Delegate)))
  140. {
  141. result |= TypeFlags.IsDelegate;
  142. }
  143. if (pt.IsByRef)
  144. {
  145. result |= TypeFlags.IsByRef;
  146. }
  147. if (pt.IsEnum)
  148. {
  149. result |= TypeFlags.IsEnum;
  150. }
  151. if (pt.IsValueType)
  152. {
  153. result |= TypeFlags.IsValueType;
  154. }
  155. typeFlags[pt] = result;
  156. }
  157. return result;
  158. }
  159. public static object CheckCLRTypes(this Type pt, object obj)
  160. {
  161. if (obj == null)
  162. return null;
  163. var typeFlags = GetTypeFlags(pt);
  164. if ((typeFlags & TypeFlags.IsPrimitive) != 0 && pt != typeof(int))
  165. {
  166. if (pt == typeof(bool) && !(obj is bool))
  167. {
  168. obj = (int)obj == 1;
  169. }
  170. else if (pt == typeof(byte) && !(obj is byte))
  171. obj = (byte)(int)obj;
  172. else if (pt == typeof(short) && !(obj is short))
  173. obj = (short)(int)obj;
  174. else if (pt == typeof(char) && !(obj is char))
  175. obj = (char)(int)obj;
  176. else if (pt == typeof(ushort) && !(obj is ushort))
  177. obj = (ushort)(int)obj;
  178. else if (pt == typeof(uint) && !(obj is uint))
  179. obj = (uint)(int)obj;
  180. else if (pt == typeof(sbyte) && !(obj is sbyte))
  181. obj = (sbyte)(int)obj;
  182. else if (pt == typeof(ulong) && !(obj is ulong))
  183. {
  184. obj = (ulong)(long)obj;
  185. }
  186. }
  187. else if (obj is ILRuntime.Reflection.ILRuntimeWrapperType)
  188. {
  189. obj = ((ILRuntime.Reflection.ILRuntimeWrapperType)obj).RealType;
  190. }
  191. else if ((typeFlags & TypeFlags.IsDelegate) != 0)
  192. {
  193. if (obj is Delegate)
  194. return obj;
  195. if (pt == typeof(Delegate))
  196. return ((IDelegateAdapter)obj).Delegate;
  197. return ((IDelegateAdapter)obj).GetConvertor(pt);
  198. }
  199. else if ((typeFlags & TypeFlags.IsByRef) != 0)
  200. {
  201. return CheckCLRTypes(pt.GetElementType(), obj);
  202. }
  203. else if ((typeFlags & TypeFlags.IsEnum) != 0)
  204. {
  205. return Enum.ToObject(pt, obj);
  206. }
  207. else if (obj is ILTypeInstance)
  208. {
  209. var adapter = obj as IDelegateAdapter;
  210. if (adapter != null && pt != typeof(ILTypeInstance))
  211. {
  212. return adapter.Delegate;
  213. }
  214. if (!(obj is ILEnumTypeInstance))
  215. {
  216. var ins = (ILTypeInstance)obj;
  217. if (ins.IsValueType)
  218. ins = ins.Clone();
  219. return ins.CLRInstance;
  220. }
  221. }
  222. return obj;
  223. }
  224. }
  225. }