2015-08-26 07:17:56 -04:00
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Text;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using Mono.Cecil;
|
2016-08-03 10:59:49 +00:00
|
|
|
using Mono.Cecil.Cil;
|
|
|
|
using Mono.Collections.Generic;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
namespace Mono
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
2016-08-03 10:59:49 +00:00
|
|
|
class AssemblyLocationProvider
|
|
|
|
{
|
|
|
|
AssemblyDefinition assembly;
|
|
|
|
Logger logger;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
public AssemblyLocationProvider (string assemblyPath, Logger logger)
|
|
|
|
{
|
|
|
|
assemblyPath = Path.GetFullPath (assemblyPath);
|
|
|
|
this.logger = logger;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
if (!File.Exists (assemblyPath))
|
|
|
|
throw new ArgumentException ("assemblyPath does not exist: "+ assemblyPath);
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
var readerParameters = new ReaderParameters { ReadSymbols = true };
|
|
|
|
assembly = AssemblyDefinition.ReadAssembly (assemblyPath, readerParameters);
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
public bool TryResolveLocation (StackFrameData sfData, SeqPointInfo seqPointInfo)
|
|
|
|
{
|
|
|
|
if (!assembly.MainModule.HasSymbols)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
TypeDefinition type = null;
|
|
|
|
var nested = sfData.TypeFullName.Split ('+');
|
|
|
|
var types = assembly.MainModule.Types;
|
|
|
|
foreach (var ntype in nested) {
|
|
|
|
if (type == null) {
|
|
|
|
// Use namespace first time.
|
|
|
|
type = types.FirstOrDefault (t => t.FullName == ntype);
|
|
|
|
} else {
|
|
|
|
type = types.FirstOrDefault (t => t.Name == ntype);
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
if (type == null) {
|
|
|
|
logger.LogWarning ("Could not find type: {0}", ntype);
|
2015-08-26 07:17:56 -04:00
|
|
|
return false;
|
2016-08-03 10:59:49 +00:00
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
types = type.NestedTypes;
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
var parensStart = sfData.MethodSignature.IndexOf ('(');
|
|
|
|
var methodName = sfData.MethodSignature.Substring (0, parensStart).TrimEnd ();
|
|
|
|
var methodParameters = sfData.MethodSignature.Substring (parensStart);
|
|
|
|
var method = type.Methods.FirstOrDefault (m => CompareName (m, methodName) && CompareParameters (m.Parameters, methodParameters));
|
|
|
|
if (method == null) {
|
|
|
|
logger.LogWarning ("Could not find method: {0}", methodName);
|
|
|
|
return false;
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
int ilOffset;
|
|
|
|
if (sfData.IsILOffset) {
|
|
|
|
ilOffset = sfData.Offset;
|
|
|
|
} else {
|
|
|
|
if (seqPointInfo == null)
|
|
|
|
return false;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
ilOffset = seqPointInfo.GetILOffset (method.MetadataToken.ToInt32 (), sfData.MethodIndex, sfData.Offset);
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
if (ilOffset < 0)
|
|
|
|
return false;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
SequencePoint sp = null;
|
|
|
|
foreach (var instr in method.Body.Instructions) {
|
|
|
|
if (instr.SequencePoint != null)
|
|
|
|
sp = instr.SequencePoint;
|
|
|
|
|
|
|
|
if (instr.Offset >= ilOffset) {
|
|
|
|
sfData.SetLocation (sp.Document.Url, sp.StartLine);
|
|
|
|
return true;
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
return false;
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
static bool CompareName (MethodDefinition candidate, string expected)
|
|
|
|
{
|
|
|
|
if (candidate.Name == expected)
|
|
|
|
return true;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
if (!candidate.HasGenericParameters)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
var genStart = expected.IndexOf ('[');
|
|
|
|
if (genStart < 0)
|
|
|
|
return false;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
if (candidate.Name != expected.Substring (0, genStart))
|
|
|
|
return false;
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
int arity = 1;
|
|
|
|
for (int pos = genStart; pos < expected.Length; ++pos) {
|
|
|
|
if (expected [pos] == ',')
|
|
|
|
++arity;
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
return candidate.GenericParameters.Count == arity;
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
static bool CompareParameters (Collection<ParameterDefinition> candidate, string expected)
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
2016-08-03 10:59:49 +00:00
|
|
|
var builder = new StringBuilder ();
|
|
|
|
builder.Append ("(");
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
for (int i = 0; i < candidate.Count; i++) {
|
|
|
|
var parameter = candidate [i];
|
|
|
|
if (i > 0)
|
|
|
|
builder.Append (", ");
|
|
|
|
|
|
|
|
if (parameter.ParameterType.IsSentinel)
|
|
|
|
builder.Append ("...,");
|
2015-08-26 07:17:56 -04:00
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
var pt = parameter.ParameterType;
|
|
|
|
if (!string.IsNullOrEmpty (pt.Namespace)) {
|
|
|
|
builder.Append (pt.Namespace);
|
|
|
|
builder.Append (".");
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
FormatElementType (pt, builder);
|
|
|
|
|
|
|
|
builder.Append (" ");
|
|
|
|
builder.Append (parameter.Name);
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
builder.Append (")");
|
|
|
|
|
|
|
|
return builder.ToString () == expected;
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
static void FormatElementType (TypeReference tr, StringBuilder builder)
|
2015-08-26 07:17:56 -04:00
|
|
|
{
|
2016-08-03 10:59:49 +00:00
|
|
|
var ts = tr as TypeSpecification;
|
|
|
|
if (ts != null) {
|
|
|
|
if (ts.IsByReference) {
|
|
|
|
FormatElementType (ts.ElementType, builder);
|
|
|
|
builder.Append ("&");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var array = ts as ArrayType;
|
|
|
|
if (array != null) {
|
|
|
|
FormatElementType (ts.ElementType, builder);
|
|
|
|
builder.Append ("[");
|
|
|
|
|
|
|
|
for (int ii = 0; ii < array.Rank - 1; ++ii) {
|
|
|
|
builder.Append (",");
|
|
|
|
}
|
|
|
|
|
|
|
|
builder.Append ("]");
|
|
|
|
return;
|
|
|
|
}
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
|
2016-08-03 10:59:49 +00:00
|
|
|
builder.Append (tr.Name);
|
2015-08-26 07:17:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|