You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			426 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			426 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //
 | |
| // 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.
 | |
| //
 | |
| // Copyright (C) Lluis Sanchez Gual, 2004
 | |
| //
 | |
| 
 | |
| #if !FULL_AOT_RUNTIME
 | |
| using System;
 | |
| using System.IO;
 | |
| using System.Collections;
 | |
| using System.Reflection.Emit;
 | |
| using System.Reflection;
 | |
| 
 | |
| namespace Mono.CodeGeneration
 | |
| {
 | |
| 	public class CodeBuilder
 | |
| 	{
 | |
| 		CodeBlock mainBlock;
 | |
| 		CodeBlock currentBlock;
 | |
| 		Stack blockStack = new Stack ();
 | |
| 		int varId;
 | |
| 		Label returnLabel;
 | |
| 		ArrayList nestedIfs = new ArrayList();
 | |
| 		int currentIfSerie = -1;
 | |
| 		CodeClass codeClass;
 | |
| 
 | |
| 		public CodeBuilder (CodeClass codeClass)
 | |
| 		{
 | |
| 			this.codeClass = codeClass;
 | |
| 			mainBlock = new CodeBlock ();
 | |
| 			currentBlock = mainBlock;
 | |
| 		}
 | |
| 		
 | |
| 		CodeBuilder (CodeBlock block)
 | |
| 		{
 | |
| 			currentBlock = block;
 | |
| 		}
 | |
| 		
 | |
| 		public CodeBlock CurrentBlock
 | |
| 		{
 | |
| 			get {
 | |
| 				return currentBlock;
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		public CodeClass OwnerClass
 | |
| 		{
 | |
| 			get { return codeClass; }
 | |
| 		}
 | |
| 		
 | |
| 		public void Generate (ILGenerator gen)
 | |
| 		{
 | |
| //			try {
 | |
| 				mainBlock.Generate (gen);
 | |
| /*
 | |
| 			}
 | |
| 			catch (Exception ex) {
 | |
| 				string m = ex.Message + "\nCode block:\n";
 | |
| 				m += "-----------------------\n";
 | |
| 				m += PrintCode ();
 | |
| 				m += "-----------------------\n";
 | |
| 				throw new Exception (m, ex);
 | |
| 			}
 | |
| */
 | |
| 		}
 | |
| 		
 | |
| 		public string PrintCode ()
 | |
| 		{
 | |
| 			StringWriter sw = new StringWriter ();
 | |
| 			CodeWriter cw = new CodeWriter (sw);
 | |
| 			PrintCode (cw);
 | |
| 			return sw.ToString ();
 | |
| 		}
 | |
| 		
 | |
| 		public void PrintCode (CodeWriter cp)
 | |
| 		{
 | |
| 			mainBlock.PrintCode (cp);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeVariableReference DeclareVariable (Type type)
 | |
| 		{
 | |
| 			return DeclareVariable (type, null);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeVariableReference DeclareVariable (Type type, object ob)
 | |
| 		{
 | |
| 			return DeclareVariable (type, Exp.Literal(ob));
 | |
| 		}
 | |
| 		
 | |
| 		public CodeVariableReference DeclareVariable (CodeExpression initValue)
 | |
| 		{
 | |
| 			return DeclareVariable (initValue.GetResultType(), initValue);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeVariableReference DeclareVariable (Type type, CodeExpression initValue)
 | |
| 		{
 | |
| 			string name = "v" + (varId++);
 | |
| 			CodeVariableDeclaration var = new CodeVariableDeclaration (type, name);
 | |
| 			currentBlock.Add (var);
 | |
| 			if (!object.ReferenceEquals (initValue, null)) 
 | |
| 				Assign (var.Variable, initValue);
 | |
| 			return var.Variable;
 | |
| 		}
 | |
| 		
 | |
| 		public void Assign (CodeValueReference var, CodeExpression val)
 | |
| 		{
 | |
| 			currentBlock.Add (new CodeAssignment (var, val)); 
 | |
| 		}
 | |
| 		
 | |
| 		public void If (CodeExpression condition)
 | |
| 		{
 | |
| 			currentBlock.Add (new CodeIf (condition));
 | |
| 			PushNewBlock ();
 | |
| 			nestedIfs.Add (0);
 | |
| 		}
 | |
| 		
 | |
| 		public void ElseIf (CodeExpression condition)
 | |
| 		{
 | |
| 			if (nestedIfs.Count == 0)
 | |
| 				throw new InvalidOperationException ("'Else' not allowed here");
 | |
| 
 | |
| 			Else ();
 | |
| 			currentBlock.Add (new CodeIf (condition));
 | |
| 			PushNewBlock ();
 | |
| 			nestedIfs [nestedIfs.Count-1] = 1 + (int)nestedIfs [nestedIfs.Count-1];
 | |
| 		}
 | |
| 		
 | |
| 		public void Else ()
 | |
| 		{
 | |
| 			CodeBlock block = PopBlock ();
 | |
| 			CodeIf cif = currentBlock.GetLastItem () as CodeIf;
 | |
| 			
 | |
| 			if (cif == null || cif.TrueBlock != null)
 | |
| 				throw new InvalidOperationException ("'Else' not allowed here");
 | |
| 				
 | |
| 			cif.TrueBlock = block;
 | |
| 			PushNewBlock ();
 | |
| 		}
 | |
| 		
 | |
| 		public void EndIf ()
 | |
| 		{
 | |
| 			CodeBlock block = PopBlock ();
 | |
| 			CodeIf cif = currentBlock.GetLastItem () as CodeIf;
 | |
| 			
 | |
| 			if (cif == null || cif.FalseBlock != null || nestedIfs.Count == 0)
 | |
| 				throw new InvalidOperationException ("'EndIf' not allowed here");
 | |
| 			
 | |
| 			if (cif.TrueBlock == null)
 | |
| 				cif.TrueBlock = block;
 | |
| 			else
 | |
| 				cif.FalseBlock = block;
 | |
| 				
 | |
| 			int num = (int) nestedIfs [nestedIfs.Count-1];
 | |
| 			if (num > 0) {
 | |
| 				nestedIfs [nestedIfs.Count-1] = --num;
 | |
| 				EndIf ();
 | |
| 			}
 | |
| 			else {
 | |
| 				nestedIfs.RemoveAt (nestedIfs.Count - 1);
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		public void Select ()
 | |
| 		{
 | |
| 			currentBlock.Add (new CodeSelect ());
 | |
| 			PushNewBlock ();
 | |
| 		}
 | |
| 		
 | |
| 		public void Case (CodeExpression condition)
 | |
| 		{
 | |
| 			PopBlock ();
 | |
| 			CodeSelect select = currentBlock.GetLastItem () as CodeSelect;
 | |
| 			if (select == null)
 | |
| 				throw new InvalidOperationException ("'Case' not allowed here");
 | |
| 
 | |
| 			PushNewBlock ();
 | |
| 			select.AddCase (condition, currentBlock);
 | |
| 		}
 | |
| 		
 | |
| 		public void EndSelect ()
 | |
| 		{
 | |
| 			PopBlock ();
 | |
| 			CodeSelect select = currentBlock.GetLastItem () as CodeSelect;
 | |
| 			if (select == null)
 | |
| 				throw new InvalidOperationException ("'EndSelect' not allowed here");
 | |
| 		}
 | |
| 		
 | |
| 		
 | |
| 		public void While (CodeExpression condition)
 | |
| 		{
 | |
| 			currentBlock.Add (new CodeWhile (condition));
 | |
| 			PushNewBlock ();
 | |
| 		}
 | |
| 		
 | |
| 		public void EndWhile ()
 | |
| 		{
 | |
| 			CodeBlock block = PopBlock ();
 | |
| 			CodeWhile cif = currentBlock.GetLastItem () as CodeWhile;
 | |
| 			
 | |
| 			if (cif == null || cif.WhileBlock != null)
 | |
| 				throw new InvalidOperationException ("'EndWhile' not allowed here");
 | |
| 			
 | |
| 			cif.WhileBlock = block;
 | |
| 		}
 | |
| 		
 | |
| 		public void Foreach (Type type, out CodeExpression item, CodeExpression array)
 | |
| 		{
 | |
| 			CodeForeach cfe = new CodeForeach (array, type);
 | |
| 			item = cfe.ItemExpression;
 | |
| 			currentBlock.Add (cfe);
 | |
| 			PushNewBlock ();
 | |
| 		}
 | |
| 		
 | |
| 		public void EndForeach ()
 | |
| 		{
 | |
| 			CodeBlock block = PopBlock ();
 | |
| 			CodeForeach cif = currentBlock.GetLastItem () as CodeForeach;
 | |
| 			
 | |
| 			if (cif == null || cif.ForBlock != null)
 | |
| 				throw new InvalidOperationException ("'EndForeach' not allowed here");
 | |
| 			
 | |
| 			cif.ForBlock = block;
 | |
| 		}
 | |
| 		
 | |
| 		public void For (CodeExpression initExp, CodeExpression conditionExp, CodeExpression nextExp)
 | |
| 		{
 | |
| 			currentBlock.Add (new CodeFor (initExp, conditionExp, nextExp));
 | |
| 			PushNewBlock ();
 | |
| 		}
 | |
| 		
 | |
| 		public void EndFor ()
 | |
| 		{
 | |
| 			CodeBlock block = PopBlock ();
 | |
| 			CodeFor cif = currentBlock.GetLastItem () as CodeFor;
 | |
| 			
 | |
| 			if (cif == null || cif.ForBlock != null)
 | |
| 				throw new InvalidOperationException ("'EndFor' not allowed here");
 | |
| 			
 | |
| 			cif.ForBlock = block;
 | |
| 		}
 | |
| 		
 | |
| 		
 | |
| 		public void Call (CodeExpression target, string name, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) target == null)
 | |
| 				throw new ArgumentNullException ("target");
 | |
| 			if (name == null)
 | |
| 				throw new ArgumentNullException ("name");
 | |
| 			currentBlock.Add (new CodeMethodCall (target, name, parameters));
 | |
| 		}
 | |
| 		
 | |
| 		public void Call (CodeExpression target, MethodBase method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) target == null)
 | |
| 				throw new ArgumentNullException ("target");
 | |
| 			if (method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			currentBlock.Add (new CodeMethodCall (target, method, parameters));
 | |
| 		}
 | |
| 		
 | |
| 		public void Call (CodeExpression target, CodeMethod method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) target == null)
 | |
| 				throw new ArgumentNullException ("target");
 | |
| 			if (method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			currentBlock.Add (new CodeMethodCall (target, method, parameters));
 | |
| 		}
 | |
| 		
 | |
| 		public void Call (Type type, string name, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if (type == null)
 | |
| 				throw new ArgumentNullException ("type");
 | |
| 			if (name == null)
 | |
| 				throw new ArgumentNullException ("name");
 | |
| 			currentBlock.Add (new CodeMethodCall (type, name, parameters));
 | |
| 		}
 | |
| 		
 | |
| 		public void Call (MethodInfo method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if (method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			currentBlock.Add (new CodeMethodCall (method, parameters));
 | |
| 		}
 | |
| 		
 | |
| 		public void Call (CodeMethod method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			currentBlock.Add (new CodeMethodCall (method, parameters));
 | |
| 		}
 | |
| 		
 | |
| 		public CodeExpression CallFunc (CodeExpression target, string name, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) target == null)
 | |
| 				throw new ArgumentNullException ("target");
 | |
| 			if (name == null)
 | |
| 				throw new ArgumentNullException ("name");
 | |
| 			return new CodeMethodCall (target, name, parameters);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeExpression CallFunc (CodeExpression target, MethodInfo method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) target == null)
 | |
| 				throw new ArgumentNullException ("target");
 | |
| 			if (method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			return new CodeMethodCall (target, method, parameters);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeExpression CallFunc (CodeExpression target, CodeMethod method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) target == null)
 | |
| 				throw new ArgumentNullException ("target");
 | |
| 			if (method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			return new CodeMethodCall (target, method, parameters);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeExpression CallFunc (Type type, string name, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if (type == null)
 | |
| 				throw new ArgumentNullException ("type");
 | |
| 			if (name == null)
 | |
| 				throw new ArgumentNullException ("name");
 | |
| 			return new CodeMethodCall (type, name, parameters);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeExpression CallFunc (MethodInfo method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if (method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			return new CodeMethodCall (method, parameters);
 | |
| 		}
 | |
| 		
 | |
| 		public CodeExpression CallFunc (CodeMethod method, params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			if ((object) method == null)
 | |
| 				throw new ArgumentNullException ("method");
 | |
| 			return new CodeMethodCall (method, parameters);
 | |
| 		}
 | |
| 		
 | |
| 		public void Inc (CodeValueReference val)
 | |
| 		{
 | |
| 			Assign (val, new CodeIncrement (val));
 | |
| 		}
 | |
| 		
 | |
| 		public void Dec (CodeValueReference val)
 | |
| 		{
 | |
| 			Assign (val, new CodeDecrement (val));
 | |
| 		}
 | |
| 		
 | |
| 		public CodeExpression When (CodeExpression condition, CodeExpression trueResult, CodeExpression falseResult)
 | |
| 		{
 | |
| 			return new CodeWhen (condition, trueResult, falseResult);
 | |
| 		}
 | |
| 		
 | |
| 		public void ConsoleWriteLine (params CodeExpression[] parameters)
 | |
| 		{
 | |
| 			Call (typeof(Console), "WriteLine", parameters);
 | |
| 		}
 | |
| 		
 | |
| 		public void ConsoleWriteLine (params object[] parameters)
 | |
| 		{
 | |
| 			CodeExpression[] exps = new CodeExpression [parameters.Length];
 | |
| 			for (int n=0; n<exps.Length; n++)
 | |
| 				exps[n] = Exp.Literal (parameters[n]);
 | |
| 				
 | |
| 			ConsoleWriteLine (exps);
 | |
| 		}
 | |
| 		
 | |
| 		public void Return (CodeExpression exp)
 | |
| 		{
 | |
| 			currentBlock.Add (new CodeReturn (this, exp));
 | |
| 		}
 | |
| 		
 | |
| 		public void Return ()
 | |
| 		{
 | |
| 			currentBlock.Add (new CodeReturn (this));
 | |
| 		}
 | |
| 		
 | |
| 		public static CodeBuilder operator+(CodeBuilder cb, CodeItem e)
 | |
| 		{
 | |
| 			cb.currentBlock.Add (e);
 | |
| 			return cb;
 | |
| 		}
 | |
| 		
 | |
| 		internal Label ReturnLabel
 | |
| 		{
 | |
| 			get { return returnLabel; }
 | |
| 			set { returnLabel = value; }
 | |
| 		}
 | |
| 		
 | |
| 		void PushNewBlock ()
 | |
| 		{
 | |
| 			blockStack.Push (currentBlock);
 | |
| 			currentBlock = new CodeBlock ();
 | |
| 		}
 | |
| 		
 | |
| 		CodeBlock PopBlock ()
 | |
| 		{
 | |
| 			CodeBlock block = currentBlock;
 | |
| 			currentBlock = (CodeBlock) blockStack.Pop ();
 | |
| 			return block;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| #endif
 |