146 lines
4.3 KiB
C#
Raw Normal View History

2021-06-18 21:54:32 +01:00
using Harmony;
using OWML.Common;
using OWML.Utils;
using QSB.Utility;
using System;
using System.Collections.Generic;
using System.Linq;
2021-06-18 21:54:32 +01:00
using System.Reflection;
namespace QSB.Patches
2020-11-03 21:11:10 +00:00
{
2020-12-02 21:23:01 +00:00
public abstract class QSBPatch
{
public abstract QSBPatchTypes Type { get; }
public abstract void DoPatches();
2021-06-18 21:54:32 +01:00
public void DoUnpatches()
{
foreach (var item in _patchedMethods)
{
2021-08-08 20:04:55 +01:00
//DebugLog.DebugWrite($"[Unpatch] {item.DeclaringType}.{item.Name}", MessageType.Info);
Unpatch(item);
2021-06-18 21:54:32 +01:00
}
2021-06-18 22:38:32 +01:00
2021-06-18 21:54:32 +01:00
_patchedMethods.Clear();
}
private List<MethodInfo> _patchedMethods = new List<MethodInfo>();
public void Empty(string patchName)
{
2021-08-08 20:04:55 +01:00
//DebugLog.DebugWrite($"[Empty] {patchName}", MessageType.Success);
2021-06-18 21:54:32 +01:00
var method = GetMethodInfo(patchName);
QSBCore.Helper.HarmonyHelper.EmptyMethod(method);
_patchedMethods.Add(method);
2021-06-18 21:54:32 +01:00
}
2021-08-21 19:52:35 +01:00
public void Prefix(string patchName, params Type[] args)
=> DoPrefixPostfix(true, patchName, args);
2021-06-18 21:54:32 +01:00
2021-08-21 19:52:35 +01:00
public void Postfix(string patchName, params Type[] args)
=> DoPrefixPostfix(false, patchName, args);
2021-06-18 21:54:32 +01:00
2021-08-21 19:52:35 +01:00
private void DoPrefixPostfix(bool isPrefix, string patchName, params Type[] args)
2021-06-18 21:54:32 +01:00
{
2021-08-21 19:52:35 +01:00
var method = GetMethodInfo(patchName, args);
2021-06-18 21:54:32 +01:00
if (method != null)
{
if (isPrefix)
{
QSBCore.Helper.HarmonyHelper.AddPrefix(method, GetType(), patchName);
}
else
{
QSBCore.Helper.HarmonyHelper.AddPostfix(method, GetType(), patchName);
}
2021-06-18 22:38:32 +01:00
2021-06-18 21:54:32 +01:00
_patchedMethods.Add(method);
}
2021-08-08 20:04:55 +01:00
//DebugLog.DebugWrite($"{(isPrefix ? "[Prefix]" : "[Postfix]")} {patchName}", method == null ? MessageType.Error : MessageType.Success);
2021-06-18 21:54:32 +01:00
}
2021-08-21 19:52:35 +01:00
private MethodInfo GetMethodInfo(string patchName, params Type[] args)
2021-06-18 21:54:32 +01:00
{
var splitName = patchName.Split('_');
var typeName = splitName[0];
var methodName = splitName[1];
var type = GetFirstTypeByName(typeName);
if (type == null)
{
DebugLog.ToConsole($"Error - Couldn't find type for patch name {patchName}!", MessageType.Error);
2021-06-18 21:54:32 +01:00
return null;
}
2021-08-21 19:52:35 +01:00
var allMethodsOfName = type.GetMethods(BindingFlags.Instance | BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.FlattenHierarchy).Where(x => x.Name == methodName);
if (allMethodsOfName.Count() == 0)
2021-06-18 21:54:32 +01:00
{
2021-08-21 19:52:35 +01:00
DebugLog.ToConsole($"Error - Could not find method {methodName} in type {typeName}.", MessageType.Error);
2021-06-18 21:54:32 +01:00
return null;
}
2021-08-21 19:52:35 +01:00
if (allMethodsOfName.Count() == 1)
{
return allMethodsOfName.First();
}
DebugLog.DebugWrite($"More than one method found with name {methodName} in type {typeName}");
foreach (var method in allMethodsOfName)
{
DebugLog.DebugWrite($"checking {method.Name}");
var paramList = method.GetParameters().Select(x => x.ParameterType);
if (Enumerable.SequenceEqual(args, paramList))
{
DebugLog.DebugWrite($"match!");
return method;
}
}
DebugLog.DebugWrite($"nothing found");
DebugLog.ToConsole($"Error - Could not find method {methodName} in type {typeName} with parameter list of {string.Join(", ", args.Select(x => x.FullName).ToArray())}", MessageType.Error);
foreach (var method in allMethodsOfName)
{
var paramList = method.GetParameters().Select(x => x.ParameterType);
DebugLog.ToConsole($"- Found {method.Name}, but with params {string.Join(", ", paramList.Select(x => x.FullName).ToArray())}", MessageType.Error);
}
return null;
2021-06-18 21:54:32 +01:00
}
private Type GetFirstTypeByName(string typeName)
{
var a = typeof(OWRigidbody).Assembly;
var assemblyTypes = a.GetTypes();
for (var j = 0; j < assemblyTypes.Length; j++)
2021-06-18 21:54:32 +01:00
{
if (assemblyTypes[j].Name == typeName)
{
return assemblyTypes[j];
}
}
2021-06-18 22:38:32 +01:00
2021-06-18 21:54:32 +01:00
return null;
}
private void Unpatch(MethodInfo method)
{
var dictionary = typeof(HarmonySharedState).Invoke<Dictionary<MethodBase, byte[]>>("GetState", new object[0]);
var methodBase = dictionary.Keys.First(m =>
2021-07-04 22:34:38 +01:00
m.DeclaringType == method.DeclaringType
&& m.Name == method.Name);
var patchInfo = PatchInfoSerialization.Deserialize(dictionary.GetValueSafe(methodBase));
patchInfo.RemovePostfix(QSBCore.Helper.Manifest.UniqueName);
patchInfo.RemovePrefix(QSBCore.Helper.Manifest.UniqueName);
patchInfo.RemoveTranspiler(QSBCore.Helper.Manifest.UniqueName);
PatchFunctions.UpdateWrapper(methodBase, patchInfo, QSBCore.Helper.Manifest.UniqueName);
dictionary[methodBase] = patchInfo.Serialize();
}
2020-12-02 21:23:01 +00:00
}
2020-12-03 08:28:05 +00:00
}