You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			216 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			216 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //  | ||
|  | // EnsuresBlock.cs | ||
|  | //  | ||
|  | // Authors: | ||
|  | // 	Alexander Chebaturkin (chebaturkin@gmail.com) | ||
|  | //  | ||
|  | // Copyright (C) 2011 Alexander Chebaturkin | ||
|  | //  | ||
|  | // 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 Mono.CodeContracts.Static.AST; | ||
|  | using Mono.CodeContracts.Static.AST.Visitors; | ||
|  | using Mono.CodeContracts.Static.ControlFlow.Subroutines; | ||
|  | using Mono.CodeContracts.Static.DataStructures; | ||
|  | 
 | ||
|  | namespace Mono.CodeContracts.Static.ControlFlow.Blocks { | ||
|  | 	class EnsuresBlock<Label> : BlockWithLabels<Label> { | ||
|  | 		private const uint Mask = 0xC0000000u; | ||
|  | 		private const uint BeginOldMask = 0x80000000u; | ||
|  | 		private const uint EndOldMask = 0x40000000u; | ||
|  | 
 | ||
|  | 		private List<uint> overridingLabels; | ||
|  | 
 | ||
|  | 		public EnsuresBlock (SubroutineBase<Label> subroutine, ref int idGen) | ||
|  | 			: base (subroutine, ref idGen) | ||
|  | 		{ | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private new EnsuresSubroutine<Label> Subroutine | ||
|  | 		{ | ||
|  | 			get { return (EnsuresSubroutine<Label>) base.Subroutine; } | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public override int Count | ||
|  | 		{ | ||
|  | 			get | ||
|  | 			{ | ||
|  | 				if (this.overridingLabels != null) | ||
|  | 					return this.overridingLabels.Count; | ||
|  | 				return base.Count; | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public bool UsesOverriding | ||
|  | 		{ | ||
|  | 			get { return this.overridingLabels != null; } | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public override bool TryGetLabel (int index, out Label label) | ||
|  | 		{ | ||
|  | 			int originalOffset; | ||
|  | 			if (IsOriginal (index, out originalOffset)) | ||
|  | 				return base.TryGetLabel (originalOffset, out label); | ||
|  | 			label = default(Label); | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public Result OriginalForwardDecode<Data, Result, Visitor> (int index, Visitor visitor, Data data) | ||
|  | 			where Visitor : IAggregateVisitor<Label, Data, Result> | ||
|  | 		{ | ||
|  | 			Label label; | ||
|  | 			if (base.TryGetLabel (index, out label)) | ||
|  | 				return Subroutine.CodeProvider.Decode<Visitor, Data, Result> (label, visitor, data); | ||
|  | 
 | ||
|  | 			throw new InvalidOperationException ("should not happen"); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public override Result ForwardDecode<Data, Result, Visitor> (APC pc, Visitor visitor, Data data) | ||
|  | 		{ | ||
|  | 			Label label; | ||
|  | 			if (TryGetLabel (pc.Index, out label)) | ||
|  | 				return base.ForwardDecode<Data, Result, Visitor> (pc, visitor, data); | ||
|  | 
 | ||
|  | 			int endOldIndex; | ||
|  | 			if (IsBeginOld (pc.Index, out endOldIndex)) { | ||
|  | 				CFGBlock block = Subroutine.InferredBeginEndBijection (pc); | ||
|  | 				return visitor.BeginOld (pc, new APC (block, endOldIndex, pc.SubroutineContext), data); | ||
|  | 			} | ||
|  | 
 | ||
|  | 			int beginOldIndex; | ||
|  | 			if (IsEndOld (pc.Index, out beginOldIndex)) { | ||
|  | 				TypeNode endOldType; | ||
|  | 				CFGBlock block = Subroutine.InferredBeginEndBijection (pc, out endOldType); | ||
|  | 				return visitor.EndOld (pc, new APC (block, beginOldIndex, pc.SubroutineContext), endOldType, Dummy.Value, Dummy.Value, data); | ||
|  | 			} | ||
|  | 
 | ||
|  | 			return visitor.Nop (pc, data); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private bool IsEndOld (int index, out int beginOldIndex) | ||
|  | 		{ | ||
|  | 			if (this.overridingLabels != null && index < this.overridingLabels.Count && (this.overridingLabels [index] & EndOldMask) != 0) { | ||
|  | 				beginOldIndex = (int) (this.overridingLabels [index] & (EndOldMask - 1)); | ||
|  | 				return true; | ||
|  | 			} | ||
|  | 
 | ||
|  | 			beginOldIndex = 0; | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private bool IsBeginOld (int index, out int endOldIndex) | ||
|  | 		{ | ||
|  | 			if (this.overridingLabels != null && index < this.overridingLabels.Count && (this.overridingLabels [index] & BeginOldMask) != 0) { | ||
|  | 				endOldIndex = (int) (this.overridingLabels [index] & (EndOldMask - 1)); | ||
|  | 				return true; | ||
|  | 			} | ||
|  | 
 | ||
|  | 			endOldIndex = 0; | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private bool IsOriginal (int index, out int originalOffset) | ||
|  | 		{ | ||
|  | 			if (this.overridingLabels == null) { | ||
|  | 				originalOffset = index; | ||
|  | 				return true; | ||
|  | 			} | ||
|  | 			if (index < this.overridingLabels.Count && (this.overridingLabels [index] & Mask) == 0) { | ||
|  | 				originalOffset = (int) (this.overridingLabels [index] & (EndOldMask - 1)); | ||
|  | 				return true; | ||
|  | 			} | ||
|  | 
 | ||
|  | 			originalOffset = 0; | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public void StartOverridingLabels () | ||
|  | 		{ | ||
|  | 			this.overridingLabels = new List<uint> (); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public void BeginOld (int index) | ||
|  | 		{ | ||
|  | 			if (this.overridingLabels == null) { | ||
|  | 				StartOverridingLabels (); | ||
|  | 				for (int i = 0; i < index; ++i) | ||
|  | 					this.overridingLabels.Add ((uint) i); | ||
|  | 			} | ||
|  | 			this.overridingLabels.Add (BeginOldMask); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public void AddInstruction (int index) | ||
|  | 		{ | ||
|  | 			this.overridingLabels.Add ((uint) index); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public void EndOld (int index, TypeNode nextEndOldType) | ||
|  | 		{ | ||
|  | 			AddInstruction (index); | ||
|  | 			EndOldWithoutInstruction (nextEndOldType); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public void EndOldWithoutInstruction (TypeNode nextEndOldType) | ||
|  | 		{ | ||
|  | 			int endOldIndex = this.overridingLabels.Count; | ||
|  | 			CFGBlock beginBlock; | ||
|  | 			this.overridingLabels.Add ((uint) (EndOldMask | PatchPriorBeginOld (this, endOldIndex, out beginBlock))); | ||
|  | 			Subroutine.AddInferredOldMap (this.Index, endOldIndex, beginBlock, nextEndOldType); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private int PatchPriorBeginOld (CFGBlock endBlock, int endOldIndex, out CFGBlock beginBlock) | ||
|  | 		{ | ||
|  | 			for (int i = this == endBlock ? endOldIndex - 2 : Count - 1; i >= 0; i--) { | ||
|  | 				int endOldI; | ||
|  | 				if (IsBeginOld (i, out endOldI)) { | ||
|  | 					this.overridingLabels [i] = BeginOldMask | (uint)endOldIndex; | ||
|  | 					beginBlock = this; | ||
|  | 					Subroutine.AddInferredOldMap (this.Index, i, endBlock, default(TypeNode)); | ||
|  | 					return i; | ||
|  | 				} | ||
|  | 			} | ||
|  | 
 | ||
|  | 			IEnumerator<CFGBlock> enumerator = Subroutine.PredecessorBlocks (this).GetEnumerator (); | ||
|  | 			if (!enumerator.MoveNext ()) | ||
|  | 				throw new InvalidOperationException ("missing begin_old"); | ||
|  | 			int result = PatchPriorBeginOld (endBlock, endOldIndex, enumerator.Current, out beginBlock); | ||
|  | 			enumerator.MoveNext (); | ||
|  | 			return result; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private int PatchPriorBeginOld (CFGBlock endBlock, int endOldIndex, CFGBlock current, out CFGBlock beginBlock) | ||
|  | 		{ | ||
|  | 			var ensuresBlock = current as EnsuresBlock<Label>; | ||
|  | 			if (ensuresBlock != null) | ||
|  | 				return ensuresBlock.PatchPriorBeginOld (endBlock, endOldIndex, out beginBlock); | ||
|  | 			IEnumerator<CFGBlock> enumerator = current.Subroutine.PredecessorBlocks (current).GetEnumerator (); | ||
|  | 			if (!enumerator.MoveNext ()) | ||
|  | 				throw new InvalidOperationException ("missing begin_old"); | ||
|  | 
 | ||
|  | 			int result = PatchPriorBeginOld (endBlock, endOldIndex, enumerator.Current, out beginBlock); | ||
|  | 			enumerator.MoveNext (); | ||
|  | 			return result; | ||
|  | 		} | ||
|  | 	} | ||
|  | } |