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.
473 lines
12 KiB
473 lines
12 KiB
// 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.CSharp; |
|
using ICSharpCode.Decompiler.CSharp.OutputVisitor; |
|
using ICSharpCode.Decompiler.CSharp.Resolver; |
|
using ICSharpCode.Decompiler.CSharp.Syntax; |
|
using ICSharpCode.Decompiler.IL; |
|
using ICSharpCode.Decompiler.Semantics; |
|
using ICSharpCode.Decompiler.TypeSystem; |
|
using ICSharpCode.Decompiler.TypeSystem.Implementation; |
|
|
|
namespace ICSharpCode.Decompiler |
|
{ |
|
public class TextTokenWriter : TokenWriter |
|
{ |
|
readonly ITextOutput output; |
|
readonly DecompilerSettings settings; |
|
readonly IDecompilerTypeSystem typeSystem; |
|
readonly Stack<AstNode> nodeStack = new Stack<AstNode>(); |
|
int braceLevelWithinType = -1; |
|
bool inDocumentationComment = false; |
|
bool firstUsingDeclaration; |
|
bool lastUsingDeclaration; |
|
|
|
public TextTokenWriter(ITextOutput output, DecompilerSettings settings, IDecompilerTypeSystem typeSystem) |
|
{ |
|
if (output == null) |
|
throw new ArgumentNullException(nameof(output)); |
|
if (settings == null) |
|
throw new ArgumentNullException(nameof(settings)); |
|
if (typeSystem == null) |
|
throw new ArgumentNullException(nameof(typeSystem)); |
|
this.output = output; |
|
this.settings = settings; |
|
this.typeSystem = typeSystem; |
|
} |
|
|
|
public override void WriteIdentifier(Identifier identifier) |
|
{ |
|
if (identifier.IsVerbatim || CSharpOutputVisitor.IsKeyword(identifier.Name, identifier)) |
|
{ |
|
output.Write('@'); |
|
} |
|
|
|
var definition = GetCurrentDefinition(); |
|
string name = TextWriterTokenWriter.EscapeIdentifier(identifier.Name); |
|
switch (definition) |
|
{ |
|
case IType t: |
|
output.WriteReference(t, name, true); |
|
return; |
|
case IMember m: |
|
output.WriteReference(m, name, true); |
|
return; |
|
} |
|
|
|
var member = GetCurrentMemberReference(); |
|
switch (member) |
|
{ |
|
case IType t: |
|
output.WriteReference(t, name, false); |
|
return; |
|
case IMember m: |
|
output.WriteReference(m, name, false); |
|
return; |
|
} |
|
|
|
var localDefinition = GetCurrentLocalDefinition(identifier); |
|
if (localDefinition != null) |
|
{ |
|
output.WriteLocalReference(name, localDefinition, isDefinition: true); |
|
return; |
|
} |
|
|
|
var localRef = GetCurrentLocalReference(); |
|
if (localRef != null) |
|
{ |
|
output.WriteLocalReference(name, localRef); |
|
return; |
|
} |
|
|
|
if (firstUsingDeclaration && !lastUsingDeclaration) |
|
{ |
|
output.MarkFoldStart(defaultCollapsed: !settings.ExpandUsingDeclarations); |
|
firstUsingDeclaration = false; |
|
} |
|
|
|
output.Write(name); |
|
} |
|
|
|
ISymbol GetCurrentMemberReference() |
|
{ |
|
AstNode node = nodeStack.Peek(); |
|
var symbol = node.GetSymbol(); |
|
if (symbol == null && node.Role == Roles.TargetExpression && node.Parent is InvocationExpression) |
|
{ |
|
symbol = node.Parent.GetSymbol(); |
|
} |
|
if (symbol != null && node.Role == Roles.Type && node.Parent is ObjectCreateExpression) |
|
{ |
|
symbol = node.Parent.GetSymbol(); |
|
} |
|
|
|
if (node is IdentifierExpression && node.Role == Roles.TargetExpression && node.Parent is InvocationExpression && symbol is IMember member) |
|
{ |
|
var declaringType = member.DeclaringType; |
|
if (declaringType != null && declaringType.Kind == TypeKind.Delegate) |
|
return null; |
|
} |
|
return FilterMember(symbol); |
|
} |
|
|
|
ISymbol FilterMember(ISymbol symbol) |
|
{ |
|
if (symbol == null) |
|
return null; |
|
|
|
if (symbol is LocalFunctionMethod) |
|
return null; |
|
|
|
return symbol; |
|
} |
|
|
|
object GetCurrentLocalReference() |
|
{ |
|
AstNode node = nodeStack.Peek(); |
|
ILVariable variable = node.Annotation<ILVariableResolveResult>()?.Variable; |
|
if (variable != null) |
|
return variable; |
|
|
|
var letClauseVariable = node.Annotation<CSharp.Transforms.LetIdentifierAnnotation>(); |
|
if (letClauseVariable != null) |
|
return letClauseVariable; |
|
|
|
if (node is GotoStatement gotoStatement) |
|
{ |
|
var method = nodeStack.Select(nd => nd.GetSymbol() as IMethod).FirstOrDefault(mr => mr != null); |
|
if (method != null) |
|
return method + gotoStatement.Label; |
|
} |
|
|
|
if (node.Role == Roles.TargetExpression && node.Parent is InvocationExpression) |
|
{ |
|
var symbol = node.Parent.GetSymbol(); |
|
if (symbol is LocalFunctionMethod) |
|
return symbol; |
|
} |
|
|
|
return null; |
|
} |
|
|
|
object GetCurrentLocalDefinition(Identifier id) |
|
{ |
|
AstNode node = nodeStack.Peek(); |
|
if (node is Identifier && node.Parent != null) |
|
node = node.Parent; |
|
|
|
if (node is ParameterDeclaration || node is VariableInitializer || node is CatchClause || node is VariableDesignation) |
|
{ |
|
var variable = node.Annotation<ILVariableResolveResult>()?.Variable; |
|
if (variable != null) |
|
return variable; |
|
} |
|
|
|
if (id.Role == QueryJoinClause.IntoIdentifierRole || id.Role == QueryJoinClause.JoinIdentifierRole) |
|
{ |
|
var variable = id.Annotation<ILVariableResolveResult>()?.Variable; |
|
if (variable != null) |
|
return variable; |
|
} |
|
|
|
if (node is QueryLetClause) |
|
{ |
|
var variable = node.Annotation<CSharp.Transforms.LetIdentifierAnnotation>(); |
|
if (variable != null) |
|
return variable; |
|
} |
|
|
|
if (node is LabelStatement label) |
|
{ |
|
var method = nodeStack.Select(nd => nd.GetSymbol() as IMethod).FirstOrDefault(mr => mr != null); |
|
if (method != null) |
|
return method + label.Label; |
|
} |
|
|
|
if (node is MethodDeclaration && node.Parent is LocalFunctionDeclarationStatement) |
|
{ |
|
var localFunction = node.Parent.GetResolveResult() as MemberResolveResult; |
|
if (localFunction != null) |
|
return localFunction.Member; |
|
} |
|
|
|
return null; |
|
} |
|
|
|
ISymbol GetCurrentDefinition() |
|
{ |
|
if (nodeStack == null || nodeStack.Count == 0) |
|
return null; |
|
|
|
var node = nodeStack.Peek(); |
|
if (node is Identifier) |
|
node = node.Parent; |
|
if (IsDefinition(ref node)) |
|
return node.GetSymbol(); |
|
|
|
return null; |
|
} |
|
|
|
public override void WriteKeyword(Role role, string keyword) |
|
{ |
|
//To make reference for 'this' and 'base' keywords in the ClassName():this() expression |
|
if (role == ConstructorInitializer.ThisKeywordRole || role == ConstructorInitializer.BaseKeywordRole) |
|
{ |
|
if (nodeStack.Peek() is ConstructorInitializer initializer && initializer.GetSymbol() is IMember member) |
|
{ |
|
output.WriteReference(member, keyword); |
|
return; |
|
} |
|
} |
|
output.Write(keyword); |
|
} |
|
|
|
public override void WriteToken(Role role, string token) |
|
{ |
|
switch (token) |
|
{ |
|
case "{": |
|
if (role != Roles.LBrace) |
|
{ |
|
output.Write("{"); |
|
break; |
|
} |
|
if (braceLevelWithinType >= 0 || nodeStack.Peek() is TypeDeclaration) |
|
braceLevelWithinType++; |
|
if (nodeStack.OfType<BlockStatement>().Count() <= 1 || settings.FoldBraces) |
|
{ |
|
output.MarkFoldStart(defaultCollapsed: !settings.ExpandMemberDefinitions && braceLevelWithinType == 1, isDefinition: braceLevelWithinType == 1); |
|
} |
|
output.Write("{"); |
|
break; |
|
case "}": |
|
output.Write('}'); |
|
if (role != Roles.RBrace) |
|
break; |
|
if (nodeStack.OfType<BlockStatement>().Count() <= 1 || settings.FoldBraces) |
|
output.MarkFoldEnd(); |
|
if (braceLevelWithinType >= 0) |
|
braceLevelWithinType--; |
|
break; |
|
default: |
|
// Attach member reference to token only if there's no identifier in the current node. |
|
var member = GetCurrentMemberReference(); |
|
var node = nodeStack.Peek(); |
|
if (member != null && node.GetChildByRole(Roles.Identifier).IsNull) |
|
{ |
|
switch (member) |
|
{ |
|
case IType t: |
|
output.WriteReference(t, token, false); |
|
return; |
|
case IMember m: |
|
output.WriteReference(m, token, false); |
|
return; |
|
} |
|
} |
|
else |
|
output.Write(token); |
|
break; |
|
} |
|
} |
|
|
|
public override void Space() |
|
{ |
|
output.Write(' '); |
|
} |
|
|
|
public override void Indent() |
|
{ |
|
output.Indent(); |
|
} |
|
|
|
public override void Unindent() |
|
{ |
|
output.Unindent(); |
|
} |
|
|
|
public override void NewLine() |
|
{ |
|
if (!firstUsingDeclaration && lastUsingDeclaration) |
|
{ |
|
output.MarkFoldEnd(); |
|
lastUsingDeclaration = false; |
|
} |
|
output.WriteLine(); |
|
} |
|
|
|
public override 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 override 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(); |
|
} |
|
|
|
public override void WritePrimitiveValue(object value, LiteralFormat format = LiteralFormat.None) |
|
{ |
|
new TextWriterTokenWriter(new TextOutputWriter(output)).WritePrimitiveValue(value, format); |
|
} |
|
|
|
public override void WriteInterpolatedText(string text) |
|
{ |
|
output.Write(TextWriterTokenWriter.ConvertString(text)); |
|
} |
|
|
|
public override void WritePrimitiveType(string type) |
|
{ |
|
switch (type) |
|
{ |
|
case "new": |
|
output.Write(type); |
|
output.Write("()"); |
|
break; |
|
case "bool": |
|
case "byte": |
|
case "sbyte": |
|
case "short": |
|
case "ushort": |
|
case "int": |
|
case "uint": |
|
case "long": |
|
case "ulong": |
|
case "float": |
|
case "double": |
|
case "decimal": |
|
case "char": |
|
case "string": |
|
case "object": |
|
var node = nodeStack.Peek(); |
|
ISymbol symbol; |
|
if (node.Role == Roles.Type && node.Parent is ObjectCreateExpression) |
|
{ |
|
symbol = node.Parent.GetSymbol(); |
|
} |
|
else |
|
{ |
|
symbol = nodeStack.Peek().GetSymbol(); |
|
} |
|
if (symbol == null) |
|
goto default; |
|
switch (symbol) |
|
{ |
|
case IType t: |
|
output.WriteReference(t, type, false); |
|
return; |
|
case IMember m: |
|
output.WriteReference(m, type, false); |
|
return; |
|
} |
|
break; |
|
default: |
|
output.Write(type); |
|
break; |
|
} |
|
} |
|
|
|
public override 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); |
|
} |
|
|
|
private bool IsUsingDeclaration(AstNode node) |
|
{ |
|
return node is UsingDeclaration || node is UsingAliasDeclaration; |
|
} |
|
|
|
public override void EndNode(AstNode node) |
|
{ |
|
if (nodeStack.Pop() != node) |
|
throw new InvalidOperationException(); |
|
} |
|
|
|
public static bool IsDefinition(ref AstNode node) |
|
{ |
|
if (node is EntityDeclaration && !(node.Parent is LocalFunctionDeclarationStatement)) |
|
return true; |
|
if (node is VariableInitializer && node.Parent is FieldDeclaration or EventDeclaration) |
|
{ |
|
node = node.Parent; |
|
return true; |
|
} |
|
if (node is FixedVariableInitializer && node.Parent is FixedFieldDeclaration) |
|
{ |
|
node = node.Parent; |
|
return true; |
|
} |
|
return false; |
|
} |
|
} |
|
}
|
|
|