8075207: Nashorn parser API returns StatementTree objects in out of order
Reviewed-by: lagergren, forax, hannesw
This commit is contained in:
parent
4c6eb2b310
commit
cf37646b3f
@ -52,7 +52,6 @@ var Parser = Java.type("jdk.nashorn.api.tree.Parser");
|
|||||||
var SimpleTreeVisitor = Java.type("jdk.nashorn.api.tree.SimpleTreeVisitorES5_1");
|
var SimpleTreeVisitor = Java.type("jdk.nashorn.api.tree.SimpleTreeVisitorES5_1");
|
||||||
|
|
||||||
var parser = Parser.create("-scripting", "--const-as-var");
|
var parser = Parser.create("-scripting", "--const-as-var");
|
||||||
var protoFound = false;
|
|
||||||
|
|
||||||
function checkFile(file) {
|
function checkFile(file) {
|
||||||
// print("checking " + file);
|
// print("checking " + file);
|
||||||
@ -92,7 +91,3 @@ if (file.isDirectory()) {
|
|||||||
} else {
|
} else {
|
||||||
checkFile(file);
|
checkFile(file);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (protoFound) {
|
|
||||||
print("__proto__ is non-standard. Use Object.get/setPrototypeOf instead");
|
|
||||||
}
|
|
||||||
|
@ -44,6 +44,7 @@ public interface DoWhileLoopTree extends ConditionalLoopTree {
|
|||||||
*
|
*
|
||||||
* @return the condition expression
|
* @return the condition expression
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
ExpressionTree getCondition();
|
ExpressionTree getCondition();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -51,5 +52,6 @@ public interface DoWhileLoopTree extends ConditionalLoopTree {
|
|||||||
*
|
*
|
||||||
* @return the statement
|
* @return the statement
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
StatementTree getStatement();
|
StatementTree getStatement();
|
||||||
}
|
}
|
||||||
|
@ -57,6 +57,7 @@ public interface ForInLoopTree extends LoopTree {
|
|||||||
*
|
*
|
||||||
* @return the statement
|
* @return the statement
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
StatementTree getStatement();
|
StatementTree getStatement();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -25,6 +25,7 @@
|
|||||||
package jdk.nashorn.api.tree;
|
package jdk.nashorn.api.tree;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Comparator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import jdk.nashorn.internal.ir.AccessNode;
|
import jdk.nashorn.internal.ir.AccessNode;
|
||||||
import jdk.nashorn.internal.ir.BinaryNode;
|
import jdk.nashorn.internal.ir.BinaryNode;
|
||||||
@ -92,7 +93,7 @@ final class IRTranslator extends NodeVisitor<LexicalContext> {
|
|||||||
|
|
||||||
final Block body = node.getBody();
|
final Block body = node.getBody();
|
||||||
return new CompilationUnitTreeImpl(node,
|
return new CompilationUnitTreeImpl(node,
|
||||||
translateStats(body != null? body.getStatements() : null));
|
translateStats(body != null? getOrderedStatements(body.getStatements()) : null));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -103,25 +104,7 @@ final class IRTranslator extends NodeVisitor<LexicalContext> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean enterBlock(final Block block) {
|
public boolean enterBlock(final Block block) {
|
||||||
// FIXME: revisit this!
|
return handleBlock(block, false);
|
||||||
if (block.isSynthetic()) {
|
|
||||||
final int statCount = block.getStatementCount();
|
|
||||||
switch (statCount) {
|
|
||||||
case 0: {
|
|
||||||
final EmptyNode emptyNode = new EmptyNode(-1, block.getToken(), block.getFinish());
|
|
||||||
curStat = new EmptyStatementTreeImpl(emptyNode);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
case 1: {
|
|
||||||
curStat = translateStat(block.getStatements().get(0));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
curStat = new BlockTreeImpl(block,
|
|
||||||
translateStats(block.getStatements()));
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -245,7 +228,7 @@ final class IRTranslator extends NodeVisitor<LexicalContext> {
|
|||||||
|
|
||||||
final List<? extends ExpressionTree> paramTrees
|
final List<? extends ExpressionTree> paramTrees
|
||||||
= translateExprs(functionNode.getParameters());
|
= translateExprs(functionNode.getParameters());
|
||||||
final BlockTree blockTree = (BlockTree) translateBlock(functionNode.getBody());
|
final BlockTree blockTree = (BlockTree) translateBlock(functionNode.getBody(), true);
|
||||||
curExpr = new FunctionExpressionTreeImpl(functionNode, paramTrees, blockTree);
|
curExpr = new FunctionExpressionTreeImpl(functionNode, paramTrees, blockTree);
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
@ -420,7 +403,7 @@ final class IRTranslator extends NodeVisitor<LexicalContext> {
|
|||||||
|
|
||||||
final List<? extends ExpressionTree> paramTrees
|
final List<? extends ExpressionTree> paramTrees
|
||||||
= translateExprs(funcNode.getParameters());
|
= translateExprs(funcNode.getParameters());
|
||||||
final BlockTree blockTree = (BlockTree) translateBlock(funcNode.getBody());
|
final BlockTree blockTree = (BlockTree) translateBlock(funcNode.getBody(), true);
|
||||||
curStat = new FunctionDeclarationTreeImpl(varNode, paramTrees, blockTree);
|
curStat = new FunctionDeclarationTreeImpl(varNode, paramTrees, blockTree);
|
||||||
} else {
|
} else {
|
||||||
curStat = new VariableTreeImpl(varNode, translateExpr(initNode));
|
curStat = new VariableTreeImpl(varNode, translateExpr(initNode));
|
||||||
@ -453,14 +436,51 @@ final class IRTranslator extends NodeVisitor<LexicalContext> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private StatementTree translateBlock(final Block blockNode) {
|
private StatementTree translateBlock(final Block blockNode) {
|
||||||
|
return translateBlock(blockNode, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private StatementTree translateBlock(final Block blockNode, final boolean sortStats) {
|
||||||
if (blockNode == null) {
|
if (blockNode == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
curStat = null;
|
curStat = null;
|
||||||
blockNode.accept(this);
|
handleBlock(blockNode, sortStats);
|
||||||
return curStat;
|
return curStat;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean handleBlock(final Block block, final boolean sortStats) {
|
||||||
|
// FIXME: revisit this!
|
||||||
|
if (block.isSynthetic()) {
|
||||||
|
final int statCount = block.getStatementCount();
|
||||||
|
switch (statCount) {
|
||||||
|
case 0: {
|
||||||
|
final EmptyNode emptyNode = new EmptyNode(-1, block.getToken(), block.getFinish());
|
||||||
|
curStat = new EmptyStatementTreeImpl(emptyNode);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
case 1: {
|
||||||
|
curStat = translateStat(block.getStatements().get(0));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
// fall through
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
final List<? extends Statement> stats = block.getStatements();
|
||||||
|
curStat = new BlockTreeImpl(block,
|
||||||
|
translateStats(sortStats? getOrderedStatements(stats) : stats));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<? extends Statement> getOrderedStatements(final List<? extends Statement> stats) {
|
||||||
|
final List<? extends Statement> statList = new ArrayList<>(stats);
|
||||||
|
statList.sort(Comparator.comparingInt(Node::getStart));
|
||||||
|
return statList;
|
||||||
|
}
|
||||||
|
|
||||||
private List<? extends StatementTree> translateStats(final List<? extends Statement> stats) {
|
private List<? extends StatementTree> translateStats(final List<? extends Statement> stats) {
|
||||||
if (stats == null) {
|
if (stats == null) {
|
||||||
return null;
|
return null;
|
||||||
@ -511,7 +531,7 @@ final class IRTranslator extends NodeVisitor<LexicalContext> {
|
|||||||
return curStat;
|
return curStat;
|
||||||
}
|
}
|
||||||
|
|
||||||
private IdentifierTree translateIdent(final IdentNode ident) {
|
private static IdentifierTree translateIdent(final IdentNode ident) {
|
||||||
return new IdentifierTreeImpl(ident);
|
return new IdentifierTreeImpl(ident);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -139,7 +139,7 @@ final class ParserImpl implements Parser {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private CompilationUnitTree translate(final FunctionNode node) {
|
private static CompilationUnitTree translate(final FunctionNode node) {
|
||||||
return new IRTranslator().translate(node);
|
return new IRTranslator().translate(node);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -43,6 +43,7 @@ public interface WhileLoopTree extends ConditionalLoopTree {
|
|||||||
*
|
*
|
||||||
* @return the condition expression
|
* @return the condition expression
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
ExpressionTree getCondition();
|
ExpressionTree getCondition();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -50,5 +51,6 @@ public interface WhileLoopTree extends ConditionalLoopTree {
|
|||||||
*
|
*
|
||||||
* @return the statement contained
|
* @return the statement contained
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
StatementTree getStatement();
|
StatementTree getStatement();
|
||||||
}
|
}
|
||||||
|
76
nashorn/test/script/basic/JDK-8075207.js
Normal file
76
nashorn/test/script/basic/JDK-8075207.js
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JDK-8075207: Nashorn parser API returns StatementTree objects in out of order
|
||||||
|
*
|
||||||
|
* @test
|
||||||
|
* @option -scripting
|
||||||
|
* @run
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Parser = Java.type("jdk.nashorn.api.tree.Parser");
|
||||||
|
var ExpressionStatementTree = Java.type("jdk.nashorn.api.tree.ExpressionStatementTree");
|
||||||
|
var FunctionDeclarationTree = Java.type("jdk.nashorn.api.tree.FunctionDeclarationTree");
|
||||||
|
var VariableTree = Java.type("jdk.nashorn.api.tree.VariableTree");
|
||||||
|
|
||||||
|
var parser = Parser.create();
|
||||||
|
|
||||||
|
var ast = parser.parse("hello.js", <<CODE
|
||||||
|
|
||||||
|
var hello = 'hello';
|
||||||
|
|
||||||
|
function print_hello() {
|
||||||
|
var x = 2;
|
||||||
|
print(hello);
|
||||||
|
function inner_func() {}
|
||||||
|
var y = function() {
|
||||||
|
var PI = Math.PI;
|
||||||
|
function inner2() {}
|
||||||
|
var E = Math.E;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var hello = "hello 2";
|
||||||
|
|
||||||
|
CODE, print);
|
||||||
|
|
||||||
|
var stats = ast.sourceElements;
|
||||||
|
Assert.assertTrue(stats.get(0) instanceof VariableTree);
|
||||||
|
Assert.assertTrue(stats.get(1) instanceof FunctionDeclarationTree);
|
||||||
|
Assert.assertTrue(stats.get(2) instanceof VariableTree);
|
||||||
|
|
||||||
|
var print_hello = stats.get(1);
|
||||||
|
Assert.assertEquals(print_hello.name, "print_hello");
|
||||||
|
var print_hello_stats = print_hello.body.statements;
|
||||||
|
Assert.assertTrue(print_hello_stats.get(0) instanceof VariableTree);
|
||||||
|
Assert.assertTrue(print_hello_stats.get(1) instanceof ExpressionStatementTree);
|
||||||
|
Assert.assertTrue(print_hello_stats.get(2) instanceof FunctionDeclarationTree);
|
||||||
|
Assert.assertTrue(print_hello_stats.get(3) instanceof VariableTree);
|
||||||
|
|
||||||
|
var anonFunc = print_hello_stats.get(3).initializer;
|
||||||
|
var anonFunc_stats = anonFunc.body.statements;
|
||||||
|
Assert.assertTrue(anonFunc_stats.get(0) instanceof VariableTree);
|
||||||
|
Assert.assertTrue(anonFunc_stats.get(1) instanceof FunctionDeclarationTree);
|
||||||
|
Assert.assertTrue(anonFunc_stats.get(2) instanceof VariableTree);
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
Loading…
x
Reference in New Issue
Block a user