using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Cosmos.Compiler.XSharp { public class TokenPatterns { public delegate string CodeFunc(Token[] aTokens); protected Dictionary mList = new Dictionary(); public TokenPatterns() { Add(new TokenType[] { TokenType.LiteralAsm }, "new LiteralAssemblerCode(\"{0}\");" ); Add(new TokenType[] { TokenType.Comment }, "new Comment(\"{0}\");" ); Add("ABC:" , "new Label(\"{0}\");" ); Add("REG = 123", "new Mov{{ DestinationReg = RegistersEnum.{0}, SourceValue = {2} }};" ); Add("REG = REG", "new Mov{{ DestinationReg = RegistersEnum.{0}, SourceReg = RegistersEnum.{2} }};" ); Add("REG = [REG + 0]", "new Mov {{" + " DestinationReg = RegistersEnum.{0}" + ", SourceReg = RegistersEnum.{3}, SourceIsIndirect = true, SourceDisplacement = {5}" + "}};" ); Add("ABC = REG", "new Mov {{" + " DestinationRef = Cosmos.Assembler.ElementReference.New(RegistersEnum.{0})" + " , DestinationIsIndirect = true" + " , SourceValue = value.Value.GetValueOrDefault()" + " , SourceRef = value.Reference" + " , SourceReg = value.Register" + " , SourceIsIndirect = value.IsIndirect" + " }};" ); // TODO: Allow asm to optimize these to Inc/Dec Add("REG + 1", delegate(Token[] aTokens) { if (aTokens[2].Value == "1") { return "new Inc {{ DestinationReg = RegistersEnum.{0} }};"; } else { return "new Add {{ DestinationReg = RegistersEnum.{0}, SourceValue = {2} }};"; } }); Add("REG - 1", delegate(Token[] aTokens) { if (aTokens[2].Value == "1") { return "new Dec {{ DestinationReg = RegistersEnum.{0} }};"; } else { return "new Sub {{ DestinationReg = RegistersEnum.{0}, SourceValue = {2} }};"; } }); Add(new TokenType[] { TokenType.OpCode }, delegate(Token[] aTokens) { string xOp = aTokens[0].Value.ToUpper(); if (xOp == "POPALL") { return "new Popad();"; } else if (xOp == "PUSHALL") { return "new Pushad();"; } else { throw new Exception("Unrecognized op: " + aTokens[0].Value); } }); } public string GetCode(List aTokens) { var xPattern = aTokens.Select(c => c.Type).ToArray(); CodeFunc xAction; if (!mList.TryGetValue(new TokenPattern(xPattern), out xAction)) { throw new Exception("Token pattern not found."); } string xResult = xAction(aTokens.ToArray()); return string.Format(xResult, aTokens.Select(c => c.Value).ToArray()); } public void Add(string aPattern, CodeFunc aCode) { mList.Add(new TokenPattern(aPattern), aCode); } public void Add(TokenType[] aPattern, CodeFunc aCode) { mList.Add(new TokenPattern(aPattern), aCode); } public void Add(string aPattern, string aCode) { Add(aPattern, delegate(Token[] aTokens) { return aCode; }); } public void Add(TokenType[] aPattern, string aCode) { Add(aPattern, delegate(Token[] aTokens) { return aCode; }); } } }