2025-10-29 20:40:37 +08:00
|
|
|
|
// Copyright 2025 Code Philosophy
|
|
|
|
|
|
//
|
|
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
|
|
//
|
|
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
|
|
//
|
|
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
|
|
// SOFTWARE.
|
|
|
|
|
|
|
2025-05-21 09:23:29 +08:00
|
|
|
|
using dnlib.DotNet;
|
|
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
|
|
namespace Obfuz.Utils
|
|
|
|
|
|
{
|
|
|
|
|
|
public sealed class GenericArgumentContext
|
|
|
|
|
|
{
|
2025-06-17 20:21:28 +08:00
|
|
|
|
public readonly List<TypeSig> typeArgsStack;
|
|
|
|
|
|
public readonly List<TypeSig> methodArgsStack;
|
2025-05-21 09:23:29 +08:00
|
|
|
|
|
|
|
|
|
|
public GenericArgumentContext(IList<TypeSig> typeArgsStack, IList<TypeSig> methodArgsStack)
|
|
|
|
|
|
{
|
|
|
|
|
|
this.typeArgsStack = typeArgsStack?.ToList();
|
|
|
|
|
|
this.methodArgsStack = methodArgsStack?.ToList();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public TypeSig Resolve(TypeSig typeSig)
|
|
|
|
|
|
{
|
2025-05-30 13:32:29 +08:00
|
|
|
|
if (!typeSig.ContainsGenericParameter)
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
2025-05-30 13:32:29 +08:00
|
|
|
|
return typeSig;
|
2025-05-21 09:23:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
typeSig = typeSig.RemovePinnedAndModifiers();
|
2025-05-30 13:32:29 +08:00
|
|
|
|
switch (typeSig.ElementType)
|
|
|
|
|
|
{
|
|
|
|
|
|
case ElementType.Ptr: return new PtrSig(Resolve(typeSig.Next));
|
|
|
|
|
|
case ElementType.ByRef: return new ByRefSig(Resolve(typeSig.Next));
|
2025-05-21 09:23:29 +08:00
|
|
|
|
|
2025-05-22 19:44:13 +08:00
|
|
|
|
case ElementType.SZArray: return new SZArraySig(Resolve(typeSig.Next));
|
2025-05-30 13:32:29 +08:00
|
|
|
|
case ElementType.Array:
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
|
|
|
|
|
var ara = (ArraySig)typeSig;
|
|
|
|
|
|
return new ArraySig(Resolve(typeSig.Next), ara.Rank, ara.Sizes, ara.LowerBounds);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-30 13:32:29 +08:00
|
|
|
|
case ElementType.Var:
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
|
|
|
|
|
GenericVar genericVar = (GenericVar)typeSig;
|
2025-05-22 19:44:13 +08:00
|
|
|
|
var newSig = Resolve(typeArgsStack, genericVar.Number);
|
2025-05-21 09:23:29 +08:00
|
|
|
|
if (newSig == null)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new Exception();
|
|
|
|
|
|
}
|
|
|
|
|
|
return newSig;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-30 13:32:29 +08:00
|
|
|
|
case ElementType.MVar:
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
|
|
|
|
|
GenericMVar genericVar = (GenericMVar)typeSig;
|
2025-05-22 19:44:13 +08:00
|
|
|
|
var newSig = Resolve(methodArgsStack, genericVar.Number);
|
2025-05-21 09:23:29 +08:00
|
|
|
|
if (newSig == null)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new Exception();
|
|
|
|
|
|
}
|
|
|
|
|
|
return newSig;
|
|
|
|
|
|
}
|
2025-05-30 13:32:29 +08:00
|
|
|
|
case ElementType.GenericInst:
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
|
|
|
|
|
var gia = (GenericInstSig)typeSig;
|
|
|
|
|
|
return new GenericInstSig(gia.GenericType, gia.GenericArguments.Select(ga => Resolve(ga)).ToList());
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-30 13:32:29 +08:00
|
|
|
|
case ElementType.FnPtr:
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
|
|
|
|
|
var fptr = (FnPtrSig)typeSig;
|
|
|
|
|
|
var cs = fptr.Signature;
|
|
|
|
|
|
CallingConventionSig ccs;
|
|
|
|
|
|
switch (cs)
|
|
|
|
|
|
{
|
|
|
|
|
|
case MethodSig ms:
|
|
|
|
|
|
{
|
|
|
|
|
|
ccs = new MethodSig(ms.GetCallingConvention(), ms.GenParamCount, Resolve(ms.RetType), ms.Params.Select(p => Resolve(p)).ToList());
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
case PropertySig ps:
|
|
|
|
|
|
{
|
|
|
|
|
|
ccs = new PropertySig(ps.HasThis, Resolve(ps.RetType));
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
case GenericInstMethodSig gims:
|
|
|
|
|
|
{
|
|
|
|
|
|
ccs = new GenericInstMethodSig(gims.GenericArguments.Select(ga => Resolve(ga)).ToArray());
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
default: throw new NotSupportedException(cs.ToString());
|
|
|
|
|
|
}
|
|
|
|
|
|
return new FnPtrSig(ccs);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-30 13:32:29 +08:00
|
|
|
|
case ElementType.ValueArray:
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
|
|
|
|
|
var vas = (ValueArraySig)typeSig;
|
|
|
|
|
|
return new ValueArraySig(Resolve(vas.Next), vas.Size);
|
|
|
|
|
|
}
|
|
|
|
|
|
default: return typeSig;
|
2025-05-30 13:32:29 +08:00
|
|
|
|
}
|
2025-05-21 09:23:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-05-22 19:44:13 +08:00
|
|
|
|
private TypeSig Resolve(List<TypeSig> args, uint number)
|
2025-05-21 09:23:29 +08:00
|
|
|
|
{
|
2025-06-17 20:21:28 +08:00
|
|
|
|
if (args == null)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new ArgumentNullException(nameof(args));
|
|
|
|
|
|
}
|
2025-05-22 19:44:13 +08:00
|
|
|
|
return args[(int)number];
|
2025-05-21 09:23:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|