mirror of https://github.com/icsharpcode/ILSpy.git
9 changed files with 444 additions and 337 deletions
@ -0,0 +1,42 @@ |
|||||||
|
using System; |
||||||
|
using System.Collections.Generic; |
||||||
|
using System.Diagnostics; |
||||||
|
using System.Linq; |
||||||
|
using System.Text; |
||||||
|
using System.Threading.Tasks; |
||||||
|
|
||||||
|
namespace ICSharpCode.Decompiler.IL |
||||||
|
{ |
||||||
|
partial class Branch : SimpleInstruction |
||||||
|
{ |
||||||
|
public readonly int TargetILOffset; |
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Pops the specified number of arguments from the evaluation stack during the branching operation.
|
||||||
|
/// Note that the Branch instruction does not set InstructionFlags.MayPop -- the pop instead is considered
|
||||||
|
/// to happen after the branch was taken.
|
||||||
|
/// </summary>
|
||||||
|
public int PopCount; |
||||||
|
|
||||||
|
public Branch(int targetILOffset) : base(OpCode.Branch) |
||||||
|
{ |
||||||
|
this.TargetILOffset = targetILOffset; |
||||||
|
} |
||||||
|
|
||||||
|
public string TargetLabel { |
||||||
|
get { return CecilExtensions.OffsetToString(TargetILOffset); } |
||||||
|
} |
||||||
|
|
||||||
|
public override void WriteTo(ITextOutput output) |
||||||
|
{ |
||||||
|
output.Write(OpCode); |
||||||
|
output.Write(' '); |
||||||
|
output.WriteReference(TargetLabel, TargetILOffset, isLocal: true); |
||||||
|
if (PopCount != 0) { |
||||||
|
output.Write(" (pop "); |
||||||
|
output.Write(PopCount.ToString()); |
||||||
|
output.Write(')'); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -1,156 +0,0 @@ |
|||||||
using System; |
|
||||||
using System.Collections.Generic; |
|
||||||
using System.Diagnostics; |
|
||||||
using System.Linq; |
|
||||||
using System.Text; |
|
||||||
using System.Threading.Tasks; |
|
||||||
|
|
||||||
namespace ICSharpCode.Decompiler.IL |
|
||||||
{ |
|
||||||
/// <summary>
|
|
||||||
/// Base class for unconditional and conditional branches.
|
|
||||||
/// </summary>
|
|
||||||
public abstract class BranchInstruction : ILInstruction |
|
||||||
{ |
|
||||||
public readonly int TargetILOffset; |
|
||||||
|
|
||||||
protected BranchInstruction(OpCode opCode, int targetILOffset) : base(opCode) |
|
||||||
{ |
|
||||||
this.TargetILOffset = targetILOffset; |
|
||||||
} |
|
||||||
|
|
||||||
public string TargetLabel { |
|
||||||
get { return CecilExtensions.OffsetToString(TargetILOffset); } |
|
||||||
} |
|
||||||
|
|
||||||
public override void WriteTo(ITextOutput output) |
|
||||||
{ |
|
||||||
output.Write(OpCode.ToString()); |
|
||||||
output.Write(' '); |
|
||||||
output.WriteReference(TargetLabel, TargetILOffset, isLocal: true); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
partial class Branch |
|
||||||
{ |
|
||||||
public override TAccumulate AggregateChildren<TSource, TAccumulate>(TAccumulate initial, ILVisitor<TSource> visitor, Func<TAccumulate, TSource, TAccumulate> func) |
|
||||||
{ |
|
||||||
return initial; |
|
||||||
} |
|
||||||
|
|
||||||
public override void TransformChildren(ILVisitor<ILInstruction> visitor) |
|
||||||
{ |
|
||||||
} |
|
||||||
|
|
||||||
internal override ILInstruction Inline(InstructionFlags flagsBefore, Stack<ILInstruction> instructionStack, out bool finished) |
|
||||||
{ |
|
||||||
finished = true; |
|
||||||
return this; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
partial class ConditionalBranch |
|
||||||
{ |
|
||||||
ILInstruction condition; |
|
||||||
|
|
||||||
public ConditionalBranch(ILInstruction condition, int targetILOffset) : base(OpCode.ConditionalBranch, targetILOffset) |
|
||||||
{ |
|
||||||
this.Condition = condition; |
|
||||||
} |
|
||||||
|
|
||||||
public ILInstruction Condition { |
|
||||||
get { |
|
||||||
return condition; |
|
||||||
} |
|
||||||
set { |
|
||||||
ValidateArgument(value); |
|
||||||
SetChildInstruction(ref condition, value); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
public override TAccumulate AggregateChildren<TSource, TAccumulate>(TAccumulate initial, ILVisitor<TSource> visitor, Func<TAccumulate, TSource, TAccumulate> func) |
|
||||||
{ |
|
||||||
return func(initial, condition.AcceptVisitor(visitor)); |
|
||||||
} |
|
||||||
|
|
||||||
public override void TransformChildren(ILVisitor<ILInstruction> visitor) |
|
||||||
{ |
|
||||||
this.Condition = condition.AcceptVisitor(visitor); |
|
||||||
} |
|
||||||
|
|
||||||
internal override ILInstruction Inline(InstructionFlags flagsBefore, Stack<ILInstruction> instructionStack, out bool finished) |
|
||||||
{ |
|
||||||
this.Condition = condition.Inline(flagsBefore, instructionStack, out finished); |
|
||||||
return this; |
|
||||||
} |
|
||||||
|
|
||||||
protected override InstructionFlags ComputeFlags() |
|
||||||
{ |
|
||||||
return condition.Flags | InstructionFlags.MayBranch; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/* |
|
||||||
/// <summary>
|
|
||||||
/// Special instruction for unresolved branches.
|
|
||||||
/// Created by ILReader phase, replaced with TODO when building basic blocks.
|
|
||||||
/// </summary>
|
|
||||||
class ConditionalBranch(public ILInstruction Condition, int targetILOffset) : Branch(OpCode.ConditionalBranch, targetILOffset) |
|
||||||
{ |
|
||||||
public override bool IsPeeking { get { return Condition.IsPeeking; } } |
|
||||||
|
|
||||||
public override void WriteTo(ITextOutput output) |
|
||||||
{ |
|
||||||
base.WriteTo(output); |
|
||||||
output.Write('('); |
|
||||||
Condition.WriteTo(output); |
|
||||||
output.Write(')'); |
|
||||||
} |
|
||||||
|
|
||||||
public override void TransformChildren(Func<ILInstruction, ILInstruction> transformFunc) |
|
||||||
{ |
|
||||||
Condition = transformFunc(Condition); |
|
||||||
} |
|
||||||
|
|
||||||
internal override ILInstruction Inline(InstructionFlags flagsBefore, Stack<ILInstruction> instructionStack, out bool finished) |
|
||||||
{ |
|
||||||
Condition = Condition.Inline(flagsBefore, instructionStack, out finished); |
|
||||||
return this; |
|
||||||
} |
|
||||||
|
|
||||||
public override InstructionFlags Flags |
|
||||||
{ |
|
||||||
get { return InstructionFlags.MayJump | Condition.Flags; } |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
class ReturnVoidInstruction() : SimpleInstruction(OpCode.Ret) |
|
||||||
{ |
|
||||||
public override bool IsEndReachable { get { return false; } } |
|
||||||
|
|
||||||
public override InstructionFlags Flags |
|
||||||
{ |
|
||||||
get { return InstructionFlags.MayJump; } |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
class ReturnInstruction() : UnaryInstruction(OpCode.Ret) |
|
||||||
{ |
|
||||||
public override bool IsEndReachable { get { return false; } } |
|
||||||
|
|
||||||
public override InstructionFlags Flags |
|
||||||
{ |
|
||||||
get { return InstructionFlags.MayJump | Operand.Flags; } |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
class ThrowInstruction() : UnaryInstruction(OpCode.Throw) |
|
||||||
{ |
|
||||||
public override bool IsEndReachable { get { return false; } } |
|
||||||
|
|
||||||
public override InstructionFlags Flags |
|
||||||
{ |
|
||||||
get { return InstructionFlags.MayJump | Operand.Flags; } |
|
||||||
} |
|
||||||
}*/ |
|
||||||
} |
|
@ -0,0 +1,116 @@ |
|||||||
|
// Copyright (c) 2014 Daniel Grunwald
|
||||||
|
//
|
||||||
|
// 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; |
||||||
|
|
||||||
|
namespace ICSharpCode.Decompiler.IL |
||||||
|
{ |
||||||
|
partial class IfInstruction : ILInstruction |
||||||
|
{ |
||||||
|
ILInstruction condition; |
||||||
|
ILInstruction trueInst; |
||||||
|
ILInstruction falseInst; |
||||||
|
|
||||||
|
public IfInstruction(ILInstruction condition, ILInstruction trueInst, ILInstruction falseInst = null) : base(OpCode.IfInstruction) |
||||||
|
{ |
||||||
|
this.Condition = condition; |
||||||
|
this.TrueInst = trueInst; |
||||||
|
this.FalseInst = falseInst ?? new Nop(); |
||||||
|
} |
||||||
|
|
||||||
|
public ILInstruction Condition { |
||||||
|
get { return condition; } |
||||||
|
set { |
||||||
|
ValidateArgument(value); |
||||||
|
SetChildInstruction(ref condition, value); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
public ILInstruction TrueInst { |
||||||
|
get { return trueInst; } |
||||||
|
set { |
||||||
|
if (value == null) |
||||||
|
throw new ArgumentNullException(); |
||||||
|
SetChildInstruction(ref trueInst, value); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
public ILInstruction FalseInst { |
||||||
|
get { return falseInst; } |
||||||
|
set { |
||||||
|
if (value == null) |
||||||
|
throw new ArgumentNullException(); |
||||||
|
SetChildInstruction(ref falseInst, value); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
public override StackType ResultType { |
||||||
|
get { |
||||||
|
return trueInst.ResultType; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
public override TAccumulate AggregateChildren<TSource, TAccumulate>(TAccumulate initial, ILVisitor<TSource> visitor, Func<TAccumulate, TSource, TAccumulate> func) |
||||||
|
{ |
||||||
|
TAccumulate value = initial; |
||||||
|
value = func(value, condition.AcceptVisitor(visitor)); |
||||||
|
value = func(value, trueInst.AcceptVisitor(visitor)); |
||||||
|
value = func(value, falseInst.AcceptVisitor(visitor)); |
||||||
|
return value; |
||||||
|
} |
||||||
|
|
||||||
|
public override void TransformChildren(ILVisitor<ILInstruction> visitor) |
||||||
|
{ |
||||||
|
this.Condition = condition.AcceptVisitor(visitor); |
||||||
|
this.TrueInst = trueInst.AcceptVisitor(visitor); |
||||||
|
this.FalseInst = falseInst.AcceptVisitor(visitor); |
||||||
|
} |
||||||
|
|
||||||
|
internal override ILInstruction Inline(InstructionFlags flagsBefore, Stack<ILInstruction> instructionStack, out bool finished) |
||||||
|
{ |
||||||
|
this.Condition = condition.Inline(flagsBefore, instructionStack, out finished); |
||||||
|
// don't continue inlining if this instruction still contains peek/pop instructions
|
||||||
|
if (HasFlag(InstructionFlags.MayPeek | InstructionFlags.MayPop)) |
||||||
|
finished = false; |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
protected override InstructionFlags ComputeFlags() |
||||||
|
{ |
||||||
|
// the endpoint of the 'if' is only unreachable if both branches have an unreachable endpoint
|
||||||
|
const InstructionFlags combineWithAnd = InstructionFlags.EndPointUnreachable; |
||||||
|
InstructionFlags trueFlags = trueInst.Flags; |
||||||
|
InstructionFlags falseFlags = falseInst.Flags; |
||||||
|
return condition.Flags | (trueFlags & falseFlags) | ((trueFlags | falseFlags) & ~combineWithAnd); |
||||||
|
} |
||||||
|
|
||||||
|
public override void WriteTo(ITextOutput output) |
||||||
|
{ |
||||||
|
output.Write(OpCode); |
||||||
|
output.Write(" ("); |
||||||
|
condition.WriteTo(output); |
||||||
|
output.Write(") "); |
||||||
|
trueInst.WriteTo(output); |
||||||
|
if (falseInst.OpCode != OpCode.Nop) { |
||||||
|
output.Write(" else "); |
||||||
|
falseInst.WriteTo(output); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue