mirror of https://github.com/icsharpcode/ILSpy.git
45 changed files with 1408 additions and 567 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,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,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