From a9b421dd421a0bcfe48c0d34e3836be4d29d66e9 Mon Sep 17 00:00:00 2001 From: Attila Szegedi Date: Tue, 8 Jul 2014 13:13:31 +0200 Subject: [PATCH] 8047067: all eval arguments need to be copied in Lower Reviewed-by: lagergren, sundar --- .../internal/codegen/CodeGenerator.java | 23 ++++---- .../jdk/nashorn/internal/codegen/Lower.java | 13 ++--- .../src/jdk/nashorn/internal/ir/CallNode.java | 57 ++++--------------- .../jdk/nashorn/internal/objects/Global.java | 6 +- nashorn/test/script/basic/JDK-8047057.js | 12 ++-- nashorn/test/script/basic/JDK-8047067.js | 36 ++++++++++++ 6 files changed, 72 insertions(+), 75 deletions(-) create mode 100644 nashorn/test/script/basic/JDK-8047067.js diff --git a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java index c890208879c..46a6b86bfd3 100644 --- a/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java +++ b/nashorn/src/jdk/nashorn/internal/codegen/CodeGenerator.java @@ -1317,20 +1317,14 @@ final class CodeGenerator extends NodeOperatorVisitor evalArgs = callNode.getEvalArgs().getArgs(); // load evaluated code - loadExpressionAsObject(evalArgs.getCode()); + loadExpressionAsObject(evalArgs.get(0)); // load second and subsequent args for side-effect - final List callArgs = callNode.getArgs(); - final int numArgs = callArgs.size(); + final int numArgs = evalArgs.size(); for (int i = 1; i < numArgs; i++) { - loadExpressionUnbounded(callArgs.get(i)).pop(); + loadAndDiscard(evalArgs.get(i)); } - // special/extra 'eval' arguments - loadExpressionUnbounded(evalArgs.getThis()); - method.load(evalArgs.getLocation()); - method.load(evalArgs.getStrictMode()); - method.convert(Type.OBJECT); method._goto(invoke_direct_eval); method.label(is_not_eval); @@ -1339,7 +1333,7 @@ final class CodeGenerator extends NodeOperatorVisitor implements Lo // 'eval' call with at least one argument if (args.size() >= 1 && EVAL.symbolName().equals(callee.getName())) { - final FunctionNode currentFunction = lc.getCurrentFunction(); - return callNode.setEvalArgs( - new CallNode.EvalArgs( - (Expression)ensureUniqueNamesIn(args.get(0)).accept(this), - compilerConstant(THIS), - evalLocation(callee), - currentFunction.isStrict())); + final List evalArgs = new ArrayList<>(args.size()); + for(final Expression arg: args) { + evalArgs.add((Expression)ensureUniqueNamesIn(arg).accept(this)); + } + return callNode.setEvalArgs(new CallNode.EvalArgs(evalArgs, evalLocation(callee))); } } diff --git a/nashorn/src/jdk/nashorn/internal/ir/CallNode.java b/nashorn/src/jdk/nashorn/internal/ir/CallNode.java index d8bfd47adf3..0751ce4087d 100644 --- a/nashorn/src/jdk/nashorn/internal/ir/CallNode.java +++ b/nashorn/src/jdk/nashorn/internal/ir/CallNode.java @@ -65,61 +65,35 @@ public final class CallNode extends LexicalContextExpression implements Optimist * Arguments to be passed to builtin {@code eval} function */ public static class EvalArgs { - /** evaluated code */ - private final Expression code; - - /** 'this' passed to evaluated code */ - private final IdentNode evalThis; + private final List args; /** location string for the eval call */ private final String location; - /** is this call from a strict context? */ - private final boolean strictMode; - /** * Constructor * - * @param code code to evaluate - * @param evalThis this node - * @param location location for the eval call - * @param strictMode is this a call from a strict context? + * @param args arguments to eval + * @param location location for the eval call */ - public EvalArgs(final Expression code, final IdentNode evalThis, final String location, final boolean strictMode) { - this.code = code; - this.evalThis = evalThis; + public EvalArgs(final List args, final String location) { + this.args = args; this.location = location; - this.strictMode = strictMode; } /** * Return the code that is to be eval:ed by this eval function * @return code as an AST node */ - public Expression getCode() { - return code; + public List getArgs() { + return Collections.unmodifiableList(args); } - private EvalArgs setCode(final Expression code) { - if (this.code == code) { + private EvalArgs setArgs(final List args) { + if (this.args == args) { return this; } - return new EvalArgs(code, evalThis, location, strictMode); - } - - /** - * Get the {@code this} symbol used to invoke this eval call - * @return the {@code this} symbol - */ - public IdentNode getThis() { - return this.evalThis; - } - - private EvalArgs setThis(final IdentNode evalThis) { - if (this.evalThis == evalThis) { - return this; - } - return new EvalArgs(code, evalThis, location, strictMode); + return new EvalArgs(args, location); } /** @@ -129,14 +103,6 @@ public final class CallNode extends LexicalContextExpression implements Optimist public String getLocation() { return this.location; } - - /** - * Check whether this eval call is executed in strict mode - * @return true if executed in strict mode, false otherwise - */ - public boolean getStrictMode() { - return this.strictMode; - } } /** arguments for 'eval' call. Non-null only if this call node is 'eval' */ @@ -212,8 +178,7 @@ public final class CallNode extends LexicalContextExpression implements Optimist setArgs(Node.accept(visitor, Expression.class, args)). setEvalArgs(evalArgs == null ? null : - evalArgs.setCode((Expression)evalArgs.getCode().accept(visitor)). - setThis((IdentNode)evalArgs.getThis().accept(visitor)))); + evalArgs.setArgs(Node.accept(visitor, Expression.class, evalArgs.getArgs())))); // Theoretically, we'd need to instead pass lc to every setter and do a replacement on each. In practice, // setType from TypeOverride can't accept a lc, and we don't necessarily want to go there now. if (this != newCallNode) { diff --git a/nashorn/src/jdk/nashorn/internal/objects/Global.java b/nashorn/src/jdk/nashorn/internal/objects/Global.java index d9219211109..5d28c1efd74 100644 --- a/nashorn/src/jdk/nashorn/internal/objects/Global.java +++ b/nashorn/src/jdk/nashorn/internal/objects/Global.java @@ -908,7 +908,7 @@ public final class Global extends ScriptObject implements Scope { * @return the result of eval */ public static Object eval(final Object self, final Object str) { - return directEval(self, str, UNDEFINED, UNDEFINED, UNDEFINED); + return directEval(self, str, UNDEFINED, UNDEFINED, false); } /** @@ -924,14 +924,14 @@ public final class Global extends ScriptObject implements Scope { * * This is directly invoked from generated when eval(code) is called in user code */ - public static Object directEval(final Object self, final Object str, final Object callThis, final Object location, final Object strict) { + public static Object directEval(final Object self, final Object str, final Object callThis, final Object location, final boolean strict) { if (!(str instanceof String || str instanceof ConsString)) { return str; } final Global global = Global.instanceFrom(self); final ScriptObject scope = self instanceof ScriptObject ? (ScriptObject)self : global; - return global.getContext().eval(scope, str.toString(), callThis, location, Boolean.TRUE.equals(strict), true); + return global.getContext().eval(scope, str.toString(), callThis, location, strict, true); } /** diff --git a/nashorn/test/script/basic/JDK-8047057.js b/nashorn/test/script/basic/JDK-8047057.js index f13e36e4941..bd38ee901e3 100644 --- a/nashorn/test/script/basic/JDK-8047057.js +++ b/nashorn/test/script/basic/JDK-8047057.js @@ -49,7 +49,7 @@ makeFuncAndCall("switch(0) { default: {break;} return }"); makeFuncAndCall("L: { { break L; } return; }"); makeFuncAndCall("L: { while(0) break L; return; }"); makeFuncExpectError("L: {while(0) break L; return [](); }", TypeError); -// makeFuncAndCall("do with({}) break ; while(0);"); +makeFuncAndCall("do with({}) break ; while(0);"); makeFuncAndCall("while(0) with({}) continue ;"); makeFuncAndCall("eval([]);"); makeFuncAndCall("try{} finally{[]}"); @@ -59,10 +59,10 @@ makeFuncAndCall("Error() * (false)[-0]--"); makeFuncAndCall("try { var x = 1, x = null; } finally { }"); makeFuncAndCall("try { var x = {}, x = []; } catch(x3) { }"); makeFuncAndCall("[delete this]"); -// makeFuncAndCall("if(eval('', eval('', function() {}))) { }"); -// makeFuncAndCall("if(eval('', eval('', function() {}))) { }"); -// makeFuncAndCall("eval(\"[,,];\", [11,12,13,14].some)"); -// makeFuncAndCall("eval(\"1.2e3\", ({})[ /x/ ])"); +makeFuncAndCall("if(eval('', eval('', function() {}))) { }"); +makeFuncAndCall("if(eval('', eval('', function() {}))) { }"); +makeFuncAndCall("eval(\"[,,];\", [11,12,13,14].some)"); +makeFuncAndCall("eval(\"1.2e3\", ({})[ /x/ ])"); makeFuncExpectError("eval(\"x4\", x3);", ReferenceError); makeFuncAndCall("with({5.0000000000000000000000: String()}){(false); }"); makeFuncAndCall("try { var x = undefined, x = 5.0000000000000000000000; } catch(x) { x = undefined; }"); @@ -72,4 +72,4 @@ makeFuncAndCall("(false % !this) && 0"); makeFuncAndCall("with({8: 'fafafa'.replace()}){ }"); makeFuncAndCall("(function (x) '' )(true)"); makeFuncExpectError("new eval(function(){})", TypeError); -//** makeFuncAndCall('eval("23", ({})[/x/])'); +makeFuncAndCall('eval("23", ({})[/x/])'); diff --git a/nashorn/test/script/basic/JDK-8047067.js b/nashorn/test/script/basic/JDK-8047067.js new file mode 100644 index 00000000000..2b68611290c --- /dev/null +++ b/nashorn/test/script/basic/JDK-8047067.js @@ -0,0 +1,36 @@ +/* + * Copyright (c) 2010, 2014, 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-8047067: all eval arguments need to be copied in Lower + * + * @test + * @run + */ + +// The second expression triggers optimistic deoptimization, and if not +// all eval arguments were copied in Lower, we'd end up with duplicate +// program points that'd cause incorrect continuation program point in +// the rest-of, and therefore a bad stack, and therefore an AIOOBE in +// the continuation setup code. +eval("23", ({})[/x/])