You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

157 lines
5.9 KiB

/*
* [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;
}
}
}