AssemblyCacheBase.cs 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. using dnlib.DotNet;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Text;
  7. using System.Threading.Tasks;
  8. namespace HybridCLR.Editor.Meta
  9. {
  10. public abstract class AssemblyCacheBase
  11. {
  12. private readonly IAssemblyResolver _assemblyPathResolver;
  13. private readonly ModuleContext _modCtx;
  14. private readonly AssemblyResolver _asmResolver;
  15. private bool _loadedNetstandard;
  16. public ModuleContext ModCtx => _modCtx;
  17. public Dictionary<string, ModuleDefMD> LoadedModules { get; } = new Dictionary<string, ModuleDefMD>();
  18. private readonly List<ModuleDefMD> _loadedModulesIncludeNetstandard = new List<ModuleDefMD>();
  19. protected AssemblyCacheBase(IAssemblyResolver assemblyResolver)
  20. {
  21. _assemblyPathResolver = assemblyResolver;
  22. _modCtx = ModuleDef.CreateModuleContext();
  23. _asmResolver = (AssemblyResolver)_modCtx.AssemblyResolver;
  24. _asmResolver.EnableTypeDefCache = true;
  25. _asmResolver.UseGAC = false;
  26. }
  27. public ModuleDefMD TryLoadModule(string moduleName, bool loadReferenceAssemblies = true)
  28. {
  29. string dllPath = _assemblyPathResolver.ResolveAssembly(moduleName, false);
  30. if (string.IsNullOrEmpty(dllPath))
  31. {
  32. return null;
  33. }
  34. return LoadModule(moduleName, loadReferenceAssemblies);
  35. }
  36. public ModuleDefMD LoadModule(string moduleName, bool loadReferenceAssemblies = true)
  37. {
  38. // Debug.Log($"load module:{moduleName}");
  39. if (LoadedModules.TryGetValue(moduleName, out var mod))
  40. {
  41. return mod;
  42. }
  43. if (moduleName == "netstandard")
  44. {
  45. if (!_loadedNetstandard)
  46. {
  47. LoadNetStandard();
  48. }
  49. return null;
  50. }
  51. mod = DoLoadModule(_assemblyPathResolver.ResolveAssembly(moduleName, true));
  52. LoadedModules.Add(moduleName, mod);
  53. if (loadReferenceAssemblies)
  54. {
  55. foreach (var refAsm in mod.GetAssemblyRefs())
  56. {
  57. LoadModule(refAsm.Name);
  58. }
  59. }
  60. return mod;
  61. }
  62. private void LoadNetStandard()
  63. {
  64. string netstandardDllPath = _assemblyPathResolver.ResolveAssembly("netstandard", false);
  65. if (!string.IsNullOrEmpty(netstandardDllPath))
  66. {
  67. DoLoadModule(netstandardDllPath);
  68. }
  69. else
  70. {
  71. DoLoadModule(MetaUtil.ResolveNetStandardAssemblyPath("netstandard2.0"));
  72. DoLoadModule(MetaUtil.ResolveNetStandardAssemblyPath("netstandard2.1"));
  73. }
  74. _loadedNetstandard = true;
  75. }
  76. private ModuleDefMD DoLoadModule(string dllPath)
  77. {
  78. //Debug.Log($"do load module:{dllPath}");
  79. ModuleDefMD mod = ModuleDefMD.Load(File.ReadAllBytes(dllPath), _modCtx);
  80. mod.EnableTypeDefFindCache = true;
  81. _asmResolver.AddToCache(mod);
  82. _loadedModulesIncludeNetstandard.Add(mod);
  83. return mod;
  84. }
  85. }
  86. }