HotUpdateCodeLoader.cs 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. using System;
  2. using System.Collections;
  3. using System.Reflection;
  4. using UnityEngine;
  5. using GFGGame.Launcher;
  6. using System.Collections.Generic;
  7. using HybridCLR;
  8. using System.Linq;
  9. namespace GFGGame
  10. {
  11. public class HotUpdateCodeLoader : SingletonMonoBase<HotUpdateCodeLoader>
  12. {
  13. public static List<string> AOTMetaAssemblyNames { get; } = new List<string>()
  14. {
  15. "mscorlib.dll",
  16. "System.dll",
  17. "System.Core.dll",
  18. };
  19. public Type[] allTypes;
  20. public Type[] GetTypes()
  21. {
  22. return this.allTypes;
  23. }
  24. public void StartLoad()
  25. {
  26. #if !UNITY_EDITOR
  27. LoadMetadataForAOTAssemblies();
  28. StartCoroutine(StartLoadAssemblyHotfix());
  29. #else
  30. var gameAss = AppDomain.CurrentDomain.GetAssemblies().First(assembly => assembly.GetName().Name == "Game.HotUpdate");
  31. LoadAssemblyJustInTime(gameAss);
  32. #endif
  33. }
  34. IEnumerator StartLoadAssemblyHotfix()
  35. {
  36. LogServerHelperHttp.SendNodeLog((int)LogNode.StartLoadGameDll);
  37. //ET.Log.Debug("ILRuntimeLauncher StartLoadAssemblyHotfix");
  38. yield return new WaitForSeconds(0.1f);
  39. var dllPath = $"{LauncherConfig.DllDirHotfix}Game.HotUpdate.dll.bytes";
  40. var asset = GFGAsset.Load<TextAsset>(dllPath);
  41. byte[] assBytes = asset.bytes;
  42. var assembly = Assembly.Load(assBytes);
  43. LoadAssemblyJustInTime(assembly);
  44. GFGAsset.Release(dllPath);
  45. }
  46. void LoadAssemblyJustInTime(Assembly assembly)
  47. {
  48. //mono模式
  49. this.allTypes = assembly.GetTypes();
  50. System.Type type = assembly.GetType("GFGGame.HotUpdate.HotUpdateEntry");
  51. var method = type.GetMethod("Start");
  52. var startDel = (Action)System.Delegate.CreateDelegate(typeof(Action), null, method);
  53. startDel();
  54. }
  55. /// <summary>
  56. /// 为aot assembly加载原始metadata, 这个代码放aot或者热更新都行。
  57. /// 一旦加载后,如果AOT泛型函数对应native实现不存在,则自动替换为解释模式执行
  58. /// </summary>
  59. private void LoadMetadataForAOTAssemblies()
  60. {
  61. // 可以加载任意aot assembly的对应的dll。但要求dll必须与unity build过程中生成的裁剪后的dll一致,而不能直接使用原始dll。
  62. // 我们在BuildProcessors里添加了处理代码,这些裁剪后的dll在打包时自动被复制到 {项目目录}/HybridCLRData/AssembliesPostIl2CppStrip/{Target} 目录。
  63. /// 注意,补充元数据是给AOT dll补充元数据,而不是给热更新dll补充元数据。
  64. /// 热更新dll不缺元数据,不需要补充,如果调用LoadMetadataForAOTAssembly会返回错误
  65. ///
  66. HomologousImageMode mode = HomologousImageMode.SuperSet;
  67. foreach (var aotDllName in AOTMetaAssemblyNames)
  68. {
  69. var dllPath = $"{LauncherConfig.DllDirAOT}{aotDllName}.bytes";
  70. var asset = GFGAsset.Load<TextAsset>(dllPath);
  71. byte[] dllBytes = asset.bytes;
  72. // 加载assembly对应的dll,会自动为它hook。一旦aot泛型函数的native函数不存在,用解释器版本代码
  73. LoadImageErrorCode err = RuntimeApi.LoadMetadataForAOTAssembly(dllBytes, mode);
  74. Debug.Log($"LoadMetadataForAOTAssembly:{aotDllName}. mode:{mode} ret:{err}");
  75. }
  76. }
  77. }
  78. }