966bba02bb
Former-commit-id: bb0468d0f257ff100aa895eb5fe583fb5dfbf900
451 lines
15 KiB
C#
451 lines
15 KiB
C#
#if !READ_ONLY
|
|
using System.IO;
|
|
using System.Linq;
|
|
|
|
using Mono.Cecil.Cil;
|
|
using Mono.Cecil.Pdb;
|
|
|
|
using NUnit.Framework;
|
|
|
|
namespace Mono.Cecil.Tests {
|
|
|
|
[TestFixture]
|
|
public class PdbTests : BaseTestFixture {
|
|
|
|
[Test]
|
|
public void Main ()
|
|
{
|
|
TestModule ("test.exe", module => {
|
|
var type = module.GetType ("Program");
|
|
var main = type.GetMethod ("Main");
|
|
|
|
AssertCode (@"
|
|
.locals init (System.Int32 i, System.Int32 CS$1$0000, System.Boolean CS$4$0001)
|
|
.line 6,6:2,3 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_0000: nop
|
|
.line 7,7:8,18 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_0001: ldc.i4.0
|
|
IL_0002: stloc.0
|
|
.line hidden 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_0003: br.s IL_0012
|
|
.line 8,8:4,21 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_0005: ldarg.0
|
|
IL_0006: ldloc.0
|
|
IL_0007: ldelem.ref
|
|
IL_0008: call System.Void Program::Print(System.String)
|
|
IL_000d: nop
|
|
.line 7,7:36,39 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_000e: ldloc.0
|
|
IL_000f: ldc.i4.1
|
|
IL_0010: add
|
|
IL_0011: stloc.0
|
|
.line 7,7:19,34 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_0012: ldloc.0
|
|
IL_0013: ldarg.0
|
|
IL_0014: ldlen
|
|
IL_0015: conv.i4
|
|
IL_0016: clt
|
|
IL_0018: stloc.2
|
|
.line hidden 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_0019: ldloc.2
|
|
IL_001a: brtrue.s IL_0005
|
|
.line 10,10:3,12 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_001c: ldc.i4.0
|
|
IL_001d: stloc.1
|
|
IL_001e: br.s IL_0020
|
|
.line 11,11:2,3 'c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs'
|
|
IL_0020: ldloc.1
|
|
IL_0021: ret
|
|
", main);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof(PdbReaderProvider), symbolWriterProvider: typeof(PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void DebuggerHiddenVariable ()
|
|
{
|
|
TestModule ("test.exe", module => {
|
|
var type = module.GetType ("Program");
|
|
var method = type.GetMethod ("Main");
|
|
|
|
var scope = method.DebugInformation.Scope;
|
|
|
|
Assert.IsTrue (scope.HasVariables);
|
|
var variables = scope.Variables;
|
|
|
|
Assert.AreEqual ("CS$1$0000", variables [0].Name);
|
|
Assert.IsTrue (variables [0].IsDebuggerHidden);
|
|
Assert.AreEqual ("CS$4$0001", variables [1].Name);
|
|
Assert.IsTrue (variables [1].IsDebuggerHidden);
|
|
|
|
Assert.AreEqual (1, scope.Scopes.Count);
|
|
scope = scope.Scopes [0];
|
|
variables = scope.Variables;
|
|
|
|
Assert.AreEqual ("i", variables [0].Name);
|
|
Assert.IsFalse (variables [0].IsDebuggerHidden);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof(PdbReaderProvider), symbolWriterProvider: typeof(PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void Document ()
|
|
{
|
|
TestModule ("test.exe", module => {
|
|
var type = module.GetType ("Program");
|
|
var method = type.GetMethod ("Main");
|
|
|
|
var sequence_point = method.DebugInformation.SequencePoints.First (sp => sp != null);
|
|
var document = sequence_point.Document;
|
|
|
|
Assert.IsNotNull (document);
|
|
|
|
Assert.AreEqual (@"c:\sources\cecil\symbols\Mono.Cecil.Pdb\Test\Resources\assemblies\test.cs", document.Url);
|
|
Assert.AreEqual (DocumentType.Text, document.Type);
|
|
Assert.AreEqual (DocumentHashAlgorithm.None, document.HashAlgorithm);
|
|
Assert.AreEqual (DocumentLanguage.CSharp, document.Language);
|
|
Assert.AreEqual (DocumentLanguageVendor.Microsoft, document.LanguageVendor);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof(PdbReaderProvider), symbolWriterProvider: typeof(PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void BasicDocument ()
|
|
{
|
|
TestModule ("VBConsApp.exe", module => {
|
|
var type = module.GetType ("VBConsApp.Program");
|
|
var method = type.GetMethod ("Main");
|
|
|
|
var sequence_point = method.DebugInformation.SequencePoints.First (sp => sp != null);
|
|
var document = sequence_point.Document;
|
|
|
|
Assert.IsNotNull (document);
|
|
|
|
Assert.AreEqual (@"c:\tmp\VBConsApp\Program.vb", document.Url);
|
|
Assert.AreEqual (DocumentType.Text, document.Type);
|
|
Assert.AreEqual (DocumentHashAlgorithm.None, document.HashAlgorithm);
|
|
Assert.AreEqual (DocumentLanguage.Basic, document.Language);
|
|
Assert.AreEqual (DocumentLanguageVendor.Microsoft, document.LanguageVendor);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof(PdbReaderProvider), symbolWriterProvider: typeof(PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void FSharpDocument ()
|
|
{
|
|
TestModule ("fsapp.exe", module => {
|
|
var type = module.GetType ("Program");
|
|
var method = type.GetMethod ("fact");
|
|
|
|
var sequence_point = method.DebugInformation.SequencePoints.First (sp => sp != null);
|
|
var document = sequence_point.Document;
|
|
|
|
Assert.IsNotNull (document);
|
|
|
|
Assert.AreEqual (@"c:\tmp\fsapp\Program.fs", document.Url);
|
|
Assert.AreEqual (DocumentType.Text, document.Type);
|
|
Assert.AreEqual (DocumentHashAlgorithm.None, document.HashAlgorithm);
|
|
Assert.AreEqual (DocumentLanguage.FSharp, document.Language);
|
|
Assert.AreEqual (DocumentLanguageVendor.Microsoft, document.LanguageVendor);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof(PdbReaderProvider), symbolWriterProvider: typeof(PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void EmptyEnumerable ()
|
|
{
|
|
TestModule ("empty-iterator.dll", module => {
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof (PdbReaderProvider), symbolWriterProvider: typeof (PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void LocalVariables ()
|
|
{
|
|
TestModule ("ComplexPdb.dll", module => {
|
|
var type = module.GetType ("ComplexPdb.Program");
|
|
var method = type.GetMethod ("Bar");
|
|
var debug_info = method.DebugInformation;
|
|
|
|
Assert.IsNotNull (debug_info.Scope);
|
|
Assert.IsTrue (debug_info.Scope.HasScopes);
|
|
Assert.AreEqual (2, debug_info.Scope.Scopes.Count);
|
|
|
|
var scope = debug_info.Scope.Scopes [0];
|
|
|
|
Assert.IsNotNull (scope);
|
|
Assert.IsTrue (scope.HasVariables);
|
|
Assert.AreEqual (1, scope.Variables.Count);
|
|
|
|
var variable = scope.Variables [0];
|
|
|
|
Assert.AreEqual ("s", variable.Name);
|
|
Assert.IsFalse (variable.IsDebuggerHidden);
|
|
Assert.AreEqual (2, variable.Index);
|
|
|
|
scope = debug_info.Scope.Scopes [1];
|
|
|
|
Assert.IsNotNull (scope);
|
|
Assert.IsTrue (scope.HasVariables);
|
|
Assert.AreEqual (1, scope.Variables.Count);
|
|
|
|
variable = scope.Variables [0];
|
|
|
|
Assert.AreEqual ("s", variable.Name);
|
|
Assert.IsFalse (variable.IsDebuggerHidden);
|
|
Assert.AreEqual (3, variable.Index);
|
|
|
|
Assert.IsTrue (scope.HasScopes);
|
|
Assert.AreEqual (1, scope.Scopes.Count);
|
|
|
|
scope = scope.Scopes [0];
|
|
|
|
Assert.IsNotNull (scope);
|
|
Assert.IsTrue (scope.HasVariables);
|
|
Assert.AreEqual (1, scope.Variables.Count);
|
|
|
|
variable = scope.Variables [0];
|
|
|
|
Assert.AreEqual ("u", variable.Name);
|
|
Assert.IsFalse (variable.IsDebuggerHidden);
|
|
Assert.AreEqual (5, variable.Index);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof (PdbReaderProvider), symbolWriterProvider: typeof (PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void LocalConstants ()
|
|
{
|
|
TestModule ("ComplexPdb.dll", module => {
|
|
var type = module.GetType ("ComplexPdb.Program");
|
|
var method = type.GetMethod ("Bar");
|
|
var debug_info = method.DebugInformation;
|
|
|
|
Assert.IsNotNull (debug_info.Scope);
|
|
Assert.IsTrue (debug_info.Scope.HasScopes);
|
|
Assert.AreEqual (2, debug_info.Scope.Scopes.Count);
|
|
|
|
var scope = debug_info.Scope.Scopes [1];
|
|
|
|
Assert.IsNotNull (scope);
|
|
Assert.IsTrue (scope.HasConstants);
|
|
Assert.AreEqual (2, scope.Constants.Count);
|
|
|
|
var constant = scope.Constants [0];
|
|
|
|
Assert.AreEqual ("b", constant.Name);
|
|
Assert.AreEqual (12, constant.Value);
|
|
Assert.AreEqual (MetadataType.Int32, constant.ConstantType.MetadataType);
|
|
|
|
constant = scope.Constants [1];
|
|
Assert.AreEqual ("c", constant.Name);
|
|
Assert.AreEqual ((decimal) 74, constant.Value);
|
|
Assert.AreEqual (MetadataType.ValueType, constant.ConstantType.MetadataType);
|
|
|
|
method = type.GetMethod ("Foo");
|
|
debug_info = method.DebugInformation;
|
|
|
|
Assert.IsNotNull (debug_info.Scope);
|
|
Assert.IsTrue (debug_info.Scope.HasConstants);
|
|
Assert.AreEqual (4, debug_info.Scope.Constants.Count);
|
|
|
|
constant = debug_info.Scope.Constants [0];
|
|
Assert.AreEqual ("s", constant.Name);
|
|
Assert.AreEqual ("const string", constant.Value);
|
|
Assert.AreEqual (MetadataType.String, constant.ConstantType.MetadataType);
|
|
|
|
constant = debug_info.Scope.Constants [1];
|
|
Assert.AreEqual ("f", constant.Name);
|
|
Assert.AreEqual (1, constant.Value);
|
|
Assert.AreEqual (MetadataType.Int32, constant.ConstantType.MetadataType);
|
|
|
|
constant = debug_info.Scope.Constants [2];
|
|
Assert.AreEqual ("o", constant.Name);
|
|
Assert.AreEqual (null, constant.Value);
|
|
Assert.AreEqual (MetadataType.Object, constant.ConstantType.MetadataType);
|
|
|
|
constant = debug_info.Scope.Constants [3];
|
|
Assert.AreEqual ("u", constant.Name);
|
|
Assert.AreEqual (null, constant.Value);
|
|
Assert.AreEqual (MetadataType.String, constant.ConstantType.MetadataType);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof (PdbReaderProvider), symbolWriterProvider: typeof (PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void ImportScope ()
|
|
{
|
|
TestModule ("ComplexPdb.dll", module => {
|
|
var type = module.GetType ("ComplexPdb.Program");
|
|
var method = type.GetMethod ("Bar");
|
|
var debug_info = method.DebugInformation;
|
|
|
|
Assert.IsNotNull (debug_info.Scope);
|
|
|
|
var import = debug_info.Scope.Import;
|
|
Assert.IsNotNull (import);
|
|
|
|
Assert.IsTrue (import.HasTargets);
|
|
Assert.AreEqual (6, import.Targets.Count);
|
|
var target = import.Targets [0];
|
|
|
|
Assert.AreEqual (ImportTargetKind.ImportNamespace, target.Kind);
|
|
Assert.AreEqual ("System", target.Namespace);
|
|
|
|
target = import.Targets [1];
|
|
|
|
Assert.AreEqual (ImportTargetKind.ImportNamespace, target.Kind);
|
|
Assert.AreEqual ("System.Collections.Generic", target.Namespace);
|
|
|
|
target = import.Targets [2];
|
|
|
|
Assert.AreEqual (ImportTargetKind.ImportNamespace, target.Kind);
|
|
Assert.AreEqual ("System.Threading.Tasks", target.Namespace);
|
|
|
|
target = import.Targets [3];
|
|
|
|
Assert.AreEqual (ImportTargetKind.ImportType, target.Kind);
|
|
Assert.AreEqual ("System.Console", target.Type.FullName);
|
|
|
|
target = import.Targets [4];
|
|
|
|
Assert.AreEqual (ImportTargetKind.DefineTypeAlias, target.Kind);
|
|
Assert.AreEqual ("Foo1", target.Alias);
|
|
Assert.AreEqual ("System.Console", target.Type.FullName);
|
|
|
|
target = import.Targets [5];
|
|
|
|
Assert.AreEqual (ImportTargetKind.DefineNamespaceAlias, target.Kind);
|
|
Assert.AreEqual ("Foo2", target.Alias);
|
|
Assert.AreEqual ("System.Reflection", target.Namespace);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof (PdbReaderProvider), symbolWriterProvider: typeof (PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void StateMachineKickOff ()
|
|
{
|
|
TestModule ("ComplexPdb.dll", module => {
|
|
var state_machine = module.GetType ("ComplexPdb.Program/<TestAsync>d__2");
|
|
var move_next = state_machine.GetMethod ("MoveNext");
|
|
var symbol = move_next.DebugInformation;
|
|
|
|
Assert.IsNotNull (symbol);
|
|
Assert.IsNotNull (symbol.StateMachineKickOffMethod);
|
|
Assert.AreEqual ("System.Threading.Tasks.Task ComplexPdb.Program::TestAsync()", symbol.StateMachineKickOffMethod.FullName);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof (PdbReaderProvider), symbolWriterProvider: typeof (PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void Iterators ()
|
|
{
|
|
TestModule ("ComplexPdb.dll", module => {
|
|
var state_machine = module.GetType ("ComplexPdb.Program/<TestAsync>d__2");
|
|
var move_next = state_machine.GetMethod ("MoveNext");
|
|
|
|
Assert.IsTrue (move_next.DebugInformation.HasCustomDebugInformations);
|
|
Assert.AreEqual (2, move_next.DebugInformation.CustomDebugInformations.Count);
|
|
|
|
var state_machine_scope = move_next.DebugInformation.CustomDebugInformations [0] as StateMachineScopeDebugInformation;
|
|
Assert.IsNotNull (state_machine_scope);
|
|
Assert.AreEqual (142, state_machine_scope.Start.Offset);
|
|
Assert.AreEqual (319, state_machine_scope.End.Offset);
|
|
|
|
var async_body = move_next.DebugInformation.CustomDebugInformations [1] as AsyncMethodBodyDebugInformation;
|
|
Assert.IsNotNull (async_body);
|
|
Assert.AreEqual (-1, async_body.CatchHandler.Offset);
|
|
|
|
Assert.AreEqual (2, async_body.Yields.Count);
|
|
Assert.AreEqual (68, async_body.Yields [0].Offset);
|
|
Assert.AreEqual (197, async_body.Yields [1].Offset);
|
|
|
|
Assert.AreEqual (2, async_body.Resumes.Count);
|
|
Assert.AreEqual (98, async_body.Resumes [0].Offset);
|
|
Assert.AreEqual (227, async_body.Resumes [1].Offset);
|
|
|
|
Assert.AreEqual (move_next, async_body.MoveNextMethod);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof (PdbReaderProvider), symbolWriterProvider: typeof (PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void ImportsForFirstMethod ()
|
|
{
|
|
TestModule ("CecilTest.exe", module => {
|
|
var type = module.GetType ("CecilTest.Program");
|
|
var method = type.GetMethod ("Main");
|
|
|
|
var debug = method.DebugInformation;
|
|
var scope = debug.Scope;
|
|
|
|
Assert.IsTrue (scope.End.IsEndOfMethod);
|
|
|
|
var import = scope.Import;
|
|
|
|
Assert.IsNotNull (import);
|
|
Assert.AreEqual (5, import.Targets.Count);
|
|
|
|
var ns = new [] {
|
|
"System",
|
|
"System.Collections.Generic",
|
|
"System.Linq",
|
|
"System.Text",
|
|
"System.Threading.Tasks",
|
|
};
|
|
|
|
for (int i = 0; i < import.Targets.Count; i++) {
|
|
var target = import.Targets [i];
|
|
|
|
Assert.AreEqual (ImportTargetKind.ImportNamespace, target.Kind);
|
|
Assert.AreEqual (ns [i], target.Namespace);
|
|
}
|
|
|
|
Assert.AreEqual ("System", import.Targets [0].Namespace);
|
|
}, readOnly: Platform.OnMono, symbolReaderProvider: typeof (PdbReaderProvider), symbolWriterProvider: typeof (PdbWriterProvider));
|
|
}
|
|
|
|
[Test]
|
|
public void CreateMethodFromScratch ()
|
|
{
|
|
IgnoreOnMono ();
|
|
|
|
var module = ModuleDefinition.CreateModule ("Pan", ModuleKind.Dll);
|
|
var type = new TypeDefinition ("Pin", "Pon", TypeAttributes.Public | TypeAttributes.Abstract | TypeAttributes.Sealed, module.ImportReference (typeof (object)));
|
|
module.Types.Add (type);
|
|
|
|
var method = new MethodDefinition ("Pang", MethodAttributes.Public | MethodAttributes.Static, module.ImportReference (typeof (string)));
|
|
type.Methods.Add (method);
|
|
|
|
var body = method.Body;
|
|
|
|
body.InitLocals = true;
|
|
|
|
var il = body.GetILProcessor ();
|
|
var temp = new VariableDefinition (module.ImportReference (typeof (string)));
|
|
body.Variables.Add (temp);
|
|
|
|
il.Emit (OpCodes.Nop);
|
|
il.Emit (OpCodes.Ldstr, "hello");
|
|
il.Emit (OpCodes.Stloc, temp);
|
|
il.Emit (OpCodes.Ldloc, temp);
|
|
il.Emit (OpCodes.Ret);
|
|
|
|
var sequence_point = new SequencePoint (body.Instructions [0], new Document (@"C:\test.cs")) {
|
|
StartLine = 0,
|
|
StartColumn = 0,
|
|
EndLine = 0,
|
|
EndColumn = 4,
|
|
};
|
|
|
|
method.DebugInformation.SequencePoints.Add (sequence_point);
|
|
|
|
method.DebugInformation.Scope = new ScopeDebugInformation (body.Instructions [0], null) {
|
|
Variables = { new VariableDebugInformation (temp, "temp") }
|
|
};
|
|
|
|
var file = Path.Combine (Path.GetTempPath (), "Pan.dll");
|
|
module.Write (file, new WriterParameters {
|
|
SymbolWriterProvider = new PdbWriterProvider (),
|
|
});
|
|
|
|
module = ModuleDefinition.ReadModule (file, new ReaderParameters {
|
|
SymbolReaderProvider = new PdbReaderProvider (),
|
|
});
|
|
|
|
method = module.GetType ("Pin.Pon").GetMethod ("Pang");
|
|
|
|
Assert.AreEqual ("temp", method.DebugInformation.Scope.Variables [0].Name);
|
|
}
|
|
}
|
|
}
|
|
#endif |