You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			1351 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			1351 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | using System.IO; | ||
|  | using System.Collections; | ||
|  | 
 | ||
|  | namespace PEAPI { | ||
|  | 
 | ||
|  | 	/**************************************************************************/   | ||
|  | 	/// <summary> | ||
|  | 	/// Descriptor for an IL instruction | ||
|  | 	/// </summary> | ||
|  | 	internal abstract class CILInstruction { | ||
|  | 		protected static readonly sbyte maxByteVal = 127; | ||
|  | 		protected static readonly sbyte minByteVal = -128; | ||
|  | 		protected static readonly byte leadByte = 0xFE; | ||
|  | 		protected static readonly uint USHeapIndex = 0x70000000; | ||
|  | 		protected static readonly int longInstrStart = (int)Op.arglist; | ||
|  | 		public bool twoByteInstr = false; | ||
|  | 		public uint size = 0; | ||
|  | 		public uint offset; | ||
|  | 
 | ||
|  | 		internal virtual bool Check(MetaData md)  | ||
|  | 		{ | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal virtual void Write(FileImage output) { } | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class CILByte : CILInstruction { | ||
|  | 		byte byteVal; | ||
|  | 
 | ||
|  | 		internal CILByte(byte bVal)  | ||
|  | 		{ | ||
|  | 			byteVal = bVal; | ||
|  | 			size = 1; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			output.Write(byteVal); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class Instr : CILInstruction { | ||
|  | 		protected int instr; | ||
|  | 
 | ||
|  | 		internal Instr(int inst)  | ||
|  | 		{ | ||
|  | 			if (inst >= longInstrStart) { | ||
|  | 				instr = inst - longInstrStart; | ||
|  | 				twoByteInstr = true; | ||
|  | 				size = 2; | ||
|  | 			} else { | ||
|  | 				instr = inst; | ||
|  | 				size = 1; | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			//Console.WriteLine("Writing instruction " + instr + " with size " + size); | ||
|  | 			if (twoByteInstr) output.Write(leadByte); | ||
|  | 			output.Write((byte)instr); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class IntInstr : Instr { | ||
|  | 		int val; | ||
|  | 		bool byteNum; | ||
|  | 
 | ||
|  | 		internal IntInstr(int inst, int num, bool byteSize) : base(inst)  | ||
|  | 		{ | ||
|  | 			val = num; | ||
|  | 			byteNum = byteSize; | ||
|  | 			if (byteNum) size++; | ||
|  | 			else size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			if (byteNum)  | ||
|  | 				output.Write((sbyte)val); | ||
|  | 			else  | ||
|  | 				output.Write(val);  | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class UIntInstr : Instr { | ||
|  | 		int val; | ||
|  | 		bool byteNum; | ||
|  | 
 | ||
|  | 		internal UIntInstr(int inst, int num, bool byteSize) : base(inst)  | ||
|  | 		{ | ||
|  | 			val = num; | ||
|  | 			byteNum = byteSize; | ||
|  | 			if (byteNum) size++; | ||
|  | 			else size += 2; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			if (byteNum) | ||
|  | 				output.Write((byte)val); | ||
|  | 			else | ||
|  | 				output.Write((ushort)val);  | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class LongInstr : Instr { | ||
|  | 		long val; | ||
|  | 
 | ||
|  | 		internal LongInstr(int inst, long l) : base(inst)  | ||
|  | 		{ | ||
|  | 			val = l; | ||
|  | 			size += 8; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(val); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class FloatInstr : Instr { | ||
|  | 		float fVal; | ||
|  | 
 | ||
|  | 		internal FloatInstr(int inst, float f) : base(inst)  | ||
|  | 		{ | ||
|  | 			fVal = f; | ||
|  | 			size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(fVal); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class DoubleInstr : Instr { | ||
|  | 		double val; | ||
|  | 
 | ||
|  | 		internal DoubleInstr(int inst, double d) : base(inst)  | ||
|  | 		{ | ||
|  | 			val = d; | ||
|  | 			size += 8; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(val); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class StringInstr : Instr { | ||
|  | 		string val; | ||
|  | 		byte[] bval;                                                   | ||
|  | 		uint strIndex; | ||
|  | 
 | ||
|  | 		internal StringInstr(int inst, string str) : base(inst)  | ||
|  | 		{ | ||
|  | 			val = str;   | ||
|  | 			size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal StringInstr (int inst, byte[] str) : base (inst)  | ||
|  | 		{ | ||
|  | 			bval = str; | ||
|  | 			size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override bool Check(MetaData md)  | ||
|  | 		{ | ||
|  | 			if (val != null) | ||
|  | 				strIndex = md.AddToUSHeap(val); | ||
|  | 			else | ||
|  | 				strIndex = md.AddToUSHeap (bval); | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(USHeapIndex  | strIndex); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class LabelInstr : CILInstruction { | ||
|  | 		CILLabel label; | ||
|  | 
 | ||
|  | 		internal LabelInstr(CILLabel lab)  | ||
|  | 		{ | ||
|  | 			label = lab; | ||
|  | 			label.AddLabelInstr(this); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class FieldInstr : Instr { | ||
|  | 		Field field; | ||
|  | 
 | ||
|  | 		internal FieldInstr(int inst, Field f) : base(inst)  | ||
|  | 		{ | ||
|  | 			field = f; | ||
|  | 			size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(field.Token()); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class MethInstr : Instr { | ||
|  | 		Method meth; | ||
|  | 
 | ||
|  | 		internal MethInstr(int inst, Method m) : base(inst)  | ||
|  | 		{ | ||
|  | 			meth = m; | ||
|  | 			size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(meth.Token()); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class SigInstr : Instr { | ||
|  | 		CalliSig signature; | ||
|  | 
 | ||
|  | 		internal SigInstr(int inst, CalliSig sig) : base(inst)  | ||
|  | 		{ | ||
|  | 			signature = sig; | ||
|  | 			size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override bool Check(MetaData md)  | ||
|  | 		{ | ||
|  | 			md.AddToTable(MDTable.StandAloneSig,signature); | ||
|  | 			signature.BuildTables(md); | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(signature.Token()); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class TypeInstr : Instr { | ||
|  | 		MetaDataElement theType; | ||
|  | 
 | ||
|  | 		internal TypeInstr(int inst, Type aType, MetaData md) : base(inst)  | ||
|  | 		{ | ||
|  | 			theType = aType.GetTypeSpec(md); | ||
|  | 			size += 4; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(theType.Token()); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class BranchInstr : Instr { | ||
|  | 		CILLabel dest; | ||
|  | 		private bool shortVer = true; | ||
|  | 		private int target = 0; | ||
|  | 
 | ||
|  | 		internal BranchInstr(int inst, CILLabel dst) : base(inst)  | ||
|  | 		{ | ||
|  | 			dest = dst; | ||
|  | 			dest.AddBranch(this); | ||
|  | 			size++; | ||
|  | 
 | ||
|  | 			if (inst >= (int) BranchOp.br && inst != (int) BranchOp.leave_s) { | ||
|  | 				shortVer = false; | ||
|  | 				size += 3; | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override bool Check(MetaData md)  | ||
|  | 		{ | ||
|  | 			target = (int)dest.GetLabelOffset() - (int)(offset + size); | ||
|  | 			return false; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			if (shortVer) | ||
|  | 				output.Write((sbyte)target); | ||
|  | 			else | ||
|  | 				output.Write(target); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	internal class SwitchInstr : Instr { | ||
|  | 		CILLabel[] cases; | ||
|  | 		uint numCases = 0; | ||
|  | 
 | ||
|  | 		internal SwitchInstr(int inst, CILLabel[] dsts) : base(inst)  | ||
|  | 		{ | ||
|  | 			cases = dsts; | ||
|  | 			if (cases != null) numCases = (uint)cases.Length; | ||
|  | 			size += 4 + (numCases * 4); | ||
|  | 			for (int i=0; i < numCases; i++) { | ||
|  | 				cases[i].AddBranch(this); | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			base.Write(output); | ||
|  | 			output.Write(numCases); | ||
|  | 			for (int i=0; i < numCases; i++) { | ||
|  | 				int target = (int)cases[i].GetLabelOffset() - (int)(offset + size); | ||
|  | 				output.Write(target); | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/**************************************************************************/   | ||
|  | 	/// <summary> | ||
|  | 	/// The IL instructions for a method | ||
|  | 	/// </summary> | ||
|  | 	public class CILInstructions  { | ||
|  | 		private static readonly uint ExHeaderSize = 4; | ||
|  | 		private static readonly uint FatExClauseSize = 24; | ||
|  | 		private static readonly uint SmlExClauseSize = 12; | ||
|  | 		private static readonly sbyte maxByteVal = 127; | ||
|  | 		private static readonly sbyte minByteVal = -128; | ||
|  | 		private static readonly byte maxUByteVal = 255; | ||
|  | 		private static readonly int smallSize = 64; | ||
|  | 		private static readonly ushort TinyFormat = 0x2; | ||
|  | 		private static readonly ushort FatFormat = 0x3003; | ||
|  | 		private static readonly ushort MoreSects = 0x8; | ||
|  | 		private static readonly ushort InitLocals = 0x10; | ||
|  | 		private static readonly uint FatSize = 12; | ||
|  | 		private static readonly byte FatExceptTable = 0x41; | ||
|  | 		private static readonly byte SmlExceptTable = 0x01;  | ||
|  | 
 | ||
|  | 		private MetaData metaData; | ||
|  | 		private ArrayList exceptions, blockStack; | ||
|  | 		//private bool codeChecked = false; | ||
|  | 		private static readonly int INITSIZE = 5; | ||
|  | 		private CILInstruction[] buffer = new CILInstruction[INITSIZE]; | ||
|  | 		private int tide = 0; | ||
|  | 		private uint offset = 0; | ||
|  | 		private ushort headerFlags = 0; | ||
|  | 		private short maxStack; | ||
|  | 		private uint paddingNeeded = 0; | ||
|  | 		private byte exceptHeader = 0; | ||
|  | 		uint localSigIx = 0; | ||
|  | 		uint codeSize = 0, exceptSize = 0; | ||
|  | 		bool tinyFormat, fatExceptionFormat = false; | ||
|  | 
 | ||
|  | 		public uint Offset { | ||
|  | 			get { return offset; } | ||
|  | 		}	 | ||
|  | 
 | ||
|  | 		internal CILInstructions(MetaData md)  | ||
|  | 		{ | ||
|  | 			metaData = md; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		private void AddToBuffer(CILInstruction inst)  | ||
|  | 		{ | ||
|  | 			if (tide >= buffer.Length) { | ||
|  | 				CILInstruction[] tmp = buffer; | ||
|  | 				buffer = new CILInstruction[tmp.Length * 2]; | ||
|  | 				for (int i=0; i < tide; i++) { | ||
|  | 					buffer[i] = tmp[i]; | ||
|  | 				} | ||
|  | 			} | ||
|  | 			//Console.WriteLine("Adding instruction at offset " + offset + " with size " + inst.size); | ||
|  | 			inst.offset = offset; | ||
|  | 			offset += inst.size; | ||
|  | 			buffer[tide++] = inst; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add a simple IL instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="inst">the IL instruction</param> | ||
|  | 		public void Inst(Op inst)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new Instr((int)inst)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add an IL instruction with an integer parameter | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="inst">the IL instruction</param> | ||
|  | 		/// <param name="val">the integer parameter value</param> | ||
|  | 		public void IntInst(IntOp inst, int val)  | ||
|  | 		{ | ||
|  | 			int instr = (int)inst; | ||
|  | 			if ((inst == IntOp.ldc_i4_s) || (inst == IntOp.ldc_i4))  | ||
|  | 				AddToBuffer(new IntInstr(instr,val,(inst == IntOp.ldc_i4_s))); | ||
|  | 			else | ||
|  | 				AddToBuffer(new UIntInstr(instr,val,((inst < IntOp.ldc_i4_s) || | ||
|  | 								(inst == IntOp.unaligned)))); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the load long instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="cVal">the long value</param> | ||
|  | 		public void ldc_i8(long cVal)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new LongInstr(0x21,cVal)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the load float32 instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="cVal">the float value</param> | ||
|  | 		public void ldc_r4(float cVal)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new FloatInstr(0x22,cVal)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the load float64 instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="cVal">the float value</param> | ||
|  | 		public void ldc_r8(double cVal)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new DoubleInstr(0x23,cVal)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the load string instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="str">the string value</param> | ||
|  | 		public void ldstr(string str)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new StringInstr(0x72,str)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the load string instruction | ||
|  | 		/// </summary> | ||
|  | 		public void ldstr (byte[] str)  | ||
|  | 		{ | ||
|  | 			AddToBuffer (new StringInstr (0x72, str)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the calli instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="sig">the signature for the calli</param> | ||
|  | 		public void calli(CalliSig sig)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new SigInstr(0x29,sig)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add a label to the CIL instructions | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="lab">the label to be added</param> | ||
|  | 		public void CodeLabel(CILLabel lab)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new LabelInstr(lab)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add an instruction with a field parameter | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="inst">the CIL instruction</param> | ||
|  | 		/// <param name="f">the field parameter</param> | ||
|  | 		public void FieldInst(FieldOp inst, Field f)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new FieldInstr((int)inst,f)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add an instruction with a method parameter | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="inst">the CIL instruction</param> | ||
|  | 		/// <param name="m">the method parameter</param> | ||
|  | 		public void MethInst(MethodOp inst, Method m)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new MethInstr((int)inst,m)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add an instruction with a type parameter | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="inst">the CIL instruction</param> | ||
|  | 		/// <param name="t">the type argument for the CIL instruction</param> | ||
|  | 		public void TypeInst(TypeOp inst, Type aType)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new TypeInstr((int)inst,aType,metaData)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add a branch instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="inst">the branch instruction</param> | ||
|  | 		/// <param name="lab">the label that is the target of the branch</param> | ||
|  | 		public void Branch(BranchOp inst,  CILLabel lab)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new BranchInstr((int)inst,lab)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add a switch instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="labs">the target labels for the switch</param> | ||
|  | 		public void Switch(CILLabel[] labs)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new SwitchInstr(0x45,labs)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add a byte to the CIL instructions (.emitbyte) | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="bVal"></param> | ||
|  | 		public void emitbyte(byte bVal)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new CILByte(bVal)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add an instruction which puts an integer on TOS.  This method | ||
|  | 		/// selects the correct instruction based on the value of the integer. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="i">the integer value</param> | ||
|  | 		public void PushInt(int i)  | ||
|  | 		{ | ||
|  | 			if (i == -1) { | ||
|  | 				AddToBuffer(new Instr((int)Op.ldc_i4_m1)); | ||
|  | 			} else if ((i >= 0) && (i <= 8)) { | ||
|  | 				Op op = (Op)(Op.ldc_i4_0 + i); | ||
|  | 				AddToBuffer(new Instr((int)op)); | ||
|  | 			} else if ((i >= minByteVal) && (i <= maxByteVal)) { | ||
|  | 				AddToBuffer(new IntInstr((int)IntOp.ldc_i4_s,i,true)); | ||
|  | 			} else { | ||
|  | 				AddToBuffer(new IntInstr((int)IntOp.ldc_i4,i,false));  | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the instruction to load a long on TOS | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="l">the long value</param> | ||
|  | 		public void PushLong(long l)  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new LongInstr(0x21,l)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add an instruction to push the boolean value true on TOS | ||
|  | 		/// </summary> | ||
|  | 		public void PushTrue()  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new Instr((int)Op.ldc_i4_1)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		///  Add an instruction to push the boolean value false on TOS | ||
|  | 		/// </summary> | ||
|  | 		public void PushFalse()  | ||
|  | 		{ | ||
|  | 			AddToBuffer(new Instr((int)Op.ldc_i4_0)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the instruction to load an argument on TOS.  This method | ||
|  | 		/// selects the correct instruction based on the value of argNo | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="argNo">the number of the argument</param> | ||
|  | 		public void LoadArg(int argNo)  | ||
|  | 		{ | ||
|  | 			if (argNo < 4) { | ||
|  | 				int op = (int)Op.ldarg_0 + argNo; | ||
|  | 				AddToBuffer(new Instr(op)); | ||
|  | 			} else if (argNo <= maxUByteVal) { | ||
|  | 				AddToBuffer(new UIntInstr((int)IntOp.ldarg,argNo,true)); | ||
|  | 			} else { | ||
|  | 				AddToBuffer(new UIntInstr(0x09,argNo,false));  | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the instruction to load the address of an argument on TOS. | ||
|  | 		/// This method selects the correct instruction based on the value | ||
|  | 		/// of argNo. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="argNo">the number of the argument</param> | ||
|  | 		public void LoadArgAdr(int argNo)  | ||
|  | 		{ | ||
|  | 			if (argNo <= maxUByteVal) { | ||
|  | 				AddToBuffer(new UIntInstr((int)IntOp.ldarga,argNo,true)); | ||
|  | 			} else { | ||
|  | 				AddToBuffer(new UIntInstr(0x0A,argNo,false));  | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the instruction to load a local on TOS.  This method selects | ||
|  | 		/// the correct instruction based on the value of locNo. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="locNo">the number of the local to load</param> | ||
|  | 		public void LoadLocal(int locNo)  | ||
|  | 		{ | ||
|  | 			if (locNo < 4) { | ||
|  | 				int op = (int)Op.ldloc_0 + locNo; | ||
|  | 				AddToBuffer(new Instr(op)); | ||
|  | 			} else if (locNo <= maxUByteVal) { | ||
|  | 				AddToBuffer(new UIntInstr((int)IntOp.ldloc,locNo,true)); | ||
|  | 			} else { | ||
|  | 				AddToBuffer(new UIntInstr(0x0C,locNo,false));  | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the instruction to load the address of a local on TOS. | ||
|  | 		/// This method selects the correct instruction based on the  | ||
|  | 		/// value of locNo. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="locNo">the number of the local</param> | ||
|  | 		public void LoadLocalAdr(int locNo)  | ||
|  | 		{ | ||
|  | 			if (locNo <= maxUByteVal) { | ||
|  | 				AddToBuffer(new UIntInstr((int)IntOp.ldloca,locNo,true)); | ||
|  | 			} else { | ||
|  | 				AddToBuffer(new UIntInstr(0x0D,locNo,false));  | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the instruction to store to an argument.  This method | ||
|  | 		/// selects the correct instruction based on the value of argNo. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="argNo">the argument to be stored to</param> | ||
|  | 		public void StoreArg(int argNo)  | ||
|  | 		{ | ||
|  | 			if (argNo <= maxUByteVal) { | ||
|  | 				AddToBuffer(new UIntInstr((int)IntOp.starg,argNo,true)); | ||
|  | 			} else { | ||
|  | 				AddToBuffer(new UIntInstr(0x0B,argNo,false));  | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the instruction to store to a local.  This method selects | ||
|  | 		/// the correct instruction based on the value of locNo. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="locNo">the local to be stored to</param> | ||
|  | 		public void StoreLocal(int locNo)  | ||
|  | 		{ | ||
|  | 			if (locNo < 4) { | ||
|  | 				int op = (int)Op.stloc_0 + locNo; | ||
|  | 				AddToBuffer(new Instr(op)); | ||
|  | 			} else if (locNo <= maxUByteVal) { | ||
|  | 				AddToBuffer(new UIntInstr((int)IntOp.stloc,locNo,true)); | ||
|  | 			} else { | ||
|  | 				AddToBuffer(new UIntInstr(0x0E,locNo,false));  | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new CIL label.  To place the label in the CIL instruction | ||
|  | 		/// stream use CodeLabel. | ||
|  | 		/// </summary> | ||
|  | 		/// <returns>a new CIL label</returns> | ||
|  | 		public CILLabel NewLabel()  | ||
|  | 		{ | ||
|  | 			return new CILLabel(); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public void AddTryBlock(TryBlock tryBlock)  | ||
|  | 		{ | ||
|  | 			if (exceptions == null)  | ||
|  | 				exceptions = new ArrayList(); | ||
|  | 			else if (exceptions.Contains(tryBlock)) return; | ||
|  | 			exceptions.Add(tryBlock); | ||
|  | 			tryBlock.ResolveCatchBlocks (metaData); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new label at this position in the code buffer | ||
|  | 		/// </summary> | ||
|  | 		/// <returns>the label at the current position</returns> | ||
|  | 		public CILLabel NewCodedLabel()  | ||
|  | 		{ | ||
|  | 			CILLabel lab = new CILLabel(); | ||
|  | 			AddToBuffer(new LabelInstr(lab)); | ||
|  | 			return lab; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Mark this position as the start of a new block | ||
|  | 		/// (try, catch, filter, finally or fault) | ||
|  | 		/// </summary> | ||
|  | 		public void StartBlock()  | ||
|  | 		{ | ||
|  | 			if (blockStack == null) blockStack = new ArrayList(); | ||
|  | 			blockStack.Insert(0,NewCodedLabel()); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Mark this position as the end of the last started block and | ||
|  | 		/// make it a try block.  This try block is added to the current  | ||
|  | 		/// instructions (ie do not need to call AddTryBlock) | ||
|  | 		/// </summary> | ||
|  | 		/// <returns>The try block just ended</returns> | ||
|  | 		public TryBlock EndTryBlock()  | ||
|  | 		{ | ||
|  | 			TryBlock tBlock = new TryBlock((CILLabel)blockStack[0],NewCodedLabel()); | ||
|  | 			blockStack.RemoveAt(0); | ||
|  | 			AddTryBlock(tBlock); | ||
|  | 			return tBlock; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Mark this position as the end of the last started block and | ||
|  | 		/// make it a catch block.  This catch block is associated with the | ||
|  | 		/// specified try block. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="exceptType">the exception type to be caught</param> | ||
|  | 		/// <param name="tryBlock">the try block associated with this catch block</param> | ||
|  | 		public void EndCatchBlock(Class exceptType, TryBlock tryBlock)  | ||
|  | 		{ | ||
|  | 			Catch catchBlock = new Catch(exceptType,(CILLabel)blockStack[0], | ||
|  | 					NewCodedLabel()); | ||
|  | 			tryBlock.AddHandler(catchBlock); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Mark this position as the end of the last started block and | ||
|  | 		/// make it a filter block.  This filter block is associated with the | ||
|  | 		/// specified try block. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="filterLab">the label where the filter code is</param> | ||
|  | 		/// <param name="tryBlock">the try block associated with this filter block</param> | ||
|  | 		public void EndFilterBlock(CILLabel filterLab, TryBlock tryBlock)  | ||
|  | 		{ | ||
|  | 			Filter filBlock = new Filter(filterLab,(CILLabel)blockStack[0],NewCodedLabel()); | ||
|  | 			tryBlock.AddHandler(filBlock); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Mark this position as the end of the last started block and | ||
|  | 		/// make it a finally block.  This finally block is associated with the | ||
|  | 		/// specified try block. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="tryBlock">the try block associated with this finally block</param> | ||
|  | 		public void EndFinallyBlock(TryBlock tryBlock)  | ||
|  | 		{ | ||
|  | 			Finally finBlock= new Finally((CILLabel)blockStack[0],NewCodedLabel()); | ||
|  | 			tryBlock.AddHandler(finBlock); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Mark this position as the end of the last started block and | ||
|  | 		/// make it a fault block.  This fault block is associated with the | ||
|  | 		/// specified try block. | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="tryBlock">the try block associated with this fault block</param> | ||
|  | 		public void EndFaultBlock(TryBlock tryBlock)  | ||
|  | 		{ | ||
|  | 			Fault fBlock= new Fault((CILLabel)blockStack[0],NewCodedLabel()); | ||
|  | 			tryBlock.AddHandler(fBlock); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal uint GetCodeSize()  | ||
|  | 		{ | ||
|  | 			return codeSize + paddingNeeded + exceptSize; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal void CheckCode(uint locSigIx, bool initLocals, int maxStack)  | ||
|  | 		{ | ||
|  | 			if (tide == 0) return; | ||
|  | 			bool changed = true; | ||
|  | 			while (changed) { | ||
|  | 				changed = false; | ||
|  | 				for (int i=0; i < tide; i++) { | ||
|  | 					changed = buffer[i].Check(metaData) || changed; | ||
|  | 				} | ||
|  | 				if (changed) { | ||
|  | 					for (int i=1; i < tide; i++) { | ||
|  | 						buffer[i].offset = buffer[i-1].offset + buffer[i-1].size; | ||
|  | 					} | ||
|  | 					offset = buffer[tide-1].offset + buffer[tide-1].size; | ||
|  | 				} | ||
|  | 			} | ||
|  | 			codeSize = offset; | ||
|  | 			// Console.WriteLine("codeSize before header added = " + codeSize); | ||
|  | 			if ((offset < smallSize) && (maxStack <= 8) && (locSigIx == 0) && (exceptions == null)) { | ||
|  | 				// can use tiny header | ||
|  | 				//Console.WriteLine("Tiny Header"); | ||
|  | 				tinyFormat = true; | ||
|  | 				headerFlags = (ushort)(TinyFormat | ((ushort)codeSize << 2)); | ||
|  | 				codeSize++; | ||
|  | 				if ((codeSize % 4) != 0) { paddingNeeded = 4 - (codeSize % 4); } | ||
|  | 			} else { | ||
|  | 				//Console.WriteLine("Fat Header"); | ||
|  | 				tinyFormat = false; | ||
|  | 				localSigIx = locSigIx; | ||
|  | 				this.maxStack = (short)maxStack; | ||
|  | 				headerFlags = FatFormat; | ||
|  | 				if (exceptions != null) { | ||
|  | 					// Console.WriteLine("Got exceptions"); | ||
|  | 					headerFlags |= MoreSects; | ||
|  | 					uint numExceptClauses = 0; | ||
|  | 					for (int i=0; i < exceptions.Count; i++) { | ||
|  | 						TryBlock tryBlock = (TryBlock)exceptions[i]; | ||
|  | 						tryBlock.SetSize(); | ||
|  | 						numExceptClauses += (uint)tryBlock.NumHandlers(); | ||
|  | 						if (tryBlock.isFat()) fatExceptionFormat = true; | ||
|  | 					} | ||
|  | 
 | ||
|  | 					uint data_size = ExHeaderSize + numExceptClauses * | ||
|  | 						(fatExceptionFormat ? FatExClauseSize : SmlExClauseSize); | ||
|  | 
 | ||
|  | 					if (data_size > 255) | ||
|  | 						fatExceptionFormat = true; | ||
|  | 
 | ||
|  | 					// Console.WriteLine("numexceptclauses = " + numExceptClauses); | ||
|  | 					if (fatExceptionFormat) { | ||
|  | 						// Console.WriteLine("Fat exception format"); | ||
|  | 						exceptHeader = FatExceptTable; | ||
|  | 						exceptSize = ExHeaderSize + numExceptClauses * FatExClauseSize; | ||
|  | 					} else { | ||
|  | 						// Console.WriteLine("Tiny exception format"); | ||
|  | 						exceptHeader = SmlExceptTable; | ||
|  | 						exceptSize = ExHeaderSize + numExceptClauses * SmlExClauseSize; | ||
|  | 					} | ||
|  | 					// Console.WriteLine("exceptSize = " + exceptSize); | ||
|  | 				} | ||
|  | 				if (initLocals) headerFlags |= InitLocals; | ||
|  | 				if ((offset % 4) != 0) { paddingNeeded = 4 - (offset % 4); } | ||
|  | 				codeSize += FatSize; | ||
|  | 			} | ||
|  | 			// Console.WriteLine("codeSize = " + codeSize + "  headerFlags = " +  | ||
|  | 			//                   Hex.Short(headerFlags)); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal void Write(FileImage output)  | ||
|  | 		{ | ||
|  | 			// Console.WriteLine("Writing header flags = " + Hex.Short(headerFlags)); | ||
|  | 			if (tinyFormat) { | ||
|  | 				// Console.WriteLine("Writing tiny code"); | ||
|  | 				output.Write((byte)headerFlags); | ||
|  | 			} else { | ||
|  | 				// Console.WriteLine("Writing fat code"); | ||
|  | 				output.Write(headerFlags); | ||
|  | 				output.Write((ushort)maxStack); | ||
|  | 				output.Write(offset); | ||
|  | 				output.Write(localSigIx); | ||
|  | 			} | ||
|  | 			// Console.WriteLine(Hex.Int(tide) + " CIL instructions"); | ||
|  | 			// Console.WriteLine("starting instructions at " + output.Seek(0,SeekOrigin.Current)); | ||
|  | 			for (int i=0; i < tide; i++) { | ||
|  | 				buffer[i].Write(output); | ||
|  | 			} | ||
|  | 			// Console.WriteLine("ending instructions at " + output.Seek(0,SeekOrigin.Current)); | ||
|  | 			for (int i=0; i < paddingNeeded; i++) { output.Write((byte)0); } | ||
|  | 			if (exceptions != null) { | ||
|  | 				// Console.WriteLine("Writing exceptions"); | ||
|  | 				// Console.WriteLine("header = " + Hex.Short(exceptHeader) + " exceptSize = " + Hex.Int(exceptSize)); | ||
|  | 				output.Write(exceptHeader); | ||
|  | 				output.Write3Bytes((uint)exceptSize); | ||
|  | 				for (int i=0; i < exceptions.Count; i++) { | ||
|  | 					TryBlock tryBlock = (TryBlock)exceptions[i]; | ||
|  | 					tryBlock.Write(output,fatExceptionFormat); | ||
|  | 				} | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/**************************************************************************/   | ||
|  | 	public abstract class CodeBlock { | ||
|  | 
 | ||
|  | 		private static readonly int maxCodeSize = 255; | ||
|  | 		protected CILLabel start, end; | ||
|  | 		protected bool small = true; | ||
|  | 
 | ||
|  | 		public CodeBlock(CILLabel start, CILLabel end)  | ||
|  | 		{ | ||
|  | 			this.start = start; | ||
|  | 			this.end = end; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal virtual bool isFat()  | ||
|  | 		{ | ||
|  | 			// Console.WriteLine("block start = " + start.GetLabelOffset() + | ||
|  | 			//                  "  block end = " + end.GetLabelOffset()); | ||
|  | 			return (end.GetLabelOffset() - start.GetLabelOffset()) > maxCodeSize; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal virtual void Write(FileImage output, bool fatFormat)  | ||
|  | 		{ | ||
|  | 			if (fatFormat) output.Write(start.GetLabelOffset()); | ||
|  | 			else output.Write((short)start.GetLabelOffset()); | ||
|  | 			uint len = end.GetLabelOffset() - start.GetLabelOffset(); | ||
|  | 			if (fatFormat) output.Write(len); | ||
|  | 			else output.Write((byte)len); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/// <summary> | ||
|  | 	/// The descriptor for a guarded block (.try) | ||
|  | 	/// </summary> | ||
|  | 	public class TryBlock : CodeBlock { | ||
|  | 		protected bool fatFormat = false; | ||
|  | 		protected int flags = 0; | ||
|  | 		ArrayList handlers = new ArrayList(); | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new try block | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="start">start label for the try block</param> | ||
|  | 		/// <param name="end">end label for the try block</param> | ||
|  | 		public TryBlock(CILLabel start, CILLabel end) : base(start,end) { } | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add a handler to this try block | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="handler">a handler to be added to the try block</param> | ||
|  | 		public void AddHandler(HandlerBlock handler)  | ||
|  | 		{ | ||
|  | 			flags = handler.GetFlag(); | ||
|  | 			handlers.Add(handler); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal void SetSize()  | ||
|  | 		{ | ||
|  | 			fatFormat = base.isFat(); | ||
|  | 			if (fatFormat) return; | ||
|  | 			for (int i=0; i < handlers.Count; i++) { | ||
|  | 				HandlerBlock handler = (HandlerBlock)handlers[i]; | ||
|  | 				if (handler.isFat()) { | ||
|  | 					fatFormat = true; | ||
|  | 					return; | ||
|  | 				} | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal int NumHandlers()  | ||
|  | 		{ | ||
|  | 			return handlers.Count; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override bool isFat()  | ||
|  | 		{ | ||
|  | 			return fatFormat; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		//Hackish | ||
|  | 		internal void ResolveCatchBlocks (MetaData md) | ||
|  | 		{ | ||
|  | 			for (int i=0; i < handlers.Count; i++) { | ||
|  | 				Catch c = handlers [i] as Catch; | ||
|  | 				if (c != null) | ||
|  | 					c.ResolveType (md); | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output, bool fatFormat)  | ||
|  | 		{ | ||
|  | 			// Console.WriteLine("writing exception details"); | ||
|  | 			for (int i=0; i < handlers.Count; i++) { | ||
|  | 				// Console.WriteLine("Except block " + i); | ||
|  | 				HandlerBlock handler = (HandlerBlock)handlers[i]; | ||
|  | 				if (fatFormat) output.Write(flags); | ||
|  | 				else output.Write((short)flags); | ||
|  | 				// Console.WriteLine("flags = " + Hex.Short(flags)); | ||
|  | 				base.Write(output,fatFormat); | ||
|  | 				handler.Write(output,fatFormat); | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	public abstract class HandlerBlock : CodeBlock  { | ||
|  | 
 | ||
|  | 		protected static readonly short ExceptionFlag = 0; | ||
|  | 		protected static readonly short FilterFlag = 0x01; | ||
|  | 		protected static readonly short FinallyFlag = 0x02; | ||
|  | 		protected static readonly short FaultFlag = 0x04; | ||
|  | 
 | ||
|  | 		public HandlerBlock(CILLabel start, CILLabel end) : base(start,end) { } | ||
|  | 
 | ||
|  | 		internal virtual short GetFlag() { return ExceptionFlag; } | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output, bool fatFormat)  | ||
|  | 		{ | ||
|  | 			base.Write(output,fatFormat); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/// <summary> | ||
|  | 	/// The descriptor for a catch clause (.catch) | ||
|  | 	/// </summary> | ||
|  | 	public class Catch : HandlerBlock  { | ||
|  | 
 | ||
|  | 		MetaDataElement exceptType; | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new catch clause | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="except">the exception to be caught</param> | ||
|  | 		/// <param name="handlerStart">start of the handler code</param> | ||
|  | 		/// <param name="handlerEnd">end of the handler code</param> | ||
|  | 		public Catch(Class except, CILLabel handlerStart, CILLabel handlerEnd) | ||
|  | 			: base(handlerStart, handlerEnd) | ||
|  | 		{ | ||
|  | 			exceptType = except; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public Catch(Type except, CILLabel handlerStart, CILLabel handlerEnd) | ||
|  | 			: base(handlerStart,handlerEnd)  | ||
|  | 		{ | ||
|  | 			exceptType = except; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal void ResolveType (MetaData md) | ||
|  | 		{ | ||
|  | 		       exceptType = ((Type) exceptType).GetTypeSpec (md); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output, bool fatFormat)  | ||
|  | 		{ | ||
|  | 			base.Write(output,fatFormat); | ||
|  | 			output.Write(exceptType.Token()); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/// <summary> | ||
|  | 	/// The descriptor for a filter clause (.filter) | ||
|  | 	/// </summary> | ||
|  | 	public class Filter : HandlerBlock  { | ||
|  | 
 | ||
|  | 		CILLabel filterLabel; | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new filter clause | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="filterLabel">the label where the filter code starts</param> | ||
|  | 		/// <param name="handlerStart">the start of the handler code</param> | ||
|  | 		/// <param name="handlerEnd">the end of the handler code</param> | ||
|  | 		public Filter(CILLabel filterLabel, CILLabel handlerStart,  | ||
|  | 				CILLabel handlerEnd) : base(handlerStart,handlerEnd)  | ||
|  | 				{ | ||
|  | 			this.filterLabel = filterLabel; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override short GetFlag()  | ||
|  | 		{ | ||
|  | 			return FilterFlag;  | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output, bool fatFormat)  | ||
|  | 		{ | ||
|  | 			base.Write(output,fatFormat); | ||
|  | 			output.Write(filterLabel.GetLabelOffset()); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/// <summary> | ||
|  | 	/// Descriptor for a finally block (.finally) | ||
|  | 	/// </summary> | ||
|  | 	public class Finally : HandlerBlock  { | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new finally clause | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="finallyStart">start of finally code</param> | ||
|  | 		/// <param name="finallyEnd">end of finally code</param> | ||
|  | 		public Finally(CILLabel finallyStart, CILLabel finallyEnd) | ||
|  | 			: base(finallyStart,finallyEnd) { } | ||
|  | 
 | ||
|  | 		internal override short GetFlag()  | ||
|  | 		{ | ||
|  | 			return FinallyFlag;  | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output, bool fatFormat)  | ||
|  | 		{ | ||
|  | 			base.Write(output,fatFormat); | ||
|  | 			output.Write((int)0); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/// <summary> | ||
|  | 	/// Descriptor for a fault block (.fault) | ||
|  | 	/// </summary> | ||
|  | 	public class Fault : HandlerBlock  { | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new fault clause | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="faultStart">start of the fault code</param> | ||
|  | 		/// <param name="faultEnd">end of the fault code</param> | ||
|  | 		public Fault(CILLabel faultStart, CILLabel faultEnd) | ||
|  | 			: base(faultStart,faultEnd) { } | ||
|  | 
 | ||
|  | 		internal override short GetFlag()  | ||
|  | 		{ | ||
|  | 			return FaultFlag;  | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal override void Write(FileImage output, bool fatFormat)  | ||
|  | 		{ | ||
|  | 			base.Write(output,fatFormat); | ||
|  | 			output.Write((int)0); | ||
|  | 
 | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/**************************************************************************/   | ||
|  | 	/// <summary> | ||
|  | 	/// Descriptor for the locals for a method | ||
|  | 	/// </summary> | ||
|  | 	public class LocalSig : Signature { | ||
|  | 
 | ||
|  | 		private static readonly byte LocalSigByte = 0x7; | ||
|  | 		Local[] locals; | ||
|  | 
 | ||
|  | 		public LocalSig(Local[] locals)          | ||
|  | 		{ | ||
|  | 			this.locals = locals; | ||
|  | 			tabIx = MDTable.StandAloneSig; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void BuildTables(MetaData md)  | ||
|  | 		{ | ||
|  | 			if (done) return; | ||
|  | 			MemoryStream sig = new MemoryStream(); | ||
|  | 			sig.WriteByte(LocalSigByte); | ||
|  | 			MetaData.CompressNum((uint)locals.Length,sig); | ||
|  | 			for (int i=0; i < locals.Length; i++) { | ||
|  | 				((Local)locals[i]).TypeSig(sig); | ||
|  | 			} | ||
|  | 			sigIx = md.AddToBlobHeap(sig.ToArray()); | ||
|  | 			done = true; | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/**************************************************************************/   | ||
|  | 	/// <summary> | ||
|  | 	/// Signature for calli instruction | ||
|  | 	/// </summary> | ||
|  | 	public class CalliSig : Signature { | ||
|  | 
 | ||
|  | 		private static readonly byte Sentinel = 0x41; | ||
|  | 		CallConv callConv; | ||
|  | 		Type returnType; | ||
|  | 		Type[] parameters, optParams; | ||
|  | 		uint numPars = 0, numOptPars = 0; | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a signature for a calli instruction | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="cconv">calling conventions</param> | ||
|  | 		/// <param name="retType">return type</param> | ||
|  | 		/// <param name="pars">parameter types</param> | ||
|  | 		public CalliSig(CallConv cconv, Type retType, Type[] pars)  | ||
|  | 		{ | ||
|  | 			tabIx = MDTable.StandAloneSig; | ||
|  | 			callConv = cconv; | ||
|  | 			returnType = retType; | ||
|  | 			parameters = pars; | ||
|  | 			if (pars != null) numPars = (uint)pars.Length; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add the optional parameters to a vararg method | ||
|  | 		/// This method sets the vararg calling convention | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="optPars">the optional pars for the vararg call</param> | ||
|  | 		public void AddVarArgs(Type[] optPars)  | ||
|  | 		{ | ||
|  | 			optParams = optPars; | ||
|  | 			if (optPars != null) numOptPars = (uint)optPars.Length; | ||
|  | 			callConv |= CallConv.Vararg; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Add extra calling conventions to this callsite signature | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="cconv"></param> | ||
|  | 		public void AddCallingConv(CallConv cconv)  | ||
|  | 		{ | ||
|  | 			callConv |= cconv; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal sealed override void BuildTables(MetaData md)  | ||
|  | 		{ | ||
|  | 			if (done) return; | ||
|  | 			MemoryStream sig = new MemoryStream(); | ||
|  | 			sig.WriteByte((byte)callConv); | ||
|  | 			MetaData.CompressNum(numPars+numOptPars,sig); | ||
|  | 			returnType.TypeSig(sig); | ||
|  | 			for (int i=0; i < numPars; i++) { | ||
|  | 				parameters[i].TypeSig(sig); | ||
|  | 			} | ||
|  | 			sigIx = md.AddToBlobHeap(sig.ToArray()); | ||
|  | 			if (numOptPars > 0) { | ||
|  | 				sig.WriteByte(Sentinel); | ||
|  | 				for (int i=0; i < numOptPars; i++) { | ||
|  | 					optParams[i].TypeSig(sig); | ||
|  | 				} | ||
|  | 			} | ||
|  | 			done = true; | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/**************************************************************************/   | ||
|  | 	/// <summary> | ||
|  | 	/// Descriptor for a local of a method | ||
|  | 	/// </summary> | ||
|  | 	public class Local { | ||
|  | 
 | ||
|  | 		private static readonly byte Pinned = 0x45; | ||
|  | 		string name; | ||
|  | 		Type type; | ||
|  | 		bool pinned = false, byref = false; | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new local variable  | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="lName">name of the local variable</param> | ||
|  | 		/// <param name="lType">type of the local variable</param> | ||
|  | 		public Local(string lName, Type lType)  | ||
|  | 		{ | ||
|  | 			name = lName; | ||
|  | 			type = lType; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		/// <summary> | ||
|  | 		/// Create a new local variable that is byref and/or pinned | ||
|  | 		/// </summary> | ||
|  | 		/// <param name="lName">local name</param> | ||
|  | 		/// <param name="lType">local type</param> | ||
|  | 		/// <param name="byRef">is byref</param> | ||
|  | 		/// <param name="isPinned">has pinned attribute</param> | ||
|  | 		public Local(string lName, Type lType, bool byRef, bool isPinned) | ||
|  | 		{ | ||
|  | 			name = lName; | ||
|  | 			type = lType; | ||
|  | 			byref = byRef; | ||
|  | 			pinned = isPinned; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal void TypeSig(MemoryStream str)  | ||
|  | 		{ | ||
|  | 			if (pinned) str.WriteByte(Pinned); | ||
|  | 			type.TypeSig(str); | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/**************************************************************************/   | ||
|  | 	/// <summary> | ||
|  | 	/// A label in the IL | ||
|  | 	/// </summary> | ||
|  | 	public class CILLabel { | ||
|  | 
 | ||
|  | 		CILInstruction branch; | ||
|  | 		CILInstruction[] multipleBranches; | ||
|  | 		int tide = 0; | ||
|  | 		CILInstruction labInstr; | ||
|  | 		uint offset = 0; | ||
|  | 		bool absolute; | ||
|  | 
 | ||
|  | 
 | ||
|  | 		public CILLabel (uint offset, bool absolute)  | ||
|  | 		{ | ||
|  | 			this.offset = offset; | ||
|  | 			this.absolute = absolute; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		public CILLabel (uint offset) : this (offset, false) | ||
|  | 		{ | ||
|  | 		} | ||
|  | 
 | ||
|  | 
 | ||
|  | 		internal CILLabel()  | ||
|  | 		{ | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal void AddBranch(CILInstruction instr)  | ||
|  | 		{ | ||
|  | 			if (branch == null) { | ||
|  | 				branch = instr; | ||
|  | 				return; | ||
|  | 			} | ||
|  | 			if (multipleBranches == null) { | ||
|  | 				multipleBranches = new CILInstruction[2]; | ||
|  | 			} else if (tide >= multipleBranches.Length) { | ||
|  | 				CILInstruction[] tmp = multipleBranches; | ||
|  | 				multipleBranches = new CILInstruction[tmp.Length*2]; | ||
|  | 				for (int i=0; i < tide; i++) { | ||
|  | 					multipleBranches[i] = tmp[i]; | ||
|  | 				} | ||
|  | 			} | ||
|  | 			multipleBranches[tide++] = instr; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal void AddLabelInstr(LabelInstr lInstr)  | ||
|  | 		{ | ||
|  | 			labInstr = lInstr; | ||
|  | 		} | ||
|  | 
 | ||
|  | 		internal uint GetLabelOffset()  | ||
|  | 		{ | ||
|  | 			if (absolute) return offset; | ||
|  | 			if (labInstr == null) return 0; | ||
|  | 			return labInstr.offset + offset; | ||
|  | 		} | ||
|  | 
 | ||
|  | 	} | ||
|  | 
 | ||
|  | 
 | ||
|  | } | ||
|  | 
 | ||
|  | 
 |