using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Text; namespace Indy.IL2CPU.Assembler { public class Assembler: IDisposable { public enum OutputTypeEnum { DLL, Console, GUI } public const string EntryPointLabelName = "___ENTRYPOINT___"; private static Assembler mCurrentInstance; private List mInstructions = new List(); private List mDataMembers = new List(); private OutputTypeEnum mOutputType = OutputTypeEnum.DLL; private StreamWriter mOutputWriter; private List mIncludes = new List(); private List mImportMembers = new List(); private uint mDataMemberCounter = 0; public string GetIdentifier(string aPrefix) { return aPrefix + mDataMemberCounter++.ToString("X8").ToUpper(); } public Assembler(StreamWriter aOutputWriter) { if (mCurrentInstance != null) { throw new Exception("There already is an Assembler Instance!"); } mCurrentInstance = this; mOutputWriter = aOutputWriter; } public static Assembler Current { get { if (mCurrentInstance == null) { throw new Exception("No current Assembler Instance!"); } return mCurrentInstance; } } public OutputTypeEnum OutputType { get { return mOutputType; } set { mOutputType = value; } } public List Includes { get { return mIncludes; } } public List DataMembers { get { return mDataMembers; } } public List ImportMembers { get { return mImportMembers; } } #region IDisposable Members public void Dispose() { // MtW: I know, IDisposable usage for this isn't really nice, but for now this should be fine. mCurrentInstance = null; mInstructions.Clear(); mDataMembers.Clear(); } #endregion public void Add(Instruction aInstruction) { mInstructions.Add(aInstruction); } public void Flush() { // write .asm header // debug: we're generating .dll so we can test the outpu switch (mOutputType) { case OutputTypeEnum.Console: mOutputWriter.WriteLine("format PE console"); break; case OutputTypeEnum.GUI: mOutputWriter.WriteLine("format PE GUI 4.0"); break; default: mOutputWriter.WriteLine("format PE dll"); break; } if (mOutputType != OutputTypeEnum.DLL) { mOutputWriter.WriteLine("entry " + EntryPointLabelName); } mOutputWriter.WriteLine(); foreach (string xInclude in mIncludes) { mOutputWriter.WriteLine("include '{0}'", xInclude); } mOutputWriter.WriteLine(); mOutputWriter.WriteLine("section '.data' data readable writeable"); mOutputWriter.WriteLine(); foreach (DataMember xMember in mDataMembers) { mOutputWriter.WriteLine("\t" + xMember); } mOutputWriter.WriteLine(); mOutputWriter.WriteLine("section '.code' code readable executable"); mOutputWriter.WriteLine(); mOutputWriter.WriteLine(" " + EntryPointLabelName + ":"); foreach (Instruction x in mInstructions) { string prefix = "\t"; if (x is Label) { mOutputWriter.WriteLine(); prefix = " "; } mOutputWriter.WriteLine(prefix + x); } mOutputWriter.WriteLine(); mOutputWriter.WriteLine("section '.idata' import data readable writeable"); mOutputWriter.WriteLine(); foreach (ImportMember xImportMember in mImportMembers) { mOutputWriter.WriteLine("\t" + xImportMember); } } public string GetLabelName(string aType, string aReturnType, string aMethodName, params string[] aParamTypes) { StringBuilder xSB = new StringBuilder(); xSB.Append(aReturnType.Replace('.', '_').Replace('+', '_')); xSB.Append("___"); xSB.Append(aType.Replace('.', '_').Replace('+', '_')); xSB.Append("_"); xSB.Append(aMethodName.Replace('.', '_')); xSB.Append("__"); foreach(string s in aParamTypes) { xSB.Append("_"); xSB.Append(s.Replace('.', '_').Replace('+', '_')); xSB.Append("_"); } xSB.Append("__"); return xSB.ToString(); } } }