2021-05-03 19:13:03 +00:00
|
|
|
|
using Mono.Cecil;
|
|
|
|
|
using Mono.Cecil.Cil;
|
|
|
|
|
using Mono.Cecil.Mdb;
|
|
|
|
|
using Mono.Cecil.Pdb;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Reflection;
|
|
|
|
|
|
|
|
|
|
namespace QNetWeaver
|
|
|
|
|
{
|
|
|
|
|
internal class Helpers
|
|
|
|
|
{
|
|
|
|
|
public static string UnityEngineDLLDirectoryName()
|
|
|
|
|
{
|
|
|
|
|
var directoryName = Path.GetDirectoryName(Assembly.GetExecutingAssembly().CodeBase);
|
|
|
|
|
return (directoryName == null) ? null : directoryName.Replace("file:\\", "");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static ISymbolReaderProvider GetSymbolReaderProvider(string inputFile)
|
|
|
|
|
{
|
|
|
|
|
var text = inputFile.Substring(0, inputFile.Length - 4);
|
|
|
|
|
ISymbolReaderProvider result;
|
|
|
|
|
if (File.Exists(text + ".pdb"))
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("Symbols will be read from " + text + ".pdb");
|
|
|
|
|
result = new PdbReaderProvider();
|
|
|
|
|
}
|
|
|
|
|
else if (File.Exists(text + ".dll.mdb"))
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("Symbols will be read from " + text + ".dll.mdb");
|
|
|
|
|
result = new MdbReaderProvider();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("No symbols for " + inputFile);
|
|
|
|
|
result = null;
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static bool InheritsFromSyncList(TypeReference typeRef)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
if (typeRef.IsValueType)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 21:35:18 +00:00
|
|
|
|
foreach (var typeReference in ResolveInheritanceHierarchy(typeRef))
|
2021-05-03 19:13:03 +00:00
|
|
|
|
{
|
|
|
|
|
if (typeReference.IsGenericInstance)
|
|
|
|
|
{
|
|
|
|
|
var typeDefinition = typeReference.Resolve();
|
|
|
|
|
if (typeDefinition.HasGenericParameters && typeDefinition.FullName == Weaver.SyncListType.FullName)
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static IEnumerable<TypeReference> ResolveInheritanceHierarchy(TypeReference type)
|
|
|
|
|
{
|
|
|
|
|
if (type.IsValueType)
|
|
|
|
|
{
|
|
|
|
|
yield return type;
|
|
|
|
|
yield return Weaver.valueTypeType;
|
|
|
|
|
yield return Weaver.objectType;
|
|
|
|
|
yield break;
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
while (type != null && type.FullName != Weaver.objectType.FullName)
|
|
|
|
|
{
|
|
|
|
|
yield return type;
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var typeDefinition = type.Resolve();
|
|
|
|
|
if (typeDefinition == null)
|
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
type = typeDefinition.BaseType;
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
yield return Weaver.objectType;
|
|
|
|
|
yield break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static string DestinationFileFor(string outputDir, string assemblyPath)
|
|
|
|
|
{
|
|
|
|
|
var fileName = Path.GetFileName(assemblyPath);
|
|
|
|
|
return Path.Combine(outputDir, fileName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static string PrettyPrintType(TypeReference type)
|
|
|
|
|
{
|
|
|
|
|
string result;
|
|
|
|
|
if (type.IsGenericInstance)
|
|
|
|
|
{
|
|
|
|
|
var genericInstanceType = (GenericInstanceType)type;
|
|
|
|
|
var text = genericInstanceType.Name.Substring(0, genericInstanceType.Name.Length - 2);
|
|
|
|
|
var text2 = "<";
|
|
|
|
|
var text3 = ", ";
|
|
|
|
|
IEnumerable<TypeReference> genericArguments = genericInstanceType.GenericArguments;
|
2021-05-03 21:35:18 +00:00
|
|
|
|
result = text + text2 + string.Join(text3, Enumerable.ToArray<string>(Enumerable.Select<TypeReference, string>(genericArguments, new Func<TypeReference, string>(PrettyPrintType)))) + ">";
|
2021-05-03 19:13:03 +00:00
|
|
|
|
}
|
|
|
|
|
else if (type.HasGenericParameters)
|
|
|
|
|
{
|
|
|
|
|
result = type.Name.Substring(0, type.Name.Length - 2) + "<" + string.Join(", ", Enumerable.ToArray<string>(Enumerable.Select<GenericParameter, string>(type.GenericParameters, (GenericParameter x) => x.Name))) + ">";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
result = type.Name;
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static ReaderParameters ReaderParameters(string assemblyPath, IEnumerable<string> extraPaths, IAssemblyResolver assemblyResolver, string unityEngineDLLPath, string unityUNetDLLPath)
|
|
|
|
|
{
|
|
|
|
|
var readerParameters = new ReaderParameters();
|
|
|
|
|
if (assemblyResolver == null)
|
|
|
|
|
{
|
|
|
|
|
assemblyResolver = new DefaultAssemblyResolver();
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
var addSearchDirectoryHelper = new Helpers.AddSearchDirectoryHelper(assemblyResolver);
|
|
|
|
|
addSearchDirectoryHelper.AddSearchDirectory(Path.GetDirectoryName(assemblyPath));
|
2021-05-03 21:35:18 +00:00
|
|
|
|
addSearchDirectoryHelper.AddSearchDirectory(UnityEngineDLLDirectoryName());
|
2021-05-03 19:13:03 +00:00
|
|
|
|
addSearchDirectoryHelper.AddSearchDirectory(Path.GetDirectoryName(unityEngineDLLPath));
|
|
|
|
|
addSearchDirectoryHelper.AddSearchDirectory(Path.GetDirectoryName(unityUNetDLLPath));
|
|
|
|
|
if (extraPaths != null)
|
|
|
|
|
{
|
|
|
|
|
foreach (var directory in extraPaths)
|
|
|
|
|
{
|
|
|
|
|
addSearchDirectoryHelper.AddSearchDirectory(directory);
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
readerParameters.AssemblyResolver = assemblyResolver;
|
2021-05-03 21:35:18 +00:00
|
|
|
|
readerParameters.SymbolReaderProvider = GetSymbolReaderProvider(assemblyPath);
|
2021-05-03 19:13:03 +00:00
|
|
|
|
return readerParameters;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static WriterParameters GetWriterParameters(ReaderParameters readParams)
|
|
|
|
|
{
|
|
|
|
|
var writerParameters = new WriterParameters();
|
|
|
|
|
if (readParams.SymbolReaderProvider is PdbReaderProvider)
|
|
|
|
|
{
|
|
|
|
|
writerParameters.SymbolWriterProvider = new PdbWriterProvider();
|
|
|
|
|
}
|
|
|
|
|
else if (readParams.SymbolReaderProvider is MdbReaderProvider)
|
|
|
|
|
{
|
|
|
|
|
writerParameters.SymbolWriterProvider = new MdbWriterProvider();
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
return writerParameters;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static TypeReference MakeGenericType(TypeReference self, params TypeReference[] arguments)
|
|
|
|
|
{
|
|
|
|
|
if (self.GenericParameters.Count != arguments.Length)
|
|
|
|
|
{
|
|
|
|
|
throw new ArgumentException();
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
var genericInstanceType = new GenericInstanceType(self);
|
|
|
|
|
foreach (var item in arguments)
|
|
|
|
|
{
|
|
|
|
|
genericInstanceType.GenericArguments.Add(item);
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
return genericInstanceType;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static MethodReference MakeHostInstanceGeneric(MethodReference self, params TypeReference[] arguments)
|
|
|
|
|
{
|
2021-05-03 21:35:18 +00:00
|
|
|
|
var methodReference = new MethodReference(self.Name, self.ReturnType, MakeGenericType(self.DeclaringType, arguments))
|
2021-05-03 19:13:03 +00:00
|
|
|
|
{
|
|
|
|
|
HasThis = self.HasThis,
|
|
|
|
|
ExplicitThis = self.ExplicitThis,
|
|
|
|
|
CallingConvention = self.CallingConvention
|
|
|
|
|
};
|
|
|
|
|
foreach (var parameterDefinition in self.Parameters)
|
|
|
|
|
{
|
|
|
|
|
methodReference.Parameters.Add(new ParameterDefinition(parameterDefinition.ParameterType));
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
foreach (var genericParameter in self.GenericParameters)
|
|
|
|
|
{
|
|
|
|
|
methodReference.GenericParameters.Add(new GenericParameter(genericParameter.Name, methodReference));
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:13:03 +00:00
|
|
|
|
return methodReference;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private class AddSearchDirectoryHelper
|
|
|
|
|
{
|
|
|
|
|
public AddSearchDirectoryHelper(IAssemblyResolver assemblyResolver)
|
|
|
|
|
{
|
|
|
|
|
var method = assemblyResolver.GetType().GetMethod("AddSearchDirectory", (BindingFlags)20, null, new Type[]
|
|
|
|
|
{
|
|
|
|
|
typeof(string)
|
|
|
|
|
}, null);
|
|
|
|
|
if (method == null)
|
|
|
|
|
{
|
|
|
|
|
throw new Exception("Assembly resolver doesn't implement AddSearchDirectory method.");
|
|
|
|
|
}
|
2021-06-18 21:40:38 +00:00
|
|
|
|
|
2021-05-03 19:27:52 +00:00
|
|
|
|
_addSearchDirectory = (Helpers.AddSearchDirectoryHelper.AddSearchDirectoryDelegate)Delegate.CreateDelegate(typeof(Helpers.AddSearchDirectoryHelper.AddSearchDirectoryDelegate), assemblyResolver, method);
|
2021-05-03 19:13:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-05-03 19:27:52 +00:00
|
|
|
|
public void AddSearchDirectory(string directory) => _addSearchDirectory(directory);
|
2021-05-03 19:13:03 +00:00
|
|
|
|
|
|
|
|
|
private readonly Helpers.AddSearchDirectoryHelper.AddSearchDirectoryDelegate _addSearchDirectory;
|
|
|
|
|
|
|
|
|
|
private delegate void AddSearchDirectoryDelegate(string directory);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|