8131929: Add option for debuggable scopes

Reviewed-by: attila, lagergren
This commit is contained in:
Hannes Wallnöfer 2015-11-18 17:12:47 +01:00
parent f65591ae29
commit 9e8fa2b8c2
14 changed files with 209 additions and 50 deletions

View File

@ -550,7 +550,10 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
if (swap) {
method.swap();
}
for (int i = 0; i < depth; i++) {
if (depth > 1) {
method.load(depth);
method.invoke(ScriptObject.GET_PROTO_DEPTH);
} else {
method.invoke(ScriptObject.GET_PROTO);
}
if (swap) {
@ -1379,7 +1382,10 @@ final class CodeGenerator extends NodeOperatorVisitor<CodeGeneratorLexicalContex
return;
}
method.loadCompilerConstant(SCOPE);
for(int i = 0; i < count; ++i) {
if (count > 1) {
method.load(count);
method.invoke(ScriptObject.GET_PROTO_DEPTH);
} else {
method.invoke(ScriptObject.GET_PROTO);
}
method.storeCompilerConstant(SCOPE);

View File

@ -239,7 +239,7 @@ public final class FunctionNode extends LexicalContextExpression implements Flag
private static final int HAS_DEEP_EVAL = HAS_EVAL | HAS_NESTED_EVAL;
/** Does this function need to store all its variables in scope? */
private static final int HAS_ALL_VARS_IN_SCOPE = HAS_DEEP_EVAL;
public static final int HAS_ALL_VARS_IN_SCOPE = HAS_DEEP_EVAL;
/** Does this function potentially need "arguments"? Note that this is not a full test, as further negative check of REDEFINES_ARGS is needed. */
private static final int MAYBE_NEEDS_ARGUMENTS = USES_ARGUMENTS | HAS_EVAL;

View File

@ -223,6 +223,18 @@ public final class NativeDebug extends ScriptObject {
return obj.getClass() + "@" + Integer.toHexString(hash);
}
/**
* Returns {@code true} if passed object is a function that is fully debuggable (has all vars in scope).
*
* @param self self reference
* @param obj object
* @return true {@code obj} is a debuggable function
*/
@Function(attributes = Attribute.NOT_ENUMERABLE, where = Where.CONSTRUCTOR)
public static Object isDebuggableFunction(final Object self, final Object obj) {
return (obj instanceof ScriptFunction && ((ScriptFunction) obj).hasAllVarsInScope());
}
/**
* Returns the property listener count for a script object
*

View File

@ -2941,6 +2941,10 @@ loop:
try {
// Create a new function block.
body = newBlock();
if (env._debug_scopes) {
// debug scope options forces everything to be in scope
markEval(lc);
}
assert functionNode != null;
final int functionId = functionNode.getId();
parseBody = reparsedFunction == null || functionId <= reparsedFunction.getFunctionNodeId();

View File

@ -81,6 +81,9 @@ public final class ScriptEnvironment {
/** Generate line number table in class files */
public final boolean _debug_lines;
/** Put all variables in scopes to make them debuggable */
public final boolean _debug_scopes;
/** Directory in which source files and generated class files are dumped */
public final String _dest_dir;
@ -246,6 +249,7 @@ public final class ScriptEnvironment {
_compile_only = options.getBoolean("compile.only");
_const_as_var = options.getBoolean("const.as.var");
_debug_lines = options.getBoolean("debug.lines");
_debug_scopes = options.getBoolean("debug.scopes");
_dest_dir = options.getString("d");
_dump_on_error = options.getBoolean("doe");
_early_lvalue_error = options.getBoolean("early.lvalue.error");

View File

@ -51,6 +51,7 @@ import jdk.internal.dynalink.linker.support.Guards;
import jdk.nashorn.internal.codegen.ApplySpecialization;
import jdk.nashorn.internal.codegen.Compiler;
import jdk.nashorn.internal.codegen.CompilerConstants.Call;
import jdk.nashorn.internal.ir.FunctionNode;
import jdk.nashorn.internal.objects.Global;
import jdk.nashorn.internal.objects.NativeFunction;
import jdk.nashorn.internal.objects.annotations.SpecializedFunction.LinkLogic;
@ -471,6 +472,15 @@ public class ScriptFunction extends ScriptObject {
return data.isStrict();
}
/**
* Is this is a function with all variables in scope?
* @return true if function has all
*/
public boolean hasAllVarsInScope() {
return data instanceof RecompilableScriptFunctionData &&
(((RecompilableScriptFunctionData) data).getFunctionFlags() & FunctionNode.HAS_ALL_VARS_IN_SCOPE) != 0;
}
/**
* Returns true if this is a non-strict, non-built-in function that requires
* non-primitive this argument according to ECMA 10.4.3.

View File

@ -1244,7 +1244,7 @@ public abstract class ScriptObject implements PropertyAccess, Cloneable {
public final ScriptObject getProto(final int n) {
assert n > 0;
ScriptObject p = getProto();
for (int i = n; i-- > 0;) {
for (int i = n; --i > 0;) {
p = p.getProto();
}
return p;

View File

@ -77,6 +77,15 @@ nashorn.option.xhelp = { \
desc="Print extended help for command line flags." \
}
nashorn.option.anonymous.classes = { \
name="--anonymous-classes", \
is_undocumented=true, \
params=[auto|true|false], \
default=auto, \
type=string, \
desc="Use VM anonymous classes for compiled scripts." \
}
nashorn.option.class.cache.size ={ \
name="--class-cache-size", \
short_name="-ccs", \
@ -118,6 +127,31 @@ nashorn.option.d = { \
type=String \
}
nashorn.option.D = { \
name="-D", \
desc="-Dname=value. Set a system property. This option can be repeated.", \
type=String \
}
nashorn.option.debug.lines = { \
name="--debug-lines", \
is_undocumented=true, \
desc="Generate line number table in .class files.", \
default=true \
}
nashorn.option.debug.locals = { \
name="--debug-locals", \
is_undocumented=true, \
desc="Generate local variable table in .class files." \
}
nashorn.option.debug.scopes = { \
name="--debug-scopes", \
is_undocumented=true, \
desc="Put all variables in scopes to make them debuggable." \
}
nashorn.option.doe = { \
name="-dump-on-error", \
short_name="-doe", \
@ -172,6 +206,14 @@ nashorn.option.global.per.engine = { \
default=false \
}
nashorn.option.language = { \
name="--language", \
type=String, \
params=[es5|es6], \
default=es5, \
desc="Specify ECMAScript language version." \
}
nashorn.option.log = { \
name="--log", \
is_undocumented=true, \
@ -181,19 +223,6 @@ nashorn.option.log = { \
type=Log \
}
nashorn.option.debug.lines = { \
name="--debug-lines", \
is_undocumented=true, \
desc="Generate line number table in .class files.", \
default=true \
}
nashorn.option.debug.locals = { \
name="--debug-locals", \
is_undocumented=true, \
desc="Generate local variable table in .class files." \
}
nashorn.option.lazy.compilation = { \
name="--lazy-compilation", \
is_undocumented=true, \
@ -201,13 +230,6 @@ nashorn.option.lazy.compilation = {
default=true \
}
nashorn.option.optimistic.types = { \
name="--optimistic-types", \
short_name="-ot", \
desc="Use optimistic type assumptions with deoptimizing recompilation. This makes the compiler try, for any program symbol whose type cannot be proven at compile time, to type it as narrow and primitive as possible. If the runtime encounters an error because symbol type is too narrow, a wider method will be generated until steady stage is reached. While this produces as optimal Java Bytecode as possible, erroneous type guesses will lead to longer warmup. Optimistic typing is currently enabled by default, but can be disabled for faster startup performance.", \
default=true \
}
nashorn.option.loader.per.compile = { \
name="--loader-per-compile", \
is_undocumented=true, \
@ -239,6 +261,13 @@ nashorn.option.no.typed.arrays = { \
default=false \
}
nashorn.option.optimistic.types = { \
name="--optimistic-types", \
short_name="-ot", \
desc="Use optimistic type assumptions with deoptimizing recompilation. This makes the compiler try, for any program symbol whose type cannot be proven at compile time, to type it as narrow and primitive as possible. If the runtime encounters an error because symbol type is too narrow, a wider method will be generated until steady stage is reached. While this produces as optimal Java Bytecode as possible, erroneous type guesses will lead to longer warmup. Optimistic typing is currently enabled by default, but can be disabled for faster startup performance.", \
default=true \
}
nashorn.option.parse.only = { \
name="--parse-only", \
is_undocumented=true, \
@ -313,12 +342,6 @@ nashorn.option.print.symbols = { \
desc="Print the symbol table." \
}
nashorn.option.D = { \
name="-D", \
desc="-Dname=value. Set a system property. This option can be repeated.", \
type=String \
}
nashorn.option.strict = { \
name="-strict", \
desc="Run scripts in strict mode." \
@ -329,14 +352,6 @@ nashorn.option.scripting = { \
desc="Enable scripting features." \
}
nashorn.option.language = { \
name="--language", \
type=String, \
params=[es5|es6], \
default=es5, \
desc="Specify ECMAScript language version." \
}
nashorn.option.stdout = { \
name="--stdout", \
is_undocumented=true, \
@ -380,15 +395,6 @@ nashorn.option.trace.callsites = {
enterexit [trace callsite enter/exit], objects [print object properties]." \
}
nashorn.option.anonymous.classes = { \
name="--anonymous-classes", \
is_undocumented=true, \
params=[auto|true|false], \
default=auto, \
type=string, \
desc="Use VM anonymous classes for compiled scripts." \
}
nashorn.option.unstable.relink.threshold ={ \
name="--unstable-relink-threshold", \
short_name="-urt", \

View File

@ -0,0 +1,46 @@
/*
* 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-8131929: Add option for debuggable scopes
*
* @test
* @run
* @option --debug-scopes=true
* @option -Dnashorn.debug
* @fork
*/
function f1(x, y) {
var a = x * 2 , b = y + 3;
print(a, b);
return a + b;
}
function f2() {}
Assert.assertTrue(Debug.isDebuggableFunction(f1));
Assert.assertTrue(Debug.isDebuggableFunction(f2));
Assert.assertTrue(Debug.isDebuggableFunction((function() { return function() {}})()));

View File

@ -0,0 +1,34 @@
/*
* 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-8131929: Add option for debuggable scopes
*
* @test
* @runif external.prototype
* @option --debug-scopes=true
* @fork
*/
load(__DIR__ + 'prototype.js');

View File

@ -0,0 +1 @@
parsed and compiled ok prototype.js

View File

@ -0,0 +1,34 @@
/*
* 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-8131929: Add option for debuggable scopes
*
* @test
* @runif external.yui
* @option --debug-scopes=true
* @fork
*/
load(__DIR__ + 'yui.js');

View File

@ -0,0 +1,2 @@
parsed and compiled ok yui-min.js
parsed and compiled ok yui.js

View File

@ -22,7 +22,7 @@
*/
/**
* NASHORN-467 - check that prootype.js parses and compiles
* NASHORN-467 - check that prototype.js parses and compiles
*
* @test
* @runif external.prototype