mirror of https://github.com/icsharpcode/ILSpy.git
30 changed files with 2763 additions and 54 deletions
@ -0,0 +1,29 @@
@@ -0,0 +1,29 @@
|
||||
using System; |
||||
|
||||
namespace ICSharpCode.Decompiler.Tests.TestCases.Correctness |
||||
{ |
||||
class NullPropagation |
||||
{ |
||||
static void Main() |
||||
{ |
||||
new NullPropagation().TestNotCoalescing(); |
||||
} |
||||
|
||||
class MyClass |
||||
{ |
||||
public string Text; |
||||
} |
||||
|
||||
void TestNotCoalescing() |
||||
{ |
||||
Console.WriteLine("TestNotCoalescing:"); |
||||
Console.WriteLine(NotCoalescing(null)); |
||||
Console.WriteLine(NotCoalescing(new MyClass())); |
||||
} |
||||
|
||||
string NotCoalescing(MyClass c) |
||||
{ |
||||
return c != null ? c.Text : "Hello"; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,213 @@
@@ -0,0 +1,213 @@
|
||||
using System; |
||||
|
||||
namespace ICSharpCode.Decompiler.Tests.TestCases.Pretty |
||||
{ |
||||
internal class NullPropagation |
||||
{ |
||||
private class MyClass |
||||
{ |
||||
public int IntVal; |
||||
public string Text; |
||||
public MyClass Field; |
||||
public MyClass Property { |
||||
get; |
||||
set; |
||||
} |
||||
public MyClass this[int index] { |
||||
get { |
||||
return null; |
||||
} |
||||
} |
||||
public MyClass Method(int arg) |
||||
{ |
||||
return null; |
||||
} |
||||
|
||||
public void Done() |
||||
{ |
||||
} |
||||
} |
||||
|
||||
private struct MyStruct |
||||
{ |
||||
public int IntVal; |
||||
public MyClass Field; |
||||
public MyStruct? Property1 { |
||||
get { |
||||
return null; |
||||
} |
||||
} |
||||
public MyStruct Property2 { |
||||
get { |
||||
return default(MyStruct); |
||||
} |
||||
} |
||||
public MyStruct? this[int index] { |
||||
get { |
||||
return null; |
||||
} |
||||
} |
||||
public MyStruct? Method1(int arg) |
||||
{ |
||||
return null; |
||||
} |
||||
public MyStruct Method2(int arg) |
||||
{ |
||||
return default(MyStruct); |
||||
} |
||||
|
||||
public void Done() |
||||
{ |
||||
} |
||||
} |
||||
|
||||
private int GetInt() |
||||
{ |
||||
return 9; |
||||
} |
||||
|
||||
private string GetString() |
||||
{ |
||||
return null; |
||||
} |
||||
|
||||
private MyClass GetMyClass() |
||||
{ |
||||
return null; |
||||
} |
||||
|
||||
private MyStruct? GetMyStruct() |
||||
{ |
||||
return null; |
||||
} |
||||
|
||||
public string Substring() |
||||
{ |
||||
return this.GetString()?.Substring(this.GetInt()); |
||||
} |
||||
|
||||
public void CallSubstringAndIgnoreResult() |
||||
{ |
||||
this.GetString()?.Substring(this.GetInt()); |
||||
} |
||||
|
||||
private void Use<T>(T t) |
||||
{ |
||||
} |
||||
|
||||
public void CallDone() |
||||
{ |
||||
this.GetMyClass()?.Done(); |
||||
this.GetMyClass()?.Field?.Done(); |
||||
this.GetMyClass()?.Field.Done(); |
||||
this.GetMyClass()?.Property?.Done(); |
||||
this.GetMyClass()?.Property.Done(); |
||||
this.GetMyClass()?.Method(this.GetInt())?.Done(); |
||||
this.GetMyClass()?.Method(this.GetInt()).Done(); |
||||
this.GetMyClass()?[this.GetInt()]?.Done(); |
||||
this.GetMyClass()?[this.GetInt()].Done(); |
||||
} |
||||
|
||||
public void CallDoneStruct() |
||||
{ |
||||
this.GetMyStruct()?.Done(); |
||||
#if STRUCT_SPLITTING_IMPROVED
|
||||
this.GetMyStruct()?.Field?.Done(); |
||||
this.GetMyStruct()?.Field.Done(); |
||||
this.GetMyStruct()?.Property1?.Done(); |
||||
this.GetMyStruct()?.Property2.Done(); |
||||
this.GetMyStruct()?.Method1(this.GetInt())?.Done(); |
||||
this.GetMyStruct()?.Method2(this.GetInt()).Done(); |
||||
this.GetMyStruct()?[this.GetInt()]?.Done(); |
||||
#endif
|
||||
} |
||||
|
||||
public void RequiredParentheses() |
||||
{ |
||||
(this.GetMyClass()?.Field).Done(); |
||||
(this.GetMyClass()?.Method(this.GetInt())).Done(); |
||||
#if STRUCT_SPLITTING_IMPROVED
|
||||
(GetMyStruct()?.Property2)?.Done(); |
||||
#endif
|
||||
} |
||||
|
||||
public int?[] ChainsOnClass() |
||||
{ |
||||
return new int?[9] { |
||||
this.GetMyClass()?.IntVal, |
||||
this.GetMyClass()?.Field.IntVal, |
||||
this.GetMyClass()?.Field?.IntVal, |
||||
this.GetMyClass()?.Property.IntVal, |
||||
this.GetMyClass()?.Property?.IntVal, |
||||
this.GetMyClass()?.Method(this.GetInt()).IntVal, |
||||
this.GetMyClass()?.Method(this.GetInt())?.IntVal, |
||||
this.GetMyClass()?[this.GetInt()].IntVal, |
||||
this.GetMyClass()?[this.GetInt()]?.IntVal |
||||
}; |
||||
} |
||||
|
||||
#if STRUCT_SPLITTING_IMPROVED
|
||||
public int?[] ChainsStruct() |
||||
{ |
||||
return new int?[8] { |
||||
this.GetMyStruct()?.IntVal, |
||||
this.GetMyStruct()?.Field.IntVal, |
||||
this.GetMyStruct()?.Field?.IntVal, |
||||
this.GetMyStruct()?.Property2.IntVal, |
||||
this.GetMyStruct()?.Property1?.IntVal, |
||||
this.GetMyStruct()?.Method2(this.GetInt()).IntVal, |
||||
this.GetMyStruct()?.Method1(this.GetInt())?.IntVal, |
||||
this.GetMyStruct()?[this.GetInt()]?.IntVal |
||||
}; |
||||
} |
||||
#endif
|
||||
|
||||
public int CoalescingReturn() |
||||
{ |
||||
return this.GetMyClass()?.IntVal ?? 1; |
||||
} |
||||
|
||||
public void Coalescing() |
||||
{ |
||||
this.Use(this.GetMyClass()?.IntVal ?? 1); |
||||
} |
||||
|
||||
public void CoalescingString() |
||||
{ |
||||
this.Use(this.GetMyClass()?.Text ?? "Hello"); |
||||
} |
||||
|
||||
public void InvokeDelegate(EventHandler eh) |
||||
{ |
||||
eh?.Invoke(null, EventArgs.Empty); |
||||
} |
||||
|
||||
public int? InvokeDelegate(Func<int> f) |
||||
{ |
||||
return f?.Invoke(); |
||||
} |
||||
|
||||
private void NotNullPropagation(MyClass c) |
||||
{ |
||||
// don't decompile this to "(c?.IntVal ?? 0) != 0"
|
||||
if (c != null && c.IntVal != 0) { |
||||
Console.WriteLine("non-zero"); |
||||
} |
||||
if (c == null || c.IntVal == 0) { |
||||
Console.WriteLine("null or zero"); |
||||
} |
||||
Console.WriteLine("end of method"); |
||||
} |
||||
|
||||
private void Setter(MyClass c) |
||||
{ |
||||
if (c != null) { |
||||
c.IntVal = 1; |
||||
} |
||||
Console.WriteLine(); |
||||
if (c != null) { |
||||
c.Property = null; |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,863 @@
@@ -0,0 +1,863 @@
|
||||
|
||||
// Microsoft (R) .NET Framework IL Disassembler. Version 4.6.1055.0 |
||||
// Copyright (c) Microsoft Corporation. All rights reserved. |
||||
|
||||
|
||||
|
||||
// Metadata version: v4.0.30319 |
||||
.assembly extern mscorlib |
||||
{ |
||||
.publickeytoken = (B7 7A 5C 56 19 34 E0 89 ) // .z\V.4.. |
||||
.ver 4:0:0:0 |
||||
} |
||||
.assembly NullPropagation |
||||
{ |
||||
.custom instance void [mscorlib]System.Runtime.CompilerServices.CompilationRelaxationsAttribute::.ctor(int32) = ( 01 00 08 00 00 00 00 00 ) |
||||
.custom instance void [mscorlib]System.Runtime.CompilerServices.RuntimeCompatibilityAttribute::.ctor() = ( 01 00 01 00 54 02 16 57 72 61 70 4E 6F 6E 45 78 // ....T..WrapNonEx |
||||
63 65 70 74 69 6F 6E 54 68 72 6F 77 73 01 ) // ceptionThrows. |
||||
|
||||
// --- The following custom attribute is added automatically, do not uncomment ------- |
||||
// .custom instance void [mscorlib]System.Diagnostics.DebuggableAttribute::.ctor(valuetype [mscorlib]System.Diagnostics.DebuggableAttribute/DebuggingModes) = ( 01 00 02 00 00 00 00 00 ) |
||||
|
||||
.permissionset reqmin |
||||
= {[mscorlib]System.Security.Permissions.SecurityPermissionAttribute = {property bool 'SkipVerification' = bool(true)}} |
||||
.hash algorithm 0x00008004 |
||||
.ver 0:0:0:0 |
||||
} |
||||
.module NullPropagation.dll |
||||
// MVID: {DDAB2C82-C901-450A-B55D-2D0D7BC34C48} |
||||
.custom instance void [mscorlib]System.Security.UnverifiableCodeAttribute::.ctor() = ( 01 00 00 00 ) |
||||
.imagebase 0x10000000 |
||||
.file alignment 0x00000200 |
||||
.stackreserve 0x00100000 |
||||
.subsystem 0x0003 // WINDOWS_CUI |
||||
.corflags 0x00000001 // ILONLY |
||||
// Image base: 0x003E0000 |
||||
|
||||
|
||||
// =============== CLASS MEMBERS DECLARATION =================== |
||||
|
||||
.class private auto ansi beforefieldinit ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation |
||||
extends [mscorlib]System.Object |
||||
{ |
||||
.class auto ansi nested private beforefieldinit MyClass |
||||
extends [mscorlib]System.Object |
||||
{ |
||||
.custom instance void [mscorlib]System.Reflection.DefaultMemberAttribute::.ctor(string) = ( 01 00 04 49 74 65 6D 00 00 ) // ...Item.. |
||||
.field public int32 IntVal |
||||
.field public string Text |
||||
.field public class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass Field |
||||
.field private class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass '<Property>k__BackingField' |
||||
.custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor() = ( 01 00 00 00 ) |
||||
.method public hidebysig specialname |
||||
instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass |
||||
get_Property() cil managed |
||||
{ |
||||
.custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor() = ( 01 00 00 00 ) |
||||
// Code size 7 (0x7) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: ldfld class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::'<Property>k__BackingField' |
||||
IL_0006: ret |
||||
} // end of method MyClass::get_Property |
||||
|
||||
.method public hidebysig specialname |
||||
instance void set_Property(class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass 'value') cil managed |
||||
{ |
||||
.custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor() = ( 01 00 00 00 ) |
||||
// Code size 8 (0x8) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: ldarg.1 |
||||
IL_0002: stfld class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::'<Property>k__BackingField' |
||||
IL_0007: ret |
||||
} // end of method MyClass::set_Property |
||||
|
||||
.method public hidebysig specialname |
||||
instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass |
||||
get_Item(int32 index) cil managed |
||||
{ |
||||
// Code size 2 (0x2) |
||||
.maxstack 8 |
||||
IL_0000: ldnull |
||||
IL_0001: ret |
||||
} // end of method MyClass::get_Item |
||||
|
||||
.method public hidebysig instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass |
||||
Method(int32 arg) cil managed |
||||
{ |
||||
// Code size 2 (0x2) |
||||
.maxstack 8 |
||||
IL_0000: ldnull |
||||
IL_0001: ret |
||||
} // end of method MyClass::Method |
||||
|
||||
.method public hidebysig instance void |
||||
Done() cil managed |
||||
{ |
||||
// Code size 1 (0x1) |
||||
.maxstack 8 |
||||
IL_0000: ret |
||||
} // end of method MyClass::Done |
||||
|
||||
.method public hidebysig specialname rtspecialname |
||||
instance void .ctor() cil managed |
||||
{ |
||||
// Code size 7 (0x7) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance void [mscorlib]System.Object::.ctor() |
||||
IL_0006: ret |
||||
} // end of method MyClass::.ctor |
||||
|
||||
.property instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass |
||||
Property() |
||||
{ |
||||
.get instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Property() |
||||
.set instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::set_Property(class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass) |
||||
} // end of property MyClass::Property |
||||
.property instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass |
||||
Item(int32) |
||||
{ |
||||
.get instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Item(int32) |
||||
} // end of property MyClass::Item |
||||
} // end of class MyClass |
||||
|
||||
.class sequential ansi sealed nested private beforefieldinit MyStruct |
||||
extends [mscorlib]System.ValueType |
||||
{ |
||||
.custom instance void [mscorlib]System.Reflection.DefaultMemberAttribute::.ctor(string) = ( 01 00 04 49 74 65 6D 00 00 ) // ...Item.. |
||||
.field public int32 IntVal |
||||
.field public class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass Field |
||||
.method public hidebysig specialname |
||||
instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
get_Property1() cil managed |
||||
{ |
||||
// Code size 10 (0xa) |
||||
.maxstack 1 |
||||
.locals init (valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> V_0) |
||||
IL_0000: ldloca.s V_0 |
||||
IL_0002: initobj valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
IL_0008: ldloc.0 |
||||
IL_0009: ret |
||||
} // end of method MyStruct::get_Property1 |
||||
|
||||
.method public hidebysig specialname |
||||
instance valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct |
||||
get_Property2() cil managed |
||||
{ |
||||
// Code size 10 (0xa) |
||||
.maxstack 1 |
||||
.locals init (valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct V_0) |
||||
IL_0000: ldloca.s V_0 |
||||
IL_0002: initobj ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct |
||||
IL_0008: ldloc.0 |
||||
IL_0009: ret |
||||
} // end of method MyStruct::get_Property2 |
||||
|
||||
.method public hidebysig specialname |
||||
instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
get_Item(int32 index) cil managed |
||||
{ |
||||
// Code size 10 (0xa) |
||||
.maxstack 1 |
||||
.locals init (valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> V_0) |
||||
IL_0000: ldloca.s V_0 |
||||
IL_0002: initobj valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
IL_0008: ldloc.0 |
||||
IL_0009: ret |
||||
} // end of method MyStruct::get_Item |
||||
|
||||
.method public hidebysig instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
Method1(int32 arg) cil managed |
||||
{ |
||||
// Code size 10 (0xa) |
||||
.maxstack 1 |
||||
.locals init (valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> V_0) |
||||
IL_0000: ldloca.s V_0 |
||||
IL_0002: initobj valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
IL_0008: ldloc.0 |
||||
IL_0009: ret |
||||
} // end of method MyStruct::Method1 |
||||
|
||||
.method public hidebysig instance valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct |
||||
Method2(int32 arg) cil managed |
||||
{ |
||||
// Code size 10 (0xa) |
||||
.maxstack 1 |
||||
.locals init (valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct V_0) |
||||
IL_0000: ldloca.s V_0 |
||||
IL_0002: initobj ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct |
||||
IL_0008: ldloc.0 |
||||
IL_0009: ret |
||||
} // end of method MyStruct::Method2 |
||||
|
||||
.method public hidebysig instance void |
||||
Done() cil managed |
||||
{ |
||||
// Code size 1 (0x1) |
||||
.maxstack 8 |
||||
IL_0000: ret |
||||
} // end of method MyStruct::Done |
||||
|
||||
.property instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
Property1() |
||||
{ |
||||
.get instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct::get_Property1() |
||||
} // end of property MyStruct::Property1 |
||||
.property instance valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct |
||||
Property2() |
||||
{ |
||||
.get instance valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct::get_Property2() |
||||
} // end of property MyStruct::Property2 |
||||
.property instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
Item(int32) |
||||
{ |
||||
.get instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct::get_Item(int32) |
||||
} // end of property MyStruct::Item |
||||
} // end of class MyStruct |
||||
|
||||
.method private hidebysig instance int32 |
||||
GetInt() cil managed |
||||
{ |
||||
// Code size 3 (0x3) |
||||
.maxstack 8 |
||||
IL_0000: ldc.i4.s 9 |
||||
IL_0002: ret |
||||
} // end of method NullPropagation::GetInt |
||||
|
||||
.method private hidebysig instance string |
||||
GetString() cil managed |
||||
{ |
||||
// Code size 2 (0x2) |
||||
.maxstack 8 |
||||
IL_0000: ldnull |
||||
IL_0001: ret |
||||
} // end of method NullPropagation::GetString |
||||
|
||||
.method private hidebysig instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass |
||||
GetMyClass() cil managed |
||||
{ |
||||
// Code size 2 (0x2) |
||||
.maxstack 8 |
||||
IL_0000: ldnull |
||||
IL_0001: ret |
||||
} // end of method NullPropagation::GetMyClass |
||||
|
||||
.method private hidebysig instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
GetMyStruct() cil managed |
||||
{ |
||||
// Code size 10 (0xa) |
||||
.maxstack 1 |
||||
.locals init (valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> V_0) |
||||
IL_0000: ldloca.s V_0 |
||||
IL_0002: initobj valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> |
||||
IL_0008: ldloc.0 |
||||
IL_0009: ret |
||||
} // end of method NullPropagation::GetMyStruct |
||||
|
||||
.method public hidebysig instance string |
||||
Substring() cil managed |
||||
{ |
||||
// Code size 24 (0x18) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance string ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetString() |
||||
IL_0006: dup |
||||
IL_0007: brtrue.s IL_000c |
||||
|
||||
IL_0009: pop |
||||
IL_000a: ldnull |
||||
IL_000b: ret |
||||
|
||||
IL_000c: ldarg.0 |
||||
IL_000d: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_0012: call instance string [mscorlib]System.String::Substring(int32) |
||||
IL_0017: ret |
||||
} // end of method NullPropagation::Substring |
||||
|
||||
.method public hidebysig instance void |
||||
CallSubstringAndIgnoreResult() cil managed |
||||
{ |
||||
// Code size 24 (0x18) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance string ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetString() |
||||
IL_0006: dup |
||||
IL_0007: brtrue.s IL_000b |
||||
|
||||
IL_0009: pop |
||||
IL_000a: ret |
||||
|
||||
IL_000b: ldarg.0 |
||||
IL_000c: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_0011: call instance string [mscorlib]System.String::Substring(int32) |
||||
IL_0016: pop |
||||
IL_0017: ret |
||||
} // end of method NullPropagation::CallSubstringAndIgnoreResult |
||||
|
||||
.method private hidebysig instance void |
||||
Use<T>(!!T t) cil managed |
||||
{ |
||||
// Code size 1 (0x1) |
||||
.maxstack 8 |
||||
IL_0000: ret |
||||
} // end of method NullPropagation::Use |
||||
|
||||
.method public hidebysig instance void |
||||
CallDone() cil managed |
||||
{ |
||||
// Code size 241 (0xf1) |
||||
.maxstack 2 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0006: dup |
||||
IL_0007: brtrue.s IL_000c |
||||
|
||||
IL_0009: pop |
||||
IL_000a: br.s IL_0011 |
||||
|
||||
IL_000c: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_0011: ldarg.0 |
||||
IL_0012: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0017: dup |
||||
IL_0018: brtrue.s IL_001d |
||||
|
||||
IL_001a: pop |
||||
IL_001b: br.s IL_002d |
||||
|
||||
IL_001d: ldfld class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Field |
||||
IL_0022: dup |
||||
IL_0023: brtrue.s IL_0028 |
||||
|
||||
IL_0025: pop |
||||
IL_0026: br.s IL_002d |
||||
|
||||
IL_0028: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_002d: ldarg.0 |
||||
IL_002e: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0033: dup |
||||
IL_0034: brtrue.s IL_0039 |
||||
|
||||
IL_0036: pop |
||||
IL_0037: br.s IL_0043 |
||||
|
||||
IL_0039: ldfld class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Field |
||||
IL_003e: callvirt instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_0043: ldarg.0 |
||||
IL_0044: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0049: dup |
||||
IL_004a: brtrue.s IL_004f |
||||
|
||||
IL_004c: pop |
||||
IL_004d: br.s IL_005f |
||||
|
||||
IL_004f: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Property() |
||||
IL_0054: dup |
||||
IL_0055: brtrue.s IL_005a |
||||
|
||||
IL_0057: pop |
||||
IL_0058: br.s IL_005f |
||||
|
||||
IL_005a: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_005f: ldarg.0 |
||||
IL_0060: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0065: dup |
||||
IL_0066: brtrue.s IL_006b |
||||
|
||||
IL_0068: pop |
||||
IL_0069: br.s IL_0075 |
||||
|
||||
IL_006b: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Property() |
||||
IL_0070: callvirt instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_0075: ldarg.0 |
||||
IL_0076: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_007b: dup |
||||
IL_007c: brtrue.s IL_0081 |
||||
|
||||
IL_007e: pop |
||||
IL_007f: br.s IL_0097 |
||||
|
||||
IL_0081: ldarg.0 |
||||
IL_0082: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_0087: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Method(int32) |
||||
IL_008c: dup |
||||
IL_008d: brtrue.s IL_0092 |
||||
|
||||
IL_008f: pop |
||||
IL_0090: br.s IL_0097 |
||||
|
||||
IL_0092: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_0097: ldarg.0 |
||||
IL_0098: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_009d: dup |
||||
IL_009e: brtrue.s IL_00a3 |
||||
|
||||
IL_00a0: pop |
||||
IL_00a1: br.s IL_00b3 |
||||
|
||||
IL_00a3: ldarg.0 |
||||
IL_00a4: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_00a9: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Method(int32) |
||||
IL_00ae: callvirt instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_00b3: ldarg.0 |
||||
IL_00b4: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_00b9: dup |
||||
IL_00ba: brtrue.s IL_00bf |
||||
|
||||
IL_00bc: pop |
||||
IL_00bd: br.s IL_00d5 |
||||
|
||||
IL_00bf: ldarg.0 |
||||
IL_00c0: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_00c5: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Item(int32) |
||||
IL_00ca: dup |
||||
IL_00cb: brtrue.s IL_00d0 |
||||
|
||||
IL_00cd: pop |
||||
IL_00ce: br.s IL_00d5 |
||||
|
||||
IL_00d0: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_00d5: ldarg.0 |
||||
IL_00d6: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_00db: dup |
||||
IL_00dc: brtrue.s IL_00e0 |
||||
|
||||
IL_00de: pop |
||||
IL_00df: ret |
||||
|
||||
IL_00e0: ldarg.0 |
||||
IL_00e1: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_00e6: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Item(int32) |
||||
IL_00eb: callvirt instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_00f0: ret |
||||
} // end of method NullPropagation::CallDone |
||||
|
||||
.method public hidebysig instance void |
||||
CallDoneStruct() cil managed |
||||
{ |
||||
// Code size 33 (0x21) |
||||
.maxstack 2 |
||||
.locals init (valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> V_0, |
||||
valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct V_1) |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct> ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyStruct() |
||||
IL_0006: stloc.0 |
||||
IL_0007: ldloca.s V_0 |
||||
IL_0009: dup |
||||
IL_000a: call instance bool valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct>::get_HasValue() |
||||
IL_000f: brtrue.s IL_0013 |
||||
|
||||
IL_0011: pop |
||||
IL_0012: ret |
||||
|
||||
IL_0013: call instance !0 valuetype [mscorlib]System.Nullable`1<valuetype ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct>::GetValueOrDefault() |
||||
IL_0018: stloc.1 |
||||
IL_0019: ldloca.s V_1 |
||||
IL_001b: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyStruct::Done() |
||||
IL_0020: ret |
||||
} // end of method NullPropagation::CallDoneStruct |
||||
|
||||
.method public hidebysig instance void |
||||
RequiredParentheses() cil managed |
||||
{ |
||||
// Code size 53 (0x35) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0006: dup |
||||
IL_0007: brtrue.s IL_000d |
||||
|
||||
IL_0009: pop |
||||
IL_000a: ldnull |
||||
IL_000b: br.s IL_0012 |
||||
|
||||
IL_000d: ldfld class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Field |
||||
IL_0012: callvirt instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_0017: ldarg.0 |
||||
IL_0018: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_001d: dup |
||||
IL_001e: brtrue.s IL_0024 |
||||
|
||||
IL_0020: pop |
||||
IL_0021: ldnull |
||||
IL_0022: br.s IL_002f |
||||
|
||||
IL_0024: ldarg.0 |
||||
IL_0025: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_002a: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Method(int32) |
||||
IL_002f: callvirt instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Done() |
||||
IL_0034: ret |
||||
} // end of method NullPropagation::RequiredParentheses |
||||
|
||||
.method public hidebysig instance valuetype [mscorlib]System.Nullable`1<int32>[] |
||||
ChainsOnClass() cil managed |
||||
{ |
||||
// Code size 474 (0x1da) |
||||
.maxstack 5 |
||||
.locals init (valuetype [mscorlib]System.Nullable`1<int32> V_0) |
||||
IL_0000: ldc.i4.s 9 |
||||
IL_0002: newarr valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0007: dup |
||||
IL_0008: ldc.i4.0 |
||||
IL_0009: ldarg.0 |
||||
IL_000a: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_000f: dup |
||||
IL_0010: brtrue.s IL_001e |
||||
|
||||
IL_0012: pop |
||||
IL_0013: ldloca.s V_0 |
||||
IL_0015: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_001b: ldloc.0 |
||||
IL_001c: br.s IL_0028 |
||||
|
||||
IL_001e: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_0023: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_0028: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_002d: dup |
||||
IL_002e: ldc.i4.1 |
||||
IL_002f: ldarg.0 |
||||
IL_0030: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0035: dup |
||||
IL_0036: brtrue.s IL_0044 |
||||
|
||||
IL_0038: pop |
||||
IL_0039: ldloca.s V_0 |
||||
IL_003b: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0041: ldloc.0 |
||||
IL_0042: br.s IL_0053 |
||||
|
||||
IL_0044: ldfld class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Field |
||||
IL_0049: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_004e: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_0053: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0058: dup |
||||
IL_0059: ldc.i4.2 |
||||
IL_005a: ldarg.0 |
||||
IL_005b: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0060: dup |
||||
IL_0061: brtrue.s IL_006f |
||||
|
||||
IL_0063: pop |
||||
IL_0064: ldloca.s V_0 |
||||
IL_0066: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_006c: ldloc.0 |
||||
IL_006d: br.s IL_008d |
||||
|
||||
IL_006f: ldfld class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Field |
||||
IL_0074: dup |
||||
IL_0075: brtrue.s IL_0083 |
||||
|
||||
IL_0077: pop |
||||
IL_0078: ldloca.s V_0 |
||||
IL_007a: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0080: ldloc.0 |
||||
IL_0081: br.s IL_008d |
||||
|
||||
IL_0083: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_0088: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_008d: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0092: dup |
||||
IL_0093: ldc.i4.3 |
||||
IL_0094: ldarg.0 |
||||
IL_0095: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_009a: dup |
||||
IL_009b: brtrue.s IL_00a9 |
||||
|
||||
IL_009d: pop |
||||
IL_009e: ldloca.s V_0 |
||||
IL_00a0: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_00a6: ldloc.0 |
||||
IL_00a7: br.s IL_00b8 |
||||
|
||||
IL_00a9: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Property() |
||||
IL_00ae: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_00b3: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_00b8: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_00bd: dup |
||||
IL_00be: ldc.i4.4 |
||||
IL_00bf: ldarg.0 |
||||
IL_00c0: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_00c5: dup |
||||
IL_00c6: brtrue.s IL_00d4 |
||||
|
||||
IL_00c8: pop |
||||
IL_00c9: ldloca.s V_0 |
||||
IL_00cb: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_00d1: ldloc.0 |
||||
IL_00d2: br.s IL_00f2 |
||||
|
||||
IL_00d4: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Property() |
||||
IL_00d9: dup |
||||
IL_00da: brtrue.s IL_00e8 |
||||
|
||||
IL_00dc: pop |
||||
IL_00dd: ldloca.s V_0 |
||||
IL_00df: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_00e5: ldloc.0 |
||||
IL_00e6: br.s IL_00f2 |
||||
|
||||
IL_00e8: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_00ed: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_00f2: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_00f7: dup |
||||
IL_00f8: ldc.i4.5 |
||||
IL_00f9: ldarg.0 |
||||
IL_00fa: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_00ff: dup |
||||
IL_0100: brtrue.s IL_010e |
||||
|
||||
IL_0102: pop |
||||
IL_0103: ldloca.s V_0 |
||||
IL_0105: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_010b: ldloc.0 |
||||
IL_010c: br.s IL_0123 |
||||
|
||||
IL_010e: ldarg.0 |
||||
IL_010f: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_0114: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Method(int32) |
||||
IL_0119: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_011e: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_0123: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0128: dup |
||||
IL_0129: ldc.i4.6 |
||||
IL_012a: ldarg.0 |
||||
IL_012b: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0130: dup |
||||
IL_0131: brtrue.s IL_013f |
||||
|
||||
IL_0133: pop |
||||
IL_0134: ldloca.s V_0 |
||||
IL_0136: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_013c: ldloc.0 |
||||
IL_013d: br.s IL_0163 |
||||
|
||||
IL_013f: ldarg.0 |
||||
IL_0140: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_0145: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Method(int32) |
||||
IL_014a: dup |
||||
IL_014b: brtrue.s IL_0159 |
||||
|
||||
IL_014d: pop |
||||
IL_014e: ldloca.s V_0 |
||||
IL_0150: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0156: ldloc.0 |
||||
IL_0157: br.s IL_0163 |
||||
|
||||
IL_0159: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_015e: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_0163: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0168: dup |
||||
IL_0169: ldc.i4.7 |
||||
IL_016a: ldarg.0 |
||||
IL_016b: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0170: dup |
||||
IL_0171: brtrue.s IL_017f |
||||
|
||||
IL_0173: pop |
||||
IL_0174: ldloca.s V_0 |
||||
IL_0176: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_017c: ldloc.0 |
||||
IL_017d: br.s IL_0194 |
||||
|
||||
IL_017f: ldarg.0 |
||||
IL_0180: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_0185: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Item(int32) |
||||
IL_018a: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_018f: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_0194: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_0199: dup |
||||
IL_019a: ldc.i4.8 |
||||
IL_019b: ldarg.0 |
||||
IL_019c: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_01a1: dup |
||||
IL_01a2: brtrue.s IL_01b0 |
||||
|
||||
IL_01a4: pop |
||||
IL_01a5: ldloca.s V_0 |
||||
IL_01a7: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_01ad: ldloc.0 |
||||
IL_01ae: br.s IL_01d4 |
||||
|
||||
IL_01b0: ldarg.0 |
||||
IL_01b1: call instance int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetInt() |
||||
IL_01b6: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::get_Item(int32) |
||||
IL_01bb: dup |
||||
IL_01bc: brtrue.s IL_01ca |
||||
|
||||
IL_01be: pop |
||||
IL_01bf: ldloca.s V_0 |
||||
IL_01c1: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_01c7: ldloc.0 |
||||
IL_01c8: br.s IL_01d4 |
||||
|
||||
IL_01ca: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_01cf: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_01d4: stelem valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_01d9: ret |
||||
} // end of method NullPropagation::ChainsOnClass |
||||
|
||||
.method public hidebysig instance int32 |
||||
CoalescingReturn() cil managed |
||||
{ |
||||
// Code size 18 (0x12) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0006: dup |
||||
IL_0007: brtrue.s IL_000c |
||||
|
||||
IL_0009: pop |
||||
IL_000a: ldc.i4.1 |
||||
IL_000b: ret |
||||
|
||||
IL_000c: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_0011: ret |
||||
} // end of method NullPropagation::CoalescingReturn |
||||
|
||||
.method public hidebysig instance void |
||||
Coalescing() cil managed |
||||
{ |
||||
// Code size 25 (0x19) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: ldarg.0 |
||||
IL_0002: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0007: dup |
||||
IL_0008: brtrue.s IL_000e |
||||
|
||||
IL_000a: pop |
||||
IL_000b: ldc.i4.1 |
||||
IL_000c: br.s IL_0013 |
||||
|
||||
IL_000e: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_0013: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::Use<int32>(!!0) |
||||
IL_0018: ret |
||||
} // end of method NullPropagation::Coalescing |
||||
|
||||
.method public hidebysig instance void |
||||
CoalescingString() cil managed |
||||
{ |
||||
// Code size 34 (0x22) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: ldarg.0 |
||||
IL_0002: call instance class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::GetMyClass() |
||||
IL_0007: dup |
||||
IL_0008: brtrue.s IL_000e |
||||
|
||||
IL_000a: pop |
||||
IL_000b: ldnull |
||||
IL_000c: br.s IL_0013 |
||||
|
||||
IL_000e: ldfld string ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::Text |
||||
IL_0013: dup |
||||
IL_0014: brtrue.s IL_001c |
||||
|
||||
IL_0016: pop |
||||
IL_0017: ldstr "Hello" |
||||
IL_001c: call instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation::Use<string>(!!0) |
||||
IL_0021: ret |
||||
} // end of method NullPropagation::CoalescingString |
||||
|
||||
.method public hidebysig instance void |
||||
InvokeDelegate(class [mscorlib]System.EventHandler eh) cil managed |
||||
{ |
||||
// Code size 16 (0x10) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.1 |
||||
IL_0001: brfalse.s IL_000f |
||||
|
||||
IL_0003: ldarg.1 |
||||
IL_0004: ldnull |
||||
IL_0005: ldsfld class [mscorlib]System.EventArgs [mscorlib]System.EventArgs::Empty |
||||
IL_000a: callvirt instance void [mscorlib]System.EventHandler::Invoke(object, |
||||
class [mscorlib]System.EventArgs) |
||||
IL_000f: ret |
||||
} // end of method NullPropagation::InvokeDelegate |
||||
|
||||
.method public hidebysig instance valuetype [mscorlib]System.Nullable`1<int32> |
||||
InvokeDelegate(class [mscorlib]System.Func`1<int32> f) cil managed |
||||
{ |
||||
// Code size 25 (0x19) |
||||
.maxstack 1 |
||||
.locals init (valuetype [mscorlib]System.Nullable`1<int32> V_0) |
||||
IL_0000: ldarg.1 |
||||
IL_0001: brtrue.s IL_000d |
||||
|
||||
IL_0003: ldloca.s V_0 |
||||
IL_0005: initobj valuetype [mscorlib]System.Nullable`1<int32> |
||||
IL_000b: ldloc.0 |
||||
IL_000c: ret |
||||
|
||||
IL_000d: ldarg.1 |
||||
IL_000e: callvirt instance !0 class [mscorlib]System.Func`1<int32>::Invoke() |
||||
IL_0013: newobj instance void valuetype [mscorlib]System.Nullable`1<int32>::.ctor(!0) |
||||
IL_0018: ret |
||||
} // end of method NullPropagation::InvokeDelegate |
||||
|
||||
.method private hidebysig instance void |
||||
NotNullPropagation(class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass c) cil managed |
||||
{ |
||||
// Code size 53 (0x35) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.1 |
||||
IL_0001: brfalse.s IL_0015 |
||||
|
||||
IL_0003: ldarg.1 |
||||
IL_0004: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_0009: brfalse.s IL_0015 |
||||
|
||||
IL_000b: ldstr "non-zero" |
||||
IL_0010: call void [mscorlib]System.Console::WriteLine(string) |
||||
IL_0015: ldarg.1 |
||||
IL_0016: brfalse.s IL_0020 |
||||
|
||||
IL_0018: ldarg.1 |
||||
IL_0019: ldfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_001e: brtrue.s IL_002a |
||||
|
||||
IL_0020: ldstr "null or zero" |
||||
IL_0025: call void [mscorlib]System.Console::WriteLine(string) |
||||
IL_002a: ldstr "end of method" |
||||
IL_002f: call void [mscorlib]System.Console::WriteLine(string) |
||||
IL_0034: ret |
||||
} // end of method NullPropagation::NotNullPropagation |
||||
|
||||
.method private hidebysig instance void |
||||
Setter(class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass c) cil managed |
||||
{ |
||||
// Code size 26 (0x1a) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.1 |
||||
IL_0001: brfalse.s IL_000a |
||||
|
||||
IL_0003: ldarg.1 |
||||
IL_0004: ldc.i4.1 |
||||
IL_0005: stfld int32 ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::IntVal |
||||
IL_000a: call void [mscorlib]System.Console::WriteLine() |
||||
IL_000f: ldarg.1 |
||||
IL_0010: brfalse.s IL_0019 |
||||
|
||||
IL_0012: ldarg.1 |
||||
IL_0013: ldnull |
||||
IL_0014: callvirt instance void ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass::set_Property(class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation/MyClass) |
||||
IL_0019: ret |
||||
} // end of method NullPropagation::Setter |
||||
|
||||
.method public hidebysig specialname rtspecialname |
||||
instance void .ctor() cil managed |
||||
{ |
||||
// Code size 7 (0x7) |
||||
.maxstack 8 |
||||
IL_0000: ldarg.0 |
||||
IL_0001: call instance void [mscorlib]System.Object::.ctor() |
||||
IL_0006: ret |
||||
} // end of method NullPropagation::.ctor |
||||
|
||||
} // end of class ICSharpCode.Decompiler.Tests.TestCases.Pretty.NullPropagation |
||||
|
||||
|
||||
// ============================================================= |
||||
|
||||
// *********** DISASSEMBLY COMPLETE *********************** |
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,79 @@
@@ -0,0 +1,79 @@
|
||||
// Copyright (c) 2018 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.Diagnostics; |
||||
using System.Linq; |
||||
|
||||
namespace ICSharpCode.Decompiler.IL |
||||
{ |
||||
partial class NullableUnwrap |
||||
{ |
||||
public NullableUnwrap(StackType unwrappedType, ILInstruction argument) |
||||
: base(OpCode.NullableUnwrap, argument) |
||||
{ |
||||
this.ResultType = unwrappedType; |
||||
} |
||||
|
||||
internal override void CheckInvariant(ILPhase phase) |
||||
{ |
||||
base.CheckInvariant(phase); |
||||
Debug.Assert(Argument.ResultType == StackType.O, "nullable.unwrap expects nullable type as input"); |
||||
Debug.Assert(Ancestors.Any(a => a is NullableRewrap)); |
||||
} |
||||
|
||||
public override void WriteTo(ITextOutput output, ILAstWritingOptions options) |
||||
{ |
||||
output.Write("nullable.unwrap "); |
||||
output.Write(ResultType); |
||||
output.Write('('); |
||||
Argument.WriteTo(output, options); |
||||
output.Write(')'); |
||||
} |
||||
|
||||
public override StackType ResultType { get; } |
||||
} |
||||
|
||||
partial class NullableRewrap |
||||
{ |
||||
internal override void CheckInvariant(ILPhase phase) |
||||
{ |
||||
base.CheckInvariant(phase); |
||||
Debug.Assert(Argument.HasFlag(InstructionFlags.MayUnwrapNull)); |
||||
} |
||||
|
||||
public override InstructionFlags DirectFlags => InstructionFlags.ControlFlow; |
||||
|
||||
protected override InstructionFlags ComputeFlags() |
||||
{ |
||||
// Convert MayUnwrapNull flag to ControlFlow flag.
|
||||
// Also, remove EndpointUnreachable flag, because the end-point is reachable through
|
||||
// the implicit nullable.unwrap branch.
|
||||
const InstructionFlags flagsToRemove = InstructionFlags.MayUnwrapNull | InstructionFlags.EndPointUnreachable; |
||||
return (Argument.Flags & ~flagsToRemove) | InstructionFlags.ControlFlow; |
||||
} |
||||
|
||||
public override StackType ResultType { |
||||
get { |
||||
if (Argument.ResultType == StackType.Void) |
||||
return StackType.Void; |
||||
else |
||||
return StackType.O; |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,236 @@
@@ -0,0 +1,236 @@
|
||||
// Copyright (c) 2018 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; |
||||
using System.Diagnostics; |
||||
using System.Text; |
||||
using ICSharpCode.Decompiler.TypeSystem; |
||||
using ICSharpCode.Decompiler.Util; |
||||
|
||||
namespace ICSharpCode.Decompiler.IL.Transforms |
||||
{ |
||||
/// <summary>
|
||||
/// Transform that converts code patterns like "v != null ? v.M() : null" to "v?.M()"
|
||||
/// </summary>
|
||||
struct NullPropagationTransform |
||||
{ |
||||
internal static bool IsProtectedIfInst(IfInstruction ifInst) |
||||
{ |
||||
// We exclude logic.and to avoid turning
|
||||
// "logic.and(comp(interfaces != ldnull), call get_Count(interfaces))"
|
||||
// into "if ((interfaces?.Count ?? 0) != 0)".
|
||||
return (ifInst.MatchLogicAnd(out _, out _) || ifInst.MatchLogicOr(out _, out _)) |
||||
&& IfInstruction.IsInConditionSlot(ifInst); |
||||
} |
||||
|
||||
readonly ILTransformContext context; |
||||
|
||||
public NullPropagationTransform(ILTransformContext context) |
||||
{ |
||||
this.context = context; |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Check if "condition ? trueInst : falseInst" can be simplified using the null-conditional operator.
|
||||
/// Returns the replacement instruction, or null if no replacement is possible.
|
||||
/// </summary>
|
||||
internal ILInstruction Run(ILInstruction condition, ILInstruction trueInst, ILInstruction falseInst, Interval ilRange) |
||||
{ |
||||
Debug.Assert(context.Settings.NullPropagation); |
||||
Debug.Assert(!condition.MatchLogicNot(out _), "Caller should pass in positive condition"); |
||||
if (condition is Comp comp && comp.Left.MatchLdLoc(out var testedVar) && comp.Right.MatchLdNull()) { |
||||
if (comp.LiftingKind != ComparisonLiftingKind.None) |
||||
return null; |
||||
if (comp.Kind == ComparisonKind.Equality) { |
||||
// testedVar == null ? trueInst : falseInst
|
||||
return TryNullPropagation(testedVar, falseInst, trueInst, true, ilRange); |
||||
} else if (comp.Kind == ComparisonKind.Inequality) { |
||||
return TryNullPropagation(testedVar, trueInst, falseInst, true, ilRange); |
||||
} |
||||
} else if (NullableLiftingTransform.MatchHasValueCall(condition, out testedVar)) { |
||||
// testedVar.HasValue ? trueInst : falseInst
|
||||
return TryNullPropagation(testedVar, trueInst, falseInst, false, ilRange); |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// testedVar != null ? nonNullInst : nullInst
|
||||
/// </summary>
|
||||
ILInstruction TryNullPropagation(ILVariable testedVar, ILInstruction nonNullInst, ILInstruction nullInst, |
||||
bool testedVarHasReferenceType, Interval ilRange) |
||||
{ |
||||
bool removedRewrapOrNullableCtor = false; |
||||
if (NullableLiftingTransform.MatchNullableCtor(nonNullInst, out _, out var arg)) { |
||||
nonNullInst = arg; |
||||
removedRewrapOrNullableCtor = true; |
||||
} else if (nonNullInst.MatchNullableRewrap(out arg)) { |
||||
nonNullInst = arg; |
||||
removedRewrapOrNullableCtor = true; |
||||
} |
||||
if (!IsValidAccessChain(testedVar, testedVarHasReferenceType, nonNullInst, out var varLoad)) |
||||
return null; |
||||
// note: InferType will be accurate in this case because the access chain consists of calls and field accesses
|
||||
IType returnType = nonNullInst.InferType(); |
||||
if (nullInst.MatchLdNull()) { |
||||
context.Step("Null propagation (reference type)", nonNullInst); |
||||
// testedVar != null ? testedVar.AccessChain : null
|
||||
// => testedVar?.AccessChain
|
||||
IntroduceUnwrap(testedVar, varLoad); |
||||
return new NullableRewrap(nonNullInst) { ILRange = ilRange }; |
||||
} else if (nullInst.MatchDefaultValue(out var type) && type.IsKnownType(KnownTypeCode.NullableOfT)) { |
||||
context.Step("Null propagation (value type)", nonNullInst); |
||||
// testedVar != null ? testedVar.AccessChain : default(T?)
|
||||
// => testedVar?.AccessChain
|
||||
IntroduceUnwrap(testedVar, varLoad); |
||||
return new NullableRewrap(nonNullInst) { ILRange = ilRange }; |
||||
} else if (!removedRewrapOrNullableCtor && NullableType.IsNonNullableValueType(returnType)) { |
||||
context.Step("Null propagation with null coalescing", nonNullInst); |
||||
// testedVar != null ? testedVar.AccessChain : nullInst
|
||||
// => testedVar?.AccessChain ?? nullInst
|
||||
// (only valid if AccessChain returns a non-nullable value)
|
||||
IntroduceUnwrap(testedVar, varLoad); |
||||
return new NullCoalescingInstruction( |
||||
NullCoalescingKind.NullableWithValueFallback, |
||||
new NullableRewrap(nonNullInst), |
||||
nullInst |
||||
) { |
||||
UnderlyingResultType = nullInst.ResultType, |
||||
ILRange = ilRange |
||||
}; |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// if (x != null) x.AccessChain();
|
||||
/// => x?.AccessChain();
|
||||
/// </summary>
|
||||
internal void RunStatements(Block block, int pos) |
||||
{ |
||||
var ifInst = block.Instructions[pos] as IfInstruction; |
||||
if (ifInst == null || !ifInst.FalseInst.MatchNop()) |
||||
return; |
||||
if (ifInst.Condition is Comp comp && comp.Kind == ComparisonKind.Inequality |
||||
&& comp.Left.MatchLdLoc(out var testedVar) && comp.Right.MatchLdNull()) { |
||||
TryNullPropForVoidCall(testedVar, true, ifInst.TrueInst as Block, ifInst); |
||||
} else if (NullableLiftingTransform.MatchHasValueCall(ifInst.Condition, out testedVar)) { |
||||
TryNullPropForVoidCall(testedVar, false, ifInst.TrueInst as Block, ifInst); |
||||
} |
||||
} |
||||
|
||||
void TryNullPropForVoidCall(ILVariable testedVar, bool testedVarHasReferenceType, Block body, IfInstruction ifInst) |
||||
{ |
||||
if (body == null || body.Instructions.Count != 1) |
||||
return; |
||||
var bodyInst = body.Instructions[0]; |
||||
if (bodyInst.MatchNullableRewrap(out var arg)) { |
||||
bodyInst = arg; |
||||
} |
||||
if (!IsValidAccessChain(testedVar, testedVarHasReferenceType, bodyInst, out var varLoad)) |
||||
return; |
||||
context.Step("Null-propagation (void call)", body); |
||||
// if (testedVar != null) { testedVar.AccessChain(); }
|
||||
// => testedVar?.AccessChain();
|
||||
IntroduceUnwrap(testedVar, varLoad); |
||||
ifInst.ReplaceWith(new NullableRewrap( |
||||
bodyInst |
||||
) { ILRange = ifInst.ILRange }); |
||||
} |
||||
|
||||
bool IsValidAccessChain(ILVariable testedVar, bool testedVarHasReferenceType, ILInstruction inst, out ILInstruction finalLoad) |
||||
{ |
||||
finalLoad = null; |
||||
int chainLength = 0; |
||||
while (true) { |
||||
if (IsValidEndOfChain()) { |
||||
// valid end of chain
|
||||
finalLoad = inst; |
||||
return chainLength >= 1; |
||||
} else if (inst.MatchLdFld(out var target, out _)) { |
||||
inst = target; |
||||
} else if (inst is CallInstruction call && call.OpCode != OpCode.NewObj) { |
||||
if (call.Arguments.Count == 0) { |
||||
return false; |
||||
} |
||||
if (call.Method.IsStatic && !call.Method.IsExtensionMethod) { |
||||
return false; // only instance or extension methods can be called with ?. syntax
|
||||
} |
||||
if (call.Method.IsAccessor && !IsGetter(call.Method)) { |
||||
return false; // setter/adder/remover cannot be called with ?. syntax
|
||||
} |
||||
inst = call.Arguments[0]; |
||||
if (call.Method.DeclaringType.IsReferenceType == false && inst.MatchAddressOf(out var arg)) { |
||||
inst = arg; |
||||
} |
||||
// ensure the access chain does not contain any 'nullable.unwrap' that aren't directly part of the chain
|
||||
for (int i = 1; i < call.Arguments.Count; ++i) { |
||||
if (call.Arguments[i].HasFlag(InstructionFlags.MayUnwrapNull)) { |
||||
return false; |
||||
} |
||||
} |
||||
} else if (inst is NullableUnwrap unwrap) { |
||||
inst = unwrap.Argument; |
||||
} else { |
||||
// unknown node -> invalid chain
|
||||
return false; |
||||
} |
||||
chainLength++; |
||||
} |
||||
|
||||
bool IsValidEndOfChain() |
||||
{ |
||||
if (testedVarHasReferenceType) { |
||||
return inst.MatchLdLoc(testedVar); |
||||
} else { |
||||
return NullableLiftingTransform.MatchGetValueOrDefault(inst, testedVar); |
||||
} |
||||
} |
||||
} |
||||
|
||||
static bool IsGetter(IMethod method) |
||||
{ |
||||
return method.AccessorOwner is IProperty p && p.Getter == method; |
||||
} |
||||
|
||||
private void IntroduceUnwrap(ILVariable testedVar, ILInstruction varLoad) |
||||
{ |
||||
if (NullableLiftingTransform.MatchGetValueOrDefault(varLoad, testedVar)) { |
||||
varLoad.ReplaceWith(new NullableUnwrap( |
||||
varLoad.ResultType, |
||||
new LdLoc(testedVar) { ILRange = varLoad.Children[0].ILRange } |
||||
) { ILRange = varLoad.ILRange }); |
||||
} else { |
||||
// Wrap varLoad in nullable.unwrap:
|
||||
var children = varLoad.Parent.Children; |
||||
children[varLoad.ChildIndex] = new NullableUnwrap(testedVar.StackType, varLoad); |
||||
} |
||||
} |
||||
} |
||||
|
||||
class NullPropagationStatementTransform : IStatementTransform |
||||
{ |
||||
public void Run(Block block, int pos, StatementTransformContext context) |
||||
{ |
||||
if (!context.Settings.NullPropagation) |
||||
return; |
||||
new NullPropagationTransform(context).RunStatements(block, pos); |
||||
} |
||||
} |
||||
} |
Loading…
Reference in new issue