e5cd25ff4f
Former-commit-id: 2a5b9df2014f72665850c7f885e7aed54704a53a
143 lines
3.4 KiB
C#
143 lines
3.4 KiB
C#
//
|
|
// Author:
|
|
// Jb Evain (jbevain@gmail.com)
|
|
//
|
|
// Copyright (c) 2008 - 2015 Jb Evain
|
|
// Copyright (c) 2008 - 2011 Novell, Inc.
|
|
//
|
|
// Licensed under the MIT/X11 license.
|
|
//
|
|
|
|
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<string, SymDocumentWriter> documents;
|
|
|
|
internal PdbWriter (ModuleDefinition module, SymWriter writer)
|
|
{
|
|
this.module = module;
|
|
this.writer = writer;
|
|
this.documents = new Dictionary<string, SymDocumentWriter> ();
|
|
}
|
|
|
|
public bool GetDebugHeader (out ImageDebugDirectory directory, out byte [] header)
|
|
{
|
|
header = writer.GetDebugInfo (out directory);
|
|
return true;
|
|
}
|
|
|
|
public void Write (MethodDebugInformation info)
|
|
{
|
|
var method_token = info.method.MetadataToken;
|
|
var sym_token = new SymbolToken (method_token.ToInt32 ());
|
|
|
|
writer.OpenMethod (sym_token);
|
|
|
|
if (!info.sequence_points.IsNullOrEmpty ())
|
|
DefineSequencePoints (info.sequence_points);
|
|
|
|
if (info.scope != null)
|
|
DefineScope (info.scope, info);
|
|
|
|
writer.CloseMethod ();
|
|
}
|
|
|
|
void DefineScope (ScopeDebugInformation scope, MethodDebugInformation info)
|
|
{
|
|
var start_offset = scope.Start.Offset;
|
|
var end_offset = scope.End.IsEndOfMethod
|
|
? info.code_size
|
|
: scope.End.Offset;
|
|
|
|
writer.OpenScope (start_offset);
|
|
|
|
var sym_token = new SymbolToken (info.local_var_token.ToInt32 ());
|
|
|
|
if (!scope.variables.IsNullOrEmpty ()) {
|
|
for (int i = 0; i < scope.variables.Count; i++) {
|
|
var variable = scope.variables [i];
|
|
CreateLocalVariable (variable, sym_token, start_offset, end_offset);
|
|
}
|
|
}
|
|
|
|
if (!scope.scopes.IsNullOrEmpty ()) {
|
|
for (int i = 0; i < scope.scopes.Count; i++)
|
|
DefineScope (scope.scopes [i], info);
|
|
}
|
|
|
|
writer.CloseScope (end_offset);
|
|
}
|
|
|
|
void DefineSequencePoints (Collection<SequencePoint> sequence_points)
|
|
{
|
|
for (int i = 0; i < sequence_points.Count; i++) {
|
|
var sequence_point = sequence_points [i];
|
|
|
|
writer.DefineSequencePoints (
|
|
GetDocument (sequence_point.Document),
|
|
new [] { sequence_point.Offset },
|
|
new [] { sequence_point.StartLine },
|
|
new [] { sequence_point.StartColumn },
|
|
new [] { sequence_point.EndLine },
|
|
new [] { sequence_point.EndColumn });
|
|
}
|
|
}
|
|
|
|
void CreateLocalVariable (VariableDebugInformation variable, SymbolToken local_var_token, int start_offset, int end_offset)
|
|
{
|
|
writer.DefineLocalVariable2 (
|
|
variable.Name,
|
|
variable.Attributes,
|
|
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 Dispose ()
|
|
{
|
|
var entry_point = module.EntryPoint;
|
|
if (entry_point != null)
|
|
writer.SetUserEntryPoint (new SymbolToken (entry_point.MetadataToken.ToInt32 ()));
|
|
|
|
writer.Close ();
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|