// Copyright (c) 2011 AlphaSierraPapa for the SharpDevelop Team // // 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.Linq; using ICSharpCode.Decompiler; using ICSharpCode.Decompiler.ILAst; using ICSharpCode.NRefactory; using ICSharpCode.NRefactory.CSharp; using Mono.Cecil; namespace ICSharpCode.Decompiler.Ast { public class TextOutputFormatter : IOutputFormatter { readonly ITextOutput output; readonly Stack nodeStack = new Stack(); int braceLevelWithinType = -1; bool inDocumentationComment = false; bool firstUsingDeclaration; bool lastUsingDeclaration; public TextOutputFormatter(ITextOutput output) { if (output == null) throw new ArgumentNullException("output"); this.output = output; } public void WriteIdentifier(string identifier) { var definition = GetCurrentDefinition(); if (definition != null) { output.WriteDefinition(identifier, definition, false); return; } object memberRef = GetCurrentMemberReference(); if (memberRef != null) { output.WriteReference(identifier, memberRef); return; } definition = GetCurrentLocalDefinition(); if (definition != null) { output.WriteDefinition(identifier, definition); return; } memberRef = GetCurrentLocalReference(); if (memberRef != null) { output.WriteReference(identifier, memberRef, true); return; } if (firstUsingDeclaration) { output.MarkFoldStart(defaultCollapsed: true); firstUsingDeclaration = false; } output.Write(identifier); } MemberReference GetCurrentMemberReference() { AstNode node = nodeStack.Peek(); MemberReference memberRef = node.Annotation(); if (memberRef == null && node.Role == Roles.TargetExpression && (node.Parent is InvocationExpression || node.Parent is ObjectCreateExpression)) { memberRef = node.Parent.Annotation(); } return memberRef; } object GetCurrentLocalReference() { AstNode node = nodeStack.Peek(); ILVariable variable = node.Annotation(); if (variable != null) { if (variable.OriginalParameter != null) return variable.OriginalParameter; //if (variable.OriginalVariable != null) // return variable.OriginalVariable; return variable; } var gotoStatement = node as GotoStatement; if (gotoStatement != null) { var method = nodeStack.Select(nd => nd.Annotation()).FirstOrDefault(mr => mr != null); if (method != null) return method.ToString() + gotoStatement.Label; } return null; } object GetCurrentLocalDefinition() { AstNode node = nodeStack.Peek(); var parameterDef = node.Annotation(); if (parameterDef != null) return parameterDef; if (node is VariableInitializer || node is CatchClause || node is ForeachStatement) { var variable = node.Annotation(); if (variable != null) { if (variable.OriginalParameter != null) return variable.OriginalParameter; //if (variable.OriginalVariable != null) // return variable.OriginalVariable; return variable; } else { } } var label = node as LabelStatement; if (label != null) { var method = nodeStack.Select(nd => nd.Annotation()).FirstOrDefault(mr => mr != null); if (method != null) return method.ToString() + label.Label; } return null; } object GetCurrentDefinition() { if (nodeStack == null || nodeStack.Count == 0) return null; var node = nodeStack.Peek(); if (IsDefinition(node)) return node.Annotation(); var fieldDef = node.Parent.Annotation(); if (fieldDef != null) return node.Parent.Annotation(); return null; } public void WriteKeyword(string keyword) { output.Write(keyword); } public void WriteToken(string token) { // Attach member reference to token only if there's no identifier in the current node. MemberReference memberRef = GetCurrentMemberReference(); var node = nodeStack.Peek(); if (memberRef != null && node.GetChildByRole(Roles.Identifier).IsNull) output.WriteReference(token, memberRef); else output.Write(token); } public void Space() { output.Write(' '); } public void OpenBrace(BraceStyle style) { if (braceLevelWithinType >= 0 || nodeStack.Peek() is TypeDeclaration) braceLevelWithinType++; if (nodeStack.OfType().Count() <= 1) { output.MarkFoldStart(defaultCollapsed: braceLevelWithinType == 1); } output.WriteLine(); output.WriteLine("{"); output.Indent(); } public void CloseBrace(BraceStyle style) { output.Unindent(); output.Write('}'); if (nodeStack.OfType().Count() <= 1) output.MarkFoldEnd(); if (braceLevelWithinType >= 0) braceLevelWithinType--; } public void Indent() { output.Indent(); } public void Unindent() { output.Unindent(); } public void NewLine() { if (lastUsingDeclaration) { output.MarkFoldEnd(); lastUsingDeclaration = false; } output.WriteLine(); } public void WriteComment(CommentType commentType, string content) { switch (commentType) { case CommentType.SingleLine: output.Write("//"); output.WriteLine(content); break; case CommentType.MultiLine: output.Write("/*"); output.Write(content); output.Write("*/"); break; case CommentType.Documentation: bool isLastLine = !(nodeStack.Peek().NextSibling is Comment); if (!inDocumentationComment && !isLastLine) { inDocumentationComment = true; output.MarkFoldStart("///" + content, true); } output.Write("///"); output.Write(content); if (inDocumentationComment && isLastLine) { inDocumentationComment = false; output.MarkFoldEnd(); } output.WriteLine(); break; default: output.Write(content); break; } } public void WritePreProcessorDirective(PreProcessorDirectiveType type, string argument) { // pre-processor directive must start on its own line output.Write('#'); output.Write(type.ToString().ToLowerInvariant()); if (!string.IsNullOrEmpty(argument)) { output.Write(' '); output.Write(argument); } output.WriteLine(); } Stack startLocations = new Stack(); MemberMapping currentMemberMapping; Stack parentMemberMappings = new Stack(); public void StartNode(AstNode node) { if (nodeStack.Count == 0) { if (IsUsingDeclaration(node)) { firstUsingDeclaration = !IsUsingDeclaration(node.PrevSibling); lastUsingDeclaration = !IsUsingDeclaration(node.NextSibling); } else { firstUsingDeclaration = false; lastUsingDeclaration = false; } } nodeStack.Push(node); startLocations.Push(output.Location); if (node is EntityDeclaration && node.Annotation() != null && node.GetChildByRole(Roles.Identifier).IsNull) output.WriteDefinition("", node.Annotation(), false); MemberMapping mapping = node.Annotation(); if (mapping != null) { parentMemberMappings.Push(currentMemberMapping); currentMemberMapping = mapping; } } private bool IsUsingDeclaration(AstNode node) { return node is UsingDeclaration || node is UsingAliasDeclaration; } public void EndNode(AstNode node) { if (nodeStack.Pop() != node) throw new InvalidOperationException(); var startLocation = startLocations.Pop(); // code mappings if (currentMemberMapping != null) { var ranges = node.Annotation>(); if (ranges != null && ranges.Count > 0) { // add all ranges foreach (var range in ranges) { currentMemberMapping.MemberCodeMappings.Add( new SourceCodeMapping { ILInstructionOffset = range, StartLocation = startLocation, EndLocation = output.Location, MemberMapping = currentMemberMapping }); } } } if (node.Annotation() != null) { output.AddDebuggerMemberMapping(currentMemberMapping); currentMemberMapping = parentMemberMappings.Pop(); } } private static bool IsDefinition(AstNode node) { return node is EntityDeclaration; } } }