1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192 |
- using System;
- using System.Collections;
- using System.Reflection;
- using UnityEngine;
- using GFGGame.Launcher;
- using System.Collections.Generic;
- using HybridCLR;
- using System.Linq;
- namespace GFGGame
- {
- public class HotUpdateCodeLoader : SingletonMonoBase<HotUpdateCodeLoader>
- {
- public static List<string> AOTMetaAssemblyNames { get; } = new List<string>()
- {
- "mscorlib.dll",
- "System.dll",
- "System.Core.dll",
- "Game.Launcher.dll",
- "ThirdParty.dll"
- };
- public Type[] allTypes;
- public Type[] GetTypes()
- {
- return this.allTypes;
- }
- public void StartLoad()
- {
- #if !UNITY_EDITOR
- LoadMetadataForAOTAssemblies();
- StartCoroutine(StartLoadAssemblyHotfix());
- #else
- var gameAss = AppDomain.CurrentDomain.GetAssemblies().First(assembly => assembly.GetName().Name == "Game.HotUpdate");
- LoadAssemblyJustInTime(gameAss);
- #endif
- }
- IEnumerator StartLoadAssemblyHotfix()
- {
- LogServerHelperHttp.SendNodeLog((int)LogNode.StartLoadGameDll);
- //ET.Log.Debug("ILRuntimeLauncher StartLoadAssemblyHotfix");
- yield return new WaitForSeconds(0.1f);
- var dllPath = $"{LauncherConfig.DllDirHotfix}Game.HotUpdate.dll.bytes";
- var asset = GFGAsset.Load<TextAsset>(dllPath);
- byte[] assBytes = asset.bytes;
- var assembly = Assembly.Load(assBytes);
- LoadAssemblyJustInTime(assembly);
- GFGAsset.Release(dllPath);
- }
- void LoadAssemblyJustInTime(Assembly assembly)
- {
- //mono模式
- this.allTypes = assembly.GetTypes();
- System.Type type = assembly.GetType("GFGGame.HotUpdate.HotUpdateEntry");
- var method = type.GetMethod("Start");
- var startDel = (Action)System.Delegate.CreateDelegate(typeof(Action), null, method);
- startDel();
- }
- /// <summary>
- /// 为aot assembly加载原始metadata, 这个代码放aot或者热更新都行。
- /// 一旦加载后,如果AOT泛型函数对应native实现不存在,则自动替换为解释模式执行
- /// </summary>
- private void LoadMetadataForAOTAssemblies()
- {
- // 可以加载任意aot assembly的对应的dll。但要求dll必须与unity build过程中生成的裁剪后的dll一致,而不能直接使用原始dll。
- // 我们在BuildProcessors里添加了处理代码,这些裁剪后的dll在打包时自动被复制到 {项目目录}/HybridCLRData/AssembliesPostIl2CppStrip/{Target} 目录。
- /// 注意,补充元数据是给AOT dll补充元数据,而不是给热更新dll补充元数据。
- /// 热更新dll不缺元数据,不需要补充,如果调用LoadMetadataForAOTAssembly会返回错误
- ///
- HomologousImageMode mode = HomologousImageMode.SuperSet;
- foreach (var aotDllName in AOTMetaAssemblyNames)
- {
- var dllPath = $"{LauncherConfig.DllDirAOT}{aotDllName}.bytes";
- var asset = GFGAsset.Load<TextAsset>(dllPath);
- byte[] dllBytes = asset.bytes;
- // 加载assembly对应的dll,会自动为它hook。一旦aot泛型函数的native函数不存在,用解释器版本代码
- LoadImageErrorCode err = RuntimeApi.LoadMetadataForAOTAssembly(dllBytes, mode);
- Debug.Log($"LoadMetadataForAOTAssembly:{aotDllName}. mode:{mode} ret:{err}");
- }
- }
- }
- }
|