2016-08-03 10:59:49 +00:00
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Text;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using Mono.Cecil;
|
|
|
|
using Mono.Cecil.Cil;
|
|
|
|
using Mono.Collections.Generic;
|
|
|
|
|
|
|
|
namespace Mono
|
|
|
|
{
|
|
|
|
public class SymbolManager
|
|
|
|
{
|
|
|
|
string msymDir;
|
|
|
|
Logger logger;
|
|
|
|
|
|
|
|
public SymbolManager (string msymDir, Logger logger) {
|
|
|
|
this.msymDir = msymDir;
|
|
|
|
this.logger = logger;
|
|
|
|
}
|
|
|
|
|
2016-08-23 13:20:38 +00:00
|
|
|
internal bool TryResolveLocation (StackFrameData sfData)
|
2016-08-03 10:59:49 +00:00
|
|
|
{
|
2016-08-23 13:20:38 +00:00
|
|
|
if (sfData.Mvid == null)
|
2016-08-03 10:59:49 +00:00
|
|
|
return false;
|
|
|
|
|
2016-08-23 13:20:38 +00:00
|
|
|
var assemblyLocProvider = GetOrCreateAssemblyLocationProvider (sfData.Mvid);
|
2016-08-03 10:59:49 +00:00
|
|
|
if (assemblyLocProvider == null)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
SeqPointInfo seqPointInfo = null;
|
2016-08-23 13:20:38 +00:00
|
|
|
if (!sfData.IsILOffset && sfData.Aotid != null)
|
|
|
|
seqPointInfo = GetOrCreateSeqPointInfo (sfData.Aotid);
|
2016-08-03 10:59:49 +00:00
|
|
|
|
|
|
|
return assemblyLocProvider.TryResolveLocation (sfData, seqPointInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
Dictionary<string, AssemblyLocationProvider> assemblies = new Dictionary<string, AssemblyLocationProvider> ();
|
|
|
|
|
|
|
|
private AssemblyLocationProvider GetOrCreateAssemblyLocationProvider (string mvid)
|
|
|
|
{
|
|
|
|
if (assemblies.ContainsKey (mvid))
|
|
|
|
return assemblies[mvid];
|
|
|
|
|
|
|
|
var mvidDir = Path.Combine (msymDir, mvid);
|
|
|
|
if (!Directory.Exists (mvidDir)) {
|
|
|
|
logger.LogWarning ("MVID directory does not exist: {0}", mvidDir);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
string assemblyPath = null;
|
|
|
|
var exeFiles = Directory.GetFiles (mvidDir, "*.exe");
|
|
|
|
var dllFiles = Directory.GetFiles (mvidDir, "*.dll");
|
|
|
|
|
|
|
|
if (exeFiles.Length + dllFiles.Length != 1) {
|
|
|
|
logger.LogError ("MVID directory should include one assembly: {0}", mvidDir);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
assemblyPath = exeFiles.Length > 0 ? exeFiles[0] : dllFiles[0];
|
2016-08-03 10:59:49 +00:00
|
|
|
|
|
|
|
var locProvider = new AssemblyLocationProvider (assemblyPath, logger);
|
|
|
|
|
|
|
|
assemblies.Add (mvid, locProvider);
|
|
|
|
|
|
|
|
return locProvider;
|
|
|
|
}
|
|
|
|
|
|
|
|
Dictionary<string, SeqPointInfo> seqPointInfos = new Dictionary<string, SeqPointInfo> ();
|
|
|
|
|
|
|
|
private SeqPointInfo GetOrCreateSeqPointInfo (string aotid)
|
|
|
|
{
|
|
|
|
if (seqPointInfos.ContainsKey (aotid))
|
|
|
|
return seqPointInfos[aotid];
|
|
|
|
|
|
|
|
var aotidDir = Path.Combine (msymDir, aotid);
|
|
|
|
if (!Directory.Exists (aotidDir)) {
|
|
|
|
logger.LogError ("AOTID directory does not exist: {0}", aotidDir);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
string msymFile = null;
|
|
|
|
var msymFiles = Directory.GetFiles(aotidDir, "*.msym");
|
|
|
|
msymFile = msymFiles[0];
|
|
|
|
|
|
|
|
var seqPointInfo = SeqPointInfo.Read (msymFile);
|
|
|
|
|
|
|
|
seqPointInfos.Add (aotid, seqPointInfo);
|
|
|
|
|
|
|
|
return seqPointInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void StoreSymbols (params string[] lookupDirs)
|
|
|
|
{
|
|
|
|
foreach (var dir in lookupDirs) {
|
|
|
|
var exeFiles = Directory.GetFiles (dir, "*.exe");
|
|
|
|
var dllFiles = Directory.GetFiles (dir, "*.dll");
|
|
|
|
var assemblies = exeFiles.Concat (dllFiles);
|
|
|
|
foreach (var assemblyPath in assemblies) {
|
2016-11-10 13:04:39 +00:00
|
|
|
|
|
|
|
// TODO: Ignore embedded pdb
|
|
|
|
var symbolFile = GetSymbolFile (assemblyPath);
|
|
|
|
|
|
|
|
if (symbolFile == null) {
|
|
|
|
logger.LogWarning ("Directory {0} contains {1} but no debug symbols file was found.", dir, Path.GetFileName (assemblyPath));
|
|
|
|
// assemblies without debug symbols are useless
|
2016-08-03 10:59:49 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
var assembly = AssemblyDefinition.ReadAssembly (assemblyPath);
|
|
|
|
|
|
|
|
var mvid = assembly.MainModule.Mvid.ToString ("N");
|
|
|
|
var mvidDir = Path.Combine (msymDir, mvid);
|
|
|
|
|
|
|
|
if (Directory.Exists (mvidDir)) {
|
|
|
|
try {
|
|
|
|
Directory.Delete (mvidDir, true);
|
2016-11-10 13:04:39 +00:00
|
|
|
} catch (DirectoryNotFoundException) {}
|
2016-08-03 10:59:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Directory.CreateDirectory (mvidDir);
|
|
|
|
|
|
|
|
var mvidAssemblyPath = Path.Combine (mvidDir, Path.GetFileName (assemblyPath));
|
|
|
|
File.Copy (assemblyPath, mvidAssemblyPath);
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
var mvidDebugPath = Path.Combine (mvidDir, Path.GetFileName (symbolFile));
|
|
|
|
File.Copy (symbolFile, mvidDebugPath);
|
2016-08-03 10:59:49 +00:00
|
|
|
|
|
|
|
// TODO create MVID dir for non main modules with links to main module MVID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-11-10 13:04:39 +00:00
|
|
|
|
|
|
|
static string GetSymbolFile (string assembly)
|
|
|
|
{
|
|
|
|
var pdbName = Path.ChangeExtension (assembly, "pdb");
|
|
|
|
if (File.Exists (pdbName))
|
|
|
|
return pdbName;
|
|
|
|
|
|
|
|
var mdbName = assembly + ".mdb";
|
|
|
|
|
|
|
|
if (File.Exists (mdbName))
|
|
|
|
return mdbName;
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
2016-08-03 10:59:49 +00:00
|
|
|
}
|
|
|
|
}
|