mirror of https://github.com/icsharpcode/ILSpy.git
				
				
			
				 10 changed files with 137 additions and 6 deletions
			
			
		| @ -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 @@ | |||||||
|  | // 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