X-Git-Url: http://git.phpeclipse.com diff --git a/net.sourceforge.phpeclipse/src/test/PHPParser.jj b/net.sourceforge.phpeclipse/src/test/PHPParser.jj index 0f04260..b61df9a 100644 --- a/net.sourceforge.phpeclipse/src/test/PHPParser.jj +++ b/net.sourceforge.phpeclipse/src/test/PHPParser.jj @@ -1,8 +1,9 @@ + options { LOOKAHEAD = 1; CHOICE_AMBIGUITY_CHECK = 2; OTHER_AMBIGUITY_CHECK = 1; - STATIC = true; + STATIC = false; DEBUG_PARSER = false; DEBUG_LOOKAHEAD = false; DEBUG_TOKEN_MANAGER = false; @@ -17,6 +18,7 @@ options { BUILD_TOKEN_MANAGER = true; SANITY_CHECK = true; FORCE_LA_CHECK = false; + COMMON_TOKEN_ACTION = true; } PARSER_BEGIN(PHPParser) @@ -40,6 +42,7 @@ import net.sourceforge.phpdt.internal.compiler.ast.*; import net.sourceforge.phpdt.internal.compiler.parser.OutlineableWithChildren; import net.sourceforge.phpdt.internal.compiler.parser.Outlineable; import net.sourceforge.phpdt.internal.compiler.parser.PHPOutlineInfo; +import net.sourceforge.phpdt.internal.corext.Assert; /** * A new php parser. @@ -47,12 +50,12 @@ import net.sourceforge.phpdt.internal.compiler.parser.PHPOutlineInfo; * given with JavaCC. You can get JavaCC at http://www.webgain.com * You can test the parser with the PHPParserTestCase2.java * @author Matthieu Casanova - * @version $Reference: 1.0$ */ public final class PHPParser extends PHPParserSuperclass { - /** The file that is parsed. */ - private static IFile fileToParse; +//todo : fix the variables names bug +//todo : handle tilde operator + /** The current segment. */ private static OutlineableWithChildren currentSegment; @@ -70,7 +73,7 @@ public final class PHPParser extends PHPParserSuperclass { private static int errorEnd = -1; private static PHPDocument phpDocument; - private static final char[] SYNTAX_ERROR_CHAR = {'s','y','n','t','a','x',' ','e','r','r','o','r'}; + private static final String SYNTAX_ERROR_CHAR = "syntax error"; /** * The point where html starts. * It will be used by the token manager to create HTMLCode objects @@ -84,8 +87,10 @@ public final class PHPParser extends PHPParserSuperclass { /** The cursor in expression stack. */ private static int nodePtr; + public static final boolean PARSER_DEBUG = false; + public final void setFileToParse(final IFile fileToParse) { - this.fileToParse = fileToParse; + PHPParser.fileToParse = fileToParse; } public PHPParser() { @@ -93,7 +98,50 @@ public final class PHPParser extends PHPParserSuperclass { public PHPParser(final IFile fileToParse) { this(new StringReader("")); - this.fileToParse = fileToParse; + PHPParser.fileToParse = fileToParse; + } + + public final void phpParserTester(final String strEval) throws ParseException { + final StringReader stream = new StringReader(strEval); + if (jj_input_stream == null) { + jj_input_stream = new SimpleCharStream(stream, 1, 1); + token_source = new PHPParserTokenManager(jj_input_stream); + } + ReInit(new StringReader(strEval)); + init(); + phpDocument = new PHPDocument(null,"_root".toCharArray()); + currentSegment = phpDocument; + outlineInfo = new PHPOutlineInfo(null, currentSegment); + token_source.SwitchTo(PHPParserTokenManager.PHPPARSING); + phpTest(); + } + + public final void htmlParserTester(final File fileName) throws FileNotFoundException, ParseException { + final Reader stream = new FileReader(fileName); + if (jj_input_stream == null) { + jj_input_stream = new SimpleCharStream(stream, 1, 1); + token_source = new PHPParserTokenManager(jj_input_stream); + } + ReInit(stream); + init(); + phpDocument = new PHPDocument(null,"_root".toCharArray()); + currentSegment = phpDocument; + outlineInfo = new PHPOutlineInfo(null, currentSegment); + phpFile(); + } + + public final void htmlParserTester(final String strEval) throws ParseException { + final StringReader stream = new StringReader(strEval); + if (jj_input_stream == null) { + jj_input_stream = new SimpleCharStream(stream, 1, 1); + token_source = new PHPParserTokenManager(jj_input_stream); + } + ReInit(stream); + init(); + phpDocument = new PHPDocument(null,"_root".toCharArray()); + currentSegment = phpDocument; + outlineInfo = new PHPOutlineInfo(null, currentSegment); + phpFile(); } /** @@ -129,6 +177,7 @@ public final class PHPParser extends PHPParserSuperclass { final StringReader stream = new StringReader(s); if (jj_input_stream == null) { jj_input_stream = new SimpleCharStream(stream, 1, 1); + token_source = new PHPParserTokenManager(jj_input_stream); } ReInit(stream); init(); @@ -146,6 +195,19 @@ public final class PHPParser extends PHPParserSuperclass { } /** + * This function will throw the exception if we are in debug mode + * and process it if we are in production mode. + * this should be fast since the PARSER_DEBUG is static final so the difference will be at compile time + * @param e the exception + * @throws ParseException the thrown exception + */ + private static void processParseExceptionDebug(final ParseException e) throws ParseException { + if (PARSER_DEBUG) { + throw e; + } + processParseException(e); + } + /** * This method will process the parse exception. * If the error message is null, the parse exception wasn't catched and a trace is written in the log * @param e the ParseException @@ -154,16 +216,16 @@ public final class PHPParser extends PHPParserSuperclass { if (errorMessage == null) { PHPeclipsePlugin.log(e); errorMessage = "this exception wasn't handled by the parser please tell us how to reproduce it"; - errorStart = SimpleCharStream.getPosition(); - errorEnd = errorStart + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; } setMarker(e); errorMessage = null; - if (PHPeclipsePlugin.DEBUG) PHPeclipsePlugin.log(e); + // if (PHPeclipsePlugin.DEBUG) PHPeclipsePlugin.log(e); } /** - * Create marker for the parse error + * Create marker for the parse error. * @param e the ParseException */ private static void setMarker(final ParseException e) { @@ -171,17 +233,17 @@ public final class PHPParser extends PHPParserSuperclass { if (errorStart == -1) { setMarker(fileToParse, errorMessage, - SimpleCharStream.tokenBegin, - SimpleCharStream.tokenBegin + e.currentToken.image.length(), + e.currentToken.sourceStart, + e.currentToken.sourceEnd, errorLevel, - "Line " + e.currentToken.beginLine); + "Line " + e.currentToken.beginLine+", "+e.currentToken.sourceStart+':'+e.currentToken.sourceEnd); } else { setMarker(fileToParse, errorMessage, errorStart, errorEnd, errorLevel, - "Line " + e.currentToken.beginLine); + "Line " + e.currentToken.beginLine+", "+errorStart+':'+errorEnd); errorStart = -1; errorEnd = -1; } @@ -195,12 +257,12 @@ public final class PHPParser extends PHPParserSuperclass { final int indx, final int brIndx) throws CoreException { String current; - StringBuffer lineNumberBuffer = new StringBuffer(10); + final StringBuffer lineNumberBuffer = new StringBuffer(10); char ch; current = output.substring(indx, brIndx); if (current.indexOf(PARSE_WARNING_STRING) != -1 || current.indexOf(PARSE_ERROR_STRING) != -1) { - int onLine = current.indexOf("on line <b>"); + final int onLine = current.indexOf("on line <b>"); if (onLine != -1) { lineNumberBuffer.delete(0, lineNumberBuffer.length()); for (int i = onLine; i < current.length(); i++) { @@ -210,9 +272,9 @@ public final class PHPParser extends PHPParserSuperclass { } } - int lineNumber = Integer.parseInt(lineNumberBuffer.toString()); + final int lineNumber = Integer.parseInt(lineNumberBuffer.toString()); - Hashtable attributes = new Hashtable(); + final Hashtable attributes = new Hashtable(); current = current.replaceAll("\n", ""); current = current.replaceAll("<b>", ""); @@ -231,10 +293,11 @@ public final class PHPParser extends PHPParserSuperclass { } } - public final void parse(final String s) throws CoreException { + public final void parse(final String s) { final StringReader stream = new StringReader(s); if (jj_input_stream == null) { jj_input_stream = new SimpleCharStream(stream, 1, 1); + token_source = new PHPParserTokenManager(jj_input_stream); } ReInit(stream); init(); @@ -270,51 +333,64 @@ public final class PHPParser extends PHPParserSuperclass { /** * Put a new html block in the stack. */ - public static final void createNewHTMLCode() { - final int currentPosition = SimpleCharStream.getPosition(); - if (currentPosition == htmlStart || currentPosition > SimpleCharStream.currentBuffer.length()) { + public final void createNewHTMLCode() { + final int currentPosition = token.sourceStart; + if (currentPosition == htmlStart || + currentPosition < htmlStart || + currentPosition > jj_input_stream.getCurrentBuffer().length()) { return; } - final char[] chars = SimpleCharStream.currentBuffer.substring(htmlStart,currentPosition+1).toCharArray(); - pushOnAstNodes(new HTMLCode(chars, htmlStart,currentPosition)); + final String html = jj_input_stream.getCurrentBuffer().substring(htmlStart, currentPosition); + pushOnAstNodes(new HTMLCode(html, htmlStart,currentPosition)); } - /** - * Create a new task. - */ - public static final void createNewTask() { - final int currentPosition = SimpleCharStream.getPosition(); - final String todo = SimpleCharStream.currentBuffer.substring(currentPosition+1, - SimpleCharStream.currentBuffer.indexOf("\n", - currentPosition)-1); - try { - setMarker(fileToParse, - "todo : " + todo, - SimpleCharStream.getBeginLine(), - TASK, - "Line "+SimpleCharStream.getBeginLine()); - } catch (CoreException e) { - PHPeclipsePlugin.log(e); + /** Create a new task. */ + public final void createNewTask(final int todoStart) { + final String todo = jj_input_stream.getCurrentBuffer().substring(todoStart, + jj_input_stream.getCurrentBuffer().indexOf("\n", + todoStart)-1); + if (!PARSER_DEBUG) { + try { + setMarker(fileToParse, + todo, + jj_input_stream.getBeginLine(), + TASK, + "Line "+jj_input_stream.getBeginLine()); + } catch (CoreException e) { + PHPeclipsePlugin.log(e); + } } } - private static final void parse() throws ParseException { + private final void parse() throws ParseException { phpFile(); } } PARSER_END(PHPParser) +TOKEN_MGR_DECLS: +{ + // CommonTokenAction: use the begins/ends fields added to the Jack + // CharStream class to set corresponding fields in each Token (which was + // also extended with new fields). By default Jack doesn't supply absolute + // offsets, just line/column offsets + void CommonTokenAction(Token t) { + t.sourceStart = input_stream.getBeginOffset(); + t.sourceEnd = input_stream.getBeginOffset(); + } // CommonTokenAction +} // TOKEN_MGR_DECLS + <DEFAULT> TOKEN : { - <PHPSTARTSHORT : "<?"> {PHPParser.createNewHTMLCode();} : PHPPARSING -| <PHPSTARTLONG : "<?php"> {PHPParser.createNewHTMLCode();} : PHPPARSING -| <PHPECHOSTART : "<?="> {PHPParser.createNewHTMLCode();} : PHPPARSING + <PHPSTARTSHORT : "<?"> : PHPPARSING +| <PHPSTARTLONG : "<?php"> : PHPPARSING +| <PHPECHOSTART : "<?="> : PHPPARSING } -<PHPPARSING> TOKEN : +<PHPPARSING, IN_SINGLE_LINE_COMMENT,IN_VARIABLE> TOKEN : { - <PHPEND :"?>"> {PHPParser.htmlStart = SimpleCharStream.getPosition();} : DEFAULT + <PHPEND :"?>"> : DEFAULT } /* Skip any character if we are not in php mode */ @@ -334,6 +410,14 @@ PARSER_END(PHPParser) | "\f" } +<IN_VARIABLE> SPECIAL_TOKEN : +{ + " " : PHPPARSING +| "\t" : PHPPARSING +| "\n" : PHPPARSING +| "\r" : PHPPARSING +| "\f" : PHPPARSING +} /* COMMENTS */ <PHPPARSING> SPECIAL_TOKEN : { @@ -346,14 +430,19 @@ PARSER_END(PHPParser) <IN_SINGLE_LINE_COMMENT> SPECIAL_TOKEN : { <SINGLE_LINE_COMMENT: "\n" | "\r" | "\r\n" > : PHPPARSING -| "?>" : DEFAULT +| < ~[] > } <IN_SINGLE_LINE_COMMENT,IN_FORMAL_COMMENT,IN_MULTI_LINE_COMMENT> SPECIAL_TOKEN : { - "todo" {PHPParser.createNewTask();} + "todo" } +void todo() : +{Token todoToken;} +{ + todoToken = "TODO" {createNewTask(todoToken.sourceStart);} +} <IN_FORMAL_COMMENT> SPECIAL_TOKEN : { "*/" : PHPPARSING @@ -396,9 +485,13 @@ MORE : | <GLOBAL : "global"> | <DEFINE : "define"> | <STATIC : "static"> -| <CLASSACCESS : "->"> -| <STATICCLASSACCESS : "::"> -| <ARRAYASSIGN : "=>"> +} + +<PHPPARSING,IN_VARIABLE> TOKEN : +{ + <CLASSACCESS : "->"> : PHPPARSING +| <STATICCLASSACCESS : "::"> : PHPPARSING +| <ARRAYASSIGN : "=>"> : PHPPARSING } /* RESERVED WORDS AND LITERALS */ @@ -445,36 +538,35 @@ MORE : } //Misc token -<PHPPARSING> TOKEN : +<PHPPARSING,IN_VARIABLE> TOKEN : { - <AT : "@"> -| <DOLLAR : "$"> -| <BANG : "!"> -| <TILDE : "~"> -| <HOOK : "?"> -| <COLON : ":"> + <AT : "@"> : PHPPARSING +| <BANG : "!"> : PHPPARSING +| <TILDE : "~"> : PHPPARSING +| <HOOK : "?"> : PHPPARSING +| <COLON : ":"> : PHPPARSING } /* OPERATORS */ -<PHPPARSING> TOKEN : -{ - <OR_OR : "||"> -| <AND_AND : "&&"> -| <PLUS_PLUS : "++"> -| <MINUS_MINUS : "--"> -| <PLUS : "+"> -| <MINUS : "-"> -| <STAR : "*"> -| <SLASH : "/"> -| <BIT_AND : "&"> -| <BIT_OR : "|"> -| <XOR : "^"> -| <REMAINDER : "%"> -| <LSHIFT : "<<"> -| <RSIGNEDSHIFT : ">>"> -| <RUNSIGNEDSHIFT : ">>>"> -| <_ORL : "OR"> -| <_ANDL : "AND"> +<PHPPARSING,IN_VARIABLE> TOKEN : +{ + <OR_OR : "||"> : PHPPARSING +| <AND_AND : "&&"> : PHPPARSING +| <PLUS_PLUS : "++"> : PHPPARSING +| <MINUS_MINUS : "--"> : PHPPARSING +| <PLUS : "+"> : PHPPARSING +| <MINUS : "-"> : PHPPARSING +| <STAR : "*"> : PHPPARSING +| <SLASH : "/"> : PHPPARSING +| <BIT_AND : "&"> : PHPPARSING +| <BIT_OR : "|"> : PHPPARSING +| <XOR : "^"> : PHPPARSING +| <REMAINDER : "%"> : PHPPARSING +| <LSHIFT : "<<"> : PHPPARSING +| <RSIGNEDSHIFT : ">>"> : PHPPARSING +| <RUNSIGNEDSHIFT : ">>>"> : PHPPARSING +| <_ORL : "OR"> : PHPPARSING +| <_ANDL : "AND"> : PHPPARSING } /* LITERALS */ @@ -501,16 +593,80 @@ MORE : | <#EXPONENT: ["e","E"] (["+","-"])? (["0"-"9"])+ > | - <STRING_LITERAL: (<STRING_1> | <STRING_2> | <STRING_3>)> -| <STRING_1: "\"" ( ~["\"","\\"] | "\\" ~[] )* "\""> + <STRING_LITERAL: (<STRING_2> | <STRING_3>)> +//| <STRING_1: "\"" ( ~["\"","\\"] | "\\" ~[] )* "\""> | <STRING_2: "'" ( ~["'","\\"] | "\\" ~[] )* "'"> | <STRING_3: "`" ( ~["`","\\"] | "\\" ~[] )* "`"> } -/* IDENTIFIERS */ +<IN_STRING,DOLLAR_IN_STRING> SKIP : +{ + <ESCAPED : ("\\" ~[])> : IN_STRING +} <PHPPARSING> TOKEN : { + <DOUBLEQUOTE : "\""> : IN_STRING +} + + +<IN_STRING> TOKEN : +{ + <DOLLARS : "$"> : DOLLAR_IN_STRING +} + +<IN_STRING,DOLLAR_IN_STRING> TOKEN : +{ + <DOUBLEQUOTE2 : "\""> : PHPPARSING +} + +<DOLLAR_IN_STRING> TOKEN : +{ + <LBRACE1 : "{"> : DOLLAR_IN_STRING_EXPR +} + +<IN_STRING> SPECIAL_TOKEN : +{ + <"{"> : SKIPSTRING +} + +<SKIPSTRING> SPECIAL_TOKEN : +{ + <"}"> : IN_STRING +} + +<SKIPSTRING> SKIP : +{ + <~[]> +} + +<DOLLAR_IN_STRING_EXPR> TOKEN : +{ + <RBRACE1 : "}"> : DOLLAR_IN_STRING +} + +<DOLLAR_IN_STRING_EXPR> TOKEN : +{ + <ID : (~["}"])*> +} + +<IN_STRING> SKIP : +{ + <~[]> +} + +<DOLLAR_IN_STRING_EXPR,IN_STRING> SKIP : +{ + <~[]> +} +/* IDENTIFIERS */ + + +<PHPPARSING,IN_VARIABLE> TOKEN : {<DOLLAR : "$"> : IN_VARIABLE} + + +<PHPPARSING, IN_VARIABLE, DOLLAR_IN_STRING> TOKEN : +{ <IDENTIFIER: (<LETTER>|<SPECIAL>) (<LETTER>|<DIGIT>|<SPECIAL>)* > | < #LETTER: @@ -526,57 +682,63 @@ MORE : > } +<DOLLAR_IN_STRING> SPECIAL_TOKEN : +{ + < ~[] > : IN_STRING +} /* SEPARATORS */ -<PHPPARSING> TOKEN : +<PHPPARSING,IN_VARIABLE> TOKEN : { - <LPAREN : "("> -| <RPAREN : ")"> -| <LBRACE : "{"> -| <RBRACE : "}"> -| <LBRACKET : "["> -| <RBRACKET : "]"> -| <SEMICOLON : ";"> -| <COMMA : ","> -| <DOT : "."> + <LPAREN : "("> : PHPPARSING +| <RPAREN : ")"> : PHPPARSING +| <LBRACE : "{"> : PHPPARSING +| <RBRACE : "}"> : PHPPARSING +| <LBRACKET : "["> : PHPPARSING +| <RBRACKET : "]"> : PHPPARSING +| <SEMICOLON : ";"> : PHPPARSING +| <COMMA : ","> : PHPPARSING +| <DOT : "."> : PHPPARSING } /* COMPARATOR */ -<PHPPARSING> TOKEN : +<PHPPARSING,IN_VARIABLE> TOKEN : { - <GT : ">"> -| <LT : "<"> -| <EQUAL_EQUAL : "=="> -| <LE : "<="> -| <GE : ">="> -| <NOT_EQUAL : "!="> -| <DIF : "<>"> -| <BANGDOUBLEEQUAL : "!=="> -| <TRIPLEEQUAL : "==="> + <GT : ">"> : PHPPARSING +| <LT : "<"> : PHPPARSING +| <EQUAL_EQUAL : "=="> : PHPPARSING +| <LE : "<="> : PHPPARSING +| <GE : ">="> : PHPPARSING +| <NOT_EQUAL : "!="> : PHPPARSING +| <DIF : "<>"> : PHPPARSING +| <BANGDOUBLEEQUAL : "!=="> : PHPPARSING +| <TRIPLEEQUAL : "==="> : PHPPARSING } /* ASSIGNATION */ -<PHPPARSING> TOKEN : -{ - <ASSIGN : "="> -| <PLUSASSIGN : "+="> -| <MINUSASSIGN : "-="> -| <STARASSIGN : "*="> -| <SLASHASSIGN : "/="> -| <ANDASSIGN : "&="> -| <ORASSIGN : "|="> -| <XORASSIGN : "^="> -| <DOTASSIGN : ".="> -| <REMASSIGN : "%="> -| <TILDEEQUAL : "~="> -| <LSHIFTASSIGN : "<<="> -| <RSIGNEDSHIFTASSIGN : ">>="> -} - -<PHPPARSING> TOKEN : +<PHPPARSING,IN_VARIABLE> TOKEN : +{ + <ASSIGN : "="> : PHPPARSING +| <PLUSASSIGN : "+="> : PHPPARSING +| <MINUSASSIGN : "-="> : PHPPARSING +| <STARASSIGN : "*="> : PHPPARSING +| <SLASHASSIGN : "/="> : PHPPARSING +| <ANDASSIGN : "&="> : PHPPARSING +| <ORASSIGN : "|="> : PHPPARSING +| <XORASSIGN : "^="> : PHPPARSING +| <DOTASSIGN : ".="> : PHPPARSING +| <REMASSIGN : "%="> : PHPPARSING +| <TILDEEQUAL : "~="> : PHPPARSING +| <LSHIFTASSIGN : "<<="> : PHPPARSING +| <RSIGNEDSHIFTASSIGN : ">>="> : PHPPARSING +} + +void phpTest() : +{} { - <DOLLAR_ID: <DOLLAR> <IDENTIFIER>> + Php() + <EOF> } void phpFile() : @@ -584,11 +746,11 @@ void phpFile() : { try { (PhpBlock())* - {PHPParser.createNewHTMLCode();} + {createNewHTMLCode();} } catch (TokenMgrError e) { PHPeclipsePlugin.log(e); - errorStart = SimpleCharStream.getPosition(); - errorEnd = errorStart + 1; + errorStart = jj_input_stream.getBeginOffset(); + errorEnd = jj_input_stream.getEndOffset(); errorMessage = e.getMessage(); errorLevel = ERROR; throw generateParseException(); @@ -602,48 +764,53 @@ void phpFile() : */ void PhpBlock() : { - final int start = SimpleCharStream.getPosition(); final PHPEchoBlock phpEchoBlock; + final Token token,phpEnd; } { phpEchoBlock = phpEchoBlock() {pushOnAstNodes(phpEchoBlock);} | [ <PHPSTARTLONG> - | <PHPSTARTSHORT> + | token = <PHPSTARTSHORT> {try { setMarker(fileToParse, "You should use '<?php' instead of '<?' it will avoid some problems with XML", - start, - SimpleCharStream.getPosition(), + token.sourceStart, + token.sourceEnd, INFO, "Line " + token.beginLine); } catch (CoreException e) { PHPeclipsePlugin.log(e); }} ] + {createNewHTMLCode();} Php() try { - <PHPEND> + phpEnd = <PHPEND> + {htmlStart = phpEnd.sourceEnd;} } catch (ParseException e) { errorMessage = "'?>' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } } PHPEchoBlock phpEchoBlock() : { final Expression expr; - final int pos = SimpleCharStream.getPosition(); final PHPEchoBlock echoBlock; + final Token token, token2; } { - <PHPECHOSTART> expr = Expression() [ <SEMICOLON> ] <PHPEND> + token = <PHPECHOSTART> {createNewHTMLCode();} + expr = Expression() [ <SEMICOLON> ] token2 = <PHPEND> { - echoBlock = new PHPEchoBlock(expr,pos,SimpleCharStream.getPosition()); + htmlStart = token2.sourceEnd; + + echoBlock = new PHPEchoBlock(expr,token.sourceStart,token2.sourceEnd); pushOnAstNodes(echoBlock); return echoBlock;} } @@ -657,82 +824,95 @@ void Php() : ClassDeclaration ClassDeclaration() : { final ClassDeclaration classDeclaration; - final Token className,superclassName; - final int pos; - char[] classNameImage = SYNTAX_ERROR_CHAR; - char[] superclassNameImage = null; + Token className = null; + final Token superclassName, token, extendsToken; + String classNameImage = SYNTAX_ERROR_CHAR; + String superclassNameImage = null; + final int classEnd; } { - <CLASS> - {pos = SimpleCharStream.getPosition();} + token = <CLASS> try { className = <IDENTIFIER> - {classNameImage = className.image.toCharArray();} + {classNameImage = className.image;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', identifier expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = token.sourceEnd+1; + errorEnd = token.sourceEnd+1; + processParseExceptionDebug(e); } [ - <EXTENDS> + extendsToken = <EXTENDS> try { superclassName = <IDENTIFIER> - {superclassNameImage = superclassName.image.toCharArray();} + {superclassNameImage = superclassName.image;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', identifier expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = extendsToken.sourceEnd+1; + errorEnd = extendsToken.sourceEnd+1; + processParseExceptionDebug(e); superclassNameImage = SYNTAX_ERROR_CHAR; } ] { + int start, end; + if (className == null) { + start = token.sourceStart; + end = token.sourceEnd; + } else { + start = className.sourceStart; + end = className.sourceEnd; + } if (superclassNameImage == null) { + classDeclaration = new ClassDeclaration(currentSegment, classNameImage, - pos, - 0); + start, + end); } else { classDeclaration = new ClassDeclaration(currentSegment, classNameImage, superclassNameImage, - pos, - 0); + start, + end); } currentSegment.add(classDeclaration); currentSegment = classDeclaration; } - ClassBody(classDeclaration) + classEnd = ClassBody(classDeclaration) {currentSegment = (OutlineableWithChildren) currentSegment.getParent(); - classDeclaration.sourceEnd = SimpleCharStream.getPosition(); + classDeclaration.sourceEnd = classEnd; pushOnAstNodes(classDeclaration); return classDeclaration;} } -void ClassBody(final ClassDeclaration classDeclaration) : -{} +int ClassBody(final ClassDeclaration classDeclaration) : +{ +Token token; +} { try { <LBRACE> } catch (ParseException e) { - errorMessage = "unexpected token : '"+ e.currentToken.next.image + "', '{' expected"; + errorMessage = "unexpected token : '"+ e.currentToken.next.image + "'. '{' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } ( ClassBodyDeclaration(classDeclaration) )* try { - <RBRACE> + token = <RBRACE> + {return token.sourceEnd;} } catch (ParseException e) { - errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', 'var', 'function' or '}' expected"; + errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. 'var', 'function' or '}' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); + return this.token.sourceEnd; } } @@ -745,77 +925,144 @@ void ClassBodyDeclaration(final ClassDeclaration classDeclaration) : final FieldDeclaration field; } { - method = MethodDeclaration() {classDeclaration.addMethod(method);} + method = MethodDeclaration() {method.analyzeCode(); + classDeclaration.addMethod(method);} | field = FieldDeclaration() {classDeclaration.addField(field);} } /** * A class field declaration : it's var VariableDeclarator() (, VariableDeclarator())*;. + * it is only used by ClassBodyDeclaration() */ FieldDeclaration FieldDeclaration() : { VariableDeclaration variableDeclaration; final VariableDeclaration[] list; final ArrayList arrayList = new ArrayList(); - final int pos = SimpleCharStream.getPosition(); + final Token token; + Token token2 = null; + int pos; } { - <VAR> variableDeclaration = VariableDeclarator() - {arrayList.add(variableDeclaration); - outlineInfo.addVariable(new String(variableDeclaration.name));} - ( <COMMA> variableDeclaration = VariableDeclarator() - {arrayList.add(variableDeclaration); - outlineInfo.addVariable(new String(variableDeclaration.name));} + token = <VAR> variableDeclaration = VariableDeclaratorNoSuffix() + { + arrayList.add(variableDeclaration); + pos = variableDeclaration.sourceEnd; + } + ( + <COMMA> variableDeclaration = VariableDeclaratorNoSuffix() + { + arrayList.add(variableDeclaration); + outlineInfo.addVariable(variableDeclaration.name()); + pos = variableDeclaration.sourceEnd; + } )* try { - <SEMICOLON> + token2 = <SEMICOLON> } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. A ';' was expected after variable declaration"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = pos+1; + errorEnd = pos+1; + processParseExceptionDebug(e); } {list = new VariableDeclaration[arrayList.size()]; arrayList.toArray(list); + int end; + if (token2 == null) { + end = list[list.length-1].sourceEnd; + } else { + end = token2.sourceEnd; + } return new FieldDeclaration(list, - pos, - SimpleCharStream.getPosition(), + token.sourceStart, + end, currentSegment);} } +/** + * a strict variable declarator : there cannot be a suffix here. + * It will be used by fields and formal parameters + */ +VariableDeclaration VariableDeclaratorNoSuffix() : +{ + final Token token, lbrace,rbrace; + Expression expr, initializer = null; + Token assignToken; + Variable variable; +} +{ + <DOLLAR> + ( + token = <IDENTIFIER> + {variable = new Variable(token.image,token.sourceStart,token.sourceEnd);} + | + lbrace = <LBRACE> expr = Expression() rbrace = <RBRACE> + {variable = new Variable(expr,lbrace.sourceStart,rbrace.sourceEnd);} + ) + [ + assignToken = <ASSIGN> + try { + initializer = VariableInitializer() + } catch (ParseException e) { + errorMessage = "Literal expression expected in variable initializer"; + errorLevel = ERROR; + errorStart = assignToken.sourceEnd +1; + errorEnd = assignToken.sourceEnd +1; + processParseExceptionDebug(e); + } + ] + { + if (initializer == null) { + return new VariableDeclaration(currentSegment, + variable, + variable.sourceStart, + variable.sourceEnd); + } + return new VariableDeclaration(currentSegment, + variable, + initializer, + VariableDeclaration.EQUAL, + variable.sourceStart); + } +} + +/** + * this will be used by static statement + */ VariableDeclaration VariableDeclarator() : { - final String varName; + final AbstractVariable variable; Expression initializer = null; - final int pos = SimpleCharStream.getPosition(); + final Token token; } { - varName = VariableDeclaratorId() + variable = VariableDeclaratorId() [ - <ASSIGN> + token = <ASSIGN> try { initializer = VariableInitializer() } catch (ParseException e) { errorMessage = "Literal expression expected in variable initializer"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = token.sourceEnd+1; + errorEnd = token.sourceEnd+1; + processParseExceptionDebug(e); } ] { if (initializer == null) { return new VariableDeclaration(currentSegment, - varName.toCharArray(), - pos, - SimpleCharStream.getPosition()); + variable, + variable.sourceStart, + variable.sourceEnd); } return new VariableDeclaration(currentSegment, - varName.toCharArray(), - initializer, - pos); + variable, + initializer, + VariableDeclaration.EQUAL, + variable.sourceStart); } } @@ -823,134 +1070,90 @@ VariableDeclaration VariableDeclarator() : * A Variable name. * @return the variable name (with suffix) */ -String VariableDeclaratorId() : +AbstractVariable VariableDeclaratorId() : { - final String expr; - Expression expression = null; - final int pos = SimpleCharStream.getPosition(); - ConstantIdentifier ex; + AbstractVariable var; } { try { - expr = Variable() - ( LOOKAHEAD(2) - {ex = new ConstantIdentifier(expr.toCharArray(), - pos, - SimpleCharStream.getPosition());} - expression = VariableSuffix(ex) + var = Variable() + ( + LOOKAHEAD(2) + var = VariableSuffix(var) )* { - if (expression == null) { - return expr; - } - return expression.toStringExpression(); + return var; } } catch (ParseException e) { errorMessage = "'$' expected for variable identifier"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } } -/** - * Return a variablename without the $. - * @return a variable name - */ -String Variable(): +Variable Variable() : { - final StringBuffer buff; - Expression expression = null; + Variable variable = null; final Token token; - final String expr; } { - token = <DOLLAR_ID> [<LBRACE> expression = Expression() <RBRACE>] + token = <DOLLAR> variable = Var() { - if (expression == null) { - return token.image.substring(1); - } - buff = new StringBuffer(token.image); - buff.append("{"); - buff.append(expression.toStringExpression()); - buff.append("}"); - return buff.toString(); + return variable; } -| - <DOLLAR> expr = VariableName() - {return expr;} } -/** - * A Variable name (without the $) - * @return a variable name String - */ -String VariableName(): +Variable Var() : { - final StringBuffer buff; - final String expr; - Expression expression = null; - final Token token; + Variable variable = null; + final Token token,token2; + ConstantIdentifier constant; + Expression expression; } { - <LBRACE> expression = Expression() <RBRACE> - {buff = new StringBuffer("{"); - buff.append(expression.toStringExpression()); - buff.append("}"); - return buff.toString();} + token = <DOLLAR> variable = Var() + {return new Variable(variable,variable.sourceStart,variable.sourceEnd);} | - token = <IDENTIFIER> [<LBRACE> expression = Expression() <RBRACE>] + token = <LBRACE> expression = Expression() token2 = <RBRACE> { - if (expression == null) { - return token.image; - } - buff = new StringBuffer(token.image); - buff.append("{"); - buff.append(expression.toStringExpression()); - buff.append("}"); - return buff.toString(); + return new Variable(expression, + token.sourceStart, + token2.sourceEnd); } | - <DOLLAR> expr = VariableName() + token = <IDENTIFIER> { - buff = new StringBuffer("$"); - buff.append(expr); - return buff.toString(); + outlineInfo.addVariable('$' + token.image); + return new Variable(token.image,token.sourceStart,token.sourceEnd); } -| - token = <DOLLAR_ID> {return token.image;} } Expression VariableInitializer() : { final Expression expr; - final Token token; - final int pos = SimpleCharStream.getPosition(); + final Token token, token2; } { expr = Literal() {return expr;} | - <MINUS> (token = <INTEGER_LITERAL> | token = <FLOATING_POINT_LITERAL>) - {return new PrefixedUnaryExpression(new NumberLiteral(token.image.toCharArray(), - pos, - SimpleCharStream.getPosition()), + token2 = <MINUS> (token = <INTEGER_LITERAL> | token = <FLOATING_POINT_LITERAL>) + {return new PrefixedUnaryExpression(new NumberLiteral(token), OperatorIds.MINUS, - pos);} + token2.sourceStart);} | - <PLUS> (token = <INTEGER_LITERAL> | token = <FLOATING_POINT_LITERAL>) - {return new PrefixedUnaryExpression(new NumberLiteral(token.image.toCharArray(), - pos, - SimpleCharStream.getPosition()), + token2 = <PLUS> (token = <INTEGER_LITERAL> | token = <FLOATING_POINT_LITERAL>) + {return new PrefixedUnaryExpression(new NumberLiteral(token), OperatorIds.PLUS, - pos);} + token2.sourceStart);} | expr = ArrayDeclarator() {return expr;} | token = <IDENTIFIER> - {return new ConstantIdentifier(token.image.toCharArray(),pos,SimpleCharStream.getPosition());} + {return new ConstantIdentifier(token);} } ArrayVariableDeclaration ArrayVariable() : @@ -959,10 +1162,11 @@ final Expression expr,expr2; } { expr = Expression() - [<ARRAYASSIGN> expr2 = Expression() - {return new ArrayVariableDeclaration(expr,expr2);} + [ + <ARRAYASSIGN> expr2 = Expression() + {return new ArrayVariableDeclaration(expr,expr2);} ] - {return new ArrayVariableDeclaration(expr,SimpleCharStream.getPosition());} + {return new ArrayVariableDeclaration(expr,jj_input_stream.getPosition());} } ArrayVariableDeclaration[] ArrayInitializer() : @@ -971,13 +1175,17 @@ ArrayVariableDeclaration[] ArrayInitializer() : final ArrayList list = new ArrayList(); } { - <LPAREN> [ expr = ArrayVariable() - {list.add(expr);} - ( LOOKAHEAD(2) <COMMA> expr = ArrayVariable() - {list.add(expr);} - )* - ] - [<COMMA> {list.add(null);}] + <LPAREN> + [ + expr = ArrayVariable() + {list.add(expr);} + ( LOOKAHEAD(2) <COMMA> expr = ArrayVariable() + {list.add(expr);} + )* + ] + [ + <COMMA> {list.add(null);} + ] <RPAREN> { final ArrayVariableDeclaration[] vars = new ArrayVariableDeclaration[list.size()]; @@ -994,18 +1202,19 @@ MethodDeclaration MethodDeclaration() : final MethodDeclaration functionDeclaration; final Block block; final OutlineableWithChildren seg = currentSegment; + final Token token; } { - <FUNCTION> + token = <FUNCTION> try { - functionDeclaration = MethodDeclarator() - {outlineInfo.addVariable(new String(functionDeclaration.name));} + functionDeclaration = MethodDeclarator(token.sourceStart) + {outlineInfo.addVariable(functionDeclaration.name);} } catch (ParseException e) { if (errorMessage != null) throw e; errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', function identifier expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } {currentSegment = functionDeclaration;} @@ -1020,71 +1229,96 @@ MethodDeclaration MethodDeclaration() : * [&] IDENTIFIER(parameters ...). * @return a function description for the outline */ -MethodDeclaration MethodDeclarator() : +MethodDeclaration MethodDeclarator(final int start) : { - final Token identifier; + Token identifier = null; Token reference = null; - final Hashtable formalParameters; - final int pos = SimpleCharStream.getPosition(); - char[] identifierChar = SYNTAX_ERROR_CHAR; + final ArrayList formalParameters = new ArrayList(); + String identifierChar = SYNTAX_ERROR_CHAR; + int end = start; } { - [reference = <BIT_AND>] + [reference = <BIT_AND> {end = reference.sourceEnd;}] try { identifier = <IDENTIFIER> - {identifierChar = identifier.image.toCharArray();} + { + identifierChar = identifier.image; + end = identifier.sourceEnd; + } } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', function identifier expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = e.currentToken.sourceEnd; + errorEnd = e.currentToken.next.sourceStart; + processParseExceptionDebug(e); + } + end = FormalParameters(formalParameters) + { + int nameStart, nameEnd; + if (identifier == null) { + if (reference == null) { + nameStart = start + 9; + nameEnd = start + 10; + } else { + nameStart = reference.sourceEnd + 1; + nameEnd = reference.sourceEnd + 2; + } + } else { + nameStart = identifier.sourceStart; + nameEnd = identifier.sourceEnd; + } + return new MethodDeclaration(currentSegment, + identifierChar, + formalParameters, + reference != null, + nameStart, + nameEnd, + start, + end); } - formalParameters = FormalParameters() - {return new MethodDeclaration(currentSegment, - identifierChar, - formalParameters, - reference != null, - pos, - SimpleCharStream.getPosition());} } /** * FormalParameters follows method identifier. * (FormalParameter()) */ -Hashtable FormalParameters() : +int FormalParameters(final ArrayList parameters) : { VariableDeclaration var; - final Hashtable parameters = new Hashtable(); + final Token token; + Token tok = this.token; + int end = tok.sourceEnd; } { try { - <LPAREN> + tok = <LPAREN> + {end = tok.sourceEnd;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', '(' expected after function identifier"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = e.currentToken.next.sourceStart; + errorEnd = e.currentToken.next.sourceEnd; + processParseExceptionDebug(e); } - [ var = FormalParameter() - {parameters.put(new String(var.name),var);} - ( - <COMMA> var = FormalParameter() - {parameters.put(new String(var.name),var);} - )* - ] + [ + var = FormalParameter() + {parameters.add(var);end = var.sourceEnd;} + ( + <COMMA> var = FormalParameter() + {parameters.add(var);end = var.sourceEnd;} + )* + ] try { - <RPAREN> + token = <RPAREN> + {end = token.sourceEnd;} } catch (ParseException e) { errorMessage = "')' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = e.currentToken.next.sourceStart; + errorEnd = e.currentToken.next.sourceEnd; + processParseExceptionDebug(e); } - {return parameters;} + {return end;} } /** @@ -1097,8 +1331,9 @@ VariableDeclaration FormalParameter() : Token token = null; } { - [token = <BIT_AND>] variableDeclaration = VariableDeclarator() + [token = <BIT_AND>] variableDeclaration = VariableDeclaratorNoSuffix() { + outlineInfo.addVariable('$'+variableDeclaration.name()); if (token != null) { variableDeclaration.setReference(true); } @@ -1106,26 +1341,17 @@ VariableDeclaration FormalParameter() : } ConstantIdentifier Type() : -{final int pos;} -{ - <STRING> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.STRING,pos,pos-6);} -| <BOOL> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.BOOL,pos,pos-4);} -| <BOOLEAN> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.BOOLEAN,pos,pos-7);} -| <REAL> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.REAL,pos,pos-4);} -| <DOUBLE> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.DOUBLE,pos,pos-5);} -| <FLOAT> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.FLOAT,pos,pos-5);} -| <INT> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.INT,pos,pos-3);} -| <INTEGER> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.INTEGER,pos,pos-7);} -| <OBJECT> {pos = SimpleCharStream.getPosition(); - return new ConstantIdentifier(Types.OBJECT,pos,pos-6);} +{final Token token;} +{ + token = <STRING> {return new ConstantIdentifier(token);} +| token = <BOOL> {return new ConstantIdentifier(token);} +| token = <BOOLEAN> {return new ConstantIdentifier(token);} +| token = <REAL> {return new ConstantIdentifier(token);} +| token = <DOUBLE> {return new ConstantIdentifier(token);} +| token = <FLOAT> {return new ConstantIdentifier(token);} +| token = <INT> {return new ConstantIdentifier(token);} +| token = <INTEGER> {return new ConstantIdentifier(token);} +| token = <OBJECT> {return new ConstantIdentifier(token);} } Expression Expression() : @@ -1133,53 +1359,65 @@ Expression Expression() : final Expression expr; Expression initializer = null; int assignOperator = -1; - final int pos = SimpleCharStream.getPosition(); } { LOOKAHEAD(1) expr = ConditionalExpression() - [ assignOperator = AssignmentOperator() + [ + assignOperator = AssignmentOperator() try { - {final int errorStart = SimpleCharStream.getPosition();} - initializer = Expression() + initializer = Expression() } catch (ParseException e) { if (errorMessage != null) { throw e; } errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', expression expected"; errorLevel = ERROR; - errorEnd = SimpleCharStream.getPosition(); + errorEnd = jj_input_stream.getPosition(); throw e; } ] { - if (assignOperator == -1) return expr; - return new VarAssignation(expr, - initializer, - assignOperator, - pos, - SimpleCharStream.getPosition()); - return expr;} + if (assignOperator != -1) {// todo : change this, very very bad :( + if (expr instanceof AbstractVariable) { + return new VariableDeclaration(currentSegment, + (AbstractVariable) expr, + initializer, + expr.sourceStart, + initializer.sourceEnd); + } + String varName = expr.toStringExpression().substring(1); + return new VariableDeclaration(currentSegment, + new Variable(varName, + expr.sourceStart, + expr.sourceEnd), + expr.sourceStart, + initializer.sourceEnd); + } + return expr; + } | expr = ExpressionWBang() {return expr;} } Expression ExpressionWBang() : { final Expression expr; - final int pos = SimpleCharStream.getPosition(); + final Token token; } { - <BANG> expr = ExpressionWBang() {return new PrefixedUnaryExpression(expr,OperatorIds.NOT,pos);} + token = <BANG> expr = ExpressionWBang() + {return new PrefixedUnaryExpression(expr,OperatorIds.NOT,token.sourceStart);} | expr = ExpressionNoBang() {return expr;} } Expression ExpressionNoBang() : { - final Expression expr; + Expression expr; } { - expr = PrintExpression() {return expr;} -| expr = ListExpression() {return expr;} + expr = ListExpression() {return expr;} +| + expr = PrintExpression() {return expr;} } /** @@ -1189,19 +1427,19 @@ Expression ExpressionNoBang() : int AssignmentOperator() : {} { - <ASSIGN> {return VarAssignation.EQUAL;} -| <STARASSIGN> {return VarAssignation.STAR_EQUAL;} -| <SLASHASSIGN> {return VarAssignation.SLASH_EQUAL;} -| <REMASSIGN> {return VarAssignation.REM_EQUAL;} -| <PLUSASSIGN> {return VarAssignation.PLUS_EQUAL;} -| <MINUSASSIGN> {return VarAssignation.MINUS_EQUAL;} -| <LSHIFTASSIGN> {return VarAssignation.LSHIFT_EQUAL;} -| <RSIGNEDSHIFTASSIGN> {return VarAssignation.RSIGNEDSHIFT_EQUAL;} -| <ANDASSIGN> {return VarAssignation.AND_EQUAL;} -| <XORASSIGN> {return VarAssignation.XOR_EQUAL;} -| <ORASSIGN> {return VarAssignation.OR_EQUAL;} -| <DOTASSIGN> {return VarAssignation.DOT_EQUAL;} -| <TILDEEQUAL> {return VarAssignation.TILDE_EQUAL;} + <ASSIGN> {return VariableDeclaration.EQUAL;} +| <STARASSIGN> {return VariableDeclaration.STAR_EQUAL;} +| <SLASHASSIGN> {return VariableDeclaration.SLASH_EQUAL;} +| <REMASSIGN> {return VariableDeclaration.REM_EQUAL;} +| <PLUSASSIGN> {return VariableDeclaration.PLUS_EQUAL;} +| <MINUSASSIGN> {return VariableDeclaration.MINUS_EQUAL;} +| <LSHIFTASSIGN> {return VariableDeclaration.LSHIFT_EQUAL;} +| <RSIGNEDSHIFTASSIGN> {return VariableDeclaration.RSIGNEDSHIFT_EQUAL;} +| <ANDASSIGN> {return VariableDeclaration.AND_EQUAL;} +| <XORASSIGN> {return VariableDeclaration.XOR_EQUAL;} +| <ORASSIGN> {return VariableDeclaration.OR_EQUAL;} +| <DOTASSIGN> {return VariableDeclaration.DOT_EQUAL;} +| <TILDEEQUAL> {return VariableDeclaration.TILDE_EQUAL;} } Expression ConditionalExpression() : @@ -1231,7 +1469,8 @@ Expression ConditionalOrExpression() : ( <OR_OR> {operator = OperatorIds.OR_OR;} | <_ORL> {operator = OperatorIds.ORL;} - ) expr2 = ConditionalAndExpression() + ) + expr2 = ConditionalAndExpression() { expr = new BinaryExpression(expr,expr2,operator); } @@ -1299,6 +1538,7 @@ Expression AndExpression() : { expr = EqualityExpression() ( + LOOKAHEAD(1) <BIT_AND> expr2 = EqualityExpression() {expr = new BinaryExpression(expr,expr2,OperatorIds.AND);} )* @@ -1309,15 +1549,16 @@ Expression EqualityExpression() : { Expression expr,expr2; int operator; + Token token; } { expr = RelationalExpression() ( - ( <EQUAL_EQUAL> {operator = OperatorIds.EQUAL_EQUAL;} - | <DIF> {operator = OperatorIds.DIF;} - | <NOT_EQUAL> {operator = OperatorIds.DIF;} - | <BANGDOUBLEEQUAL> {operator = OperatorIds.BANG_EQUAL_EQUAL;} - | <TRIPLEEQUAL> {operator = OperatorIds.EQUAL_EQUAL_EQUAL;} + ( token = <EQUAL_EQUAL> {operator = OperatorIds.EQUAL_EQUAL;} + | token = <DIF> {operator = OperatorIds.DIF;} + | token = <NOT_EQUAL> {operator = OperatorIds.DIF;} + | token = <BANGDOUBLEEQUAL> {operator = OperatorIds.BANG_EQUAL_EQUAL;} + | token = <TRIPLEEQUAL> {operator = OperatorIds.EQUAL_EQUAL_EQUAL;} ) try { expr2 = RelationalExpression() @@ -1327,9 +1568,10 @@ Expression EqualityExpression() : } errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', expression expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = token.sourceEnd +1; + errorEnd = token.sourceEnd +1; + expr2 = new ConstantIdentifier(SYNTAX_ERROR_CHAR,token.sourceEnd +1,token.sourceEnd +1); + processParseExceptionDebug(e); } { expr = new BinaryExpression(expr,expr2,operator); @@ -1381,8 +1623,10 @@ Expression AdditiveExpression() : { expr = MultiplicativeExpression() ( - ( <PLUS> {operator = OperatorIds.PLUS;} - | <MINUS> {operator = OperatorIds.MINUS;} ) + LOOKAHEAD(1) + ( <PLUS> {operator = OperatorIds.PLUS;} + | <MINUS> {operator = OperatorIds.MINUS;} + ) expr2 = MultiplicativeExpression() {expr = new BinaryExpression(expr,expr2,operator);} )* @@ -1399,10 +1643,10 @@ Expression MultiplicativeExpression() : expr = UnaryExpression() } catch (ParseException e) { if (errorMessage != null) throw e; - errorMessage = "unexpected token '"+e.currentToken.next.image+"'"; + errorMessage = "unexpected token '"+e.currentToken.next.image+'\''; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = this.token.sourceStart; + errorEnd = this.token.sourceEnd; throw e; } ( @@ -1421,41 +1665,71 @@ Expression MultiplicativeExpression() : Expression UnaryExpression() : { final Expression expr; - final int pos = SimpleCharStream.getPosition(); } { - <BIT_AND> expr = UnaryExpressionNoPrefix() + /* <BIT_AND> expr = UnaryExpressionNoPrefix() //why did I had that ? {return new PrefixedUnaryExpression(expr,OperatorIds.AND,pos);} -| - expr = AtUnaryExpression() {return expr;} +| */ + expr = AtNotTildeUnaryExpression() {return expr;} } -Expression AtUnaryExpression() : +Expression AtNotTildeUnaryExpression() : { final Expression expr; - final int pos = SimpleCharStream.getPosition(); + final Token token; } { - <AT> - expr = AtUnaryExpression() - {return new PrefixedUnaryExpression(expr,OperatorIds.AT,pos);} + token = <AT> + expr = AtNotTildeUnaryExpression() + {return new PrefixedUnaryExpression(expr,OperatorIds.AT,token.sourceStart);} +| + token = <TILDE> + expr = AtNotTildeUnaryExpression() + {return new PrefixedUnaryExpression(expr,OperatorIds.TWIDDLE,token.sourceStart);} +| + token = <BANG> + expr = AtNotUnaryExpression() + {return new PrefixedUnaryExpression(expr,OperatorIds.NOT,token.sourceStart);} | expr = UnaryExpressionNoPrefix() {return expr;} } +/** + * An expression prefixed (or not) by one or more @ and !. + * @return the expression + */ +Expression AtNotUnaryExpression() : +{ + final Expression expr; + final Token token; +} +{ + token = <AT> + expr = AtNotUnaryExpression() + {return new PrefixedUnaryExpression(expr,OperatorIds.AT,token.sourceStart);} +| + token = <BANG> + expr = AtNotUnaryExpression() + {return new PrefixedUnaryExpression(expr,OperatorIds.NOT,token.sourceStart);} +| + expr = UnaryExpressionNoPrefix() + {return expr;} +} Expression UnaryExpressionNoPrefix() : { final Expression expr; - final int operator; - final int pos = SimpleCharStream.getPosition(); + final Token token; } { - ( <PLUS> {operator = OperatorIds.PLUS;} - | <MINUS> {operator = OperatorIds.MINUS;}) - expr = UnaryExpression() - {return new PrefixedUnaryExpression(expr,operator,pos);} + token = <PLUS> expr = AtNotTildeUnaryExpression() {return new PrefixedUnaryExpression(expr, + OperatorIds.PLUS, + token.sourceStart);} +| + token = <MINUS> expr = AtNotTildeUnaryExpression() {return new PrefixedUnaryExpression(expr, + OperatorIds.MINUS, + token.sourceStart);} | expr = PreIncDecExpression() {return expr;} @@ -1469,24 +1743,25 @@ Expression PreIncDecExpression() : { final Expression expr; final int operator; - final int pos = SimpleCharStream.getPosition(); +final Token token; } { - ( <PLUS_PLUS> {operator = OperatorIds.PLUS_PLUS;} - | <MINUS_MINUS> {operator = OperatorIds.MINUS_MINUS;}) - expr = PrimaryExpression() - {return new PrefixedUnaryExpression(expr,operator,pos);} + ( + token = <PLUS_PLUS> {operator = OperatorIds.PLUS_PLUS;} + | + token = <MINUS_MINUS> {operator = OperatorIds.MINUS_MINUS;} + ) + expr = PrimaryExpression() + {return new PrefixedUnaryExpression(expr,operator,token.sourceStart);} } Expression UnaryExpressionNotPlusMinus() : { final Expression expr; - final int pos = SimpleCharStream.getPosition(); } { LOOKAHEAD( <LPAREN> (Type() | <ARRAY>) <RPAREN> ) expr = CastExpression() {return expr;} -| <BANG> expr = UnaryExpression() {return new PrefixedUnaryExpression(expr,OperatorIds.NOT,pos);} | expr = PostfixExpression() {return expr;} | expr = Literal() {return expr;} | <LPAREN> expr = Expression() @@ -1495,9 +1770,9 @@ Expression UnaryExpressionNotPlusMinus() : } catch (ParseException e) { errorMessage = "')' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = expr.sourceEnd +1; + errorEnd = expr.sourceEnd +1; + processParseExceptionDebug(e); } {return expr;} } @@ -1506,84 +1781,105 @@ CastExpression CastExpression() : { final ConstantIdentifier type; final Expression expr; -final int pos = SimpleCharStream.getPosition(); +final Token token,token1; } { - <LPAREN> - (type = Type() - | <ARRAY> {type = new ConstantIdentifier(Types.ARRAY,pos,SimpleCharStream.getPosition());}) + token1 = <LPAREN> + ( + type = Type() + | + token = <ARRAY> {type = new ConstantIdentifier(token);} + ) <RPAREN> expr = UnaryExpression() - {return new CastExpression(type,expr,pos,SimpleCharStream.getPosition());} + {return new CastExpression(type,expr,token1.sourceStart,expr.sourceEnd);} } Expression PostfixExpression() : { final Expression expr; int operator = -1; - final int pos = SimpleCharStream.getPosition(); + Token token = null; } { expr = PrimaryExpression() - [ <PLUS_PLUS> {operator = OperatorIds.PLUS_PLUS;} - | <MINUS_MINUS> {operator = OperatorIds.MINUS_MINUS;}] + [ + token = <PLUS_PLUS> {operator = OperatorIds.PLUS_PLUS;} + | + token = <MINUS_MINUS> {operator = OperatorIds.MINUS_MINUS;} + ] { if (operator == -1) { return expr; } - return new PostfixedUnaryExpression(expr,operator,pos); + return new PostfixedUnaryExpression(expr,operator,token.sourceEnd); } } Expression PrimaryExpression() : { - final Token identifier; Expression expr; - final int pos = SimpleCharStream.getPosition(); + Token token = null; } { - expr = PrimaryPrefix() - ( - LOOKAHEAD(PrimarySuffix()) - expr = PrimarySuffix(expr))* + [token = <BIT_AND>] expr = refPrimaryExpression(token) {return expr;} | expr = ArrayDeclarator() {return expr;} } -Expression PrimaryPrefix() : -{ - final Expression expr; - final Token token; - final String var; - final int pos = SimpleCharStream.getPosition(); -} -{ - token = <IDENTIFIER> {return new ConstantIdentifier(token.image.toCharArray(), - pos, - SimpleCharStream.getPosition());} -| <NEW> expr = ClassIdentifier() {return new PrefixedUnaryExpression(expr, - OperatorIds.NEW, - pos);} -| var = VariableDeclaratorId() {return new VariableDeclaration(currentSegment, - var.toCharArray(), - pos, - SimpleCharStream.getPosition());} -} - -AbstractSuffixExpression PrimarySuffix(final Expression prefix) : +Expression refPrimaryExpression(final Token reference) : { - final AbstractSuffixExpression suffix; - final Expression expr; + Expression expr; + Expression expr2 = null; + final Token identifier; } { - suffix = Arguments(prefix) {return suffix;} -| suffix = VariableSuffix(prefix) {return suffix;} -| <STATICCLASSACCESS> expr = ClassIdentifier() - {suffix = new ClassAccess(prefix, - expr, - ClassAccess.STATIC); - return suffix;} + identifier = <IDENTIFIER> + { + expr = new ConstantIdentifier(identifier); + } + ( + <STATICCLASSACCESS> expr2 = ClassIdentifier() + {expr = new ClassAccess(expr, + expr2, + ClassAccess.STATIC);} + )* + [ expr2 = Arguments(expr) ] + { + if (expr2 == null) { + if (reference != null) { + ParseException e = generateParseException(); + errorMessage = "you cannot use a constant by reference"; + errorLevel = ERROR; + errorStart = reference.sourceStart; + errorEnd = reference.sourceEnd; + processParseExceptionDebug(e); + } + return expr; + } + return expr2; + } +| + expr = VariableDeclaratorId() //todo use the reference parameter ... + [ expr = Arguments(expr) ] + {return expr;} +| + token = <NEW> + expr = ClassIdentifier() + { + int start; + if (reference == null) { + start = token.sourceStart; + } else { + start = reference.sourceStart; + } + expr = new ClassInstantiation(expr, + reference != null, + start); + } + [ expr = Arguments(expr) ] + {return expr;} } /** @@ -1594,121 +1890,170 @@ AbstractSuffixExpression PrimarySuffix(final Expression prefix) : ArrayInitializer ArrayDeclarator() : { final ArrayVariableDeclaration[] vars; - final int pos = SimpleCharStream.getPosition(); -} -{ - <ARRAY> vars = ArrayInitializer() - {return new ArrayInitializer(vars,pos,SimpleCharStream.getPosition());} -} - -PrefixedUnaryExpression classInstantiation() : -{ - Expression expr; - final StringBuffer buff; - final int pos = SimpleCharStream.getPosition(); + final Token token; } { - <NEW> expr = ClassIdentifier() - [ - {buff = new StringBuffer(expr.toStringExpression());} - expr = PrimaryExpression() - {buff.append(expr.toStringExpression()); - expr = new ConstantIdentifier(buff.toString().toCharArray(), - pos, - SimpleCharStream.getPosition());} - ] - {return new PrefixedUnaryExpression(expr, - OperatorIds.NEW, - pos);} + token = <ARRAY> vars = ArrayInitializer() + {return new ArrayInitializer(vars, + token.sourceStart, + this.token.sourceEnd);} } -ConstantIdentifier ClassIdentifier(): +Expression ClassIdentifier(): { - final String expr; + final Expression expr; final Token token; - final int pos = SimpleCharStream.getPosition(); - final ConstantIdentifier type; } { - token = <IDENTIFIER> {return new ConstantIdentifier(token.image.toCharArray(), - pos, - SimpleCharStream.getPosition());} -| type = Type() {return type;} -| expr = VariableDeclaratorId() {return new ConstantIdentifier(expr.toCharArray(), - pos, - SimpleCharStream.getPosition());} + token = <IDENTIFIER> {return new ConstantIdentifier(token);} +| expr = Type() {return expr;} +| expr = VariableDeclaratorId() {return expr;} } -AbstractSuffixExpression VariableSuffix(final Expression prefix) : +/** + * Used by Variabledeclaratorid and primarysuffix + */ +AbstractVariable VariableSuffix(final AbstractVariable prefix) : { - String expr = null; - final int pos = SimpleCharStream.getPosition(); Expression expression = null; + final Token classAccessToken,lbrace,rbrace; + Token token; + int pos; } { - <CLASSACCESS> + classAccessToken = <CLASSACCESS> try { - expr = VariableName() + ( + lbrace = <LBRACE> expression = Expression() rbrace = <RBRACE> + { + expression = new Variable(expression, + lbrace.sourceStart, + rbrace.sourceEnd); + } + | + token = <IDENTIFIER> + {expression = new ConstantIdentifier(token.image,token.sourceStart,token.sourceEnd);} + | + expression = Variable() + ) } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', function call or field access expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = classAccessToken.sourceEnd +1; + errorEnd = classAccessToken.sourceEnd +1; + processParseExceptionDebug(e); } {return new ClassAccess(prefix, - new ConstantIdentifier(expr.toCharArray(),pos,SimpleCharStream.getPosition()), + expression, ClassAccess.NORMAL);} | - <LBRACKET> [ expression = Expression() | expression = Type() ] //Not good + token = <LBRACKET> {pos = token.sourceEnd+1;} + [ expression = Expression() {pos = expression.sourceEnd+1;} + | expression = Type() {pos = expression.sourceEnd+1;}] //Not good try { - <RBRACKET> + token = <RBRACKET> + {pos = token.sourceEnd;} } catch (ParseException e) { errorMessage = "']' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); + } + {return new ArrayDeclarator(prefix,expression,pos);} +| + token = <LBRACE> {pos = token.sourceEnd+1;} + [ expression = Expression() {pos = expression.sourceEnd+1;} + | expression = Type() {pos = expression.sourceEnd+1;}] //Not good + try { + token = <RBRACE> + {pos = token.sourceEnd;} + } catch (ParseException e) { + errorMessage = "']' expected"; + errorLevel = ERROR; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } - {return new ArrayDeclarator(prefix,expression,SimpleCharStream.getPosition());} + {return new ArrayDeclarator(prefix,expression,pos);}//todo : check braces here } Literal Literal() : { final Token token; - final int pos; + StringLiteral literal; } { - token = <INTEGER_LITERAL> {pos = SimpleCharStream.getPosition(); - return new NumberLiteral(token.image.toCharArray(),pos-token.image.length(),pos);} -| token = <FLOATING_POINT_LITERAL> {pos = SimpleCharStream.getPosition(); - return new NumberLiteral(token.image.toCharArray(),pos-token.image.length(),pos);} -| token = <STRING_LITERAL> {pos = SimpleCharStream.getPosition(); - return new StringLiteral(token.image.toCharArray(),pos-token.image.length());} -| <TRUE> {pos = SimpleCharStream.getPosition(); - return new TrueLiteral(pos-4,pos);} -| <FALSE> {pos = SimpleCharStream.getPosition(); - return new FalseLiteral(pos-4,pos);} -| <NULL> {pos = SimpleCharStream.getPosition(); - return new NullLiteral(pos-4,pos);} + token = <INTEGER_LITERAL> {return new NumberLiteral(token);} +| token = <FLOATING_POINT_LITERAL> {return new NumberLiteral(token);} +| token = <STRING_LITERAL> {return new StringLiteral(token);} +| token = <TRUE> {return new TrueLiteral(token);} +| token = <FALSE> {return new FalseLiteral(token);} +| token = <NULL> {return new NullLiteral(token);} +| literal = evaluableString() {return literal;} +} + +StringLiteral evaluableString() : +{ + ArrayList list = new ArrayList(); + Token start,end; + Token token,lbrace,rbrace; + AbstractVariable var; + Expression expr; +} +{ + start = <DOUBLEQUOTE> + ( + <DOLLARS> + ( + token = <IDENTIFIER> {list.add(new Variable(token.image, + token.sourceStart, + token.sourceEnd));} + | + lbrace = <LBRACE1> + token = <ID> + {list.add(new Variable(token.image, + token.sourceStart, + token.sourceEnd));} + rbrace = <RBRACE1> + ) + )* + end = <DOUBLEQUOTE2> + { + AbstractVariable[] vars = new AbstractVariable[list.size()]; + list.toArray(vars); + return new StringLiteral(jj_input_stream.getCurrentBuffer().substring(start.sourceEnd,end.sourceStart), + start.sourceStart, + end.sourceEnd, + vars); + } } FunctionCall Arguments(final Expression func) : { Expression[] args = null; +final Token token,lparen; } { - <LPAREN> [ args = ArgumentList() ] + lparen = <LPAREN> [ args = ArgumentList() ] try { - <RPAREN> + token = <RPAREN> + {return new FunctionCall(func,args,token.sourceEnd);} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', ')' expected to close the argument list"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + if (args == null) { + errorStart = lparen.sourceEnd+1; + errorEnd = lparen.sourceEnd+2; + } else { + errorStart = args[args.length-1].sourceEnd+1; + errorEnd = args[args.length-1].sourceEnd+2; + } + processParseExceptionDebug(e); } - {return new FunctionCall(func,args,SimpleCharStream.getPosition());} + { + int sourceEnd = (args == null && args.length != 0) ? lparen.sourceEnd+1 : args[args.length-1].sourceEnd; + return new FunctionCall(func,args,sourceEnd);} } /** @@ -1720,20 +2065,23 @@ Expression[] ArgumentList() : { Expression arg; final ArrayList list = new ArrayList(); +int pos; +Token token; } { arg = Expression() - {list.add(arg);} - ( <COMMA> + {list.add(arg);pos = arg.sourceEnd;} + ( token = <COMMA> {pos = token.sourceEnd;} try { arg = Expression() - {list.add(arg);} + {list.add(arg); + pos = arg.sourceEnd;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. An expression expected after a comma in argument list"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos+1; + errorEnd = pos+1; + processParseException(e); } )* { @@ -1744,6 +2092,7 @@ final ArrayList list = new ArrayList(); /** * A Statement without break. + * @return a statement */ Statement StatementNoBreak() : { @@ -1752,34 +2101,11 @@ Statement StatementNoBreak() : } { LOOKAHEAD(2) - statement = Expression() - try { - <SEMICOLON> - } catch (ParseException e) { - if (e.currentToken.next.kind != PHPParserConstants.PHPEND) { - errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. A ';' was expected"; - errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; - } - } - {return statement;} -| LOOKAHEAD(2) - statement = LabeledStatement() {return statement;} -| statement = Block() {return statement;} -| statement = EmptyStatement() {return statement;} -| statement = StatementExpression() - try { - <SEMICOLON> - } catch (ParseException e) { - errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. A ';' was expected"; - errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; - } - {return statement;} + statement = expressionStatement() {return statement;} +| LOOKAHEAD(1) + statement = LabeledStatement() {return statement;} +| statement = Block() {return statement;} +| statement = EmptyStatement() {return statement;} | statement = SwitchStatement() {return statement;} | statement = IfStatement() {return statement;} | statement = WhileStatement() {return statement;} @@ -1792,6 +2118,7 @@ Statement StatementNoBreak() : | [token=<AT>] statement = IncludeStatement() {if (token != null) { ((InclusionStatement)statement).silent = true; + statement.sourceStart = token.sourceStart; } return statement;} | statement = StaticStatement() {return statement;} @@ -1799,63 +2126,99 @@ Statement StatementNoBreak() : | statement = defineStatement() {currentSegment.add((Outlineable)statement);return statement;} } +/** + * A statement expression. + * expression ; + * @return an expression + */ +Statement expressionStatement() : +{ + final Statement statement; + final Token token; +} +{ + statement = Expression() + try { + token = <SEMICOLON> + {statement.sourceEnd = token.sourceEnd;} + } catch (ParseException e) { + if (e.currentToken.next.kind != PHPParserConstants.PHPEND) { + errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. A ';' was expected"; + errorLevel = ERROR; + errorStart = statement.sourceEnd+1; + errorEnd = statement.sourceEnd+1; + processParseExceptionDebug(e); + } + } + {return statement;} +} + Define defineStatement() : { - final int start = SimpleCharStream.getPosition(); Expression defineName,defineValue; + final Token defineToken; + Token token; + int pos; } { - <DEFINE> + defineToken = <DEFINE> {pos = defineToken.sourceEnd+1;} try { - <LPAREN> + token = <LPAREN> + {pos = token.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', '(' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } try { defineName = Expression() + {pos = defineName.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', expression expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); + defineName = new StringLiteral(SYNTAX_ERROR_CHAR,pos,pos); } try { - <COMMA> + token = <COMMA> + {pos = defineName.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', ',' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } try { defineValue = Expression() + {pos = defineValue.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', expression expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); + defineValue = new StringLiteral(SYNTAX_ERROR_CHAR,pos,pos); } try { - <RPAREN> + token = <RPAREN> + {pos = token.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', ')' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } {return new Define(currentSegment, defineName, defineValue, - start, - SimpleCharStream.getPosition());} + defineToken.sourceStart, + pos);} } /** @@ -1878,22 +2241,29 @@ HTMLBlock htmlBlock() : final int startIndex = nodePtr; final AstNode[] blockNodes; final int nbNodes; + final Token phpEnd; } { - <PHPEND> (phpEchoBlock())* + phpEnd = <PHPEND> + {htmlStart = phpEnd.sourceEnd;} + (phpEchoBlock())* try { (<PHPSTARTLONG> | <PHPSTARTSHORT>) + {createNewHTMLCode();} } catch (ParseException e) { errorMessage = "unexpected end of file , '<?php' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition(); - errorEnd = SimpleCharStream.getPosition(); + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } { nbNodes = nodePtr - startIndex; + if (nbNodes == 0) { + return null; + } blockNodes = new AstNode[nbNodes]; - System.arraycopy(nodes,startIndex,blockNodes,0,nbNodes); + System.arraycopy(nodes,startIndex+1,blockNodes,0,nbNodes); nodePtr = startIndex; return new HTMLBlock(blockNodes);} } @@ -1903,112 +2273,123 @@ HTMLBlock htmlBlock() : */ InclusionStatement IncludeStatement() : { - final Expression expr; + Expression expr; final int keyword; - final int pos = SimpleCharStream.getPosition(); final InclusionStatement inclusionStatement; + final Token token, token2; + int pos; } { - ( <REQUIRE> {keyword = InclusionStatement.REQUIRE;} - | <REQUIRE_ONCE> {keyword = InclusionStatement.REQUIRE_ONCE;} - | <INCLUDE> {keyword = InclusionStatement.INCLUDE;} - | <INCLUDE_ONCE> {keyword = InclusionStatement.INCLUDE_ONCE;}) + ( token = <REQUIRE> {keyword = InclusionStatement.REQUIRE;pos=token.sourceEnd;} + | token = <REQUIRE_ONCE> {keyword = InclusionStatement.REQUIRE_ONCE;pos=token.sourceEnd;} + | token = <INCLUDE> {keyword = InclusionStatement.INCLUDE;pos=token.sourceEnd;} + | token = <INCLUDE_ONCE> {keyword = InclusionStatement.INCLUDE_ONCE;pos=token.sourceEnd;}) try { expr = Expression() + {pos = expr.sourceEnd;} } catch (ParseException e) { if (errorMessage != null) { throw e; } errorMessage = "unexpected token '"+ e.currentToken.next.image+"', expression expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; - } - {inclusionStatement = new InclusionStatement(currentSegment, - keyword, - expr, - pos); - currentSegment.add(inclusionStatement); + errorStart = e.currentToken.next.sourceStart; + errorEnd = e.currentToken.next.sourceEnd; + expr = new ConstantIdentifier(SYNTAX_ERROR_CHAR,pos,pos); + processParseExceptionDebug(e); } try { - <SEMICOLON> + token2 = <SEMICOLON> + {pos=token2.sourceEnd;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. A ';' was expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.next.sourceStart; + errorEnd = e.currentToken.next.sourceEnd; + processParseExceptionDebug(e); + } + { + inclusionStatement = new InclusionStatement(currentSegment, + keyword, + expr, + token.sourceStart, + pos); + currentSegment.add(inclusionStatement); + return inclusionStatement; } - {return inclusionStatement;} } PrintExpression PrintExpression() : { final Expression expr; - final int pos = SimpleCharStream.getPosition(); + final Token printToken; } { - <PRINT> expr = Expression() {return new PrintExpression(expr,pos,SimpleCharStream.getPosition());} + token = <PRINT> expr = Expression() + {return new PrintExpression(expr,token.sourceStart,expr.sourceEnd);} } ListExpression ListExpression() : { - String expr = null; + Expression expr = null; final Expression expression; final ArrayList list = new ArrayList(); - final int pos = SimpleCharStream.getPosition(); + int pos; + final Token listToken, rParen; + Token token; } { - <LIST> + listToken = <LIST> {pos = listToken.sourceEnd;} try { - <LPAREN> + token = <LPAREN> {pos = token.sourceEnd;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', '(' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = listToken.sourceEnd+1; + errorEnd = listToken.sourceEnd+1; + processParseExceptionDebug(e); } [ expr = VariableDeclaratorId() - {list.add(expr);} + {list.add(expr);pos = expr.sourceEnd;} ] {if (expr == null) list.add(null);} ( try { - <COMMA> + token = <COMMA> + {pos = token.sourceEnd;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', ',' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos+1; + errorEnd = pos+1; + processParseExceptionDebug(e); } - [expr = VariableDeclaratorId() {list.add(expr);}] + [expr = VariableDeclaratorId() {list.add(expr);pos = expr.sourceEnd;}] )* try { - <RPAREN> + rParen = <RPAREN> + {pos = rParen.sourceEnd;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"', ')' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos+1; + errorEnd = pos+1; + processParseExceptionDebug(e); } [ <ASSIGN> expression = Expression() { - final String[] strings = new String[list.size()]; - list.toArray(strings); - return new ListExpression(strings, + final AbstractVariable[] vars = new AbstractVariable[list.size()]; + list.toArray(vars); + return new ListExpression(vars, expression, - pos, - SimpleCharStream.getPosition());} + listToken.sourceStart, + expression.sourceEnd);} ] { - final String[] strings = new String[list.size()]; - list.toArray(strings); - return new ListExpression(strings,pos,SimpleCharStream.getPosition());} + final AbstractVariable[] vars = new AbstractVariable[list.size()]; + list.toArray(vars); + return new ListExpression(vars,listToken.sourceStart,pos);} } /** @@ -2019,101 +2400,112 @@ EchoStatement EchoStatement() : { final ArrayList expressions = new ArrayList(); Expression expr; - final int pos = SimpleCharStream.getPosition(); + Token token; + Token token2 = null; } { - <ECHO> expr = Expression() + token = <ECHO> expr = Expression() {expressions.add(expr);} ( <COMMA> expr = Expression() {expressions.add(expr);} )* try { - <SEMICOLON> + token2 = <SEMICOLON> } catch (ParseException e) { if (e.currentToken.next.kind != 4) { errorMessage = "';' expected after 'echo' statement"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceEnd; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } } - {final Expression[] exprs = new Expression[expressions.size()]; + { + final Expression[] exprs = new Expression[expressions.size()]; expressions.toArray(exprs); - return new EchoStatement(exprs,pos);} + if (token2 == null) { + return new EchoStatement(exprs,token.sourceStart, exprs[exprs.length-1].sourceEnd); + } + return new EchoStatement(exprs,token.sourceStart, token2.sourceEnd); + } } GlobalStatement GlobalStatement() : { - final int pos = SimpleCharStream.getPosition(); - String expr; + Variable expr; final ArrayList vars = new ArrayList(); final GlobalStatement global; + final Token token, token2; + int pos; } { - <GLOBAL> - expr = VariableDeclaratorId() - {vars.add(expr);} + token = <GLOBAL> + expr = Variable() + {vars.add(expr);pos = expr.sourceEnd+1;} (<COMMA> - expr = VariableDeclaratorId() - {vars.add(expr);} + expr = Variable() + {vars.add(expr);pos = expr.sourceEnd+1;} )* try { - <SEMICOLON> - { - final String[] strings = new String[vars.size()]; - vars.toArray(strings); - global = new GlobalStatement(currentSegment, - strings, - pos, - SimpleCharStream.getPosition()); - currentSegment.add(global); - return global;} + token2 = <SEMICOLON> + {pos = token2.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. a ';' was expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } + { + final Variable[] variables = new Variable[vars.size()]; + vars.toArray(variables); + global = new GlobalStatement(currentSegment, + variables, + token.sourceStart, + pos); + currentSegment.add(global); + return global;} } StaticStatement StaticStatement() : { - final int pos = SimpleCharStream.getPosition(); final ArrayList vars = new ArrayList(); VariableDeclaration expr; + final Token token, token2; + int pos; } { - <STATIC> expr = VariableDeclarator() {vars.add(new String(expr.name));} - (<COMMA> expr = VariableDeclarator() {vars.add(new String(expr.name));})* + token = <STATIC> expr = VariableDeclarator() {vars.add(expr);pos = expr.sourceEnd+1;} + ( + <COMMA> expr = VariableDeclarator() {vars.add(expr);pos = expr.sourceEnd+1;} + )* try { - <SEMICOLON> - { - final String[] strings = new String[vars.size()]; - vars.toArray(strings); - return new StaticStatement(strings, - pos, - SimpleCharStream.getPosition());} + token2 = <SEMICOLON> + {pos = token2.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. a ';' was expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseException(e); } + { + final VariableDeclaration[] variables = new VariableDeclaration[vars.size()]; + vars.toArray(variables); + return new StaticStatement(variables, + token.sourceStart, + pos);} } LabeledStatement LabeledStatement() : { - final int pos = SimpleCharStream.getPosition(); final Token label; final Statement statement; } { label = <IDENTIFIER> <COLON> statement = Statement() - {return new LabeledStatement(label.image.toCharArray(),statement,pos,SimpleCharStream.getPosition());} + {return new LabeledStatement(label.image,statement,label.sourceStart,statement.sourceEnd);} } /** @@ -2125,35 +2517,46 @@ LabeledStatement LabeledStatement() : */ Block Block() : { - final int pos = SimpleCharStream.getPosition(); final ArrayList list = new ArrayList(); Statement statement; + final Token token, token2; + int pos,start; } { try { - <LBRACE> + token = <LBRACE> + {pos = token.sourceEnd+1;start=token.sourceStart;} } catch (ParseException e) { errorMessage = "'{' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; - } - ( statement = BlockStatement() {list.add(statement);} - | statement = htmlBlock() {list.add(statement);})* + pos = this.token.sourceEnd+1; + start=pos; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); + } + ( statement = BlockStatement() {list.add(statement);pos = statement.sourceEnd+1;} + | statement = htmlBlock() {if (statement != null) { + list.add(statement); + pos = statement.sourceEnd+1; + } + pos = this.token.sourceEnd+1; + } + )* try { - <RBRACE> + token2 = <RBRACE> + {pos = token2.sourceEnd+1;} } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.image +"', '}' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } { final Statement[] statements = new Statement[list.size()]; list.toArray(statements); - return new Block(statements,pos,SimpleCharStream.getPosition());} + return new Block(statements,start,pos);} } Statement BlockStatement() : @@ -2161,11 +2564,20 @@ Statement BlockStatement() : final Statement statement; } { - statement = Statement() {if (phpDocument == currentSegment) pushOnAstNodes(statement); - return statement;} + try { + statement = Statement() {if (phpDocument == currentSegment) pushOnAstNodes(statement); + return statement;} + } catch (ParseException e) { + errorMessage = "unexpected token : '"+ e.currentToken.image +"', a statement was expected"; + errorLevel = ERROR; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + throw e; + } | statement = ClassDeclaration() {return statement;} | statement = MethodDeclaration() {if (phpDocument == currentSegment) pushOnAstNodes(statement); currentSegment.add((MethodDeclaration) statement); + ((MethodDeclaration) statement).analyzeCode(); return statement;} } @@ -2180,141 +2592,161 @@ Statement BlockStatementNoBreak() : statement = StatementNoBreak() {return statement;} | statement = ClassDeclaration() {return statement;} | statement = MethodDeclaration() {currentSegment.add((MethodDeclaration) statement); + ((MethodDeclaration) statement).analyzeCode(); return statement;} } -VariableDeclaration[] LocalVariableDeclaration() : +/** + * used only by ForInit() + */ +Expression[] LocalVariableDeclaration() : { final ArrayList list = new ArrayList(); - VariableDeclaration var; + Expression var; } { - var = LocalVariableDeclarator() + var = Expression() {list.add(var);} - ( <COMMA> var = LocalVariableDeclarator() {list.add(var);})* + ( <COMMA> var = Expression() {list.add(var);})* { - final VariableDeclaration[] vars = new VariableDeclaration[list.size()]; + final Expression[] vars = new Expression[list.size()]; list.toArray(vars); - return vars;} + return vars; + } } +/** + * used only by LocalVariableDeclaration(). + */ VariableDeclaration LocalVariableDeclarator() : { - final String varName; + final Variable varName; Expression initializer = null; - final int pos = SimpleCharStream.getPosition(); } { - varName = VariableDeclaratorId() [ <ASSIGN> initializer = Expression() ] + varName = Variable() [ <ASSIGN> initializer = Expression() ] { if (initializer == null) { return new VariableDeclaration(currentSegment, - varName.toCharArray(), - pos, - SimpleCharStream.getPosition()); + varName, + varName.sourceStart, + varName.sourceEnd); } return new VariableDeclaration(currentSegment, - varName.toCharArray(), - initializer, - pos); + varName, + initializer, + VariableDeclaration.EQUAL, + varName.sourceStart); } } EmptyStatement EmptyStatement() : { - final int pos; + final Token token; } { - <SEMICOLON> - {pos = SimpleCharStream.getPosition(); - return new EmptyStatement(pos-1,pos);} + token = <SEMICOLON> + {return new EmptyStatement(token.sourceStart,token.sourceEnd);} } +/** + * used only by StatementExpressionList() which is used only by ForInit() and ForStatement() + */ Expression StatementExpression() : { - final Expression expr,expr2; - final int operator; + final Expression expr; + final Token operator; } { expr = PreIncDecExpression() {return expr;} | expr = PrimaryExpression() - [ <PLUS_PLUS> {return new PostfixedUnaryExpression(expr, - OperatorIds.PLUS_PLUS, - SimpleCharStream.getPosition());} - | <MINUS_MINUS> {return new PostfixedUnaryExpression(expr, - OperatorIds.MINUS_MINUS, - SimpleCharStream.getPosition());} - | operator = AssignmentOperator() expr2 = Expression() - {return new BinaryExpression(expr,expr2,operator);} + [ operator = <PLUS_PLUS> {return new PostfixedUnaryExpression(expr, + OperatorIds.PLUS_PLUS, + operator.sourceEnd);} + | operator = <MINUS_MINUS> {return new PostfixedUnaryExpression(expr, + OperatorIds.MINUS_MINUS, + operator.sourceEnd);} ] {return expr;} } SwitchStatement SwitchStatement() : { - final Expression variable; + Expression variable; final AbstractCase[] cases; - final int pos = SimpleCharStream.getPosition(); + final Token switchToken,lparenToken,rparenToken; + int pos; } { - <SWITCH> + switchToken = <SWITCH> {pos = switchToken.sourceEnd+1;} try { - <LPAREN> + lparenToken = <LPAREN> + {pos = lparenToken.sourceEnd+1;} } catch (ParseException e) { errorMessage = "'(' expected after 'switch'"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } try { - variable = Expression() + variable = Expression() {pos = variable.sourceEnd+1;} } catch (ParseException e) { if (errorMessage != null) { throw e; } errorMessage = "expression expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); + variable = new ConstantIdentifier(SYNTAX_ERROR_CHAR,pos,pos); } try { - <RPAREN> + rparenToken = <RPAREN> {pos = rparenToken.sourceEnd+1;} } catch (ParseException e) { errorMessage = "')' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } - (cases = switchStatementBrace() | cases = switchStatementColon(pos, pos + 6)) - {return new SwitchStatement(variable,cases,pos,SimpleCharStream.getPosition());} + ( cases = switchStatementBrace() + | cases = switchStatementColon(switchToken.sourceStart, switchToken.sourceEnd)) + {return new SwitchStatement(variable, + cases, + switchToken.sourceStart, + this.token.sourceEnd);} } AbstractCase[] switchStatementBrace() : { AbstractCase cas; final ArrayList cases = new ArrayList(); + Token token; + int pos; } { - <LBRACE> - ( cas = switchLabel0() {cases.add(cas);})* + token = <LBRACE> {pos = token.sourceEnd;} + ( cas = switchLabel0() {cases.add(cas);pos = cas.sourceEnd;})* try { - <RBRACE> - { - final AbstractCase[] abcase = new AbstractCase[cases.size()]; - cases.toArray(abcase); - return abcase;} + token = <RBRACE> + {pos = token.sourceEnd;} } catch (ParseException e) { errorMessage = "'}' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos+1; + errorEnd = pos+1; + processParseExceptionDebug(e); + } + { + final AbstractCase[] abcase = new AbstractCase[cases.size()]; + cases.toArray(abcase); + return abcase; } } + /** * A Switch statement with : ... endswitch; * @param start the begin offset of the switch @@ -2324,9 +2756,11 @@ AbstractCase[] switchStatementColon(final int start, final int end) : { AbstractCase cas; final ArrayList cases = new ArrayList(); + Token token; + int pos; } { - <COLON> + token = <COLON> {pos = token.sourceEnd;} {try { setMarker(fileToParse, "Ugly syntax detected, you should switch () {...} instead of switch (): ... enswitch;", @@ -2337,28 +2771,29 @@ AbstractCase[] switchStatementColon(final int start, final int end) : } catch (CoreException e) { PHPeclipsePlugin.log(e); }} - ( cas = switchLabel0() {cases.add(cas);})* + ( cas = switchLabel0() {cases.add(cas);pos = cas.sourceEnd;})* try { - <ENDSWITCH> + token = <ENDSWITCH> {pos = token.sourceEnd;} } catch (ParseException e) { errorMessage = "'endswitch' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos+1; + errorEnd = pos+1; + processParseExceptionDebug(e); } try { - <SEMICOLON> - { - final AbstractCase[] abcase = new AbstractCase[cases.size()]; - cases.toArray(abcase); - return abcase;} + token = <SEMICOLON> {pos = token.sourceEnd;} } catch (ParseException e) { errorMessage = "';' expected after 'endswitch' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos+1; + errorEnd = pos+1; + processParseExceptionDebug(e); + } + { + final AbstractCase[] abcase = new AbstractCase[cases.size()]; + cases.toArray(abcase); + return abcase; } } @@ -2367,20 +2802,29 @@ AbstractCase switchLabel0() : final Expression expr; Statement statement; final ArrayList stmts = new ArrayList(); - final int pos = SimpleCharStream.getPosition(); + final Token token = this.token; + final int start = this.token.next.sourceStart; } { expr = SwitchLabel() ( statement = BlockStatementNoBreak() {stmts.add(statement);} - | statement = htmlBlock() {stmts.add(statement);})* - [ statement = BreakStatement() {stmts.add(statement);}] + | statement = htmlBlock() {if (statement != null) {stmts.add(statement);}} + | statement = BreakStatement() {stmts.add(statement);})* + //[ statement = BreakStatement() {stmts.add(statement);}] { - final Statement[] stmtsArray = new Statement[stmts.size()]; - stmts.toArray(stmtsArray); - if (expr == null) {//it's a default - return new DefaultCase(stmtsArray,pos,SimpleCharStream.getPosition()); + final int listSize = stmts.size(); + final Statement[] stmtsArray = new Statement[listSize]; + stmts.toArray(stmtsArray); + if (expr == null) {//it's a default + final int end = this.token.next.sourceStart; + return new DefaultCase(stmtsArray,start,end); + } + if (listSize != 0) { + return new Case(expr,stmtsArray,expr.sourceStart,stmtsArray[listSize-1].sourceEnd); + } else { + return new Case(expr,stmtsArray,expr.sourceStart,expr.sourceEnd); + } } - return new Case(expr,stmtsArray,pos,SimpleCharStream.getPosition());} } /** @@ -2401,61 +2845,65 @@ Expression SwitchLabel() : if (errorMessage != null) throw e; errorMessage = "expression expected after 'case' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = token.sourceEnd +1; + errorEnd = token.sourceEnd +1; throw e; } try { - <COLON> - {return expr;} + token = <COLON> } catch (ParseException e) { errorMessage = "':' expected after case expression"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = expr.sourceEnd+1; + errorEnd = expr.sourceEnd+1; + processParseExceptionDebug(e); } + {return expr;} | token = <_DEFAULT> try { <COLON> - {return null;} } catch (ParseException e) { errorMessage = "':' expected after 'default' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = token.sourceEnd+1; + errorEnd = token.sourceEnd+1; + processParseExceptionDebug(e); } + {return null;} } Break BreakStatement() : { Expression expression = null; - final int start = SimpleCharStream.getPosition(); + final Token token, token2; + int pos; } { - <BREAK> [ expression = Expression() ] + token = <BREAK> {pos = token.sourceEnd+1;} + [ expression = Expression() {pos = expression.sourceEnd+1;}] try { - <SEMICOLON> + token2 = <SEMICOLON> + {pos = token2.sourceEnd;} } catch (ParseException e) { errorMessage = "';' expected after 'break' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } - {return new Break(expression, start, SimpleCharStream.getPosition());} + {return new Break(expression, token.sourceStart, pos);} } IfStatement IfStatement() : { - final int pos = SimpleCharStream.getPosition(); final Expression condition; final IfStatement ifStatement; + Token token; } { - <IF> condition = Condition("if") ifStatement = IfStatement0(condition, pos,pos+2) + token = <IF> condition = Condition("if") + ifStatement = IfStatement0(condition,token.sourceStart,token.sourceEnd) {return ifStatement;} } @@ -2470,9 +2918,9 @@ Expression Condition(final String keyword) : } catch (ParseException e) { errorMessage = "'(' expected after " + keyword + " keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length(); - errorEnd = errorStart +1; - processParseException(e); + errorStart = this.token.sourceEnd + 1; + errorEnd = this.token.sourceEnd + 1; + processParseExceptionDebug(e); } condition = Expression() try { @@ -2480,9 +2928,9 @@ Expression Condition(final String keyword) : } catch (ParseException e) { errorMessage = "')' expected after " + keyword + " keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - processParseException(e); + errorStart = condition.sourceEnd+1; + errorEnd = condition.sourceEnd+1; + processParseExceptionDebug(e); } {return condition;} } @@ -2497,15 +2945,15 @@ IfStatement IfStatement0(final Expression condition, final int start,final int e final ArrayList stmts; final ArrayList elseIfList = new ArrayList(); final ElseIf[] elseIfs; - int pos = SimpleCharStream.getPosition(); + int pos = jj_input_stream.getPosition(); final int endStatements; } { <COLON> {stmts = new ArrayList();} ( statement = Statement() {stmts.add(statement);} - | statement = htmlBlock() {stmts.add(statement);})* - {endStatements = SimpleCharStream.getPosition();} + | statement = htmlBlock() {if (statement != null) {stmts.add(statement);}})* + {endStatements = jj_input_stream.getPosition();} (elseifStatement = ElseIfStatementColon() {elseIfList.add(elseifStatement);})* [elseStatement = ElseStatementColon()] @@ -2524,8 +2972,8 @@ IfStatement IfStatement0(final Expression condition, final int start,final int e } catch (ParseException e) { errorMessage = "'endif' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } try { @@ -2533,8 +2981,8 @@ IfStatement IfStatement0(final Expression condition, final int start,final int e } catch (ParseException e) { errorMessage = "';' expected after 'endif' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } { @@ -2546,7 +2994,7 @@ IfStatement IfStatement0(final Expression condition, final int start,final int e elseIfs, elseStatement, pos, - SimpleCharStream.getPosition()); + jj_input_stream.getPosition()); } else { statementsArray = new Statement[stmts.size()]; stmts.toArray(statementsArray); @@ -2555,7 +3003,7 @@ IfStatement IfStatement0(final Expression condition, final int start,final int e elseIfs, elseStatement, pos, - SimpleCharStream.getPosition()); + jj_input_stream.getPosition()); } } @@ -2565,17 +3013,17 @@ IfStatement IfStatement0(final Expression condition, final int start,final int e [ LOOKAHEAD(1) <ELSE> try { - {pos = SimpleCharStream.getPosition();} + {pos = jj_input_stream.getPosition();} statement = Statement() - {elseStatement = new Else(statement,pos,SimpleCharStream.getPosition());} + {elseStatement = new Else(statement,pos,jj_input_stream.getPosition());} } catch (ParseException e) { if (errorMessage != null) { throw e; } errorMessage = "unexpected token '"+e.currentToken.next.image+"', a statement was expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } ] @@ -2587,7 +3035,7 @@ IfStatement IfStatement0(final Expression condition, final int start,final int e elseIfs, elseStatement, pos, - SimpleCharStream.getPosition());} + jj_input_stream.getPosition());} } ElseIf ElseIfStatementColon() : @@ -2595,66 +3043,68 @@ ElseIf ElseIfStatementColon() : final Expression condition; Statement statement; final ArrayList list = new ArrayList(); - final int pos = SimpleCharStream.getPosition(); + final Token elseifToken; } { - <ELSEIF> condition = Condition("elseif") + elseifToken = <ELSEIF> condition = Condition("elseif") <COLON> ( statement = Statement() {list.add(statement);} - | statement = htmlBlock() {list.add(statement);})* + | statement = htmlBlock() {if (statement != null) {list.add(statement);}})* { - final Statement[] stmtsArray = new Statement[list.size()]; + final int sizeList = list.size(); + final Statement[] stmtsArray = new Statement[sizeList]; list.toArray(stmtsArray); - return new ElseIf(condition,stmtsArray ,pos,SimpleCharStream.getPosition());} + return new ElseIf(condition,stmtsArray , + elseifToken.sourceStart, + stmtsArray[sizeList-1].sourceEnd);} } Else ElseStatementColon() : { Statement statement; final ArrayList list = new ArrayList(); - final int pos = SimpleCharStream.getPosition(); + final Token elseToken; } { - <ELSE> <COLON> ( statement = Statement() {list.add(statement);} - | statement = htmlBlock() {list.add(statement);})* + elseToken = <ELSE> <COLON> ( statement = Statement() {list.add(statement);} + | statement = htmlBlock() {if (statement != null) {list.add(statement);}})* { - final Statement[] stmtsArray = new Statement[list.size()]; + final int sizeList = list.size(); + final Statement[] stmtsArray = new Statement[sizeList]; list.toArray(stmtsArray); - return new Else(stmtsArray,pos,SimpleCharStream.getPosition());} + return new Else(stmtsArray,elseToken.sourceStart,stmtsArray[sizeList-1].sourceEnd);} } ElseIf ElseIfStatement() : { final Expression condition; - final Statement statement; - final ArrayList list = new ArrayList(); - final int pos = SimpleCharStream.getPosition(); + //final Statement statement; + final Token elseifToken; + final Statement[] statement = new Statement[1]; } { - <ELSEIF> condition = Condition("elseif") statement = Statement() {list.add(statement);/*todo:do better*/} + elseifToken = <ELSEIF> condition = Condition("elseif") statement[0] = Statement() { - final Statement[] stmtsArray = new Statement[list.size()]; - list.toArray(stmtsArray); - return new ElseIf(condition,stmtsArray,pos,SimpleCharStream.getPosition());} + return new ElseIf(condition,statement,elseifToken.sourceStart,statement[0].sourceEnd);} } WhileStatement WhileStatement() : { final Expression condition; final Statement action; - final int pos = SimpleCharStream.getPosition(); + final Token whileToken; } { - <WHILE> + whileToken = <WHILE> condition = Condition("while") - action = WhileStatement0(pos,pos + 5) - {return new WhileStatement(condition,action,pos,SimpleCharStream.getPosition());} + action = WhileStatement0(whileToken.sourceStart,whileToken.sourceEnd) + {return new WhileStatement(condition,action,whileToken.sourceStart,action.sourceEnd);} } Statement WhileStatement0(final int start, final int end) : { Statement statement; final ArrayList stmts = new ArrayList(); - final int pos = SimpleCharStream.getPosition(); + final int pos = jj_input_stream.getPosition(); } { <COLON> (statement = Statement() {stmts.add(statement);})* @@ -2673,8 +3123,8 @@ Statement WhileStatement0(final int start, final int end) : } catch (ParseException e) { errorMessage = "'endwhile' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } try { @@ -2682,12 +3132,12 @@ Statement WhileStatement0(final int start, final int end) : { final Statement[] stmtsArray = new Statement[stmts.size()]; stmts.toArray(stmtsArray); - return new Block(stmtsArray,pos,SimpleCharStream.getPosition());} + return new Block(stmtsArray,pos,jj_input_stream.getPosition());} } catch (ParseException e) { errorMessage = "';' expected after 'endwhile' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; throw e; } | @@ -2699,104 +3149,126 @@ DoStatement DoStatement() : { final Statement action; final Expression condition; - final int pos = SimpleCharStream.getPosition(); + final Token token; + Token token2 = null; } { - <DO> action = Statement() <WHILE> condition = Condition("while") + token = <DO> action = Statement() <WHILE> condition = Condition("while") try { - <SEMICOLON> - {return new DoStatement(condition,action,pos,SimpleCharStream.getPosition());} + token2 = <SEMICOLON> } catch (ParseException e) { errorMessage = "unexpected token : '"+ e.currentToken.next.image +"'. A ';' was expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = condition.sourceEnd+1; + errorEnd = condition.sourceEnd+1; + processParseExceptionDebug(e); + } + { + if (token2 == null) { + return new DoStatement(condition,action,token.sourceStart,condition.sourceEnd); + } + return new DoStatement(condition,action,token.sourceStart,token2.sourceEnd); } } ForeachStatement ForeachStatement() : { - Statement statement; - Expression expression; - final int pos = SimpleCharStream.getPosition(); - ArrayVariableDeclaration variable; + Statement statement = null; + Expression expression = null; + ArrayVariableDeclaration variable = null; + Token foreachToken; + Token lparenToken = null; + Token asToken = null; + Token rparenToken = null; + int pos; } { - <FOREACH> - try { - <LPAREN> + foreachToken = <FOREACH> + try { + lparenToken = <LPAREN> + {pos = lparenToken.sourceEnd+1;} } catch (ParseException e) { errorMessage = "'(' expected after 'foreach' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); + {pos = foreachToken.sourceEnd+1;} } try { expression = Expression() + {pos = expression.sourceEnd+1;} } catch (ParseException e) { errorMessage = "variable expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } try { - <AS> + asToken = <AS> + {pos = asToken.sourceEnd+1;} } catch (ParseException e) { errorMessage = "'as' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } try { variable = ArrayVariable() + {pos = variable.sourceEnd+1;} } catch (ParseException e) { + if (errorMessage != null) throw e; errorMessage = "variable expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } try { - <RPAREN> + rparenToken = <RPAREN> + {pos = rparenToken.sourceEnd+1;} } catch (ParseException e) { errorMessage = "')' expected after 'foreach' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } try { statement = Statement() + {pos = statement.sourceEnd+1;} } catch (ParseException e) { if (errorMessage != null) throw e; errorMessage = "statement expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = e.currentToken.sourceStart; + errorEnd = e.currentToken.sourceEnd; + processParseExceptionDebug(e); } - {return new ForeachStatement(expression, + { + return new ForeachStatement(expression, variable, statement, - pos, - SimpleCharStream.getPosition());} + foreachToken.sourceStart, + pos);} } +/** + * a for declaration. + * @return a node representing the for statement + */ ForStatement ForStatement() : { -final Token token; -final int pos = SimpleCharStream.getPosition(); +final Token token,tokenEndFor,token2,tokenColon; +int pos; Expression[] initializations = null; Expression condition = null; Expression[] increments = null; Statement action; final ArrayList list = new ArrayList(); -final int startBlock, endBlock; } { token = <FOR> @@ -2805,55 +3277,67 @@ final int startBlock, endBlock; } catch (ParseException e) { errorMessage = "'(' expected after 'for' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = token.sourceEnd; + errorEnd = token.sourceEnd +1; + processParseExceptionDebug(e); } [ initializations = ForInit() ] <SEMICOLON> [ condition = Expression() ] <SEMICOLON> [ increments = StatementExpressionList() ] <RPAREN> ( action = Statement() - {return new ForStatement(initializations,condition,increments,action,pos,SimpleCharStream.getPosition());} + {return new ForStatement(initializations, + condition, + increments, + action, + token.sourceStart, + action.sourceEnd);} | - <COLON> - {startBlock = SimpleCharStream.getPosition();} - (action = Statement() {list.add(action);})* + tokenColon = <COLON> {pos = tokenColon.sourceEnd+1;} + (action = Statement() {list.add(action);pos = action.sourceEnd+1;})* { try { setMarker(fileToParse, "Ugly syntax detected, you should for () {...} instead of for (): ... endfor;", - pos, - pos+token.image.length(), + token.sourceStart, + token.sourceEnd, INFO, "Line " + token.beginLine); } catch (CoreException e) { PHPeclipsePlugin.log(e); } } - {endBlock = SimpleCharStream.getPosition();} try { - <ENDFOR> + tokenEndFor = <ENDFOR> + {pos = tokenEndFor.sourceEnd+1;} } catch (ParseException e) { errorMessage = "'endfor' expected"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } try { - <SEMICOLON> - { - final Statement[] stmtsArray = new Statement[list.size()]; - list.toArray(stmtsArray); - return new ForStatement(initializations,condition,increments,new Block(stmtsArray,startBlock,endBlock),pos,SimpleCharStream.getPosition());} + token2 = <SEMICOLON> + {pos = token2.sourceEnd+1;} } catch (ParseException e) { errorMessage = "';' expected after 'endfor' keyword"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + errorStart = pos; + errorEnd = pos; + processParseExceptionDebug(e); } + { + final Statement[] stmtsArray = new Statement[list.size()]; + list.toArray(stmtsArray); + return new ForStatement(initializations, + condition, + increments, + new Block(stmtsArray, + stmtsArray[0].sourceStart, + stmtsArray[stmtsArray.length-1].sourceEnd), + token.sourceStart, + pos);} ) } @@ -2876,48 +3360,78 @@ Expression[] StatementExpressionList() : final Expression expr; } { - expr = StatementExpression() {list.add(expr);} - (<COMMA> StatementExpression() {list.add(expr);})* + expr = Expression() {list.add(expr);} + (<COMMA> Expression() {list.add(expr);})* { - final Expression[] exprsArray = new Expression[list.size()]; - list.toArray(exprsArray); - return exprsArray;} + final Expression[] exprsArray = new Expression[list.size()]; + list.toArray(exprsArray); + return exprsArray; + } } Continue ContinueStatement() : { Expression expr = null; - final int pos = SimpleCharStream.getPosition(); + final Token token; + Token token2 = null; } { - <CONTINUE> [ expr = Expression() ] + token = <CONTINUE> [ expr = Expression() ] try { - <SEMICOLON> - {return new Continue(expr,pos,SimpleCharStream.getPosition());} + token2 = <SEMICOLON> } catch (ParseException e) { errorMessage = "';' expected after 'continue' statement"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + if (expr == null) { + errorStart = token.sourceEnd+1; + errorEnd = token.sourceEnd+1; + } else { + errorStart = expr.sourceEnd+1; + errorEnd = expr.sourceEnd+1; + } + processParseExceptionDebug(e); + } + { + if (token2 == null) { + if (expr == null) { + return new Continue(expr,token.sourceStart,token.sourceEnd); + } + return new Continue(expr,token.sourceStart,expr.sourceEnd); + } + return new Continue(expr,token.sourceStart,token2.sourceEnd); } } ReturnStatement ReturnStatement() : { Expression expr = null; - final int pos = SimpleCharStream.getPosition(); + final Token token; + Token token2 = null; } { - <RETURN> [ expr = Expression() ] + token = <RETURN> [ expr = Expression() ] try { - <SEMICOLON> - {return new ReturnStatement(expr,pos,SimpleCharStream.getPosition());} + token2 = <SEMICOLON> } catch (ParseException e) { errorMessage = "';' expected after 'return' statement"; errorLevel = ERROR; - errorStart = SimpleCharStream.getPosition() - e.currentToken.next.image.length() + 1; - errorEnd = SimpleCharStream.getPosition() + 1; - throw e; + if (expr == null) { + errorStart = token.sourceEnd+1; + errorEnd = token.sourceEnd+1; + } else { + errorStart = expr.sourceEnd+1; + errorEnd = expr.sourceEnd+1; + } + processParseExceptionDebug(e); + } + { + if (token2 == null) { + if (expr == null) { + return new ReturnStatement(expr,token.sourceStart,token.sourceEnd); + } + return new ReturnStatement(expr,token.sourceStart,expr.sourceEnd); + } + return new ReturnStatement(expr,token.sourceStart,token2.sourceEnd); } -} \ No newline at end of file +} +