mirror of https://github.com/icsharpcode/ILSpy.git
				
				
			
				 10 changed files with 137 additions and 6 deletions
			
			
		| @ -0,0 +1,36 @@@@ -0,0 +1,36 @@ | ||||
| using System; | ||||
| 
 | ||||
| namespace ICSharpCode.Decompiler.Tests.TestCases.Pretty | ||||
| { | ||||
| 	public class PatternMatching | ||||
| 	{ | ||||
| 		public bool SimpleTypePattern(object x) | ||||
| 		{ | ||||
| 			Use(x is string y); | ||||
| 			if (x is string z) | ||||
| 			{ | ||||
| 				Console.WriteLine(z); | ||||
| 			} | ||||
| 			return x is string w; | ||||
| 		} | ||||
| 
 | ||||
| 		public bool SimpleTypePatternWithShortcircuit(object x) | ||||
| 		{ | ||||
| 			Use(F() && x is string y && y.Contains("a")); | ||||
| 			if (F() && x is string z && z.Contains("a")) | ||||
| 			{ | ||||
| 				Console.WriteLine(z); | ||||
| 			} | ||||
| 			return F() && x is string w && w.Contains("a"); | ||||
| 		} | ||||
| 
 | ||||
| 		private bool F() | ||||
| 		{ | ||||
| 			return true; | ||||
| 		} | ||||
| 
 | ||||
| 		private void Use(bool x) | ||||
| 		{ | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @ -0,0 +1,73 @@@@ -0,0 +1,73 @@ | ||||
| // Copyright (c) 2021 Daniel Grunwald, 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.
 | ||||
| 
 | ||||
| #nullable enable | ||||
| 
 | ||||
| 
 | ||||
| namespace ICSharpCode.Decompiler.IL.Transforms | ||||
| { | ||||
| 
 | ||||
| 	class PatternMatchingTransform : IStatementTransform | ||||
| 	{ | ||||
| 		/// <summary>
 | ||||
| 		/// stloc V(isinst T(testedOperand))
 | ||||
| 		/// call Use(..., comp.o(ldloc V != ldnull))
 | ||||
| 		/// =>
 | ||||
| 		/// call Use(..., match.type[T](V = testedOperand))
 | ||||
| 		/// </summary>
 | ||||
| 		void IStatementTransform.Run(Block block, int pos, StatementTransformContext context) | ||||
| 		{ | ||||
| 			if (pos + 1 >= block.Instructions.Count) | ||||
| 				return; | ||||
| 			if (block.Instructions[pos] is not StLoc | ||||
| 				{ | ||||
| 					Variable: var v, | ||||
| 					Value: IsInst { Argument: var testedOperand, Type: var type } | ||||
| 				}) | ||||
| 			{ | ||||
| 				return; | ||||
| 			} | ||||
| 			if (!v.IsSingleDefinition) | ||||
| 				return; | ||||
| 			if (v.Kind is not (VariableKind.Local or VariableKind.StackSlot)) | ||||
| 				return; | ||||
| 			if (!v.Type.Equals(type)) | ||||
| 				return; | ||||
| 
 | ||||
| 			var result = ILInlining.FindLoadInNext(block.Instructions[pos + 1], v, testedOperand, InliningOptions.None); | ||||
| 			if (result.Type != ILInlining.FindResultType.Found) | ||||
| 				return; | ||||
| 			if (result.LoadInst is not LdLoc) | ||||
| 				return; | ||||
| 			if (!result.LoadInst.Parent!.MatchCompNotEqualsNull(out _)) | ||||
| 				return; | ||||
| 
 | ||||
| 			context.Step($"Type pattern matching {v.Name}", block.Instructions[pos]); | ||||
| 			// call Use(..., match.type[T](V = testedOperand))			
 | ||||
| 
 | ||||
| 			var target = result.LoadInst.Parent; | ||||
| 			var matchInstruction = new MatchInstruction(v, testedOperand) { | ||||
| 				CheckNotNull = true, | ||||
| 				CheckType = true | ||||
| 			}; | ||||
| 			target.ReplaceWith(matchInstruction.WithILRange(target)); | ||||
| 			block.Instructions.RemoveAt(pos); | ||||
| 			v.Kind = VariableKind.PatternLocal; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
					Loading…
					
					
				
		Reference in new issue