mirror of https://github.com/icsharpcode/ILSpy.git
88 changed files with 2579 additions and 713 deletions
@ -1,76 +0,0 @@
@@ -1,76 +0,0 @@
|
||||
using System; |
||||
using System.Collections.Generic; |
||||
using System.Linq; |
||||
using System.Text; |
||||
using System.Threading.Tasks; |
||||
|
||||
namespace LocalFunctions |
||||
{ |
||||
class LocalFunctions |
||||
{ |
||||
int field; |
||||
|
||||
public static void Main(string[] args) |
||||
{ |
||||
StaticContextNoCapture(10); |
||||
StaticContextSimpleCapture(10); |
||||
StaticContextCaptureInForLoop(10); |
||||
var inst = new LocalFunctions() { field = 10 }; |
||||
inst.ContextNoCapture(); |
||||
inst.ContextSimpleCapture(); |
||||
inst.ContextCaptureInForLoop(); |
||||
} |
||||
|
||||
public static void StaticContextNoCapture(int length) |
||||
{ |
||||
for (int i = 0; i < length; i++) { |
||||
LocalWrite("Hello " + i); |
||||
} |
||||
|
||||
void LocalWrite(string s) => Console.WriteLine(s); |
||||
} |
||||
|
||||
public static void StaticContextSimpleCapture(int length) |
||||
{ |
||||
for (int i = 0; i < length; i++) { |
||||
LocalWrite(); |
||||
} |
||||
|
||||
void LocalWrite() => Console.WriteLine("Hello " + length); |
||||
} |
||||
|
||||
public static void StaticContextCaptureInForLoop(int length) |
||||
{ |
||||
for (int i = 0; i < length; i++) { |
||||
void LocalWrite() => Console.WriteLine("Hello " + i + "/" + length); |
||||
LocalWrite(); |
||||
} |
||||
} |
||||
|
||||
public void ContextNoCapture() |
||||
{ |
||||
for (int i = 0; i < field; i++) { |
||||
LocalWrite("Hello " + i); |
||||
} |
||||
|
||||
void LocalWrite(string s) => Console.WriteLine(s); |
||||
} |
||||
|
||||
public void ContextSimpleCapture() |
||||
{ |
||||
for (int i = 0; i < field; i++) { |
||||
LocalWrite(); |
||||
} |
||||
|
||||
void LocalWrite() => Console.WriteLine("Hello " + field); |
||||
} |
||||
|
||||
public void ContextCaptureInForLoop() |
||||
{ |
||||
for (int i = 0; i < field; i++) { |
||||
void LocalWrite() => Console.WriteLine("Hello " + i + "/" + field); |
||||
LocalWrite(); |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,14 @@
@@ -0,0 +1,14 @@
|
||||
using System; |
||||
|
||||
public sealed class TestClass : IDisposable |
||||
{ |
||||
void IDisposable.Dispose() |
||||
{ |
||||
} |
||||
|
||||
public void Test(TestClass other) |
||||
{ |
||||
((IDisposable)this).Dispose(); |
||||
((IDisposable)other).Dispose(); |
||||
} |
||||
} |
@ -0,0 +1,36 @@
@@ -0,0 +1,36 @@
|
||||
.assembly extern mscorlib |
||||
{ |
||||
.publickeytoken = (B7 7A 5C 56 19 34 E0 89 ) // .z\V.4.. |
||||
.ver 4:0:0:0 |
||||
} |
||||
.assembly DirectCallToExplicitInterfaceImpl |
||||
{ |
||||
.ver 1:0:0:0 |
||||
} |
||||
.module DirectCallToExplicitInterfaceImpl.exe |
||||
|
||||
.class public auto ansi sealed TestClass |
||||
extends [mscorlib]System.Object |
||||
implements [mscorlib]System.IDisposable |
||||
{ |
||||
// Methods |
||||
|
||||
.method private final hidebysig newslot virtual |
||||
instance void System.IDisposable.Dispose () cil managed |
||||
{ |
||||
.override method instance void [mscorlib]System.IDisposable::Dispose() |
||||
ret |
||||
} |
||||
|
||||
.method public hidebysig void Test (class TestClass other) cil managed |
||||
{ |
||||
ldarg.0 |
||||
call instance void TestClass::System.IDisposable.Dispose() |
||||
|
||||
ldarg.1 |
||||
call instance void TestClass::System.IDisposable.Dispose() |
||||
|
||||
ret |
||||
} |
||||
|
||||
} // end of class TestClass |
@ -0,0 +1,39 @@
@@ -0,0 +1,39 @@
|
||||
using System; |
||||
|
||||
public static class Issue684 |
||||
{ |
||||
static int Main(string[] A_0) |
||||
{ |
||||
int[] array = new int[1000]; |
||||
int num = int.Parse(Console.ReadLine()); |
||||
// Point of this test was to ensure the stack slot here uses an appropriate type,
|
||||
// (bool instead of int). Unfortunately our type fixup runs too late to affect variable names.
|
||||
bool num2 = num >= 1000; |
||||
if (!num2) { |
||||
num2 = (num < 2); |
||||
} |
||||
if (num2) { |
||||
Console.WriteLine(-1); |
||||
} else { |
||||
int i = 2; |
||||
for (int num3 = 2; num3 <= num; num3 = i) { |
||||
Console.WriteLine(num3); |
||||
for (; i <= num; i += num3) { |
||||
int num4 = array[i] = 1; |
||||
} |
||||
i = num3; |
||||
while (true) { |
||||
bool num5 = i <= num; |
||||
if (num5) { |
||||
num5 = (array[i] != 0); |
||||
} |
||||
if (!num5) { |
||||
break; |
||||
} |
||||
i++; |
||||
} |
||||
} |
||||
} |
||||
return 0; |
||||
} |
||||
} |
@ -0,0 +1,133 @@
@@ -0,0 +1,133 @@
|
||||
.assembly extern mscorlib |
||||
{ |
||||
.publickeytoken = (B7 7A 5C 56 19 34 E0 89 ) // .z\V.4.. |
||||
.ver 4:0:0:0 |
||||
} |
||||
.assembly Issue684 |
||||
{ |
||||
.ver 1:0:0:0 |
||||
} |
||||
.module Issue684.exe |
||||
|
||||
.class public auto ansi abstract sealed Issue684 |
||||
extends [mscorlib]System.Object |
||||
{ |
||||
// Methods |
||||
|
||||
.method static privatescope |
||||
int32 Main$PST06000001 ( |
||||
string[] '' |
||||
) cil managed |
||||
{ |
||||
// Method begins at RVA 0x2050 |
||||
// Code size 196 (0xc4) |
||||
.maxstack 11 |
||||
.entrypoint |
||||
.locals init ( |
||||
[0] int32, |
||||
[1] int32, |
||||
[2] int32, |
||||
[3] int32[], |
||||
[4] int32 |
||||
) |
||||
|
||||
IL_0000: ldc.i4 1000 |
||||
IL_0005: newarr [mscorlib]System.Int32 |
||||
IL_000a: stloc.3 |
||||
IL_000b: call string [mscorlib]System.Console::ReadLine() |
||||
IL_0010: call int32 [mscorlib]System.Int32::Parse(string) |
||||
IL_0015: stloc.2 |
||||
IL_0016: ldloc.2 |
||||
IL_0017: ldc.i4 1000 |
||||
IL_001c: clt |
||||
IL_001e: ldc.i4.0 |
||||
IL_001f: ceq |
||||
IL_0021: dup |
||||
IL_0022: brtrue IL_0030 |
||||
|
||||
IL_0027: pop |
||||
IL_0028: ldloc.2 |
||||
IL_0029: ldc.i4 2 |
||||
IL_002e: clt |
||||
|
||||
IL_0030: brfalse IL_0045 |
||||
|
||||
IL_0035: ldc.i4 1 |
||||
IL_003a: neg |
||||
IL_003b: call void [mscorlib]System.Console::WriteLine(int32) |
||||
IL_0040: br IL_00c2 |
||||
|
||||
IL_0045: ldc.i4 2 |
||||
IL_004a: stloc.0 |
||||
IL_004b: ldc.i4 2 |
||||
IL_0050: stloc.1 |
||||
// loop start (head: IL_0051) |
||||
IL_0051: ldloc.1 |
||||
IL_0052: ldloc.2 |
||||
IL_0053: cgt |
||||
IL_0055: ldc.i4.0 |
||||
IL_0056: ceq |
||||
IL_0058: brfalse IL_00c2 |
||||
|
||||
IL_005d: ldloc.1 |
||||
IL_005e: call void [mscorlib]System.Console::WriteLine(int32) |
||||
// loop start (head: IL_0063) |
||||
IL_0063: ldloc.0 |
||||
IL_0064: ldloc.2 |
||||
IL_0065: cgt |
||||
IL_0067: ldc.i4.0 |
||||
IL_0068: ceq |
||||
IL_006a: brfalse IL_0088 |
||||
|
||||
IL_006f: ldc.i4 1 |
||||
IL_0074: stloc.s 4 |
||||
IL_0076: ldloc.3 |
||||
IL_0077: ldloc.0 |
||||
IL_0078: ldloc.s 4 |
||||
IL_007a: stelem.any [mscorlib]System.Int32 |
||||
IL_007f: ldloc.0 |
||||
IL_0080: ldloc.1 |
||||
IL_0081: add |
||||
IL_0082: stloc.0 |
||||
IL_0083: br IL_0063 |
||||
// end loop |
||||
|
||||
IL_0088: ldloc.1 |
||||
IL_0089: stloc.0 |
||||
// loop start (head: IL_008a) |
||||
IL_008a: ldloc.0 |
||||
IL_008b: ldloc.2 |
||||
IL_008c: cgt |
||||
IL_008e: ldc.i4.0 |
||||
IL_008f: ceq |
||||
IL_0091: dup |
||||
IL_0092: brfalse IL_00a9 |
||||
|
||||
IL_0097: pop |
||||
IL_0098: ldloc.3 |
||||
IL_0099: ldloc.0 |
||||
IL_009a: ldelem.any [mscorlib]System.Int32 |
||||
IL_009f: ldc.i4 0 |
||||
IL_00a4: ceq |
||||
IL_00a6: ldc.i4.0 |
||||
IL_00a7: ceq |
||||
|
||||
IL_00a9: brfalse IL_00bb |
||||
|
||||
IL_00ae: ldloc.0 |
||||
IL_00af: ldc.i4 1 |
||||
IL_00b4: add |
||||
IL_00b5: stloc.0 |
||||
IL_00b6: br IL_008a |
||||
// end loop |
||||
|
||||
IL_00bb: ldloc.0 |
||||
IL_00bc: stloc.1 |
||||
IL_00bd: br IL_0051 |
||||
// end loop |
||||
|
||||
IL_00c2: ldc.i4.0 |
||||
IL_00c3: ret |
||||
} // end of method Program::Main |
||||
|
||||
} // end of class Issue684 |
@ -0,0 +1,288 @@
@@ -0,0 +1,288 @@
|
||||
// Copyright (c) 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; |
||||
|
||||
namespace LocalFunctions |
||||
{ |
||||
internal class LocalFunctions |
||||
{ |
||||
private int field; |
||||
|
||||
private static void Test(int x) |
||||
{ |
||||
} |
||||
|
||||
private static int GetInt(string a) |
||||
{ |
||||
return a.Length; |
||||
} |
||||
|
||||
private static string GetString(int a) |
||||
{ |
||||
return a.ToString(); |
||||
} |
||||
|
||||
public static void StaticContextNoCapture(int length) |
||||
{ |
||||
for (int i = 0; i < length; i++) { |
||||
LocalWrite("Hello " + i); |
||||
} |
||||
|
||||
void LocalWrite(string s) |
||||
{ |
||||
Console.WriteLine(s); |
||||
} |
||||
} |
||||
|
||||
public static void StaticContextSimpleCapture(int length) |
||||
{ |
||||
for (int i = 0; i < length; i++) { |
||||
LocalWrite(); |
||||
} |
||||
|
||||
void LocalWrite() |
||||
{ |
||||
Console.WriteLine("Hello " + length); |
||||
} |
||||
} |
||||
|
||||
public static void StaticContextCaptureForLoopVariable(int length) |
||||
{ |
||||
int i; |
||||
for (i = 0; i < length; i++) { |
||||
LocalWrite(); |
||||
} |
||||
void LocalWrite() |
||||
{ |
||||
Console.WriteLine("Hello " + i + "/" + length); |
||||
} |
||||
} |
||||
|
||||
public void ContextNoCapture() |
||||
{ |
||||
for (int i = 0; i < field; i++) { |
||||
LocalWrite("Hello " + i); |
||||
} |
||||
|
||||
void LocalWrite(string s) |
||||
{ |
||||
Console.WriteLine(s); |
||||
} |
||||
} |
||||
|
||||
public void ContextSimpleCapture() |
||||
{ |
||||
for (int i = 0; i < field; i++) { |
||||
LocalWrite(); |
||||
} |
||||
|
||||
void LocalWrite() |
||||
{ |
||||
Console.WriteLine("Hello " + field); |
||||
} |
||||
} |
||||
|
||||
public void ContextCaptureForLoopVariable() |
||||
{ |
||||
int i; |
||||
for (i = 0; i < field; i++) { |
||||
LocalWrite(); |
||||
} |
||||
void LocalWrite() |
||||
{ |
||||
Console.WriteLine("Hello " + i + "/" + field); |
||||
} |
||||
} |
||||
|
||||
public void CapturedOutsideLoop() |
||||
{ |
||||
int i = 0; |
||||
while (i < field) { |
||||
i = GetInt("asdf"); |
||||
LocalWrite(); |
||||
} |
||||
|
||||
void LocalWrite() |
||||
{ |
||||
Console.WriteLine("Hello " + i + "/" + field); |
||||
} |
||||
} |
||||
|
||||
public void CapturedInForeachLoop(IEnumerable<string> args) |
||||
{ |
||||
foreach (string arg2 in args) { |
||||
string arg = arg2; |
||||
LocalWrite(); |
||||
void LocalWrite() |
||||
{ |
||||
Console.WriteLine("Hello " + arg); |
||||
} |
||||
} |
||||
} |
||||
|
||||
public void Overloading() |
||||
{ |
||||
Test(5); |
||||
LocalFunctions.Test(2); |
||||
|
||||
void Test(int x) |
||||
{ |
||||
Console.WriteLine("x: {0}", x); |
||||
} |
||||
} |
||||
|
||||
private void Name() |
||||
{ |
||||
|
||||
} |
||||
|
||||
private void LocalFunctionHidingMethod() |
||||
{ |
||||
Action action = this.Name; |
||||
Name(); |
||||
action(); |
||||
|
||||
void Name() |
||||
{ |
||||
|
||||
} |
||||
} |
||||
|
||||
public void NamedArgument() |
||||
{ |
||||
Use(Get(1), Get(2), Get(3)); |
||||
Use(Get(1), c: Get(2), b: Get(3)); |
||||
|
||||
int Get(int i) |
||||
{ |
||||
return i; |
||||
} |
||||
|
||||
void Use(int a, int b, int c) |
||||
{ |
||||
Console.WriteLine(a + b + c); |
||||
} |
||||
} |
||||
|
||||
public static Func<int> LambdaInLocalFunction() |
||||
{ |
||||
int x = (int)Math.Pow(2.0, 10.0); |
||||
return Create(); |
||||
|
||||
Func<int> Create() |
||||
{ |
||||
return () => x; |
||||
} |
||||
} |
||||
|
||||
public static Func<int> MethodRef() |
||||
{ |
||||
int x = (int)Math.Pow(2.0, 10.0); |
||||
Enumerable.Range(1, 100).Select(LocalFunction); |
||||
return null; |
||||
|
||||
int LocalFunction(int y) |
||||
{ |
||||
return x * y; |
||||
} |
||||
} |
||||
|
||||
public static int Fib(int i) |
||||
{ |
||||
return FibHelper(i); |
||||
|
||||
int FibHelper(int n) |
||||
{ |
||||
if (n <= 0) { |
||||
return 0; |
||||
} |
||||
|
||||
return FibHelper(n - 1) + FibHelper(n - 2); |
||||
} |
||||
} |
||||
public int MutuallyRecursiveLocalFunctions() |
||||
{ |
||||
return B(4) + C(3); |
||||
|
||||
int A(int i) |
||||
{ |
||||
if (i > 0) { |
||||
return A(i - 1) + 2 * B(i - 1) + 3 * C(i - 1); |
||||
} |
||||
return 1; |
||||
} |
||||
|
||||
int B(int i) |
||||
{ |
||||
if (i > 0) { |
||||
return 3 * A(i - 1) + B(i - 1); |
||||
} |
||||
return 1; |
||||
} |
||||
|
||||
int C(int i) |
||||
{ |
||||
if (i > 0) { |
||||
return 2 * A(i - 1) + C(i - 1); |
||||
} |
||||
return 1; |
||||
} |
||||
} |
||||
|
||||
public static int NestedLocalFunctions(int i) |
||||
{ |
||||
return A(); |
||||
|
||||
int A() |
||||
{ |
||||
double x = Math.Pow(10.0, 2.0); |
||||
return B(); |
||||
|
||||
int B() |
||||
{ |
||||
return i + (int)x; |
||||
} |
||||
} |
||||
} |
||||
|
||||
public static int LocalFunctionInLambda(IEnumerable<int> xs) |
||||
{ |
||||
return xs.First(delegate(int x) { |
||||
return Do(); |
||||
|
||||
bool Do() |
||||
{ |
||||
return x == 3; |
||||
} |
||||
}); |
||||
} |
||||
//public static void LocalFunctionInUsing()
|
||||
//{
|
||||
// using (MemoryStream memoryStream = new MemoryStream()) {
|
||||
// Do();
|
||||
|
||||
// void Do()
|
||||
// {
|
||||
// memoryStream.WriteByte(42);
|
||||
// }
|
||||
// }
|
||||
//}
|
||||
} |
||||
} |
@ -1,306 +0,0 @@
@@ -1,306 +0,0 @@
|
||||
//
|
||||
// ReducedExtensionMethod.cs
|
||||
//
|
||||
// Author:
|
||||
// Mike Krüger <mkrueger@xamarin.com>
|
||||
//
|
||||
// Copyright (c) 2013 Xamarin Inc. (http://xamarin.com)
|
||||
//
|
||||
// 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 System.Reflection; |
||||
using ICSharpCode.Decompiler.TypeSystem; |
||||
|
||||
namespace ICSharpCode.Decompiler.CSharp.Resolver |
||||
{ |
||||
/// <summary>
|
||||
/// An invocated extension method hides the extension parameter in its parameter list.
|
||||
/// It's used to hide the internals of extension method invocation in certain situation to simulate the
|
||||
/// syntactic way of writing extension methods on semantic level.
|
||||
/// </summary>
|
||||
public class ReducedExtensionMethod : IMethod |
||||
{ |
||||
readonly IMethod baseMethod; |
||||
|
||||
public ReducedExtensionMethod(IMethod baseMethod) |
||||
{ |
||||
this.baseMethod = baseMethod; |
||||
} |
||||
|
||||
public bool Equals(IMember obj, TypeVisitor typeNormalization) |
||||
{ |
||||
var other = obj as ReducedExtensionMethod; |
||||
if (other == null) |
||||
return false; |
||||
return baseMethod.Equals(other.baseMethod, typeNormalization); |
||||
} |
||||
|
||||
public override bool Equals(object obj) |
||||
{ |
||||
var other = obj as ReducedExtensionMethod; |
||||
if (other == null) |
||||
return false; |
||||
return baseMethod.Equals(other.baseMethod); |
||||
} |
||||
|
||||
public override int GetHashCode() |
||||
{ |
||||
unchecked { |
||||
return baseMethod.GetHashCode() + 1; |
||||
} |
||||
} |
||||
|
||||
public override string ToString() |
||||
{ |
||||
return string.Format("[ReducedExtensionMethod: ReducedFrom={0}]", ReducedFrom); |
||||
} |
||||
|
||||
#region IMember implementation
|
||||
public IMember MemberDefinition { |
||||
get { |
||||
return baseMethod.MemberDefinition; |
||||
} |
||||
} |
||||
|
||||
public IType ReturnType { |
||||
get { |
||||
return baseMethod.ReturnType; |
||||
} |
||||
} |
||||
|
||||
public IEnumerable<IMember> ExplicitlyImplementedInterfaceMembers { |
||||
get { |
||||
return baseMethod.ExplicitlyImplementedInterfaceMembers; |
||||
} |
||||
} |
||||
|
||||
public bool IsExplicitInterfaceImplementation { |
||||
get { |
||||
return baseMethod.IsExplicitInterfaceImplementation; |
||||
} |
||||
} |
||||
|
||||
public bool IsVirtual { |
||||
get { |
||||
return baseMethod.IsVirtual; |
||||
} |
||||
} |
||||
|
||||
public bool IsOverride { |
||||
get { |
||||
return baseMethod.IsOverride; |
||||
} |
||||
} |
||||
|
||||
public bool IsOverridable { |
||||
get { |
||||
return baseMethod.IsOverridable; |
||||
} |
||||
} |
||||
|
||||
public TypeParameterSubstitution Substitution { |
||||
get { |
||||
return baseMethod.Substitution; |
||||
} |
||||
} |
||||
|
||||
public IMethod Specialize(TypeParameterSubstitution substitution) |
||||
{ |
||||
return new ReducedExtensionMethod((IMethod)baseMethod.Specialize(substitution)); |
||||
} |
||||
|
||||
IMember IMember.Specialize(TypeParameterSubstitution substitution) |
||||
{ |
||||
return Specialize(substitution); |
||||
} |
||||
|
||||
#endregion
|
||||
|
||||
#region IMethod implementation
|
||||
|
||||
public IReadOnlyList<ITypeParameter> TypeParameters { |
||||
get { |
||||
return baseMethod.TypeParameters; |
||||
} |
||||
} |
||||
|
||||
public bool IsExtensionMethod { |
||||
get { |
||||
return true; |
||||
} |
||||
} |
||||
|
||||
public bool IsConstructor { |
||||
get { |
||||
return baseMethod.IsConstructor; |
||||
} |
||||
} |
||||
|
||||
public bool IsDestructor { |
||||
get { |
||||
return baseMethod.IsDestructor; |
||||
} |
||||
} |
||||
|
||||
public bool IsOperator { |
||||
get { |
||||
return baseMethod.IsOperator; |
||||
} |
||||
} |
||||
|
||||
public bool HasBody { |
||||
get { |
||||
return baseMethod.HasBody; |
||||
} |
||||
} |
||||
|
||||
public bool IsAccessor => baseMethod.IsAccessor; |
||||
public IMember AccessorOwner => baseMethod.AccessorOwner; |
||||
public MethodSemanticsAttributes AccessorKind => baseMethod.AccessorKind; |
||||
|
||||
public IMethod ReducedFrom { |
||||
get { |
||||
return baseMethod; |
||||
} |
||||
} |
||||
|
||||
public IReadOnlyList<IType> TypeArguments { |
||||
get { |
||||
return baseMethod.TypeArguments; |
||||
} |
||||
} |
||||
#endregion
|
||||
|
||||
#region IParameterizedMember implementation
|
||||
List<IParameter> parameters; |
||||
public IReadOnlyList<IParameter> Parameters { |
||||
get { |
||||
if (parameters == null) |
||||
parameters = new List<IParameter> (baseMethod.Parameters.Skip (1)); |
||||
return parameters; |
||||
} |
||||
} |
||||
|
||||
#endregion
|
||||
|
||||
#region IEntity implementation
|
||||
|
||||
public System.Reflection.Metadata.EntityHandle MetadataToken => baseMethod.MetadataToken; |
||||
|
||||
public SymbolKind SymbolKind { |
||||
get { |
||||
return baseMethod.SymbolKind; |
||||
} |
||||
} |
||||
|
||||
public ITypeDefinition DeclaringTypeDefinition { |
||||
get { |
||||
return baseMethod.DeclaringTypeDefinition; |
||||
} |
||||
} |
||||
|
||||
public IType DeclaringType { |
||||
get { |
||||
return baseMethod.DeclaringType; |
||||
} |
||||
} |
||||
|
||||
public IModule ParentModule { |
||||
get { |
||||
return baseMethod.ParentModule; |
||||
} |
||||
} |
||||
|
||||
IEnumerable<IAttribute> IEntity.GetAttributes() => baseMethod.GetAttributes(); |
||||
IEnumerable<IAttribute> IMethod.GetReturnTypeAttributes() => baseMethod.GetReturnTypeAttributes(); |
||||
bool IMethod.ReturnTypeIsRefReadOnly => baseMethod.ReturnTypeIsRefReadOnly; |
||||
|
||||
public bool IsStatic { |
||||
get { |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
public bool IsAbstract { |
||||
get { |
||||
return baseMethod.IsAbstract; |
||||
} |
||||
} |
||||
|
||||
public bool IsSealed { |
||||
get { |
||||
return baseMethod.IsSealed; |
||||
} |
||||
} |
||||
|
||||
#endregion
|
||||
|
||||
#region IHasAccessibility implementation
|
||||
|
||||
public Accessibility Accessibility { |
||||
get { |
||||
return baseMethod.Accessibility; |
||||
} |
||||
} |
||||
|
||||
#endregion
|
||||
|
||||
#region INamedElement implementation
|
||||
|
||||
public string FullName { |
||||
get { |
||||
return baseMethod.FullName; |
||||
} |
||||
} |
||||
|
||||
public string Name { |
||||
get { |
||||
return baseMethod.Name; |
||||
} |
||||
} |
||||
|
||||
public string ReflectionName { |
||||
get { |
||||
return baseMethod.ReflectionName; |
||||
} |
||||
} |
||||
|
||||
public string Namespace { |
||||
get { |
||||
return baseMethod.Namespace; |
||||
} |
||||
} |
||||
|
||||
#endregion
|
||||
|
||||
#region ICompilationProvider implementation
|
||||
|
||||
public ICompilation Compilation { |
||||
get { |
||||
return baseMethod.Compilation; |
||||
} |
||||
} |
||||
|
||||
#endregion
|
||||
} |
||||
} |
||||
|
@ -0,0 +1,110 @@
@@ -0,0 +1,110 @@
|
||||
// Copyright (c) 2019 Siegfried Pammer
|
||||
//
|
||||
// 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.Collections.Generic; |
||||
using ICSharpCode.Decompiler.CSharp.Syntax.PatternMatching; |
||||
|
||||
namespace ICSharpCode.Decompiler.CSharp.Syntax |
||||
{ |
||||
public class LocalFunctionDeclarationStatement : Statement |
||||
{ |
||||
public AstNodeCollection<TypeParameterDeclaration> TypeParameters { |
||||
get { return GetChildrenByRole(Roles.TypeParameter); } |
||||
} |
||||
|
||||
public CSharpTokenNode LParToken { |
||||
get { return GetChildByRole(Roles.LPar); } |
||||
} |
||||
|
||||
public AstNodeCollection<ParameterDeclaration> Parameters { |
||||
get { return GetChildrenByRole(Roles.Parameter); } |
||||
} |
||||
|
||||
public CSharpTokenNode RParToken { |
||||
get { return GetChildByRole(Roles.RPar); } |
||||
} |
||||
|
||||
public AstNodeCollection<Constraint> Constraints { |
||||
get { return GetChildrenByRole(Roles.Constraint); } |
||||
} |
||||
|
||||
public BlockStatement Body { |
||||
get { return GetChildByRole(Roles.Body); } |
||||
set { SetChildByRole(Roles.Body, value); } |
||||
} |
||||
|
||||
public Modifiers Modifiers { |
||||
get { return EntityDeclaration.GetModifiers(this); } |
||||
set { EntityDeclaration.SetModifiers(this, value); } |
||||
} |
||||
|
||||
public bool HasModifier(Modifiers mod) |
||||
{ |
||||
return (Modifiers & mod) == mod; |
||||
} |
||||
|
||||
public IEnumerable<CSharpModifierToken> ModifierTokens { |
||||
get { return GetChildrenByRole(EntityDeclaration.ModifierRole); } |
||||
} |
||||
|
||||
public virtual string Name { |
||||
get { |
||||
return GetChildByRole(Roles.Identifier).Name; |
||||
} |
||||
set { |
||||
SetChildByRole(Roles.Identifier, Identifier.Create(value, TextLocation.Empty)); |
||||
} |
||||
} |
||||
|
||||
public virtual Identifier NameToken { |
||||
get { return GetChildByRole(Roles.Identifier); } |
||||
set { SetChildByRole(Roles.Identifier, value); } |
||||
} |
||||
|
||||
public virtual AstType ReturnType { |
||||
get { return GetChildByRole(Roles.Type); } |
||||
set { SetChildByRole(Roles.Type, value); } |
||||
} |
||||
|
||||
public override void AcceptVisitor(IAstVisitor visitor) |
||||
{ |
||||
visitor.VisitLocalFunctionDeclarationStatement(this); |
||||
} |
||||
|
||||
public override T AcceptVisitor<T>(IAstVisitor<T> visitor) |
||||
{ |
||||
return visitor.VisitLocalFunctionDeclarationStatement(this); |
||||
} |
||||
|
||||
public override S AcceptVisitor<T, S>(IAstVisitor<T, S> visitor, T data) |
||||
{ |
||||
return visitor.VisitLocalFunctionDeclarationStatement(this, data); |
||||
} |
||||
|
||||
protected internal override bool DoMatch(AstNode other, Match match) |
||||
{ |
||||
LocalFunctionDeclarationStatement o = other as LocalFunctionDeclarationStatement; |
||||
return o != null && MatchString(this.Name, o.Name) |
||||
&& (this.Modifiers == Modifiers.Any || this.Modifiers == o.Modifiers) |
||||
&& this.ReturnType.DoMatch(o.ReturnType, match) |
||||
&& this.TypeParameters.DoMatch(o.TypeParameters, match) |
||||
&& this.Parameters.DoMatch(o.Parameters, match) && this.Constraints.DoMatch(o.Constraints, match) |
||||
&& this.Body.DoMatch(o.Body, match); |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,65 @@
@@ -0,0 +1,65 @@
|
||||
using System; |
||||
using System.Linq; |
||||
using System.Collections.Generic; |
||||
using System.Text; |
||||
using System.Diagnostics; |
||||
using ICSharpCode.Decompiler.TypeSystem; |
||||
|
||||
namespace ICSharpCode.Decompiler.IL.Transforms |
||||
{ |
||||
public class FixRemainingIncrements : IILTransform |
||||
{ |
||||
void IILTransform.Run(ILFunction function, ILTransformContext context) |
||||
{ |
||||
var callsToFix = new List<Call>(); |
||||
foreach (var call in function.Descendants.OfType<Call>()) { |
||||
if (!(call.Method.IsOperator && (call.Method.Name == "op_Increment" || call.Method.Name == "op_Decrement"))) |
||||
continue; |
||||
if (call.Arguments.Count != 1) |
||||
continue; |
||||
if (call.Method.DeclaringType.IsKnownType(KnownTypeCode.Decimal)) { |
||||
// For decimal, legacy csc can optimize "d + 1m" to "op_Increment(d)".
|
||||
// We can handle these calls in ReplaceMethodCallsWithOperators.
|
||||
continue; |
||||
} |
||||
callsToFix.Add(call); |
||||
} |
||||
foreach (var call in callsToFix) { |
||||
// A user-defined increment/decrement that was not handled by TransformAssignment.
|
||||
// This can happen because the variable-being-incremented was optimized out by Roslyn,
|
||||
// e.g.
|
||||
// public void Issue1552Pre(UserType a, UserType b)
|
||||
// {
|
||||
// UserType num = a + b;
|
||||
// Console.WriteLine(++num);
|
||||
// }
|
||||
// can end up being compiled to:
|
||||
// Console.WriteLine(UserType.op_Increment(a + b));
|
||||
if (call.SlotInfo == StLoc.ValueSlot && call.Parent.SlotInfo == Block.InstructionSlot) { |
||||
var store = (StLoc)call.Parent; |
||||
var block = (Block)store.Parent; |
||||
context.Step($"Fix {call.Method.Name} call at 0x{call.StartILOffset:x4} using {store.Variable.Name}", call); |
||||
// stloc V(call op_Increment(...))
|
||||
// ->
|
||||
// stloc V(...)
|
||||
// compound.assign op_Increment(V)
|
||||
call.ReplaceWith(call.Arguments[0]); |
||||
block.Instructions.Insert(store.ChildIndex + 1, |
||||
new UserDefinedCompoundAssign(call.Method, CompoundEvalMode.EvaluatesToNewValue, |
||||
new LdLoca(store.Variable), CompoundTargetKind.Address, new LdcI4(1)).WithILRange(call)); |
||||
} else { |
||||
context.Step($"Fix {call.Method.Name} call at 0x{call.StartILOffset:x4} using new local", call); |
||||
var newVariable = call.Arguments[0].Extract(); |
||||
if (newVariable == null) { |
||||
Debug.Fail("Failed to extract argument of remaining increment/decrement"); |
||||
continue; |
||||
} |
||||
newVariable.Type = call.GetParameter(0).Type; |
||||
Debug.Assert(call.Arguments[0].MatchLdLoc(newVariable)); |
||||
call.ReplaceWith(new UserDefinedCompoundAssign(call.Method, CompoundEvalMode.EvaluatesToNewValue, |
||||
new LdLoca(newVariable), CompoundTargetKind.Address, new LdcI4(1)).WithILRange(call)); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,103 @@
@@ -0,0 +1,103 @@
|
||||
using System; |
||||
using System.Collections.Generic; |
||||
using System.Diagnostics; |
||||
using System.Linq; |
||||
using System.Threading; |
||||
|
||||
namespace ICSharpCode.Decompiler.IL.Transforms |
||||
{ |
||||
/// <summary>
|
||||
/// Context object for the ILInstruction.Extract() operation.
|
||||
/// </summary>
|
||||
class ExtractionContext |
||||
{ |
||||
/// <summary>
|
||||
/// Nearest function, used for registering the new locals that are created by extraction.
|
||||
/// </summary>
|
||||
readonly ILFunction Function; |
||||
|
||||
/// <summary>
|
||||
/// Combined flags of all instructions being moved.
|
||||
/// </summary>
|
||||
internal InstructionFlags FlagsBeingMoved; |
||||
|
||||
/// <summary>
|
||||
/// List of actions to be executed when performing the extraction.
|
||||
///
|
||||
/// Each function in this list has the side-effect of replacing the instruction-to-be-moved
|
||||
/// with a load of a fresh temporary variable; and returns the the store to the temporary variable,
|
||||
/// which will be inserted at block-level.
|
||||
/// </summary>
|
||||
readonly List<Func<ILInstruction>> MoveActions = new List<Func<ILInstruction>>(); |
||||
|
||||
ExtractionContext(ILFunction function) |
||||
{ |
||||
Debug.Assert(function != null); |
||||
this.Function = function; |
||||
} |
||||
|
||||
internal void RegisterMove(ILInstruction predecessor) |
||||
{ |
||||
FlagsBeingMoved |= predecessor.Flags; |
||||
MoveActions.Add(delegate { |
||||
var v = Function.RegisterVariable(VariableKind.StackSlot, predecessor.ResultType); |
||||
predecessor.ReplaceWith(new LdLoc(v)); |
||||
return new StLoc(v, predecessor); |
||||
}); |
||||
} |
||||
|
||||
internal void RegisterMoveIfNecessary(ILInstruction predecessor) |
||||
{ |
||||
if (!CanReorderWithInstructionsBeingMoved(predecessor)) { |
||||
RegisterMove(predecessor); |
||||
} |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Currently, <c>predecessor</c> is evaluated before the instructions being moved.
|
||||
/// If this function returns true, <c>predecessor</c> can stay as-is, despite the move changing the evaluation order.
|
||||
/// If this function returns false, <c>predecessor</c> will need to also move, to ensure the evaluation order stays unchanged.
|
||||
/// </summary>
|
||||
public bool CanReorderWithInstructionsBeingMoved(ILInstruction predecessor) |
||||
{ |
||||
// We could track the instructions being moved and be smarter about unnecessary moves,
|
||||
// but given the limited scenarios where extraction is used so far,
|
||||
// this seems unnecessary.
|
||||
return predecessor.Flags == InstructionFlags.None; |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Extracts the specified instruction:
|
||||
/// The instruction is replaced with a load of a new temporary variable;
|
||||
/// and the instruction is moved to a store to said variable at block-level.
|
||||
///
|
||||
/// May return null if extraction is not possible.
|
||||
/// </summary>
|
||||
public static ILVariable Extract(ILInstruction instToExtract) |
||||
{ |
||||
var function = instToExtract.Ancestors.OfType<ILFunction>().First(); |
||||
ExtractionContext ctx = new ExtractionContext(function); |
||||
ctx.FlagsBeingMoved = instToExtract.Flags; |
||||
ILInstruction inst = instToExtract; |
||||
while (inst != null) { |
||||
if (inst.Parent is Block block && block.Kind == BlockKind.ControlFlow) { |
||||
// We've reached the target block, and extraction is possible all the way.
|
||||
int insertIndex = inst.ChildIndex; |
||||
// Move instToExtract itself:
|
||||
var v = function.RegisterVariable(VariableKind.StackSlot, instToExtract.ResultType); |
||||
instToExtract.ReplaceWith(new LdLoc(v)); |
||||
block.Instructions.Insert(insertIndex, new StLoc(v, instToExtract)); |
||||
// Apply the other move actions:
|
||||
foreach (var moveAction in ctx.MoveActions) { |
||||
block.Instructions.Insert(insertIndex, moveAction()); |
||||
} |
||||
return v; |
||||
} |
||||
if (!inst.Parent.PrepareExtract(inst.ChildIndex, ctx)) |
||||
return null; |
||||
inst = inst.Parent; |
||||
} |
||||
return null; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,138 @@
@@ -0,0 +1,138 @@
|
||||
// Copyright (c) 2019 Siegfried Pammer
|
||||
//
|
||||
// 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.Reflection; |
||||
using ICSharpCode.Decompiler.Util; |
||||
|
||||
namespace ICSharpCode.Decompiler.TypeSystem.Implementation |
||||
{ |
||||
/// <summary>
|
||||
/// A local function has zero or more compiler-generated parameters added at the end.
|
||||
/// </summary>
|
||||
class LocalFunctionMethod : IMethod |
||||
{ |
||||
readonly IMethod baseMethod; |
||||
|
||||
public LocalFunctionMethod(IMethod baseMethod, int numberOfCompilerGeneratedParameters) |
||||
{ |
||||
this.baseMethod = baseMethod; |
||||
this.NumberOfCompilerGeneratedParameters = numberOfCompilerGeneratedParameters; |
||||
} |
||||
|
||||
|
||||
public bool Equals(IMember obj, TypeVisitor typeNormalization) |
||||
{ |
||||
if (!(obj is LocalFunctionMethod other)) |
||||
return false; |
||||
return baseMethod.Equals(other.baseMethod, typeNormalization) |
||||
&& NumberOfCompilerGeneratedParameters == other.NumberOfCompilerGeneratedParameters; |
||||
} |
||||
|
||||
public override bool Equals(object obj) |
||||
{ |
||||
if (!(obj is LocalFunctionMethod other)) |
||||
return false; |
||||
return baseMethod.Equals(other.baseMethod) |
||||
&& NumberOfCompilerGeneratedParameters == other.NumberOfCompilerGeneratedParameters; |
||||
} |
||||
|
||||
public override int GetHashCode() |
||||
{ |
||||
unchecked { |
||||
return baseMethod.GetHashCode() + NumberOfCompilerGeneratedParameters + 1; |
||||
} |
||||
} |
||||
|
||||
public override string ToString() |
||||
{ |
||||
return string.Format("[LocalFunctionMethod: ReducedFrom={0}, NumberOfGeneratedParameters={1}]", ReducedFrom, NumberOfCompilerGeneratedParameters); |
||||
} |
||||
|
||||
internal int NumberOfCompilerGeneratedParameters { get; } |
||||
|
||||
public IMember MemberDefinition => this; |
||||
|
||||
public IType ReturnType => baseMethod.ReturnType; |
||||
IEnumerable<IMember> IMember.ExplicitlyImplementedInterfaceMembers => baseMethod.ExplicitlyImplementedInterfaceMembers; |
||||
bool IMember.IsExplicitInterfaceImplementation => baseMethod.IsExplicitInterfaceImplementation; |
||||
public bool IsVirtual => baseMethod.IsVirtual; |
||||
public bool IsOverride => baseMethod.IsOverride; |
||||
public bool IsOverridable => baseMethod.IsOverridable; |
||||
public TypeParameterSubstitution Substitution => baseMethod.Substitution; |
||||
|
||||
public IMethod Specialize(TypeParameterSubstitution substitution) |
||||
{ |
||||
return SpecializedMethod.Create(this, substitution); |
||||
} |
||||
|
||||
IMember IMember.Specialize(TypeParameterSubstitution substitution) |
||||
{ |
||||
return Specialize(substitution); |
||||
} |
||||
|
||||
public IReadOnlyList<ITypeParameter> TypeParameters => baseMethod.TypeParameters; |
||||
public bool IsExtensionMethod => baseMethod.IsExtensionMethod; |
||||
public bool IsLocalFunction => true; |
||||
public bool IsConstructor => baseMethod.IsConstructor; |
||||
public bool IsDestructor => baseMethod.IsDestructor; |
||||
public bool IsOperator => baseMethod.IsOperator; |
||||
public bool HasBody => baseMethod.HasBody; |
||||
public bool IsAccessor => baseMethod.IsAccessor; |
||||
public IMember AccessorOwner => baseMethod.AccessorOwner; |
||||
public MethodSemanticsAttributes AccessorKind => baseMethod.AccessorKind; |
||||
public IMethod ReducedFrom => baseMethod; |
||||
public IReadOnlyList<IType> TypeArguments => baseMethod.TypeArguments; |
||||
|
||||
List<IParameter> parameters; |
||||
public IReadOnlyList<IParameter> Parameters { |
||||
get { |
||||
if (parameters == null) |
||||
parameters = new List<IParameter>(baseMethod.Parameters.SkipLast(NumberOfCompilerGeneratedParameters)); |
||||
return parameters; |
||||
} |
||||
} |
||||
|
||||
public System.Reflection.Metadata.EntityHandle MetadataToken => baseMethod.MetadataToken; |
||||
public SymbolKind SymbolKind => baseMethod.SymbolKind; |
||||
public ITypeDefinition DeclaringTypeDefinition => baseMethod.DeclaringTypeDefinition; |
||||
public IType DeclaringType => baseMethod.DeclaringType; |
||||
public IModule ParentModule => baseMethod.ParentModule; |
||||
IEnumerable<IAttribute> IEntity.GetAttributes() => baseMethod.GetAttributes(); |
||||
IEnumerable<IAttribute> IMethod.GetReturnTypeAttributes() => baseMethod.GetReturnTypeAttributes(); |
||||
bool IMethod.ReturnTypeIsRefReadOnly => baseMethod.ReturnTypeIsRefReadOnly; |
||||
/// <summary>
|
||||
/// We consider local functions as always static, because they do not have a "this parameter".
|
||||
/// Even local functions in instance methods capture this.
|
||||
/// </summary>
|
||||
public bool IsStatic => true; |
||||
public bool IsAbstract => baseMethod.IsAbstract; |
||||
public bool IsSealed => baseMethod.IsSealed; |
||||
|
||||
public Accessibility Accessibility => baseMethod.Accessibility; |
||||
|
||||
public string FullName => baseMethod.FullName; |
||||
public string Name => baseMethod.Name; |
||||
public string ReflectionName => baseMethod.ReflectionName; |
||||
public string Namespace => baseMethod.Namespace; |
||||
|
||||
public ICompilation Compilation => baseMethod.Compilation; |
||||
} |
||||
} |
||||
|
Loading…
Reference in new issue