104 lines
3.4 KiB
C#
104 lines
3.4 KiB
C#
using dnlib.DotNet;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace HybridCLR.Editor.Meta
|
|
{
|
|
/// <summary>
|
|
/// Replaces generic type/method var with its generic argument
|
|
/// </summary>
|
|
public sealed class GenericArgumentContext
|
|
{
|
|
List<TypeSig> typeArgsStack = new List<TypeSig>();
|
|
List<TypeSig> methodArgsStack = new List<TypeSig>();
|
|
|
|
public GenericArgumentContext(List<TypeSig> typeArgsStack, List<TypeSig> methodArgsStack)
|
|
{
|
|
this.typeArgsStack = typeArgsStack;
|
|
this.methodArgsStack = methodArgsStack;
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
/// Replaces a generic type/method var with its generic argument (if any). If
|
|
/// <paramref name="typeSig"/> isn't a generic type/method var or if it can't
|
|
/// be resolved, it itself is returned. Else the resolved type is returned.
|
|
/// </summary>
|
|
/// <param name="typeSig">Type signature</param>
|
|
/// <returns>New <see cref="TypeSig"/> which is never <c>null</c> unless
|
|
/// <paramref name="typeSig"/> is <c>null</c></returns>
|
|
public TypeSig Resolve(TypeSig typeSig)
|
|
{
|
|
if (!typeSig.ContainsGenericParameter)
|
|
{
|
|
return typeSig;
|
|
}
|
|
switch (typeSig.ElementType)
|
|
{
|
|
case ElementType.Ptr: return new PtrSig(Resolve(typeSig.Next));
|
|
case ElementType.ByRef: return new PtrSig(Resolve(typeSig.Next));
|
|
|
|
case ElementType.SZArray: return new PtrSig(Resolve(typeSig.Next));
|
|
case ElementType.Array:
|
|
{
|
|
var ara = (ArraySig)typeSig;
|
|
return new ArraySig(Resolve(typeSig.Next), ara.Rank, ara.Sizes, ara.LowerBounds);
|
|
}
|
|
|
|
case ElementType.Var:
|
|
{
|
|
GenericVar genericVar = (GenericVar)typeSig;
|
|
var newSig = Resolve(typeArgsStack, genericVar.Number, true);
|
|
if (newSig == null)
|
|
{
|
|
throw new Exception();
|
|
}
|
|
return newSig;
|
|
}
|
|
|
|
case ElementType.MVar:
|
|
{
|
|
GenericMVar genericVar = (GenericMVar)typeSig;
|
|
var newSig = Resolve(methodArgsStack, genericVar.Number, true);
|
|
if (newSig is null)
|
|
{
|
|
throw new Exception();
|
|
}
|
|
return newSig;
|
|
}
|
|
case ElementType.GenericInst:
|
|
{
|
|
var gia = (GenericInstSig)typeSig;
|
|
return new GenericInstSig(gia.GenericType, gia.GenericArguments.Select(ga => Resolve(ga)).ToList());
|
|
}
|
|
|
|
case ElementType.FnPtr:
|
|
{
|
|
throw new NotSupportedException(typeSig.ToString());
|
|
}
|
|
|
|
case ElementType.ValueArray:
|
|
{
|
|
var vas = (ValueArraySig)typeSig;
|
|
return new ValueArraySig(Resolve(vas.Next), vas.Size);
|
|
}
|
|
default: return typeSig;
|
|
}
|
|
}
|
|
|
|
private TypeSig Resolve(List<TypeSig> args, uint number, bool isTypeVar)
|
|
{
|
|
var typeSig = args[(int)number];
|
|
var gvar = typeSig as GenericSig;
|
|
if (gvar is null || gvar.IsTypeVar != isTypeVar)
|
|
return typeSig;
|
|
return gvar;
|
|
}
|
|
}
|
|
|
|
}
|