#develop (short for SharpDevelop) is a free IDE for .NET programming languages.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

284 lines
6.3 KiB

// <file>
// <copyright see="prj:///doc/copyright.txt"/>
// <license see="prj:///doc/license.txt"/>
// <owner name="David Srbecký" email="dsrbecky@gmail.com"/>
// <version>$Revision$</version>
// </file>
using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
using Debugger.Interop;
using Debugger.Interop.CorDebug;
using Debugger.Interop.CorSym;
using Debugger.Interop.MetaData;
using Debugger.MetaData;
namespace Debugger
{
public class Module: DebuggerObject, IDisposable
{
AppDomain appDomain;
Process process;
bool unloaded = false;
string fullPath;
int orderOfLoading = 0;
ICorDebugModule corModule;
ISymUnmanagedReader symReader;
MetaDataImport metaData;
internal Dictionary<string, DebugType> LoadedDebugTypes = new Dictionary<string, DebugType>();
public event EventHandler<ModuleEventArgs> SymbolsLoaded;
protected virtual void OnSymbolsLoaded(ModuleEventArgs e)
{
if (SymbolsLoaded != null) {
SymbolsLoaded(this, e);
}
}
public AppDomain AppDomain {
get { return appDomain; }
}
public Process Process {
get { return process; }
}
NDebugger Debugger {
get { return this.AppDomain.Process.Debugger; }
}
public MetaDataImport MetaData {
get {
return metaData;
}
}
public bool Unloaded {
get {
return unloaded;
}
}
public ISymUnmanagedReader SymReader {
get {
return symReader;
}
}
public ISymUnmanagedDocument[] SymDocuments {
get {
ISymUnmanagedDocument[] docs;
uint maxCount = 2;
uint fetched;
do {
maxCount *= 8;
docs = new ISymUnmanagedDocument[maxCount];
symReader.GetDocuments(maxCount, out fetched, docs);
} while (fetched == maxCount);
Array.Resize(ref docs, (int)fetched);
return docs;
}
}
[Debugger.Tests.Ignore]
public ICorDebugModule CorModule {
get { return corModule; }
}
[Debugger.Tests.Ignore]
public ICorDebugModule2 CorModule2 {
get { return (ICorDebugModule2)corModule; }
}
public ulong BaseAdress {
get {
return this.CorModule.GetBaseAddress();
}
}
public bool IsDynamic {
get {
return this.CorModule.IsDynamic() == 1;
}
}
public bool IsInMemory {
get {
return this.CorModule.IsInMemory() == 1;
}
}
internal uint AppDomainID {
get {
return this.CorModule.GetAssembly().GetAppDomain().GetID();
}
}
[Debugger.Tests.Ignore]
public string FullPath {
get {
return fullPath;
}
}
public string Filename {
get {
if (IsDynamic || IsInMemory) return String.Empty;
return System.IO.Path.GetFileName(FullPath);
}
}
[Debugger.Tests.Ignore]
public string DirectoryName {
get {
if (IsDynamic || IsInMemory) return String.Empty;
return System.IO.Path.GetDirectoryName(FullPath);
}
}
public bool HasSymbols {
get {
return symReader != null;
}
}
public int OrderOfLoading {
get {
return orderOfLoading;
}
set {
orderOfLoading = value;
}
}
/// <summary> Returns all non-generic types defined in the module </summary>
/// <remarks> Generic types can not be returned, because we do not know how to instanciate them </remarks>
public List<DebugType> GetDefinedTypes()
{
List<DebugType> types = new List<DebugType>();
foreach(TypeDefProps typeDef in this.MetaData.EnumTypeDefProps()) {
if (this.MetaData.EnumGenericParams(typeDef.Token).Length == 0) {
types.Add(DebugType.CreateFromTypeDefOrRef(this, null, typeDef.Token, null));
}
}
return types;
}
/// <summary> Get names of all generic and non-generic types defined in this module </summary>
public List<string> GetNamesOfDefinedTypes()
{
List<string> names = new List<string>();
foreach(TypeDefProps typeProps in this.MetaData.EnumTypeDefProps()) {
names.Add(typeProps.Name);
}
return names;
}
internal Module(AppDomain appDomain, ICorDebugModule corModule)
{
this.appDomain = appDomain;
this.process = appDomain.Process;
this.corModule = corModule;
metaData = new MetaDataImport(corModule);
fullPath = corModule.GetName();
LoadSymbols(process.Options.SymbolsSearchPaths);
ResetJustMyCodeStatus();
}
/// <summary> Try to load the debugging symbols (.pdb) from the given path </summary>
public void LoadSymbols(string[] searchPath)
{
if (symReader == null) {
symReader = metaData.GetSymReader(fullPath, string.Join("; ", searchPath ?? new string[0]));
if (symReader != null) {
OnSymbolsLoaded(new ModuleEventArgs(this));
ResetJustMyCodeStatus();
}
}
}
public void UpdateSymbolsFromStream(IStream pSymbolStream)
{
if (symReader != null) {
((ISymUnmanagedDispose)symReader).Destroy();
}
symReader = metaData.GetSymReader(pSymbolStream);
if (symReader != null) {
OnSymbolsLoaded(new ModuleEventArgs(this));
}
ResetJustMyCodeStatus();
}
/// <summary> Sets all code as being 'my code'. The code will be gradually
/// set to not-user-code as encountered acording to stepping options </summary>
public void ResetJustMyCodeStatus()
{
uint unused = 0;
if (process.Options.StepOverNoSymbols && !this.HasSymbols) {
// Optimization - set the code as non-user right away
this.CorModule2.SetJMCStatus(0, 0, ref unused);
return;
}
try {
this.CorModule2.SetJMCStatus(1, 0, ref unused);
} catch (COMException e) {
// Cannot use JMC on this code (likely wrong JIT settings).
if ((uint)e.ErrorCode == 0x80131323) {
process.TraceMessage("Cannot use JMC on this code. Release build?");
return;
}
throw;
}
}
public void ApplyChanges(byte[] metadata, byte[] il)
{
this.CorModule2.ApplyChanges((uint)metadata.Length, metadata, (uint)il.Length, il);
}
public void Dispose()
{
metaData.Dispose();
if (symReader != null) {
((ISymUnmanagedDispose)symReader).Destroy();
}
unloaded = true;
}
public override string ToString()
{
return string.Format("{0}", this.Filename);
}
}
[Serializable]
public class ModuleEventArgs : ProcessEventArgs
{
Module module;
public Module Module {
get {
return module;
}
}
public ModuleEventArgs(Module module): base(module.Process)
{
this.module = module;
}
}
}