// // PdbWriter.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.Diagnostics.SymbolStore; using Mono.Cecil.Cil; using Mono.Collections.Generic; #if !READ_ONLY namespace Mono.Cecil.Pdb { public class PdbWriter : Cil.ISymbolWriter { readonly ModuleDefinition module; readonly SymWriter writer; readonly Dictionary documents; internal PdbWriter (ModuleDefinition module, SymWriter writer) { this.module = module; this.writer = writer; this.documents = new Dictionary (); } public bool GetDebugHeader (out ImageDebugDirectory directory, out byte [] header) { header = writer.GetDebugInfo (out directory); return true; } public void Write (MethodBody body) { var method_token = body.Method.MetadataToken; var sym_token = new SymbolToken (method_token.ToInt32 ()); var instructions = CollectInstructions (body); if (instructions.Count == 0) return; var start_offset = 0; var end_offset = body.CodeSize; writer.OpenMethod (sym_token); writer.OpenScope (start_offset); DefineSequencePoints (instructions); DefineVariables (body, start_offset, end_offset); writer.CloseScope (end_offset); writer.CloseMethod (); } Collection CollectInstructions (MethodBody body) { var collection = new Collection (); var instructions = body.Instructions; for (int i = 0; i < instructions.Count; i++) { var instruction = instructions [i]; var sequence_point = instruction.SequencePoint; if (sequence_point == null) continue; GetDocument (sequence_point.Document); collection.Add (instruction); } return collection; } void DefineVariables (MethodBody body, int start_offset, int end_offset) { if (!body.HasVariables) return; var sym_token = new SymbolToken (body.LocalVarToken.ToInt32 ()); var variables = body.Variables; for (int i = 0; i < variables.Count; i++) { var variable = variables [i]; CreateLocalVariable (variable, sym_token, start_offset, end_offset); } } void DefineSequencePoints (Collection instructions) { for (int i = 0; i < instructions.Count; i++) { var instruction = instructions [i]; var sequence_point = instruction.SequencePoint; writer.DefineSequencePoints ( GetDocument (sequence_point.Document), new [] { instruction.Offset }, new [] { sequence_point.StartLine }, new [] { sequence_point.StartColumn }, new [] { sequence_point.EndLine }, new [] { sequence_point.EndColumn }); } } void CreateLocalVariable (VariableDefinition variable, SymbolToken local_var_token, int start_offset, int end_offset) { writer.DefineLocalVariable2 ( variable.Name, 0, local_var_token, SymAddressKind.ILOffset, variable.Index, 0, 0, start_offset, end_offset); } SymDocumentWriter GetDocument (Document document) { if (document == null) return null; SymDocumentWriter doc_writer; if (documents.TryGetValue (document.Url, out doc_writer)) return doc_writer; doc_writer = writer.DefineDocument ( document.Url, document.Language.ToGuid (), document.LanguageVendor.ToGuid (), document.Type.ToGuid ()); documents [document.Url] = doc_writer; return doc_writer; } public void Write (MethodSymbols symbols) { var sym_token = new SymbolToken (symbols.MethodToken.ToInt32 ()); var start_offset = 0; var end_offset = symbols.CodeSize; writer.OpenMethod (sym_token); writer.OpenScope (start_offset); DefineSequencePoints (symbols); DefineVariables (symbols, start_offset, end_offset); writer.CloseScope (end_offset); writer.CloseMethod (); } void DefineSequencePoints (MethodSymbols symbols) { var instructions = symbols.instructions; for (int i = 0; i < instructions.Count; i++) { var instruction = instructions [i]; var sequence_point = instruction.SequencePoint; writer.DefineSequencePoints ( GetDocument (sequence_point.Document), new [] { instruction.Offset }, new [] { sequence_point.StartLine }, new [] { sequence_point.StartColumn }, new [] { sequence_point.EndLine }, new [] { sequence_point.EndColumn }); } } void DefineVariables (MethodSymbols symbols, int start_offset, int end_offset) { if (!symbols.HasVariables) return; var sym_token = new SymbolToken (symbols.LocalVarToken.ToInt32 ()); var variables = symbols.Variables; for (int i = 0; i < variables.Count; i++) { var variable = variables [i]; CreateLocalVariable (variable, sym_token, start_offset, end_offset); } } public void Dispose () { var entry_point = module.EntryPoint; if (entry_point != null) writer.SetUserEntryPoint (new SymbolToken (entry_point.MetadataToken.ToInt32 ())); writer.Close (); } } } #endif