/* Copyright (c) 2001-2007, The HSQL Development Group
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * Redistributions of source code must retain the above copyright notice, this
 * list of conditions and the following disclaimer.
 *
 * 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.
 *
 * Neither the name of the HSQL Development Group nor the names of its
 * contributors may be used to endorse or promote products derived from this
 * software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "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 HSQL DEVELOPMENT GROUP, HSQLDB.ORG,
 * OR CONTRIBUTORS 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.
 */


package org.hsqldb.util.preprocessor;

/* $Id: Parser.java 5793 2018-01-06 13:12:38Z fredt $ */

Simple preprocessor directive parser.

Author:Campbell Burnet (campbell-burnet@users dot sourceforge.net)
Version:1.8.1
Since:1.8.1
/** * Simple preprocessor directive parser. <p> * * @author Campbell Burnet (campbell-burnet@users dot sourceforge.net) * @version 1.8.1 * @since 1.8.1 */
class Parser { Defines defines; Tokenizer tokenizer; Parser(Defines defines, Tokenizer tokenizer) { this.defines = defines; this.tokenizer = tokenizer; } boolean parseExpression() throws PreprocessorException { boolean result = parseTerm(); while (true) { switch(this.tokenizer.getTokenType()) { case Token.OR : { this.tokenizer.next(); result = result | parseTerm(); break; } case Token.XOR : { this.tokenizer.next(); result = result ^ parseTerm(); break; } default : { return result; } } } } boolean parseTerm() throws PreprocessorException { boolean result = parseFactor(); while (this.tokenizer.isToken(Token.AND)) { this.tokenizer.next(); result = result & parseFactor(); } return result; } boolean parseFactor() throws PreprocessorException { boolean result; switch(this.tokenizer.getTokenType()) { case Token.IDENT : { String ident = this.tokenizer.getIdent(); int type = this.tokenizer.next(); if ((type == Token.EOI) || (type == Token.RPAREN) || Token.isLogicalOperator(type)) { result = this.defines.isDefined(ident); } else if (Token.isComparisonOperator(type)) { result = parseComparison(ident, type); } else { throw new PreprocessorException("Logical or comparison " + "operator token required at position " + this.tokenizer.getStartIndex() + " in [" + this.tokenizer.getSource() + "]"); // NOI18N } break; } case Token.NOT :{ this.tokenizer.next(); result = !parseFactor(); break; } case Token.LPAREN : { this.tokenizer.next(); result = parseExpression(); if (!this.tokenizer.isToken(Token.RPAREN)) { throw new PreprocessorException("RPAREN token required at " + "position " + this.tokenizer.getStartIndex() + " in [" + this.tokenizer.getSource() + "]"); // NOI18N } this.tokenizer.next(); break; } default : { throw new PreprocessorException("IDENT, NOT or LPAREN " + "token required at position " + this.tokenizer.getStartIndex() + " in [" + this.tokenizer.getSource() + "]"); // NOI18N } } return result; } boolean parseComparison(String ident, int opType) throws PreprocessorException { // checkIsComparisonOperator(opType); boolean result; Object lhs = this.defines.getDefintion(ident); int pos = this.tokenizer.getStartIndex(); Object rhs = parseValue(); if (lhs == null) { throw new PreprocessorException("IDENT " + ident + " is not defined at position" + pos + "in [" + this.tokenizer.getSource() + "]"); // NOI18N } switch(opType) { case Token.EQ :{ result = (compare(lhs, rhs) == 0); break; } case Token.LT : { result = (compare(lhs, rhs) < 0); break; } case Token.LTE : { result = (compare(lhs, rhs) <= 0); break; } case Token.GT : { result = (compare(lhs, rhs) > 0); break; } case Token.GTE : { result = (compare(lhs, rhs) >= 0); break; } default : { // Stupid compiler trick. // Can't actually happen because this case will cause an // exception to be thrown in method parseFactor (or in // method checkIsComparisonOperator when uncommented) throw new PreprocessorException("Internal error"); // NOI18N } } this.tokenizer.next(); return result; } // void checkIsComparisonOperator(int opType) throws PreprocessorException { // if (!Token.isComparisonOperator(opType)) { // throw new PreprocessorException("Comparison " // + "operator token required at position " // + tokenizer.getBeginIndex() // + " in [" // + tokenizer.getSource() // + "]"); // NOI18N // } // } static int compare(Object o1, Object o2) { // nulls are basically 'illegal' so yes: // we want to throw NPE here if o1 or o2 is null if (o1 instanceof Comparable) { return (o1.getClass().isAssignableFrom(o2.getClass())) ? ((Comparable)o1).compareTo(o2) : String.valueOf(o1).compareTo(String.valueOf(o2)); } else { return o1.toString().compareTo(o2.toString()); } } Object parseValue() throws PreprocessorException { Object value; switch(this.tokenizer.next()) { case Token.IDENT : { String ident = this.tokenizer.getIdent(); value = this.defines.getDefintion(ident); if (value == null) { throw new PreprocessorException("IDENT " + ident + " is not defined at position" + this.tokenizer.getStartIndex() + "in [" + this.tokenizer.getSource() + "]"); // NOI18N } break; } case Token.STRING : { value = this.tokenizer.getString(); break; } case Token.NUMBER : { value = this.tokenizer.getNumber(); break; } default :{ throw new PreprocessorException("IDENT, STRING" + "or NUMBER token required at position " + this.tokenizer.getStartIndex() + " in: [" + this.tokenizer.getSource() + "]"); // NOI18N } } return value; } }