/* * [The "BSD licence"] * Copyright (c) 2005-2008 Terence Parr * All rights reserved. * * Conversion to C#: * Copyright (c) 2008-2009 Sam Harwell, Pixel Mine, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ namespace Antlr.Runtime.Tree { using InvalidOperationException = System.InvalidOperationException; public class TreePatternParser { protected TreePatternLexer tokenizer; protected int ttype; protected TreeWizard wizard; protected ITreeAdaptor adaptor; public TreePatternParser(TreePatternLexer tokenizer, TreeWizard wizard, ITreeAdaptor adaptor) { this.tokenizer = tokenizer; this.wizard = wizard; this.adaptor = adaptor; ttype = tokenizer.NextToken(); // kickstart } public virtual object Pattern() { if (ttype == TreePatternLexer.BEGIN) { return ParseTree(); } else if (ttype == TreePatternLexer.ID) { object node = ParseNode(); if (ttype == TreePatternLexer.EOF) { return node; } return null; // extra junk on end } return null; } public virtual object ParseTree() { if (ttype != TreePatternLexer.BEGIN) throw new InvalidOperationException("No beginning."); ttype = tokenizer.NextToken(); object root = ParseNode(); if (root == null) { return null; } while (ttype == TreePatternLexer.BEGIN || ttype == TreePatternLexer.ID || ttype == TreePatternLexer.PERCENT || ttype == TreePatternLexer.DOT) { if (ttype == TreePatternLexer.BEGIN) { object subtree = ParseTree(); adaptor.AddChild(root, subtree); } else { object child = ParseNode(); if (child == null) { return null; } adaptor.AddChild(root, child); } } if (ttype != TreePatternLexer.END) throw new InvalidOperationException("No end."); ttype = tokenizer.NextToken(); return root; } public virtual object ParseNode() { // "%label:" prefix string label = null; if (ttype == TreePatternLexer.PERCENT) { ttype = tokenizer.NextToken(); if (ttype != TreePatternLexer.ID) { return null; } label = tokenizer.sval.ToString(); ttype = tokenizer.NextToken(); if (ttype != TreePatternLexer.COLON) { return null; } ttype = tokenizer.NextToken(); // move to ID following colon } // Wildcard? if (ttype == TreePatternLexer.DOT) { ttype = tokenizer.NextToken(); IToken wildcardPayload = new CommonToken(0, "."); TreeWizard.TreePattern node = new TreeWizard.WildcardTreePattern(wildcardPayload); if (label != null) { node.label = label; } return node; } // "ID" or "ID[arg]" if (ttype != TreePatternLexer.ID) { return null; } string tokenName = tokenizer.sval.ToString(); ttype = tokenizer.NextToken(); if (tokenName.Equals("nil")) { return adaptor.Nil(); } string text = tokenName; // check for arg string arg = null; if (ttype == TreePatternLexer.ARG) { arg = tokenizer.sval.ToString(); text = arg; ttype = tokenizer.NextToken(); } // create node int treeNodeType = wizard.GetTokenType(tokenName); if (treeNodeType == TokenTypes.Invalid) { return null; } object node2; node2 = adaptor.Create(treeNodeType, text); if (label != null && node2.GetType() == typeof(TreeWizard.TreePattern)) { ((TreeWizard.TreePattern)node2).label = label; } if (arg != null && node2.GetType() == typeof(TreeWizard.TreePattern)) { ((TreeWizard.TreePattern)node2).hasTextArg = true; } return node2; } } }