mirror of https://github.com/icsharpcode/ILSpy.git
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.
278 lines
7.0 KiB
278 lines
7.0 KiB
// |
|
// PdbReader.cs |
|
// |
|
// Author: |
|
// Jb Evain (jbevain@gmail.com) |
|
// |
|
// Copyright (c) 2008 - 2011 Jb Evain |
|
// |
|
// 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. |
|
// |
|
|
|
using System; |
|
using System.Collections.Generic; |
|
using System.IO; |
|
|
|
using Microsoft.Cci.Pdb; |
|
|
|
using Mono.Cecil.Cil; |
|
|
|
namespace Mono.Cecil.Pdb { |
|
|
|
public class PdbReader : ISymbolReader { |
|
|
|
int age; |
|
Guid guid; |
|
|
|
readonly Stream pdb_file; |
|
readonly Dictionary<string, Document> documents = new Dictionary<string, Document> (); |
|
readonly Dictionary<uint, PdbFunction> functions = new Dictionary<uint, PdbFunction> (); |
|
|
|
internal PdbReader (Stream file) |
|
{ |
|
this.pdb_file = file; |
|
} |
|
|
|
/* |
|
uint Magic = 0x53445352; |
|
Guid Signature; |
|
uint Age; |
|
string FileName; |
|
*/ |
|
|
|
public bool ProcessDebugHeader (ImageDebugDirectory directory, byte [] header) |
|
{ |
|
if (header.Length < 24) |
|
return false; |
|
|
|
var magic = ReadInt32 (header, 0); |
|
if (magic != 0x53445352) |
|
return false; |
|
|
|
var guid_bytes = new byte [16]; |
|
Buffer.BlockCopy (header, 4, guid_bytes, 0, 16); |
|
|
|
this.guid = new Guid (guid_bytes); |
|
this.age = ReadInt32 (header, 20); |
|
|
|
return PopulateFunctions (); |
|
} |
|
|
|
static int ReadInt32 (byte [] bytes, int start) |
|
{ |
|
return (bytes [start] |
|
| (bytes [start + 1] << 8) |
|
| (bytes [start + 2] << 16) |
|
| (bytes [start + 3] << 24)); |
|
} |
|
|
|
bool PopulateFunctions () |
|
{ |
|
using (pdb_file) { |
|
int age; |
|
Guid guid; |
|
var funcs = PdbFile.LoadFunctions (pdb_file, true, out age, out guid); |
|
|
|
if (this.age != 0 && this.guid != guid) |
|
return false; |
|
|
|
foreach (PdbFunction function in funcs) |
|
functions.Add (function.token, function); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
public void Read (MethodBody body, InstructionMapper mapper) |
|
{ |
|
var method_token = body.Method.MetadataToken; |
|
|
|
PdbFunction function; |
|
if (!functions.TryGetValue (method_token.ToUInt32 (), out function)) |
|
return; |
|
|
|
ReadSequencePoints (function, mapper); |
|
ReadScopeAndLocals (function.scopes, null, body, mapper); |
|
} |
|
|
|
static void ReadScopeAndLocals (PdbScope [] scopes, Scope parent, MethodBody body, InstructionMapper mapper) |
|
{ |
|
foreach (PdbScope scope in scopes) |
|
ReadScopeAndLocals (scope, parent, body, mapper); |
|
|
|
CreateRootScope (body); |
|
} |
|
|
|
static void CreateRootScope (MethodBody body) |
|
{ |
|
if (!body.HasVariables) |
|
return; |
|
|
|
var instructions = body.Instructions; |
|
|
|
var root = new Scope (); |
|
root.Start = instructions [0]; |
|
root.End = instructions [instructions.Count - 1]; |
|
|
|
var variables = body.Variables; |
|
for (int i = 0; i < variables.Count; i++) |
|
root.Variables.Add (variables [i]); |
|
|
|
body.Scope = root; |
|
} |
|
|
|
static void ReadScopeAndLocals (PdbScope scope, Scope parent, MethodBody body, InstructionMapper mapper) |
|
{ |
|
//Scope s = new Scope (); |
|
//s.Start = GetInstruction (body, instructions, (int) scope.address); |
|
//s.End = GetInstruction (body, instructions, (int) scope.length - 1); |
|
|
|
//if (parent != null) |
|
// parent.Scopes.Add (s); |
|
//else |
|
// body.Scopes.Add (s); |
|
|
|
if (scope == null) |
|
return; |
|
|
|
foreach (PdbSlot slot in scope.slots) { |
|
int index = (int) slot.slot; |
|
if (index < 0 || index >= body.Variables.Count) |
|
continue; |
|
|
|
VariableDefinition variable = body.Variables [index]; |
|
variable.Name = slot.name; |
|
|
|
//s.Variables.Add (variable); |
|
} |
|
|
|
ReadScopeAndLocals (scope.scopes, null /* s */, body, mapper); |
|
} |
|
|
|
void ReadSequencePoints (PdbFunction function, InstructionMapper mapper) |
|
{ |
|
if (function.lines == null) |
|
return; |
|
|
|
foreach (PdbLines lines in function.lines) |
|
ReadLines (lines, mapper); |
|
} |
|
|
|
void ReadLines (PdbLines lines, InstructionMapper mapper) |
|
{ |
|
var document = GetDocument (lines.file); |
|
|
|
foreach (var line in lines.lines) |
|
ReadLine (line, document, mapper); |
|
} |
|
|
|
static void ReadLine (PdbLine line, Document document, InstructionMapper mapper) |
|
{ |
|
var instruction = mapper ((int) line.offset); |
|
if (instruction == null) |
|
return; |
|
|
|
var sequence_point = new SequencePoint (document); |
|
sequence_point.StartLine = (int) line.lineBegin; |
|
sequence_point.StartColumn = (int) line.colBegin; |
|
sequence_point.EndLine = (int) line.lineEnd; |
|
sequence_point.EndColumn = (int) line.colEnd; |
|
|
|
instruction.SequencePoint = sequence_point; |
|
} |
|
|
|
Document GetDocument (PdbSource source) |
|
{ |
|
string name = source.name; |
|
Document document; |
|
if (documents.TryGetValue (name, out document)) |
|
return document; |
|
|
|
document = new Document (name) { |
|
Language = source.language.ToLanguage (), |
|
LanguageVendor = source.vendor.ToVendor (), |
|
Type = source.doctype.ToType (), |
|
}; |
|
documents.Add (name, document); |
|
return document; |
|
} |
|
|
|
public void Read (MethodSymbols symbols) |
|
{ |
|
PdbFunction function; |
|
if (!functions.TryGetValue (symbols.MethodToken.ToUInt32 (), out function)) |
|
return; |
|
|
|
ReadSequencePoints (function, symbols); |
|
ReadLocals (function.scopes, symbols); |
|
} |
|
|
|
void ReadLocals (PdbScope [] scopes, MethodSymbols symbols) |
|
{ |
|
foreach (var scope in scopes) |
|
ReadLocals (scope, symbols); |
|
} |
|
|
|
void ReadLocals (PdbScope scope, MethodSymbols symbols) |
|
{ |
|
if (scope == null) |
|
return; |
|
|
|
foreach (var slot in scope.slots) { |
|
int index = (int) slot.slot; |
|
if (index < 0 || index >= symbols.Variables.Count) |
|
continue; |
|
|
|
var variable = symbols.Variables [index]; |
|
variable.Name = slot.name; |
|
} |
|
|
|
ReadLocals (scope.scopes, symbols); |
|
} |
|
|
|
void ReadSequencePoints (PdbFunction function, MethodSymbols symbols) |
|
{ |
|
if (function.lines == null) |
|
return; |
|
|
|
foreach (PdbLines lines in function.lines) |
|
ReadLines (lines, symbols); |
|
} |
|
|
|
void ReadLines (PdbLines lines, MethodSymbols symbols) |
|
{ |
|
for (int i = 0; i < lines.lines.Length; i++) { |
|
var line = lines.lines [i]; |
|
|
|
symbols.Instructions.Add (new InstructionSymbol ((int) line.offset, new SequencePoint (GetDocument (lines.file)) { |
|
StartLine = (int) line.lineBegin, |
|
StartColumn = (int) line.colBegin, |
|
EndLine = (int) line.lineEnd, |
|
EndColumn = (int) line.colEnd, |
|
})); |
|
} |
|
} |
|
|
|
public void Dispose () |
|
{ |
|
pdb_file.Close (); |
|
} |
|
} |
|
}
|
|
|